001 /*
002 * Copyright 2010-2013 JetBrains s.r.o.
003 *
004 * Licensed under the Apache License, Version 2.0 (the "License");
005 * you may not use this file except in compliance with the License.
006 * You may obtain a copy of the License at
007 *
008 * http://www.apache.org/licenses/LICENSE-2.0
009 *
010 * Unless required by applicable law or agreed to in writing, software
011 * distributed under the License is distributed on an "AS IS" BASIS,
012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013 * See the License for the specific language governing permissions and
014 * limitations under the License.
015 */
016
017 package org.jetbrains.jet.lang.diagnostics;
018
019 import com.intellij.lang.ASTNode;
020 import com.intellij.openapi.util.TextRange;
021 import com.intellij.psi.PsiElement;
022 import com.intellij.psi.PsiErrorElement;
023 import org.jetbrains.annotations.NotNull;
024
025 import java.util.Collections;
026 import java.util.List;
027
028 public class PositioningStrategy<E extends PsiElement> {
029 @NotNull
030 public List<TextRange> mark(@NotNull E element) {
031 return markElement(element);
032 }
033
034 public boolean isValid(@NotNull E element) {
035 return !hasSyntaxErrors(element);
036 }
037
038 @NotNull
039 protected static List<TextRange> markElement(@NotNull PsiElement element) {
040 return Collections.singletonList(element.getTextRange());
041 }
042
043 @NotNull
044 protected static List<TextRange> markNode(@NotNull ASTNode node) {
045 return Collections.singletonList(node.getTextRange());
046 }
047
048 @NotNull
049 protected static List<TextRange> markRange(@NotNull TextRange range) {
050 return Collections.singletonList(range);
051 }
052
053 protected static boolean hasSyntaxErrors(@NotNull PsiElement psiElement) {
054 if (psiElement instanceof PsiErrorElement) return true;
055
056 PsiElement lastChild = psiElement.getLastChild();
057 if (lastChild != null && hasSyntaxErrors(lastChild)) return true;
058
059 PsiElement[] children = psiElement.getChildren();
060 if (children.length > 0 && hasSyntaxErrors(children[children.length - 1])) return true;
061
062 return false;
063 }
064 }