about summary refs log tree commit diff stats
path: root/src/buffer/container.nim
diff options
context:
space:
mode:
Diffstat (limited to 'src/buffer/container.nim')
-rw-r--r--src/buffer/container.nim285
1 files changed, 141 insertions, 144 deletions
diff --git a/src/buffer/container.nim b/src/buffer/container.nim
index 2e9d9618..f17267c3 100644
--- a/src/buffer/container.nim
+++ b/src/buffer/container.nim
@@ -52,6 +52,7 @@ type
     clear*: bool
 
   Container* = ref object
+    iface: BufferInterface
     attrs*: WindowAttributes
     width*: int
     height*: int
@@ -81,6 +82,7 @@ type
     redraw*: bool
     cmdvalid: array[ContainerCommand, bool]
     needslines*: bool
+    events*: seq[ContainerEvent]
 
 proc newBuffer*(dispatcher: Dispatcher, config: Config, source: BufferSource, title = ""): Container =
   let attrs = getWindowAttributes(stdout)
@@ -240,20 +242,35 @@ func findHighlights*(container: Container, y: int): seq[Highlight] =
     if y in hl:
       result.add(hl)
 
-proc expect(container: Container, cmd: ContainerCommand) =
-  container.cmdvalid[cmd] = true
+proc triggerEvent(container: Container, event: ContainerEvent) =
+  container.events.add(event)
+
+proc triggerEvent(container: Container, t: ContainerEventType) =
+  container.triggerEvent(ContainerEvent(t: t))
+
+proc updateCursor(container: Container)
 
 proc requestLines*(container: Container, w = container.lineWindow) =
-  container.ostream.getLines(w)
-  container.expect(SET_LINES)
+  container.iface.getLines(w).then(proc(res: seq[SimpleFlexibleLine]) =
+    container.lines.setLen(w.len)
+    container.lineshift = w.a
+    for y in 0 ..< min(res.len, w.len):
+      container.lines[y] = res[y]
+    container.updateCursor()
+    let cw = container.fromy ..< container.fromy + container.height
+    if w.a in cw or w.b in cw or cw.a in w or cw.b in w:
+      container.triggerEvent(UPDATE))
 
 proc redraw*(container: Container) {.jsfunc.} =
   container.redraw = true
 
 proc sendCursorPosition*(container: Container) =
-  container.ostream.moveCursor(container.cursorx, container.cursory)
-  container.expect(SET_HOVER)
-  container.expect(RESHAPE)
+  container.iface.updateHover(container.cursorx, container.cursory).then(proc(res: UpdateHoverResult) =
+    if res.hover.isSome:
+      container.hovertext = res.hover.get
+      container.triggerEvent(STATUS)
+    if res.repaint:
+      container.needslines = true)
 
 proc setFromY*(container: Container, y: int) {.jsfunc.} =
   if container.pos.fromy != y:
@@ -504,49 +521,100 @@ proc popCursorPos*(container: Container, nojump = false) =
     container.needslines = true
 
 proc cursorNextLink*(container: Container) {.jsfunc.} =
-  container.ostream.findNextLink(container.cursorx, container.cursory)
-  container.expect(JUMP)
+  container.iface
+    .findNextLink(container.cursorx, container.cursory)
+    .then(proc(res: tuple[x, y: int]) =
+      container.setCursorXY(res.x, res.y))
 
 proc cursorPrevLink*(container: Container) {.jsfunc.} =
-  container.ostream.findPrevLink(container.cursorx, container.cursory)
-  container.expect(JUMP)
+  container.iface
+    .findPrevLink(container.cursorx, container.cursory)
+    .then(proc(res: tuple[x, y: int]) =
+      container.setCursorXY(res.x, res.y))
+
+proc clearSearchHighlights*(container: Container) =
+  for i in countdown(container.highlights.high, 0):
+    if container.highlights[i].clear:
+      container.highlights.del(i)
 
 proc cursorNextMatch*(container: Container, regex: Regex, wrap: bool) {.jsfunc.} =
