lnav/src/ansi_scrubber.cc

166 lines
5.6 KiB
C++

/**
* Copyright (c) 2013, Timothy Stack
*
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* * Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
* * Neither the name of Timothy Stack nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ''AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
* ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
* @file ansi_scrubber.cc
*/
#include "config.h"
#include <vector>
#include "pcrepp.hh"
#include "ansi_scrubber.hh"
using namespace std;
static pcrepp &ansi_regex(void)
{
static pcrepp retval("\x1b\\[([\\d=;]*)([a-zA-Z])");
return retval;
}
void scrub_ansi_string(std::string &str, string_attrs_t &sa)
{
view_colors &vc = view_colors::singleton();
pcre_context_static<60> context;
pcrepp & regex = ansi_regex();
pcre_input pi(str);
replace(str.begin(), str.end(), '\0', ' ');
while (regex.match(context, pi)) {
pcre_context::capture_t *caps = context.all();
struct line_range lr;
bool has_attrs = false;
int attrs = 0;
int bg = 0;
int fg = 0;
size_t lpc;
switch (pi.get_substr_start(&caps[2])[0]) {
case 'm':
for (lpc = caps[1].c_begin;
lpc != string::npos && lpc < (size_t) caps[1].c_end;) {
int ansi_code = 0;
if (sscanf(&(str[lpc]), "%d", &ansi_code) == 1) {
if (90 <= ansi_code && ansi_code <= 97) {
ansi_code -= 60;
attrs |= A_STANDOUT;
}
if (30 <= ansi_code && ansi_code <= 37) {
fg = ansi_code - 30;
}
if (40 <= ansi_code && ansi_code <= 47) {
bg = ansi_code - 40;
}
switch (ansi_code) {
case 1:
attrs |= A_BOLD;
break;
case 2:
attrs |= A_DIM;
break;
case 4:
attrs |= A_UNDERLINE;
break;
case 7:
attrs |= A_REVERSE;
break;
}
}
lpc = str.find(";", lpc);
if (lpc != string::npos) {
lpc += 1;
}
}
if (fg != 0 || bg != 0) {
attrs |= vc.ansi_color_pair(fg, bg);
}
has_attrs = true;
break;
case 'C': {
unsigned int spaces = 0;
if (sscanf(&(str[caps[1].c_begin]), "%u", &spaces) == 1 &&
spaces > 0) {
str.insert((unsigned long) caps[0].c_end, spaces, ' ');
}
break;
}
case 'O': {
int role_int;
if (sscanf(&(str[caps[1].c_begin]), "%d", &role_int) == 1) {
if (role_int >= 0 && role_int < view_colors::VCR__MAX) {
attrs = vc.attrs_for_role(
(view_colors::role_t) role_int);
has_attrs = true;
}
}
break;
}
}
str.erase(str.begin() + caps[0].c_begin,
str.begin() + caps[0].c_end);
if (has_attrs) {
if (!sa.empty()) {
sa.back().sa_range.lr_end = caps[0].c_begin;
}
lr.lr_start = caps[0].c_begin;
lr.lr_end = -1;
sa.push_back(string_attr(lr, &view_curses::VC_STYLE, attrs));
}
pi.reset(str);
}
}
void add_ansi_vars(std::map<std::string, std::string> &vars)
{
vars["ansi_csi"] = "\x1b[";
vars["ansi_norm"] = "\x1b[0m";
vars["ansi_bold"] = "\x1b[1m";
vars["ansi_underline"] = "\x1b[4m";
vars["ansi_black"] = "\x1b[30m";
vars["ansi_red"] = "\x1b[31m";
vars["ansi_green"] = "\x1b[32m";
vars["ansi_yellow"] = "\x1b[33m";
vars["ansi_blue"] = "\x1b[34m";
vars["ansi_magenta"] = "\x1b[35m";
vars["ansi_cyan"] = "\x1b[36m";
vars["ansi_white"] = "\x1b[37m";
}