15 public class TokenStream : Object
18 Gee.ArrayList<Token> tokens;
19 int cursor; // where are we in the stream.
22 public TokenStream(Gee.ArrayList<Token> tokens) {
31 public void rewind() {
38 public Token? look (int n, bool considerWhitespace)
42 if (considerWhitespace == true) {
44 if (this.cursor+n < 0 || this.cursor+n > (this.tokens.size -1)) {
45 return new Token("", "VOID", "START_OF_STREAM");
47 return this.tokens.get(this.cursor+n);
56 return new Token("", "VOID", "START_OF_STREAM");
58 if (i > this.tokens.size) {
59 return new Token("", "VOID", "END_OF_STREAM");
62 if (i != this.cursor && this.tokens.get(i).is("WHIT")) {
63 i += (n < 0) ? -1 : 1;
68 return this.tokens.get(i);
71 i += (n < 0) ? -1 : 1;
74 return new Token("", "VOID", "STREAM_ERROR"); // because null isn't an object and caller always expects an object
78 public int lookFor (string data)
81 var i = this.cursor < 0 ? 0 : this.cursor ;
84 if (i >= this.tokens.size) {
87 if (this.tokens.get(i).data == data) {
93 // should not get here!
100 * look ahead (or back) x number of tokens (which are not comment or whitespace)
103 public Token lookTok (int n) {
119 return new Token("", "VOID", "END_OF_STREAM");
121 if (i > this.tokens.length) {
122 return new Token("", "VOID", "END_OF_STREAM");
125 if (i != this.cursor && ( this.tokens.get(i).is("WHIT") || this.tokens.get(i).is("COMM"))) {
126 i += (n < 0) ? -1 : 1;
131 return this.tokens.get(i);
134 i += (n < 0) ? -1 : 1;
136 // should never get here..
137 return new Token("", "VOID", "END_OF_STREAM");; // because null isn't an object and caller always expects an object;
142 * @return {Token|null}
143 * next token (with white space)
147 public Token? next() {
150 //if (typeof howMany == "undefined") howMany = 1;
151 // if (howMany < 1) { return null; }
153 if (this.cursor+1 >= this.tokens.size) {
157 return this.tokens.get(this.cursor);
161 public Token? nextM(uint howMany) {
163 //if (typeof howMany == "undefined") howMany = 1;
164 // if (howMany < 1) { return null; }
166 var got = new Gee.ArrayList<Token>();
168 for (var i = 1; i <= howMany; i++) {
169 if (this.cursor+i >= this.tokens.size) {
172 got.add(this.tokens.get(this.cursor+i));
174 this.cursor += howMany;
185 // what about comments after 'function'...
187 nextTok : function() {
188 return this.nextNonSpace();
190 nextNonSpace : function ()
198 if (tok.is('WHIT') || tok.is('COMM')) {
205 * @type JSDOC.Token[]
206 * @param start {String} token name or data (eg. '{'
207 * @param stop {String} (Optional) token name or data (eg. '}'
209 balance : function(/**String*/start, /**String*/stop) {
212 start = typeof(Lang.punc(start)) == 'undefined' ? start : Lang.punc(start);
214 if (!stop) stop = Lang.matching(start);
219 //Seed.print("START:" + start);
220 //Seed.print("STOP:" + stop);
221 while ((token = this.look())) {
222 if (token.is(start)) {
223 // Seed.print("balance: START : " + depth + " " + token.data);
232 if (token.is(stop)) {
234 // Seed.print("balance: STOP: " + depth + " " + token.data);
235 if (depth < 1) return got;
237 if (!this.next()) break;
242 getMatchingToken : function(/**String*/start, /**String*/stop) {
244 var cursor = this.cursor;
247 start = Lang.matching(stop);
250 if (!stop) stop = Lang.matching(start);
252 while ((token = this.tokens[cursor])) {
253 if (token.is(start)) {
257 if (token.is(stop) && cursor) {
259 if (depth == 0) return this.tokens[cursor];
266 insertAhead : function(/**JSDOC.Token*/token) {
267 this.tokens.splice(this.cursor+1, 0, token);
270 remaining : function() {
273 var tok = this.look(1,true);
274 if (!tok || !tok.is || tok.is('VOID')) {
277 ret.push(this.next(1));
282 arrayToString : function(ar) {
283 console.log(typeof(ar));
285 ar.forEach(function(e) {
290 dump: function(start, end)
292 start = Math.max(start || 0, 0);
293 end = Math.min(end || this.tokens.length, this.tokens.length);
295 for (var i =start;i < end; i++) {
297 out += (this.tokens[i].outData == false) ? this.tokens[i].data : this.tokens[i].outData;