/* Copyright 2018, 2019 Simon Forman This file is part of Thun Thun is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. Thun is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with Thun. If not see . Parser */ :- set_prolog_flag(double_quotes, codes). joy_parse([T|J]) --> blanks, joy_term(T), blanks, joy_parse(J). joy_parse([]) --> []. joy_term(N) --> num(N), !. joy_term(J) --> "[", !, joy_parse(J), "]". joy_term(C) --> symbol(C). symbol(C) --> chars(Chars), !, {Chars \= "==", atom_codes(C, Chars)}. % Line is the next new-line delimited line from standard input stream as % a list of character codes. line(Line) :- get_code(X), line(X, Line). line(10, []) :- !. % break on new-lines. line(-1, [eof]) :- !. % break on EOF line(X, [X|Line]) :- get_code(Y), !, line(Y, Line). chars([Ch|Rest]) --> char(Ch), chars(Rest). chars([Ch]) --> char(Ch). char(Ch) --> [Ch], { Ch \== 0'[, Ch \== 0'], Ch >= 33, Ch =< 126 }. blanks --> blank, !, blanks. blanks --> []. blank --> [32] | [13] | [10]. % TODO: negative numbers, floats, scientific notation. num(N) --> digits(Codes), !, { num(N, Codes) }. num(_, []) :- fail, !. num(N, [C|Codes]) :- number_codes(N, [C|Codes]). digits([H|T]) --> digit(H), !, digits(T). digits([]) --> []. digit(C) --> [C], { nonvar(C), C =< 57, C >= 48 }.