/*
* Copyright 2011-present Greg Shrago
*
* 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 org.intellij.grammar.editor;
import com.intellij.codeInsight.intention.IntentionAction;
import com.intellij.lang.annotation.Annotation;
import com.intellij.lang.annotation.AnnotationHolder;
import com.intellij.lang.annotation.Annotator;
import com.intellij.openapi.editor.markup.TextAttributes;
import com.intellij.openapi.project.DumbAware;
import com.intellij.openapi.util.TextRange;
import com.intellij.openapi.util.text.StringUtil;
import com.intellij.psi.*;
import com.intellij.psi.util.PsiTreeUtil;
import com.intellij.util.ObjectUtils;
import org.intellij.grammar.KnownAttribute;
import org.intellij.grammar.generator.BnfConstants;
import org.intellij.grammar.generator.ParserGeneratorUtil;
import org.intellij.grammar.generator.RuleGraphHelper;
import org.intellij.grammar.java.JavaHelper;
import org.intellij.grammar.psi.*;
import org.intellij.grammar.psi.impl.BnfRefOrTokenImpl;
import org.intellij.grammar.psi.impl.GrammarUtil;
import org.jetbrains.annotations.NotNull;
import java.util.StringTokenizer;
/**
* @author gregsh
*/
public class BnfAnnotator implements Annotator, DumbAware {
@Override
public void annotate(@NotNull PsiElement psiElement, @NotNull AnnotationHolder annotationHolder) {
PsiElement parent = psiElement.getParent();
if (parent instanceof BnfRule && ((BnfRule)parent).getId() == psiElement) {
addRuleHighlighting((BnfRule)parent, psiElement, annotationHolder);
}
else if (parent instanceof BnfAttr && ((BnfAttr)parent).getId() == psiElement) {
annotationHolder.createInfoAnnotation(psiElement, null).setTextAttributes(BnfSyntaxHighlighter.ATTRIBUTE);
}
else if (parent instanceof BnfModifier) {
annotationHolder.createInfoAnnotation(psiElement, null).setTextAttributes(BnfSyntaxHighlighter.KEYWORD);
}
else if (parent instanceof BnfListEntry && ((BnfListEntry)parent).getId() == psiElement) {
boolean hasValue = ((BnfListEntry)parent).getLiteralExpression() != null;
BnfAttr attr = PsiTreeUtil.getParentOfType(parent, BnfAttr.class);
KnownAttribute attribute = attr != null ? KnownAttribute.getCompatibleAttribute(attr.getName()) : null;
if (attribute == KnownAttribute.METHODS && !hasValue) {
PsiReference reference = parent.findReferenceAt(psiElement.getStartOffsetInParent());
PsiElement resolve = reference == null ? null : reference.resolve();
if (resolve == null) {
annotationHolder.createWarningAnnotation(psiElement, "Unresolved method reference");
}
else {
annotationHolder.createInfoAnnotation(psiElement, null).setTextAttributes(BnfSyntaxHighlighter.EXTERNAL);
}
}
}
else if (psiElement instanceof BnfReferenceOrToken) {
if (parent instanceof BnfAttr) {
String text = psiElement.getText();
if (text.equals("true") || text.equals("false")) {
annotationHolder.createInfoAnnotation(psiElement, null).setTextAttributes(BnfSyntaxHighlighter.KEYWORD);
return;
}
}
PsiReference reference = psiElement.getReference();
Object resolve = reference == null ? null : reference.resolve();
if (resolve instanceof BnfRule) {
addRuleHighlighting((BnfRule)resolve, psiElement, annotationHolder);
}
else if (resolve instanceof BnfAttr) {
annotationHolder.createInfoAnnotation(psiElement, null).setTextAttributes(BnfSyntaxHighlighter.ATTRIBUTE);
}
else if (resolve == null && parent instanceof BnfAttr) {
annotationHolder.createWarningAnnotation(psiElement, "Unresolved rule reference");
}
else if (GrammarUtil.isExternalReference(psiElement)) {
if (resolve == null && parent instanceof BnfExternalExpression && ((BnfExternalExpression)parent).getExpressionList().size() == 1 &&
ParserGeneratorUtil.Rule.isMeta(ParserGeneratorUtil.Rule.of((BnfRefOrTokenImpl)psiElement))) {
annotationHolder.createInfoAnnotation(parent, null).setTextAttributes(BnfSyntaxHighlighter.META_PARAM);
}
else {
annotationHolder.createInfoAnnotation(psiElement, null).setTextAttributes(BnfSyntaxHighlighter.EXTERNAL);
}
}
else if (resolve == null) {
annotationHolder.createInfoAnnotation(psiElement, null).setTextAttributes(BnfSyntaxHighlighter.TOKEN);
}
}
else if (psiElement instanceof BnfStringLiteralExpression) {
if (parent instanceof BnfAttrPattern || parent instanceof BnfAttr || parent instanceof BnfListEntry) {
annotationHolder.createInfoAnnotation(psiElement, null).setEnforcedTextAttributes(TextAttributes.ERASE_MARKER);
annotationHolder.createInfoAnnotation(psiElement, null).setTextAttributes(BnfSyntaxHighlighter.PATTERN);
}
if (parent instanceof BnfAttrPattern) {
PsiReference reference = psiElement.getReference();
if (reference instanceof PsiPolyVariantReference && ((PsiPolyVariantReference)reference).multiResolve(false).length == 0) {
annotationHolder.createWarningAnnotation(psiElement, "Pattern doesn't match any rule");
}
}
else if (parent instanceof BnfAttr || parent instanceof BnfListEntry) {
final String attrName = ObjectUtils.assertNotNull(PsiTreeUtil.getParentOfType(psiElement, BnfAttr.class)).getName();
KnownAttribute attribute = KnownAttribute.getCompatibleAttribute(attrName);
if (attribute != null) {
String value = (String)ParserGeneratorUtil.getAttributeValue((BnfExpression)psiElement);
Object resolve;
String refType = "";
JavaHelper javaHelper = JavaHelper.getJavaHelper(psiElement);
if (attribute.getName().endsWith("Class") || attribute == KnownAttribute.MIXIN) {
resolve = checkJavaResolve(value, javaHelper);
refType = "class ";
}
else if (attribute.getName().endsWith("Package")) {
resolve = javaHelper.findPackage(value);
refType = "package ";
}
else if (attribute.getName().endsWith("Factory")) {
resolve = Boolean.TRUE; // todo
}
else if (attribute == KnownAttribute.EXTENDS || attribute == KnownAttribute.IMPLEMENTS) {
resolve = value.contains(".") ? checkJavaResolve(value, javaHelper) :
((BnfFile)psiElement.getContainingFile()).getRule(value);
refType = "rule or class ";
}
else if (attribute == KnownAttribute.ELEMENT_TYPE || attribute == KnownAttribute.NAME) {
resolve = ObjectUtils.chooseNotNull(((BnfFile)psiElement.getContainingFile()).getRule(value), Boolean.TRUE);
refType = "rule or constant ";
}
else if (attribute == KnownAttribute.RECOVER_WHILE && !BnfConstants.RECOVER_AUTO.equals(value)) {
if (GrammarUtil.isDoubleAngles(value) &&
ParserGeneratorUtil.Rule.isMeta(ParserGeneratorUtil.Rule.of((BnfStringLiteralExpression)psiElement))) {
resolve = Boolean.TRUE;
}
else {
resolve = ((BnfFile)psiElement.getContainingFile()).getRule(value);
refType = "rule ";
}
}
else {
resolve = Boolean.TRUE;
}
TextRange range = ElementManipulators.getValueTextRange(psiElement).shiftRight(psiElement.getTextRange().getStartOffset());
if (resolve instanceof BnfRule) {
annotationHolder.createInfoAnnotation(range, null).setTextAttributes(BnfSyntaxHighlighter.RULE);
}
else if (resolve == null) {
Annotation annotation = annotationHolder.createWarningAnnotation(range, "Unresolved " + refType + "reference");
if (refType.contains("class")) {
boolean intf = attribute == KnownAttribute.IMPLEMENTS;
IntentionAction fix = JavaHelper.getJavaHelper(parent).getCreateClassQuickFix(parent, value, intf, null);
if (fix != null) annotation.registerFix(fix, null, null);
}
}
}
}
else {
String text = ParserGeneratorUtil.getLiteralValue((BnfStringLiteralExpression)psiElement);
if (!RuleGraphHelper.getTokenTextToNameMap((BnfFile)psiElement.getContainingFile()).containsKey(text)) {
String message = "Tokens matched by text are slower than tokens matched by types";
annotationHolder.createInfoAnnotation(psiElement, null).setEnforcedTextAttributes(TextAttributes.ERASE_MARKER);
annotationHolder.createInfoAnnotation(psiElement, message).setTextAttributes(BnfSyntaxHighlighter.PATTERN);
}
}
}
}
private static void addRuleHighlighting(BnfRule rule, PsiElement psiElement, AnnotationHolder annotationHolder) {
if (ParserGeneratorUtil.Rule.isMeta(rule)) {
annotationHolder.createInfoAnnotation(psiElement, null).setTextAttributes(BnfSyntaxHighlighter.META_RULE);
}
else {
annotationHolder.createInfoAnnotation(psiElement, null).setTextAttributes(BnfSyntaxHighlighter.RULE);
}
PsiFile file = rule.getContainingFile();
if (StringUtil.isNotEmpty(((BnfFile)file).findAttributeValue(rule, KnownAttribute.RECOVER_WHILE, null))) {
annotationHolder.createInfoAnnotation(psiElement, null).setTextAttributes(BnfSyntaxHighlighter.RECOVER_MARKER);
}
}
private static Object checkJavaResolve(String value, JavaHelper javaHelper) {
Object resolve = null;
for (String s : StringUtil.tokenize(new StringTokenizer(value, "<>,?", false))) {
resolve = javaHelper.findClass(s.trim());
if (resolve == null) break;
}
return resolve;
}
}