-  container.ostream.findNextMatch(container.cursorx, container.cursory, regex, wrap)
-  container.expect(JUMP)
+  container.iface
+    .findNextMatch(regex, container.cursorx, container.cursory, wrap)
+    .then(proc(res: BufferMatch) =
+      if container.hlon:
+        container.setCursorXY(res.x, res.y)
+        container.clearSearchHighlights()
+        let ex = res.x + res.str.width() - 1
+        let hl = Highlight(x: res.x, y: res.y, endx: ex, endy: res.y, clear: true)
+        container.highlights.add(hl)
+        container.hlon = false)
 
 proc cursorPrevMatch*(container: Container, regex: Regex, wrap: bool) {.jsfunc.} =
-  container.ostream.findPrevMatch(container.cursorx, container.cursory, regex, wrap)
-  container.expect(JUMP)
+  container.iface
+    .findPrevMatch(regex, container.cursorx, container.cursory, wrap)
+    .then(proc(res: BufferMatch) =
+      if container.hlon:
+        container.setCursorXY(res.x, res.y)
+        container.clearSearchHighlights()
+        let ex = res.x + res.str.width() - 1
+        let hl = Highlight(x: res.x, y: res.y, endx: ex, endy: res.y, clear: true)
+        container.highlights.add(hl)
+        container.hlon = false)
+
+proc setLoadInfo(container: Container, msg: string) =
+  container.loadinfo = msg
+  container.triggerEvent(STATUS)
 
 proc load*(container: Container) =
-  container.ostream.load()
-  container.expect(LOAD_DONE)
-  container.expect(SET_LOAD_INFO)
-  container.expect(SET_NEEDS_AUTH)
-  container.expect(SET_REDIRECT)
-  container.expect(SET_CONTENT_TYPE)
-  container.expect(SET_TITLE)
-  if container.source.location.anchor != "":
-    container.expect(JUMP)
-
-proc gotoAnchor*(container: Container, anchor: string) =
-  container.ostream.findAnchor(anchor)
-  container.expect(ANCHOR_FOUND)
-  container.expect(ANCHOR_FAIL)
+  container.loadinfo = "Connecting to " & $container.source.location
+  container.iface.connect().then(proc(res: ConnectResult): auto =
+    container.code = res.code
+    if res.code != -2:
+      if res.code == 0:
+        if res.needsAuth:
+          container.triggerEvent(NEEDS_AUTH)
+        if res.redirect.isSome:
+          container.triggerEvent(REDIRECT)
+        if res.contentType != "":
+          container.contenttype = some(res.contentType)
+        container.setLoadInfo("Downloading " & $container.source.location)
+        return container.iface.load()
+      else:
+        container.triggerEvent(FAIL)
+  ).then(proc(res: tuple[success: bool, title: string]): auto =
+    if res.success:
+      container.title = res.title
+      container.setLoadInfo("Rendering " & $container.source.location)
+      return container.iface.render()
+  ).then(proc(lines: int): auto =
+    container.numLines = lines
+    container.setLoadInfo("")
+    container.needslines = true
+    return container.iface.gotoAnchor()
+  ).then(proc(res: tuple[x, y: int]) =
+    container.setCursorXY(res.x, res.y)
+  )
+
+proc findAnchor*(container: Container, anchor: string) =
+  container.iface.findAnchor(anchor).then(proc(found: bool) =
+    if found:
+      container.triggerEvent(ANCHOR)
+    else:
+      container.triggerEvent(NO_ANCHOR))
 
 proc readCanceled*(container: Container) =
-  container.ostream.readCanceled()
+  container.iface.readCanceled()
 
 proc readSuccess*(container: Container, s: string) =
-  container.ostream.readSuccess(s)
-  container.expect(OPEN)
-  container.expect(RESHAPE)
+  container.iface.readSuccess(s).then(proc(res: ReadSuccessResult) =
+    if res.reshape:
+      container.needslines = true
+    if res.open.isSome:
+      container.triggerEvent(ContainerEvent(t: OPEN, request: res.open.get)))
 
 proc reshape*(container: Container, noreq = false) {.jsfunc.} =
