about summary refs log tree commit diff stats
diff options
context:
space:
mode:
authorbptato <nincsnevem662@gmail.com>2022-11-26 11:31:56 +0100
committerbptato <nincsnevem662@gmail.com>2022-11-26 11:36:12 +0100
commita58d2eff7f68bf98ee3b1ba1b59de6d80743f97e (patch)
treef6cc4b662e06567e4ded0409b734045e5e227b54
parent8e20aaaf7e1e73c27c664c640b49bdf0b6c96757 (diff)
downloadchawan-a58d2eff7f68bf98ee3b1ba1b59de6d80743f97e.tar.gz
Get rid of writeCommand in container
We have a type system, so let's use it.
-rw-r--r--src/buffer/buffer.nim405
-rw-r--r--src/buffer/container.nim43
2 files changed, 245 insertions, 203 deletions
diff --git a/src/buffer/buffer.nim b/src/buffer/buffer.nim
index b6acf2b0..9c016c0f 100644
--- a/src/buffer/buffer.nim
+++ b/src/buffer/buffer.nim
@@ -40,7 +40,7 @@ type
     CONNECT, DOWNLOAD, RENDER, DONE
 
   BufferCommand* = enum
-    LOAD, RENDER, WINDOW_CHANGE, GOTO_ANCHOR, READ_SUCCESS, READ_CANCELED,
+    LOAD, RENDER, WINDOW_CHANGE, FIND_ANCHOR, READ_SUCCESS, READ_CANCELED,
     CLICK, FIND_NEXT_LINK, FIND_PREV_LINK, FIND_NEXT_MATCH, FIND_PREV_MATCH,
     GET_SOURCE, GET_LINES, MOVE_CURSOR, PASS_FD
 
@@ -85,20 +85,82 @@ type
     userstyle: CSSStylesheet
 
 macro writeCommand(buffer: Buffer, cmd: ContainerCommand, args: varargs[typed]) =
-  result = newStmtList()
+  let cmdblock = newStmtList()
+  var idlist: seq[NimNode]
+  var i = 0
+  for arg in args:
+    let id = ident("arg_" & $i)
+    idlist.add(id)
+    cmdblock.add(quote do:
+      let `id` = `arg`)
+    inc i
   let lens = ident("lens")
