blob: 01b694e22b5f69a2260f8c16541bb92f400962bd [file] [log] [blame]
/*
* Copyright 2003-2009 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.psi.*;
import com.intellij.psi.util.PsiTreeUtil;
import com.siyeh.InspectionGadgetsBundle;
import com.siyeh.ig.BaseInspection;
import com.siyeh.ig.BaseInspectionVisitor;
import com.siyeh.ig.psiutils.InitializationUtils;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
public class UnnecessaryDefaultInspection extends BaseInspection {
@Override
@NotNull
public String getDisplayName() {
return InspectionGadgetsBundle.message(
"unnecessary.default.display.name");
}
@Override
@NotNull
protected String buildErrorString(Object... infos) {
return InspectionGadgetsBundle.message(
"unnecessary.default.problem.descriptor");
}
@Override
public BaseInspectionVisitor buildVisitor() {
return new UnnecessaryDefaultVisitor();
}
private static class UnnecessaryDefaultVisitor
extends BaseInspectionVisitor {
@Override
public void visitSwitchStatement(
@NotNull PsiSwitchStatement statement) {
super.visitSwitchStatement(statement);
final PsiSwitchLabelStatement defaultStatement =
retrieveUnnecessaryDefault(statement);
if (defaultStatement == null) {
return;
}
PsiStatement nextStatement = PsiTreeUtil.getNextSiblingOfType(
defaultStatement, PsiStatement.class);
while (nextStatement != null &&
!(nextStatement instanceof PsiBreakStatement) &&
!(nextStatement instanceof PsiSwitchLabelStatement)) {
if (nextStatement instanceof PsiThrowStatement ||
isStatementNeededForInitializationOfVariable(statement,
nextStatement)) {
return;
}
nextStatement = PsiTreeUtil.getNextSiblingOfType(
nextStatement, PsiStatement.class);
}
registerStatementError(defaultStatement);
}
private static boolean isStatementNeededForInitializationOfVariable(
PsiSwitchStatement switchStatement, PsiStatement statement) {
if (!(statement instanceof PsiExpressionStatement)) {
return false;
}
final PsiExpressionStatement expressionStatement =
(PsiExpressionStatement)statement;
final PsiExpression expression =
expressionStatement.getExpression();
if (!(expression instanceof PsiAssignmentExpression)) {
return false;
}
final PsiAssignmentExpression assignmentExpression =
(PsiAssignmentExpression)expression;
final PsiExpression lhs = assignmentExpression.getLExpression();
if (!(lhs instanceof PsiReferenceExpression)) {
return false;
}
final PsiReferenceExpression referenceExpression =
(PsiReferenceExpression)lhs;
final PsiElement target = referenceExpression.resolve();
if (!(target instanceof PsiLocalVariable)) {
return false;
}
final PsiLocalVariable variable = (PsiLocalVariable)target;
return InitializationUtils.switchStatementAssignsVariableOrFails(
switchStatement, variable, true);
}
@Nullable
private static PsiSwitchLabelStatement retrieveUnnecessaryDefault(
PsiSwitchStatement statement) {
final PsiExpression expression = statement.getExpression();
if (expression == null) {
return null;
}
final PsiType type = expression.getType();
if (!(type instanceof PsiClassType)) {
return null;
}
final PsiClassType classType = (PsiClassType)type;
final PsiClass aClass = classType.resolve();
if (aClass == null || !aClass.isEnum()) {
return null;
}
final PsiCodeBlock body = statement.getBody();
if (body == null) {
return null;
}
final PsiStatement[] statements = body.getStatements();
int numCases = 0;
PsiSwitchLabelStatement result = null;
for (final PsiStatement child : statements) {
if (!(child instanceof PsiSwitchLabelStatement)) {
continue;
}
final PsiSwitchLabelStatement labelStatement =
(PsiSwitchLabelStatement)child;
if (labelStatement.isDefaultCase()) {
result = labelStatement;
}
else {
numCases++;
}
}
if (result == null) {
return null;
}
final PsiField[] fields = aClass.getFields();
int numEnums = 0;
for (final PsiField field : fields) {
final PsiType fieldType = field.getType();
if (fieldType.equals(type)) {
numEnums++;
}
}
if (numEnums != numCases) {
return null;
}
return result;
}
}
}