blob: 4ddd2240ae03e1960b5cbb49899bd66d4022356c [file] [log] [blame]
/*
* Copyright 2000-2013 JetBrains s.r.o.
*
* 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.jetbrains.python.codeInsight.highlighting;
import com.intellij.codeInsight.CodeInsightBundle;
import com.intellij.codeInsight.controlflow.ControlFlow;
import com.intellij.codeInsight.controlflow.Instruction;
import com.intellij.codeInsight.highlighting.HighlightUsagesHandler;
import com.intellij.codeInsight.highlighting.HighlightUsagesHandlerBase;
import com.intellij.featureStatistics.FeatureUsageTracker;
import com.intellij.featureStatistics.ProductivityFeatureNames;
import com.intellij.openapi.editor.Editor;
import com.intellij.psi.PsiElement;
import com.intellij.psi.PsiFile;
import com.intellij.psi.util.PsiTreeUtil;
import com.intellij.util.Consumer;
import com.jetbrains.python.codeInsight.controlflow.ControlFlowCache;
import com.jetbrains.python.psi.*;
import com.jetbrains.python.psi.impl.PyPsiUtils;
import org.jetbrains.annotations.Nullable;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
/**
* @author oleg
*/
public class PyHighlightExitPointsHandler extends HighlightUsagesHandlerBase<PsiElement> {
private final PsiElement myTarget;
public PyHighlightExitPointsHandler(final Editor editor, final PsiFile file, final PsiElement target) {
super(editor, file);
myTarget = target;
}
public List<PsiElement> getTargets() {
return Collections.singletonList(myTarget);
}
protected void selectTargets(final List<PsiElement> targets, final Consumer<List<PsiElement>> selectionConsumer) {
selectionConsumer.consume(targets);
}
public void computeUsages(final List<PsiElement> targets) {
FeatureUsageTracker.getInstance().triggerFeatureUsed(ProductivityFeatureNames.CODEASSISTS_HIGHLIGHT_RETURN);
final PsiElement parent = myTarget.getParent();
if (!(parent instanceof PyReturnStatement)) {
return;
}
final PyFunction function = PsiTreeUtil.getParentOfType(myTarget, PyFunction.class);
if (function == null) {
return;
}
highlightExitPoints((PyReturnStatement)parent, function);
}
@Nullable
private static PsiElement getExitTarget(PsiElement exitStatement) {
if (exitStatement instanceof PyReturnStatement) {
return PsiTreeUtil.getParentOfType(exitStatement, PyFunction.class);
}
else if (exitStatement instanceof PyBreakStatement) {
return ((PyBreakStatement)exitStatement).getLoopStatement();
}
else if (exitStatement instanceof PyContinueStatement) {
return ((PyContinueStatement)exitStatement).getLoopStatement();
}
else if (exitStatement instanceof PyRaiseStatement) {
// TODO[oleg]: Implement better logic here!
return null;
}
return null;
}
private void highlightExitPoints(final PyReturnStatement statement,
final PyFunction function) {
final ControlFlow flow = ControlFlowCache.getControlFlow(function);
final Collection<PsiElement> exitStatements = findExitPointsAndStatements(flow);
if (!exitStatements.contains(statement)) {
return;
}
final PsiElement originalTarget = getExitTarget(statement);
for (PsiElement exitStatement : exitStatements) {
if (getExitTarget(exitStatement) == originalTarget) {
addOccurrence(exitStatement);
}
}
myStatusText = CodeInsightBundle.message("status.bar.exit.points.highlighted.message",
exitStatements.size(),
HighlightUsagesHandler.getShortcutText());
}
private static Collection<PsiElement> findExitPointsAndStatements(final ControlFlow flow) {
final List<PsiElement> statements = new ArrayList<PsiElement>();
final Instruction[] instructions = flow.getInstructions();
for (Instruction instruction : instructions[instructions.length - 1].allPred()){
final PsiElement element = instruction.getElement();
if (element == null){
continue;
}
final PsiElement statement = PyPsiUtils.getStatement(element);
if (statement != null){
statements.add(statement);
}
}
return statements;
}
}