nxu/femto.tal

1708 lines
52 KiB
Tal
Raw Normal View History

2022-02-09 20:04:12 -05:00
( femto.tal )
( )
( requires terminal to be in raw mode )
2022-02-07 22:57:34 -05:00
( see femto launcher script for more details )
2022-02-06 17:07:11 -05:00
2022-02-11 19:15:40 -05:00
( TODO: )
( - get long line truncation/scrolling working )
2022-03-24 01:42:20 -04:00
( - allow line numbers to be toggled off )
2022-02-11 19:15:40 -05:00
( - open file command? )
( - close file command? )
( - search&replace )
2022-02-15 21:19:44 -05:00
2022-02-06 17:07:11 -05:00
|00 @System [ &vector $2 &wst $1 &rst $1 &pad $4 &r $2 &g $2 &b $2 &debug $1 &halt $1 ]
|10 @Console [ &vector $2 &read $5 &type $1 &write $1 &error $1 ]
2022-02-06 17:07:11 -05:00
|a0 @File [ &vector $2 &success $2 &stat $2 &delete $1 &append $1 &name $2 &length $2 &read $2 &write $2 ]
2022-04-10 02:05:09 -04:00
( MAX file size is currently #ce80, i.e. 52864 bytes )
2022-04-02 23:14:18 -04:00
2022-02-06 17:07:11 -05:00
%dbg { #ff .System/debug DEO }
%emit { .Console/write DEO }
%sp { #2018 DEO }
%nl { #0a18 DEO }
%cr { #0d18 DEO }
2022-04-12 12:15:42 -04:00
%crlf { cr nl }
2022-02-06 17:07:11 -05:00
%ansi { #1b18 DEO #5b18 DEO }
%alternate-buffer-on { ( \e[?1049h )
ansi #3f18 DEO #3118 DEO #3018 DEO #3418 DEO #3918 DEO #6818 DEO }
%alternate-buffer-off { ( \e[?1049l )
ansi #3f18 DEO #3118 DEO #3018 DEO #3418 DEO #3918 DEO #6c18 DEO }
2022-02-06 17:07:11 -05:00
( emit macros )
( )
( these save one byte and are easier to read. )
2022-09-10 13:30:15 -04:00
%emit-! { LIT2 "! 18 DEO }
%emit-$ { LIT2 "$ 18 DEO }
2022-04-20 23:02:27 -04:00
%emit-lpar { LIT2 28 18 DEO }
%emit-rpar { LIT2 29 18 DEO }
2022-09-10 13:30:15 -04:00
%emit-, { LIT2 ", 18 DEO }
%emit-0 { LIT2 "0 18 DEO }
%emit-1 { LIT2 "1 18 DEO }
%emit-2 { LIT2 "2 18 DEO }
%emit-3 { LIT2 "3 18 DEO }
%emit-6 { LIT2 "6 18 DEO }
%emit-7 { LIT2 "7 18 DEO }
%emit-: { LIT2 ": 18 DEO }
%emit-; { LIT2 "; 18 DEO }
%emit-C { LIT2 "C 18 DEO }
%emit-H { LIT2 "H 18 DEO }
%emit-J { LIT2 "J 18 DEO }
%emit-K { LIT2 "K 18 DEO }
%emit-[ { LIT2 "[ 18 DEO }
%emit-] { LIT2 "] 18 DEO }
%emit-m { LIT2 "m 18 DEO }
%emit-n { LIT2 "n 18 DEO }
%emit-~ { LIT2 "~ 18 DEO }
2022-03-08 23:31:12 -05:00
%quit! { #01 .System/halt DEO BRK }
%quit-restore! { alternate-buffer-off quit! }
2022-04-01 23:47:37 -04:00
%lmargin { #0006 }
2022-03-07 12:56:52 -05:00
( zero page )
|0000
2022-04-04 00:40:37 -04:00
( terminal size information )
2022-04-07 00:16:25 -04:00
( )
( for now these are constant but eventually we could detect terminal resizes )
2022-03-07 22:38:02 -05:00
@term [
2022-03-24 01:42:20 -04:00
&cols $2 ( relative x coordinate of cursor, from 0 )
&rows $2 ( relative y coordinate of cursor, from 1 )
2022-03-07 22:38:02 -05:00
]
2022-04-04 00:40:37 -04:00
( configuration settings used when editing )
2022-04-07 00:16:25 -04:00
( )
( these can be changed at any time without breaking anything else )
2022-03-07 22:38:02 -05:00
@config [
2022-04-06 20:39:02 -04:00
&tab-width $2 ( how many spaces to display tab chars )
&tab-adjust $2 ( how many "extra" spaces; tab-width - 1 )
2022-04-06 20:39:02 -04:00
&insert-tabs $1 ( tab key inserts tabs when true )
&color $2 ( digits of highlight color in reverse order )
2023-11-28 10:33:25 -05:00
&red $2 ( digits of color for line/EOF markers in reverse order )
2022-03-07 22:38:02 -05:00
]
2022-04-04 00:40:37 -04:00
( tracks information related to the buffer's view of data )
2022-04-07 00:16:25 -04:00
( )
( limit and line-count change when we modify the buffer. )
( offset and line-offset change when we move the view port. )
2022-03-07 22:38:02 -05:00
@buffer [
&limit $2 ( last byte of actual data (not including \0) + 1 )
&line-count $2 ( total number of lines in file )
2022-04-07 00:16:25 -04:00
&offset $2 ( first byte of data visible in terminal )
2022-03-07 22:38:02 -05:00
&line-offset $2 ( first line of text visible in terminal )
]
( relative cursor positions, e.g. 0 to cols-1 )
2022-04-07 00:16:25 -04:00
( )
( when these change we may need to move the view port as well )
2022-03-07 22:38:02 -05:00
@cursor [
&col $2 ( current column value 0-n (may exceed lenght of row) )
&row $2 ( current relative row value, 0-(height-1) )
]
2022-04-04 00:40:37 -04:00
( tracks overall editor state between events )
2022-03-07 22:38:02 -05:00
@state [
&in-help $1 ( are we showing help? )
2022-04-08 23:52:07 -04:00
&in-undo $1 ( are we currently in undo? )
2022-03-30 23:41:16 -04:00
&key $1 ( last key read )
2022-03-07 22:38:02 -05:00
&saw-esc $1 ( did we just see ESC? )
2022-03-24 01:26:06 -04:00
&saw-xterm $1 ( did we just see an ESC [ xterm sequence? )
&saw-vt $1 ( did we just see an ESC [ $N vt sequence? )
2022-03-30 01:26:33 -04:00
&redraw $1 ( redrawing: bits determine which parts )
( 0x01 cursor )
( 0x02 statusbar )
( 0x04 prompt )
( 0x08 matches )
( 0x10 body and everything else )
2022-04-01 23:47:37 -04:00
&message $1 ( did we just print a message? )
2022-03-23 22:55:03 -04:00
&modified $1 ( has the buffer been modified? )
&quitting $1 ( are we in the process of quitting? )
2022-03-07 22:38:02 -05:00
]
2024-07-07 23:45:18 -04:00
( prompt uses .tmp/pos and .tmp to track user input )
2022-03-07 22:38:02 -05:00
@prompt [
&active $1 ( is prompt currently active? )
&vector $2 ( what code to run when user responds )
2022-03-23 21:46:59 -04:00
&string $2 ( string to print for the prompt )
2022-03-07 22:38:02 -05:00
]
2022-04-04 00:40:37 -04:00
( temporary input buffer used for a variety of things )
2022-03-07 22:38:02 -05:00
@tmp [
2024-07-07 23:45:18 -04:00
$80 ( small scratch pad when reading data )
2022-03-07 22:38:02 -05:00
&pos $2 ( temporary pointer to address when reading data )
]
2024-07-07 23:45:18 -04:00
( search uses .tmp/pos and .tmp to track query string )
2022-03-29 00:02:22 -04:00
@searching [
2022-03-31 23:21:49 -04:00
&active $1 ( are we displaying search results? )
&orig-row $2 ( row we began the search at )
&orig-col $2 ( col we began the search at )
&regex $2 ( regex to be stored if any )
&start $2 ( absolute start pos of match )
&end $2 ( absolute limit pos of match )
2022-03-29 00:02:22 -04:00
]
2022-02-17 23:33:48 -05:00
( startup )
2022-03-07 22:38:02 -05:00
|0100
.Console/type DEI ?{ ;messages/usage print crlf quit! }
alternate-buffer-on
2022-04-07 00:16:25 -04:00
( init zero page )
2022-03-07 22:38:02 -05:00
#0050 .term/cols STZ2
#0018 .term/rows STZ2
#0004 .config/tab-width STZ2
#0003 .config/tab-adjust STZ2
2022-03-25 00:12:09 -04:00
#00 .config/insert-tabs STZ
2024-07-07 23:45:18 -04:00
#3333 .config/color STZ2
2023-11-28 10:33:25 -05:00
( #3133 .config/red STZ2 )
2024-07-06 15:33:54 -04:00
( #3033 .config/color STZ2 )
2023-11-28 10:33:25 -05:00
#3033 .config/red STZ2
2022-03-07 22:38:02 -05:00
;data .buffer/offset STZ2
2022-02-06 17:07:11 -05:00
2022-04-07 00:16:25 -04:00
( start reading the filename from argv )
2022-03-07 22:38:02 -05:00
;filename .tmp/pos STZ2
;read-filename .Console/vector DEO2
BRK
2022-02-09 02:35:46 -05:00
2022-04-07 00:16:25 -04:00
( import uxn regex library )
~regex.tal
2022-02-13 14:46:50 -05:00
( ERROR HANDLING )
2023-11-28 12:08:42 -05:00
( using errorq will print the given message before causing )
2022-02-13 14:46:50 -05:00
( the interpreter to halt. )
2023-11-28 12:08:42 -05:00
@errorq ( msg* -> )
emit-! sp print nl dbg BRK
2022-02-13 14:46:50 -05:00
2022-04-04 00:40:37 -04:00
( open the given file at editor start up )
( )
( this is called during startup by ;read-filename )
( )
( TODO: enable closing/opening files with editor already running )
2022-02-09 02:35:46 -05:00
@open-file ( filename* -> )
2022-02-09 20:04:12 -05:00
.File/name DEO2
2022-04-10 02:05:09 -04:00
#ce81 .File/length DEO2
2022-03-07 22:38:02 -05:00
;data .File/read DEO2
2022-02-06 17:07:11 -05:00
2022-04-09 00:05:12 -04:00
.File/success DEI2 #0000 EQU2 .state/modified STZ
2023-11-28 12:08:42 -05:00
.File/success DEI2 #ce81 LTH2 ?&ok
2022-04-12 12:15:42 -04:00
crlf
2023-11-28 12:08:42 -05:00
;messages/input-error print
;filename print crlf quit!
2022-02-06 17:07:11 -05:00
2022-02-09 20:04:12 -05:00
( calculate buffer limit address using start + size )
2022-03-07 22:38:02 -05:00
&ok .File/success DEI2 ;data ADD2 .buffer/limit STZ2
2022-02-09 02:35:46 -05:00
JMP2r
2022-02-09 01:05:11 -05:00
2022-04-04 00:40:37 -04:00
( ask the terminal for its size )
( )
( called during editor initialization by ;read-filename )
( )
( TODO: consider supporting terminal resizing )
2022-02-18 00:20:09 -05:00
@setup-terminal-size ( -> )
2023-11-28 12:08:42 -05:00
#03e7 DUP2 term-move-cursor
term-get-cursor-position
2024-07-07 23:45:18 -04:00
;tmp .tmp/pos STZ2
2022-02-09 02:35:46 -05:00
;receive-terminal-size .Console/vector DEO2
JMP2r
2022-04-04 00:40:37 -04:00
( receive size information from the terminal )
( )
( called from Console/vector after ;setup-terminal-size )
2022-02-18 00:20:09 -05:00
@receive-terminal-size ( -> )
2022-03-30 23:41:16 -04:00
.Console/read DEI .state/key STZ
.state/key LDZ .tmp/pos LDZ2 STA
2022-03-07 22:38:02 -05:00
.tmp/pos LDZ2 INC2 .tmp/pos STZ2
2023-11-28 12:08:42 -05:00
.state/key LDZ LIT "R EQU ?parse-terminal-size
2022-02-09 02:35:46 -05:00
BRK
2022-04-04 00:40:37 -04:00
( parse and store terminal size information )
( )
( called by ;receive-terminal-size after complete message received )
2022-02-09 02:35:46 -05:00
@parse-terminal-size ( -> )
2022-04-06 20:39:02 -04:00
#0000 ,&acc STR2
2024-07-07 23:45:18 -04:00
.tmp LDZk #1b NEQ ?&parse-error ( i ) INC
LDZk LIT "[ NEQ ?&parse-error ( i ) INC
2022-02-09 02:35:46 -05:00
&loop
2023-11-28 12:08:42 -05:00
LDZk LIT "; EQU ?&parse-col
2024-07-07 23:45:18 -04:00
LIT2r =&loop !&read
2022-02-09 02:35:46 -05:00
&parse-col
2022-04-06 20:39:02 -04:00
INC ,&acc LDR2 #0002 SUB2 .term/rows STZ2
#0000 ,&acc STR2
2022-02-09 02:35:46 -05:00
&loop2
2023-11-28 12:08:42 -05:00
LDZk LIT "R EQU ?&done
2024-07-07 23:45:18 -04:00
LIT2r =&loop2 !&read
2022-04-06 20:39:02 -04:00
&read
2022-09-10 13:30:15 -04:00
LDZk LIT "0 SUB #00 SWP
2022-04-06 20:39:02 -04:00
,&acc LDR2 #000a MUL2 ADD2 ,&acc STR2
INC JMP2r
2022-02-09 02:35:46 -05:00
&done
2022-04-06 20:39:02 -04:00
,&acc LDR2 .term/cols STZ2 POP
2022-02-09 02:35:46 -05:00
;on-key .Console/vector DEO2
2023-11-28 12:08:42 -05:00
draw-all
2022-02-09 02:35:46 -05:00
BRK
2022-04-06 20:39:02 -04:00
[ &acc $2 ]
2024-07-07 23:45:18 -04:00
&parse-error POP .tmp LDZ2
2023-11-28 12:08:42 -05:00
;messages/term-size-parse-error !errorq
2022-02-09 02:35:46 -05:00
2022-04-06 20:39:02 -04:00
@count-c ( c^ -> n* )
2024-07-07 23:45:18 -04:00
STH #0000 ;data ( 0* data* [c^] )
&loop LDAk #00 EQU ?&done ( n* data* [c^] )
LDAk STHkr NEQ ?&next ( n* data* [c^] )
SWP2 INC2 SWP2 ( n+1* data* [c^] )
&next INC2 !&loop ( n+1* data+1* [c^] )
&done POP2 POPr JMP2r ( n* )
2022-04-06 20:39:02 -04:00
2022-04-04 00:40:37 -04:00
( save count of number of lines in input file )
( )
( this method also detects whether \t characters are used, )
( and uses this to initialize config/insert-tabs. )
2022-02-11 15:39:49 -05:00
@setup-linecount ( -> )
2023-11-28 12:08:42 -05:00
#0a count-c INC2 .buffer/line-count STZ2
#09 count-c #0000 GTH2 .config/insert-tabs STZ
2022-04-06 22:41:25 -04:00
JMP2r
2022-02-11 15:39:49 -05:00
2022-04-04 00:40:37 -04:00
( reads filename from the program's argv )
( )
( currently femto must be given a file to edit, and reading this )
( filename is the first thing that happens in ;startup. )
( )
( TODO: support other situations, such as: )
( - launching femto without a file name )
( - closing the given file and opening a new one )
2022-02-09 02:35:46 -05:00
@read-filename ( -> )
2023-11-28 12:08:42 -05:00
#12 DEI #0a EQU ?&execute ( did we read \n ? )
2022-03-07 22:38:02 -05:00
#12 DEI .tmp/pos LDZ2 STA ( no, so save in buffer )
.tmp/pos LDZ2 INC2 .tmp/pos STZ2 ( pos++ )
2022-02-11 15:39:49 -05:00
BRK ( return )
&execute ( we saw a newline, so do something )
2022-03-07 22:38:02 -05:00
#00 .tmp/pos LDZ2 STA ( null terminate str )
2023-11-28 12:08:42 -05:00
;filename open-file ( open file )
setup-linecount ( determine # of lines )
setup-terminal-size ( detect terminal dimensions )
2022-02-06 17:07:11 -05:00
BRK
2022-04-04 00:40:37 -04:00
( jump to beginning of line )
2022-02-18 00:20:09 -05:00
@bol ( -> )
2022-03-07 22:38:02 -05:00
#0000 .cursor/col STZ2
2023-11-28 12:08:42 -05:00
redraw-statusbar-and-cursor
!return
2022-02-06 17:07:11 -05:00
2022-04-04 00:40:37 -04:00
( jump to beginning of line )
2022-02-18 00:20:09 -05:00
@eol ( -> )
2023-11-28 12:08:42 -05:00
cur-len .cursor/col STZ2
redraw-statusbar-and-cursor
!return
2022-02-06 17:07:11 -05:00
2022-02-18 00:20:09 -05:00
@forward ( -> )
2023-11-28 12:08:42 -05:00
go-forward !return
2022-04-06 23:26:07 -04:00
( move forward by one character )
@go-forward ( -> )
2023-11-28 12:08:42 -05:00
cur-pos last-pos GTH2 ( ?return ) ?&noop
redraw-statusbar-and-cursor
cur-col cur-len LTH2 ?&normal
#0000 .cursor/col STZ2
.cursor/row LDZ2 INC2 .cursor/row STZ2
2023-11-28 12:08:42 -05:00
!ensure-visible-cursor
2022-03-25 23:17:34 -04:00
&normal
2023-11-28 12:08:42 -05:00
cur-col INC2 .cursor/col STZ2
2022-04-06 23:26:07 -04:00
&noop JMP2r
2022-02-06 17:07:11 -05:00
2022-04-04 00:40:37 -04:00
( move backward by one character )
@back ( -> )
2023-11-28 12:08:42 -05:00
go-back !return
2022-04-04 00:40:37 -04:00
( internal implementation shared by ;back and ;backspace )
2022-03-30 01:26:33 -04:00
@go-back ( -> )
2023-11-28 12:08:42 -05:00
cur-pos ;data EQU2 ?&noop
cur-col #0001 LTH2 ?&next-line
cur-col #0001 SUB2 .cursor/col STZ2
!redraw-statusbar-and-cursor
2022-03-25 23:17:34 -04:00
&next-line
2022-04-01 23:47:37 -04:00
.cursor/row LDZ2k #0001 SUB2 ROT STZ2
2023-11-28 12:08:42 -05:00
cur-len .cursor/col STZ2
ensure-visible-cursor
redraw-statusbar-and-cursor
2022-04-01 23:47:37 -04:00
&noop JMP2r
2022-03-30 01:26:33 -04:00
2022-04-04 00:40:37 -04:00
( move up by one line )
2022-02-18 00:20:09 -05:00
@up ( -> )
2023-11-28 12:08:42 -05:00
.cursor/row LDZ2 #0000 EQU2 ?return
2022-04-01 00:44:07 -04:00
.cursor/row LDZ2 #0001 SUB2 .cursor/row STZ2
2023-11-28 12:08:42 -05:00
ensure-visible-cursor
redraw-statusbar-and-cursor
!return
2022-04-01 23:47:37 -04:00
2022-04-04 00:40:37 -04:00
( move down by one line )
2022-02-18 00:20:09 -05:00
@down ( -> )
2022-04-04 00:40:37 -04:00
.cursor/row LDZ2
2023-11-28 12:08:42 -05:00
.buffer/line-count LDZ2 #0001 SUB2 EQU2 ?return
2022-04-01 00:44:07 -04:00
.cursor/row LDZ2 INC2 .cursor/row STZ2
2023-11-28 12:08:42 -05:00
ensure-visible-cursor
redraw-statusbar-and-cursor
!return
2022-02-06 17:07:11 -05:00
2022-04-06 23:26:07 -04:00
@is-word-char ( c^ -> bool^ )
DUP #2f GTH OVR #3a LTH AND STH
DUP #40 GTH OVR #5b LTH AND STH ORAr
DUP #60 GTH SWP #7b LTH AND STHr ORA JMP2r
@not-word-char ( c^ -> bool^ )
2023-11-28 12:08:42 -05:00
is-word-char #00 EQU JMP2r
2022-04-06 23:26:07 -04:00
@forward-by-word ( -> )
2023-11-28 12:08:42 -05:00
cur-pos
2022-04-06 23:26:07 -04:00
&first
2023-11-28 12:08:42 -05:00
LDAk #00 EQU ?&done
LDAk is-word-char ?&second
INC2 go-forward !&first
2022-04-06 23:26:07 -04:00
&second
2023-11-28 12:08:42 -05:00
LDAk #00 EQU ?&done
LDAk not-word-char ?&done
INC2 go-forward !&second
2022-04-06 23:26:07 -04:00
&done
2023-11-28 12:08:42 -05:00
POP2 !return
2022-04-06 23:26:07 -04:00
@back-by-word ( -> )
2023-11-28 12:08:42 -05:00
cur-pos #0001 SUB2
2022-04-06 23:26:07 -04:00
&first
2023-11-28 12:08:42 -05:00
DUP2 ;data LTH2 ?&done
LDAk is-word-char ?&second
#0001 SUB2 go-back !&first
2022-04-06 23:26:07 -04:00
&second
2023-11-28 12:08:42 -05:00
DUP2 ;data LTH2 ?&done
LDAk not-word-char ?&done
#0001 SUB2 go-back !&second
2022-04-06 23:26:07 -04:00
&done
2023-11-28 12:08:42 -05:00
POP2 !return
2022-04-06 23:26:07 -04:00
@help
#01 .state/in-help STZ
2023-11-28 12:08:42 -05:00
term-erase-all
#0000 #0000 term-move-cursor
emit-color-bold
;help-text print
emit-reset
redraw-all
BRK
2022-04-04 00:40:37 -04:00
( center buffer view on the current line )
@center-view
2022-03-31 01:52:20 -04:00
.term/rows LDZ2 INC2 #0002 DIV2 STH2k
2023-11-28 12:08:42 -05:00
.cursor/row LDZ2 LTH2 ?&standard
POP2r
#0000 .buffer/line-offset STZ2
;data .buffer/offset STZ2
2023-11-28 12:08:42 -05:00
!&done
&standard
.cursor/row LDZ2 STH2r SUB2
DUP2 .buffer/line-offset STZ2
2023-11-28 12:08:42 -05:00
abs-line .buffer/offset STZ2
&done
2023-11-28 12:08:42 -05:00
redraw-all !return
2022-04-04 00:40:37 -04:00
( move up by one page )
@page-up ( -> )
.term/rows LDZ2 #0002 SUB2 STH2k
2023-11-28 12:08:42 -05:00
.buffer/line-offset LDZ2 LTH2 ?&move-full
POP2r
2023-11-28 12:08:42 -05:00
zero-row
2022-03-07 22:38:02 -05:00
#0000 .cursor/col STZ2
2023-11-28 12:08:42 -05:00
!&done
&move-full
.cursor/row LDZ2 STH2kr SUB2 .cursor/row STZ2
.buffer/line-offset LDZ2 STH2r SUB2
DUP2 .buffer/line-offset STZ2
2023-11-28 12:08:42 -05:00
abs-line .buffer/offset STZ2
&done
2023-11-28 12:08:42 -05:00
redraw-all !return
2022-03-30 01:26:33 -04:00
2022-04-04 00:40:37 -04:00
( move down by one page )
2022-02-15 21:58:12 -05:00
@page-down
2023-11-28 12:08:42 -05:00
eof-is-visible ?&near-eof
.term/rows LDZ2 #0002 SUB2 STH2k
.buffer/line-offset LDZ2 ADD2
DUP2 .buffer/line-offset STZ2
2023-11-28 12:08:42 -05:00
abs-line .buffer/offset STZ2
.cursor/row LDZ2 STH2r ADD2 .cursor/row STZ2
2023-11-28 12:08:42 -05:00
redraw-all !return
2022-02-15 21:58:12 -05:00
&near-eof
2022-04-01 23:47:37 -04:00
.buffer/line-count LDZ2 #0001 SUB2 .cursor/row STZ2
2023-11-28 12:08:42 -05:00
cur-len .cursor/col STZ2
redraw-cursor !return
2022-02-15 21:58:12 -05:00
2022-04-04 00:40:37 -04:00
( return true if the end of the file is visible )
@eof-is-visible ( -> bool^ )
.buffer/line-offset LDZ2 .term/rows LDZ2 ADD2 INC2
.buffer/line-count LDZ2
GTH2 JMP2r
( beginning quitting femto, prompting if unsaved changes )
2022-03-23 22:55:03 -04:00
@quit
#01 .state/quitting STZ
2023-11-28 12:08:42 -05:00
.state/modified LDZ #00 EQU ?quit-now
;messages/quit-prompt ;messages/null ;do-quit start-prompt
redraw-prompt-and-cursor !return
2022-02-07 22:25:32 -05:00
2022-04-04 00:40:37 -04:00
( display two strings on the message line )
( )
2024-07-07 23:45:18 -04:00
( often this involves a static messages + an argument like ;tmp. )
2022-04-04 00:40:37 -04:00
( )
( use messages/null for the second string if only one is needed. )
2022-04-01 23:47:37 -04:00
@send-message ( s1* s2* -> )
#01 .state/message STZ
2023-11-28 12:08:42 -05:00
move-to-message-line
SWP2 print !print
2022-04-01 23:47:37 -04:00
2022-04-04 00:40:37 -04:00
( callback executed in response to the quit prompt. )
2022-03-23 22:55:03 -04:00
@do-quit
2024-07-07 23:45:18 -04:00
.tmp LDZ LIT "n EQU ?quit-now
.tmp LDZ LIT "y EQU ?save
2022-03-23 22:55:03 -04:00
#00 .state/quitting STZ
2024-07-07 23:45:18 -04:00
;messages/unknown-input ;tmp send-message
2022-03-23 22:55:03 -04:00
BRK
( label that calls quit-restore! )
2022-04-04 00:40:37 -04:00
( )
( this definition is needed so the address can be used by JCN2. )
@quit-now quit-restore!
2022-03-23 22:55:03 -04:00
2022-04-04 00:40:37 -04:00
( label that calls BRK )
( )
( this definition is needed so the address can be used by JCN2. )
2022-03-23 22:55:03 -04:00
@ignore
2022-04-04 00:40:37 -04:00
BRK
2022-02-07 22:25:32 -05:00
2022-04-04 00:40:37 -04:00
( insert the given character at the cursor position )
( )
( this should not be called for newlines, see ;newline )
2022-02-06 17:07:11 -05:00
@insert ( c^ -> )
2023-11-28 12:08:42 -05:00
cur-pos shift-right
cur-col INC2 .cursor/col STZ2
redraw-all !return
2022-02-07 22:25:32 -05:00
2022-04-04 00:40:37 -04:00
( insert the given character in the prompt )
2022-03-23 22:12:17 -04:00
@insert-prompt ( c^ -> )
.tmp/pos LDZ2 STH2k STA ( data[pos] <- c )
INC2r #00 STH2kr STA ( data[pos+1] <- 0 )
STH2r .tmp/pos STZ2 ( pos <- pos+1 )
2023-11-28 12:08:42 -05:00
redraw-prompt-and-cursor !return
2022-03-23 22:12:17 -04:00
2022-04-04 00:40:37 -04:00
( insert a tab at the cursor position )
( )
( depending on the state of config/insert-tabs this will )
( either call ;insert with \t or else insert a number of )
( spaces based on .config/tab-width. )
@insert-tab ( -> )
2023-11-28 12:08:42 -05:00
.config/insert-tabs LDZ ?&use-tabs
2022-03-07 22:38:02 -05:00
#0000 .config/tab-width LDZ2 SUB2
&loop
2023-11-28 12:08:42 -05:00
DUP2 #0000 EQU2 ?&done
#20 cur-pos shift-right
INC2 !&loop
&done
2023-11-28 12:08:42 -05:00
cur-col .config/tab-width LDZ2 ADD2 .cursor/col STZ2
redraw-all !return
&use-tabs
2023-11-28 12:08:42 -05:00
#09 !insert
2022-04-04 00:40:37 -04:00
( insert a newline at the cursor position )
2022-02-07 22:25:32 -05:00
@newline ( c^ -> )
2023-11-28 12:08:42 -05:00
#0a cur-pos shift-right
2022-03-07 22:38:02 -05:00
#0000 .cursor/col STZ2
2022-04-01 00:44:07 -04:00
.cursor/row LDZ2 INC2 .cursor/row STZ2
2022-03-23 21:46:59 -04:00
.buffer/line-count LDZ2k INC2 ROT STZ2
2023-11-28 12:08:42 -05:00
ensure-visible-cursor
redraw-all !return
2022-02-07 22:25:32 -05:00
2022-04-04 00:40:37 -04:00
( delete the character to the left of the cursor, if any )
2022-02-06 17:07:11 -05:00
@backspace ( -> )
2023-11-28 12:08:42 -05:00
cur-pos ;data EQU2 ?return
go-back !delete
2022-03-25 23:17:34 -04:00
2022-04-04 00:40:37 -04:00
( delete the last character in the prompt )
2022-03-23 22:12:17 -04:00
@backspace-prompt ( -> )
2024-07-07 23:45:18 -04:00
.tmp/pos LDZ2 ;tmp EQU2 ?&skip ( ?return )
2022-03-23 22:12:17 -04:00
#00 .tmp/pos LDZ2 #0001 SUB2 ( 0 pos-1 )
STH2k STA ( data[pos-1] <- 0 )
STH2r .tmp/pos STZ2 ( pos <- pos-1 )
2023-11-28 12:08:42 -05:00
&skip redraw-prompt-and-cursor !return
2022-03-23 22:12:17 -04:00
2022-04-04 00:40:37 -04:00
( delete the character under the cursor, if any )
2022-02-07 22:25:32 -05:00
@delete ( -> )
2023-11-28 12:08:42 -05:00
last-pos cur-pos LTH2 ?return
cur-pos LDAk STH ( cur [c] )
shift-left ( [c] )
STHr #0a NEQ ?&not-newline
2022-03-23 21:46:59 -04:00
.buffer/line-count LDZ2k #0001 SUB2 ROT STZ2
2023-11-28 12:08:42 -05:00
&not-newline redraw-all !return
2022-02-06 17:07:11 -05:00
2022-04-04 00:40:37 -04:00
( used at the start of an escape sequence to set up state. )
( )
( many keys such as page-down will actually send an escape character )
( followed by others. to support these we use saw-esc to interpret )
( input characters differently. )
( )
( see also state/saw-xterm which supports such sequences. )
2022-02-08 22:22:08 -05:00
@escape ( -> )
2022-03-07 22:38:02 -05:00
#01 .state/saw-esc STZ BRK
2022-02-08 22:22:08 -05:00
2022-04-04 00:40:37 -04:00
( move to the end of the file )
2022-02-12 02:25:41 -05:00
@goto-end ( -> )
.buffer/line-count LDZ2 #0001 SUB2 .cursor/row STZ2
2023-11-28 12:08:42 -05:00
.buffer/line-count LDZ2 .term/rows LDZ2 LTH2k ?&use-zero
SUB2 #0002 ADD2 !&continue
&use-zero
POP2 POP2 #0000
2022-02-13 14:46:50 -05:00
&continue
2022-03-07 22:38:02 -05:00
DUP2 .buffer/line-offset STZ2
2023-11-28 12:08:42 -05:00
abs-line .buffer/offset STZ2
cur-len .cursor/col STZ2
redraw-all !return
2022-02-12 02:25:41 -05:00
2022-04-04 00:40:37 -04:00
( move to the start of the file )
2022-02-08 22:22:08 -05:00
@goto-start ( -> )
2023-11-28 12:08:42 -05:00
zero-row
2022-03-07 22:38:02 -05:00
#0000 .cursor/col STZ2
2023-11-28 12:08:42 -05:00
redraw-all !return
2022-03-30 01:26:33 -04:00
2022-04-04 00:40:37 -04:00
( prompt for a line number and move to that line )
2022-02-10 01:40:48 -05:00
@goto-line ( -> )
2023-11-28 12:08:42 -05:00
;messages/goto-line ;messages/null ;do-goto-line start-prompt
redraw-prompt-and-cursor !return
2022-03-24 00:09:55 -04:00
2022-04-04 00:40:37 -04:00
( parse the given string as a decimal number )
( )
( returns the number as a short followed by whether parsing was ok )
2022-03-24 00:09:55 -04:00
@parse-decimal-number ( addr* -> n* ok^ )
2023-11-28 12:08:42 -05:00
LDAk ?&non-empty
2022-03-24 00:09:55 -04:00
#00 JMP2r
&non-empty
LIT2r 0000
&loop
2023-11-28 12:08:42 -05:00
LDAk ?&continue
2022-03-24 00:09:55 -04:00
POP2 STH2r #01 JMP2r
&continue
2023-11-28 12:08:42 -05:00
LDAk LIT "0 LTH ?&fail
LDAk LIT "9 GTH ?&fail
2022-03-24 00:09:55 -04:00
LIT2r 000a MUL2r
2022-09-10 13:30:15 -04:00
LDAk LIT "0 SUB #00 SWP STH2 ADD2r
2023-11-28 12:08:42 -05:00
INC2 !&loop
2022-03-24 00:09:55 -04:00
&fail
POP2r #00 JMP2r
2022-04-04 00:40:37 -04:00
( go to the given line number )
( )
( this is used as a callback from the goto-line prompt )
@do-goto-line ( n* -> )
2024-07-07 23:45:18 -04:00
;tmp parse-decimal-number
2023-11-28 12:08:42 -05:00
?&ok
2024-07-07 23:45:18 -04:00
;messages/unknown-input ;tmp send-message
2023-11-28 12:08:42 -05:00
!return
2022-03-24 00:09:55 -04:00
&ok
#0001 SUB2 ( convert 1-indexing to 0-indexing )
2023-11-28 12:08:42 -05:00
DUP2 .buffer/line-count LDZ2 LTH2 ?&within
POP2 !goto-end
2022-03-24 00:09:55 -04:00
&within
2023-11-28 12:08:42 -05:00
jump-to-line
redraw-all !return
2022-02-10 01:40:48 -05:00
2022-04-04 00:40:37 -04:00
( move the cursor to the given coordinates )
( )
( this won't move the display if the given coordinates are visible. )
2022-03-31 13:48:08 -04:00
@move-to-coord ( col* row* -> )
2023-11-28 12:08:42 -05:00
DUP2 line-is-visible ?jump-to-coord/short
!jump-to-coord
2022-03-31 13:48:08 -04:00
2022-04-04 00:40:37 -04:00
( move the cursor to the given coordinates )
( )
( this will always ensure the display is centered on the given coordinates )
2022-03-31 13:48:08 -04:00
@jump-to-coord ( x* y* -> )
2023-11-28 12:08:42 -05:00
.term/rows LDZ2 INC2 #0002 DIV2 LTH2k ( x y rows/2 y<rows/2? ) ?&early
2022-03-31 13:48:08 -04:00
OVR2 SWP2 SUB2 ( x y y-rows/2 )
.buffer/line-count LDZ2 ( x y y-rows/2 lines )
.term/rows LDZ2 SUB2 ( x y y-rows/2 lines-rows )
GTH2k ( x y y-rows/2 lines-rows y-rows/2>lines-rows? )
2023-11-28 12:08:42 -05:00
?&late ( x y y-rows/2 lines-rows )
POP2 !&finish
2022-03-31 13:48:08 -04:00
&early ( x y rows/2 )
2023-11-28 12:08:42 -05:00
POP2 #0000 !&finish ( x y 0000 )
2022-03-31 13:48:08 -04:00
&late ( x y y-rows/2 lines-rows )
2022-02-14 16:47:36 -05:00
NIP2
2022-03-31 13:48:08 -04:00
&finish ( x y o )
2023-11-28 12:08:42 -05:00
redraw-all
2022-03-31 13:48:08 -04:00
SUB2k STH2 DUP2 ( x y o o [y-o] )
.buffer/line-offset STZ2 ( x y o [y-o] )
2023-11-28 12:08:42 -05:00
abs-line .buffer/offset STZ2 ( x y [y-o] )
2022-03-31 13:48:08 -04:00
POP2r
&short
2023-11-28 12:08:42 -05:00
redraw-statusbar-and-cursor
2022-03-31 13:48:08 -04:00
.cursor/row STZ2 ( x )
.cursor/col STZ2
2022-02-10 01:40:48 -05:00
JMP2r
2022-04-04 00:40:37 -04:00
( jump to the given line number )
2022-03-31 13:48:08 -04:00
@jump-to-line ( n* -> )
2023-11-28 12:08:42 -05:00
#0000 SWP2 !jump-to-coord
2022-03-31 13:48:08 -04:00
2022-04-04 00:40:37 -04:00
( ensure the cursor is visibe )
( )
( if the cursor is not already visible the screen will be )
( centered on the cursor's coordinates. )
2022-03-07 22:38:02 -05:00
@ensure-visible-cursor
.cursor/row LDZ2 .buffer/line-offset LDZ2
2023-11-28 12:08:42 -05:00
SUB2 .term/rows LDZ2 LTH2 ?&noop
.cursor/row LDZ2 jump-to-line
redraw-all
2022-02-14 16:47:36 -05:00
&noop JMP2r
2022-04-04 00:40:37 -04:00
( currently used to print stack information. )
2022-02-13 14:46:50 -05:00
@debug
2023-11-28 12:08:42 -05:00
;messages/input-error !errorq
2022-02-13 14:46:50 -05:00
2022-04-04 00:40:37 -04:00
( move the terminal's cursor to the message line )
( )
( this low level method does not change any editor state (such as )
( the cursor) but is used to display messages. )
2022-02-15 21:19:44 -05:00
@move-to-message-line ( -> )
2023-11-28 12:08:42 -05:00
#0000 .term/rows LDZ2 #0002 ADD2 !term-move-cursor
2022-02-15 21:19:44 -05:00
2022-04-04 00:40:37 -04:00
( start a prompt on the message line )
( )
( the arguments are as follows: )
( - the prompt string will printed in bold )
( - the default string will be editable )
( - the vector address will be used on return )
( )
( prompts can always be cancelled using C-g. )
( )
( when called vector should end in a BRK instructinon. )
2022-03-23 21:46:59 -04:00
@start-prompt ( prompt* default* vector* -> )
2023-11-28 12:08:42 -05:00
.prompt/active LDZ ?&is-active
2022-03-23 21:46:59 -04:00
#01 .prompt/active STZ ( prompt/active <- 1 )
.prompt/vector STZ2 ( prompt/vector <- vector )
2024-07-07 23:45:18 -04:00
;tmp str-copy ( tmp <- default )
;tmp ADD2 .tmp/pos STZ2 ( tmp/pos <- len(default)+data )
2022-03-23 21:46:59 -04:00
.prompt/string STZ2 ( prompt/string <- prompt )
JMP2r
&is-active
#0000 DIV
( ends prompt without calling vector )
@cancel-prompt ( -> )
2022-04-01 23:47:37 -04:00
#00 .prompt/active STZ
#00 .state/quitting STZ
2023-11-28 12:08:42 -05:00
clear-message-line
redraw-prompt-and-cursor
!return
2022-03-23 21:46:59 -04:00
2022-04-04 00:40:37 -04:00
( finishes prompt and executes callback )
( )
( when called vector should end in a BRK instruction )
2022-03-23 21:46:59 -04:00
@finish-prompt ( -> )
2022-04-01 23:47:37 -04:00
#00 .prompt/active STZ
2023-11-28 12:08:42 -05:00
clear-message-line
redraw-prompt-and-cursor
2022-04-01 23:47:37 -04:00
.prompt/vector LDZ2 JMP2
2022-03-23 21:46:59 -04:00
2022-04-04 00:40:37 -04:00
( begin saving the file, prompting the user for a fiel name )
2022-03-23 21:46:59 -04:00
@save
2023-11-28 12:08:42 -05:00
;messages/save-prompt ;filename ;do-save start-prompt
redraw-prompt-and-cursor
!return
2022-03-23 21:46:59 -04:00
2024-07-07 23:45:18 -04:00
( save the file with the filename found in tmp )
2022-03-23 21:46:59 -04:00
@do-save ( -> )
2022-03-07 22:38:02 -05:00
.buffer/limit LDZ2 ;data SUB2 STH2 ( [size] )
2024-07-07 23:45:18 -04:00
;tmp .File/name DEO2
2022-03-23 21:46:59 -04:00
STH2kr .File/length DEO2
;data .File/write DEO2
2022-02-15 21:19:44 -05:00
2023-11-28 12:08:42 -05:00
.File/success DEI2 STH2r EQU2 ?&ok
;messages/save-failed !&finish
2022-03-24 01:26:06 -04:00
&ok
#00 .state/modified STZ
2024-07-07 23:45:18 -04:00
;tmp ;filename str-copy POP2
2022-04-01 23:47:37 -04:00
;messages/save-ok
2022-03-23 22:55:03 -04:00
&finish
2024-07-07 23:45:18 -04:00
;tmp send-message
2023-11-28 12:08:42 -05:00
.state/quitting LDZ ?quit-now
2022-04-09 00:44:14 -04:00
#03 .state/redraw STZ ( FIXME: why do we have to do this? )
2023-11-28 12:08:42 -05:00
!return
2022-02-15 21:19:44 -05:00
2022-04-04 00:40:37 -04:00
( begin a search, prompting for a search string )
2022-03-29 00:02:22 -04:00
@search ( -> )
2023-11-28 12:08:42 -05:00
;messages/search-prompt ;messages/null ;do-search start-prompt
redraw-prompt-and-cursor
!return
2022-03-29 00:02:22 -04:00
2022-04-04 00:40:37 -04:00
( execute a search, using the given search string )
2022-03-29 00:02:22 -04:00
@do-search ( -> )
.cursor/row LDZ2 .searching/orig-row STZ2
.cursor/col LDZ2 .searching/orig-col STZ2
#0000 .searching/regex STZ2
2023-11-28 12:08:42 -05:00
move-to-next-match ?&found
move-to-prev-match ?&found
2024-07-07 23:45:18 -04:00
;messages/no-matches-found ;tmp send-message BRK
2022-04-01 23:47:37 -04:00
&found
#01 .searching/active STZ
2023-11-28 12:08:42 -05:00
redraw-matches
!return
2022-03-31 23:21:49 -04:00
2022-04-04 00:40:37 -04:00
( begin a search, prompting for a regular expression )
@regex-search ( -> )
2023-11-28 12:08:42 -05:00
;messages/regex-search-prompt ;messages/null ;do-regex-search start-prompt
redraw-prompt-and-cursor !return
2022-03-31 23:21:49 -04:00
2022-04-04 00:40:37 -04:00
( execute a search, using the given regular expressions )
( )
( TODO: handle invalid regular expressions that fail to compile )
2022-03-31 23:21:49 -04:00
@do-regex-search ( -> )
2023-11-28 12:08:42 -05:00
cur-pos DUP2 .searching/start STZ2 .searching/end STZ2
.cursor/row LDZ2 .searching/orig-row STZ2
.cursor/col LDZ2 .searching/orig-col STZ2
2024-07-07 23:45:18 -04:00
;tmp compile .searching/regex STZ2
2023-11-28 12:08:42 -05:00
move-to-next-regex-match ?&found
move-to-prev-regex-match ?&found
2024-07-07 23:45:18 -04:00
;messages/no-matches-found ;tmp send-message BRK
2022-04-01 23:47:37 -04:00
&found
#01 .searching/active STZ
2023-11-28 12:08:42 -05:00
redraw-matches
!return
2022-03-29 00:02:22 -04:00
2022-04-04 00:40:37 -04:00
( toggle the color used by the terminal )
( )
( available colors are: )
( - black )
( - red )
( - green )
( - yellow )
( - blue )
( - magenta )
( - cyan )
( - white )
2022-03-24 01:35:35 -04:00
@toggle-color ( -> )
2023-11-28 12:08:42 -05:00
.config/color LDZ2 #3733 EQU2 ?{
.config/color LDZ2 #0100 ADD2 DUP2 .config/color STZ2
DUP2 #3733 LTH2 ?{ .config/red STZ2 !&done }
POP2 #3133 .config/red STZ2 !&done
}
#3033 DUP2 .config/color STZ2 .config/red STZ2
&done redraw-all !return
2022-03-24 01:35:35 -04:00
2022-04-04 00:40:37 -04:00
( toggle whether to use literal tab characters )
( )
( when opening a file, this defaults to 01 if existing tab )
( characters are found, and 00 otherwise. )
2022-03-25 00:12:09 -04:00
@toggle-tabs ( -> )
2022-04-06 20:39:02 -04:00
.config/insert-tabs LDZk #00 EQU SWP STZ
2023-11-28 12:08:42 -05:00
redraw-statusbar-and-cursor !return
2022-04-06 20:39:02 -04:00
2022-04-04 00:40:37 -04:00
( interpret user input as an escaped sequence )
( )
( called by on-key with state/saw-esc is true )
( )
2022-04-06 23:26:07 -04:00
( TODO: maybe M-% for search&replace )
2022-03-23 21:46:59 -04:00
@on-key-escaped ( -> )
#00 .state/saw-esc STZ
2023-11-28 12:08:42 -05:00
.state/key LDZ LIT "< EQU ( M-< ) ?goto-start
.state/key LDZ LIT "> EQU ( M-> ) ?goto-end
.state/key LDZ LIT "b EQU ( M-b ) ?back-by-word
.state/key LDZ LIT "c EQU ( M-c ) ?toggle-color
.state/key LDZ LIT "f EQU ( M-f ) ?forward-by-word
.state/key LDZ LIT "g EQU ( M-g ) ?goto-line
.state/key LDZ LIT "h EQU ( M-h ) ?help
.state/key LDZ LIT "s EQU ( M-s ) ?regex-search
.state/key LDZ LIT "t EQU ( M-t ) ?toggle-tabs
.state/key LDZ LIT "u EQU ( M-u ) ?undo
.state/key LDZ LIT "v EQU ( M-v ) ?page-up
.state/key LDZ LIT "[ EQU ( M-[ ) ?xterm
2022-03-24 01:26:06 -04:00
BRK
2022-04-04 00:40:37 -04:00
( set our input to expect xterm control sequences )
( )
( after seeing ESC followed by [ we expect various )
( ANSI or xterm control sequences. these include )
( things like: )
( - up/down/left/right arrow keys )
( - page up/page down keys )
( - end/home keys )
2022-03-24 01:26:06 -04:00
@xterm
#01 .state/saw-xterm STZ BRK
2022-04-04 00:40:37 -04:00
( after seeing sequences like "ESC [ 1" we expect )
( to see a trailing ~ to complete the sequence. )
( )
( this callback checks for and if set performs )
( the relevant action. )
2022-03-24 01:26:06 -04:00
@on-key-vt ( -> )
.state/saw-vt LDZk STH #00 SWP STZ
2023-11-28 12:08:42 -05:00
.state/key LDZ LIT "~ EQU ?&ok
2022-03-24 01:26:06 -04:00
POPr BRK
&ok
2023-11-28 12:08:42 -05:00
STHr DUP LIT "1 NEQ ?&not-1
( ^[[1~ -> home ) POP !bol
&not-1 DUP LIT "2 NEQ ?&not-2
2022-03-24 01:26:06 -04:00
( ^[[2~ -> insert ) POP BRK
2023-11-28 12:08:42 -05:00
&not-2 DUP LIT "3 NEQ ?&not-3
( ^[[3~ -> delete ) POP !delete
&not-3 DUP LIT "4 NEQ ?&not-4
( ^[[4~ -> end ) POP !eol
&not-4 DUP LIT "5 NEQ ?&not-5
( ^[[5~ -> page up ) POP !page-up
&not-5 DUP LIT "6 NEQ ?&not-6
( ^[[6~ -> page down ) POP !page-down
&not-6 DUP LIT "7 NEQ ?&not-7
( ^[[7~ -> home ) POP !bol
&not-7 DUP LIT "8 NEQ ?&not-8
( ^[[8~ -> end ) POP !eol
2022-03-24 01:26:06 -04:00
&not-8
( ??? ) POP BRK
2022-04-04 00:40:37 -04:00
( after seeing sequences like "ESC [" we expect )
( to see more characters to determine the logical key. )
( )
( this callback performs the relevant action, )
( or else sets (or unsets) state as necessary )
( to continue (or end) the sequence. )
2022-03-24 01:26:06 -04:00
@on-key-xterm ( -> )
#00 .state/saw-xterm STZ
2023-11-28 12:08:42 -05:00
.state/key LDZ LIT "A EQU ( ^[[A -> up ) ?up
.state/key LDZ LIT "B EQU ( ^[[B -> down ) ?down
.state/key LDZ LIT "C EQU ( ^[[C -> right ) ?forward
.state/key LDZ LIT "D EQU ( ^[[D -> left ) ?back
.state/key LDZ LIT "F EQU ( ^[[F -> end ) ?eol
.state/key LDZ LIT "H EQU ( ^[[H -> home ) ?bol
.state/key LDZ LIT "0 LTH ?ignore
.state/key LDZ LIT "8 GTH ?ignore
2022-03-30 23:41:16 -04:00
.state/key LDZ .state/saw-vt STZ ( ^[[1 through ^[[8 )
2022-03-23 21:46:59 -04:00
BRK
2022-04-04 00:40:37 -04:00
( clear the message line )
( )
( this includes the code needed to move the cursor )
( to that line, the ANSI control sequence to clear )
( the line, and unsetting state/message. )
( )
( if state/message is unset this is a no-op. )
2022-03-23 22:55:03 -04:00
@clear-message-line
2023-11-28 12:08:42 -05:00
.state/message LDZ #00 EQU ?&done
move-to-message-line
term-erase-line
2022-04-01 23:47:37 -04:00
#00 .state/message STZ
&done JMP2r
2022-03-23 22:55:03 -04:00
2022-04-04 00:40:37 -04:00
( cancel the active search )
( )
( this method unsets searching/active and also restores )
( the original cursor position. )
2022-03-29 00:02:22 -04:00
@cancel-search
2023-11-28 12:08:42 -05:00
.searching/orig-row LDZ2 jump-to-line
.searching/orig-col LDZ2 .cursor/col STZ2
2023-11-28 12:08:42 -05:00
!finish-search
2022-03-29 00:02:22 -04:00
2022-04-04 00:40:37 -04:00
( cancel the active search )
( )
( this method unsets searching/active. unlike ;cancel-search )
( this leaves the cursor where it is. )
2022-03-29 00:02:22 -04:00
@finish-search
#00 .searching/active STZ
2023-11-28 12:08:42 -05:00
reset-arena redraw-all !return
2022-03-29 00:02:22 -04:00
2022-03-31 13:59:20 -04:00
( TODO: i haven't decided how to solve the problem of )
( overlapping matches yet. i don't really want to maintain )
( a global list of all matches, which means that currently )
( it can change in response to e.g. cursor position when )
( matches overlap. )
2022-04-04 00:40:37 -04:00
( )
( UPDATE: now that we have searching/start and searching/end )
( we can use those to resume the search after the full match. )
( this solves the problem except in some very strange cases )
( which are quite unlikely. )
2022-03-31 13:59:20 -04:00
2022-04-04 00:40:37 -04:00
( jump forward to the next match, if any. )
( )
( moves the cursor forward to the next match. if there are no )
( further matches the cursor does not move. )
2022-03-31 23:21:49 -04:00
@jump-to-next-match ( -> )
2023-11-28 12:08:42 -05:00
.searching/regex LDZ2 ORA ?&is-regex
move-to-next-match POP !return
&is-regex
2023-11-28 12:08:42 -05:00
move-to-next-regex-match POP !return
2022-03-31 23:21:49 -04:00
2022-04-04 00:40:37 -04:00
( jump backward to the previous match, if any. )
( )
( moves the cursor backward to the previous match. if there are )
( no further matches the cursor does not move. )
2022-03-31 23:21:49 -04:00
@jump-to-prev-match ( -> )
2023-11-28 12:08:42 -05:00
.searching/regex LDZ2 ORA ?&is-regex
move-to-prev-match POP !return
&is-regex
2023-11-28 12:08:42 -05:00
move-to-prev-regex-match POP !return
2022-03-31 23:21:49 -04:00
2022-04-04 00:40:37 -04:00
( move to the next substring match. )
( )
( called by ;jump-to-next-match. )
2022-03-31 23:21:49 -04:00
@move-to-next-match ( -> ok^ )
.buffer/limit LDZ2
2023-11-28 12:08:42 -05:00
cur-pos INC2
&loop
2023-11-28 12:08:42 -05:00
GTH2k #00 EQU ?&fail
DUP2 matches-at
ORA ?&found
INC2 !&loop
&found
2023-11-28 12:08:42 -05:00
NIP2 jump-to-pos #01 JMP2r
&fail
2022-04-01 23:47:37 -04:00
POP2 POP2 #00 JMP2r
2022-04-04 00:40:37 -04:00
( move to the previous substring match. )
( )
( called by ;jump-to-prev-match. )
2022-03-31 23:21:49 -04:00
@move-to-prev-match ( -> ok^ )
2022-03-31 01:52:20 -04:00
;data
2023-11-28 12:08:42 -05:00
cur-pos #0001 SUB2
2022-03-31 01:52:20 -04:00
&loop
2023-11-28 12:08:42 -05:00
GTH2k ?&fail
DUP2 matches-at
ORA ?&found
#0001 SUB2 !&loop
2022-03-31 01:52:20 -04:00
&found
2023-11-28 12:08:42 -05:00
NIP2 jump-to-pos #01 JMP2r
2022-03-31 01:52:20 -04:00
&fail
2022-04-01 23:47:37 -04:00
POP2 POP2 #00 JMP2r
2022-03-31 23:21:49 -04:00
2022-04-04 00:40:37 -04:00
( move to the next regex match. )
( )
( called by ;jump-to-next-match. )
@move-to-next-regex-match ( -> ok^ )
.searching/end LDZ2 .buffer/limit LDZ2 OVR2
2023-11-28 12:08:42 -05:00
GTH2 ?&ok
POP2 #00 JMP2r
&ok
2023-11-28 12:08:42 -05:00
.searching/regex LDZ2 rx-search ?&found
#00 JMP2r
2022-03-31 23:21:49 -04:00
&found
;search-end LDA2 .searching/end STZ2
;search-start LDA2 DUP2 .searching/start STZ2
2023-11-28 12:08:42 -05:00
jump-to-pos #01 JMP2r
2022-03-31 23:21:49 -04:00
2022-04-04 00:40:37 -04:00
( move to the previous substring match. )
( )
( called by ;jump-to-prev-match. )
( )
( compared to move-to-next-regex-match this is kind of inefficient. )
( that's because we have no easy way to search backwards from a point. )
( )
( we could do some kind of fancy thing where we search the previous )
( N bytes, then the 2N bytes before that, etc. )
( )
( however, 64K is small enough that just searching from the beginning )
( and then taking the last match before the cursor works. )
@move-to-prev-regex-match ( -> ok^ )
LITr 00
2023-11-28 12:08:42 -05:00
cur-pos ;data ( limit pos [res] )
&loop ( limit pos [res] )
2023-11-28 12:08:42 -05:00
GTH2k #00 EQU ?&done ( limit pos )
DUP2 .searching/regex LDZ2 rx-search ( limit pos match? )
#00 EQU ?&done ( limit pos )
OVR2 ;search-end LDA2 LTH2 ?&done
POP2 POPr LITr 01
;search-start LDA2 .searching/start STZ2
;search-end LDA2 DUP2 .searching/end STZ2
2023-11-28 12:08:42 -05:00
!&loop
2022-03-31 23:21:49 -04:00
&done
2023-11-28 12:08:42 -05:00
POP2 POP2 STHr DUP #00 EQU ?&fail
.searching/start LDZ2 jump-to-pos
&fail
JMP2r
2022-03-29 00:02:22 -04:00
2022-04-04 00:40:37 -04:00
( on-key event handler to use when searching )
( )
( when searching the user can: )
( - move to the next match (n or C-s) )
( - move to the previous match (p or C-r) )
( - end the search leaving the cursor where it is (enter) )
( - cancel the search restoring the cursor (C-g) )
2022-03-29 00:02:22 -04:00
@on-key-searching
2023-11-28 12:08:42 -05:00
.state/key LDZ #07 EQU ( C-g ) ?cancel-search
.state/key LDZ #08 EQU ( C-h ) ?help
.state/key LDZ #0d EQU ( \r ) ?finish-search
.state/key LDZ #12 EQU ( C-r ) ?jump-to-prev-match
.state/key LDZ #13 EQU ( C-s ) ?jump-to-next-match
.state/key LDZ #6e EQU ( n ) ?jump-to-next-match
.state/key LDZ #70 EQU ( p ) ?jump-to-prev-match
!ignore
2022-03-29 00:02:22 -04:00
2022-04-04 00:40:37 -04:00
( on-key event handler to use when prompt is active )
( )
( when the prompt is active the user can: )
( - append characters to the input string )
( - delete from the end of the input string (backspace) )
( - complete the input and act (enter) )
( - cancel the prompt without action (C-g) )
( )
( TODO: currently it's impossible to edit the prompt )
( except from the end. ideally we'd support most of the )
( same navigation commands as we do in the buffer, such as )
( C-a, C-d, etc. however, it's enough extra work to enable )
( this that for now i haven't done it. )
2022-03-23 21:46:59 -04:00
@on-key-prompt
2023-11-28 12:08:42 -05:00
.state/key LDZ #07 EQU ( C-g ) ?cancel-prompt
.state/key LDZ #08 EQU ( C-h ) ?help
.state/key LDZ #0d EQU ( \r ) ?finish-prompt
.state/key LDZ #7f EQU ( DEL ) ?backspace-prompt
.state/key LDZ #20 LTH ?ignore ( ignore for now )
.state/key LDZ #7e GTH ?ignore ( ignore for now )
.state/key LDZ ( printable ASCII ) !insert-prompt
2022-03-23 21:46:59 -04:00
BRK
2022-04-04 00:40:37 -04:00
( on-key event handler )
( )
( this is the "normal" event handler to use for editing )
( the buffer. it checks various state values to determine )
( if we're in the midst of a control sequence, if we're )
( searching or have an active prompt, etc. )
( )
2022-03-23 22:55:03 -04:00
( TODO: C-h for help )
2022-04-04 00:40:37 -04:00
( )
( you could also imagine building data structures of )
( commands to unify input strings, help text, callbacks, )
( and so on. this might ultimately be more efficient but )
( for now what we have works. )
2022-02-09 20:04:12 -05:00
@on-key
2022-04-08 23:52:07 -04:00
#00 .state/in-undo STZ
.Console/read DEI .state/key STZ
2023-11-28 12:08:42 -05:00
clear-message-line
.state/in-help LDZ #00 .state/in-help STZ ?return
.searching/active LDZ ?on-key-searching
.prompt/active LDZ ?on-key-prompt
.state/saw-vt LDZ ?on-key-vt
.state/saw-xterm LDZ ?on-key-xterm
.state/saw-esc LDZ ?on-key-escaped
.state/key LDZ #01 EQU ( C-a ) ?bol
.state/key LDZ #02 EQU ( C-b ) ?back
.state/key LDZ #04 EQU ( C-d ) ?delete
.state/key LDZ #05 EQU ( C-e ) ?eol
.state/key LDZ #06 EQU ( C-f ) ?forward
.state/key LDZ #08 EQU ( C-h ) ?help
.state/key LDZ #09 EQU ( \t ) ?insert-tab
.state/key LDZ #0c EQU ( C-l ) ?center-view
.state/key LDZ #0d EQU ( \r ) ?newline
.state/key LDZ #0e EQU ( C-n ) ?down
.state/key LDZ #0f EQU ( C-o ) ?save
.state/key LDZ #10 EQU ( C-p ) ?up
.state/key LDZ #13 EQU ( C-s ) ?search
.state/key LDZ #16 EQU ( C-v ) ?page-down
.state/key LDZ #18 EQU ( C-x ) ?quit
.state/key LDZ #1a EQU ( C-z ) ?debug
.state/key LDZ #1b EQU ( ESC ) ?escape
.state/key LDZ #7f EQU ( DEL ) ?backspace
.state/key LDZ #20 LTH ?ignore ( ignore for now )
.state/key LDZ #7e GTH ?ignore ( ignore for now )
.state/key LDZ ( printable ASCII ) !insert
2022-02-06 17:07:11 -05:00
2022-04-04 00:40:37 -04:00
( return the smaller of two short values )
2022-02-15 21:58:12 -05:00
@min2 ( x* y* -> min* )
LTH2k JMP SWP2 POP2 JMP2r
2022-04-04 00:40:37 -04:00
( method to add bits to the redraw register )
( )
( state/redraw uses 8 bits to represent which parts )
( of the screen (if any) should be redrawn. this method )
( uses logical-or (ORA) to add the bits of n to those )
( already set. )
2022-03-30 01:26:33 -04:00
@redraw-add ( n^ -> )
.state/redraw LDZk ROT ORA SWP STZ JMP2r
2022-04-04 00:40:37 -04:00
( various redrawing methods )
( )
( these don't perform a redraw right away, but instead )
( signal that the next drawing should include that part. )
2023-11-28 12:08:42 -05:00
@redraw-cursor ( -> ) #01 !redraw-add
@redraw-statusbar-and-cursor ( -> ) #03 !redraw-add
@redraw-prompt-and-cursor ( -> ) #05 !redraw-add
@redraw-matches ( -> ) #08 !redraw-add
@redraw-all ( -> ) #1f !redraw-add
2022-03-30 01:26:33 -04:00
2022-04-04 00:40:37 -04:00
( draw the current cursor location )
2022-02-18 00:20:09 -05:00
@draw-cursor ( -> )
2022-04-08 23:52:07 -04:00
.prompt/active LDZ #00 EQU JMP JMP2r
( TODO: handle long lines )
2023-11-28 12:08:42 -05:00
cur-w-col lmargin ADD2
2022-04-01 23:47:37 -04:00
.cursor/row LDZ2 .buffer/line-offset LDZ2 SUB2
2023-11-28 12:08:42 -05:00
!term-move-cursor
2022-02-09 01:05:11 -05:00
2022-04-01 23:47:37 -04:00
( current column in terms of display width )
2022-04-04 00:40:37 -04:00
( )
( this is different than ;cur-col due to tabs )
2022-04-01 23:47:37 -04:00
@cur-w-col ( -> col* )
LIT2r 0000 ( [0] )
2023-11-28 12:08:42 -05:00
cur-line DUP2 cur-col ADD2 SWP2 ( lim s [0] )
&loop GTH2k ?&next
2022-04-01 23:47:37 -04:00
POP2 POP2 STH2r JMP2r
2023-11-28 12:08:42 -05:00
&next LDAk #09 EQU ?&tabs INC2 INC2r !&loop
&tabs INC2 .config/tab-width LDZ2 STH2 ADD2r !&loop
2022-04-01 23:47:37 -04:00
2022-04-04 00:40:37 -04:00
( move the terminal cursor to the statusbar line )
@move-to-statusbar ( -> )
2023-11-28 12:08:42 -05:00
#0000 .term/rows LDZ2 !term-move-cursor
2022-03-25 00:12:09 -04:00
2022-04-04 00:40:37 -04:00
( draw the full statusbar )
2022-02-18 00:20:09 -05:00
@draw-statusbar ( -> )
2023-11-28 12:08:42 -05:00
move-to-statusbar
2024-07-06 15:33:54 -04:00
emit-color-reverse-nonbold
2022-02-11 15:39:49 -05:00
2022-04-04 00:40:37 -04:00
LIT2r 2018 .term/cols LDZ2 #0001 ( cols i [2018] )
2023-11-28 12:08:42 -05:00
&loop LTH2k ?&done DEOkr INC2 !&loop
2022-04-04 00:40:37 -04:00
&done POP2 POP2 POP2r ( )
2023-11-28 12:08:42 -05:00
move-to-statusbar
2022-04-06 22:41:25 -04:00
( display ** if the buffer has unsaved changes, -- otherwise )
#2d .state/modified LDZ #00 NEQ #03 MUL SUB DUP emit emit sp
2023-11-28 12:08:42 -05:00
;filename print
2022-04-04 00:40:37 -04:00
sp emit-[
2023-11-28 12:08:42 -05:00
.buffer/limit LDZ2 ;data SUB2 emit-dec2
;messages/bytes print
sp
2023-11-28 12:08:42 -05:00
.buffer/line-count LDZ2 emit-dec2
;messages/lines print
2022-04-20 23:02:27 -04:00
sp emit-lpar
2023-11-28 12:08:42 -05:00
cur-col INC2 emit-dec2
emit-,
2023-11-28 12:08:42 -05:00
.cursor/row LDZ2 INC2 emit-dec2
2022-04-20 23:02:27 -04:00
emit-rpar sp emit-[
2022-09-10 13:30:15 -04:00
LIT "s .config/insert-tabs LDZ ADD emit
emit-] sp
2023-11-28 12:08:42 -05:00
;messages/help-msg print
!emit-reset
2022-02-09 01:05:11 -05:00
2022-03-23 21:46:59 -04:00
@draw-prompt ( -> )
2023-11-28 12:08:42 -05:00
clear-message-line
.prompt/active LDZ ?&is-active
2022-03-23 21:46:59 -04:00
JMP2r
&is-active
2022-04-01 23:47:37 -04:00
#01 .state/message STZ
2023-11-28 12:08:42 -05:00
move-to-message-line
emit-color-bold
.prompt/string LDZ2 print
emit-reset
2024-07-07 23:45:18 -04:00
;tmp print
2022-04-01 23:47:37 -04:00
JMP2r
2022-03-23 21:46:59 -04:00
@draw-linenum ( n* -> )
2023-11-28 12:08:42 -05:00
emit-reset
emit-color
emit-dec2-pad sp
!emit-reset
2022-03-29 00:02:22 -04:00
@matches-at ( s* -> limit* )
2024-07-07 23:45:18 -04:00
LIT2r =tmp
2023-11-28 12:08:42 -05:00
&loop LDAkr STHr #00 EQU ?&done
LDAk LDAkr STHr NEQ ?&fail
INC2 INC2r !&loop
2022-04-04 00:40:37 -04:00
&fail POP2 #0000
&done POP2r JMP2r
2022-03-29 00:02:22 -04:00
2022-04-06 22:41:25 -04:00
( TODO: this doesn't handle tabs correctly )
2022-03-29 00:02:22 -04:00
@draw-region ( offset* limit* col* row* -> )
OVR2 ( offset limit col row col )
.term/cols LDZ2 SWP2 SUB2 STH2 ( offset limit col row [cols-col] )
2023-11-28 12:08:42 -05:00
term-move-cursor ( offset limit [cols-col] )
2022-03-29 00:02:22 -04:00
OVR2 STH2r ADD2 ( offset limit offset+cols-col )
2023-11-28 12:08:42 -05:00
min2 STH2 ( offset [cutoff] )
2022-03-29 00:02:22 -04:00
&loop ( i [cutoff] )
2023-11-28 12:08:42 -05:00
DUP2 STH2kr LTH2 #00 EQU ?&done
LDAk #00 EQU ?&done
LDAk #18 DEO INC2 !&loop
2022-03-29 00:02:22 -04:00
&done
POP2 POP2r JMP2r
@screen-limit ( -> sc-limit* )
.term/rows LDZ2 .buffer/line-offset LDZ2 ADD2 ( row0+rows )
2023-11-28 12:08:42 -05:00
DUP2 .buffer/line-count LDZ2 LTH2 ?&not-end
2022-03-29 00:02:22 -04:00
POP2 .buffer/limit LDZ2 JMP2r
&not-end
2023-11-28 12:08:42 -05:00
!abs-line
2022-03-29 00:02:22 -04:00
@draw-regex-matches ( -> )
2024-10-26 00:36:26 -04:00
emit-color-reverse ( )
2023-11-28 12:08:42 -05:00
screen-limit .buffer/offset LDZ2 ( limit pos )
&loop ( limit pos )
GTH2k #00 EQU ( limit pos limit>pos=0? )
2023-11-28 12:08:42 -05:00
?&done ( limit pos )
DUP2 .searching/regex LDZ2 ( limit pos pos rx )
2023-11-28 12:08:42 -05:00
rx-search #00 EQU ( limit pos found=0? )
?&done ( limit pos )
POP2 ;search-start LDA2 ( limit start )
GTH2k #00 EQU ( limit start limit>start=0? )
2023-11-28 12:08:42 -05:00
?&done ( limit start )
;search-end LDA2 OVR2 ( limit start end start )
2023-11-28 12:08:42 -05:00
pos-to-row-col ( limit start end row col )
lmargin ADD2 ( limit start end row col+lm )
SWP2 .buffer/line-offset LDZ2 SUB2 ( limit start end col+lm row-lo )
2023-11-28 12:08:42 -05:00
draw-region ( limit )
;search-end LDA2 ( limit end ) !&loop
&done ( limit pos )
POP2 POP2 JMP2r
2022-03-29 00:02:22 -04:00
@draw-matches ( -> )
( return if not searching )
2023-11-28 12:08:42 -05:00
.searching/active LDZ #00 EQU ?&return ( )
.searching/regex LDZ2 ORA ?draw-regex-matches
2024-10-26 00:36:26 -04:00
emit-color-reverse
lmargin ,&x STR2 #0000 ,&y STR2 ( x <- 0, y <- 0 )
2022-03-29 00:02:22 -04:00
.buffer/offset LDZ2 DUP2
2023-11-28 12:08:42 -05:00
screen-limit SUB2 STH2 ( offset [-count] )
2022-03-29 00:02:22 -04:00
&loop ( offset [-count] )
2023-11-28 12:08:42 -05:00
STH2kr #0000 EQU2 ?&done
DUP2 matches-at ( offset mlim [-count] )
DUP2 ORA ?&found
2022-03-29 00:02:22 -04:00
POP2 ( offset [-count] )
2023-11-28 12:08:42 -05:00
LDAk #0a EQU ?&newline
draw-matches/count-tabs
#0001 !&next ( offset n [-count] )
2022-03-29 00:02:22 -04:00
&found ( offset mlim [-count] )
STH2k ( offset mlim [mlim -count] )
OVR2 SWP2 ,&x LDR2 ,&y LDR2 ( offset offset mlim x y [mlim -count] )
2023-11-28 12:08:42 -05:00
draw-region ( offset [mlim -count] )
2022-03-29 00:02:22 -04:00
STH2r ( offset mlim [-count] )
OVR2 SUB2 ( offset mlim-offset [-count] )
&next ( offset n [-count] )
DUP2 ,&x LDR2 ADD2 ,&x STR2 ( offset n [-count )
STH2k ( offset n [n -count] )
ADD2 ADD2r ( offset+n [n-count] )
2023-11-28 12:08:42 -05:00
!&loop
2022-03-29 00:02:22 -04:00
&newline ( offset [-count] )
lmargin ,&x STR2
2022-03-29 00:02:22 -04:00
,&y LDR2 INC2 ,&y STR2
INC2 INC2r
2023-11-28 12:08:42 -05:00
!&loop
2022-03-29 00:02:22 -04:00
&done
POP2 POP2r
2023-11-28 12:08:42 -05:00
emit-reset
2022-03-29 00:02:22 -04:00
&return
JMP2r
[ &x $2 &y $2 ]
&count-tabs ( offset -> offset )
2023-11-28 12:08:42 -05:00
LDAk #09 NEQ ?&count-tabs/done
,&x LDR2 .config/tab-adjust LDZ2 ADD2 ,&x STR2
&count-tabs/done JMP2r
2022-03-29 00:02:22 -04:00
@emit-tab ( -> )
2022-03-07 22:38:02 -05:00
#0000 .config/tab-width LDZ2 SUB2
LIT2r 2018
2023-11-28 12:08:42 -05:00
&loop ORAk ?&next POP2 POP2r JMP2r
&next DEOkr INC2 !&loop
2022-04-10 02:05:09 -04:00
( ANSI terminal notes )
( )
( attrs [0-7] )
( reset, bright, dim, underscore, )
( blink, ???, reverse, hidden )
( )
( fg [30-37], bg [40-47] )
( black, red, green, yellow, )
( blue, magenta, cyan, white )
( ANSI control sequence to move the cursor to the given coord )
( ESC [ $row ; $col H )
@term-move-cursor ( col* row* -> )
2023-11-28 12:08:42 -05:00
ansi INC2 ( row+1 ) emit-dec2
emit-; INC2 ( col+1 ) emit-dec2
2022-04-10 02:05:09 -04:00
emit-H JMP2r
( ANSI control sequence to move N positions right )
( ESC [ $n C )
@term-move-right ( n* -> )
2023-11-28 12:08:42 -05:00
ansi emit-dec2 emit-C JMP2r
2022-04-10 02:05:09 -04:00
( ANSI control sequence to get the cursor position )
( ESC [ 6 n )
@term-get-cursor-position ( -> )
2023-11-28 12:08:42 -05:00
LIT2 00 "n LIT "6 !ansi-emit
2022-04-10 02:05:09 -04:00
( ANSI control sequence to erase entire screen )
( ESC [ 2 J )
@term-erase-all ( -> )
2023-11-28 12:08:42 -05:00
LIT2 00 "J LIT "2 !ansi-emit
2022-04-10 02:05:09 -04:00
( ANSI control sequence to erase the current line )
( ESC [ 2 K )
@term-erase-line ( -> )
2023-11-28 12:08:42 -05:00
LIT2 00 "K LIT "2 !ansi-emit
2022-04-10 02:05:09 -04:00
@ansi-emit ( 00 cn ... c1 c0 -> )
LITr 18 ( Console/write )
#5b1b STHkr DEO STHkr DEO ( ESC [ )
2023-11-28 12:08:42 -05:00
&loop DUP ?&next POP POPr JMP2r
&next STHkr DEO !&loop
2022-04-10 02:05:09 -04:00
( ESC [ 3 1 m )
2022-03-04 01:47:11 -05:00
@emit-red ( -> )
2023-11-28 12:08:42 -05:00
LIT2 00 "m .config/red LDZ2 !ansi-emit
2022-03-04 01:47:11 -05:00
( ESC [ 0 m )
2022-03-04 01:47:11 -05:00
@emit-reset ( -> )
2023-11-28 12:08:42 -05:00
#00 LIT2 "m "0 !ansi-emit
2022-03-04 01:47:11 -05:00
( ESC [ 1 m $ ESC [ 0 m )
2022-03-04 01:47:11 -05:00
@emit-red-dollar ( -> )
2023-11-28 12:08:42 -05:00
emit-red emit-$ !emit-reset
2022-03-04 01:47:11 -05:00
( ESC [ 3 $x ; 7 m )
( $x is 0-7 )
2022-03-04 15:30:26 -05:00
@emit-color-reverse ( -> )
2023-11-28 12:08:42 -05:00
LIT2 00 "m LIT2 "7 "; .config/color LDZ2 !ansi-emit
2022-04-10 02:05:09 -04:00
@emit-color ( -> )
2023-11-28 12:08:42 -05:00
LIT2 00 "m .config/color LDZ2 !ansi-emit
2022-03-04 15:30:26 -05:00
( ESC [ 3 $x ; 1 m )
( $x is 0-7 )
2022-03-23 21:46:59 -04:00
@emit-color-bold ( -> )
2023-11-28 12:08:42 -05:00
LIT2 00 "m LIT2 "1 "; .config/color LDZ2 !ansi-emit
2022-03-23 21:46:59 -04:00
2023-01-30 15:09:07 -05:00
( ESC [ 3 $x ; 1 ; 7 m )
( $x is 0-7 )
@emit-color-reverse-bold ( -> )
2023-11-28 12:08:42 -05:00
LIT2 00 "m LIT2 "7 "; LIT2 "1 "; .config/color LDZ2 !ansi-emit
2023-01-30 15:09:07 -05:00
2024-07-06 15:33:54 -04:00
@emit-color-reverse-nonbold ( -> )
LIT2 00 "m LIT2 "7 "; .config/color LDZ2 LIT2 "; "0 !ansi-emit
2022-02-18 00:20:09 -05:00
@draw-all ( -> )
2023-11-28 12:08:42 -05:00
term-erase-all
#0000 #0000 term-move-cursor
2022-03-07 22:38:02 -05:00
.buffer/line-offset LDZ2 STH2 LIT2r 0001 ( [k line-offset] )
.buffer/offset LDZ2
2022-03-04 15:30:26 -05:00
&bol
2023-11-28 12:08:42 -05:00
ADD2kr STH2r draw-linenum
lmargin INC2 ,&x STR2
2022-03-04 01:47:11 -05:00
&loop ( offset [k line-offset] )
2023-11-28 12:08:42 -05:00
LDAk #00 EQU ?&eof
LDAk #0a EQU ?&eol
2022-03-07 22:38:02 -05:00
,&x LDR2 .term/cols LDZ2
2023-11-28 12:08:42 -05:00
LTH2k ?&ok
GTH2 ?&skip
emit-red-dollar ,&x LDR2 INC2 ,&x STR2
2022-03-04 15:30:26 -05:00
&skip INC2
2023-11-28 12:08:42 -05:00
!&loop
2022-03-04 01:47:11 -05:00
&ok POP2 POP2
2023-11-28 12:08:42 -05:00
LDAk #09 EQU ?&do-tab
2022-03-04 15:30:26 -05:00
LDAk emit INC2 ,&x LDR2 INC2 ,&x STR2
2023-11-28 12:08:42 -05:00
!&loop
2022-03-04 15:30:26 -05:00
&eol INC2r
2023-11-28 12:08:42 -05:00
STH2kr .term/rows LDZ2 GTH2 ?&done
crlf INC2 !&bol
&do-tab emit-tab INC2
2022-03-07 22:38:02 -05:00
.config/tab-width LDZ2 ,&x LDR2 ADD2 ,&x STR2
2023-11-28 12:08:42 -05:00
!&loop
2022-03-04 01:47:11 -05:00
[ &x $2 ]
2022-02-13 14:46:50 -05:00
&eof
2023-11-28 12:08:42 -05:00
emit-red
2022-02-13 14:46:50 -05:00
&eof-loop
2023-11-28 12:08:42 -05:00
STH2kr .term/rows LDZ2 GTH2 ?&done
2022-04-12 12:15:42 -04:00
crlf
2023-11-28 12:08:42 -05:00
lmargin term-move-right
2022-03-30 01:26:33 -04:00
emit-~ INC2r
2023-11-28 12:08:42 -05:00
!&eof-loop
&done POP2 POP2r POP2r
2023-11-28 12:08:42 -05:00
emit-reset
draw-matches
draw-statusbar
draw-prompt
!draw-cursor
2022-02-06 17:07:11 -05:00
2022-03-30 01:26:33 -04:00
( handler completion code to do necessary drawing and BRK )
@return ( -> )
2022-03-31 23:21:49 -04:00
.state/redraw LDZ
2023-11-28 12:08:42 -05:00
DUP #10 AND ?&draw-all
DUP #08 AND ?&do-8 !&skip-8 &do-8 draw-matches
&skip-8 DUP #04 AND ?&do-4 !&skip-4 &do-4 draw-prompt
&skip-4 DUP #02 AND ?&do-2 !&skip-2 &do-2 draw-statusbar
&skip-2 DUP #01 AND ?&do-1 !&finish &do-1 draw-cursor !&finish
&draw-all draw-all
&finish POP #00 .state/redraw STZ BRK
2022-03-30 01:26:33 -04:00
2022-04-10 02:05:09 -04:00
@str-copy ( src* dst* -> len* )
STH2 DUP2 ( src src [dst] )
2023-11-28 12:08:42 -05:00
&loop LDAk #00 EQU ?&done
2022-02-11 15:39:49 -05:00
LDAk STH2kr STA
2023-11-28 12:08:42 -05:00
INC2 INC2r !&loop
2022-04-10 02:05:09 -04:00
&done ( src src+n [dst+n] )
SWP2 SUB2
#00 STH2r STA
2022-03-23 21:46:59 -04:00
JMP2r
2022-02-11 15:39:49 -05:00
@print ( s* -> )
2024-07-07 23:45:18 -04:00
LDAk ?{ POP2 JMP2r } LDAk #18 DEO INC2 !print
2022-02-06 17:07:11 -05:00
@cur-len ( -> n* )
2023-11-28 12:08:42 -05:00
cur-line !line-len
2022-02-06 17:07:11 -05:00
@line-len ( s* -> n* )
#0000 STH2
2023-11-28 12:08:42 -05:00
&loop LDAk #00 EQU ?&end
LDAk #0a EQU ?&end
INC2 INC2r !&loop
2022-02-06 17:07:11 -05:00
&end POP2 STH2r JMP2r
2022-03-31 13:48:08 -04:00
@line-is-visible ( n* -> bool^ )
2023-11-28 12:08:42 -05:00
.buffer/line-offset LDZ2 LTH2k ?&no
2022-03-31 13:48:08 -04:00
.term/rows LDZ2 ADD2 LTH2 JMP2r
&no POP2 POP2 #00 JMP2r
@jump-to-pos ( s* -> )
pos-to-logical-row-col SWP2 !move-to-coord
2023-12-05 21:52:32 -05:00
@pos-to-display-row-col ( s* -> row* col* )
.config/tab-width LDZ2 ;pos-to-row-col/tab-width STA2 !pos-to-row-col
@pos-to-logical-row-col ( s* -> row* col* )
#0001 ;pos-to-row-col/tab-width STA2 ( fall-through )
@pos-to-row-col ( s* -> row* col* )
#0000 ,&row STR2
#0000 ,&col STR2
;data
&loop ( s pos )
2023-11-28 12:08:42 -05:00
GTH2k #00 EQU ?&done
LDAk #0a EQU ?&newline
LDAk #09 EQU ?&tab
#0001
&inc
,&col LDR2 ADD2 ,&col STR2
2023-11-28 12:08:42 -05:00
INC2 !&loop
&newline
#0000 ,&col STR2
,&row LDR2 INC2 ,&row STR2
2023-11-28 12:08:42 -05:00
INC2 !&loop
&tab
LIT2 [ &tab-width $2 ] !&inc
&done
2022-03-31 01:52:20 -04:00
POP2 POP2
,&row LDR2 ,&col LDR2 JMP2r
[ &row $2 &col $2 ]
2022-04-01 23:47:37 -04:00
( counts y lines forward from the given address )
@line-to-pos ( addr* y* -> s* )
#0000 SWP2 SUB2 STH2 ( addr [-y] )
&newline ( addr [-y] )
2023-11-28 12:08:42 -05:00
STH2kr ORA ?&loop !&done
2022-04-01 23:47:37 -04:00
&loop ( addr [-y] )
2023-11-28 12:08:42 -05:00
LDAk #00 EQU ?&not-found ( addr [-y] )
LDAk #0a EQU ?&found ( addr [-y] )
INC2 !&loop ( addr+1 [-y] )
&found INC2 INC2r !&newline
2022-02-10 01:40:48 -05:00
&done POP2r JMP2r
&not-found POP2 POP2r #0000 JMP2r
2022-04-01 23:47:37 -04:00
( find string pointer for absolute y coordinate )
@abs-line ( y* -> s* )
2023-11-28 12:08:42 -05:00
;data SWP2 !line-to-pos
2022-04-01 23:47:37 -04:00
2022-04-04 00:40:37 -04:00
( return a pointer to the current line )
2022-02-06 17:07:11 -05:00
@cur-line ( -> s* )
2023-11-28 12:08:42 -05:00
.cursor/row LDZ2 .buffer/line-offset LDZ2 LTH2k ?&early
2022-04-06 22:41:25 -04:00
( if cursor/row is later than line-offset )
( we can save some time by starting at )
( buffer/offset instead of the beginning )
2023-11-28 12:08:42 -05:00
SUB2 .buffer/offset LDZ2 SWP2 !line-to-pos
2022-04-06 22:41:25 -04:00
&early
( if cursor/row is earlier than line-offset )
( we need to use the absolute y coordinate )
2023-11-28 12:08:42 -05:00
POP2 !abs-line
2022-02-14 16:47:36 -05:00
2022-04-04 00:40:37 -04:00
( return a pointer to the current cursor position )
2022-02-06 17:07:11 -05:00
@cur-pos ( -> s* )
2023-11-28 12:08:42 -05:00
cur-line cur-col ADD2 JMP2r
2022-02-06 17:07:11 -05:00
2022-04-04 00:40:37 -04:00
( insert one character at the cursor position )
2022-02-06 17:07:11 -05:00
@shift-right ( c^ addr* -> )
2022-04-08 23:52:07 -04:00
#01 .state/modified STZ
2022-02-06 17:07:11 -05:00
ROT STH ( addr [prev^] )
2023-11-28 12:08:42 -05:00
last-pos SWP2 ( last addr [prev^] )
.state/in-undo LDZ ?&loop
STH2k #00 STH2r u-push
&loop LTH2k ?&done ( last addr [prev^] )
2022-02-06 17:07:11 -05:00
LDAk STH SWPr ( last addr [prev^ curr^] )
DUP2 STHr ( last addr addr prev^ [curr^] )
ROT ROT STA ( last addr [curr^] )
2023-11-28 12:08:42 -05:00
INC2 !&loop ( last addr+1 [curr^] )
2022-02-06 17:07:11 -05:00
&done NIP2 DUP2 ( addr addr [prev^] )
STHr ROT ROT ( addr prev^ addr )
STA INC2 ( addr+1 )
2022-03-07 22:38:02 -05:00
.buffer/limit STZ2 ( )
2022-02-06 17:07:11 -05:00
JMP2r
2022-04-04 00:40:37 -04:00
( remove one character at the cursor position )
( )
2022-02-13 14:46:50 -05:00
( TODO: change last/addr order and GTH -> LTH to remove hack )
2022-02-06 17:07:11 -05:00
@shift-left ( addr* -> )
2022-04-08 23:52:07 -04:00
#01 .state/modified STZ
2023-11-28 12:08:42 -05:00
last-pos SWP2 ( last addr )
.state/in-undo LDZ ?&loop
STH2k cur-pos LDA STH2r u-push
&loop GTH2k ?&next ( last addr )
!&done ( last addr )
2022-02-13 14:46:50 -05:00
&next DUP2 INC2 LDAk ( last addr addr+1 c1^ )
STH SWP2 STHr ( last addr+1 addr c1^ )
ROT ROT ( last addr+1 c1^ addr )
2023-11-28 12:08:42 -05:00
STA !&loop ( last addr+1 )
2022-02-13 14:46:50 -05:00
&done POP2 ( last )
2022-03-07 22:38:02 -05:00
.buffer/limit STZ2 ( )
#00 .buffer/limit LDZ2 STA ( ensure null termination )
2022-02-06 17:07:11 -05:00
JMP2r
2022-04-01 23:47:37 -04:00
( current column in terms of bytes in buffer )
2022-02-18 00:20:09 -05:00
@cur-col ( -> col* )
2023-11-28 12:08:42 -05:00
.cursor/col LDZ2 cur-len !min2
2022-02-07 22:25:32 -05:00
2022-04-04 00:40:37 -04:00
( jump to the first line in the buffer )
@zero-row ( -> )
2022-03-07 22:38:02 -05:00
;data .buffer/offset STZ2
#0000 .buffer/line-offset STZ2
#0000 .cursor/row STZ2
JMP2r
2022-04-04 00:40:37 -04:00
( return the location of the last character in the buffer )
2022-02-18 00:20:09 -05:00
@last-pos ( -> addr* )
2022-03-07 22:38:02 -05:00
.buffer/limit LDZ2 #0001 SUB2 JMP2r
2022-02-07 22:25:32 -05:00
2022-04-04 00:40:37 -04:00
( emit a short as a decimal )
2022-02-09 01:05:11 -05:00
@emit-dec2 ( n* -> )
2022-04-04 00:40:37 -04:00
LITr 00 ( n [0] )
&read ( n [k] )
#000a DIV2k STH2k MUL2 SUB2 STH2r INCr ( n%10 n/10 [k+1] )
2023-11-28 12:08:42 -05:00
DUP2 ORA ?&read
2022-04-04 00:40:37 -04:00
POP2 ( top element was 0000 )
&write ( n0 n1 ... nk [k+1] )
NIP #30 ADD #18 DEO LITr 01 SUBr ( n0 ... n{k-1} [k] )
2023-11-28 12:08:42 -05:00
STHkr ?&write
2022-04-04 00:40:37 -04:00
POPr JMP2r
( emit a short as a decimal with leading spaces )
@emit-dec2-pad ( n* -> )
LITr 00 ( n [0] )
&read ( n [k] )
#000a DIV2k STH2k MUL2 SUB2 STH2r INCr ( n%10 n/10 [k+1] )
2023-11-28 12:08:42 -05:00
STHkr #05 LTH ?&read
POP2 ( top element was 0000 )
&write0 ( n0 n1 ... nk [k+1] )
2023-11-28 12:08:42 -05:00
DUP2 ORA ?emit-dec2/write
POP2 sp LITr 01 SUBr
2023-11-28 12:08:42 -05:00
STHkr ?&write0
POPr JMP2r
2022-03-07 22:38:02 -05:00
( various string constants used as messages for the user )
2022-03-23 22:55:03 -04:00
@messages [ &null 00
2022-04-09 00:05:12 -04:00
&input-error "Input 20 "error, 20 "file 20 "too 20 "large: 20 00
&bytes 20 "bytes, 00
2022-03-23 22:12:17 -04:00
&save-ok "Successfully 20 "saved 20 00
&save-failed "Failed 20 "to 20 "save 20 00
&lines 20 "lines] 00
2022-03-24 00:09:55 -04:00
&goto-line "Go 20 "to 20 "line: 20 00
2022-03-23 21:46:59 -04:00
&save-prompt "File 20 "Name 20 "to 20 "Write: 20 00
2022-03-29 00:02:22 -04:00
&search-prompt "Text 20 "to 20 "Search 20 "for: 20 00
&regex-search-prompt "Regex 20 "to 20 "Search 20 "for: 20 00
2022-03-23 22:55:03 -04:00
&quit-prompt "Save 20 "modified 20 "file 20 "(y/n)? 20 00
&unknown-input "Unknown 20 "input: 20 00
&no-matches-found "No 20 "matches 20 "found: 20 00
2022-04-08 23:52:07 -04:00
&term-size-parse-error "Error 20 "parsing 20 "term 20 "size 00
&help-msg "(help: 20 "C-h) 00
&usage "usage: 20 "femto 20 "<file> 00
2022-03-25 00:12:09 -04:00
]
2022-02-08 22:22:08 -05:00
@help-text
09 "femto 20 "input 20 "reference 09 "(C 20 "is 20 "Ctrl, 20 "M 20 "is 20 "Meta/Alt ") 0d 0a
0d 0a
09 "quit 09 09 "C-x 09 09 "cancel 09 09 "C-g 0d 0a
09 "save 09 09 "C-o 09 09 "undo 09 09 "C-u 0d 0a
0d 0a
09 "move 20 "up 09 09 "C-p 20 "(up) 09 "page 20 "up 09 09 "M-v 20 "(pg-up) 0d 0a
09 "move 20 "down 09 "C-n 20 "(down) 09 "page 20 "down 09 "C-v 20 "(pg-dn) 0d 0a
09 "move 20 "left 09 "C-b 20 "(left) 09 "line 20 "start 09 "C-a 20 "(home) 0d 0a
09 "move 20 "right 09 "C-f 20 "(right) 09 "line 20 "end 09 "C-e 20 "(end) 0d 0a
0d 0a
09 "goto 20 "file 20 "start 09 "M-< 09 09 "left 20 "by 20 "word 09 "M-b 0d 0a
09 "goto 20 "file 20 "end 09 "M-> 09 09 "right 20 "by 20 "word 09 "M-f 0d 0a
09 "goto 20 "line 09 "M-g 09 09 "center 20 "cursor 09 "C-l 0d 0a
0d 0a
09 "search 09 09 "C-s 09 09 "toggle 20 "colors 09 "M-c 0d 0a
09 "regex 20 "search 09 "M-s 09 09 "toggle 20 "tabs 09 "M-t 0d 0a
09 "next 20 "match 09 "C-s 20 "(n) 0d 0a
09 "prev 20 "match 09 "C-r 20 "(p) 0d 0a
09 "end 20 "search 09 "enter 0d 0a
09 "cancel 20 "search 09 "C-g 0d 0a
0d 0a
09 09 09 "press 20 "any 20 "key 20 "to 20 "continue... 0d 0a
00
2022-04-09 00:16:22 -04:00
( perform the undo action )
2022-04-08 23:52:07 -04:00
@undo ( -> )
#01 .state/in-undo STZ
2024-07-07 23:45:18 -04:00
;undo-stack/pos LDA2 ;undo-stack EQU2 ?&noop
2022-04-08 23:52:07 -04:00
;undo-stack/pos LDA2 #0003 SUB2
DUP2 ;undo-stack/pos STA2
LDA2k STH2 ( pos [addr] )
INC2 INC2 LDA STH2r ( c addr )
2023-11-28 12:08:42 -05:00
jump-to-pos
DUP #00 EQU ?&delete
DUP #0a EQU ?&newline
DUP #09 EQU ?&tab
!insert
&newline POP !newline
&tab POP !insert-tab
&delete POP !delete
2022-04-08 23:52:07 -04:00
&noop BRK
2022-04-09 00:16:22 -04:00
( free up space in the undo stack by evicting the oldest entry )
2022-04-08 23:52:07 -04:00
@u-free ( -> )
2024-07-07 23:45:18 -04:00
;undo-stack STH2k #0003 ADD2 ( st+3 [st] )
2022-04-08 23:52:07 -04:00
&loop LDAk STH2kr STA
INC2 INC2r
2023-11-28 12:08:42 -05:00
DUP2 ;undo-stack/pos LDA2 LTH2 ?&loop
2022-04-10 13:55:40 -04:00
;undo-stack/pos LDA2k #0003 SUB2 SWP2 STA2
POP2 POP2r JMP2r
2022-04-08 23:52:07 -04:00
2022-04-09 00:16:22 -04:00
( push a new item on the undo stack )
2022-04-08 23:52:07 -04:00
@u-push ( c^ addr* -> )
;undo-stack/pos DUP2 LDA2 ( c^ addr* top* pos* )
2023-11-28 12:08:42 -05:00
GTH2 ( c^ addr* top>pos^ ) ?&has-room ( c^ addr* )
u-free
2022-04-08 23:52:07 -04:00
&has-room ;undo-stack/pos LDA2 ( c^ addr* pos* )
STH2k ( c addr pos [pos] )
STA2 ( c [pos] )
STH2r INC2 INC2 STH2k ( c pos+2 [pos+2] )
STA ( [pos+2] )
STH2r INC2 ( pos+3 ) ;undo-stack/pos STA2 ( [] )
JMP2r
2022-04-04 00:40:37 -04:00
( path to file being edited )
@filename $80
2022-04-09 00:05:12 -04:00
( stack of data to undo )
( )
( each item in the stack consists of: )
( - 2 bytes: address to jump to )
( - 1 byte: character to insert, or \0 to delete )
( )
( pos points to the next open stack frame. )
2022-04-09 00:16:22 -04:00
( when pos points to data the stack is empty. )
2022-04-09 00:05:12 -04:00
( when pos points to pos the stack is full. )
2022-04-08 23:52:07 -04:00
@undo-stack [
2024-07-07 23:45:18 -04:00
$180 ( 128 steps )
&pos =undo-stack
2022-04-08 23:52:07 -04:00
]
2022-04-04 00:40:37 -04:00
( actual file data to be edited )
2022-04-10 02:05:09 -04:00
@data $ce80
2022-04-08 23:52:07 -04:00
( end of femto.tal )