retroforth/example/alternate-listener.retro
crc 6fa38ed966 fix a couple of things in example/alternate-listener.retro
FossilOrigin-Name: b3cb321551d24e23a3a1cb34bc708ac183a96cf2767fa8d8c8f51242d9233a5e
2021-11-12 23:28:23 +00:00

135 lines
3.5 KiB
Forth

# A New Listener
The basic listener is very minimalistic. There are a few reasons
for this.
- I don't use it often (I generally code in files or blocks,
using the listener only for brief tests)
- A complex listener adds surface for bugs and incompatibilites
due to terminal and host configurations
- It's harder to pipe data in and out of a more complex listener
This can be worked around in a number of ways. Tools like
`rlwrap` can be used to provide a line editor and history, and
one can always write their own custom listener...
But sometimes it's nice to experiment a bit. This is a modestly
expanded version of the listener, adding some new functionality
and providing a more flexible base to build on.
# Current Features
- character breaking input
- suggestions on hitting TAB
- show stack on hitting ESC
- use `retro-describe` to get help on CTRL+K
# Loading
retro -i -f alternate-listener.forth
new-listener
# The Code
The listener needs to determine what to treat as the end of the
token. I define `end-of-token?` for this.
~~~
:end-of-token? (c-f) hook
{ ASCII:CR ASCII:LF ASCII:SPACE } a:contains? ;
~~~
It's possible to get an empty string as an input. This isn't
any good, so I define `s:blank?` for this.
~~~
:s:blank? (s-sf) dup s:length n:zero? ;
~~~
There are certain keys I want to handle differently from others.
The initial ones are backspace, tab, escape, and CTRL+K. I am
defining handlers for these.
First is backspace. I trap ASCII:BS and ASCII:DEL for this.
~~~
:handle:backspace (c-c) hook
dup { ASCII:BS ASCII:DEL } a:contains?
[ buffer:get buffer:get drop-pair ] if ;
~~~
Tab is used to display suggestions, based on the token input so
far.
~~~
{{
:sigil?
TIB fetch 'sigil:_ [ #7 + store ] sip d:lookup n:-zero? ;
:hint
nl TIB sigil? [ n:inc ] if d:words-beginning-with nl
TIB s:put ;
---reveal---
:handle:tab (c-c) hook
dup ASCII:HT eq? [ buffer:get drop hint ] if ;
}}
~~~
Control + k (`ASCII:VT`) will display help for the word being typed.
This assumes that `retro-describe` is in your $PATH and that the
typed text is a complete word name (without a sigil).
~~~
:handle:CTRL+K (c-c) hook
dup ASCII:VT eq? [ buffer:get drop
TIB 'retro-describe_"%s" s:format nl
unix:system TIB s:put ] if ;
~~~
The escape key will be used to display the stack.
~~~
:handle:escape (c-c) hook
dup ASCII:ESC eq?
[ buffer:get drop nl &dump-stack dip nl TIB s:put ] if ;
~~~
To control the checks, I define two words. The first returns an
array of handlers, the second processes them.
~~~
:special-keys (-a) hook
{ &handle:backspace &handle:tab &handle:escape &handle:CTRL+K } ;
:check (q-) hook
&call a:for-each ;
~~~
And with these, I can quickly implement the `new-listener`.
~~~
{{
:guard (-) buffer:end TIB lt? [ TIB buffer:set ] if ;
:c:get (-c) \liii.... `1 dup buffer:add ;
:s:get (-s) [ TIB buffer:set
[ guard c:get special-keys check end-of-token? ] until
buffer:start s:chop ] buffer:preserve ;
:forever (q-) repeat &call sip again ;
---reveal---
:new-listener (-) hook
'stty_cbreak unix:system
[ s:get s:blank? &drop &interpret choose ] forever ;
[ 'stty_cbreak unix:system
#0 unix:exit ] &bye set-hook
}}
~~~
All of the exposed words are hooks: your code can patch in and
replace them as you see fit, making this much more mallable at
runtime.
Future things to (maybe) explore:
- tab completion
- line editing
- input history