Consider a simple prefix arithmetic notation; a few examples:
+ 1 2 i.e. 1+2
+ * 4 4 * 6 6 i.e. (4*4) + (6*6)
/ 8 * 2 2 i.e. (8/(2*2))
The above grammar can be represented in Prolog as:
.
operator(plus).
operator(minus).
operator(multiply).
operator(divide)
Op, num, num) :- operator(Op).
arithmetic(
Op, A, num) :-
arithmetic(Op),
operator(A = arithmetic(_,_,_).
Op, num, A) :-
arithmetic(Op),
operator(A = arithmetic(_,_,_).
Op, A, B) :-
arithmetic(Op),
operator(A = arithmetic(_,_,_),
B = arithmetic(_,_,_).
So ‘parsing’ the prior examples would produce ASTs of the form (actual digits excluded)
, num, num). % + 1 2
arithmetic(plus, arithmetic(multiply, num, num), arithmetic(multiply, num, num)). % + * 4 4 * 6 6
arithmetic(plus, num, arithmetic(multiply, num, num)). % / 8 * 2 2 arithmetic(divide
Given this AST, the objective is to spit out a sequence of HTML
<span>
for coloring the various nodes. Prolog is no
stranger to tree rewriting; here’s one approach:
Op, Str) :-
stringify_op(Op = plus, Str = '+')
(; (Op = minus, Str = '-')
; (Op = multiply, Str = '*')
; (Op = divide, Str = '/').
, Str) :-
highlight_num(numatom(Str), '<span class="number">~a</span>', [num]).
format(
Op, Str) :-
highlight_op(Op, S1),
stringify_op(atom(Str), '<span class="operator">~a</span>', [S1]).
format(
Op, num, num, Highlight) :-
highlight_arithmetic(Op, OpHL),
highlight_op(, N1HL),
highlight_num(num, N2HL),
highlight_num(numatom(Highlight), '~a ~a ~a', [OpHL, N1HL, N2HL]).
format(
Op, num, A, Highlight) :-
highlight_arithmetic(Op, OpHL),
highlight_op(, N1HL),
highlight_num(numA = arithmetic(SubOp, SubA, SubB),
SubOp, SubA, SubB, SubHL),
highlight_arithmetic(atom(Highlight), '~a ~a ~a', [OpHL, N1HL, SubHL]).
format(
Op, A, num, Highlight) :-
highlight_arithmetic(Op, OpHL),
highlight_op(A = arithmetic(SubOp, SubA, SubB),
SubOp, SubA, SubB, SubHL),
highlight_arithmetic(, N1HL),
highlight_num(numatom(Highlight), '~a ~a ~a', [OpHL, SubHL, N1HL]).
format(
Op, A, B, Highlight) :-
highlight_arithmetic(Op, OpHL),
highlight_op(A = arithmetic(AOp, AA, AB),
B = arithmetic(BOp, BB, BA),
AOp, AA, AB, AHL),
highlight_arithmetic(BOp, BA, BB, BHL),
highlight_arithmetic(atom(Highlight), '~a ~a ~a', [OpHL, AHL, BHL]). format(
For syntax highlighting, the following CSS styling is used:
.operator {
color: red;
}.number {
color: blue;
}
Now for generating the HTML. Each node is converted to its equivalent span and concatenated. Again, using the initial examples:
?- highlight_arithmetic(plus,num,num,HL).
HL = '<span class="operator">+</span> <span class="number">num</span> <span class="number">num</span>' .
+ num num
?- highlight_arithmetic(plus, arithmetic(multiply, num, num), arithmetic(multiply, num, num), HL).
HL = '<span class="operator">+</span> <span class="operator">*</span> <span class="number">num</span> <span class="number">num</span> <span class="operator">*</span> <span class="number">num</span> <span class="number">num</span>' .
+ * num num * num num
?- highlight_arithmetic(divide, num, arithmetic(multiply, num, num), HL).
HL = '<span class="operator">/</span> <span class="number">num</span> <span class="operator">*</span> <span class="number">num</span> <span class="number">num</span>' .
/ num * num num