Barely working ansi color codes parser but I _really_ don't like this one. Too much code to just get it to process correctly which means it'll be brittle as hell later.
parent
a44a9a04f9
commit
f32b39afe2
@ -0,0 +1,302 @@ |
||||
|
||||
#line 1 ".\\ansi_parser.rl" |
||||
#include <fmt/core.h> |
||||
#include <string_view> |
||||
#include <charconv> |
||||
#include "dbc.hpp" |
||||
|
||||
using namespace fmt; |
||||
|
||||
|
||||
#line 80 ".\\ansi_parser.rl" |
||||
|
||||
|
||||
|
||||
#line 11 ".\\ansi_parser.cpp" |
||||
static const char _foo_actions[] = { |
||||
0, 1, 0, 1, 1, 1, 8, 1,
|
||||
9, 1, 10, 2, 1, 2, 2, 1,
|
||||
3, 2, 1, 4, 2, 1, 5, 2,
|
||||
6, 0, 2, 7, 0, 2, 9, 0,
|
||||
2, 10, 8 |
||||
}; |
||||
|
||||
static const char _foo_key_offsets[] = { |
||||
0, 0, 1, 2, 7, 11, 13, 16,
|
||||
21, 25, 29, 33, 35, 38, 40, 43,
|
||||
45, 48, 52, 54, 57, 59, 62, 67,
|
||||
71, 75, 76 |
||||
}; |
||||
|
||||
static const char _foo_trans_keys[] = { |
||||
27, 91, 51, 52, 109, 48, 57, 59,
|
||||
109, 48, 57, 48, 57, 109, 48, 57,
|
||||
56, 59, 109, 48, 57, 59, 109, 48,
|
||||
57, 50, 53, 48, 57, 59, 109, 48,
|
||||
57, 48, 57, 59, 48, 57, 48, 57,
|
||||
59, 48, 57, 48, 57, 109, 48, 57,
|
||||
59, 109, 48, 57, 48, 57, 59, 48,
|
||||
57, 48, 57, 109, 48, 57, 56, 59,
|
||||
109, 48, 57, 59, 109, 48, 57, 50,
|
||||
53, 48, 57, 27, 27, 0 |
||||
}; |
||||
|
||||
static const char _foo_single_lengths[] = { |
||||
0, 1, 1, 3, 2, 0, 1, 3,
|
||||
2, 2, 2, 0, 1, 0, 1, 0,
|
||||
1, 2, 0, 1, 0, 1, 3, 2,
|
||||
2, 1, 1 |
||||
}; |
||||
|
||||
static const char _foo_range_lengths[] = { |
||||
0, 0, 0, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 0, 0 |
||||
}; |
||||
|
||||
static const char _foo_index_offsets[] = { |
||||
0, 0, 2, 4, 9, 13, 15, 18,
|
||||
23, 27, 31, 35, 37, 40, 42, 45,
|
||||
47, 50, 54, 56, 59, 61, 64, 69,
|
||||
73, 77, 79 |
||||
}; |
||||
|
||||
static const char _foo_trans_targs[] = { |
||||
2, 1, 3, 0, 7, 22, 26, 4,
|
||||
0, 5, 26, 4, 0, 6, 0, 26,
|
||||
6, 0, 8, 5, 26, 4, 0, 9,
|
||||
26, 4, 0, 10, 17, 6, 0, 11,
|
||||
26, 6, 0, 12, 0, 13, 12, 0,
|
||||
14, 0, 15, 14, 0, 16, 0, 26,
|
||||
16, 0, 18, 26, 6, 0, 19, 0,
|
||||
20, 19, 0, 21, 0, 26, 21, 0,
|
||||
23, 5, 26, 4, 0, 24, 26, 4,
|
||||
0, 10, 17, 6, 0, 2, 1, 2,
|
||||
1, 0 |
||||
}; |
||||
|
||||
static const char _foo_trans_actions[] = { |
||||
0, 5, 0, 0, 29, 29, 7, 29,
|
||||
0, 3, 17, 0, 0, 1, 0, 20,
|
||||
0, 0, 0, 3, 17, 0, 0, 3,
|
||||
17, 0, 0, 26, 26, 1, 0, 0,
|
||||
20, 0, 0, 1, 0, 3, 0, 0,
|
||||
1, 0, 3, 0, 0, 1, 0, 14,
|
||||
0, 0, 0, 20, 0, 0, 1, 0,
|
||||
3, 0, 0, 1, 0, 11, 0, 0,
|
||||
0, 3, 17, 0, 0, 3, 17, 0,
|
||||
0, 23, 23, 1, 0, 0, 5, 9,
|
||||
32, 0 |
||||
}; |
||||
|
||||
static const char _foo_eof_actions[] = { |
||||
0, 0, 0, 0, 0, 0, 0, 0,
|
||||
0, 0, 0, 0, 0, 0, 0, 0,
|
||||
0, 0, 0, 0, 0, 0, 0, 0,
|
||||
0, 0, 9 |
||||
}; |
||||
|
||||
static const int foo_start = 25; |
||||
static const int foo_first_final = 25; |
||||
static const int foo_error = 0; |
||||
|
||||
static const int foo_en_main = 25; |
||||
|
||||
|
||||
#line 83 ".\\ansi_parser.rl" |
||||
|
||||
bool parse_ansi(std::string_view codes) { |
||||
const char *start = NULL; |
||||
int cs = 0; |
||||
size_t value = 0; |
||||
const char *p = codes.data(); |
||||
const char *pe = p + codes.size(); |
||||
const char *eof = pe; |
||||
|
||||
|
||||
|
||||
#line 111 ".\\ansi_parser.cpp" |
||||
{ |
||||
cs = foo_start; |
||||
} |
||||
|
||||
#line 94 ".\\ansi_parser.rl" |
||||
|
||||
#line 114 ".\\ansi_parser.cpp" |
||||
{ |
||||
int _klen; |
||||
unsigned int _trans; |
||||
const char *_acts; |
||||
unsigned int _nacts; |
||||
const char *_keys; |
||||
|
||||
if ( p == pe ) |
||||
goto _test_eof; |
||||
if ( cs == 0 ) |
||||
goto _out; |
||||
_resume: |
||||
_keys = _foo_trans_keys + _foo_key_offsets[cs]; |
||||
_trans = _foo_index_offsets[cs]; |
||||
|
||||
_klen = _foo_single_lengths[cs]; |
||||
if ( _klen > 0 ) { |
||||
const char *_lower = _keys; |
||||
const char *_mid; |
||||
const char *_upper = _keys + _klen - 1; |
||||
while (1) { |
||||
if ( _upper < _lower ) |
||||
break; |
||||
|
||||
_mid = _lower + ((_upper-_lower) >> 1); |
||||
if ( (*p) < *_mid ) |
||||
_upper = _mid - 1; |
||||
else if ( (*p) > *_mid ) |
||||
_lower = _mid + 1; |
||||
else { |
||||
_trans += (unsigned int)(_mid - _keys); |
||||
goto _match; |
||||
} |
||||
} |
||||
_keys += _klen; |
||||
_trans += _klen; |
||||
} |
||||
|
||||
_klen = _foo_range_lengths[cs]; |
||||
if ( _klen > 0 ) { |
||||
const char *_lower = _keys; |
||||
const char *_mid; |
||||
const char *_upper = _keys + (_klen<<1) - 2; |
||||
while (1) { |
||||
if ( _upper < _lower ) |
||||
break; |
||||
|
||||
_mid = _lower + (((_upper-_lower) >> 1) & ~1); |
||||
if ( (*p) < _mid[0] ) |
||||
_upper = _mid - 2; |
||||
else if ( (*p) > _mid[1] ) |
||||
_lower = _mid + 2; |
||||
else { |
||||
_trans += (unsigned int)((_mid - _keys)>>1); |
||||
goto _match; |
||||
} |
||||
} |
||||
_trans += _klen; |
||||
} |
||||
|
||||
_match: |
||||
cs = _foo_trans_targs[_trans]; |
||||
|
||||
if ( _foo_trans_actions[_trans] == 0 ) |
||||
goto _again; |
||||
|
||||
_acts = _foo_actions + _foo_trans_actions[_trans]; |
||||
_nacts = (unsigned int) *_acts++; |
||||
while ( _nacts-- > 0 ) |
||||
{ |
||||
switch ( *_acts++ ) |
||||
{ |
||||
case 0: |
||||
#line 11 ".\\ansi_parser.rl" |
||||
{ |
||||
start = p; |
||||
} |
||||
break; |
||||
case 1: |
||||
#line 15 ".\\ansi_parser.rl" |
||||
{ |
||||
auto [ptr, ec] = std::from_chars(start, p, value); |
||||
dbc::check(ec == std::errc(), "error in number parsing"); |
||||
println("NUM: {}", value); |
||||
} |
||||
break; |
||||
case 2: |
||||
#line 21 ".\\ansi_parser.rl" |
||||
{ |
||||
println("COLOR256"); |
||||
} |
||||
break; |
||||
case 3: |
||||
#line 25 ".\\ansi_parser.rl" |
||||
{ |
||||
println("COLOR24B"); |
||||
} |
||||
break; |
||||
case 4: |
||||
#line 29 ".\\ansi_parser.rl" |
||||
{ |
||||
println("SINGLE"); |
||||
} |
||||
break; |
||||
case 5: |
||||
#line 33 ".\\ansi_parser.rl" |
||||
{ |
||||
println("BASIC"); |
||||
} |
||||
break; |
||||
case 6: |
||||
#line 37 ".\\ansi_parser.rl" |
||||
{ |
||||
println("BG"); |
||||
} |
||||
break; |
||||
case 7: |
||||
#line 41 ".\\ansi_parser.rl" |
||||
{ |
||||
println("FG"); |
||||
} |
||||
break; |
||||
case 8: |
||||
#line 45 ".\\ansi_parser.rl" |
||||
{ |
||||
println("ANY: {}:{}", int((*p)), (*p)); |
||||
} |
||||
break; |
||||
case 9: |
||||
#line 49 ".\\ansi_parser.rl" |
||||
{ |
||||
println("START"); |
||||
} |
||||
break; |
||||
case 10: |
||||
#line 53 ".\\ansi_parser.rl" |
||||
{ |
||||
println("END"); |
||||
} |
||||
break; |
||||
#line 243 ".\\ansi_parser.cpp" |
||||
} |
||||
} |
||||
|
||||
_again: |
||||
if ( cs == 0 ) |
||||
goto _out; |
||||
if ( ++p != pe ) |
||||
goto _resume; |
||||
_test_eof: {} |
||||
if ( p == eof ) |
||||
{ |
||||
const char *__acts = _foo_actions + _foo_eof_actions[cs]; |
||||
unsigned int __nacts = (unsigned int) *__acts++; |
||||
while ( __nacts-- > 0 ) { |
||||
switch ( *__acts++ ) { |
||||
case 10: |
||||
#line 53 ".\\ansi_parser.rl" |
||||
{ |
||||
println("END"); |
||||
} |
||||
break; |
||||
#line 263 ".\\ansi_parser.cpp" |
||||
} |
||||
} |
||||
} |
||||
|
||||
_out: {} |
||||
} |
||||
|
||||
#line 95 ".\\ansi_parser.rl" |
||||
|
||||
print("PROCESSED {} CHARS of {}", p - codes.data(), codes.size()); |
||||
|
||||
return true; |
||||
} |
@ -0,0 +1,4 @@ |
||||
#pragma once |
||||
#include <string_view> |
||||
|
||||
bool parse_ansi(std::string_view codes); |
@ -0,0 +1,99 @@ |
||||
#include <fmt/core.h> |
||||
#include <string_view> |
||||
#include <charconv> |
||||
#include "dbc.hpp" |
||||
|
||||
using namespace fmt; |
||||
|
||||
%%{ |
||||
machine foo; |
||||
|
||||
action tstart { |
||||
start = fpc; |
||||
} |
||||
|
||||
action number { |
||||
auto [ptr, ec] = std::from_chars(start, fpc, value); |
||||
dbc::check(ec == std::errc(), "error in number parsing"); |
||||
println("NUM: {}", value); |
||||
} |
||||
|
||||
action color256 { |
||||
println("COLOR256"); |
||||
} |
||||
|
||||
action color24b { |
||||
println("COLOR24B"); |
||||
} |
||||
|
||||
action colorSingle { |
||||
println("SINGLE"); |
||||
} |
||||
|
||||
action colorBasic { |
||||
println("BASIC"); |
||||
} |
||||
|
||||
action bg { |
||||
println("BG"); |
||||
} |
||||
|
||||
action fg { |
||||
println("FG"); |
||||
} |
||||
|
||||
action any { |
||||
println("ANY: {}:{}", int(fc), fc); |
||||
} |
||||
|
||||
action start { |
||||
println("START"); |
||||
} |
||||
|
||||
action end { |
||||
println("END"); |
||||
} |
||||
|
||||
start = 0x1B "["; |
||||
fg = "38;" %fg; |
||||
bg = "48;" %bg; |
||||
num = digit+ >tstart %number; |
||||
color256 = "5;" num ";" num; |
||||
color24b = "2;" num ";" num ";" num; |
||||
basic = num ";" num; |
||||
single = num; |
||||
|
||||
colorCode = ( |
||||
start %start |
||||
( |
||||
(fg|bg) color256 %color256 | |
||||
(fg|bg) color24b %color24b | |
||||
single %colorSingle | |
||||
basic %colorBasic |
||||
)** |
||||
"m" %end |
||||
); |
||||
|
||||
other = (any+ @any -- 0x1B)*; |
||||
|
||||
main := (other :>> colorCode)**; |
||||
}%% |
||||
|
||||
%% write data; |
||||
|
||||
bool parse_ansi(std::string_view codes) { |
||||
const char *start = NULL; |
||||
int cs = 0; |
||||
size_t value = 0; |
||||
const char *p = codes.data(); |
||||
const char *pe = p + codes.size(); |
||||
const char *eof = pe; |
||||
|
||||
|
||||
%% write init; |
||||
%% write exec; |
||||
|
||||
print("PROCESSED {} CHARS of {}", p - codes.data(), codes.size()); |
||||
|
||||
return true; |
||||
} |
@ -0,0 +1,59 @@ |
||||
#include <catch2/catch_test_macros.hpp> |
||||
#include <fmt/core.h> |
||||
#include <string> |
||||
#include "ansi_parser.hpp" |
||||
|
||||
|
||||
#include "ftxui/dom/elements.hpp" // for text, bgcolor, color, vbox, hbox, separator, operator|, Elements, Element, Fit, border |
||||
#include "ftxui/dom/node.hpp" // for Render |
||||
#include "ftxui/screen/color.hpp" // for Color, Color::Black, Color::Blue, Color::BlueLight, Color::Cyan, Color::CyanLight, Color::Default, Color::GrayDark, Color::GrayLight, Color::Green, Color::GreenLight, Color::Magenta, Color::MagentaLight, Color::Red, Color::RedLight, Color::White, Color::Yellow, Color::YellowLight, Color::Palette256, ftxui |
||||
#include <ftxui/screen/color_info.hpp> // for ColorInfo |
||||
#include <ftxui/screen/screen.hpp> // for Full, Screen |
||||
#include <ftxui/screen/terminal.hpp> // for ColorSupport, Color, Palette16, Palette256, TrueColor |
||||
|
||||
|
||||
using namespace fmt; |
||||
using namespace ftxui; |
||||
|
||||
std::string generate_colors() { |
||||
const int max_value = 255; |
||||
const int value_increment = 8; |
||||
const int hue_increment = 6; |
||||
int saturation = max_value; |
||||
Elements array; |
||||
int count = 0; |
||||
|
||||
for (int value = 0; value < max_value; value += 2 * value_increment) { |
||||
Elements line; |
||||
for (int hue = 0; hue < max_value; hue += hue_increment) { |
||||
line.push_back( |
||||
text("#") //
|
||||
| color(Color::HSV(hue, saturation, value)) //
|
||||
| bgcolor(Color::HSV(hue, saturation, value + value_increment))); |
||||
count++; |
||||
} |
||||
array.push_back(hbox(std::move(line))); |
||||
break; /// UNDO ME
|
||||
} |
||||
|
||||
println("MADE {} CHARS", count); |
||||
|
||||
auto true_color_display = vbox({ |
||||
vbox(std::move(array)), |
||||
}); |
||||
|
||||
auto document = vbox({hbox({ |
||||
true_color_display, |
||||
})}); |
||||
|
||||
auto screen = Screen::Create(Dimension::Full(), Dimension::Fit(document)); |
||||
Render(screen, document); |
||||
return screen.ToString(); |
||||
} |
||||
|
||||
TEST_CASE("test out ragel parser", "[gui]") { |
||||
std::string colors = generate_colors(); |
||||
|
||||
println("--- PARSING"); |
||||
REQUIRE(parse_ansi(colors)); |
||||
} |
Loading…
Reference in new issue