/* * Copyright 2000-2013 JetBrains s.r.o. * Copyright 2014-2014 AS3Boyan * Copyright 2014-2014 Elias Ku * * 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.intellij.plugins.haxe.ide.annotator; import com.intellij.openapi.progress.ProgressIndicatorProvider; import com.intellij.plugins.haxe.lang.lexer.HaxeTokenTypes; import com.intellij.plugins.haxe.lang.psi.*; import com.intellij.plugins.haxe.util.HaxeResolveUtil; import com.intellij.psi.PsiElement; import com.intellij.psi.util.PsiTreeUtil; import gnu.trove.THashSet; import org.jetbrains.annotations.NotNull; import java.util.Arrays; import java.util.Set; /** * @author: Fedor.Korotkov */ public abstract class HaxeAnnotatingVisitor extends HaxeVisitor { private static final Set<String> BUILTIN = new THashSet<String>(Arrays.asList( "trace", "__call__", "__vmem_set__", "__vmem_get__", "__vmem_sign__", "__global__", "_global", "__foreach__" )); @Override public void visitReferenceExpression(@NotNull HaxeReferenceExpression reference) { if (reference.getTokenType() != HaxeTokenTypes.REFERENCE_EXPRESSION) { return; // call, array access, this, literal, etc } final HaxeReference[] childReferences = PsiTreeUtil.getChildrenOfType(reference, HaxeReference.class); if (childReferences != null && childReferences.length > 1) { super.visitReferenceExpression(reference); return; } final HaxeReference leftSiblingReference = HaxeResolveUtil.getLeftReference(reference); final PsiElement referenceTarget = reference.resolve(); if (referenceTarget != null) { return; // OK } if (BUILTIN.contains(reference.getText()) && reference.getParent() instanceof HaxeCallExpression && !(reference.getParent().getParent() instanceof HaxeReference)) { return; } if (!(reference.getParent() instanceof HaxeReference) && !(reference.getParent() instanceof HaxePackageStatement) && !(reference.getParent() instanceof HaxeImportStatementWithWildcard)) { // whole reference expression handleUnresolvedReference(reference); } final PsiElement leftSiblingReferenceTarget = leftSiblingReference == null ? null : leftSiblingReference.resolve(); if (leftSiblingReference != null && leftSiblingReferenceTarget == null) { return; // already bad } // check all parents (ex. com.reference.Bar) PsiElement parent = reference.getParent(); while (parent instanceof HaxeReference) { if (((HaxeReference)parent).resolve() != null) return; parent = parent.getParent(); } if (parent instanceof HaxePackageStatement) { return; // package } if (parent instanceof HaxeImportStatementWithWildcard) { return; } handleUnresolvedReference(reference); } @Override public void visitElement(PsiElement element) { ProgressIndicatorProvider.checkCanceled(); element.acceptChildren(this); } abstract protected void handleUnresolvedReference(HaxeReferenceExpression reference); }