/
smart-terminal.lisp
80 lines (74 loc) · 3.05 KB
/
smart-terminal.lisp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
;; Copyright (c) 2003 Nikodemus Siivola
;;
;; Permission is hereby granted, free of charge, to any person obtaining
;; a copy of this software and associated documentation files (the
;; "Software"), to deal in the Software without restriction, including
;; without limitation the rights to use, copy, modify, merge, publish,
;; distribute, sublicense, and/or sell copies of the Software, and to
;; permit persons to whom the Software is furnished to do so, subject to
;; the following conditions:
;;
;; The above copyright notice and this permission notice shall be included
;; in all copies or substantial portions of the Software.
;;
;; THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
;; EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
;; MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
;; IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
;; CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
;; TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
;; SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
(in-package :linedit)
(defclass smart-terminal (terminal)
((point-row :initform 1 :accessor point-row)
(active-string :initform "" :accessor active-string)))
(defun smart-terminal-p ()
(and (every 'identity
(list ti:cursor-up ti:cursor-down
ti:clr-eos ti:column-address ))
(some 'identity
(list ti:auto-right-margin ti:enter-am-mode))))
(defmethod backend-init ((backend smart-terminal))
(call-next-method)
(when ti:enter-am-mode
(ti:tputs ti:enter-am-mode)))
(defmethod display ((backend smart-terminal) &key prompt line point markup)
(let* ((*terminal-io* *standard-output*)
(columns (backend-columns backend))
(marked-line (if markup
(dwim-mark-parens line point
:pre-mark ti:enter-bold-mode
:post-mark ti:exit-attribute-mode)
line)))
(flet ((find-row (n)
;; 1+ includes point in row calculations
(ceiling (1+ n) columns))
(find-col (n)
(rem n columns)))
(let* ((new (concat prompt marked-line))
(old (active-string backend))
(end (+ (length prompt) (length line))) ;; based on unmarked
(rows (find-row end))
(start (or (mismatch new old) 0))
(start-row (find-row start)) ;; markup?
(start-col (find-col start)))
;; Move to start of update and clear to eos
(ti:tputs ti:column-address start-col)
(loop repeat (- (point-row backend) start-row)
do (ti:tputs ti:cursor-up))
(ti:tputs ti:clr-eos)
;; Write updated segment
(write-string (subseq new start))
(when (and (< start end) (zerop (find-col end)))
(ti:tputs ti:cursor-down))
;; Place point
(let* ((point (+ (length prompt) point))
(point-row (find-row point))
(point-col (find-col point)))
(loop repeat (- rows point-row)
do (ti:tputs ti:cursor-up))
(ti:tputs ti:column-address point-col)
;; Save state
(setf (point-row backend) point-row
(active-string backend) new))))
(force-output *terminal-io*)))