4 * pack a javascript file, and return a shorter version!
6 * a bit picky at present with ; and crlf reading...
7 * @arg ts {TokenStream}
12 public errordomain CompressWhiteError {
16 public string CompressWhite (TokenStream ts, Packer packer, bool keepWhite) // throws CompressWhiteError,TokenStreamError
18 //keepWhite = keepWhite || false;
20 //ts.dumpAllFlat(); GLib.Process.exit(1);
22 //var str = File.read(fn);
32 if (tok.type == TokenType.WHIT) {
38 // just spaces, not \n!
39 //if (tok.data.indexOf("\n") < 0) {
45 if (tok.data == "}") {
47 if (ts.lookTok(0).type == TokenType.NAME && ts.look(1,true).name == TokenName.NEWLINE) {
48 ts.look(0,true).outData = ts.look(0,true).data+"\n";
54 // add semi-colon's where linebreaks are used... - not foolproof yet.!
55 if (tok.type == TokenType.NAME) {
56 //var tokident = ts.look(-1).data + tok.data + ts.look(1).data + ts.look(2).data;
57 // a = new function() {}
58 if (ts.lookTok(1).data == "=" && ts.lookTok(2).name == TokenName.NEW &&
59 ts.lookTok(3).name == TokenName.FUNCTION) {
63 ts.balance(TokenName.LEFT_PAREN); //"(");
66 ts.balance(TokenName.LEFT_CURLY); //"{");
67 // if next is not ';' -> make it so...
69 if (ts.lookTok(1).data != ";" && ts.lookTok(1).data != "}" && ts.lookTok(1).name == TokenName.NEWLINE) {
70 ts.look(0,true).outData = ts.lookTok(0).data +";";
76 // a = function() { ... -- add a semi colon a tthe end if not one there..
78 if (ts.lookTok(1).data == "=" && ts.lookTok(2).name == TokenName.FUNCTION) {
80 //println("got = function() ");
86 //print("NEXT = should be brac: %s\n", ts.lookTok(1).asString());
88 //print("cursor = %d", ts.cursor);
90 if (ts.lookTok(1).data != "(" || ts.balance(TokenName.LEFT_PAREN /*"("*/).size < 1 ){
91 print("balance ( issue on line %d\n", ts.toArray().get(cu).line);
93 print(">>>>>>>>>>>>>>>>>HERE>>>>>>>>>>>>");
96 throw new CompressWhiteError.BRACE( "could not find end lbrace!!!" );
98 //print("cursor = %d", ts.cursor);
99 //print("CUR = should be ): %s\n", ts.lookTok(0).asString());
102 //print("CUR = should be {: %s\n", ts.lookTok(0).asString());
103 cu = ts.cursor; // set the cursor to here.. so the next bit of the code will check inside the method.
105 //print("cursor = %d", ts.cursor);
106 // print("AFTER BALANCE (");
107 //ts.dump(cu, ts.cursor);
110 ts.cursor--; // cursor at the (
111 if (tok.data != "{" || ts.balance(TokenName.LEFT_CURLY /*"("*/).size < 1 ){
114 print(">>>>>>>>>>>>>>>>>HERE>>>>>>>>>>>>");
117 throw new CompressWhiteError.BRACE( "could not find end lbrace!!!");
119 //print('FN: '+ts.tokens[cu].toString());
120 //print('F1: '+ts.lookTok(1).toString());
121 //print('F2: '+ts.look(1,true).toString());
123 // if next is not ';' -> make it so...
124 // although this var a=function(){},v,c; causes
125 if (ts.lookTok(1).data != ";" && ts.lookTok(1).data != "}" && ts.look(1,true).name == TokenName.NEWLINE) {
127 ts.look(0,true).outData = ts.look(0,true).data+";";
128 // print("ADDING SEMI: " + ts.look(0).toString());
129 //ts.dump(cu, ts.cursor+2);
132 //ts.dump(cu, ts.cursor+2);
137 // next item is a name..
138 if ((ts.lookTok(1).type == TokenType.NAME || ts.lookTok(1).type == TokenType.KEYW )
139 && ts.look(1,true).name == TokenName.NEWLINE) {
140 // preserve linebraek
141 ts.look(0,true).outData = ts.look(0,true).data+"\n";
143 // method call followed by name..
144 if (ts.lookTok(1).data == "(") {
147 ts.balance(TokenName.LEFT_PAREN); //"(");
148 // although this var a=function(){},v,c; causes
150 if (ts.lookTok(1).type == TokenType.NAME && ts.look(1,true).name == TokenName.NEWLINE) {
152 ts.look(0,true).outData = ts.look(0,true).data+"\n";
160 // function a () { ... };
162 if (ts.look(-1).isTypeN(Script.TOKfunction) && ts.look(1).isTypeN(Script.TOKlparen)) {
164 //println("got = function() ");
167 ts.balance("lparen");
168 ts.balance("lbrace");
169 // if next is not ';' -> make it so...
170 // although this var a=function(){},v,c; causes
171 if (!ts.look(1).isData(';') && !ts.look(1).isData('}') && ts.look(1,true).isLineBreak()) {
172 ts.cur().outData = ts.cur().data+";";
182 if (ts.lookTok(1).data == "=" && ts.lookTok(2).data == "{") {
184 //println("----------*** 3 *** --------------");
187 if (ts.balance(TokenName.LEFT_CURLY /*"{" */).size < 1 ){
190 print(">>>>>>>>>>>>>>>>>HERE>>>>>>>>>>>>");
193 throw new CompressWhiteError.BRACE("could not find end lbrace!!!");
195 // if next is not ';' -> make it so...
197 if (ts.lookTok(1).data != ";" && ts.lookTok(1).data != "}" && ts.look(1,true).name==TokenName.NEWLINE) {
198 ts.look(0,true).outData = ts.look(0,true).data +";";
206 // a = function(....) { }
213 //println("got Token: " + tok.type);
218 // things that need space appending
219 case TokenName.FUNCTION:
220 case TokenName.BREAK:
221 case TokenName.CONTINUE:
222 // if next item is a identifier..
223 if (ts.lookTok(1).type == TokenType.NAME || Regex.match_simple("^[a-z]+$", ts.lookTok(1).data, GLib.RegexCompileFlags.CASELESS) ) { // as include is a keyword for us!!
224 tok.outData = tok.data + " ";
229 case TokenName.RETURN: // if next item is not a semi; (or }
230 if (ts.lookTok(1).data == ";" || ts.lookTok(1).data == "}") {
233 tok.outData = tok.data + " ";
238 case TokenName.ELSE: // if next item is not a semi; (or }
239 if (ts.lookTok(1).name != TokenName.IF) {
242 // add a space if next element is 'IF'
243 tok.outData = tok.data + " ";
246 case TokenName.INCREMENT: //"++": // if previous was a plus or next is a + add a space..
247 case TokenName.DECREMENT: //"--": // if previous was a - or next is a - add a space..
249 var p = (tok.data == "--" ? "-" : "+");
251 if (ts.lookTok(1).data == p) {
252 tok.outData = tok.data + " ";
254 if (ts.lookTok(-1).data == p) {
255 tok.outData = " " + tok.data;
260 case TokenName.IN: // before and after??
261 case TokenName.INSTANCEOF:
263 tok.outData = " " + tok.data + " ";
266 case TokenName.VAR: // always after..
268 case TokenName.DELETE:
269 case TokenName.THROW:
271 case TokenName.CONST:
273 tok.outData = tok.data + " ";
277 case TokenName.TYPEOF: // what about typeof(
278 if (ts.lookTok(1).data != "(") {
279 tok.outData = tok.data + " ";
282 case TokenName.SEMICOLON: //";":
283 //remove semicolon before brace --
284 //if(ts.look(1).isTypeN(Script.TOKrbrace)) {
296 // NOW OUTPUT THE THING.
297 //var f = new File(minfile, File.NEW);
301 //try { out.length = ts.slen; } catch (e) {} // prealloc.
307 tok = keepWhite ? ts.next() : ts.nextTok();
312 if (tok.type == TokenType.COMM) {
316 ///print(tok.type + ':' + tok.data);
318 if (tok.type == TokenType.NAME &&
319 tok.identifier != null &&
320 tok.identifier.mungedValue.length > 0) {
321 //f.write(tok.identifier.mungedValue);
322 //print("MUNGED: " + tok.identifier.mungedValue);
323 outstr += tok.identifier.mungedValue;
327 // at this point we can apply a text translation kit...
329 //if ((tok.type == "STRN") && (tok.name== "DOUBLE_QUOTE")) {
330 // if (packer && packer.stringHandler) {
331 // outstr += packer.stringHandler(tok);
336 outstr += tok.outData != "" ? tok.outData : tok.data;
338 if ((tok.name == TokenName.SEMICOLON || tok.name == TokenName.RIGHT_CURLY) && (outstr.length - outoff > 255)) {
339 outoff = outstr.length;
345 // remove the last ';' !!!
346 if (out.substring(out.length-1) == ';') {
347 return out.substring(0,out.length-1);