blob: 5e74d9f25a00db94c7c03e293e2cab0bd380e71c [file] [log] [blame]
/* Copyright 2008 Edward Yakop.
*
* 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 org.qi4j.ide.plugin.idea.concerns.inspections;
import com.intellij.codeInspection.InspectionManager;
import com.intellij.codeInspection.ProblemDescriptor;
import com.intellij.openapi.project.Project;
import com.intellij.psi.PsiAnnotation;
import com.intellij.psi.PsiAnnotationMemberValue;
import com.intellij.psi.PsiClass;
import com.intellij.psi.PsiJavaCodeReferenceElement;
import com.intellij.psi.search.GlobalSearchScope;
import org.jetbrains.annotations.NotNull;
import org.qi4j.ide.plugin.idea.common.inspections.AbstractFix;
import org.qi4j.ide.plugin.idea.common.inspections.AbstractInspection;
import org.qi4j.ide.plugin.idea.common.resource.Qi4jResourceBundle;
import java.util.LinkedList;
import java.util.List;
import static com.intellij.codeInspection.ProblemHighlightType.GENERIC_ERROR_OR_WARNING;
import static org.qi4j.ide.plugin.idea.common.psi.search.GlobalSearchScopeUtil.determineSearchScope;
import static org.qi4j.ide.plugin.idea.common.resource.Qi4jResourceBundle.message;
import static org.qi4j.ide.plugin.idea.concerns.common.Qi4jConcernUtil.*;
/**
* @author edward.yakop@gmail.com
* @since 0.1
*/
public final class ConcernsAnnotationDeclaredCorrectlyInspection extends AbstractInspection
{
@NotNull
protected final String resourceBundlePrefixId()
{
return "concerns.annotation.declared.correctly";
}
@NotNull
public final String getShortName()
{
return "ConcernsAnnotationDeclaredCorrectlyInspection";
}
@Override
public final ProblemDescriptor[] checkClass( @NotNull PsiClass psiClass,
@NotNull InspectionManager manager,
boolean isOnTheFly )
{
// If class does not have @Concerns, ignore
PsiAnnotation concernsAnnotation = getConcernsAnnotation( psiClass );
if( concernsAnnotation == null )
{
return null;
}
// If @Concerns declared in class, suggest remove @Concerns annotation
if( !psiClass.isInterface() )
{
String message = message( "concerns.annotation.declared.correctly.error.annotation.declared.in.class" );
RemoveConcernsAnnotationFix fix = new RemoveConcernsAnnotationFix( concernsAnnotation );
ProblemDescriptor problemDescriptor = manager.createProblemDescriptor( concernsAnnotation, message, fix,
GENERIC_ERROR_OR_WARNING );
return new ProblemDescriptor[]{ problemDescriptor };
}
// If @Concerns annotation is empty, ignore
List<PsiAnnotationMemberValue> concernsAnnotationValue = getConcernsAnnotationValue( concernsAnnotation );
if( concernsAnnotationValue.isEmpty() )
{
return null;
}
// If ConcernOfClass is not resolved, ignore
Project project = psiClass.getProject();
GlobalSearchScope searchScope = determineSearchScope( psiClass );
PsiClass concernOfClass = getConcernOfClass( project, searchScope );
if( concernOfClass == null )
{
return null;
}
List<ProblemDescriptor> problems = new LinkedList<ProblemDescriptor>();
for( PsiAnnotationMemberValue concernClassAnnotationValue : concernsAnnotationValue )
{
PsiJavaCodeReferenceElement concernClassReference = getConcernClassReference( concernClassAnnotationValue );
// If it's not a class reference, ignore
if( concernClassReference == null )
{
continue;
}
// If class reference can't be resolved, ignore
PsiClass concernClass = (PsiClass) concernClassReference.resolve();
if( concernClass == null )
{
continue;
}
// If concern class does not inherit concern class, suggest remove that reference.
if( !concernClass.isInheritor( concernOfClass, true ) )
{
String message = Qi4jResourceBundle.message(
"concerns.annotation.declared.correctly.error.concern.class.does.not.extend.ConcernOf",
concernClass.getQualifiedName()
);
RemoveInvalidConcernClassReferenceFix fix = new RemoveInvalidConcernClassReferenceFix(
concernClassAnnotationValue, concernClassReference
);
ProblemDescriptor problemDescriptor = manager.createProblemDescriptor(
concernClassAnnotationValue, message, fix, GENERIC_ERROR_OR_WARNING );
problems.add( problemDescriptor );
}
else
{
// TODO: Test whether it is a generic concern
// TODO: Test whether it is a specific concern
}
}
return problems.toArray( new ProblemDescriptor[problems.size()] );
}
private static class RemoveConcernsAnnotationFix extends AbstractFix
{
private final PsiAnnotation annotationToRemove;
private RemoveConcernsAnnotationFix( @NotNull PsiAnnotation annotationToRemove )
{
super( message( "concerns.annotation.declared.correctly.fix.remove.annotation" ) );
this.annotationToRemove = annotationToRemove;
}
public final void applyFix( @NotNull Project project, @NotNull ProblemDescriptor descriptor )
{
annotationToRemove.delete();
}
}
private static class RemoveInvalidConcernClassReferenceFix extends AbstractFix
{
private final PsiAnnotationMemberValue concernClassAnnotationValue;
public RemoveInvalidConcernClassReferenceFix( @NotNull PsiAnnotationMemberValue annotationValueToRemove,
@NotNull PsiJavaCodeReferenceElement concernClassReference )
{
super( message( "concerns.annotation.declared.correctly.fix.remove.concern.class.reference",
concernClassReference.getQualifiedName() ) );
this.concernClassAnnotationValue = annotationValueToRemove;
}
public final void applyFix( @NotNull Project project, @NotNull ProblemDescriptor descriptor )
{
concernClassAnnotationValue.delete();
}
}
}