blob: 58e93fb5bf84f8840daf6c914a58b7629e3ad7ef [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2011 GK Software AG and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* Stephan Herrmann - initial API and implementation
*******************************************************************************/
package org.eclipse.objectteams.internal.jdt.nullity.quickfix;
import static org.eclipse.objectteams.internal.jdt.nullity.IConstants.IProblem.DefiniteNullFromNonNullMethod;
import static org.eclipse.objectteams.internal.jdt.nullity.IConstants.IProblem.PotentialNullFromNonNullMethod;
import java.util.ArrayList;
import java.util.Hashtable;
import java.util.List;
import java.util.Map;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.jdt.core.ICompilationUnit;
import org.eclipse.jdt.core.JavaCore;
import org.eclipse.jdt.core.compiler.IProblem;
import org.eclipse.jdt.core.dom.CompilationUnit;
import org.eclipse.jdt.internal.compiler.impl.CompilerOptions;
import org.eclipse.jdt.internal.ui.fix.AbstractMultiFix;
import org.eclipse.jdt.ui.cleanup.CleanUpRequirements;
import org.eclipse.jdt.ui.cleanup.ICleanUpFix;
import org.eclipse.jdt.ui.text.java.IProblemLocation;
import org.eclipse.objectteams.internal.jdt.nullity.NullCompilerOptions;
/**
* Cleanup for adding required null annotations.
*
* Crafted after the lead of Java50CleanUp
*
* @author stephan
*/
@SuppressWarnings({ "rawtypes", "restriction", "unchecked" })
public class NullAnnotationsCleanUp extends AbstractMultiFix {
private QuickFixes master;
public NullAnnotationsCleanUp(Map<String, String> options, QuickFixes quickFixes) {
super(options);
this.master = quickFixes;
}
/**
* {@inheritDoc}
*/
public CleanUpRequirements getRequirements() {
boolean requireAST= requireAST();
Map requiredOptions= requireAST ? getRequiredOptions() : null;
return new CleanUpRequirements(requireAST, false, false, requiredOptions);
}
private boolean requireAST() {
boolean addAnotations= isEnabled(CleanUpConstants.ADD_MISSING_ANNOTATIONS);
return addAnotations &&
( isEnabled(CleanUpConstants.ADD_DEFINITELY_MISSING_RETURN_ANNOTATION_NULLABLE)
|| isEnabled(CleanUpConstants.ADD_POTENTIALLY_MISSING_RETURN_ANNOTATION_NULLABLE)
|| isEnabled(CleanUpConstants.ADD_DEFINITELY_MISSING_PARAMETER_ANNOTATION_NULLABLE));
}
/**
* {@inheritDoc}
*/
protected ICleanUpFix createFix(CompilationUnit compilationUnit) throws CoreException {
return this.createFix(compilationUnit, null);
}
/**
* {@inheritDoc}
*/
protected ICleanUpFix createFix(CompilationUnit compilationUnit, IProblemLocation[] problems) throws CoreException {
if (compilationUnit == null)
return null;
boolean addAnnotations= isEnabled(CleanUpConstants.ADD_MISSING_ANNOTATIONS);
boolean addDefintelyMissingReturnNullable = isEnabled(CleanUpConstants.ADD_DEFINITELY_MISSING_RETURN_ANNOTATION_NULLABLE);
boolean addPotentiallyMissingReturnNullable = isEnabled(CleanUpConstants.ADD_POTENTIALLY_MISSING_RETURN_ANNOTATION_NULLABLE);
boolean addDefintelyMissingParamNullable = isEnabled(CleanUpConstants.ADD_DEFINITELY_MISSING_PARAMETER_ANNOTATION_NULLABLE);
return this.master.createCleanUp(compilationUnit,
addAnnotations && addDefintelyMissingReturnNullable,
addAnnotations && addPotentiallyMissingReturnNullable,
addAnnotations && addDefintelyMissingParamNullable,
problems);
}
private Map getRequiredOptions() {
Map result= new Hashtable();
if (isEnabled(CleanUpConstants.ADD_MISSING_ANNOTATIONS) &&
( isEnabled(CleanUpConstants.ADD_DEFINITELY_MISSING_RETURN_ANNOTATION_NULLABLE)
|| isEnabled(CleanUpConstants.ADD_POTENTIALLY_MISSING_RETURN_ANNOTATION_NULLABLE)
|| isEnabled(CleanUpConstants.ADD_DEFINITELY_MISSING_PARAMETER_ANNOTATION_NULLABLE)))
{
result.put(NullCompilerOptions.OPTION_ReportNullContractViolation, JavaCore.WARNING);
result.put(CompilerOptions.OPTION_ReportRedundantNullCheck, JavaCore.WARNING);
}
return result;
}
/**
* {@inheritDoc}
*/
public String[] getStepDescriptions() {
List result= new ArrayList();
if (isEnabled(CleanUpConstants.ADD_MISSING_ANNOTATIONS) &&
( isEnabled(CleanUpConstants.ADD_DEFINITELY_MISSING_RETURN_ANNOTATION_NULLABLE)
|| isEnabled(CleanUpConstants.ADD_POTENTIALLY_MISSING_RETURN_ANNOTATION_NULLABLE)
|| isEnabled(CleanUpConstants.ADD_DEFINITELY_MISSING_PARAMETER_ANNOTATION_NULLABLE)))
{
result.add(FixMessages.NullAnnotationsCleanUp_add_nullable_annotation);
}
return (String[])result.toArray(new String[result.size()]);
}
/**
* {@inheritDoc}
*/
public String getPreview() {
StringBuffer buf= new StringBuffer();
buf.append("class E {\n"); //$NON-NLS-1$
if (isEnabled(CleanUpConstants.ADD_MISSING_ANNOTATIONS) &&
( isEnabled(CleanUpConstants.ADD_DEFINITELY_MISSING_RETURN_ANNOTATION_NULLABLE)
|| isEnabled(CleanUpConstants.ADD_POTENTIALLY_MISSING_RETURN_ANNOTATION_NULLABLE)
|| isEnabled(CleanUpConstants.ADD_DEFINITELY_MISSING_PARAMETER_ANNOTATION_NULLABLE)))
{
buf.append(" @Nullable\n"); //$NON-NLS-1$
}
buf.append(" public Object foo() { return null; }\n"); //$NON-NLS-1$
buf.append("}\n"); //$NON-NLS-1$
return buf.toString();
}
/**
* {@inheritDoc}
*/
public boolean canFix(ICompilationUnit compilationUnit, IProblemLocation problem) {
int id= problem.getProblemId();
if (QuickFixes.isMissingNullableAnnotationProblem(id)) {
if ( isEnabled(CleanUpConstants.ADD_MISSING_ANNOTATIONS)
&& ( ((id == DefiniteNullFromNonNullMethod) && isEnabled(CleanUpConstants.ADD_DEFINITELY_MISSING_RETURN_ANNOTATION_NULLABLE))
|| ((id == PotentialNullFromNonNullMethod) && isEnabled(CleanUpConstants.ADD_POTENTIALLY_MISSING_RETURN_ANNOTATION_NULLABLE))
|| ((id == IProblem.NonNullLocalVariableComparisonYieldsFalse) && isEnabled(CleanUpConstants.ADD_DEFINITELY_MISSING_PARAMETER_ANNOTATION_NULLABLE))))
{
// FIXME search specifically: return param (which??)
// if (QuickFixes.hasExplicitNullnessAnnotation(compilationUnit, problem.getOffset()))
// return false;
return true;
}
}
return false;
}
/**
* {@inheritDoc}
*/
public int computeNumberOfFixes(CompilationUnit compilationUnit) {
int result= 0;
boolean addAnnotations= isEnabled(CleanUpConstants.ADD_MISSING_ANNOTATIONS);
boolean addDefinitelyMissingReturnNullable= addAnnotations && isEnabled(CleanUpConstants.ADD_DEFINITELY_MISSING_RETURN_ANNOTATION_NULLABLE);
boolean addPotentiallyMissingReturnNullable= addAnnotations && isEnabled(CleanUpConstants.ADD_POTENTIALLY_MISSING_RETURN_ANNOTATION_NULLABLE);
boolean addDefinitelyMissingParamNullable= addAnnotations && isEnabled(CleanUpConstants.ADD_DEFINITELY_MISSING_PARAMETER_ANNOTATION_NULLABLE);
IProblem[] problems= compilationUnit.getProblems();
for (int i= 0; i < problems.length; i++) {
int id= problems[i].getID();
if ( (addDefinitelyMissingReturnNullable && id == DefiniteNullFromNonNullMethod)
|| (addPotentiallyMissingReturnNullable && id == PotentialNullFromNonNullMethod)
|| (addDefinitelyMissingParamNullable && id == IProblem.NonNullLocalVariableComparisonYieldsFalse))
if (!QuickFixes.hasExplicitNullnessAnnotation((ICompilationUnit) compilationUnit.getJavaElement(), problems[i].getSourceStart()))
result++;
}
return result;
}
}