Skip to content

Commit

Permalink
Merge remote branch into code-style-improvements
Browse files Browse the repository at this point in the history
  • Loading branch information
nettozahler committed Aug 7, 2024
2 parents 8f8e43d + 9da7de9 commit c045ce7
Show file tree
Hide file tree
Showing 7 changed files with 551 additions and 91 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@

import java.util.ArrayList;
import java.util.Collection;
import java.util.stream.Stream;

import org.eclipse.core.runtime.CoreException;

Expand Down Expand Up @@ -381,12 +382,20 @@ public void collectIncompatibleReturnTypeProposals(IInvocationContext context, I
ICompilationUnit cu= context.getCompilationUnit();
IMethodBinding methodDecl= methodDeclBinding.getMethodDeclaration();
ITypeBinding overriddenReturnType= overridden.getReturnType();
if (! JavaModelUtil.is50OrHigher(context.getCompilationUnit().getJavaProject())) {
overriddenReturnType= overriddenReturnType.getErasure();
// propose erasure
if (decl.typeParameters().isEmpty() || overriddenReturnType.getTypeBounds().length == 0 || Stream.of(overriddenReturnType.getTypeBounds()).allMatch(bound -> bound.getTypeArguments().length == 0)) {
T p1= createChangeIncompatibleReturnTypeProposal(cu, methodDecl, astRoot, overriddenReturnType.getErasure(), false, IProposalRelevance.CHANGE_RETURN_TYPE);
if (p1 != null)
proposals.add(p1);
}

// propose using (and potentially introducing) the type variable
if (overriddenReturnType.isTypeVariable()) {
T p2 = createChangeIncompatibleReturnTypeProposal(cu, methodDecl, astRoot, overriddenReturnType, false, IProposalRelevance.CHANGE_RETURN_TYPE);
if (p2 != null) {
proposals.add(p2);
}
}
T p1= createChangeIncompatibleReturnTypeProposal(cu, methodDecl, astRoot, overriddenReturnType, false, IProposalRelevance.CHANGE_RETURN_TYPE);
if (p1 != null)
proposals.add(p1);

ICompilationUnit targetCu= cu;

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -61,7 +61,6 @@
import org.eclipse.jdt.core.dom.ArrayCreation;
import org.eclipse.jdt.core.dom.ArrayInitializer;
import org.eclipse.jdt.core.dom.ArrayType;
import org.eclipse.jdt.core.dom.Assignment;
import org.eclipse.jdt.core.dom.BodyDeclaration;
import org.eclipse.jdt.core.dom.CastExpression;
import org.eclipse.jdt.core.dom.CompilationUnit;
Expand All @@ -81,7 +80,6 @@
import org.eclipse.jdt.core.dom.ParenthesizedExpression;
import org.eclipse.jdt.core.dom.QualifiedName;
import org.eclipse.jdt.core.dom.SimpleName;
import org.eclipse.jdt.core.dom.SingleVariableDeclaration;
import org.eclipse.jdt.core.dom.StructuralPropertyDescriptor;
import org.eclipse.jdt.core.dom.SuperMethodReference;
import org.eclipse.jdt.core.dom.Type;
Expand All @@ -92,7 +90,6 @@
import org.eclipse.jdt.core.dom.rewrite.ASTRewrite;
import org.eclipse.jdt.core.dom.rewrite.ImportRewrite.ImportRewriteContext;
import org.eclipse.jdt.core.dom.rewrite.ImportRewrite.TypeLocation;
import org.eclipse.jdt.core.dom.rewrite.ListRewrite;
import org.eclipse.jdt.core.manipulation.ImportReferencesCollector;
import org.eclipse.jdt.core.refactoring.CompilationUnitChange;
import org.eclipse.jdt.core.refactoring.IJavaRefactorings;
Expand Down Expand Up @@ -284,32 +281,33 @@ private void perform(Expression initializer) {
addExplicitTypeArgumentsIfNecessary(initializer);
}

private void addExplicitTypeArgumentsIfNecessary(Expression invocation) {
if (Invocations.isResolvedTypeInferredFromExpectedType(invocation)) {
ASTNode referenceContext= fNewLocation.getParent();
if (!(referenceContext instanceof VariableDeclarationFragment)
&& !(referenceContext instanceof SingleVariableDeclaration)
&& !(referenceContext instanceof Assignment)) {
ListRewrite typeArgsRewrite= Invocations.getInferredTypeArgumentsRewrite(fInitializerRewrite, invocation);
for (ITypeBinding typeArgument2 : Invocations.getInferredTypeArguments(invocation)) {
Type typeArgument= fNewLocationCuRewrite.getImportRewrite().addImport(typeArgument2, fNewLocationCuRewrite.getAST(), fNewLocationContext, TypeLocation.TYPE_ARGUMENT);
fNewLocationCuRewrite.getImportRemover().registerAddedImports(typeArgument);
typeArgsRewrite.insertLast(typeArgument, null);
}

if (invocation instanceof MethodInvocation) {
MethodInvocation methodInvocation= (MethodInvocation) invocation;
Expression expression= methodInvocation.getExpression();
if (expression == null) {
IMethodBinding methodBinding= methodInvocation.resolveMethodBinding();
if (methodBinding != null) {
expression= fNewLocationCuRewrite.getAST().newName(fNewLocationCuRewrite.getImportRewrite().addImport(methodBinding.getDeclaringClass().getTypeDeclaration(), fNewLocationContext));
fInitializerRewrite.set(invocation, MethodInvocation.EXPRESSION_PROPERTY, expression, null);
}
}
}
}
}
private void addExplicitTypeArgumentsIfNecessary(@SuppressWarnings("unused") Expression invocation) {
// TODO: this requires additional logic, for example, mark these locations and recompile new source
// if (Invocations.isResolvedTypeInferredFromExpectedType(invocation)) {
// ASTNode referenceContext= fNewLocation.getParent();
// if (!(referenceContext instanceof VariableDeclarationFragment)
// && !(referenceContext instanceof SingleVariableDeclaration)
// && !(referenceContext instanceof Assignment)) {
// ListRewrite typeArgsRewrite= Invocations.getInferredTypeArgumentsRewrite(fInitializerRewrite, invocation);
// for (ITypeBinding typeArgument2 : Invocations.getInferredTypeArguments(invocation)) {
// Type typeArgument= fNewLocationCuRewrite.getImportRewrite().addImport(typeArgument2, fNewLocationCuRewrite.getAST(), fNewLocationContext, TypeLocation.TYPE_ARGUMENT);
// fNewLocationCuRewrite.getImportRemover().registerAddedImports(typeArgument);
// typeArgsRewrite.insertLast(typeArgument, null);
// }
//
// if (invocation instanceof MethodInvocation) {
// MethodInvocation methodInvocation= (MethodInvocation) invocation;
// Expression expression= methodInvocation.getExpression();
// if (expression == null) {
// IMethodBinding methodBinding= methodInvocation.resolveMethodBinding();
// if (methodBinding != null) {
// expression= fNewLocationCuRewrite.getAST().newName(fNewLocationCuRewrite.getImportRewrite().addImport(methodBinding.getDeclaringClass().getTypeDeclaration(), fNewLocationContext));
// fInitializerRewrite.set(invocation, MethodInvocation.EXPRESSION_PROPERTY, expression, null);
// }
// }
// }
// }
// }
}

@Override
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -63,7 +63,6 @@
import org.eclipse.jdt.core.dom.ArrayCreation;
import org.eclipse.jdt.core.dom.ArrayInitializer;
import org.eclipse.jdt.core.dom.ArrayType;
import org.eclipse.jdt.core.dom.Assignment;
import org.eclipse.jdt.core.dom.CastExpression;
import org.eclipse.jdt.core.dom.CatchClause;
import org.eclipse.jdt.core.dom.CompilationUnit;
Expand All @@ -82,13 +81,11 @@
import org.eclipse.jdt.core.dom.ParenthesizedExpression;
import org.eclipse.jdt.core.dom.QualifiedName;
import org.eclipse.jdt.core.dom.SimpleName;
import org.eclipse.jdt.core.dom.SingleVariableDeclaration;
import org.eclipse.jdt.core.dom.TryStatement;
import org.eclipse.jdt.core.dom.Type;
import org.eclipse.jdt.core.dom.TypeDeclaration;
import org.eclipse.jdt.core.dom.VariableDeclaration;
import org.eclipse.jdt.core.dom.VariableDeclarationExpression;
import org.eclipse.jdt.core.dom.VariableDeclarationFragment;
import org.eclipse.jdt.core.dom.VariableDeclarationStatement;
import org.eclipse.jdt.core.dom.rewrite.ASTRewrite;
import org.eclipse.jdt.core.dom.rewrite.ImportRewrite.ImportRewriteContext;
Expand Down Expand Up @@ -417,6 +414,7 @@ private Expression getInitializerSource(CompilationUnitRewrite rewrite, SimpleNa
return copy;
}

@SuppressWarnings("unused")
private Expression getModifiedInitializerSource(CompilationUnitRewrite rewrite, SimpleName reference, TextEditGroup groupDesc) throws JavaModelException {
VariableDeclaration varDecl= getVariableDeclaration();
Expression initializer= varDecl.getInitializer();
Expand All @@ -427,18 +425,19 @@ private Expression getModifiedInitializerSource(CompilationUnitRewrite rewrite,
return replaceClashingNames(rewrite, groupDesc, initializer, replacements);
}
}
ASTNode referenceContext= reference.getParent();
if (Invocations.isResolvedTypeInferredFromExpectedType(initializer)) {
if (!(referenceContext instanceof VariableDeclarationFragment)
&& !(referenceContext instanceof SingleVariableDeclaration)
&& !(referenceContext instanceof Assignment)) {
ITypeBinding[] typeArguments= Invocations.getInferredTypeArguments(initializer);
if (typeArguments != null) {
String newSource= createParameterizedInvocation(initializer, typeArguments, rewrite);
return (Expression) rewrite.getASTRewrite().createStringPlaceholder(newSource, initializer.getNodeType());
}
}
}
// TODO: rework casting logic to mark the location and do casting after testing new source for failure
// ASTNode referenceContext= reference.getParent();
// if (Invocations.isResolvedTypeInferredFromExpectedType(initializer)) {
// if (!(referenceContext instanceof VariableDeclarationFragment)
// && !(referenceContext instanceof SingleVariableDeclaration)
// && !(referenceContext instanceof Assignment)) {
// ITypeBinding[] typeArguments= Invocations.getInferredTypeArguments(initializer);
// if (typeArguments != null) {
// String newSource= createParameterizedInvocation(initializer, typeArguments, rewrite);
// return (Expression) rewrite.getASTRewrite().createStringPlaceholder(newSource, initializer.getNodeType());
// }
// }
// }

Expression copy= (Expression) rewrite.getASTRewrite().createCopyTarget(initializer);
AST ast= rewrite.getAST();
Expand Down Expand Up @@ -706,6 +705,7 @@ private Name createFullyQualifiedName(SimpleName simpleName, ITypeBinding declar
return ast.newName(qualifiedName.toString());
}

@SuppressWarnings("unused")
private String createParameterizedInvocation(Expression invocation, ITypeBinding[] typeArguments, CompilationUnitRewrite cuRewrite) throws JavaModelException {
ASTRewrite rewrite= ASTRewrite.create(invocation.getAST());
ListRewrite typeArgsRewrite= Invocations.getInferredTypeArgumentsRewrite(rewrite, invocation);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,12 +17,15 @@
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Comparator;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.stream.Stream;

import org.eclipse.core.runtime.Assert;
import org.eclipse.core.runtime.CoreException;
Expand Down Expand Up @@ -54,13 +57,17 @@
import org.eclipse.jdt.core.dom.Modifier;
import org.eclipse.jdt.core.dom.ParameterizedType;
import org.eclipse.jdt.core.dom.PrimitiveType;
import org.eclipse.jdt.core.dom.SimpleName;
import org.eclipse.jdt.core.dom.SimpleType;
import org.eclipse.jdt.core.dom.SingleVariableDeclaration;
import org.eclipse.jdt.core.dom.TagElement;
import org.eclipse.jdt.core.dom.TextElement;
import org.eclipse.jdt.core.dom.Type;
import org.eclipse.jdt.core.dom.TypeParameter;
import org.eclipse.jdt.core.dom.VariableDeclarationExpression;
import org.eclipse.jdt.core.dom.VariableDeclarationFragment;
import org.eclipse.jdt.core.dom.VariableDeclarationStatement;
import org.eclipse.jdt.core.dom.WildcardType;
import org.eclipse.jdt.core.dom.rewrite.ASTRewrite;
import org.eclipse.jdt.core.dom.rewrite.ImportRewrite;
import org.eclipse.jdt.core.dom.rewrite.ImportRewrite.ImportRewriteContext;
Expand Down Expand Up @@ -273,9 +280,49 @@ protected ASTRewrite getRewrite() throws CoreException {
if (declNode instanceof MethodDeclaration) {
MethodDeclaration methodDecl= (MethodDeclaration) declNode;
Type origReturnType= methodDecl.getReturnType2();

if (fNewType.isTypeVariable()) {
IMethodBinding realMethodBinding= fNewType.getDeclaringMethod();
ITypeBinding[] typeParameters= realMethodBinding.getTypeParameters();

if (!methodDecl.typeParameters().isEmpty()) {
Map<String, String> typeParamNameMap= new HashMap<>();
for (int i = 0; i < methodDecl.typeParameters().size(); i++) {
typeParamNameMap.put(typeParameters[i].getName(), ((List<TypeParameter>) methodDecl.typeParameters()).get(i).getName().toString());
}
String existingTypeVarIdent= typeParamNameMap.get(fNewType.getName());
type= ast.newSimpleType(ast.newSimpleName(existingTypeVarIdent));
} else {
// add type parameters as they appear in the parent method
// notably, if you add the type variables, you must add ALL of them.
// eg. if you have <T, U> T myMethod(Class<U> clazz)
// you cannot override it with <T> T myMethod(Class<String> clazz)
ListRewrite typeParameterRewrite= rewrite.getListRewrite(methodDecl, MethodDeclaration.TYPE_PARAMETERS_PROPERTY);
for (ITypeBinding parameter : typeParameters) {
TypeParameter newTypeParameter= ast.newTypeParameter();
SimpleName newTypeParameterName= ast.newSimpleName(parameter.getName());
newTypeParameter.setName(newTypeParameterName);
Stream.of(parameter.getTypeBounds()) //
.forEach(bound -> {
newTypeParameter.typeBounds().add(getTypeNodeFromBinding(bound, ast, imports, context));
});
typeParameterRewrite.insertLast(newTypeParameter, null);
}

// Update the parameter types to match that of the resolved method.
// Some of the existing parameter types may be raw instead of containing the expected type parameters.
// Without inserting the type parameters in these cases, the signature will no longer match the overridden type.
for (int i = 0 ; i < methodDecl.parameters().size(); i++) {
SingleVariableDeclaration svd= (SingleVariableDeclaration)methodDecl.parameters().get(i);
rewrite.set(svd, SingleVariableDeclaration.TYPE_PROPERTY, getTypeNodeFromBinding(realMethodBinding.getParameterTypes()[i], ast, imports, context), null);
}
}
}

rewrite.set(methodDecl, MethodDeclaration.RETURN_TYPE2_PROPERTY, type, null);
DimensionRewrite.removeAllChildren(methodDecl, MethodDeclaration.EXTRA_DIMENSIONS2_PROPERTY, rewrite, null);
TypeAnnotationRewrite.removePureTypeAnnotations(methodDecl, MethodDeclaration.MODIFIERS2_PROPERTY, rewrite, null);

// add javadoc tag
Javadoc javadoc= methodDecl.getJavadoc();
if (javadoc != null && origReturnType != null && origReturnType.isPrimitiveType()
Expand Down Expand Up @@ -526,4 +573,43 @@ private void handledInferredParametrizedType(ASTNode node, ASTNode declaringNode
}
}

private Type getTypeNodeFromBinding(ITypeBinding typeBinding, AST ast, ImportRewrite importRewrite, ImportRewriteContext context) {

if (typeBinding.isWildcardType()) {
WildcardType wildcardType = ast.newWildcardType();
ITypeBinding bound = typeBinding.getBound();
if (bound != null) {
Type boundNode = getTypeNodeFromBinding(bound, ast, importRewrite, context);
wildcardType.setBound(boundNode);
wildcardType.setUpperBound(typeBinding.isUpperbound());
}
return wildcardType;
}

if (typeBinding.isArray()) {
Type elementTypeNode = getTypeNodeFromBinding(typeBinding.getElementType(), ast, importRewrite, context);
return ast.newArrayType(elementTypeNode, typeBinding.getDimensions());
}

if (typeBinding.isTypeVariable()) {
return ast.newSimpleType(ast.newSimpleName(typeBinding.getName()));
}

// import the simple/parameterized type if needed
importRewrite.addImport(typeBinding, ast, context, fTypeLocation);

// create the simple/parameterized
SimpleName simpleName = ast.newSimpleName(typeBinding.getErasure().getName());
SimpleType simpleType = ast.newSimpleType(simpleName);
if (typeBinding.isParameterizedType()) {
ParameterizedType parameterizedType = ast.newParameterizedType(simpleType);
for (ITypeBinding argument : typeBinding.getTypeArguments()) {
Type typeArgument = getTypeNodeFromBinding(argument, ast, importRewrite, context);
parameterizedType.typeArguments().add(typeArgument);
}
return parameterizedType;
}
return simpleType;
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -290,7 +290,7 @@ public void test30() throws Exception{
}

@Test
@Ignore("https://github.com/eclipse-jdt/eclipse.jdt.ui/issues/1552")
@Ignore //https://github.com/eclipse-jdt/eclipse.jdt.ui/issues/1565
public void test31() throws Exception{
helper1(8, 30, 8, 30);
}
Expand Down
Loading

0 comments on commit c045ce7

Please sign in to comment.