687
|
1 package luan.modules.parsers;
|
|
2
|
|
3 import luan.LuanException;
|
|
4
|
|
5
|
|
6 public final class Theme {
|
|
7
|
|
8 public static String toLuan(String source) throws LuanException {
|
|
9 try {
|
|
10 return new Theme(source).parse();
|
|
11 } catch(ParseException e) {
|
|
12 throw new LuanException(e.getMessage());
|
|
13 }
|
|
14 }
|
|
15
|
|
16 private final Parser parser;
|
|
17
|
|
18 private Theme(String source) {
|
|
19 this.parser = new Parser(source);
|
|
20 }
|
|
21
|
|
22 private ParseException exception(String msg) {
|
|
23 // parser.failure();
|
|
24 return new ParseException(parser,msg);
|
|
25 }
|
|
26
|
|
27 private String parse() throws ParseException {
|
|
28 StringBuilder stmts = new StringBuilder();
|
|
29 stmts.append( "local M = {}; " );
|
|
30 while( !parser.endOfInput() ) {
|
|
31 String def = parseDef();
|
|
32 if( def != null ) {
|
|
33 stmts.append(def);
|
|
34 } else {
|
|
35 // parser.anyChar();
|
|
36 stmts.append(parsePadding());
|
|
37 }
|
|
38 }
|
|
39 stmts.append( "\n\nreturn M\n" );
|
|
40 return stmts.toString();
|
|
41 }
|
|
42
|
|
43 private String parsePadding() throws ParseException {
|
|
44 int start = parser.currentIndex();
|
|
45 if( parser.match("--") ) {
|
|
46 while( parser.noneOf("\r\n") );
|
|
47 } else if( !parser.anyOf(" \t\r\n") ) {
|
|
48 throw exception("unexpected text");
|
|
49 }
|
|
50 return parser.textFrom(start);
|
|
51 }
|
|
52
|
|
53 private String parseDef() throws ParseException {
|
|
54 int start = parser.begin();
|
|
55 if( !parser.match("{define:") )
|
|
56 return parser.failure(null);
|
|
57 String name = parseName();
|
|
58 if( name==null )
|
|
59 throw exception("invalid block name");
|
|
60 if( !parser.match('}') )
|
|
61 throw exception("unclosed define tag");
|
|
62 String block = parseBody("define:"+name);
|
688
|
63 String rtn = "function M." + name + "(env) " + block + " end; ";
|
687
|
64 return parser.success(rtn);
|
|
65 }
|
|
66
|
|
67 private String parseBody(String tagName) throws ParseException {
|
|
68 StringBuilder stmts = new StringBuilder();
|
|
69 int start = parser.currentIndex();
|
|
70 int end = start;
|
|
71 while( !matchEndTag(tagName) ) {
|
|
72 if( parser.endOfInput() ) {
|
|
73 parser.failure();
|
|
74 throw exception("unclosed block");
|
|
75 }
|
|
76 String block = parseBlock();
|
|
77 if( block != null ) {
|
|
78 addText(start,end,stmts);
|
|
79 start = parser.currentIndex();
|
|
80 stmts.append(block);
|
|
81 continue;
|
|
82 }
|
|
83 String simpleTag = parseSimpleTag();
|
|
84 if( simpleTag != null ) {
|
|
85 addText(start,end,stmts);
|
|
86 start = parser.currentIndex();
|
|
87 stmts.append(simpleTag);
|
|
88 continue;
|
|
89 }
|
690
|
90 if( parser.match("<%") ) {
|
|
91 addText(start,end,stmts);
|
|
92 start = parser.currentIndex();
|
|
93 stmts.append("%><%='<%'%><%");
|
|
94 continue;
|
|
95 }
|
687
|
96 parser.anyChar();
|
|
97 end = parser.currentIndex();
|
|
98 }
|
|
99 addText(start,end,stmts);
|
|
100 return stmts.toString();
|
|
101 }
|
|
102
|
|
103 private boolean matchEndTag(String tagName) {
|
|
104 parser.begin();
|
|
105 /*
|
|
106 if( tagName.startsWith("define:") )
|
|
107 EndOfLine();
|
|
108 */
|
|
109 return parser.match("{/") && parser.match(tagName) && parser.match('}') ? parser.success() : parser.failure();
|
|
110 }
|
|
111
|
|
112 private void addText(int start,int end,StringBuilder stmts) {
|
|
113 if( start < end ) {
|
688
|
114 stmts.append( "%>" ).append( parser.text.substring(start,end) ).append( "<%" );
|
687
|
115 }
|
|
116 }
|
|
117
|
|
118 private String parseBlock() throws ParseException {
|
|
119 int start = parser.begin();
|
|
120 if( !parser.match("{block:") )
|
|
121 return parser.failure(null);
|
|
122 String name = parseName();
|
|
123 if( name==null ) {
|
|
124 parser.failure();
|
|
125 throw exception("invalid block name");
|
|
126 }
|
|
127 if( !parser.match('}') )
|
|
128 return parser.failure(null);
|
|
129 String block = parseBody("block:"+name);
|
688
|
130 String rtn = " env."+ name + "( env, function(env) " + block + "end); ";
|
687
|
131 // String rtn = "<% env." + tag.name + "(" + (tag.attrs.isEmpty() ? "nil" : table(tag.attrs)) + ",env,function(env) %>" + block + "<% end) %>";
|
|
132 return parser.success(rtn);
|
|
133 }
|
|
134
|
|
135 private String parseSimpleTag() throws ParseException {
|
|
136 int start = parser.begin();
|
|
137 if( !parser.match('{') )
|
|
138 return parser.failure(null);
|
|
139 String name = parseName();
|
|
140 if( name==null )
|
|
141 return parser.failure(null);
|
|
142 if( !parser.match('}') )
|
|
143 return parser.failure(null);
|
|
144 // rtn = "<% env." + name + (attrs.isEmpty() ? "()" : table(attrs)) + " %>";
|
688
|
145 String rtn = " env." + name + "(env); ";
|
687
|
146 return parser.success(rtn);
|
|
147 }
|
|
148
|
|
149 private boolean BlankLine() {
|
|
150 parser.begin();
|
|
151 while( parser.anyOf(" \t") );
|
|
152 return EndOfLine() ? parser.success() : parser.failure();
|
|
153 }
|
|
154
|
|
155 private boolean EndOfLine() {
|
|
156 return parser.match( "\r\n" ) || parser.match( '\r' ) || parser.match( '\n' );
|
|
157 }
|
|
158
|
|
159 private String parseName() throws ParseException {
|
|
160 int start = parser.begin();
|
|
161 if( parser.match('/') ) {
|
|
162 parser.failure();
|
|
163 throw exception("bad closing tag");
|
|
164 }
|
|
165 if( parser.match("define:") ) {
|
|
166 parser.failure();
|
|
167 throw exception("unexpected definition");
|
|
168 }
|
|
169 if( !FirstNameChar() )
|
|
170 return parser.failure(null);
|
|
171 while( NameChar() );
|
|
172 String match = parser.textFrom(start);
|
|
173 return parser.success(match);
|
|
174 }
|
|
175
|
|
176 private boolean FirstNameChar() {
|
|
177 return parser.inCharRange('a', 'z') || parser.inCharRange('A', 'Z') || parser.match('_');
|
|
178 }
|
|
179
|
|
180 private boolean NameChar() {
|
|
181 return FirstNameChar() || parser.inCharRange('0', '9');
|
|
182 }
|
|
183
|
|
184 }
|