/*
* Copyright 2011 Jon S Akhtar (Sylvanaar)
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.sylvanaar.idea.Lua.lang.parser.util;
import com.intellij.lang.PsiBuilder;
import com.intellij.lang.PsiBuilder.Marker;
import com.intellij.psi.tree.IElementType;
import com.intellij.psi.tree.TokenSet;
/**
* Utility classdef, that contains various useful methods for
* parser needs.
*
* @author ilyas
*/
public abstract class ParserUtils {
/**
* Auxiliary method for strict token appearance
*
* @param builder current builder
* @param elem given element
* @param errorMsg Message, that displays if element was not found; if errorMsg == null nothing displays
* @return true if element parsed
*/
public static boolean getToken(PsiBuilder builder, IElementType elem, String errorMsg) {
if (elem.equals(builder.getTokenType())) {
builder.advanceLexer();
return true;
} else {
if (errorMsg != null)
builder.error(errorMsg);
return false;
}
}
/**
* Auxiliary method for construction like
* <BNF>
* token?
* </BNF>
* parsing
*
* @param builder current builder
* @param elem given element
* @return true if element parsed
*/
public static boolean getToken(PsiBuilder builder, IElementType elem) {
if (elem.equals(builder.getTokenType())) {
builder.advanceLexer();
return true;
}
return false;
}
/**
* Same as simple getToken() method but with TokenSet
*
* @param builder
* @param tokenSet
* @return
*/
public static boolean getToken(PsiBuilder builder, TokenSet tokenSet) {
if (tokenSet.contains(builder.getTokenType())) {
return getToken(builder, builder.getTokenType(), null);
}
return false;
}
/**
* Same as simple getToken() method but with TokenSet
*
* @param builder
* @param tokenSet
* @return
*/
public static boolean getToken(PsiBuilder builder, TokenSet tokenSet, String msg) {
if (tokenSet.contains(builder.getTokenType())) {
return getToken(builder, builder.getTokenType(), msg);
}
return false;
}
/**
* Checks, that following element sequence is like given
*
* @param builder Given PsiBuilder
* @param elems Array of need elements in order
* @return true if following sequence is like a given
*/
public static boolean lookAhead(PsiBuilder builder, IElementType... elems) {
if (!elems[0].equals(builder.getTokenType())) return false;
if (elems.length == 1) return true;
Marker rb = builder.mark();
builder.advanceLexer();
int i = 1;
while (!builder.eof() && i < elems.length && elems[i].equals(builder.getTokenType())) {
builder.advanceLexer();
i++;
}
rb.rollbackTo();
return i == elems.length;
}
/**
* Wraps current token to node with specified element type
*
* @param builder Given builder
* @param elem Node element
* @return elem type
*/
public static IElementType eatElement(PsiBuilder builder, IElementType elem) {
Marker marker = builder.mark();
builder.advanceLexer();
marker.done(elem);
return elem;
}
/**
* Wraps current token with error message
*
* @param builder
* @param msg Error message
*/
public static void wrapError(PsiBuilder builder, String msg) {
Marker marker = builder.mark();
builder.advanceLexer();
marker.error(msg);
}
public static void advance(PsiBuilder builder, int count) {
for (int i = 0; i < count; i++) {
builder.getTokenText();
builder.advanceLexer();
}
}
public static void advance(PsiBuilder builder) {
advance(builder, 1);
}
}