# interactive prompt for mu recipe main [ default-space:address:array:location <- new location:type, 30:literal switch-to-display msg:address:array:character <- new [ready! type in an instruction, then hit enter. ctrl-d exits. ] 0:literal/real-screen <- print-string 0:literal/real-screen, msg:address:array:character 0:literal/real-keyboard, 0:literal/real-screen <- color-session 0:literal/real-keyboard, 0:literal/real-screen #? wait-for-key-from-keyboard #? 1 return-to-console ] recipe color-session [ default-space:address:array:location <- new location:type, 30:literal keyboard:address <- next-ingredient screen:address <- next-ingredient { inst:address:array:character, keyboard:address, screen:address <- read-instruction keyboard:address, screen:address break-unless inst:address:array:character run-interactive inst:address:array:character # assume run-interactive printed on the current line move-cursor-down-on-display #? clear-line-on-display # just to refresh the screen loop } reply keyboard:address/same-as-ingredient:0, screen:address/same-as-ingredient:1 ] # basic keyboard input; just text and enter scenario read-instruction1 [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [x <- copy y ] run [ 1:address:array:character <- read-instruction keyboard:address, screen:address 2:address:array:character <- new [=> ] print-string screen:address, 2:address:array:character print-string screen:address, 1:address:array:character ] screen-should-contain [ .x <- copy y . .=> x <- copy y . . . ] screen-should-contain-in-color 7:literal/white, [ .x copy y . .=> x <- copy y . . . ] screen-should-contain-in-color 1:literal/red, [ . <- . . . ] ] # Read characters as they're typed at the keyboard, print them to the screen, # accumulate them in a string, return the string at the end. # Most of the complexity is for the printing to screen, to highlight strings # and comments specially. Especially in the presence of backspacing. recipe read-instruction [ default-space:address:array:location <- new location:type, 60:literal k:address:keyboard <- next-ingredient x:address:screen <- next-ingredient result:address:buffer <- init-buffer 10:literal # string to maybe add to trace [app], [read-instruction] # start state machine by calling slurp-regular-characters, which will return # by calling the complete continuation complete:continuation <- current-continuation # If result is not empty, we've run slurp-regular-characters below, called # the continuation and so bounced back here. We're done. len:number <- get result:address:buffer/deref, length:offset completed?:boolean <- greater-than len:number, 0:literal jump-if completed?:boolean, +completed:label # Otherwise we're just getting started. result:address:buffer, k:address:keyboard, x:address:screen <- slurp-regular-characters result:address:buffer, k:address:keyboard, x:address:screen, complete:continuation #? $print [aaa: ], result:address:buffer #? 1 #? move-cursor-down-on-display #? 1 trace [error], [slurp-regular-characters should never return normally] +completed result2:address:array:character <- buffer-to-array result:address:buffer #? $print [bbb: ], result2:address:array:character #? 1 #? move-cursor-down-on-display #? 1 trace [app], [exiting read-instruction] reply result2:address:array:character, k:address:keyboard/same-as-ingredient:0, x:address:screen/same-as-ingredient:1 ] # read characters from the keyboard, print them to the screen in *white*. # Transition to other routines for comments and strings. recipe slurp-regular-characters [ default-space:address:array:location <- new location:type, 30:literal result:address:buffer <- next-ingredient k:address:keyboard <- next-ingredient x:address:screen <- next-ingredient complete:continuation <- next-ingredient trace [app], [slurp-regular-characters] characters-slurped:number <- copy 0:literal #? $run-depth #? 1 { +next-character trace [app], [slurp-regular-characters: next] #? $print [a0 #? 1 #? ] #? 1 #? move-cursor-down-on-display #? 1 # read character c:character, k:address:keyboard <- wait-for-key k:address:keyboard #? print-character x:address:screen, c:character #? 1 #? move-cursor-down-on-display #? 1 # quit? { #? $print [aaa] #? 1 #? move-cursor-down-on-display #? 1 ctrl-d?:boolean <- equal c:character, 4:literal/ctrl-d/eof break-unless ctrl-d?:boolean #? $print [ctrl-d] #? 1 #? move-cursor-down-on-display #? 1 trace [app], [slurp-regular-characters: ctrl-d] reply 0:literal, k:address:keyboard/same-as-ingredient:1, x:address:screen/same-as-ingredient:2 } { null?:boolean <- equal c:character, 0:literal/null break-unless null?:boolean trace [app], [slurp-regular-characters: null] reply 0:literal, k:address:keyboard/same-as-ingredient:1, x:address:screen/same-as-ingredient:2 } # comment? { comment?:boolean <- equal c:character, 35:literal/hash break-unless comment?:boolean print-character x:address:screen, c:character, 4:literal/blue result:address:buffer <- buffer-append result:address:buffer, c:character result:address:buffer, k:address:keyboard, x:address:screen <- slurp-comment result:address:buffer, k:address:keyboard, x:address:screen, complete:continuation # continue appending to this instruction, whether comment ended or was backspaced out of loop +next-character:label } # string { string?:boolean <- equal c:character, 91:literal/open-bracket break-unless string?:boolean print-character x:address:screen, c:character, 6:literal/cyan result:address:buffer <- buffer-append result:address:buffer, c:character result:address:buffer, _, k:address:keyboard, x:address:screen <- slurp-string result:address:buffer, k:address:keyboard, x:address:screen, complete:continuation loop +next-character:label } # assignment { assign?:boolean <- equal c:character, 60:literal/less-than break-unless assign?:boolean print-character x:address:screen, c:character, 1:literal/red trace [app], [start of assignment: <] result:address:buffer <- buffer-append result:address:buffer, c:character result:address:buffer, k:address:keyboard, x:address:screen <- slurp-assignment result:address:buffer, k:address:keyboard, x:address:screen, complete:continuation loop +next-character:label } # print print-character x:address:screen, c:character # default color # append result:address:buffer <- buffer-append result:address:buffer, c:character #? $print [a1 #? 1 #? ] #? 1 #? move-cursor-down-on-display #? 1 # backspace? decrement and maybe return { #? $print [a2 #? 1 #? ] #? 1 #? move-cursor-down-on-display #? 1 backspace?:boolean <- equal c:character, 8:literal/backspace break-unless backspace?:boolean #? $print [a3 #? 1 #? ] #? 1 #? move-cursor-down-on-display #? 1 characters-slurped:number <- subtract characters-slurped:number, 1:literal { #? $print [a4 #? 1 #? ] #? 1 #? move-cursor-down-on-display #? 1 done?:boolean <- lesser-or-equal characters-slurped:number, -1:literal break-unless done?:boolean #? $print [a5 #? 1 #? ] #? 1 #? move-cursor-down-on-display #? 1 trace [app], [slurp-regular-characters: too many backspaces; returning] #? $print [a6 #? 1 #? ] #? 1 #? move-cursor-down-on-display #? 1 reply result:address:buffer, k:address:keyboard/same-as-ingredient:1, x:address:screen/same-as-ingredient:2 } loop +next-character:label } #? $print [a9 #? 1 #? ] #? 1 #? move-cursor-down-on-display #? 1 # otherwise increment characters-slurped:number <- add characters-slurped:number, 1:literal # done with this instruction? done?:boolean <- equal c:character, 10:literal/newline break-if done?:boolean loop } # newline encountered; terminate all recursive calls #? xx:address:array:character <- new [completing!] #? 1 #? print-string x:address:screen, xx:address:array:character #? 1 trace [app], [slurp-regular-characters: newline encountered; unwinding stack] continue-from complete:continuation ] # read characters from keyboard, print them to screen in the comment color. # # Simpler version of slurp-regular-characters; doesn't handle comments or # strings. Tracks an extra count in case we backspace out of it recipe slurp-comment [ default-space:address:array:location <- new location:type, 30:literal result:address:buffer <- next-ingredient k:address:keyboard <- next-ingredient x:address:screen <- next-ingredient complete:continuation <- next-ingredient trace [app], [slurp-comment] # use this to track when backspace should reset color characters-slurped:number <- copy 1:literal # for the initial '#' that's already appended to result { +next-character # read character c:character, k:address:keyboard <- wait-for-key k:address:keyboard # quit? { ctrl-d?:boolean <- equal c:character, 4:literal/ctrl-d/eof break-unless ctrl-d?:boolean trace [app], [slurp-comment: ctrl-d] reply 0:literal, k:address:keyboard/same-as-ingredient:1, x:address:screen/same-as-ingredient:2 } { null?:boolean <- equal c:character, 0:literal/null break-unless null?:boolean trace [app], [slurp-comment: null] reply 0:literal, k:address:keyboard/same-as-ingredient:1, x:address:screen/same-as-ingredient:2 } # print print-character x:address:screen, c:character, 4:literal/blue # append result:address:buffer <- buffer-append result:address:buffer, c:character # backspace? decrement { backspace?:boolean <- equal c:character, 8:literal/backspace break-unless backspace?:boolean characters-slurped:number <- subtract characters-slurped:number, 1:literal { reset-color?:boolean <- lesser-or-equal characters-slurped:number, 0:literal break-unless reset-color?:boolean trace [app], [slurp-comment: too many backspaces; returning] reply result:address:buffer, k:address:keyboard/same-as-ingredient:1, x:address:screen/same-as-ingredient:2 } loop +next-character:label } # otherwise increment characters-slurped:number <- add characters-slurped:number, 1:literal # done with this instruction? done?:boolean <- equal c:character, 10:literal/newline break-if done?:boolean loop } trace [app], [slurp-regular-characters: newline encountered; returning] reply result:address:buffer, k:address:keyboard/same-as-ingredient:1, x:address:screen/same-as-ingredient:2 ] # read characters from keyboard, print them to screen in the string color and # accumulate them into a buffer. # # Version of slurp-regular-characters that: # a) doesn't handle comments # b) handles nested strings using recursive calls to itself. Tracks an extra # count in case we backspace out of it. recipe slurp-string [ default-space:address:array:location <- new location:type, 30:literal result:address:buffer <- next-ingredient k:address:keyboard <- next-ingredient x:address:screen <- next-ingredient complete:continuation <- next-ingredient nested-string?:boolean <- next-ingredient trace [app], [slurp-string] # use this to track when backspace should reset color characters-slurped:number <- copy 1:literal # for the initial '[' that's already appended to result { +next-character # read character c:character, k:address:keyboard <- wait-for-key k:address:keyboard # quit? { ctrl-d?:boolean <- equal c:character, 4:literal/ctrl-d/eof break-unless ctrl-d?:boolean trace [app], [slurp-string: ctrl-d] reply 0:literal, 0:literal, k:address:keyboard/same-as-ingredient:1, x:address:screen/same-as-ingredient:2 } { null?:boolean <- equal c:character, 0:literal/null break-unless null?:boolean trace [app], [slurp-string: null] reply 0:literal, 0:literal, k:address:keyboard/same-as-ingredient:1, x:address:screen/same-as-ingredient:2 } # string { string?:boolean <- equal c:character, 91:literal/open-bracket break-unless string?:boolean trace [app], [slurp-string: open-bracket encountered; recursing] print-character x:address:screen, c:character, 6:literal/cyan result:address:buffer <- buffer-append result:address:buffer, c:character # make a recursive call to handle nested strings result:address:buffer, tmp:number, k:address:keyboard, x:address:screen <- slurp-string result:address:buffer, k:address:keyboard, x:address:screen, complete:continuation, 1:literal/nested? # but if we backspace over a completed nested string, handle it in the caller characters-slurped:number <- add characters-slurped:number, tmp:number, 1:literal # for the leading '[' loop +next-character:label } # print print-character x:address:screen, c:character, 6:literal/cyan # append result:address:buffer <- buffer-append result:address:buffer, c:character # backspace? decrement { backspace?:boolean <- equal c:character, 8:literal/backspace break-unless backspace?:boolean characters-slurped:number <- subtract characters-slurped:number, 1:literal { reset-color?:boolean <- lesser-or-equal characters-slurped:number, 0:literal break-unless reset-color?:boolean trace [app], [slurp-string: too many backspaces; returning] reply result:address:buffer/same-as-ingredient:0, 0:literal, k:address:keyboard/same-as-ingredient:1, x:address:screen/same-as-ingredient:2 } loop +next-character:label } # otherwise increment characters-slurped:number <- add characters-slurped:number, 1:literal # done with this instruction? done?:boolean <- equal c:character, 93:literal/close-bracket break-if done?:boolean loop } { break-unless nested-string?:boolean # nested string? return like a normal recipe reply result:address:buffer, characters-slurped:number, k:address:keyboard, x:address:screen } # top-level string call? recurse trace [app], [slurp-string: close-bracket encountered; recursing to regular characters] result:address:buffer, k:address:keyboard, x:address:screen <- slurp-regular-characters result:address:buffer, k:address:keyboard, x:address:screen, complete:continuation # backspaced back into this string trace [app], [slurp-string: backspaced back into string; restarting] jump +next-character:label ] recipe slurp-assignment [ default-space:address:array:location <- new location:type, 30:literal result:address:buffer <- next-ingredient k:address:keyboard <- next-ingredient x:address:screen <- next-ingredient complete:continuation <- next-ingredient { +next-character # read character c:character, k:address:keyboard <- wait-for-key k:address:keyboard # quit? { ctrl-d?:boolean <- equal c:character, 4:literal/ctrl-d/eof break-unless ctrl-d?:boolean trace [app], [slurp-assignment: ctrl-d] reply 0:literal, k:address:keyboard/same-as-ingredient:1, x:address:screen/same-as-ingredient:2 } { null?:boolean <- equal c:character, 0:literal/null break-unless null?:boolean trace [app], [slurp-assignment: null] reply 0:literal, k:address:keyboard/same-as-ingredient:1, x:address:screen/same-as-ingredient:2 } # print print-character x:address:screen, c:character, 1:literal/red trace [app], [slurp-assignment: saved one character] # append result:address:buffer <- buffer-append result:address:buffer, c:character # backspace? return { backspace?:boolean <- equal c:character, 8:literal/backspace break-unless backspace?:boolean trace [app], [slurp-assignment: backspace; returning] reply result:address:buffer/same-as-ingredient:0, k:address:keyboard/same-as-ingredient:1, x:address:screen/same-as-ingredient:2 } } trace [app], [slurp-assignment: done, recursing to regular characters] result:address:buffer, k:address:keyboard, x:address:screen <- slurp-regular-characters result:address:buffer, k:address:keyboard, x:address:screen, complete:continuation # backspaced back into this string trace [app], [slurp-assignment: backspaced back into assignment; restarting] jump +next-character:label ] scenario read-instruction-color-comment [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [# comment ] run [ read-instruction keyboard:address, screen:address ] screen-should-contain-in-color 4:literal/blue, [ .# comment . . . ] screen-should-contain-in-color 7:literal/white, [ . . . . ] ] scenario read-instruction-cancel-comment-on-backspace [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [#a««z ] replace-in-keyboard 171:literal/«, 8:literal/backspace run [ read-instruction keyboard:address, screen:address ] screen-should-contain-in-color 4:literal/blue, [ . . . . ] screen-should-contain-in-color 7:literal/white, [ .z . . . ] ] scenario read-instruction-cancel-comment-on-backspace2 [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [#ab«««z ] replace-in-keyboard 171:literal/«, 8:literal/backspace run [ read-instruction keyboard:address, screen:address ] screen-should-contain-in-color 4:literal/blue, [ . . . . ] screen-should-contain-in-color 7:literal/white, [ .z . . . ] ] scenario read-instruction-cancel-comment-on-backspace3 [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [#a«z ] replace-in-keyboard 171:literal/«, 8:literal/backspace run [ read-instruction keyboard:address, screen:address ] screen-should-contain-in-color 4:literal/blue, [ .#z . . . ] screen-should-contain-in-color 7:literal/white, [ . . . . ] ] scenario read-instruction-color-string [ #? $start-tracing #? 1 assume-screen 30:literal/width, 5:literal/height assume-keyboard [abc [string] ] run [ read-instruction keyboard:address, screen:address ] screen-should-contain [ .abc [string] . . . ] screen-should-contain-in-color 6:literal/cyan, [ . [string] . . . ] screen-should-contain-in-color 7:literal/white, [ .abc . . . ] ] scenario read-instruction-color-string-multiline [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [abc [line1 line2] ] run [ read-instruction keyboard:address, screen:address ] screen-should-contain [ .abc [line1 . .line2] . . . ] screen-should-contain-in-color 6:literal/cyan, [ . [line1 . .line2] . . . ] screen-should-contain-in-color 7:literal/white, [ .abc . . . . . ] ] scenario read-instruction-color-string-and-comment [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [abc [string] # comment ] run [ read-instruction keyboard:address, screen:address ] screen-should-contain [ .abc [string] # comment . . . ] screen-should-contain-in-color 4:literal/blue, [ . # comment . . . ] screen-should-contain-in-color 6:literal/cyan, [ . [string] . . . ] screen-should-contain-in-color 7:literal/white, [ .abc . . . ] ] scenario read-instruction-ignore-comment-inside-string [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [abc [string # not a comment] ] run [ read-instruction keyboard:address, screen:address ] screen-should-contain [ .abc [string # not a comment] . . . ] screen-should-contain-in-color 6:literal/cyan, [ . [string # not a comment] . . . ] screen-should-contain-in-color 7:literal/white, [ .abc . . . ] screen-should-contain-in-color 4:literal/blue, [ . . . . ] ] scenario read-instruction-ignore-string-inside-comment [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [abc # comment [not a string] ] run [ read-instruction keyboard:address, screen:address ] screen-should-contain [ .abc # comment [not a string] . . . ] screen-should-contain-in-color 6:literal/cyan, [ . . . . ] screen-should-contain-in-color 7:literal/white, [ .abc . . . ] screen-should-contain-in-color 4:literal/blue, [ . # comment [not a string] . . . ] ] scenario read-instruction-color-string-inside-string [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [abc [string [inner string]] ] run [ #? $start-tracing #? 1 read-instruction keyboard:address, screen:address #? $stop-tracing #? 1 #? $browse-trace #? 1 ] screen-should-contain [ .abc [string [inner string]] . . . ] screen-should-contain-in-color 6:literal/cyan, [ . [string [inner string]] . . . ] screen-should-contain-in-color 7:literal/white, [ .abc . . . ] ] scenario read-instruction-cancel-string-on-backspace [ assume-screen 30:literal/width, 5:literal/height # need to escape the '[' once for 'scenario' and once for 'assume-keyboard' assume-keyboard [\\\[a««z ] replace-in-keyboard 171:literal/«, 8:literal/backspace run [ read-instruction keyboard:address, screen:address ] screen-should-contain-in-color 6:literal/cyan, [ . . . . ] screen-should-contain-in-color 7:literal/white, [ .z . . . ] ] scenario read-instruction-cancel-string-inside-string-on-backspace [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [[a[b]«««b] ] replace-in-keyboard 171:literal/«, 8:literal/backspace run [ read-instruction keyboard:address, screen:address ] screen-should-contain-in-color 6:literal/cyan, [ .[ab] . . . ] screen-should-contain-in-color 7:literal/white, [ . . . . ] ] scenario read-instruction-backspace-back-into-string [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [[a]«b ] replace-in-keyboard 171:literal/«, 8:literal/backspace run [ read-instruction keyboard:address, screen:address ] screen-should-contain [ .\\\[ab . . . ] screen-should-contain-in-color 6:literal/cyan, [ .\\\[ab . . . ] screen-should-contain-in-color 7:literal/white, [ . . . . ] # todo: trace sequence of events # slurp-regular-characters: [ # slurp-regular-characters/slurp-string: a # slurp-regular-characters/slurp-string: ] # slurp-regular-characters/slurp-string/slurp-regular-characters: backspace # slurp-regular-characters/slurp-string: b ] scenario read-instruction-highlight-start-of-assignment [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [a < ] run [ read-instruction keyboard:address, screen:address ] screen-should-contain [ .a < . . . ] screen-should-contain-in-color 1:literal/red, [ . < . . . ] screen-should-contain-in-color 7:literal/white, [ .a . . . ] ] scenario read-instruction-highlight-assignment [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [a <- b ] run [ read-instruction keyboard:address, screen:address ] screen-should-contain [ .a <- b . . . ] screen-should-contain-in-color 1:literal/red, [ . <- . . . ] screen-should-contain-in-color 7:literal/white, [ .a b . . . ] ] scenario read-instruction-backspace-over-assignment [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [a <-« ] replace-in-keyboard 171:literal/«, 8:literal/backspace run [ read-instruction keyboard:address, screen:address ] screen-should-contain [ .a < . . . ] screen-should-contain-in-color 1:literal/red, [ . < . . . ] screen-should-contain-in-color 7:literal/white, [ .a . . . ] ] scenario read-instruction-assignment-continues-after-backspace [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [a <-«- ] replace-in-keyboard 171:literal/«, 8:literal/backspace run [ read-instruction keyboard:address, screen:address ] screen-should-contain [ .a <- . . . ] screen-should-contain-in-color 1:literal/red, [ . <- . . . ] screen-should-contain-in-color 7:literal/white, [ .a . . . ] ] scenario read-instruction-assignment-continues-after-backspace2 [ assume-screen 30:literal/width, 5:literal/height assume-keyboard [a <- ««- ] replace-in-keyboard 171:literal/«, 8:literal/backspace run [ read-instruction keyboard:address, screen:address #? $browse-trace #? 1 ] screen-should-contain [ .a <- . . . ] screen-should-contain-in-color 1:literal/red, [ . <- . . . ] screen-should-contain-in-color 7:literal/white, [ .a . . . ] ]