Make Commander S use the new completion code and move this cruft to a

separate module.
This commit is contained in:
eknauel 2005-08-17 13:32:40 +00:00
parent 41b8e1dde9
commit 8c5de5ee27
3 changed files with 324 additions and 217 deletions

225
scheme/completer.scm Normal file
View File

@ -0,0 +1,225 @@
;;; #### also used in nuit-engine.scm move to some utils module
(define-syntax with-lock
(syntax-rules ()
((_ lock exp ...)
(begin
(obtain-lock lock)
(let ((val (begin exp ...)))
(release-lock lock)
val)))))
;; completion set for executables in PATH
(define executable-completions-lock (make-lock))
(define executable-completions #f)
(define (get-path-list)
(cond
((getenv "PATH")
=> (lambda (str)
(string-tokenize
str (char-set-difference char-set:full (char-set #\:)))))
(else
'("/usr/bin" "/bin" "/usr/sbin" "/sbin"))))
(define (init-executables-completion-set!)
(spawn
(lambda ()
(with-lock executable-completions-lock
(set! executable-completions
(make-completion-set-for-executables (get-path-list)))))))
;; find the part of the command line the user wants to complete
(define (make-scan-for-completions cmd-selector selector symbol)
(lambda (cmd)
(let lp ((things (cmd-selector cmd)))
(cond
((null? things)
#f)
((to-complete? (selector (car things)))
(list symbol cmd (selector (car things))))
(else
(lp (cdr things)))))))
(define scan-redirections-for-completions
(make-scan-for-completions
command-redirections redirection-dest 'redir-dest))
(define scan-args-for-completions
(make-scan-for-completions
command-args (lambda (v) v) 'arg))
(define (scan-command-for-completions cmd)
(cond
((to-complete? (command-executable cmd))
(list 'command cmd (command-executable cmd)))
((scan-redirections-for-completions cmd)
=> (lambda (v) v))
((scan-args-for-completions cmd)
=> (lambda (v) v))
(else #f)))
(define (scan-command-line-for-completions cmdln)
(cond
((scan-command-for-completions
(command-line-first-cmd cmdln))
=> (lambda (v) v))
(else
(let lp ((lst (command-line-combinator/cmds cmdln)))
(cond
((null? lst) #f)
((scan-command-for-completions (cdar lst))
=> (lambda (v) v))
(else (lp (cdr lst))))))))
;; completion functions for arguments and redirection
(define (find-completions-for-arg cmd to-complete)
(let ((prefix (to-complete-prefix to-complete)))
(if prefix
(complete-files/path (expand-file-name prefix (cwd)))
(complete-files/path prefix))))
;; #### no special treatment yet
(define find-completions-for-redir find-completions-for-arg)
;; completion functions for commands
(define (find-completions-for-command cmd to-complete)
(let ((prefix (or (to-complete-prefix to-complete) "")))
(cond
((command-contains-path? prefix)
;; #### install error handler
(complete-executables/path
(expand-file-name prefix (cwd))))
(else
(append
(completions-for (command-completions) prefix)
(with-lock executable-completions-lock
(completions-for-executables
executable-completions prefix)))))))
;; some helpers for the implementation of completion functions
(define (command-contains-path? command)
(or (string-contains command "/")
(string-contains command "~")
(string-contains command "..")))
(define (files-in-dir file-filter dir)
(with-cwd dir
(filter-map
(lambda (file)
(and (file-filter file)
(absolute-file-name file dir)))
(directory-files))))
(define (complete-path path)
(let ((dir (file-name-directory path)))
(map (lambda (p)
(if (string-prefix? "/" p)
p
(string-append dir p)))
(glob (string-append path "*")))))
(define (file-exists-and-is-directory? fname)
(call-with-current-continuation
(lambda (esc)
(with-handler
(lambda (c more)
(if (error? c)
(esc #f)
(more)))
(lambda ()
(and (file-exists? fname) (file-directory? fname)))))))
(define (complete-with-filesystem-objects filter partial-name)
(if (file-exists-and-is-directory? partial-name)
(files-in-dir filter partial-name)
(complete-path partial-name)))
(define (complete-executables/path partial-name)
(complete-with-filesystem-objects
(lambda (file)
(call-with-current-continuation
(lambda (esc)
(with-handler
(lambda (c more)
(if (error? c)
(esc #f)
(more)))
(lambda ()
(or (file-executable? file) (file-directory? file)))))))
partial-name))
(define (complete-files/path partial-name)
(complete-with-filesystem-objects
(lambda (file) #t) partial-name))
;; insert the completion into the command line
(define (assemble-redirection replacer redir)
(make-redirection
(redirection-op redir)
(replacer (redirection-dest redir))))
(define (assemble-command replacer cmd)
(let ((assemble-redirection
(lambda (obj) (assemble-redirection replacer obj))))
(make-command
(replacer (command-executable cmd))
(map replacer (command-args cmd))
(map assemble-redirection (command-redirections cmd)))))
(define (assemble-command-line replacer cmdln)
(let ((assemble-command
(lambda (cmd) (assemble-command replacer cmd))))
(make-command-line
(assemble-command (command-line-first-cmd cmdln))
(map (lambda (p)
(cons (car p) (assemble-command (cdr p))))
(command-line-combinator/cmds cmdln))
(command-line-job-ctrl cmdln))))
(define (assemble-with-completion cmdln to-complete completion)
(assemble-command-line (lambda (obj)
(if (eq? obj to-complete)
completion
obj))
cmdln))
;; the main part
(define (find-completer type)
(case type
((arg) find-completions-for-arg)
((command) find-completions-for-command)
((redir-dest) find-completions-for-redir)
(else
(error "Unknown completion type" type))))
(define (calculate-cursor-index to-complete completion)
(+ (to-complete-pos to-complete) (string-length completion)))
(define (complete cmdln cursor-index)
(debug-message "complete " cmdln ", " cursor-index)
(let* ((parsed
(lex/parse-partial-command-line cmdln cursor-index))
(completion-info
(scan-command-line-for-completions parsed)))
(debug-message "complete " completion-info)
(and completion-info
(destructure (((type cmd to-complete) completion-info))
(let ((completions ((find-completer type) cmd to-complete)))
(cond
((= (length completions) 1)
(list (unparse-command-line
(assemble-with-completion parsed to-complete
(car completions)))
(calculate-cursor-index to-complete
(car completions))
to-complete parsed))
(else
(list completions cursor-index to-complete parsed))))))))

View File

@ -29,9 +29,6 @@
(define (enter-command-mode!)
(set! *command-buffer-mode* 'command))
(define executable-completions-lock (make-lock))
(define executable-completions #f)
(define key-control-x 24)
(define key-o 111)
(define key-tab 9)
@ -465,22 +462,6 @@
(release-lock paint-lock)
(loop (wait-for-input) c-x-pressed? #f)))))))
(define (get-path-list)
(cond
((getenv "PATH")
=> (lambda (str)
(string-tokenize
str (char-set-difference char-set:full (char-set #\:)))))
(else
'("/usr/bin" "/bin" "/usr/sbin" "/sbin"))))
(define (init-executables-completion-set!)
(spawn
(lambda ()
(with-lock executable-completions-lock
(set! executable-completions
(make-completion-set-for-executables (get-path-list)))))))
(define (paint-bar-1)
(mvwaddstr (app-window-curses-win (bar-1)) 0 1 "Commander S")
(wrefresh (app-window-curses-win (bar-1))))
@ -716,84 +697,6 @@
completions)
num-lines))
(define (command-contains-path? command)
(or (string-contains command "/")
(string-contains command "~")
(string-contains command "..")))
(define (files-in-dir file-filter dir)
(with-cwd dir
(filter-map
(lambda (file)
(and (file-filter file)
(absolute-file-name file dir)))
(directory-files))))
(define (complete-path path)
(let ((dir (file-name-directory path)))
(map (lambda (p)
(if (string-prefix? "/" p)
p
(string-append dir p)))
(glob (string-append path "*")))))
(define (complete-with-filesystem-objects filter partial-name)
(if (and (file-exists? partial-name) (file-directory? partial-name))
(files-in-dir filter partial-name)
(complete-path partial-name)))
(define (complete-executables/path partial-name)
(complete-with-filesystem-objects
(lambda (file)
(or (file-executable? file) (file-directory? file)))
partial-name))
(define (complete-files/path partial-name)
(complete-with-filesystem-objects
(lambda (file) #t) partial-name))
(define (command-completer command prefix args args-pos)
(debug-message "command-mode-completer" prefix "|" args "|" args-pos)
(cond
((command-contains-path? prefix)
;; #### FIXME ignore errors here?
(complete-executables/path (expand-file-name prefix (cwd))))
(else
(append
(completions-for (command-completions) prefix)
(with-lock executable-completions-lock
(completions-for-executables executable-completions prefix))))))
(define (file-completer command prefix args args-pos)
(if (zero? (string-length prefix))
(complete-files/path prefix)
(complete-files/path (expand-file-name prefix (cwd)))))
(define (assemble-line-with-completion command arg arg-pos completion)
(debug-message "assemble-line-with-completion "
command "," arg "," arg-pos "," completion)
(let ((string-append* (lambda (s t)
(if (string=? s "")
t
(string-append s " " t)))))
(let lp ((tokens (cons command arg))
(arg-count 0)
(cursor-pos 0)
(line ""))
(cond
((null? tokens)
(values line (+ 2 cursor-pos)))
((= arg-count arg-pos)
(lp (cdr tokens)
(+ arg-count 1)
(+ cursor-pos (string-length completion))
(string-append* line completion)))
(else
(lp (cdr tokens)
(+ arg-count 1)
(+ 1 (+ cursor-pos (string-length (car tokens))))
(string-append* line (car tokens))))))))
(define (display-completed-line line cursor-pos)
(debug-message "display-completed-line " line "," cursor-pos)
(set-buffer-pos-col! (command-buffer) cursor-pos)
@ -818,128 +721,65 @@
win (result-buffer) (focus-on-result-buffer?))
(refresh-result-window)))
(define (completer-function-for-command command)
(let ((alist
(map (lambda (p)
(cons (command-plugin-command p)
(command-plugin-completer p)))
(command-plugin-list))))
(cond
((assoc command alist) => cdr)
(else #f))))
(define (call-completer command args prefix arg-pos)
(cond
((= 0 arg-pos)
(command-completer command prefix args arg-pos))
((completer-function-for-command command)
=> (lambda (completer)
(completer command prefix args arg-pos)))
(else
(file-completer command prefix args arg-pos))))
(define (current-cursor-index)
;; #### No, I will not comment on this.
(- (buffer-pos-col (command-buffer)) 2))
(define (offer-completions command)
(let* ((tokens/cursor-list (tokenize-command command))
(args (map car (cdr tokens/cursor-list)))
(command (caar tokens/cursor-list)))
(call-with-values
(lambda ()
(find-token-with-cursor tokens/cursor-list))
(lambda (prefix arg-pos)
;; #### FIXME
(if (not prefix)
(error "could not determine token with cursor position"
tokens/cursor-list command
(- (buffer-pos-col (command-buffer)) 2)))
(let ((completions
(call-completer command args
prefix arg-pos)))
(if (= (length completions) 1)
(begin
(call-with-values
(lambda ()
(assemble-line-with-completion
command args arg-pos (car completions)))
display-completed-line)
#f)
(let* ((select-list
(completions->select-list
completions
(- (result-buffer-num-lines (result-buffer)) 3)))
(selector
(make-completion-selector
select-list completions
command args arg-pos)))
(paint-completion-select-list select-list command)
(move-cursor (command-buffer) (result-buffer))
(refresh-command-window)
selector)))))))
(debug-message "offer-completions '" command "'")
(let ((completion-info (complete command (current-cursor-index))))
(if (not completion-info)
(begin
;; #### the completion mechanism was too confused to do anything
;; #### beep or so
#f)
(destructure
(((completions cursor-index to-complete cmdln) completion-info))
(cond
((string? completions)
;; #### don't ask
(display-completed-line completions (+ 2 cursor-index))
#f)
((list? completions)
(let* ((select-list
(completions->select-list
completions
(- (result-buffer-num-lines (result-buffer)) 3)))
(selector
(make-completion-selector select-list completions
cmdln to-complete)))
(paint-completion-select-list select-list command)
(move-cursor (command-buffer) (result-buffer))
(refresh-command-window)
selector))
(else
(error "COMPLETE returned an unexpected value"
completions)))))))
(define (make-completion-selector select-list completions
command arg arg-pos)
(define (make-completion-selector select-list completions
cmdln to-complete)
(lambda (key)
(cond
((= key 10)
(focus-command-buffer!)
(call-with-values
(lambda ()
(assemble-line-with-completion
command arg arg-pos
(let ((completion
(select-list-selected-entry select-list)))
display-completed-line)
#f)
((or (select-list-navigation-key? key)
(select-list-marking-key? key))
(let ((new-select-list
(select-list-handle-key-press select-list key)))
(paint-completion-select-list
new-select-list (last (buffer-text (command-buffer))))
(make-completion-selector
new-select-list completions command arg arg-pos)))
(else
;; #### FIXME we loose a character this way
(focus-command-buffer!)
#f))))
(define (find-token-with-cursor tokens/cursor-list)
(debug-message "find-token-with-cursor " tokens/cursor-list)
(let lp ((lst tokens/cursor-list) (i 0))
(cond
((null? lst)
(values #f i))
((cdar lst)
(values (caar lst) i))
(else
(lp (cdr lst) (+ i 1))))))
(define (command-token-delimiter? c)
(char-set-contains? char-set:whitespace c))
(define (skip-delimters delimiter? chars)
(let lp ((chars chars) (i 0))
(cond
((null? chars) (values '() i))
((delimiter? (car chars))
(lp (cdr chars) (+ i 1)))
(else (values chars i)))))
(define (tokenize-command command)
(let ((cursor-pos (- (buffer-pos-col (command-buffer)) 2))) ;; don't ask
(let lp ((chars (string->list command))
(token "")
(tokens '())
(i 0))
(cond
((null? chars)
(reverse (cons (cons token (= i cursor-pos)) tokens)))
((command-token-delimiter? (car chars))
(call-with-values
(lambda ()
(skip-delimters command-token-delimiter? chars))
(lambda (rest skipped)
(lp rest "" (cons (cons token (= i cursor-pos)) tokens)
(+ i skipped)))))
(else
(lp (cdr chars) (string-append token (string (car chars)))
tokens (+ i 1)))))))
(focus-command-buffer!)
;; #### No, I will not comment on this.
(display-completed-line
(unparse-command-line
(assemble-with-completion
cmdln to-complete completion))
(+ 2 (calculate-cursor-index to-complete completion)))
#f))
((or (select-list-navigation-key? key)
(select-list-marking-key? key))
(let ((new-select-list
(select-list-handle-key-press select-list key)))
(paint-completion-select-list
new-select-list (last (buffer-text (command-buffer))))
(make-completion-selector
new-select-list completions cmdln to-complete)))
(else
;; #### FIXME we loose a character this way
(focus-command-buffer!)
#f))))

View File

@ -462,6 +462,40 @@
thread-fluids)
(files complete))
;;; standard completion mechanism
(define-interface completer-interface
(export init-executables-completion-set!
complete
calculate-cursor-index
assemble-with-completion))
(define-structure completer completer-interface
(open scheme
(subset scsh
(file-name-directory glob with-cwd cwd
absolute-file-name expand-file-name
file-exists? file-directory? file-executable?
directory-files getenv))
threads
locks
signals
handle
conditions
destructuring
(subset srfi-1 (filter-map))
srfi-13
srfi-14
tty-debug
completion-sets
plugin
plugin-host
command-line-lexer
command-line-parser
command-line-absyn)
(files completer))
;;; console
(define-interface console-interface
@ -602,7 +636,12 @@
redirection?
redirection-op
redirection-dest))
redirection-dest
to-complete?
to-complete-prefix
to-complete-without-prefix?
to-complete-pos))
(define-interface command-line-absyn-constructors-interface
(export
@ -675,7 +714,6 @@
signals
handle
rt-modules
(modify ncurses (hide filter))
srfi-1
srfi-6
srfi-13
@ -689,11 +727,14 @@
(send cml-send)
(receive cml-receive)))
let-opt
destructuring
(modify ncurses (hide filter))
app-windows
initial-tty
nuit-windows
command-line-parser
focus-table
result-buffer-changes
nuit-eval/focus-table
@ -706,6 +747,7 @@
history
handle-fatal-error
completion-sets
completer
select-list
jobs
run-jobs