package org.jetbrains.plugins.clojure.parser.util; import com.intellij.lang.PsiBuilder; import com.intellij.psi.tree.IElementType; import com.intellij.psi.tree.TokenSet; /** * @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; PsiBuilder.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; } /** * Checks, that following element sequence is like given * * @param builder Given PsiBuilder * @param dropMarker to drop marker after successful checking or rollback it? * @param elems Array of need elements in order * @return true if following sequence is like a given */ public static boolean lookAhead(PsiBuilder builder, boolean dropMarker, IElementType... elems) { if (elems.length == 0) { return false; } if (elems.length == 1) { return elems[0].equals(builder.getTokenType()); } PsiBuilder.Marker rb = builder.mark(); int i = 0; while (!builder.eof() && i < elems.length && elems[i].equals(builder.getTokenType())) { builder.advanceLexer(); i++; } if (dropMarker && i == elems.length) { rb.drop(); } else { 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) { PsiBuilder.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) { PsiBuilder.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); } }