blob: 166245aaf8832579870d5cb396607d06c976c42f [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.inspections;
import com.google.common.collect.ImmutableList;
import com.intellij.codeInspection.LocalInspectionToolSession;
import com.intellij.codeInspection.ProblemsHolder;
import com.intellij.codeInspection.ui.MultipleCheckboxOptionsPanel;
import com.intellij.psi.PsiElementVisitor;
import com.jetbrains.python.PyBundle;
import com.jetbrains.python.PyTokenTypes;
import com.jetbrains.python.inspections.quickfix.SimplifyBooleanCheckQuickFix;
import com.jetbrains.python.psi.PyBinaryExpression;
import com.jetbrains.python.psi.PyConditionalStatementPart;
import com.jetbrains.python.psi.PyElementType;
import com.jetbrains.python.psi.PyExpression;
import org.jetbrains.annotations.Nls;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import javax.swing.*;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
/**
* @author Alexey.Ivanov
*/
public class PySimplifyBooleanCheckInspection extends PyInspection {
private static List<String> COMPARISON_LITERALS = ImmutableList.of("True", "False", "[]");
public boolean ignoreComparisonToZero = true;
@Nls
@NotNull
@Override
public String getDisplayName() {
return PyBundle.message("INSP.NAME.check.can.be.simplified");
}
@NotNull
@Override
public PsiElementVisitor buildVisitor(@NotNull ProblemsHolder holder,
boolean isOnTheFly,
@NotNull LocalInspectionToolSession session) {
return new Visitor(holder, session, ignoreComparisonToZero);
}
@Override
public JComponent createOptionsPanel() {
MultipleCheckboxOptionsPanel panel = new MultipleCheckboxOptionsPanel(this);
panel.addCheckbox("Ignore comparison to zero", "ignoreComparisonToZero");
return panel;
}
private static class Visitor extends PyInspectionVisitor {
private final boolean myIgnoreComparisonToZero;
public Visitor(@Nullable ProblemsHolder holder, @NotNull LocalInspectionToolSession session, boolean ignoreComparisonToZero) {
super(holder, session);
myIgnoreComparisonToZero = ignoreComparisonToZero;
}
@Override
public void visitPyConditionalStatementPart(PyConditionalStatementPart node) {
super.visitPyConditionalStatementPart(node);
final PyExpression condition = node.getCondition();
if (condition != null) {
condition.accept(new PyBinaryExpressionVisitor(getHolder(), getSession(), myIgnoreComparisonToZero));
}
}
}
private static class PyBinaryExpressionVisitor extends PyInspectionVisitor {
private final boolean myIgnoreComparisonToZero;
public PyBinaryExpressionVisitor(@Nullable ProblemsHolder holder,
@NotNull LocalInspectionToolSession session,
boolean ignoreComparisonToZero) {
super(holder, session);
myIgnoreComparisonToZero = ignoreComparisonToZero;
}
@Override
public void visitPyBinaryExpression(PyBinaryExpression node) {
super.visitPyBinaryExpression(node);
final PyElementType operator = node.getOperator();
final PyExpression rightExpression = node.getRightExpression();
if (rightExpression == null || rightExpression instanceof PyBinaryExpression ||
node.getLeftExpression() instanceof PyBinaryExpression) {
return;
}
if (PyTokenTypes.EQUALITY_OPERATIONS.contains(operator)) {
if (operandsEqualTo(node, COMPARISON_LITERALS) ||
(!myIgnoreComparisonToZero && operandsEqualTo(node, Collections.singleton("0")))) {
registerProblem(node);
}
}
}
private static boolean operandsEqualTo(@NotNull PyBinaryExpression expr, @NotNull Collection<String> literals) {
final String leftExpressionText = expr.getLeftExpression().getText();
final PyExpression rightExpression = expr.getRightExpression();
final String rightExpressionText = rightExpression != null ? rightExpression.getText() : null;
for (String literal : literals) {
if (literal.equals(leftExpressionText) || literal.equals(rightExpressionText)) {
return true;
}
}
return false;
}
private void registerProblem(PyBinaryExpression binaryExpression) {
registerProblem(binaryExpression, PyBundle.message("INSP.expression.can.be.simplified"), new SimplifyBooleanCheckQuickFix(binaryExpression));
}
}
}