blob: b31da17c653ced3fb4415fbe1f2519ad996dd761 [file] [log] [blame]
/*
* Copyright 2003-2013 Dave Griffith, Bas Leijdekkers
*
* 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.siyeh.ig.controlflow;
import com.intellij.codeInspection.ui.SingleCheckboxOptionsPanel;
import com.intellij.psi.*;
import com.siyeh.HardcodedMethodConstants;
import com.siyeh.InspectionGadgetsBundle;
import com.siyeh.ig.BaseInspection;
import com.siyeh.ig.BaseInspectionVisitor;
import com.siyeh.ig.psiutils.TypeUtils;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import javax.swing.*;
public class ForLoopWithMissingComponentInspection extends BaseInspection {
@SuppressWarnings("PublicField")
public boolean ignoreCollectionLoops = false;
@Override
@NotNull
public String getDisplayName() {
return InspectionGadgetsBundle.message("for.loop.with.missing.component.display.name");
}
@Override
@NotNull
public String buildErrorString(Object... infos) {
final boolean hasInitializer = ((Boolean)infos[0]).booleanValue();
final boolean hasCondition = ((Boolean)infos[1]).booleanValue();
final boolean hasUpdate = ((Boolean)infos[2]).booleanValue();
if (hasInitializer) {
if (hasCondition) {
return InspectionGadgetsBundle.message("for.loop.with.missing.component.problem.descriptor3");
}
else if (hasUpdate) {
return InspectionGadgetsBundle.message("for.loop.with.missing.component.problem.descriptor2");
}
else {
return InspectionGadgetsBundle.message("for.loop.with.missing.component.problem.descriptor6");
}
}
else if (hasCondition) {
if (hasUpdate) {
return InspectionGadgetsBundle.message("for.loop.with.missing.component.problem.descriptor1");
}
else {
return InspectionGadgetsBundle.message("for.loop.with.missing.component.problem.descriptor5");
}
}
else if (hasUpdate) {
return InspectionGadgetsBundle.message("for.loop.with.missing.component.problem.descriptor4");
}
else {
return InspectionGadgetsBundle.message("for.loop.with.missing.component.problem.descriptor7");
}
}
@Override
@Nullable
public JComponent createOptionsPanel() {
return new SingleCheckboxOptionsPanel(InspectionGadgetsBundle.message("for.loop.with.missing.component.collection.loop.option"),
this, "ignoreCollectionLoops");
}
@Override
public BaseInspectionVisitor buildVisitor() {
return new ForLoopWithMissingComponentVisitor();
}
private class ForLoopWithMissingComponentVisitor extends BaseInspectionVisitor {
@Override
public void visitForStatement(@NotNull PsiForStatement statement) {
super.visitForStatement(statement);
final boolean hasCondition = hasCondition(statement);
final boolean hasInitializer = hasInitializer(statement);
final boolean hasUpdate = hasUpdate(statement);
if (hasCondition && hasInitializer && hasUpdate) {
return;
}
if (ignoreCollectionLoops && isCollectionLoopStatement(statement)) {
return;
}
registerStatementError(statement, Boolean.valueOf(hasInitializer), Boolean.valueOf(hasCondition), Boolean.valueOf(hasUpdate));
}
private boolean hasCondition(PsiForStatement statement) {
return statement.getCondition() != null;
}
private boolean hasInitializer(PsiForStatement statement) {
final PsiStatement initialization = statement.getInitialization();
return initialization != null && !(initialization instanceof PsiEmptyStatement);
}
private boolean hasUpdate(PsiForStatement statement) {
final PsiStatement update = statement.getUpdate();
return update != null && !(update instanceof PsiEmptyStatement);
}
private boolean isCollectionLoopStatement(PsiForStatement forStatement) {
final PsiStatement initialization = forStatement.getInitialization();
if (!(initialization instanceof PsiDeclarationStatement)) {
return false;
}
final PsiDeclarationStatement declaration = (PsiDeclarationStatement)initialization;
final PsiElement[] declaredElements = declaration.getDeclaredElements();
for (PsiElement declaredElement : declaredElements) {
if (!(declaredElement instanceof PsiVariable)) {
continue;
}
final PsiVariable variable = (PsiVariable)declaredElement;
if (!TypeUtils.variableHasTypeOrSubtype(variable, CommonClassNames.JAVA_UTIL_ITERATOR)) {
continue;
}
final PsiExpression condition = forStatement.getCondition();
if (isHasNext(condition, variable)) {
return true;
}
}
return false;
}
private boolean isHasNext(PsiExpression condition, PsiVariable iterator) {
if (condition instanceof PsiBinaryExpression) {
final PsiBinaryExpression binaryExpression = (PsiBinaryExpression)condition;
final PsiExpression lhs = binaryExpression.getLOperand();
final PsiExpression rhs = binaryExpression.getROperand();
return isHasNext(lhs, iterator) || isHasNext(rhs, iterator);
}
if (!(condition instanceof PsiMethodCallExpression)) {
return false;
}
final PsiMethodCallExpression call = (PsiMethodCallExpression)condition;
final PsiExpressionList argumentList = call.getArgumentList();
final PsiExpression[] arguments = argumentList.getExpressions();
if (arguments.length != 0) {
return false;
}
final PsiReferenceExpression methodExpression = call.getMethodExpression();
final String methodName = methodExpression.getReferenceName();
if (!HardcodedMethodConstants.HAS_NEXT.equals(methodName)) {
return false;
}
final PsiExpression qualifier = methodExpression.getQualifierExpression();
if (!(qualifier instanceof PsiReferenceExpression)) {
return true;
}
final PsiReferenceExpression referenceExpression = (PsiReferenceExpression)qualifier;
final PsiElement target = referenceExpression.resolve();
return iterator.equals(target);
}
}
}