-  let calclens = newStmtList()
-  calclens.add(quote do:
+  cmdblock.add(quote do:
     var `lens` = slen(`cmd`))
-  for arg in args:
-    calclens.add(quote do: `lens` += slen(`arg`))
-  calclens.add(quote do:
+  for id in idlist:
+    cmdblock.add(quote do: `lens` += slen(`id`))
+  cmdblock.add(quote do:
     `buffer`.postream.swrite(`lens`))
-  result.add(newBlockStmt(calclens))
-  result.add(quote do: `buffer`.postream.swrite(`cmd`))
-  for arg in args:
-    result.add(quote do: `buffer`.postream.swrite(`arg`))
-  result.add(quote do: `buffer`.postream.flush())
+  cmdblock.add(quote do: `buffer`.postream.swrite(`cmd`))
+  for id in idlist:
+    cmdblock.add(quote do: `buffer`.postream.swrite(`id`))
+  cmdblock.add(quote do: `buffer`.postream.flush())
+  return newBlockStmt(cmdblock)
+
+proc buildInterfaceProc(fun: NimNode): tuple[fun, name: NimNode] =
+  let name = fun[0] # sym
+  let params = fun[3] # formalparams
+  let retval = params[0] # sym
+  var body = newStmtList()
+  assert params.len >= 2 # return type, this value
+  var x = name.strVal.toScreamingSnakeCase()
+  if x[^1] == '=':
+    x = "SET_" & x[0..^2]
+  let nup = ident(x) # add this to enums
+  let this2 = newIdentDefs(ident("stream"), ident("Stream"))
+  let thisval = this2[0]
+  body.add(quote do:
+    `thisval`.swrite(BufferCommand.`nup`))
+  var params2: seq[NimNode]
+  params2.add(retval)
+  params2.add(this2)
+  for i in 2 ..< params.len:
+    let param = params[i]
+    for i in 0 ..< param.len - 2:
+      let id2 = newIdentDefs(ident(param[i].strVal), param[^2])
+      params2.add(id2)
+  for c in params2[2..^1]:
+    let s = c[0] # sym e.g. url
+    body.add(quote do:
+      `thisval`.swrite(`s`))
+  body.add(quote do:
+    `thisval`.flush())
+  if retval.kind != nnkEmpty:
+    body.add(quote do:
+      `thisval`.sread(result))
+  return (newProc(name, params2, body), nup)
+
+type
+  ProxyFunction = object
+    iname: NimNode # internal name
+    ename: NimNode # enum name
+    params: seq[NimNode]
+  ProxyMap = Table[string, ProxyFunction]
+
+# Name -> ProxyFunction
+var ProxyFunctions {.compileTime.}: ProxyMap
+
+macro proxy(fun: typed) =
+  let iproc = buildInterfaceProc(fun)
+  var pfun: ProxyFunction
+  pfun.iname = ident(fun[0].strVal & "_internal")
+  pfun.ename = iproc[1]
+  for x in fun[3]: pfun.params.add(x)
+  ProxyFunctions[fun[0].strVal] = pfun
+  let ifun = newProc(pfun.iname, pfun.params, fun[6])
+  result = newStmtList()
+  result.add(iproc[0])
+  result.add(ifun)
 
 func getLink(node: StyledNode): HTMLAnchorElement =
   if node == nil:
@@ -137,33 +199,7 @@ func cursorBytes(buffer: Buffer, y: int, cc: int): int =
     w += r.width()
   return i
 
-func findNextLink(buffer: Buffer, cursorx, cursory: int): tuple[x, y: int] =
-  let line = buffer.lines[cursory]
-  var i = line.findFormatN(cursorx) - 1
-  var link: Element = nil
-  if i >= 0:
-    link = line.formats[i].node.getClickable()
-  inc i
-
-  while i < line.formats.len:
-    let format = line.formats[i]
-    let fl = format.node.getClickable()
-    if fl != nil and fl != link:
-      return (format.pos, cursory)
-    inc i
-
-  for y in (cursory + 1)..(buffer.lines.len - 1):
-    let line = buffer.lines[y]
-    i = 0
-    while i < line.formats.len:
-      let format = line.formats[i]
-      let fl = format.node.getClickable()
-      if fl != nil and fl != link:
-        return (format.pos, y)
-      inc i
-  return (-1, -1)
-
-func findPrevLink(buffer: Buffer, cursorx, cursory: int): tuple[x, y: int] =
+func findPrevLink0(buffer: Buffer, cursorx, cursory: int): tuple[x, y: int] =
   let line = buffer.lines[cursory]
   var i = line.findFormatN(cursorx) - 1
   var link: Element = nil
@@ -220,59 +256,103 @@ func findPrevLink(buffer: Buffer, cursorx, cursory: int): tuple[x, y: int] =
       dec i
   return (-1, -1)
 
-proc findNextMatch(buffer: Buffer, regex: Regex, cursorx, cursory: int, wrap: bool): BufferMatch =
+func findNextLink0(buffer: Buffer, cursorx, cursory: int): tuple[x, y: int] =
+  let line = buffer.lines[cursory]
+  var i = line.findFormatN(cursorx) - 1
+  var link: Element = nil
+  if i >= 0:
+    link = line.formats[i].node.getClickable()
+  inc i
+
+  while i < line.formats.len:
+    let format = line.formats[i]
+    let fl = format.node.getClickable()
+    if fl != nil and fl != link:
+      return (format.pos, cursory)
+    inc i
+
+  for y in (cursory + 1)..(buffer.lines.len - 1):
+    let line = buffer.lines[y]
+    i = 0
+    while i < line.formats.len:
+      let format = line.formats[i]
+      let fl = format.node.getClickable()
+      if fl != nil and fl != link:
+        return (format.pos, y)
+      inc i
+  return (-1, -1)
+
+proc findPrevMatch0(buffer: Buffer, regex: Regex, cursorx, cursory: int, wrap: bool): BufferMatch =
   template return_if_match =
     if res.success and res.captures.len > 0:
-      let cap = res.captures[0]
+      let cap = res.captures[^1]
       let x = buffer.lines[y].str.width(cap.s)
       let str = buffer.lines[y].str.substr(cap.s, cap.e - 1)
       return BufferMatch(success: true, x: x, y: y, str: str)
   var y = cursory
   let b = buffer.cursorBytes(y, cursorx)
-  let b2 = if buffer.lines[y].str.len > b: b + buffer.lines[y].str.runeLenAt(b) else: b
-  let res = regex.exec(buffer.lines[y].str, b2, buffer.lines[y].str.len)
+  let b2 = if b > 0: b - buffer.lines[y].str.lastRune(b)[1] else: 0
+  let res = regex.exec(buffer.lines[y].str, 0, b2)
   return_if_match
-  inc y
+  dec y
   while true:
-    if y > buffer.lines.high:
+    if y < 0:
       if wrap:
-        y = 0
+        y = buffer.lines.high
       else:
         break
     if y == cursory:
-      let res = regex.exec(buffer.lines[y].str, 0, b)
+      let res = regex.exec(buffer.lines[y].str, b, buffer.lines[y].str.len)
       return_if_match
       break
     let res = regex.exec(buffer.lines[y].str)
     return_if_match
-    inc y
+    dec y
 
-proc findPrevMatch(buffer: Buffer, regex: Regex, cursorx, cursory: int, wrap: bool): BufferMatch =
+proc findNextMatch0(buffer: Buffer, regex: Regex, cursorx, cursory: int, wrap: bool): BufferMatch =
   template return_if_match =
     if res.success and res.captures.len > 0:
-      let cap = res.captures[^1]
+      let cap = res.captures[0]
       let x = buffer.lines[y].str.width(cap.s)
       let str = buffer.lines[y].str.substr(cap.s, cap.e - 1)
       return BufferMatch(success: true, x: x, y: y, str: str)
   var y = cursory
   let b = buffer.cursorBytes(y, cursorx)
-  let b2 = if b > 0: b - buffer.lines[y].str.lastRune(b)[1] else: 0
-  let res = regex.exec(buffer.lines[y].str, 0, b2)
+  let b2 = if buffer.lines[y].str.len > b: b + buffer.lines[y].str.runeLenAt(b) else: b
+  let res = regex.exec(buffer.lines[y].str, b2, buffer.lines[y].str.len)
   return_if_match
-  dec y
+  inc y
   while true:
-    if y < 0:
+    if y > buffer.lines.high:
       if wrap:
-        y = buffer.lines.high
+        y = 0
       else:
         break
     if y == cursory:
-      let res = regex.exec(buffer.lines[y].str, b, buffer.lines[y].str.len)
+      let res = regex.exec(buffer.lines[y].str, 0, b)
       return_if_match
       break
     let res = regex.exec(buffer.lines[y].str)
     return_if_match
-    dec y
+    inc y
+
+proc findPrevLink*(buffer: Buffer, cursorx, cursory: int) {.proxy.} =
+  let pl = buffer.findPrevLink0(cursorx, cursory)
+  buffer.writeCommand(JUMP, pl.x, pl.y, 0)
+
+proc findNextLink*(buffer: Buffer, cursorx, cursory: int) {.proxy.} =
+  let pl = buffer.findNextLink0(cursorx, cursory)
+  buffer.writeCommand(JUMP, pl.x, pl.y, 0)
+
+proc findPrevMatch*(buffer: Buffer, cursorx, cursory: int, regex: Regex, wrap: bool) {.proxy.} =
+  let match = buffer.findPrevMatch0(regex, cursorx, cursory, wrap)
+  if match.success:
+    buffer.writeCommand(JUMP, match.x, match.y, match.x + match.str.width() - 1)
+
+proc findNextMatch*(buffer: Buffer, cursorx, cursory: int, regex: Regex, wrap: bool) {.proxy.} =
+  let match = buffer.findNextMatch0(regex, cursorx, cursory, wrap)
+  if match.success:
+    buffer.writeCommand(JUMP, match.x, match.y, match.x + match.str.width() - 1)
 
 proc gotoAnchor(buffer: Buffer) =
   if buffer.document == nil: return
@@ -286,7 +366,8 @@ proc gotoAnchor(buffer: Buffer) =
         buffer.writeCommand(JUMP, format.pos, y, 0)
         return
 
-proc windowChange(buffer: Buffer) =
+proc windowChange*(buffer: Buffer, attrs: WindowAttributes) {.proxy.} =
+  buffer.attrs = attrs
   buffer.viewport = Viewport(window: buffer.attrs)
   buffer.width = buffer.attrs.width
   buffer.height = buffer.attrs.height - 1
@@ -322,6 +403,9 @@ proc updateHover(buffer: Buffer, cursorx, cursory: int) =
 
   buffer.prevnode = thisnode
 
+proc moveCursor*(buffer: Buffer, cursorx, cursory: int) {.proxy.} =
+  buffer.updateHover(cursorx, cursory)
+
 proc loadResource(buffer: Buffer, document: Document, elem: HTMLLinkElement) =
   let url = parseUrl(elem.href, document.location.some)
   if url.isSome:
@@ -400,9 +484,8 @@ proc setupSource(buffer: Buffer): int =
   buffer.selector.registerHandle(cast[int](buffer.getFd()), {Read}, 1)
   buffer.loaded = true
 
-proc load(buffer: Buffer) =
-  case buffer.contenttype
-  of "text/html":
+proc load0(buffer: Buffer) =
+  if buffer.contenttype == "text/html":
     if not buffer.streamclosed:
       buffer.source = buffer.istream.readAll()
       buffer.istream.close()
@@ -414,15 +497,16 @@ proc load(buffer: Buffer) =
     buffer.writeCommand(SET_TITLE, buffer.document.title)
     buffer.document.location = buffer.location
     buffer.loadResources(buffer.document)
-  else:
-    discard
-    #if not buffer.streamclosed:
-    #  if buffer.bsource.t != LOAD_PIPE:
-    #    buffer.source = buffer.istream.readAll()
-    #    buffer.istream.close()
-    #    buffer.streamclosed = true
-
-proc render(buffer: Buffer) =
+
+proc load*(buffer: Buffer) {.proxy.} =
+  buffer.writeCommand(SET_LOAD_INFO, CONNECT)
+  let code = buffer.setupSource()
+  if code != -2:
+    buffer.writeCommand(SET_LOAD_INFO, DOWNLOAD)
+    buffer.load0()
+  buffer.writeCommand(LOAD_DONE, code)
+
+proc do_reshape(buffer: Buffer) =
   case buffer.contenttype
   of "text/html":
     if buffer.viewport == nil:
@@ -433,6 +517,13 @@ proc render(buffer: Buffer) =
   else:
     buffer.lines = renderPlainText(buffer.source)
 
+proc render*(buffer: Buffer) {.proxy.} =
+  buffer.writeCommand(SET_LOAD_INFO, LoadInfo.RENDER)
+  buffer.do_reshape()
+  buffer.writeCommand(SET_LOAD_INFO, DONE)
+  buffer.writeCommand(SET_NUM_LINES, buffer.lines.len)
+  buffer.gotoAnchor()
+
 proc load2(buffer: Buffer) =
   case buffer.contenttype
   of "text/html":
@@ -442,7 +533,8 @@ proc load2(buffer: Buffer) =
     # (We're basically recv'ing single bytes, but nim std/net does buffering
     # for us so we should be ok?)
     if not buffer.streamclosed:
-      buffer.source &= buffer.istream.readChar()
+      let c = buffer.istream.readChar()
+      buffer.source &= c
       buffer.reshape = true
 
 proc finishLoad(buffer: Buffer) =
@@ -634,7 +726,7 @@ template restore_focus =
     buffer.document.focus = nil
     buffer.reshape = true
 
-proc lineInput(buffer: Buffer, s: string) =
+proc readSuccess*(buffer: Buffer, s: string) {.proxy.} =
   if buffer.input != nil:
     let input = buffer.input
     case input.inputType
@@ -660,7 +752,7 @@ proc lineInput(buffer: Buffer, s: string) =
     else: discard
     buffer.input = nil
 
-proc click(buffer: Buffer, cursorx, cursory: int) =
+proc click*(buffer: Buffer, cursorx, cursory: int) {.proxy.} =
   let clickable = buffer.getCursorClickable(cursorx, cursory)
   if clickable != nil:
     case clickable.tagType
@@ -726,113 +818,72 @@ proc click(buffer: Buffer, cursorx, cursory: int) =
     else:
       restore_focus
 
+proc readCanceled*(buffer: Buffer) {.proxy.} =
+  buffer.input = nil
+
+proc findAnchor*(buffer: Buffer, anchor: string) {.proxy.} =
+  if buffer.document != nil and buffer.document.getElementById(anchor) != nil:
+    buffer.writeCommand(ANCHOR_FOUND)
+  else:
+    buffer.writeCommand(ANCHOR_FAIL)
+
+proc getLines*(buffer: Buffer, w: Slice[int]) {.proxy.} =
+  var w = w
+  if w.b < 0 or w.b > buffer.lines.high:
+    w.b = buffer.lines.high
+  var lens = sizeof(SET_LINES) + sizeof(buffer.lines.len) + sizeof(w)
+  for y in w:
+    lens += slen(buffer.lines[y])
+  buffer.postream.swrite(lens)
+  buffer.postream.swrite(SET_LINES)
+  buffer.postream.swrite(buffer.lines.len)
+  buffer.postream.swrite(w)
+  for y in w:
+    buffer.postream.swrite(buffer.lines[y])
+  buffer.postream.flush()
+
+proc passFd*(buffer: Buffer) {.proxy.} =
+  let fd = SocketStream(buffer.pistream).recvFileHandle()
+  buffer.bsource.fd = fd
+
+proc getSource*(buffer: Buffer) {.proxy.} =
+  let ssock = initServerSocket()
+  buffer.writeCommand(SOURCE_READY)
+  let stream = ssock.acceptSocketStream()
+  if not buffer.streamclosed:
+    buffer.source = buffer.istream.readAll()
+    buffer.streamclosed = true
+  stream.write(buffer.source)
+  stream.close()
+  ssock.close()
+
+macro bufferDispatcher(funs: static ProxyMap, buffer: Buffer, cmd: BufferCommand) =
+  let switch = newNimNode(nnkCaseStmt)
+  switch.add(ident("cmd"))
+  for k, v in funs:
+    let ofbranch = newNimNode(nnkOfBranch)
+    ofbranch.add(v.ename)
+    let stmts = newStmtList()
+    let call = newCall(v.iname, buffer)
+    for i in 2 ..< v.params.len:
+      let param = v.params[i]
+      for i in 0 ..< param.len - 2:
+        let id = ident(param[i].strVal)
+        let typ = param[^2]
+        stmts.add(quote do:
+          var `id`: `typ`
+          `buffer`.pistream.sread(`id`))
+        call.add(id)
+    stmts.add(call)
+    ofbranch.add(stmts)
+    switch.add(ofbranch)
+  return switch
+
 proc readCommand(buffer: Buffer) =
   let istream = buffer.pistream
-  let ostream = buffer.postream
   var cmd: BufferCommand
   istream.sread(cmd)
-  case cmd
-  of PASS_FD:
-    let fd = SocketStream(istream).recvFileHandle()
-    buffer.bsource.fd = fd
-  of LOAD:
-    buffer.writeCommand(SET_LOAD_INFO, CONNECT)
-    let code = buffer.setupSource()
-    if code != -2:
-      buffer.writeCommand(SET_LOAD_INFO, DOWNLOAD)
-      buffer.load()
-    buffer.writeCommand(LOAD_DONE, code)
-  of GOTO_ANCHOR:
-    var anchor: string
-    istream.sread(anchor)
-    if buffer.document != nil and buffer.document.getElementById(anchor) != nil:
-      buffer.writeCommand(ANCHOR_FOUND)
-    else:
-      buffer.writeCommand(ANCHOR_FAIL)
-  of RENDER:
-    buffer.writeCommand(SET_LOAD_INFO, LoadInfo.RENDER)
-    buffer.render()
-    buffer.writeCommand(SET_LOAD_INFO, DONE)
-    buffer.writeCommand(SET_NUM_LINES, buffer.lines.len)
-    buffer.gotoAnchor()
-  of GET_LINES:
-    var w: Slice[int]
-    istream.sread(w)
-    if w.b < 0 or w.b > buffer.lines.high:
-      w.b = buffer.lines.high
-    var lens = sizeof(SET_LINES) + sizeof(buffer.lines.len) + sizeof(w)
-    for y in w:
-      lens += slen(buffer.lines[y])
-    ostream.swrite(lens)
-    ostream.swrite(SET_LINES)
-    ostream.swrite(buffer.lines.len)
-    ostream.swrite(w)
-    for y in w:
-      ostream.swrite(buffer.lines[y])
-    ostream.flush()
-  of WINDOW_CHANGE:
-    istream.sread(buffer.attrs)
-    buffer.windowChange()
-  of FIND_PREV_LINK:
-    var cx, cy: int
-    istream.sread(cx)
-    istream.sread(cy)
-    let pl = buffer.findPrevLink(cx, cy)
-    buffer.writeCommand(JUMP, pl.x, pl.y, 0)
-  of FIND_NEXT_LINK:
-    var cx, cy: int
-    istream.sread(cx)
-    istream.sread(cy)
-    let nl = buffer.findNextLink(cx, cy)
-    buffer.writeCommand(JUMP, nl.x, nl.y, 0)
-  of FIND_PREV_MATCH:
-    var cx, cy: int
-    var regex: Regex
-    var wrap: bool
-    istream.sread(cx)
-    istream.sread(cy)
-    istream.sread(regex)
-    istream.sread(wrap)
-    let match = buffer.findPrevMatch(regex, cx, cy, wrap)
-    if match.success:
-      buffer.writeCommand(JUMP, match.x, match.y, match.x + match.str.width() - 1)
-  of FIND_NEXT_MATCH:
-    var cx, cy: int
-    var regex: Regex
-    var wrap: bool
-    istream.sread(cx)
-    istream.sread(cy)
-    istream.sread(regex)
-    istream.sread(wrap)
-    let match = buffer.findNextMatch(regex, cx, cy, wrap)
-    if match.success:
-      buffer.writeCommand(JUMP, match.x, match.y, match.x + match.str.width() - 1)
-  of READ_SUCCESS:
-    var s: string
-    istream.sread(s)
-    buffer.lineInput(s)
-  of READ_CANCELED:
-    buffer.input = nil
-  of CLICK:
-    var cx, cy: int
-    istream.sread(cx)
-    istream.sread(cy)
-    buffer.click(cx, cy)
-  of MOVE_CURSOR:
-    var cx, cy: int
-    istream.sread(cx)
-    istream.sread(cy)
-    buffer.updateHover(cx, cy)
-  of GET_SOURCE:
-    let ssock = initServerSocket()
-    buffer.writeCommand(SOURCE_READY)
-    let stream = ssock.acceptSocketStream()
-    if not buffer.streamclosed:
-      buffer.source = buffer.istream.readAll()
-      buffer.streamclosed = true
-    stream.write(buffer.source)
-    stream.close()
-    ssock.close()
+  bufferDispatcher(ProxyFunctions, buffer, cmd)
 
 proc runBuffer(buffer: Buffer, rfd: int) =
   block loop:
@@ -856,7 +907,7 @@ proc runBuffer(buffer: Buffer, rfd: int) =
         break loop
       if buffer.reshape:
         buffer.reshape = false
-        buffer.render()
+        buffer.do_reshape()
         buffer.writeCommand(RESHAPE)
   buffer.pistream.close()
   buffer.postream.close()
diff --git a/src/buffer/container.nim b/src/buffer/container.nim
index 20f04f62..2e9d9618 100644
--- a/src/buffer/container.nim
+++ b/src/buffer/container.nim
@@ -1,4 +1,3 @@
-import macros
 import options
 import streams
 import strformat
@@ -241,25 +240,18 @@ func findHighlights*(container: Container, y: int): seq[Highlight] =
     if y in hl:
       result.add(hl)
 
-macro writeCommand(container: Container, cmd: BufferCommand, args: varargs[typed]) =
-  result = newStmtList()
-  result.add(quote do: `container`.ostream.swrite(`cmd`))
-  for arg in args:
-    result.add(quote do: `container`.ostream.swrite(`arg`))
-  result.add(quote do: `container`.ostream.flush())
-
 proc expect(container: Container, cmd: ContainerCommand) =
   container.cmdvalid[cmd] = true
 
 proc requestLines*(container: Container, w = container.lineWindow) =
-  container.writeCommand(GET_LINES, w)
+  container.ostream.getLines(w)
   container.expect(SET_LINES)
 
 proc redraw*(container: Container) {.jsfunc.} =
   container.redraw = true
 
 proc sendCursorPosition*(container: Container) =
-  container.writeCommand(MOVE_CURSOR, container.cursorx, container.cursory)
+  container.ostream.moveCursor(container.cursorx, container.cursory)
   container.expect(SET_HOVER)
   container.expect(RESHAPE)
 
@@ -512,23 +504,23 @@ proc popCursorPos*(container: Container, nojump = false) =
     container.needslines = true
 
 proc cursorNextLink*(container: Container) {.jsfunc.} =
-  container.writeCommand(FIND_NEXT_LINK, container.cursorx, container.cursory)
+  container.ostream.findNextLink(container.cursorx, container.cursory)
   container.expect(JUMP)
 
 proc cursorPrevLink*(container: Container) {.jsfunc.} =
-  container.writeCommand(FIND_PREV_LINK, container.cursorx, container.cursory)
+  container.ostream.findPrevLink(container.cursorx, container.cursory)
   container.expect(JUMP)
 
 proc cursorNextMatch*(container: Container, regex: Regex, wrap: bool) {.jsfunc.} =
-  container.writeCommand(FIND_NEXT_MATCH, container.cursorx, container.cursory, regex, wrap)
+  container.ostream.findNextMatch(container.cursorx, container.cursory, regex, wrap)
   container.expect(JUMP)
 
 proc cursorPrevMatch*(container: Container, regex: Regex, wrap: bool) {.jsfunc.} =
-  container.writeCommand(FIND_PREV_MATCH, container.cursorx, container.cursory, regex, wrap)
+  container.ostream.findPrevMatch(container.cursorx, container.cursory, regex, wrap)
   container.expect(JUMP)
 
 proc load*(container: Container) =
-  container.writeCommand(LOAD)
+  container.ostream.load()
   container.expect(LOAD_DONE)
   container.expect(SET_LOAD_INFO)
   container.expect(SET_NEEDS_AUTH)
@@ -539,21 +531,20 @@ proc load*(container: Container) =
     container.expect(JUMP)
 
 proc gotoAnchor*(container: Container, anchor: string) =
-  container.writeCommand(GOTO_ANCHOR, anchor)
+  container.ostream.findAnchor(anchor)
   container.expect(ANCHOR_FOUND)
   container.expect(ANCHOR_FAIL)
 
 proc readCanceled*(container: Container) =
-  container.writeCommand(READ_CANCELED)
+  container.ostream.readCanceled()
 
 proc readSuccess*(container: Container, s: string) =
-  container.writeCommand(READ_SUCCESS, s)
+  container.ostream.readSuccess(s)
   container.expect(OPEN)
   container.expect(RESHAPE)
 
 proc reshape*(container: Container, noreq = false) {.jsfunc.} =
-  container.writeCommand(RENDER)
-  container.expect(RESHAPE)
+  container.ostream.render()
   container.expect(SET_NUM_LINES)
   container.expect(JUMP)
   if not noreq:
@@ -567,12 +558,12 @@ proc dupeBuffer*(dispatcher: Dispatcher, container: Container, config: Config, l
     clonepid: container.process,
   )
   container.pipeto = dispatcher.newBuffer(config, source, container.title)
-  container.writeCommand(GET_SOURCE)
+  container.ostream.getSource()
   container.expect(SOURCE_READY)
   return container.pipeto
 
 proc click*(container: Container) {.jsfunc.} =
-  container.writeCommand(CLICK, container.cursorx, container.cursory)
+  container.ostream.click(container.cursorx, container.cursory)
   container.expect(OPEN)
   container.expect(READ_LINE)
   container.expect(RESHAPE)
@@ -581,7 +572,7 @@ proc windowChange*(container: Container, attrs: WindowAttributes) =
   container.attrs = attrs
   container.width = attrs.width
   container.height = attrs.height - 1
-  container.writeCommand(WINDOW_CHANGE, attrs)
+  container.ostream.windowChange(attrs)
   container.expect(RESHAPE)
 
 proc clearSearchHighlights*(container: Container) =
@@ -590,7 +581,7 @@ proc clearSearchHighlights*(container: Container) =
       container.highlights.del(i)
 
 proc handleCommand(container: Container, cmd: ContainerCommand, len: int): ContainerEvent =
-  if not container.cmdvalid[cmd]:
+  if not container.cmdvalid[cmd] and cmd != SET_LINES:
     let len = len - sizeof(cmd)
     #TODO TODO TODO
     for i in 0 ..< len:
@@ -684,8 +675,7 @@ proc handleCommand(container: Container, cmd: ContainerCommand, len: int): Conta
     return ContainerEvent(t: OPEN, request: request)
   of BUFFER_READY:
     if container.source.t == LOAD_PIPE:
-      container.ostream.swrite(PASS_FD)
-      container.ostream.flush()
+      container.ostream.passFd()
       let s = SocketStream(container.ostream)
       s.sendFileHandle(container.source.fd)
       discard close(container.source.fd)
@@ -698,6 +688,7 @@ proc handleCommand(container: Container, cmd: ContainerCommand, len: int): Conta
   of RESHAPE:
     container.needslines = true
   if container.needslines:
+    container.expect(SET_LINES)
     container.requestLines()
 
 # Synchronously read all lines in the buffer.