import unicode import strutils import sequtils import sugar import types/color import utils/twtstr import html/dom type FormatFlags* = enum FLAG_ITALIC FLAG_BOLD FLAG_UNDERLINE FLAG_STRIKE FLAG_OVERLINE Formatting* = object fgcolor*: CellColor bgcolor*: CellColor flags: set[FormatFlags] Cell* = object of RootObj formatting*: Formatting nodes*: seq[Node] FormattingCell* = object of Cell pos*: int FlexibleLine* = object str*: string formats*: seq[FormattingCell] FlexibleGrid* = seq[FlexibleLine] FixedCell* = object of Cell runes*: seq[Rune] ow*: int FixedGrid* = seq[FixedCell] func italic(formatting: Formatting): bool = FLAG_ITALIC in formatting.flags func bold(formatting: Formatting): bool = FLAG_BOLD in formatting.flags func underline(formatting: Formatting): bool = FLAG_UNDERLINE in formatting.flags func strike(formatting: Formatting): bool = FLAG_STRIKE in formatting.flags func overline(formatting: Formatting): bool = FLAG_OVERLINE in formatting.flags proc italic_on*(formatting: var Formatting) = formatting.flags.incl(FLAG_ITALIC) proc italic_off*(formatting: var Formatting) = formatting.flags.excl(FLAG_ITALIC) proc bold_on*(formatting: var Formatting) = formatting.flags.incl(FLAG_BOLD) proc bold_off*(formatting: var Formatting) = formatting.flags.excl(FLAG_BOLD) proc underline_on*(formatting: var Formatting) = formatting.flags.incl(FLAG_UNDERLINE) proc underline_off*(formatting: var Formatting) = formatting.flags.excl(FLAG_UNDERLINE) proc strike_on*(formatting: var Formatting) = formatting.flags.incl(FLAG_STRIKE) proc strike_off*(formatting: var Formatting) = formatting.flags.excl(FLAG_STRIKE) proc overline_on*(formatting: var Formatting) = formatting.flags.incl(FLAG_OVERLINE) proc overline_off*(formatting: var Formatting) = formatting.flags.excl(FLAG_OVERLINE) proc `bold=`*(formatting: var Formatting, b: bool) = if b: formatting.flags.incl(FLAG_BOLD) else: formatting.flags.excl(FLAG_BOLD) proc `underline=`*(formatting: var Formatting, b: bool) = if b: formatting.flags.incl(FLAG_UNDERLINE) else: formatting.flags.excl(FLAG_UNDERLINE) proc `strike=`*(formatting: var Formatting, b: bool) = if b: formatting.flags.incl(FLAG_STRIKE) else: formatting.flags.excl(FLAG_STRIKE) proc `overline=`*(formatting: var Formatting, b: bool) = if b: formatting.flags.incl(FLAG_OVERLINE) else: formatting.flags.excl(FLAG_OVERLINE) #TODO ????? func `==`*(a: FixedCell, b: FixedCell): bool = return a.formatting == b.formatting and a.runes == b.runes and a.nodes == b.nodes func newFixedGrid*(w: int, h: int = 1): FixedGrid = return newSeq[FixedCell](w * h) func width*(line: FlexibleLine): int = return line.str.width() func width*(cell: FixedCell): int = return cell.runes.width() func newFormatting*(): Formatting = return Formatting(fgcolor: defaultColor, bgcolor: defaultColor) func findFormat*(line: FlexibleLine, pos: int): FormattingCell = var i = 0 while i < line.formats.len: if line.formats[i].pos > pos: break inc i dec i if i != -1: result = line.formats[i] else: result.pos = -1 func findNextFormat*(line: FlexibleLine, pos: int): FormattingCell = var i = 0 while i < line.formats.len: if line.formats[i].pos > pos: break inc i if i < line.formats.len: result = line.formats[i] else: result.pos = -1 func subformats*(formats: seq[FormattingCell], pos: int): seq[FormattingCell] = var i = 0 while i < formats.len: if formats[i].pos >= pos: if result.len == 0 and i > 0: var f = formats[i - 1] f.pos = 0 result.add(f) var f = formats[i] f.pos -= pos result.add(f) inc i if result.len == 0 and i > 0: var f = formats[i - 1] f.pos = 0 result.add(f) proc setLen*(line: var FlexibleLine, len: int) = for i in 0 ..< line.formats.len: if line.formats[i].pos >= len: line.formats.setLen(i) break line.str.setLen(len) #line.formats = line.formats.filter((x) => x.pos < len) proc add*(a: var FlexibleLine, b: FlexibleLine) = let l = a.str.len a.formats.add(b.formats.map((x) => FormattingCell(formatting: x.formatting, nodes: x.nodes, pos: l + x.pos))) a.str &= b.str proc addLine*(grid: var FlexibleGrid) = grid.add(FlexibleLine()) proc addFormat*(grid: var FlexibleGrid, y, pos: int, format: Formatting) = grid[y].formats.add(FormattingCell(formatting: format, pos: grid[y].str.len)) proc addFormat*(grid: var FlexibleGrid, y, pos: int, format: Formatting, nodes: seq[Node]) = grid[y].formats.add(FormattingCell(formatting: format, nodes: nodes, pos: pos)) proc addCell*(grid: var FlexibleGrid, y: int, r: Rune) = grid[y].str &= $r proc addCell*(grid: var FlexibleGrid, r: Rune) = grid.addCell(grid.len - 1, r) template inc_check(i: int) = inc i if i >= buf.len: return i proc parseAnsiCode*(formatting: var Formatting, buf: string, fi: int): int = var i = fi if buf[i] != '\e': return i inc_check i if 0x40 <= int(buf[i]) and int(buf[i]) <= 0x5F: if buf[i] != '[': #C1, TODO? return inc_check i let sp = i #parameter bytes while 0x30 <= int(buf[i]) and int(buf[i]) <= 0x3F: inc_check i let params = buf.substr(sp, i - 1) let si = i #intermediate bytes while 0x20 <= int(buf[i]) and int(buf[i]) <= 0x2F: inc_check i let interm = buf.substr(si, i) let final = buf[i] #final byte if 0x40 <= int(buf[i]) and int(buf[i]) <= 0x7E: inc_check i case final of 'm': if params.len == 0: formatting = newFormatting() else: let sparams = params.split(';') try: let ip = sparams.map((x) => parseInt(x)) var pi = 0 while pi < ip.len: case ip[pi] of 0: formatting = newFormatting() of 1: formatting.bold_on of 3: formatting.italic_on of 4: formatting.underline_on of 9: formatting.strike_on of 22: formatting.bold_off of 23: formatting.italic_off of 29: formatting.strike_off of 30..37: formatting.fgcolor = CellColor(rgb: false, color: uint8(ip[pi])) of 38: inc pi if pi < ip.len: if ip[pi] == 2: inc pi if pi + 2 < ip.len: let r = ip[pi] inc pi let g = ip[pi] inc pi let b = ip[pi] formatting.fgcolor = CellColor(rgb: true, rgbcolor: rgb(r, g, b)) else: #TODO inc pi continue else: break of 39: formatting.fgcolor = defaultColor of 40..47: formatting.bgcolor = CellColor(rgb: false, color: uint8(ip[0])) of 48: inc pi if pi < ip.len: if ip[pi] == 2: inc pi if pi + 2 < ip.len: let r = ip[pi] inc pi let g = ip[pi] inc pi let b = ip[pi] formatting.bgcolor = CellColor(rgb: true, rgbcolor: rgb(r, g, b)) else: #TODO inc pi continue else: break of 49: formatting.bgcolor = defaultColor of 53: formatting.overline_on of 55: formatting.overline_off else: discard inc pi except ValueError: discard else: discard return i proc processFormatting*(formatting: var Formatting, cellf: Formatting): string = if formatting.bold and not cellf.bold: result &= SGR(22) if formatting.italic and not cellf.italic: result &= SGR(23) if formatting.underline and not cellf.underline: result &= SGR(24) if formatting.strike and not cellf.strike: result &= SGR(29) if formatting.overline and not cellf.overline: result &= SGR(55) if cellf.fgcolor != formatting.fgcolor: var color = cellf.fgcolor if color.rgb: let rgb = color.rgbcolor result &= SGR(38, 2, rgb.r, rgb.g, rgb.b) elif color == defaultColor: result &= SGR() formatting = newFormatting() else: result &= SGR(color.color) if cellf.bgcolor != formatting.bgcolor: var color = cellf.bgcolor if color.rgb: let rgb = color.rgbcolor result &= SGR(48, 2, rgb.r, rgb.g, rgb.b) elif color == defaultColor: result &= SGR() formatting = newFormatting() else: result &= SGR(color.color) if not formatting.bold and cellf.bold: result &= SGR(1) if not formatting.italic and cellf.italic: result &= SGR(3) if not formatting.underline and cellf.underline: result &= SGR(4) if not formatting.strike and cellf.strike: result &= SGR(9) if not formatting.overline and cellf.overline: result &= SGR(53) formatting = cellf