-  container.ostream.render()
-  container.expect(SET_NUM_LINES)
-  container.expect(JUMP)
+  container.iface.render().then(proc(lines: int) =
+    container.numLines = lines)
   if not noreq:
     container.needslines = true
 
@@ -558,30 +626,36 @@ proc dupeBuffer*(dispatcher: Dispatcher, container: Container, config: Config, l
     clonepid: container.process,
   )
   container.pipeto = dispatcher.newBuffer(config, source, container.title)
-  container.ostream.getSource()
-  container.expect(SOURCE_READY)
+  container.iface.getSource().then(proc() =
+    if container.pipeto != nil:
+      container.pipeto.load()
+      container.pipeto = nil)
   return container.pipeto
 
 proc click*(container: Container) {.jsfunc.} =
-  container.ostream.click(container.cursorx, container.cursory)
-  container.expect(OPEN)
-  container.expect(READ_LINE)
-  container.expect(RESHAPE)
+  container.iface.click(container.cursorx, container.cursory).then(proc(res: ClickResult) =
+    if res.repaint:
+      container.needslines = true
+    if res.open.isSome:
+      container.triggerEvent(ContainerEvent(t: OPEN, request: res.open.get))
+    if res.readline.isSome:
+      let rl = res.readline.get
+      container.triggerEvent(
+        ContainerEvent(
+          t: READ_LINE,
+          prompt: rl.prompt,
+          value: rl.value,
+          password: rl.hide)))
 
 proc windowChange*(container: Container, attrs: WindowAttributes) =
   container.attrs = attrs
   container.width = attrs.width
   container.height = attrs.height - 1
-  container.ostream.windowChange(attrs)
-  container.expect(RESHAPE)
-
-proc clearSearchHighlights*(container: Container) =
-  for i in countdown(container.highlights.high, 0):
-    if container.highlights[i].clear:
-      container.highlights.del(i)
+  container.iface.windowChange(attrs).then(proc() =
+    container.needslines = true)
 
 proc handleCommand(container: Container, cmd: ContainerCommand, len: int): ContainerEvent =
-  if not container.cmdvalid[cmd] and cmd != SET_LINES:
+  if not container.cmdvalid[cmd] and false or cmd notin {FULFILL_PROMISE, BUFFER_READY}:
     let len = len - sizeof(cmd)
     #TODO TODO TODO
     for i in 0 ..< len:
@@ -590,106 +664,28 @@ proc handleCommand(container: Container, cmd: ContainerCommand, len: int): Conta
       return ContainerEvent(t: INVALID_COMMAND)
   container.cmdvalid[cmd] = false
   case cmd
