+ push-parse-idx
+ inc-parse-idx
+
+ nextchar [char] t <>
+ nextchar [char] f <>
+ and if pop-parse-idx false exit then
+
+ inc-parse-idx
+ delim? if
+ pop-parse-idx
+ true
+ else
+ pop-parse-idx
+ false
+ then
+;
+
+: str-equiv? ( str -- bool )
+
+ push-parse-idx
+
+ true -rot
+
+ swap dup rot + swap
+
+ do
+ i @ nextchar <> if
+ drop false
+ leave
+ then
+
+ inc-parse-idx
+ loop
+
+ delim? false = if drop false then
+
+ pop-parse-idx
+;
+
+: character? ( -- bool )
+ nextchar [char] # <> if false exit then
+
+ push-parse-idx
+ inc-parse-idx
+
+ nextchar [char] \ <> if pop-parse-idx false exit then
+
+ inc-parse-idx
+
+ S" newline" str-equiv? if pop-parse-idx true exit then
+ S" space" str-equiv? if pop-parse-idx true exit then
+ S" tab" str-equiv? if pop-parse-idx true exit then
+
+ charavailable? false = if pop-parse-idx false exit then
+
+ pop-parse-idx true
+;
+
+: pair? ( -- bool )
+ nextchar [char] ( = ;
+
+: string? ( -- bool )
+ nextchar [char] " = ;
+
+: readnum ( -- num-atom )
+ minus? dup if
+ inc-parse-idx
+ then
+
+ 0
+
+ begin digit? while
+ 10 * nextchar [char] 0 - +
+ inc-parse-idx
+ repeat
+
+ swap if negate then
+
+ number-type
+;
+
+: readbool ( -- bool-atom )
+ inc-parse-idx
+
+ nextchar [char] f = if
+ false
+ else
+ true
+ then
+
+ inc-parse-idx
+
+ boolean-type
+;
+
+: readchar ( -- char-atom )
+ inc-parse-idx
+ inc-parse-idx
+
+ S" newline" str-equiv? if 7 parse-idx +! '\n' character-type exit then
+ S" space" str-equiv? if 5 parse-idx +! bl character-type exit then
+ S" tab" str-equiv? if 3 parse-idx +! 9 character-type exit then
+
+ nextchar character-type
+
+ inc-parse-idx
+;
+
+: readstring ( -- str-obj )
+ nextchar [char] " = if
+ inc-parse-idx
+
+ delim? false = if
+ bold fg red
+ ." No delimiter following right double quote. Aborting." cr
+ reset-term abort
+ then
+
+ dec-parse-idx
+
+ 0 nil-type exit