import terminal import uri import strutils import unicode import streams import types/enums import css/values import css/style import utils/twtstr import html/dom import layout/box import layout/engine import config/config import io/term import io/lineedit import io/cell type Buffer* = ref BufferObj BufferObj = object title*: string lines*: FlexibleGrid display*: FixedGrid prevdisplay*: FixedGrid statusmsg*: FixedGrid hovertext*: string width*: int height*: int cursorx*: int cursory*: int xend*: int fromx*: int fromy*: int attrs*: TermAttributes document*: Document redraw*: bool reshape*: bool location*: Uri source*: string showsource*: bool rootbox*: CSSBox prevnodes*: seq[Node] func newBuffer*(attrs: TermAttributes): Buffer = new(result) result.width = attrs.width result.height = attrs.height - 1 result.attrs = attrs result.display = newFixedGrid(result.width, result.height) result.prevdisplay = newFixedGrid(result.width, result.height) result.statusmsg = newFixedGrid(result.width) func generateFullOutput*(buffer: Buffer): string = var x = 0 var w = 0 var formatting = newFormatting() result &= HVP(1, 1) for cell in buffer.display: if x >= buffer.width: if w < buffer.width: result &= EL() result &= '\n' x = 0 w = 0 result &= formatting.processFormatting(cell.formatting) result &= $cell.runes w += cell.width() inc x if w < buffer.width: result &= EL() result &= '\n' # generate a sequence of instructions to replace the previous frame with the # current one. ideally should be used when small changes are made (e.g. hover # changes underlining) func generateSwapOutput*(buffer: Buffer): string = var formatting = newFormatting() let curr = buffer.display let prev = buffer.prevdisplay var i = 0 var x = 0 var y = 0 var line = "" var lr = false while i < curr.len: if x >= buffer.width: if lr: result &= HVP(y + 1, 1) result &= EL() result &= line lr = false x = 0 inc y line = "" lr = lr or (curr[i] != prev[i]) line &= formatting.processFormatting(curr[i].formatting) line &= $curr[i].runes inc i inc x if lr: result &= HVP(y + 1, 1) result &= EL() result &= line lr = false #TODO maybe fix this #var x = 0 #var y = 0 #var cx = -1 #var cy = -1 #var i = 0 #var text = "" #while i < max: # if x >= buffer.width: # x = 0 # inc y # if curr[i] != prev[i]: # let currwidth = curr[i].runes.width() # let prevwidth = prev[i].runes.width() # if (curr[i].runes.len > 0 or currwidth < prevwidth) and (x != cx or y != cy): # if text.len > 0: # result &= text # text = "" # result &= HVP(y + 1, x + 1) # cx = x # cy = y # text &= formatting.processFormatting(curr[i].formatting) # text &= $curr[i].runes # if currwidth < prevwidth: # var j = 0 # while j < prevwidth - currwidth: # text &= ' ' # inc j # if text.len > 0: # inc cx # inc x # inc i #if text.len > 0: # result &= $text func generateStatusMessage*(buffer: Buffer): string = for cell in buffer.statusmsg: for r in cell.runes: if r != Rune(0): result &= $r func numLines*(buffer: Buffer): int = buffer.lines.len func lastVisibleLine*(buffer: Buffer): int = min(buffer.fromy + buffer.height, buffer.numLines) func acursorx(buffer: Buffer): int = return max(0, buffer.cursorx - buffer.fromx) func acursory(buffer: Buffer): int = return buffer.cursory - buffer.fromy func cellOrigin(buffer: Buffer, x: int, y: int): int = let row = y * buffer.width var ox = x while buffer.display[row + ox].runes.len == 0 and ox > 0: dec ox return ox func currentCellOrigin(buffer: Buffer): int = return buffer.cellOrigin(buffer.acursorx, buffer.acursory) #TODO counter-intuitive naming? func currentCell(buffer: Buffer): FixedCell = let row = (buffer.cursory - buffer.fromy) * buffer.width return buffer.display[row + buffer.currentCellOrigin()] func cell(buffer: Buffer): FixedCell = let row = (buffer.cursory - buffer.fromy) * buffer.width return buffer.display[row + buffer.acursorx] func currentRune(buffer: Buffer): Rune = let cell = buffer.currentCell() if cell.runes.len == 0: return Rune(' ') return cell.runes[0] func getCursorLink(buffer: Buffer): Element = let nodes = buffer.currentCell().nodes for node in nodes: if node.nodeType == ELEMENT_NODE: let elem = Element(node) if elem.tagType == TAG_A: return elem return nil func currentLineWidth*(buffer: Buffer): int = if buffer.cursory > buffer.lines.len: return 0 return buffer.lines[buffer.cursory].width() func maxScreenWidth*(buffer: Buffer): int = for line in buffer.lines[buffer.fromy..buffer.lastVisibleLine - 1]: result = max(line.width(), result) func atPercentOf*(buffer: Buffer): int = if buffer.lines.len == 0: return 100 return (100 * (buffer.cursory + 1)) div buffer.numLines func canScroll*(buffer: Buffer): bool = return buffer.numLines >= buffer.height proc addLine(buffer: Buffer) = buffer.lines.addLine() proc clearText*(buffer: Buffer) = buffer.lines.setLen(0) buffer.addLine() proc clearBuffer*(buffer: Buffer) = buffer.clearText() buffer.cursorx = 0 buffer.cursory = 0 buffer.fromx = 0 buffer.fromy = 0 buffer.hovertext = "" proc clearDisplay*(buffer: Buffer) = buffer.prevdisplay = buffer.display buffer.display = newFixedGrid(buffer.width, buffer.height) proc refreshDisplay*(buffer: Buffer) = var r: Rune var y = 0 buffer.clearDisplay() for line in buffer.lines[buffer.fromy.. buffer.lastVisibleLine - 1]: var w = 0 var i = 0 var j = 0 while w < buffer.fromx and i < line.str.len: fastRuneAt(line.str, i, r) w += r.width() inc j let dls = y * buffer.width var k = 0 var cf = line.findFormat(j) var nf = line.findNextFormat(j) if w > buffer.fromx: while k < w - buffer.fromx: buffer.display[dls + k].runes.add(Rune(' ')) buffer.display[dls + k].ow = r.width() inc k while i < line.str.len: fastRuneAt(line.str, i, r) w += r.width() if w > buffer.fromx + buffer.width: buffer.display[dls + k].ow += r.width() break if nf.pos != -1 and nf.pos <= j: cf = nf nf = line.findNextFormat(j) buffer.display[dls + k].runes.add(r) if cf.pos != -1: buffer.display[dls + k].formatting = cf.formatting buffer.display[dls + k].nodes = cf.nodes let tk = k + r.width() while k < tk: buffer.display[dls + k].ow += r.width() inc k inc j inc y proc restoreCursorX(buffer: Buffer) = buffer.cursorx = max(min(buffer.currentLineWidth() - 1, buffer.xend), 0) proc scrollTo*(buffer: Buffer, y: int) = if y == buffer.fromy: return buffer.fromy = min(max(buffer.numLines - buffer.height, 0), y) buffer.cursory = min(max(buffer.fromy, buffer.cursory), buffer.fromy + buffer.height) buffer.redraw = true buffer.restoreCursorX() proc cursorTo*(buffer: Buffer, x: int, y: int) = buffer.redraw = false buffer.cursory = min(max(y, 0), buffer.numLines - 1) if buffer.fromy > buffer.cursory: buffer.fromy = max(buffer.cursory, 0) buffer.redraw = true elif buffer.fromy + buffer.height - 1 <= buffer.cursory: buffer.fromy = max(buffer.cursory - buffer.height + 1, 0) buffer.redraw = true buffer.cursorx = min(max(x, 0), buffer.currentLineWidth()) if buffer.fromx < buffer.cursorx - buffer.width: buffer.fromx = max(0, buffer.cursorx - buffer.width) buffer.redraw = true elif buffer.fromx > buffer.cursorx: buffer.fromx = buffer.cursorx buffer.redraw = true proc cursorDown*(buffer: Buffer) = if buffer.cursory < buffer.numLines - 1: inc buffer.cursory buffer.restoreCursorX() if buffer.cursory - buffer.height >= buffer.fromy: inc buffer.fromy buffer.redraw = true proc cursorUp*(buffer: Buffer) = if buffer.cursory > 0: dec buffer.cursory buffer.restoreCursorX() if buffer.cursory < buffer.fromy: dec buffer.fromy buffer.redraw = true #TODO these don't exactly work as intended, I think proc cursorRight*(buffer: Buffer) = let cellwidth = buffer.cell().ow let lw = buffer.currentLineWidth() if buffer.cursorx < lw - 1: buffer.cursorx = min(lw - 1, buffer.cursorx + cellwidth) assert buffer.cursorx >= 0 buffer.xend = buffer.cursorx if buffer.cursorx - buffer.width + (cellwidth - 1) >= buffer.fromx: buffer.fromx += cellwidth buffer.redraw = true if buffer.cursorx == buffer.fromx: inc buffer.cursorx proc cursorLeft*(buffer: Buffer) = let cellorigin = buffer.fromx + buffer.currentCellOrigin() let lw = buffer.currentLineWidth() if buffer.fromx > buffer.cursorx: buffer.cursorx = min(max(lw - 1, 0), cellorigin - 1) buffer.fromx = buffer.cursorx buffer.redraw = true elif buffer.cursorx > 0: buffer.cursorx = max(0, cellorigin - 1) if buffer.fromx > buffer.cursorx - (buffer.cell().ow - 1): buffer.fromx = max(buffer.cursorx - max(0, buffer.cell().ow - 1), 0) buffer.redraw = true buffer.xend = buffer.cursorx proc cursorLineBegin*(buffer: Buffer) = buffer.cursorx = 0 buffer.xend = 0 if buffer.fromx > 0: buffer.fromx = 0 buffer.redraw = true proc cursorLineEnd*(buffer: Buffer) = buffer.cursorx = max(buffer.currentLineWidth() - 1, 0) buffer.xend = buffer.cursorx buffer.fromx = max(buffer.cursorx - buffer.width + 1, 0) buffer.redraw = buffer.fromx > 0 proc cursorNextWord*(buffer: Buffer) = let llen = buffer.currentLineWidth() - 1 if llen >= 0: while not buffer.currentRune().breaksWord(): if buffer.cursorx >= llen: break buffer.cursorRight() while buffer.currentRune().breaksWord(): if buffer.cursorx >= llen: break buffer.cursorRight() if buffer.cursorx >= buffer.currentLineWidth() - 1: if buffer.cursory < buffer.numLines - 1: buffer.cursorDown() buffer.cursorLineBegin() proc cursorPrevWord*(buffer: Buffer) = if buffer.currentLineWidth() > 0: while not buffer.currentRune().breaksWord(): if buffer.cursorx == 0: break buffer.cursorLeft() while buffer.currentRune().breaksWord(): if buffer.cursorx == 0: break buffer.cursorLeft() if buffer.cursorx == 0: if buffer.cursory > 0: buffer.cursorUp() buffer.cursorLineEnd() #TODO this is sloooooow #proc cursorRightOverflow(buffer: Buffer) = # buffer.cursorRight() # if buffer.cursorx >= buffer.currentLineWidth() - 1 and buffer.cursory < buffer.numLines - 1: # buffer.cursorDown() # buffer.cursorLineBegin() # buffer.refreshDisplay() # #proc cursorLeftOverflow(buffer: Buffer) = # buffer.cursorLeft() # if buffer.cursorx <= 0 and buffer.cursory > 0: # buffer.cursorUp() # buffer.cursorLineEnd() # buffer.refreshDisplay() proc cursorNextLink*(buffer: Buffer) = #TODO #let ocx = buffer.cursorx #let ocy = buffer.cursory #let ofx = buffer.fromx #let ofy = buffer.fromy #let elem = buffer.getCursorLink() #if elem != nil: # while buffer.getCursorLink() == elem: # buffer.cursorRightOverflow() # if buffer.cursorx >= buffer.currentLineWidth() - 1 and # buffer.cursory >= buffer.numLines - 1: # buffer.cursorx = ocx # buffer.cursory = ocy # buffer.fromx = ofx # buffer.fromy = ofy # break #while buffer.getCursorLink() == nil: # buffer.cursorRightOverflow() # if buffer.cursorx >= buffer.currentLineWidth() - 1 and # buffer.cursory >= buffer.numLines - 1: # buffer.cursorx = ocx # buffer.cursory = ocy # buffer.fromx = ofx # buffer.fromy = ofy # break discard proc cursorPrevLink*(buffer: Buffer) = #TODO return proc cursorFirstLine*(buffer: Buffer) = if buffer.fromy > 0: buffer.fromy = 0 buffer.redraw = true else: buffer.redraw = false buffer.cursory = 0 buffer.restoreCursorX() proc cursorLastLine*(buffer: Buffer) = if buffer.fromy < buffer.numLines - buffer.height: buffer.fromy = buffer.numLines - buffer.height buffer.redraw = true buffer.cursory = buffer.numLines - 1 buffer.restoreCursorX() proc cursorTop*(buffer: Buffer) = buffer.cursory = buffer.fromy buffer.restoreCursorX() proc cursorMiddle*(buffer: Buffer) = buffer.cursory = min(buffer.fromy + (buffer.height - 2) div 2, buffer.numLines - 1) buffer.restoreCursorX() proc cursorBottom*(buffer: Buffer) = buffer.cursory = min(buffer.fromy + buffer.height - 1, buffer.numLines - 1) buffer.restoreCursorX() proc centerLine*(buffer: Buffer) = let ny = max(min(buffer.cursory - buffer.height div 2, buffer.numLines - buffer.height), 0) if ny != buffer.fromy: buffer.fromy = ny buffer.redraw = true proc halfPageUp*(buffer: Buffer) = buffer.cursory = max(buffer.cursory - buffer.height div 2 + 1, 0) let nfy = max(0, buffer.fromy - buffer.height div 2 + 1) if nfy != buffer.fromy: buffer.fromy = nfy buffer.redraw = true buffer.restoreCursorX() proc halfPageDown*(buffer: Buffer) = buffer.cursory = min(buffer.cursory + buffer.height div 2 - 1, buffer.numLines - 1) let nfy = min(max(buffer.numLines - buffer.height, 0), buffer.fromy + buffer.height div 2 - 1) if nfy != buffer.fromy: buffer.fromy = nfy buffer.redraw = true buffer.restoreCursorX() proc pageUp*(buffer: Buffer) = buffer.cursory = max(buffer.cursory - buffer.height, 0) let nfy = max(0, buffer.fromy - buffer.height) if nfy != buffer.fromy: buffer.fromy = nfy buffer.redraw = true buffer.restoreCursorX() proc pageDown*(buffer: Buffer) = buffer.cursory = min(buffer.cursory + buffer.height, buffer.numLines - 1) let nfy = min(buffer.fromy + buffer.height, max(buffer.numLines - buffer.height, 0)) if nfy != buffer.fromy: buffer.fromy = nfy buffer.redraw = true buffer.restoreCursorX() proc pageLeft*(buffer: Buffer) = buffer.cursorx = max(buffer.cursorx - buffer.width, 0) let nfx = max(0, buffer.fromx - buffer.width) if nfx != buffer.fromx: buffer.fromx = nfx buffer.redraw = true proc pageRight*(buffer: Buffer) = buffer.cursorx = min(buffer.fromx, buffer.currentLineWidth()) let nfx = min(max(buffer.maxScreenWidth() - buffer.width, 0), buffer.fromx + buffer.width) if nfx != buffer.fromx: buffer.fromx = nfx buffer.redraw = true proc scrollDown*(buffer: Buffer) = if buffer.fromy + buffer.height < buffer.numLines: inc buffer.fromy if buffer.fromy > buffer.cursory: buffer.cursorDown() buffer.redraw = true else: buffer.cursorDown() proc scrollUp*(buffer: Buffer) = if buffer.fromy > 0: dec buffer.fromy if buffer.fromy + buffer.height <= buffer.cursory: buffer.cursorUp() buffer.redraw = true else: buffer.cursorUp() proc scrollRight*(buffer: Buffer) = if buffer.fromx + buffer.width < buffer.maxScreenWidth(): inc buffer.fromx if buffer.fromx >= buffer.cursorx: buffer.cursorRight() buffer.redraw = true proc scrollLeft*(buffer: Buffer) = if buffer.fromx > 0: dec buffer.fromx if buffer.fromx + buffer.height <= buffer.cursorx: buffer.cursorLeft() buffer.redraw = true proc gotoAnchor*(buffer: Buffer): bool = discard #TODO #if buffer.location.anchor != "": # let node = buffer.getElementById(buffer.location.anchor) # if node != nil: # buffer.scrollTo(max(node.y - buffer.height div 2, 0)) proc setLocation*(buffer: Buffer, uri: Uri) = buffer.location = uri proc gotoLocation*(buffer: Buffer, uri: Uri) = buffer.location = buffer.location.combine(uri) proc refreshTermAttrs*(buffer: Buffer): bool = let newAttrs = getTermAttributes() if newAttrs != buffer.attrs: buffer.attrs = newAttrs buffer.width = newAttrs.width buffer.height = newAttrs.height - 1 return true return false func formatFromLine(line: CSSRowBox): Formatting = result.fgcolor = line.color.cellColor() if line.fontstyle in { FONT_STYLE_ITALIC, FONT_STYLE_OBLIQUE }: result.italic = true if line.fontweight > 500: result.bold = true if line.textdecoration == TEXT_DECORATION_UNDERLINE: result.underline = true if line.textdecoration == TEXT_DECORATION_OVERLINE: result.overline = true if line.textdecoration == TEXT_DECORATION_LINE_THROUGH: result.strike = true proc setRowBox(buffer: Buffer, line: CSSRowBox) = var r: Rune let x = line.x let y = line.y while buffer.lines.len <= y: buffer.addLine() var i = 0 var j = 0 var cx = 0 while cx < x and i < buffer.lines[y].str.len: fastRuneAt(buffer.lines[y].str, i, r) cx += r.width() inc j let ostr = buffer.lines[y].str.substr(i) let oformats = buffer.lines[y].formats.subformats(j) buffer.lines[y].str.setLen(i) buffer.lines[y].setLen(j) buffer.lines.addFormat(y, j, line.formatFromLine(), line.nodes) var nx = cx if nx < x: buffer.lines[y].str &= ' '.repeat(x - nx) nx = x buffer.lines[y].str &= line.str nx += line.str.width() i = 0 j = 0 while cx < nx and i < ostr.len: fastRuneAt(ostr, i, r) cx += r.width() inc j if i < ostr.len: let oline = FlexibleLine(str: ostr.substr(i), formats: oformats.subformats(j)) buffer.lines[y].add(oline) proc updateCursor(buffer: Buffer) = if buffer.fromy > buffer.lastVisibleLine - 1: buffer.fromy = 0 buffer.cursory = buffer.lastVisibleLine - 1 if buffer.cursorx >= buffer.currentLineWidth() - 1: buffer.cursorx = max(buffer.currentLineWidth() - 1, 0) buffer.fromx = max(buffer.cursorx - buffer.width + 1, 0) if buffer.lines.len == 0: buffer.cursory = 0 #TODO this works, but reshape rearranges all CSS boxes which is a *very* #resource-intensive operation, and a significant restructuring of the layout #engine is needed to avoid this proc updateHover(buffer: Buffer) = let nodes = buffer.currentCell().nodes if nodes != buffer.prevnodes: for node in nodes: var elem: Element if node of Element: elem = Element(node) else: elem = node.parentElement assert elem != nil if not elem.hover: elem.hover = true buffer.reshape = true elem.cssapplied = false for node in buffer.prevnodes: var elem: Element if node of Element: elem = Element(node) else: elem = node.parentElement assert elem != nil if elem.hover and not (node in nodes): elem.hover = false buffer.reshape = true elem.cssapplied = false buffer.prevnodes = nodes proc renderPlainText*(buffer: Buffer, text: string) = buffer.clearText() var i = 0 var x = 0 var y = 0 var r: Rune var format = newFormatting() while i < text.len: if text[i] == '\n': if i != text.len - 1: buffer.addLine() buffer.lines.addFormat(buffer.lines.len - 1, format) inc y x = 0 inc i elif text[i] == '\r': inc i elif text[i] == '\t': for i in 0..8: buffer.lines[^1].str &= ' ' inc i elif text[i] == '\e': i = format.parseAnsiCode(text, i) elif text[i].isControlChar(): buffer.lines.addCell(Rune('^')) buffer.lines.addCell(Rune(text[i].getControlLetter())) inc i else: fastRuneAt(text, i, r) buffer.lines.addCell(r) buffer.updateCursor() const css = staticRead"res/ua.css" let ua_stylesheet = newStringStream(css).parseStylesheet() #TODO refactor var ss_init = false var user_stylesheet: ParsedStylesheet proc renderDocument*(buffer: Buffer) = buffer.clearText() if not ss_init: user_stylesheet = newStringStream(gconfig.stylesheet).parseStylesheet() ss_init = true buffer.document.applyStylesheets(ua_stylesheet, user_stylesheet) buffer.rootbox = buffer.document.alignBoxes(buffer.width, buffer.height) if buffer.rootbox == nil: return var stack: seq[CSSBox] stack.add(buffer.rootbox) while stack.len > 0: let box = stack.pop() if box of CSSInlineBox: let inline = CSSInlineBox(box) #eprint "NEW BOX", inline.context.conty for line in inline.content: #eprint line buffer.setRowBox(line) var i = box.children.len - 1 while i >= 0: stack.add(box.children[i]) dec i buffer.updateCursor() proc reshapeBuffer*(buffer: Buffer) = #TODO #buffer.statusmsg = newFixedGrid(buffer.width) if buffer.showsource: buffer.renderPlainText(buffer.source) else: buffer.renderDocument() proc cursorBufferPos(buffer: Buffer) = let x = max(buffer.cursorx - buffer.fromx, 0) let y = buffer.cursory - buffer.fromy print(HVP(y + 1, x + 1)) proc clearStatusMessage(buffer: Buffer) = buffer.statusmsg = newFixedGrid(buffer.width) proc setStatusMessage*(buffer: Buffer, str: string) = buffer.clearStatusMessage() let text = str.toRunes() var i = 0 var n = 0 while i < text.len: if text[i].width() == 0: inc n buffer.statusmsg[i - n].runes.add(text[i]) inc i proc statusMsgForBuffer(buffer: Buffer) = var msg = ($(buffer.cursory + 1) & "/" & $buffer.numLines & " (" & $buffer.atPercentOf() & "%) " & "<" & buffer.title & ">").ansiStyle(styleReverse).ansiReset().join() if buffer.hovertext.len > 0: msg &= " " & buffer.hovertext buffer.setStatusMessage(msg) proc displayBufferSwapOutput(buffer: Buffer) = print(buffer.generateSwapOutput()) proc displayBuffer(buffer: Buffer) = print(buffer.generateFullOutput()) proc displayStatusMessage(buffer: Buffer) = print(HVP(buffer.height + 1, 1)) print(SGR()) print(buffer.generateStatusMessage()) print(EL()) proc inputLoop(attrs: TermAttributes, buffer: Buffer): bool = var s = "" var feedNext = false while true: buffer.redraw = false buffer.displayStatusMessage() stdout.showCursor() buffer.cursorBufferPos() if not feedNext: s = "" else: feedNext = false let c = getch() s &= c let action = getNormalAction(s) var nostatus = false case action of ACTION_QUIT: eraseScreen() print(HVP(0, 0)) return false of ACTION_CURSOR_LEFT: buffer.cursorLeft() of ACTION_CURSOR_DOWN: buffer.cursorDown() of ACTION_CURSOR_UP: buffer.cursorUp() of ACTION_CURSOR_RIGHT: buffer.cursorRight() of ACTION_CURSOR_LINEBEGIN: buffer.cursorLineBegin() of ACTION_CURSOR_LINEEND: buffer.cursorLineEnd() of ACTION_CURSOR_NEXT_WORD: buffer.cursorNextWord() of ACTION_CURSOR_PREV_WORD: buffer.cursorPrevWord() of ACTION_CURSOR_NEXT_LINK: buffer.cursorNextLink() of ACTION_CURSOR_PREV_LINK: buffer.cursorPrevLink() of ACTION_PAGE_DOWN: buffer.pageDown() of ACTION_PAGE_UP: buffer.pageUp() of ACTION_PAGE_RIGHT: buffer.pageRight() of ACTION_PAGE_LEFT: buffer.pageLeft() of ACTION_HALF_PAGE_DOWN: buffer.halfPageDown() of ACTION_HALF_PAGE_UP: buffer.halfPageUp() of ACTION_CURSOR_FIRST_LINE: buffer.cursorFirstLine() of ACTION_CURSOR_LAST_LINE: buffer.cursorLastLine() of ACTION_CURSOR_TOP: buffer.cursorTop() of ACTION_CURSOR_MIDDLE: buffer.cursorMiddle() of ACTION_CURSOR_BOTTOM: buffer.cursorBottom() of ACTION_CENTER_LINE: buffer.centerLine() of ACTION_SCROLL_DOWN: buffer.scrollDown() of ACTION_SCROLL_UP: buffer.scrollUp() of ACTION_SCROLL_LEFT: buffer.scrollLeft() of ACTION_SCROLL_RIGHT: buffer.scrollRight() of ACTION_CLICK: discard of ACTION_CHANGE_LOCATION: var url = $buffer.location print(HVP(buffer.height + 1, 1)) print(EL()) let status = readLine("URL: ", url, buffer.width) if status: buffer.setLocation(parseUri(url)) return true of ACTION_LINE_INFO: buffer.setStatusMessage("line " & $(buffer.cursory + 1) & "/" & $buffer.numLines & " col " & $(buffer.cursorx + 1) & "/" & $buffer.currentLineWidth() & " cell width: " & $buffer.currentCell().width()) nostatus = true of ACTION_FEED_NEXT: feedNext = true of ACTION_RELOAD: return true of ACTION_RESHAPE: buffer.reshape = true of ACTION_REDRAW: buffer.redraw = true of ACTION_TOGGLE_SOURCE: buffer.showsource = not buffer.showsource buffer.reshape = true buffer.redraw = true else: discard stdout.hideCursor() if buffer.refreshTermAttrs(): buffer.redraw = true buffer.reshape = true if buffer.redraw: buffer.refreshDisplay() buffer.displayBuffer() buffer.redraw = false buffer.updateHover() if buffer.reshape: buffer.reshapeBuffer() buffer.reshape = false buffer.refreshDisplay() buffer.displayBufferSwapOutput() if not nostatus: buffer.statusMsgForBuffer() else: nostatus = false proc displayPage*(attrs: TermAttributes, buffer: Buffer): bool = discard buffer.gotoAnchor() buffer.refreshDisplay() buffer.displayBuffer() buffer.statusMsgForBuffer() return inputLoop(attrs, buffer)