about summary refs log blame commit diff stats
path: root/src/io/cell.nim
blob: d4e67196d189e5a67a3c8de4f33ee69dde60ee1a (plain) (tree)
1
2
3
4
5
6
7
8
9
10
              


               

                  
                   
               

    






                     
                      

                       
                           
 

                           
                     
 

                                  


                        
                                 

                                   


                             
            


                             




































                                                                                       





                                              



                                                   
                         
 


                                   


                                    


                                                                 
                                                                






                                 
                            
       
                   
 









                                                                    
 























                                                                               


                                                 
                                                                                                               

                
                                       



                                                                           
 

                                                                                                 
 
                                                        
                   
 

                                                
 

















































                                                                            
                                






                                       
                                 









                                                                            
                                
                        
                                
                        

                                                                                   















                                                                           
                                
                        
                                
                        

                                                                                   

















                                             













                                                                                
































                                                    
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 len*(line: FlexibleLine): int =
  return line.str.runeLen()

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.formats = line.formats.filter((x) => x.pos < len)

proc add*(a: var FlexibleLine, b: FlexibleLine) =
  let l = a.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: int, format: Formatting) =
  grid[y].formats.add(FormattingCell(formatting: format, pos: grid[y].len))

proc addFormat*(grid: var FlexibleGrid, y: int, 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 = true
          of 3:
            formatting.italic_on
          of 4:
            formatting.underline = true
          of 9:
            formatting.strike = true
          of 22:
            formatting.bold = false
          of 23:
            formatting.italic_off
          of 29:
            formatting.strike = false
          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 = true
          of 55:
            formatting.overline = false
          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