about summary refs log tree commit diff stats
path: root/src/display/lineedit.nim
diff options
context:
space:
mode:
authorbptato <nincsnevem662@gmail.com>2024-03-14 20:57:45 +0100
committerbptato <nincsnevem662@gmail.com>2024-03-14 21:05:16 +0100
commitd26766c4c4015990703e84e8136f96d222edbc97 (patch)
tree7f412f8ca98d2b04323da5cf2fd607efbd6c408d /src/display/lineedit.nim
parenta8f05f18fdd64485c26b453e62e8073b50e271ef (diff)
downloadchawan-d26766c4c4015990703e84e8136f96d222edbc97.tar.gz
Move around some modules
* extern -> gone, runproc absorbed by pager, others moved into io/
* display -> local/ (where else would we display?)
* xhr -> html/
* move out WindowAttributes from term, so we don't depend on local
  from server
Diffstat (limited to 'src/display/lineedit.nim')
-rw-r--r--src/display/lineedit.nim328
1 files changed, 0 insertions, 328 deletions
diff --git a/src/display/lineedit.nim b/src/display/lineedit.nim
deleted file mode 100644
index a69465a6..00000000
--- a/src/display/lineedit.nim
+++ /dev/null
@@ -1,328 +0,0 @@
-import std/strutils
-import std/unicode
-
-import bindings/quickjs
-import display/term
-import js/javascript
-import types/cell
-import types/opt
-import utils/strwidth
-import utils/twtstr
-
-import chagashi/charset
-import chagashi/validator
-import chagashi/decoder
-
-type
-  LineEditState* = enum
-    EDIT, FINISH, CANCEL
-
-  LineHistory* = ref object
-    lines: seq[string]
-
-  LineEdit* = ref object
-    news*: string
-    prompt*: string
-    promptw: int
-    state*: LineEditState
-    escNext*: bool
-    cursorx: int # 0 ..< news.notwidth
-    cursori: int # 0 ..< news.len
-    shiftx: int # 0 ..< news.notwidth
-    shifti: int # 0 ..< news.len
-    padding: int # 0 or 1
-    maxwidth: int
-    disallowed: set[char]
-    hide: bool
-    hist: LineHistory
-    histindex: int
-    histtmp: string
-    invalid*: bool
-
-jsDestructor(LineEdit)
-
-func newLineHistory*(): LineHistory =
-  return LineHistory()
-
-# Note: capped at edit.maxwidth.
-func getDisplayWidth(edit: LineEdit): int =
-  var dispw = 0
-  var i = edit.shifti
-  var r: Rune
-  while i < edit.news.len and dispw < edit.maxwidth:
-    fastRuneAt(edit.news, i, r)
-    dispw += r.width()
-  return dispw
-
-proc shiftView(edit: LineEdit) =
-  # Shift view so it contains the cursor.
-  if edit.cursorx < edit.shiftx:
-    edit.shiftx = edit.cursorx
-    edit.shifti = edit.cursori
-  # Shift view so it is completely filled.
-  if edit.shiftx > 0:
-    let dispw = edit.getDisplayWidth()
-    if dispw < edit.maxwidth:
-      let targetx = edit.shiftx - edit.maxwidth + dispw
-      if targetx <= 0:
-        edit.shiftx = 0
-        edit.shifti = 0
-      else:
-        while edit.shiftx > targetx:
-          let (r, len) = edit.news.lastRune(edit.shifti - 1)
-          edit.shiftx -= r.width()
-          edit.shifti -= len
-  edit.padding = 0
-  # Shift view so it contains the cursor. (act 2)
-  if edit.shiftx < edit.cursorx - edit.maxwidth:
-    while edit.shiftx < edit.cursorx - edit.maxwidth and
-        edit.shifti < edit.news.len:
-      var r: Rune
-      fastRuneAt(edit.news, edit.shifti, r)
-      edit.shiftx += r.width()
-    if edit.shiftx > edit.cursorx - edit.maxwidth:
-      # skipped over a cell because of a double-width char
-      edit.padding = 1
-
-proc generateOutput*(edit: LineEdit): FixedGrid =
-  edit.shiftView()
-  # Make the output grid +1 cell wide, so it covers the whole input area.
-  result = newFixedGrid(edit.promptw + edit.maxwidth + 1)
-  var x = 0
-  for r in edit.prompt.runes:
-    result[x].str &= $r
-    x += r.width()
-    if x >= result.width: break
-  for i in 0 ..< edit.padding:
-    if x < result.width:
-      result[x].str = " "
-      inc x
-  var i = edit.shifti
-  while i < edit.news.len:
-    var r: Rune
-    fastRuneAt(edit.news, i, r)
-    if not edit.hide:
-      let w = r.width()
-      if x + w > result.width: break
-      if r.isControlChar():
-        result[x].str &= '^'
-        inc x
-        result[x].str &= char(r).getControlLetter()
-        inc x
-      else:
-        result[x].str &= $r
-        x += w
-    else:
-      if x + 1 > result.width: break
-      result[x].str &= '*'
-      inc x
-
-proc getCursorX*(edit: LineEdit): int =
-  return edit.promptw + edit.cursorx + edit.padding - edit.shiftx
-
-proc insertCharseq(edit: LineEdit, s: string) =
-  let s = if edit.escNext:
-    s
-  else:
-    deleteChars(s, edit.disallowed)
-  edit.escNext = false
-  if s.len == 0:
-    return
-  let rem = edit.news.substr(edit.cursori)
-  edit.news.setLen(edit.cursori)
-  edit.news &= s
-  edit.news &= rem
-  edit.cursori += s.len
-  edit.cursorx += s.notwidth()
-  edit.invalid = true
-
-proc cancel(edit: LineEdit) {.jsfunc.} =
-  edit.state = CANCEL
-
-proc submit(edit: LineEdit) {.jsfunc.} =
-  if edit.hist.lines.len == 0 or edit.news != edit.hist.lines[^1]:
-    edit.hist.lines.add(edit.news)
-  edit.state = FINISH
-
-proc backspace(edit: LineEdit) {.jsfunc.} =
-  if edit.cursori > 0:
-    let (r, len) = edit.news.lastRune(edit.cursori - 1)
-    edit.news.delete(edit.cursori - len .. edit.cursori - 1)
-    edit.cursori -= len
-    edit.cursorx -= r.width()
-    edit.invalid = true
-
-proc write*(edit: LineEdit, s: string, cs: Charset): bool =
-  if cs == CHARSET_UTF_8:
-    if s.validateUTF8Surr() != -1:
-      return false
-    edit.insertCharseq(s)
-  else:
-    let td = newTextDecoder(cs)
-    var success = false
-    let s = td.decodeAll(s, success)
-    if not success:
-      return false
-    edit.insertCharseq(s)
-  return true
-
-proc write(edit: LineEdit, s: string): bool {.jsfunc.} =
-  edit.write(s, CHARSET_UTF_8)
-
-proc delete(edit: LineEdit) {.jsfunc.} =
-  if edit.cursori < edit.news.len:
-    let len = edit.news.runeLenAt(edit.cursori)
-    edit.news.delete(edit.cursori ..< edit.cursori + len)
-    edit.invalid = true
-
-proc escape(edit: LineEdit) {.jsfunc.} =
-  edit.escNext = true
-
-proc clear(edit: LineEdit) {.jsfunc.} =
-  if edit.cursori > 0:
-    edit.news.delete(0..edit.cursori - 1)
-    edit.cursori = 0
-    edit.cursorx = 0
-    edit.invalid = true
-
-proc kill(edit: LineEdit) {.jsfunc.} =
-  if edit.cursori < edit.news.len:
-    edit.news.setLen(edit.cursori)
-    edit.invalid = true
-
-proc backward(edit: LineEdit) {.jsfunc.} =
-  if edit.cursori > 0:
-    let (r, len) = edit.news.lastRune(edit.cursori - 1)
-    edit.cursori -= len
-    edit.cursorx -= r.width()
-    if edit.cursorx < edit.shiftx:
-      edit.invalid = true
-
-proc forward(edit: LineEdit) {.jsfunc.} =
-  if edit.cursori < edit.news.len:
-    var r: Rune
-    fastRuneAt(edit.news, edit.cursori, r)
-    edit.cursorx += r.width()
-    if edit.cursorx >= edit.shiftx + edit.maxwidth:
-      edit.invalid = true
-
-proc prevWord(edit: LineEdit) {.jsfunc.} =
-  if edit.cursori == 0:
-    return
-  let (r, len) = edit.news.lastRune(edit.cursori - 1)
-  if r.breaksWord():
-    edit.cursori -= len
-    edit.cursorx -= r.width()
-  while edit.cursori > 0:
-    let (r, len) = edit.news.lastRune(edit.cursori - 1)
-    if r.breaksWord():
-      break
-    edit.cursori -= len
-    edit.cursorx -= r.width()
-  if edit.cursorx < edit.shiftx:
-    edit.invalid = true
-
-proc nextWord(edit: LineEdit) {.jsfunc.} =
-  if edit.cursori >= edit.news.len:
-    return
-  let oc = edit.cursori
-  var r: Rune
-  fastRuneAt(edit.news, edit.cursori, r)
-  if r.breaksWord():
-    edit.cursorx += r.width()
-  else:
-    edit.cursori = oc
-  while edit.cursori < edit.news.len:
-    let pc = edit.cursori
-    fastRuneAt(edit.news, edit.cursori, r)
-    if r.breaksWord():
-      edit.cursori = pc
-      break
-    edit.cursorx += r.width()
-  if edit.cursorx >= edit.shiftx + edit.maxwidth:
-    edit.invalid = true
-
-proc clearWord(edit: LineEdit) {.jsfunc.} =
-  let oc = edit.cursori
-  edit.prevWord()
-  if oc != edit.cursori:
-    edit.news.delete(edit.cursori .. oc - 1)
-    edit.invalid = true
-
-proc killWord(edit: LineEdit) {.jsfunc.} =
-  if edit.cursori >= edit.news.len:
-    return
-  let oc = edit.cursori
-  let ox = edit.cursorx
-  edit.nextWord()
-  if edit.cursori != oc:
-    if edit.cursori < edit.news.len:
-      let len = edit.news.runeLenAt(edit.cursori)
-      edit.news.delete(oc ..< edit.cursori + len)
-    else:
-      edit.news.delete(oc ..< edit.cursori)
-    edit.cursori = oc
-    edit.cursorx = ox
-    edit.invalid = true
-
-proc begin(edit: LineEdit) {.jsfunc.} =
-  edit.cursori = 0
-  edit.cursorx = 0
-  if edit.shiftx > 0:
-    edit.invalid = true
-
-proc `end`(edit: LineEdit) {.jsfunc.} =
-  if edit.cursori < edit.news.len:
-    edit.cursori = edit.news.len
-    edit.cursorx = edit.news.notwidth()
-    if edit.cursorx >= edit.shiftx + edit.maxwidth:
-      edit.invalid = true
-
-proc prevHist(edit: LineEdit) {.jsfunc.} =
-  if edit.histindex > 0:
-    if edit.news.len > 0:
-      edit.histtmp = $edit.news
-    dec edit.histindex
-    edit.news = edit.hist.lines[edit.histindex]
-    # The begin call is needed so the cursor doesn't get lost outside
-    # the string.
-    edit.begin()
-    edit.end()
-    edit.invalid = true
-
-proc nextHist(edit: LineEdit) {.jsfunc.} =
-  if edit.histindex + 1 < edit.hist.lines.len:
-    inc edit.histindex
-    edit.news = edit.hist.lines[edit.histindex]
-    edit.begin()
-    edit.end()
-    edit.invalid = true
-  elif edit.histindex < edit.hist.lines.len:
-    inc edit.histindex
-    edit.news = edit.histtmp
-    edit.begin()
-    edit.end()
-    edit.histtmp = ""
-
-proc windowChange*(edit: LineEdit, attrs: WindowAttributes) =
-  edit.maxwidth = attrs.width - edit.promptw - 1
-
-proc readLine*(prompt: string, termwidth: int, current = "",
-    disallowed: set[char] = {}, hide = false, hist: LineHistory): LineEdit =
-  result = LineEdit(
-    prompt: prompt,
-    promptw: prompt.width(),
-    news: current,
-    disallowed: disallowed,
-    hide: hide,
-    invalid: true
-  )
-  result.cursori = result.news.len
-  result.cursorx = result.news.notwidth()
-  # - 1, so that the cursor always has place
-  result.maxwidth = termwidth - result.promptw - 1
-  result.hist = hist
-  result.histindex = result.hist.lines.len
-
-proc addLineEditModule*(ctx: JSContext) =
-  ctx.registerType(LineEdit)