about summary refs log tree commit diff stats
diff options
context:
space:
mode:
authorbptato <nincsnevem662@gmail.com>2024-06-21 20:33:47 +0200
committerbptato <nincsnevem662@gmail.com>2024-06-21 20:33:47 +0200
commitca295c18cda7bbdffb42e65729f4dd969fe16d69 (patch)
tree20388292b6dba2f87f14d2acb619d975f9b4835c
parent7ffce10055c6b553781e0b747506f6a3a50718a6 (diff)
downloadchawan-ca295c18cda7bbdffb42e65729f4dd969fe16d69.tar.gz
term, pager: improve image display
* basic repaint algorithm for sixel (instead of brute force "clear the
  whole screen")
* do not re-send kitty images already on the screen
-rw-r--r--src/local/container.nim14
-rw-r--r--src/local/pager.nim104
-rw-r--r--src/local/term.nim190
-rw-r--r--src/server/buffer.nim3
-rw-r--r--src/types/color.nim14
-rw-r--r--todo1
6 files changed, 224 insertions, 102 deletions
diff --git a/src/local/container.nim b/src/local/container.nim
index 7aa5d324..52498047 100644
--- a/src/local/container.nim
+++ b/src/local/container.nim
@@ -101,6 +101,10 @@ type
   ContainerFlag* = enum
     cfCloned, cfUserRequested, cfHasStart, cfCanReinterpret, cfSave, cfIsHTML
 
+  CachedImage* = ref object
+    loaded*: bool
+    bmp*: NetworkBitmap
+
   Container* = ref object
     # note: this is not the same as source.request.url (but should be synced
     # with buffer.url)
@@ -157,7 +161,7 @@ type
     mainConfig*: Config
     flags*: set[ContainerFlag]
     images*: seq[PosBitmap]
-    cachedImages*: seq[NetworkBitmap]
+    cachedImages*: seq[CachedImage]
     luctx: LUContext
 
 jsDestructor(Highlight)
@@ -1741,10 +1745,10 @@ proc highlightMarks*(container: Container; display: var FixedGrid;
       hlformat.bgcolor = hlcolor
       display[y * display.width + x].format = hlformat
 
-func findCachedImage*(container: Container; id: int): NetworkBitmap =
-  for bmp in container.cachedImages:
-    if bmp.imageId == id:
-      return bmp
+func findCachedImage*(container: Container; id: int): CachedImage =
+  for image in container.cachedImages:
+    if image.bmp.imageId == id:
+      return image
   return nil
 
 proc handleEvent*(container: Container) =
diff --git a/src/local/pager.nim b/src/local/pager.nim
index a70b2b3d..073b941c 100644
--- a/src/local/pager.nim
+++ b/src/local/pager.nim
@@ -126,6 +126,7 @@ type
     forkserver*: ForkServer
     formRequestMap*: Table[string, FormRequestType]
     hasload*: bool # has a page been successfully loaded since startup?
+    imageId: int # hack to allocate a new ID for canvas each frame, TODO remove
     inputBuffer*: string # currently uninterpreted characters
     iregex: Result[Regex, string]
     isearchpromise: EmptyPromise
@@ -366,6 +367,7 @@ proc writeStatusMessage(pager: Pager; str: string; format = Format();
   let e = min(start + maxwidth, pager.statusgrid.width)
   if i >= e:
     return i
+  pager.redraw = true
   for r in str.runes:
     let w = r.width()
     if i + w >= e:
@@ -456,9 +458,66 @@ proc redraw(pager: Pager) {.jsfunc.} =
   pager.redraw = true
   pager.term.clearCanvas()
 
+proc loadCachedImage(pager: Pager; container: Container; bmp: NetworkBitmap) =
+  #TODO this is kinda dumb, because we cannot unload cached images.
+  # ideally the filesystem cache should serve as the only cache, but right
+  # now it's just sort of a temporary place before the image is dumped to
+  # memory.
+  # maybe allow the buffer to add a cache file? or receive a separate "image
+  # load start" event in container, and then add one in the pager?
+  # the first option seems better; it's simpler, and buffers can add arbitrary
+  # cache files if they just tell the pager it's an image anyway.
+  let cacheId = pager.loader.addCacheFile(bmp.outputId,
+    pager.loader.clientPid, container.process)
+  let request = newRequest(newURL("cache:" & $cacheId).get)
+  let cachedImage = CachedImage(bmp: bmp)
+  pager.loader.fetch(request).then(proc(res: JSResult[Response]): EmptyPromise =
+    if res.isNone:
+      let i = container.cachedImages.find(cachedImage)
+      container.cachedImages.del(i)
+      return nil
+    return res.get.saveToBitmap(bmp)
+  ).then(proc() =
+    pager.redraw = true
+    cachedImage.loaded = true
+    pager.loader.removeCachedItem(cacheId)
+  )
+  pager.loader.resume(bmp.outputId) # get rid of dangling output
+  container.cachedImages.add(cachedImage)
+
+proc initImages(pager: Pager; container: Container) =
+  var newImages: seq[CanvasImage] = @[]
+  for image in container.images:
+    var imageId = -1
+    if image.bmp of NetworkBitmap:
+      # add cache file to pager, but source it from the container.
+      let bmp = NetworkBitmap(image.bmp)
+      let cached = container.findCachedImage(bmp.imageId)
+      imageId = bmp.imageId
+      if cached == nil:
+        pager.loadCachedImage(container, bmp)
+        continue
+      image.bmp = cached.bmp
+      if not cached.loaded:
+        continue # loading
+    else:
+      imageId = pager.imageId
+      inc pager.imageId
+    let canvasImage = pager.term.loadImage(image.bmp, container.process, imageId,
+      image.x - container.fromx, image.y - container.fromy, pager.attrs.width,
+      pager.attrs.height - 1)
+    if canvasImage != nil:
+      newImages.add(canvasImage)
+      canvasImage.marked = true
+  if pager.term.imageMode == imKitty:
+    for image in pager.term.canvasImages:
+      if not image.marked:
+        pager.term.imagesToClear.add(image)
+      image.marked = false
+  pager.term.canvasImages = newImages
+
 proc draw*(pager: Pager) =
   let container = pager.container
-  pager.term.hideCursor()
   if container != nil:
     if pager.redraw:
       pager.refreshDisplay()
@@ -472,39 +531,17 @@ proc draw*(pager: Pager) =
     if pager.lineedit.get.invalid:
       let x = pager.lineedit.get.generateOutput()
       pager.term.writeGrid(x, 0, pager.attrs.height - 1)
+      pager.lineedit.get.invalid = false
+      pager.redraw = true
   else:
     pager.term.writeGrid(pager.statusgrid, 0, pager.attrs.height - 1)
-  pager.term.outputGrid()
-  if container != nil and pager.redraw and pager.term.imageMode != imNone:
-    pager.term.clearImages()
-    for image in container.images:
-      if image.bmp of NetworkBitmap:
-        # add cache file to pager, but source it from the container.
-        let bmp = NetworkBitmap(image.bmp)
-        let cached = container.findCachedImage(bmp.imageId)
-        if cached == nil:
-          let cacheId = pager.loader.addCacheFile(bmp.outputId,
-            pager.loader.clientPid, container.process)
-          let request = newRequest(newURL("cache:" & $cacheId).get)
-          # capture bmp for the closure
-          (proc(bmp: Bitmap) =
-            pager.loader.fetch(request).then(proc(res: JSResult[Response]):
-                EmptyPromise =
-              if res.isNone:
-                return nil
-              return res.get.saveToBitmap(bmp)
-            ).then(proc() =
-              pager.redraw = true
-            )
-          )(bmp)
-          pager.loader.resume(bmp.outputId) # get rid of dangling output
-          container.cachedImages.add(bmp)
-          continue
-        image.bmp = cached
-      if uint64(image.bmp.px.len) < image.bmp.width * image.bmp.height:
-        continue # loading
-      pager.term.outputImage(image.bmp, image.x - container.fromx,
-        image.y - container.fromy, pager.attrs.width, pager.attrs.height - 1)
+  if pager.redraw:
+    if container != nil and pager.term.imageMode != imNone:
+      pager.initImages(container)
+    pager.term.hideCursor()
+    pager.term.outputGrid()
+    if pager.term.imageMode != imNone:
+      pager.term.outputImages()
   if pager.askpromise != nil:
     pager.term.setCursor(pager.askcursor, pager.attrs.height - 1)
   elif pager.lineedit.isSome:
@@ -516,7 +553,8 @@ proc draw*(pager: Pager) =
         container.select.getCursorY())
     else:
       pager.term.setCursor(pager.container.acursorx, pager.container.acursory)
-  pager.term.showCursor()
+  if pager.redraw:
+    pager.term.showCursor()
   pager.term.flush()
   pager.redraw = false
 
diff --git a/src/local/term.nim b/src/local/term.nim
index 3c8621a1..e4fc7429 100644
--- a/src/local/term.nim
+++ b/src/local/term.nim
@@ -54,6 +54,20 @@ type
     caps: array[TermcapCap, cstring]
     numCaps: array[TermcapCapNumeric, cint]
 
+  CanvasImage* = ref object
+    pid: int
+    imageId: int
+    x: int
+    y: int
+    offx: int
+    offy: int
+    dispw: int
+    disph: int
+    damaged: bool
+    marked*: bool
+    kittyId: int
+    bmp: Bitmap
+
   Terminal* = ref TerminalObj
   TerminalObj = object
     cs*: Charset
@@ -62,6 +76,8 @@ type
     outfile: File
     cleared: bool
     canvas: seq[FixedCell]
+    canvasImages*: seq[CanvasImage]
+    imagesToClear*: seq[CanvasImage]
     lineDamage: seq[int]
     attrs*: WindowAttributes
     colorMode: ColorMode
@@ -77,8 +93,8 @@ type
     defaultBackground: RGBColor
     defaultForeground: RGBColor
     ibuf*: string # buffer for chars when we can't process them
-    hasSixel: bool
     sixelRegisterNum: int
+    kittyId: int # counter for kitty image (*not* placement) ids.
 
 # control sequence introducer
 template CSI(s: varargs[string, `$`]): string =
@@ -461,10 +477,9 @@ proc processOutputString*(term: Terminal; str: string; w: var int): string =
   if term.cs == CHARSET_UTF_8:
     # The output encoding matches the internal representation.
     return str
-  else:
-    # Output is not utf-8, so we must encode it first.
-    var success = false
-    return newTextEncoder(term.cs).encodeAll(str, success)
+  # Output is not utf-8, so we must encode it first.
+  var success = false
+  return newTextEncoder(term.cs).encodeAll(str, success)
 
 proc generateFullOutput(term: Terminal): string =
   var format = Format()
@@ -591,21 +606,59 @@ proc outputGrid*(term: Terminal) =
   if term.config.display.force_clear or not term.cleared:
     term.outfile.write(term.generateFullOutput())
     term.cleared = true
-    term.hasSixel = false
   else:
     term.outfile.write(term.generateSwapOutput())
 
-proc clearImages*(term: Terminal) =
-  #TODO this entire function is a hack:
-  # * for kitty, we shouldn't destroy & re-write every image every frame
-  # * for sixel, we shouldn't practically set force-clear when images are on
-  #   the screen
-  case term.imageMode
-  of imNone: discard
-  of imKitty: term.write(APC & "Ga=d" & ST)
-  of imSixel:
-    if term.hasSixel:
-      term.cleared = false
+func findImage(term: Terminal; pid, imageId: int): CanvasImage =
+  for it in term.canvasImages:
+    if it.pid == pid and it.imageId == imageId:
+      return it
+  return nil
+
+# x, y, maxw, maxh in cells
+# x, y can be negative, then image starts outside the screen
+proc positionImage(term: Terminal; image: CanvasImage; x, y, maxw, maxh: int):
+    bool =
+  image.x = x
+  image.y = y
+  let xpx = x * term.attrs.ppc
+  let ypx = y * term.attrs.ppl
+  # calculate offset inside image to start from
+  image.offx = -min(xpx, 0)
+  image.offy = -min(ypx, 0)
+  # calculate maximum image size that fits on the screen relative to the image
+  # origin (*not* offx/offy)
+  let maxwpx = maxw * term.attrs.ppc
+  let maxhpx = maxh * term.attrs.ppl
+  image.dispw = min(int(image.bmp.width) + xpx, maxwpx) - xpx
+  image.disph = min(int(image.bmp.height) + ypx, maxhpx) - ypx
+  image.damaged = true
+  return image.dispw > 0 and image.disph > 0
+
+proc loadImage*(term: Terminal; bmp: Bitmap; pid, imageId, x, y, maxw,
+    maxh: int): CanvasImage =
+  if (let image = term.findImage(pid, imageId); image != nil):
+    # reuse image on screen
+    if image.x != x or image.y != y:
+      # with sixel, we must clear the image currently on the screen the same way
+      # as we clear text.
+      if term.imageMode == imSixel:
+        var y = max(image.y, 0)
+        let ey = min(image.y + int(image.bmp.height), maxh)
+        let x = image.x
+        while y < ey:
+          term.lineDamage[y] = min(x, term.lineDamage[y])
+          inc y
+      if not term.positionImage(image, x, y, maxw, maxh):
+        # no longer on screen
+        return nil
+    return image
+  # new image
+  let image = CanvasImage(bmp: bmp, pid: pid, imageId: imageId)
+  if term.positionImage(image, x, y, maxw, maxh):
+    return image
+  # no longer on screen
+  return nil
 
 # data is binary 0..63; the output is the final ASCII form.
 proc compressSixel(data: openArray[uint8]): string =
@@ -638,8 +691,12 @@ func find(bands: seq[SixelBand]; c: uint8): int =
       return i
   -1
 
-proc outputSixelImage(term: Terminal; x, y, offx, offy, dispw, disph: int;
-    bmp: Bitmap) =
+proc outputSixelImage(term: Terminal; x, y: int; image: CanvasImage) =
+  let offx = image.offx
+  let offy = image.offy
+  let dispw = image.dispw
+  let disph = image.disph
+  let bmp = image.bmp
   var outs = term.cursorGoto(x, y)
   outs &= DCSSTART & 'q'
   # set raster attributes
@@ -666,10 +723,13 @@ proc outputSixelImage(term: Terminal; x, y, offx, offy, dispw, disph: int;
       let my = m div term.attrs.ppl
       for bx in 0 ..< realw:
         let cx = (cx0 + bx) div term.attrs.ppc
-        let bgcolor0 = term.canvas[my + cx].format.bgcolor
-        let bgcolor = bgcolor0.getRGB(term.defaultBackground)
-        let c0 = bmp.px[n + bx + offx]
-        let c = uint8(RGBColor(bgcolor.blend(c0)).toEightBit())
+        var c0 = bmp.px[n + bx + offx]
+        if c0.a != 255:
+          let bgcolor0 = term.canvas[my + cx].format.bgcolor
+          let bgcolor = bgcolor0.getRGB(term.defaultBackground)
+          let c1 = bgcolor.blend(c0)
+          c0 = RGBAColorBE(r: c1.r, g: c1.g, b: c1.b, a: c1.a)
+        let c = uint8(c0.toEightBit())
         if (let j = bands.find(c); j != -1):
           bands[j].data[bx] = bands[j].data[bx] or mask
         else:
@@ -687,19 +747,29 @@ proc outputSixelImage(term: Terminal; x, y, offx, offy, dispw, disph: int;
     outs.setLen(outs.len - 1)
   outs &= ST
   term.write(outs)
-  term.hasSixel = true
 
-proc outputKittyImage(term: Terminal; x, y, offx, offy, dispw, disph: int;
-    bmp: Bitmap) =
+proc outputKittyImage(term: Terminal; x, y: int; image: CanvasImage) =
+  var outs = term.cursorGoto(x, y) &
+    APC & "GC=1,s=" & $image.bmp.width & ",v=" & $image.bmp.height &
+    ",x=" & $image.offx & ",y=" & $image.offy &
+    ",w=" & $image.dispw & ",h=" & $image.disph &
+    # for now, we always use placement id 1
+    ",p=1,q=2"
+  if image.kittyId != 0:
+    outs &= ",i=" & $image.kittyId & ",a=p;" & ST
+    term.write(outs)
+    term.flush()
+    return
+  inc term.kittyId # skip i=0
+  image.kittyId = term.kittyId
+  outs &= ",i=" & $image.kittyId
   const MaxBytes = 4096 * 3 div 4
   var i = MaxBytes
   # transcode to RGB
-  let p = cast[ptr UncheckedArray[uint8]](addr bmp.px[0])
-  let L = bmp.px.len * 4
+  let p = cast[ptr UncheckedArray[uint8]](addr image.bmp.px[0])
+  let L = image.bmp.px.len * 4
   let m = if i < L: '1' else: '0'
-  var outs = term.cursorGoto(x, y) &
-    APC & "Gf=32,m=" & m & ",a=T,C=1,s=" & $bmp.width & ",v=" & $bmp.height &
-    ",x=" & $offx & ",y=" & $offy & ",w=" & $dispw & ",h=" & $disph & ';'
+  outs &= ",a=T,f=32,m=" & m & ';'
   outs.btoa(p.toOpenArray(0, min(L, i) - 1))
   outs &= ST
   term.write(outs)
@@ -712,33 +782,39 @@ proc outputKittyImage(term: Terminal; x, y, offx, offy, dispw, disph: int;
     outs &= ST
     term.write(outs)
 
-# x, y, maxw, maxh in cells
-# x, y can be negative, then image starts outside the screen
-proc outputImage*(term: Terminal; bmp: Bitmap; x, y, maxw, maxh: int) =
-  var xpx = x * term.attrs.ppc
-  var ypx = y * term.attrs.ppl
-  # calculate offset inside image to start from
-  let offx = -min(xpx, 0)
-  let offy = -min(ypx, 0)
-  # calculate maximum image size that fits on the screen relative to the image
-  # origin (*not* offx/offy)
-  let maxwpx = maxw * term.attrs.ppc
-  let maxhpx = maxh * term.attrs.ppl
-  xpx = max(xpx, 0)
-  ypx = max(ypx, 0)
-  let dispw = min(int(bmp.width) - offx + xpx, maxwpx) + offx - xpx
-  let disph = min(int(bmp.height) - offy + ypx, maxhpx) + offy - ypx
-  if dispw <= offx or disph <= offy:
-    return
-  let x = max(x, 0)
-  let y = max(y, 0)
-  case term.imageMode
-  of imNone: assert false
-  of imSixel: term.outputSixelImage(x, y, offx, offy, dispw, disph, bmp)
-  of imKitty: term.outputKittyImage(x, y, offx, offy, dispw, disph, bmp)
+proc outputImages*(term: Terminal) =
+  if term.imageMode == imKitty:
+    # clean up unused kitty images
+    var s = ""
+    for image in term.imagesToClear:
+      if image.kittyId == 0:
+        continue # maybe it was never displayed...
+      s &= APC & "Ga=d,d=I,i=" & $image.kittyId & ",p=1,q=2;" & ST
+    term.write(s)
+    term.imagesToClear.setLen(0)
+  for image in term.canvasImages:
+    if image.damaged:
+      assert image.dispw > 0 and image.disph > 0
+      let x = max(image.x, 0)
+      let y = max(image.y, 0)
+      case term.imageMode
+      of imNone: assert false
+      of imSixel: term.outputSixelImage(x, y, image)
+      of imKitty: term.outputKittyImage(x, y, image)
+      image.damaged = false
 
 proc clearCanvas*(term: Terminal) =
   term.cleared = false
+  let maxw = term.attrs.width
+  let maxh = term.attrs.height - 1
+  var toRemove: seq[int] = @[]
+  for i, image in term.canvasImages:
+    if not term.positionImage(image, image.x, image.y, maxw, maxh):
+      toRemove.add(i)
+      if term.imageMode == imKitty:
+        term.imagesToClear.add(image)
+  for i in countdown(toRemove.high, 0):
+    term.canvasImages.delete(toRemove[i])
 
 # see https://viewsourcecode.org/snaptoken/kilo/02.enteringRawMode.html
 proc disableRawMode(term: Terminal) =
@@ -776,7 +852,7 @@ proc quit*(term: Terminal) =
     if term.set_title:
       term.write(XTPOPTITLE)
     term.showCursor()
-    term.cleared = false
+    term.clearCanvas()
     if term.stdinUnblocked:
       term.restoreStdin()
       term.stdinWasUnblocked = true
@@ -1130,7 +1206,7 @@ proc windowChange*(term: Terminal) =
   term.applyConfigDimensions()
   term.canvas = newSeq[FixedCell](term.attrs.width * term.attrs.height)
   term.lineDamage = newSeq[int](term.attrs.height)
-  term.cleared = false
+  term.clearCanvas()
 
 proc initScreen(term: Terminal) =
   # note: deinit happens in quit()
diff --git a/src/server/buffer.nim b/src/server/buffer.nim
index 53b30aac..d0602a76 100644
--- a/src/server/buffer.nim
+++ b/src/server/buffer.nim
@@ -1698,8 +1698,7 @@ proc getLines*(buffer: Buffer; w: Slice[int]): GetLinesResult {.proxy.} =
   result.bgcolor = buffer.bgcolor
   if buffer.config.images:
     for image in buffer.images:
-      if image.y <= w.b and
-          image.y + int(image.bmp.height) >= w.a * buffer.attrs.ppl:
+      if image.y <= w.b and image.y + int(image.bmp.height) >= w.a:
         result.images.add(image)
 
 proc markURL*(buffer: Buffer; schemes: seq[string]) {.proxy.} =
diff --git a/src/types/color.nim b/src/types/color.nim
index 1f9c3347..0e33ffed 100644
--- a/src/types/color.nim
+++ b/src/types/color.nim
@@ -417,10 +417,10 @@ func toRGB*(param0: EightBitColor): RGBColor =
     let n = (u - 232) * 10 + 8
     return gray(n)
 
-func toEightBit*(rgb: RGBColor): EightBitColor =
-  let r = int(rgb.r)
-  let g = int(rgb.g)
-  let b = int(rgb.b)
+func toEightBit(r, g, b: uint8): EightBitColor =
+  let r = int(r)
+  let g = int(g)
+  let b = int(b)
   # Idea from here: https://github.com/Qix-/color-convert/pull/75
   # This seems to work about as well as checking for
   # abs(U - 128) < 5 & abs(V - 128 < 5), but is definitely faster.
@@ -434,6 +434,12 @@ func toEightBit*(rgb: RGBColor): EightBitColor =
   return EightBitColor(uint8(16 + 36 * (r * 5 div 255) +
     6 * (g * 5 div 255) + (b * 5 div 255)))
 
+func toEightBit*(c: RGBColor): EightBitColor =
+  return toEightBit(c.r, c.g, c.b)
+
+func toEightBit*(c: RGBAColorBE): EightBitColor =
+  return toEightBit(c.r, c.g, c.b)
+
 template `$`*(rgbcolor: RGBColor): string =
   "rgb(" & $rgbcolor.r & ", " & $rgbcolor.g & ", " & $rgbcolor.b & ")"
 
diff --git a/todo b/todo
index 2c0be87d..fab985f5 100644
--- a/todo
+++ b/todo
@@ -73,7 +73,6 @@ layout engine:
 - iframe
 - writing-mode, grid, ruby, ... (i.e. cool new stuff)
 images:
-- more efficient kitty display (use IDs)
 - more efficient sixel display (store encoded images)
 - more efficient display in general (why are we repainting twice per keypress?)
 - document it (when performance is acceptable)