-  of SET_LOAD_INFO:
-    var li: LoadInfo
-    container.istream.sread(li)
-    case li
-    of CONNECT:
-      container.loadinfo = "Connecting to " & $container.source.location
-      container.expect(SET_LOAD_INFO)
-    of DOWNLOAD:
-      container.loadinfo = "Downloading " & $container.source.location
-      container.expect(SET_LOAD_INFO)
-    of RENDER:
-      container.loadinfo = "Rendering " & $container.source.location
-      container.expect(SET_LOAD_INFO)
-    of DONE:
-      container.loadinfo = ""
-    return ContainerEvent(t: STATUS)
-  of SET_LINES:
-    var w: Slice[int]
-    container.istream.sread(container.numLines)
-    container.istream.sread(w)
-    container.lines.setLen(w.len)
-    container.lineshift = w.a
-    for y in 0 ..< w.len:
-      container.istream.sread(container.lines[y])
-    container.updateCursor()
-    let cw = container.fromy ..< container.fromy + container.height
-    if w.a in cw or w.b in cw or cw.a in w or cw.b in w:
-      return ContainerEvent(t: UPDATE)
-  of SET_NUM_LINES:
-    container.istream.sread(container.numLines)
-  of SET_NEEDS_AUTH:
-    return ContainerEvent(t: NEEDS_AUTH)
-  of SET_CONTENT_TYPE:
-    var ctype: string
-    container.istream.sread(ctype, 128)
-    container.contenttype = some(ctype)
-  of SET_REDIRECT:
-    var redirect: URL
-    container.istream.sread(redirect)
-    if redirect != nil:
-      container.redirect = some(redirect)
-      return ContainerEvent(t: REDIRECT)
-  of SET_TITLE:
-    container.istream.sread(container.title)
-    return ContainerEvent(t: STATUS)
-  of SET_HOVER:
-    container.istream.sread(container.hovertext)
-    return ContainerEvent(t: STATUS)
-  of LOAD_DONE:
-    container.istream.sread(container.code)
-    if container.code == -2: return
-    if container.code != 0:
-      return ContainerEvent(t: FAIL)
-    return ContainerEvent(t: SUCCESS)
-  of ANCHOR_FOUND:
-    return ContainerEvent(t: ANCHOR)
-  of ANCHOR_FAIL:
-    return ContainerEvent(t: FAIL)
-  of READ_LINE:
-    var prompt, str: string
-    var pwd: bool
-    container.istream.sread(prompt, 1024)
-    container.istream.sread(str, 1024)
-    container.istream.sread(pwd)
-    container.cmdvalid[OPEN] = false
-    return ContainerEvent(t: READ_LINE, prompt: prompt, value: str, password: pwd)
-  of JUMP:
-    var x, y, ex: int
-    container.istream.sread(x)
-    container.istream.sread(y)
-    container.istream.sread(ex)
-    if x != -1 and y != -1:
-      if container.hlon:
-        container.clearSearchHighlights()
-        let hl = Highlight(x: x, y: y, endx: ex, endy: y, clear: true)
-        container.highlights.add(hl)
-        container.hlon = false
-      container.setCursorXY(x, y)
-  of OPEN:
-    var request: Request
-    container.istream.sread(request)
-    container.cmdvalid[READ_LINE] = false
-    return ContainerEvent(t: OPEN, request: request)
   of BUFFER_READY:
     if container.source.t == LOAD_PIPE:
-      container.ostream.passFd()
+      container.iface.passFd()
       let s = SocketStream(container.ostream)
       s.sendFileHandle(container.source.fd)
       discard close(container.source.fd)
       container.ostream.flush()
     container.load()
-  of SOURCE_READY:
-    if container.pipeto != nil:
-      container.pipeto.load()
-      container.pipeto = nil
   of RESHAPE:
     container.needslines = true
+  of FULFILL_PROMISE:
+    var packetid: int
+    container.istream.sread(packetid)
+    container.iface.fulfill(packetid, len - slen(packetid) - slen(FULFILL_PROMISE))
   if container.needslines:
-    container.expect(SET_LINES)
     container.requestLines()
+    container.needslines = false
+
+proc setStream*(container: Container, stream: Stream) =
+  container.istream = stream
+  container.ostream = stream
+  container.iface = newBufferInterface(stream)
 
 # Synchronously read all lines in the buffer.
 iterator readLines*(container: Container): SimpleFlexibleLine {.inline.} =
@@ -698,11 +694,12 @@ iterator readLines*(container: Container): SimpleFlexibleLine {.inline.} =
   var len: int
   container.istream.sread(len)
   container.istream.sread(cmd)
-  while cmd != SET_LINES:
-    discard container.handleCommand(cmd, len)
-    container.istream.sread(len)
-    container.istream.sread(cmd)
-  assert cmd == SET_LINES
+  #TODO TODO TODO
+  #while cmd != SET_LINES:
+  #  discard container.handleCommand(cmd, len)
+  #  container.istream.sread(len)
+  #  container.istream.sread(cmd)
+  #assert cmd == SET_LINES
   var w: Slice[int]
   container.istream.sread(container.numLines)
   container.istream.sread(w)