/*******************************************************************************
* Copyright (c) 2013, 2016 Andrew Gvozdev and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* Andrew Gvozdev - Initial API and implementation
*******************************************************************************/
package org.eclipse.cdt.make.internal.ui.text.makefile;
import org.eclipse.cdt.make.internal.core.makefile.gnu.GNUMakefileConstants;
import org.eclipse.jface.text.rules.ICharacterScanner;
import org.eclipse.jface.text.rules.IToken;
import org.eclipse.jface.text.rules.IWordDetector;
import org.eclipse.jface.text.rules.Token;
import org.eclipse.jface.text.rules.WordRule;
public class FunctionReferenceRule extends WordRule {
/** Buffer used for pattern detection. */
private StringBuilder fBuffer= new StringBuilder();
private String startSeq;
private String endSeq;
@SuppressWarnings("nls")
private final static String[] functions = {
"subst", "patsubst", "strip", "findstring",
"filter", "filter-out", "sort",
"word", "words", "wordlist", "firstword", "lastword",
"dir", "notdir",
"suffix", "basename", "addsuffix", "addprefix",
"join", "wildcard", "realpath", "abspath",
"if", "or", "and", "foreach",
"call", "value", "eval", "origin", "flavor",
"shell", "error", "warning", "info",
};
private static class TagDetector implements IWordDetector {
private char openBracket;
private char closedBracket;
private boolean isClosedBracket = false;
private int bracketNesting = 0;
public TagDetector(String endSeq) {
if (endSeq.length() > 0 && endSeq.charAt(0) == '}') {
openBracket = '{';
closedBracket = '}';
} else {
openBracket = '(';
closedBracket = ')';
}
}
@Override
public boolean isWordStart(char c) {
isClosedBracket = c == closedBracket;
return isClosedBracket || c == '$';
}
@Override
public boolean isWordPart(char c) {
return !isClosedBracket && (c == '$' || c == openBracket || Character.isJavaIdentifierPart(c) || c == '-');
}
public boolean isBracket(char c) {
return "(){}".contains(Character.toString(c)); //$NON-NLS-1$
}
}
public FunctionReferenceRule(IToken token, String startSeq, String endSeq) {
super(new TagDetector(endSeq));
this.startSeq = startSeq;
this.endSeq = endSeq;
for (String f : functions) {
addWord(startSeq + f, token);
addWord('$' + startSeq + f, token);
}
addWord(endSeq, token);
}
@Override
public IToken evaluate(ICharacterScanner scanner) {
TagDetector tagDetector = (TagDetector)fDetector;
int c = scanner.read();
if (c == tagDetector.closedBracket) {
if (tagDetector.bracketNesting > 0) {
tagDetector.bracketNesting--;
return fWords.get(endSeq);
}
return fDefaultToken;
}
if (c != ICharacterScanner.EOF && fDetector.isWordStart((char) c)) {
if (fColumn == UNDEFINED || (fColumn == scanner.getColumn() - 1)) {
fBuffer.setLength(0);
do {
fBuffer.append((char) c);
c = scanner.read();
} while (c != ICharacterScanner.EOF && fDetector.isWordPart((char) c));
scanner.unread();
String buffer= fBuffer.toString();
IToken token= fWords.get(buffer);
if (token != null) {
if (buffer.equals(startSeq + GNUMakefileConstants.FUNCTION_CALL) || buffer.equals('$' + startSeq + GNUMakefileConstants.FUNCTION_CALL)) {
if ((char)scanner.read() == ' ') {
do {
c = scanner.read();
} while (((TagDetector) fDetector).isBracket((char)c) || fDetector.isWordPart((char) c));
}
scanner.unread();
}
((TagDetector)fDetector).bracketNesting++;
return token;
}
if (fDefaultToken.isUndefined())
unreadBuffer(scanner);
return fDefaultToken;
}
}
scanner.unread();
return Token.UNDEFINED;
}
@Override
protected void unreadBuffer(ICharacterScanner scanner) {
for (int i= fBuffer.length() - 1; i >= 0; i--)
scanner.unread();
}
}