org.netbeans.spi.java.hints.JavaFixUtilities Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.netbeans.spi.java.hints;
import com.sun.source.doctree.DocCommentTree;
import com.sun.source.doctree.DocTree;
import com.sun.source.doctree.SinceTree;
import com.sun.source.tree.AnnotationTree;
import com.sun.source.tree.AssignmentTree;
import com.sun.source.tree.BinaryTree;
import com.sun.source.tree.BlockTree;
import com.sun.source.tree.CaseTree;
import com.sun.source.tree.CatchTree;
import com.sun.source.tree.ClassTree;
import com.sun.source.tree.CompilationUnitTree;
import com.sun.source.tree.CompoundAssignmentTree;
import com.sun.source.tree.DoWhileLoopTree;
import com.sun.source.tree.ExpressionStatementTree;
import com.sun.source.tree.ExpressionTree;
import com.sun.source.tree.IdentifierTree;
import com.sun.source.tree.IfTree;
import com.sun.source.tree.LambdaExpressionTree;
import com.sun.source.tree.LiteralTree;
import com.sun.source.tree.MemberSelectTree;
import com.sun.source.tree.MethodInvocationTree;
import com.sun.source.tree.MethodTree;
import com.sun.source.tree.ModifiersTree;
import com.sun.source.tree.NewArrayTree;
import com.sun.source.tree.NewClassTree;
import com.sun.source.tree.ParameterizedTypeTree;
import com.sun.source.tree.ParenthesizedTree;
import com.sun.source.tree.Scope;
import com.sun.source.tree.StatementTree;
import com.sun.source.tree.SwitchTree;
import com.sun.source.tree.Tree;
import com.sun.source.tree.Tree.Kind;
import com.sun.source.tree.TryTree;
import com.sun.source.tree.TypeParameterTree;
import com.sun.source.tree.UnaryTree;
import com.sun.source.tree.UnionTypeTree;
import com.sun.source.tree.VariableTree;
import com.sun.source.tree.WhileLoopTree;
import com.sun.source.util.TreePath;
import com.sun.source.util.TreePathScanner;
import com.sun.source.util.TreeScanner;
import org.netbeans.api.java.source.support.ErrorAwareTreePathScanner;
import org.netbeans.api.java.source.support.ErrorAwareTreeScanner;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.EnumMap;
import java.util.EnumSet;
import java.util.HashMap;
import java.util.HashSet;
import java.util.IdentityHashMap;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import java.util.concurrent.Callable;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.function.Supplier;
import java.util.logging.Level;
import java.util.logging.Logger;
import java.util.regex.Matcher;
import javax.lang.model.element.Element;
import javax.lang.model.element.ElementKind;
import javax.lang.model.element.Modifier;
import javax.lang.model.element.TypeElement;
import javax.lang.model.type.TypeKind;
import javax.lang.model.type.TypeMirror;
import org.netbeans.api.annotations.common.NonNull;
import org.netbeans.api.java.classpath.ClassPath;
import org.netbeans.api.java.classpath.ClassPath.PathConversionMode;
import org.netbeans.api.java.queries.SourceForBinaryQuery;
import org.netbeans.api.java.source.ClasspathInfo;
import org.netbeans.api.java.source.ClasspathInfo.PathKind;
import org.netbeans.api.java.source.CompilationInfo;
import org.netbeans.api.java.source.ElementHandle;
import org.netbeans.api.java.source.GeneratorUtilities;
import org.netbeans.api.java.source.SourceUtils;
import org.netbeans.api.java.source.TreeMaker;
import org.netbeans.api.java.source.TreePathHandle;
import org.netbeans.api.java.source.TreeUtilities;
import org.netbeans.api.java.source.TypeMirrorHandle;
import org.netbeans.api.java.source.WorkingCopy;
import org.netbeans.api.java.source.matching.Occurrence;
import org.netbeans.api.java.source.matching.Pattern;
import org.netbeans.api.project.FileOwnerQuery;
import org.netbeans.api.project.Project;
import org.netbeans.modules.java.hints.spiimpl.Hacks;
import org.netbeans.modules.java.hints.spiimpl.Utilities;
import org.netbeans.modules.java.hints.spiimpl.ipi.upgrade.ProjectDependencyUpgrader;
import org.netbeans.modules.refactoring.spi.SimpleRefactoringElementImplementation;
import org.netbeans.spi.editor.hints.Fix;
import org.netbeans.spi.java.classpath.support.ClassPathSupport;
import org.netbeans.spi.java.hints.JavaFix.TransformationContext;
import org.openide.filesystems.FileObject;
import org.openide.filesystems.FileUtil;
import org.openide.loaders.DataFolder;
import org.openide.loaders.DataObject;
import org.openide.modules.SpecificationVersion;
import org.openide.text.PositionBounds;
import org.openide.util.Exceptions;
import org.openide.util.Lookup;
import org.openide.util.NbBundle.Messages;
import org.openide.util.NbCollections;
/**Factory methods for various predefined {@link JavaFix} implementations.
*
* @author lahvac
*/
public class JavaFixUtilities {
/**Prepare a fix that will replace the given tree node ({@code what}) with the
* given code. Any variables in the {@code to} pattern will be replaced with their
* values from {@link HintContext#getVariables() }, {@link HintContext#getMultiVariables() }
* and {@link HintContext#getVariableNames() }.
*
* @param ctx basic context for which the fix should be created
* @param displayName the display name of the fix
* @param what the tree node that should be replaced
* @param to the new code that should replaced the {@code what} tree node
* @return an editor fix that performs the required transformation
*/
public static Fix rewriteFix(HintContext ctx, String displayName, TreePath what, final String to) {
return rewriteFix(ctx.getInfo(), displayName, what, to, ctx.getVariables(), ctx.getMultiVariables(), ctx.getVariableNames(), ctx.getConstraints(), Collections.emptyMap());
}
@SuppressWarnings("AssignmentToMethodParameter")
static Fix rewriteFix(CompilationInfo info, String displayName, TreePath what, final String to, Map parameters, Map> parametersMulti, final Map parameterNames, Map constraints, Map options, String... imports) {
final Map params = new HashMap<>();
final Map extraParamsData = new HashMap<>();
final Map> implicitThis = new HashMap<>();
for (Entry e : parameters.entrySet()) {
TreePath tp = e.getValue();
if (tp.getParentPath() != null && !immediateChildren(tp.getParentPath().getLeaf()).contains(tp.getLeaf())) {
Element el = info.getTrees().getElement(tp);
if (el != null && el.getSimpleName().contentEquals("this")) {
implicitThis.put(e.getKey(), ElementHandle.create(el.getEnclosingElement()));
continue;
}
}
params.put(e.getKey(), TreePathHandle.create(e.getValue(), info));
if (e.getValue() instanceof Callable callable) {
try {
extraParamsData.put(e.getKey(), callable.call());
} catch (Exception ex) {
Exceptions.printStackTrace(ex);
}
}
}
final Map> paramsMulti = new HashMap<>();
for (Entry> e : parametersMulti.entrySet()) {
Collection tph = new LinkedList<>();
for (TreePath tp : e.getValue()) {
TreePathHandle x = TreePathHandle.create(tp, info);
// resolve back, to save some problems later:
if (x.resolve(info) == null) {
continue;
}
tph.add(x);
}
paramsMulti.put(e.getKey(), tph);
}
final Map> constraintsHandles = new HashMap<>();
for (Entry c : constraints.entrySet()) {
constraintsHandles.put(c.getKey(), TypeMirrorHandle.create(c.getValue()));
}
Supplier lazyNamer;
if (displayName == null) {
lazyNamer = new Supplier() {
private String dn = null;
@Override public String get() {
if(dn == null) {
dn = defaultFixDisplayName(parameters, parametersMulti, to);
}
return dn;
}
};
} else {
lazyNamer = () -> displayName;
}
return new JavaFixRealImpl(info, what, options, lazyNamer, to, params, extraParamsData, implicitThis, paramsMulti, parameterNames, constraintsHandles, Arrays.asList(imports)).toEditorFix();
}
private static Set immediateChildren(Tree t) {
Set children = new HashSet<>();
t.accept(new TreeScanner() {
@Override
public Void scan(Tree tree, Void p) {
if (tree != null)
children.add(tree);
return null;
}
}, null);
return children;
}
/**Creates a fix that removes the given code corresponding to the given tree
* node from the source code.
*
* @param ctx basic context for which the fix should be created
* @param displayName the display name of the fix
* @param what the tree node that should be removed
* @return an editor fix that removes the give tree from the source code
*/
public static Fix removeFromParent(HintContext ctx, String displayName, TreePath what) {
return new RemoveFromParent(displayName, ctx.getInfo(), what, false).toEditorFix();
}
/**Creates a fix that removes the given code corresponding to the given tree
* node together with all its usages from the source code
*
* @param ctx basic context for which the fix should be created
* @param displayName the display name of the fix
* @param what the tree node that should be removed
* @return an editor fix that removes the give tree from the source code
*
* @since 1.48
*/
public static Fix safelyRemoveFromParent(HintContext ctx, String displayName, TreePath what) {
return RemoveFromParent.canSafelyRemove(ctx.getInfo(), what) ? new RemoveFromParent(displayName, ctx.getInfo(), what, true).toEditorFix() : null;
}
@SuppressWarnings("AssignmentToMethodParameter")
private static String defaultFixDisplayName(Map variables, Map> parametersMulti, String replaceTarget) {
Map stringsForVariables = new LinkedHashMap<>();
// replace multi vars first
for (Entry> e : parametersMulti.entrySet()) {
if (e.getKey().startsWith("$$")) {
continue;
}
if (e.getValue().isEmpty()) {
stringsForVariables.put(e.getKey()+";", ""); // could be a statement
stringsForVariables.put(", "+e.getKey(), ""); // or parameter
stringsForVariables.put(e.getKey(), "");
} else if (e.getValue().size() == 1) {
String text = treePathToString(e.getValue().iterator().next(), false, e.getKey());
stringsForVariables.put(e.getKey(), text);
} else {
// keep the variable in the text for more complex cases, but we have to escape it somehow
stringsForVariables.put(e.getKey(), e.getKey().replace("$", "♦"));
}
}
// regular vars next, longest first in case a var is a prefix of another var
variables.entrySet().stream()
.sorted((e1, e2) -> e2.getKey().length() - e1.getKey().length())
.forEach(e -> stringsForVariables.put(e.getKey(), treePathToString(e.getValue(), true, e.getKey())));
if (!stringsForVariables.containsKey("$this")) {
//XXX: is this correct?
stringsForVariables.put("$this", "this");
}
for (Entry e : stringsForVariables.entrySet()) {
replaceTarget = replaceTarget.replace(e.getKey(), e.getValue());
}
// cleanup and escape java code for html renderer
return "Rewrite to " + replaceTarget.replace("♦", "$")
.replace(";;", ";")
.replace("\n", " ")
.replaceAll("\\s{2,}", " ")
.replace("<", "<");
}
private static String treePathToString(TreePath tp, boolean preferVarName, String fallback) {
Tree leaf = tp.getLeaf();
if (preferVarName && leaf.getKind() == Kind.VARIABLE) {
return ((VariableTree) leaf).getName().toString();
}
String str = leaf.toString();
return str.equals("(ERROR)") ? fallback : str;
}
private static void checkDependency(CompilationInfo copy, Element e, boolean canShowUI) {
SpecificationVersion sv = computeSpecVersion(copy, e);
while (sv == null && e.getKind() != ElementKind.PACKAGE) {
e = e.getEnclosingElement();
sv = computeSpecVersion(copy, e);
}
if (sv == null) {
return ;
}
Project currentProject = FileOwnerQuery.getOwner(copy.getFileObject());
if (currentProject == null) {
return ;
}
FileObject file = getFile(copy, e);
if (file == null) {
return ;
}
FileObject root = findRootForFile(file, copy.getClasspathInfo());
if (root == null) {
return ;
}
Project referedProject = FileOwnerQuery.getOwner(file);
if (referedProject != null && currentProject.getProjectDirectory().equals(referedProject.getProjectDirectory())) {
return ;
}
for (ProjectDependencyUpgrader pdu : Lookup.getDefault().lookupAll(ProjectDependencyUpgrader.class)) {
if (pdu.ensureDependency(currentProject, root, sv, canShowUI)) {
return ;
}
}
}
private static final java.util.regex.Pattern SPEC_VERSION = java.util.regex.Pattern.compile("[0-9]+(\\.[0-9]+)+");
static SpecificationVersion computeSpecVersion(CompilationInfo info, Element el) {
if (!Utilities.isJavadocSupported(info)) return null;
DocCommentTree javaDoc = info.getDocTrees().getDocCommentTree(el);
if (javaDoc == null) return null;
for (DocTree tag : javaDoc.getBlockTags()) {
if (tag.getKind() != DocTree.Kind.SINCE) {
continue;
}
String text = ((SinceTree) tag).getBody().toString();
Matcher m = SPEC_VERSION.matcher(text);
if (!m.find()) {
continue;
}
return new SpecificationVersion(m.group()/*ver.toString()*/);
}
return null;
}
@SuppressWarnings("deprecation")
private static FileObject getFile(CompilationInfo copy, Element e) {
return SourceUtils.getFile(e, copy.getClasspathInfo());
}
private static FileObject findRootForFile(final FileObject file, final ClasspathInfo cpInfo) {
ClassPath cp = ClassPathSupport.createProxyClassPath(
new ClassPath[] {
cpInfo.getClassPath(ClasspathInfo.PathKind.SOURCE),
cpInfo.getClassPath(ClasspathInfo.PathKind.BOOT),
cpInfo.getClassPath(ClasspathInfo.PathKind.COMPILE),
});
FileObject root = cp.findOwnerRoot(file);
if (root != null) {
return root;
}
for (ClassPath.Entry e : cp.entries()) {
FileObject[] sourceRoots = SourceForBinaryQuery.findSourceRoots(e.getURL()).getRoots();
if (sourceRoots.length == 0) continue;
ClassPath sourcePath = ClassPathSupport.createClassPath(sourceRoots);
root = sourcePath.findOwnerRoot(file);
if (root != null) {
return root;
}
}
return null;
}
private static boolean isStaticElement(Element el) {
if (el == null) return false;
if (el.asType() == null || el.asType().getKind() == TypeKind.ERROR) {
return false;
}
if (el.getModifiers().contains(Modifier.STATIC)) {
//XXX:
if (!el.getKind().isClass() && !el.getKind().isInterface()) {
return false;
}
return true;
}
if (el.getKind().isClass() || el.getKind().isInterface()) {
return el.getEnclosingElement().getKind() == ElementKind.PACKAGE;
}
return false;
}
private static class JavaFixRealImpl extends JavaFix {
private final Supplier displayName;
private final Map params;
private final Map extraParamsData;
private final Map> implicitThis;
private final Map> paramsMulti;
private final Map parameterNames;
private final Map> constraintsHandles;
private final Iterable extends String> imports;
private final String to;
public JavaFixRealImpl(CompilationInfo info, TreePath what, Map options, Supplier displayName, String to, Map params, Map extraParamsData, Map> implicitThis, Map> paramsMulti, final Map parameterNames, Map> constraintsHandles, Iterable extends String> imports) {
super(info, what, options);
this.displayName = displayName;
this.to = to;
this.params = params;
this.extraParamsData = extraParamsData;
this.implicitThis = implicitThis;
this.paramsMulti = paramsMulti;
this.parameterNames = parameterNames;
this.constraintsHandles = constraintsHandles;
this.imports = imports;
}
@Override
protected String getText() {
return displayName.get();
}
@Override
@SuppressWarnings("NestedAssignment")
protected void performRewrite(TransformationContext ctx) {
final WorkingCopy wc = ctx.getWorkingCopy();
TreePath tp = ctx.getPath();
final GeneratorUtilities gen = GeneratorUtilities.get(wc);
tp = new TreePath(tp.getParentPath(), gen.importComments(tp.getLeaf(), tp.getCompilationUnit()));
final Map parameters = new HashMap<>();
for (Entry e : params.entrySet()) {
TreePath p = e.getValue().resolve(wc);
if (p == null) {
Logger.getLogger(JavaFix.class.getName()).log(Level.SEVERE, "Cannot resolve handle={0}", e.getValue());
}
parameters.put(e.getKey(), p);
}
Map implicitThis = new HashMap<>();
for (Entry> e : this.implicitThis.entrySet()) {
Element clazz = e.getValue().resolve(wc);
if (clazz == null) {
Logger.getLogger(JavaFix.class.getName()).log(Level.SEVERE, "Cannot resolve handle={0}", e.getValue());
continue;
}
implicitThis.put(e.getKey(), clazz);
}
final Map> parametersMulti = new HashMap<>();
for (Entry> e : paramsMulti.entrySet()) {
Collection tps = new LinkedList<>();
for (TreePathHandle tph : e.getValue()) {
TreePath p = tph.resolve(wc);
if (p == null) {
Logger.getLogger(JavaFix.class.getName()).log(Level.SEVERE, "Cannot resolve handle={0}", e.getValue());
continue;
}
tps.add(p);
}
parametersMulti.put(e.getKey(), tps);
}
Map constraints = new HashMap<>();
for (Entry> c : constraintsHandles.entrySet()) {
constraints.put(c.getKey(), c.getValue().resolve(wc));
}
Scope scope = Utilities.constructScope(wc, constraints, imports);
assert scope != null;
Tree parsed = Utilities.parseAndAttribute(wc, to, scope);
if (parsed.getKind() == Kind.EXPRESSION_STATEMENT && ExpressionTree.class.isAssignableFrom(tp.getLeaf().getKind().asInterface())) {
parsed = ((ExpressionStatementTree) parsed).getExpression();
}
Map rewriteFromTo = new IdentityHashMap<>();
List order = new ArrayList<>(7);
Tree original;
if (Utilities.isFakeBlock(parsed)) {
TreePath parent = tp.getParentPath();
List extends StatementTree> statements = ((BlockTree) parsed).getStatements();
if (tp.getLeaf().getKind() == Kind.BLOCK) {
BlockTree real = (BlockTree) tp.getLeaf();
rewriteFromTo.put(original = real, wc.getTreeMaker().Block(statements, real.isStatic()));
} else {
statements = statements.subList(1, statements.size() - 1);
if (parent.getLeaf().getKind() == Kind.BLOCK) {
List newStatements = new LinkedList<>();
for (StatementTree st : ((BlockTree) parent.getLeaf()).getStatements()) {
if (st == tp.getLeaf()) {
newStatements.addAll(statements);
} else {
newStatements.add(st);
}
}
rewriteFromTo.put(original = parent.getLeaf(), wc.getTreeMaker().Block(newStatements, ((BlockTree) parent.getLeaf()).isStatic()));
} else {
rewriteFromTo.put(original = tp.getLeaf(), wc.getTreeMaker().Block(statements, false));
}
}
} else if (Utilities.isFakeClass(parsed)) {
TreePath parent = tp.getParentPath();
List extends Tree> members = ((ClassTree) parsed).getMembers();
members = members.subList(1, members.size());
assert parent.getLeaf().getKind() == Kind.CLASS;
List newMembers = new LinkedList<>();
ClassTree ct = (ClassTree) parent.getLeaf();
for (Tree t : ct.getMembers()) {
if (t == tp.getLeaf()) {
newMembers.addAll(members);
} else {
newMembers.add(t);
}
}
rewriteFromTo.put(original = parent.getLeaf(), wc.getTreeMaker().Class(ct.getModifiers(), ct.getSimpleName(), ct.getTypeParameters(), ct.getExtendsClause(), ct.getImplementsClause(), newMembers));
} else if (tp.getLeaf().getKind() == Kind.BLOCK && parametersMulti.containsKey("$$1$") && parsed.getKind() != Kind.BLOCK && StatementTree.class.isAssignableFrom(parsed.getKind().asInterface())) {
List newStatements = new LinkedList<>();
newStatements.add(wc.getTreeMaker().ExpressionStatement(wc.getTreeMaker().Identifier("$$1$")));
newStatements.add((StatementTree) parsed);
newStatements.add(wc.getTreeMaker().ExpressionStatement(wc.getTreeMaker().Identifier("$$2$")));
parsed = wc.getTreeMaker().Block(newStatements, ((BlockTree) tp.getLeaf()).isStatic());
rewriteFromTo.put(original = tp.getLeaf(), parsed);
} else {
while ( tp.getParentPath().getLeaf().getKind() == Kind.PARENTHESIZED
&& tp.getLeaf().getKind() != parsed.getKind()
&& tp.getParentPath() != null
&& tp.getParentPath().getParentPath() != null
&& !requiresParenthesis(parsed, tp.getParentPath().getLeaf(), tp.getParentPath().getParentPath().getLeaf())
&& requiresParenthesis(tp.getLeaf(), tp.getParentPath().getLeaf(), tp.getParentPath().getParentPath().getLeaf()))
tp = tp.getParentPath();
rewriteFromTo.put(original = tp.getLeaf(), parsed);
}
order.add(original);
//prevent generating QualIdents inside import clauses - might be better to solve that inside ImportAnalysis2,
//but that seems not to be straightforward:
boolean inImport = parsed.getKind() == Kind.IMPORT;
boolean inPackage = false;
TreePath w = tp;
while (!inImport && w != null) {
inImport |= w.getLeaf().getKind() == Kind.IMPORT;
inPackage |= w.getParentPath() != null && w.getParentPath().getLeaf().getKind() == Kind.COMPILATION_UNIT && ((CompilationUnitTree) w.getParentPath().getLeaf()).getPackageName() == w.getLeaf();
w = w.getParentPath();
}
final Set originalTrees = Collections.newSetFromMap(new IdentityHashMap<>());
new ErrorAwareTreeScanner() {
@Override public Void scan(Tree tree, Void p) {
originalTrees.add(tree);
return super.scan(tree, p);
}
}.scan(original, null);
new ReplaceParameters(wc, ctx.isCanShowUI(), inImport, parameters, extraParamsData, implicitThis, parametersMulti, parameterNames, rewriteFromTo, order, originalTrees).scan(new TreePath(tp.getParentPath(), rewriteFromTo.get(original)), null);
if (inPackage) {
String newPackage = wc.getTreeUtilities().translate(wc.getCompilationUnit().getPackageName(), new IdentityHashMap<>(rewriteFromTo))./*XXX: not correct*/toString();
ClassPath source = wc.getClasspathInfo().getClassPath(PathKind.SOURCE);
FileObject ownerRoot = source.findOwnerRoot(wc.getFileObject());
if (ownerRoot != null) {
ctx.getFileChanges().add(new MoveFile(wc.getFileObject(), ownerRoot, newPackage.replace('.', '/')));
} else {
Logger.getLogger(JavaFix.class.getName()).log(Level.WARNING, "{0} not on its source path ({1})", new Object[] {FileUtil.getFileDisplayName(wc.getFileObject()), source.toString(PathConversionMode.PRINT)});
}
}
for (Tree from : order) {
Tree to = rewriteFromTo.get(from);
// gen.copyComments(from, to, true);
// gen.copyComments(from, to, false);
wc.rewrite(from, to);
}
}
}
private static class IK {
private final Object o;
public IK(Object o) {
this.o = o;
}
@Override
public int hashCode() {
int hash = 7;
if (o != null) {
hash = 59 * hash + System.identityHashCode(o);
}
return hash;
}
@Override
public boolean equals(Object obj) {
if (!(obj instanceof IK)) {
return false;
}
final IK other = (IK) obj;
return other.o == this.o;
}
}
private static final Set NUMBER_LITERAL_KINDS = EnumSet.of(Kind.FLOAT_LITERAL, Kind.DOUBLE_LITERAL, Kind.INT_LITERAL, Kind.LONG_LITERAL);
private static class ReplaceParameters extends ErrorAwareTreePathScanner {
private final CompilationInfo info;
private final TreeMaker make;
private final boolean canShowUI;
private final boolean inImport;
private final Map parameters;
private final Map extraParamsData;
private final Map implicitThis;
private final Map> parametersMulti;
private final Map parameterNames;
private final Map rewriteFromTo;
private final Set originalTrees;
private final List order;
private final List nestedTypes = new ArrayList<>();
public ReplaceParameters(WorkingCopy wc, boolean canShowUI, boolean inImport, Map parameters, Map extraParamsData, Map implicitThis, Map> parametersMulti, Map parameterNames, Map rewriteFromTo, List order, Set originalTrees) {
this.parameters = parameters;
this.info = wc;
this.make = wc.getTreeMaker();
this.canShowUI = canShowUI;
this.inImport = inImport;
this.extraParamsData = extraParamsData;
this.implicitThis = implicitThis;
this.parametersMulti = parametersMulti;
this.parameterNames = parameterNames;
this.rewriteFromTo = rewriteFromTo;
this.order = order;
this.originalTrees = originalTrees;
}
@Override
public Number visitIdentifier(IdentifierTree node, Void p) {
String name = node.getName().toString();
Tree newNode = handleIdentifier(name, node);
if (newNode != null) {
rewrite(node, newNode);
if (NUMBER_LITERAL_KINDS.contains(newNode.getKind())) {
return (Number) ((LiteralTree) newNode).getValue();
}
} else {
Element implicitThisClass = implicitThis.get(name);
if (implicitThisClass != null) {
Element enclClass = findEnclosingClass();
if (enclClass == implicitThisClass) {
rewrite(node, make.Identifier("this"));
} else {
rewrite(node, make.MemberSelect(make.QualIdent(implicitThisClass), "this"));
}
return null;
}
}
Element e = info.getTrees().getElement(getCurrentPath());
if (e != null && isStaticElement(e) && !inImport) {
rewrite(node, make.QualIdent(e));
}
return super.visitIdentifier(node, p);
}
private Element findEnclosingClass() {
TreePath findClass = getCurrentPath();
while (findClass != null && !TreeUtilities.CLASS_TREE_KINDS.contains(findClass.getLeaf().getKind())) {
findClass = findClass.getParentPath();
}
return findClass != null ? info.getTrees().getElement(findClass) : null;
}
@Override
public Number visitTypeParameter(TypeParameterTree node, Void p) {
String name = node.getName().toString();
Tree newNode = handleIdentifier(name, node);
if (newNode != null) {
rewrite(node, newNode);
if (NUMBER_LITERAL_KINDS.contains(newNode.getKind())) {
return (Number) ((LiteralTree) newNode).getValue();
}
}
return super.visitTypeParameter(node, p);
}
private static final EnumSet COMPLEX_OPS = EnumSet.of(Kind.CONDITIONAL_AND, Kind.CONDITIONAL_OR);
private Tree handleIdentifier(String name, Tree node) {
TreePath tp = parameters.get(name);
if (tp != null) {
if (tp.getLeaf() instanceof Hacks.RenameTree rt) {
return make.setLabel(rt.originalTree, rt.newName);
}
if (!parameterNames.containsKey(name)) {
Tree target = tp.getLeaf();
if (NUMBER_LITERAL_KINDS.contains(target.getKind())) {
return target;
}
//TODO: might also remove parenthesis, but needs to ensure the diff will still be minimal
// while (target.getKind() == Kind.PARENTHESIZED
// && !requiresParenthesis(((ParenthesizedTree) target).getExpression(), getCurrentPath().getParentPath().getLeaf())) {
// target = ((ParenthesizedTree) target).getExpression();
// }
if ( getCurrentPath().getParentPath() != null
&& getCurrentPath().getParentPath().getLeaf().getKind() == Kind.LOGICAL_COMPLEMENT) {
boolean rewriteNegated;
if (tp.getParentPath() == null) {
rewriteNegated = true;
} else {
Tree parent = tp.getParentPath().getLeaf();
TreePath aboveNewComplement = getCurrentPath().getParentPath().getParentPath();
// Do not try to optimize too complex expressions, following the principle of the least surprise.
// 1/ optimization is OK if the target is without parenthesis - extra level of parens can be avoided
// 2/ if both target and original are parenthesized, then optimization should be done to avoid one extra paren level
// 3/ do not optimize complex expressions - least surprise
rewriteNegated = (parent.getKind() != Kind.LOGICAL_COMPLEMENT) &&
(!COMPLEX_OPS.contains(target.getKind()) ||
parent.getKind() != Kind.PARENTHESIZED ||
(aboveNewComplement != null && aboveNewComplement.getLeaf().getKind() == Kind.PARENTHESIZED));
}
if (rewriteNegated) {
Tree negated = negate((ExpressionTree) tp.getLeaf(), getCurrentPath().getParentPath().getParentPath().getLeaf(), true);
if (negated != null) {
rewrite(getCurrentPath().getParentPath().getLeaf(), negated);
}
}
}
if (requiresParenthesis(target, node, getCurrentPath().getParentPath().getLeaf())) {
target = make.Parenthesized((ExpressionTree) target);
}
return target;
}
}
String variableName = parameterNames.get(name);
if (variableName != null) {
return make.Identifier(variableName);
}
return null;
}
@Override
public Number visitMemberSelect(MemberSelectTree node, Void p) {
Element e = info.getTrees().getElement(getCurrentPath());
if (e != null && (e.getKind() != ElementKind.CLASS || ((TypeElement) e).asType().getKind() != TypeKind.ERROR)) {
//check correct dependency:
checkDependency(info, e, canShowUI);
if (isStaticElement(e) && !inImport) {
rewrite(node, make.QualIdent(e));
return null;
}
}
MemberSelectTree nue = node;
String selectedName = node.getIdentifier().toString();
if (selectedName.startsWith("$") && parameterNames.get(selectedName) != null) {
nue = make.MemberSelect(node.getExpression(), parameterNames.get(selectedName));
}
if (nue.getExpression().getKind() == Kind.IDENTIFIER) {
String name = ((IdentifierTree) nue.getExpression()).getName().toString();
if (name.startsWith("$") && parameters.get(name) == null) {
Element implicitThisClass = implicitThis.get(name);
if (implicitThisClass != null) {
TreePath findClass = getCurrentPath();
OUTER: while (findClass != null) {
if (TreeUtilities.CLASS_TREE_KINDS.contains(findClass.getLeaf().getKind())) {
Element clazz = info.getTrees().getElement(findClass);
if (implicitThisClass.equals(clazz)) {
//this.<...>, the this may be implicit:
rewrite(node, make.Identifier(nue.getIdentifier()));
return null;
}
if (clazz.getKind().isClass() || clazz.getKind().isInterface()) {
for (Element currentClassElement : info.getElements().getAllMembers((TypeElement) clazz)) {
if (currentClassElement.getSimpleName().equals(node.getIdentifier())) {
//there may be a resolution conflict, let the member select be qualified
//TODO: no conflicts between fields and methods of the same name
//but we current still qualify the name
break OUTER;
}
}
}
}
findClass = findClass.getParentPath();
}
//let visitIdent handle this
} else {
//XXX: unbound variable, use identifier instead of member select - may cause problems?
rewrite(node, make.Identifier(nue.getIdentifier()));
return null;
}
}
}
if (nue != node) {
rewrite(node, nue);
}
return super.visitMemberSelect(node, p);
}
@Override
public Number visitVariable(VariableTree node, Void p) {
String name = node.getName().toString();
if (name.startsWith("$")) {
String nueName = parameterNames.get(name);
if (nueName != null) {
name = nueName;
}
}
VariableTree nue = make.Variable(node.getModifiers(), name, node.getType(), resolveOptionalValue(node.getInitializer()));
rewrite(node, nue);
return super.visitVariable(nue, p);
}
@Override
public Number visitIf(IfTree node, Void p) {
IfTree nue = make.If(node.getCondition(), node.getThenStatement(), resolveOptionalValue(node.getElseStatement()));
rewrite(node, nue);
return super.visitIf(nue, p);
}
@Override
public Number visitMethod(MethodTree node, Void p) {
String name = node.getName().toString();
String newName = name;
if (name.startsWith("$")) {
if (parameterNames.containsKey(name)) {
newName = parameterNames.get(name);
}
}
List extends TypeParameterTree> typeParams = resolveMultiParameters(node.getTypeParameters());
List extends VariableTree> params = resolveMultiParameters(node.getParameters());
List extends ExpressionTree> thrown = resolveMultiParameters(node.getThrows());
MethodTree nue = make.Method(node.getModifiers(), newName, node.getReturnType(), typeParams, params, thrown, node.getBody(), (ExpressionTree) node.getDefaultValue());
rewrite(node, nue);
return super.visitMethod(nue, p);
}
@Override
public Number visitClass(ClassTree node, Void p) {
String name = node.getSimpleName().toString();
String newName = name;
if (name.startsWith("$")) {
if (parameterNames.containsKey(name)) {
newName = parameterNames.get(name);
}
}
List extends TypeParameterTree> typeParams = resolveMultiParameters(node.getTypeParameters());
List extends Tree> implementsClauses = resolveMultiParameters(node.getImplementsClause());
List extends Tree> members = resolveMultiParameters(Utilities.filterHidden(getCurrentPath(), node.getMembers()));
Tree extend = resolveOptionalValue(node.getExtendsClause());
ClassTree nue = make.Class(node.getModifiers(), newName, typeParams, extend, implementsClauses, members);
rewrite(node, nue);
Element el = info.getTrees().getElement(getCurrentPath());
nestedTypes.add(el);
try {
return super.visitClass(nue, p);
} finally {
nestedTypes.remove(nestedTypes.size() - 1);
}
}
@Override
public Number visitExpressionStatement(ExpressionStatementTree node, Void p) {
CharSequence name = Utilities.getWildcardTreeName(node);
if (name != null) {
TreePath tp = parameters.get(name.toString());
if (tp != null && StatementTree.class.isAssignableFrom(tp.getLeaf().getKind().asInterface())) {
rewrite(node, tp.getLeaf());
return null;
}
}
return super.visitExpressionStatement(node, p);
}
@Override
public Number visitLiteral(LiteralTree node, Void p) {
if (node.getValue() instanceof Number number) {
return number;
}
return super.visitLiteral(node, p);
}
@Override
public Number visitBinary(BinaryTree node, Void p) {
Number left = scan(node.getLeftOperand(), p);
Number right = scan(node.getRightOperand(), p);
if (left != null && right != null) {
Number result = switch (node.getKind()) {
case MULTIPLY -> {
if (left instanceof Double || right instanceof Double) {
yield left.doubleValue() * right.doubleValue();
} else if (left instanceof Float || right instanceof Float) {
yield left.floatValue() * right.floatValue();
} else if (left instanceof Long || right instanceof Long) {
yield left.longValue() * right.longValue();
} else if (left instanceof Integer || right instanceof Integer) {
yield left.intValue() * right.intValue();
} else {
throw new IllegalStateException("left=" + left.getClass() + ", right=" + right.getClass());
}
}
case DIVIDE -> {
if (left instanceof Double || right instanceof Double) {
yield left.doubleValue() / right.doubleValue();
} else if (left instanceof Float || right instanceof Float) {
yield left.floatValue() / right.floatValue();
} else if (left instanceof Long || right instanceof Long) {
yield left.longValue() / right.longValue();
} else if (left instanceof Integer || right instanceof Integer) {
yield left.intValue() / right.intValue();
} else {
throw new IllegalStateException("left=" + left.getClass() + ", right=" + right.getClass());
}
}
case REMAINDER -> {
if (left instanceof Double || right instanceof Double) {
yield left.doubleValue() % right.doubleValue();
} else if (left instanceof Float || right instanceof Float) {
yield left.floatValue() % right.floatValue();
} else if (left instanceof Long || right instanceof Long) {
yield left.longValue() % right.longValue();
} else if (left instanceof Integer || right instanceof Integer) {
yield left.intValue() % right.intValue();
} else {
throw new IllegalStateException("left=" + left.getClass() + ", right=" + right.getClass());
}
}
case PLUS -> {
if (left instanceof Double || right instanceof Double) {
yield left.doubleValue() + right.doubleValue();
} else if (left instanceof Float || right instanceof Float) {
yield left.floatValue() + right.floatValue();
} else if (left instanceof Long || right instanceof Long) {
yield left.longValue() + right.longValue();
} else if (left instanceof Integer || right instanceof Integer) {
yield left.intValue() + right.intValue();
} else {
throw new IllegalStateException("left=" + left.getClass() + ", right=" + right.getClass());
}
}
case MINUS -> {
if (left instanceof Double || right instanceof Double) {
yield left.doubleValue() - right.doubleValue();
} else if (left instanceof Float || right instanceof Float) {
yield left.floatValue() - right.floatValue();
} else if (left instanceof Long || right instanceof Long) {
yield left.longValue() - right.longValue();
} else if (left instanceof Integer || right instanceof Integer) {
yield left.intValue() - right.intValue();
} else {
throw new IllegalStateException("left=" + left.getClass() + ", right=" + right.getClass());
}
}
case LEFT_SHIFT -> {
if (left instanceof Long || right instanceof Long) {
yield left.longValue() << right.longValue();
} else if (left instanceof Integer || right instanceof Integer) {
yield left.intValue() << right.intValue();
} else {
throw new IllegalStateException("left=" + left.getClass() + ", right=" + right.getClass());
}
}
case RIGHT_SHIFT -> {
if (left instanceof Long || right instanceof Long) {
yield left.longValue() >> right.longValue();
} else if (left instanceof Integer || right instanceof Integer) {
yield left.intValue() >> right.intValue();
} else {
throw new IllegalStateException("left=" + left.getClass() + ", right=" + right.getClass());
}
}
case UNSIGNED_RIGHT_SHIFT -> {
if (left instanceof Long || right instanceof Long) {
yield left.longValue() >>> right.longValue();
} else if (left instanceof Integer || right instanceof Integer) {
yield left.intValue() >>> right.intValue();
} else {
throw new IllegalStateException("left=" + left.getClass() + ", right=" + right.getClass());
}
}
case AND -> {
if (left instanceof Long || right instanceof Long) {
yield left.longValue() & right.longValue();
} else if (left instanceof Integer || right instanceof Integer) {
yield left.intValue() & right.intValue();
} else {
throw new IllegalStateException("left=" + left.getClass() + ", right=" + right.getClass());
}
}
case XOR -> {
if (left instanceof Long || right instanceof Long) {
yield left.longValue() ^ right.longValue();
} else if (left instanceof Integer || right instanceof Integer) {
yield left.intValue() ^ right.intValue();
} else {
throw new IllegalStateException("left=" + left.getClass() + ", right=" + right.getClass());
}
}
case OR -> {
if (left instanceof Long || right instanceof Long) {
yield left.longValue() | right.longValue();
} else if (left instanceof Integer || right instanceof Integer) {
yield left.intValue() | right.intValue();
} else {
throw new IllegalStateException("left=" + left.getClass() + ", right=" + right.getClass());
}
}
default -> null;
};
if (result != null) {
rewrite(node, make.Literal(result));
return result;
}
}
return null;
}
@Override
public Number visitUnary(UnaryTree node, Void p) {
Number op = scan(node.getExpression(), p);
if (op != null) {
Number result = switch (node.getKind()) {
case UNARY_MINUS -> {
if (op instanceof Double) {
yield -op.doubleValue();
} else if (op instanceof Float) {
yield -op.floatValue();
} else if (op instanceof Long) {
yield -op.longValue();
} else if (op instanceof Integer) {
yield -op.intValue();
} else {
throw new IllegalStateException("op=" + op.getClass());
}
}
case UNARY_PLUS -> op;
default -> null;
};
if (result != null) {
rewrite(node, make.Literal(result));
return result;
}
}
return super.visitUnary(node, p);
}
@Override
public Number visitBlock(BlockTree node, Void p) {
List extends StatementTree> nueStatement = resolveMultiParameters(node.getStatements());
BlockTree nue = make.Block(nueStatement, node.isStatic());
rewrite(node, nue);
return super.visitBlock(nue, p);
}
@Override
public Number visitCase(CaseTree node, Void p) {
List extends StatementTree> statements = (List extends StatementTree>) resolveMultiParameters(node.getStatements());
CaseTree nue = make.Case(node.getExpression(), statements);
rewrite(node, nue);
return super.visitCase(node, p);
}
@Override
@SuppressWarnings("unchecked")
public Number visitMethodInvocation(MethodInvocationTree node, Void p) {
List extends ExpressionTree> typeArgs = (List extends ExpressionTree>) resolveMultiParameters(node.getTypeArguments());
List extends ExpressionTree> args = resolveMultiParameters(node.getArguments());
MethodInvocationTree nue = make.MethodInvocation(typeArgs, node.getMethodSelect(), args);
rewrite(node, nue);
return super.visitMethodInvocation(nue, p);
}
@Override
@SuppressWarnings("unchecked")
public Number visitNewClass(NewClassTree node, Void p) {
List extends ExpressionTree> typeArgs = (List extends ExpressionTree>) resolveMultiParameters(node.getTypeArguments());
List extends ExpressionTree> args = resolveMultiParameters(node.getArguments());
NewClassTree nue = make.NewClass(node.getEnclosingExpression(), typeArgs, node.getIdentifier(), args, node.getClassBody());
rewrite(node, nue);
return super.visitNewClass(nue, p);
}
@Override
@SuppressWarnings("unchecked")
public Number visitParameterizedType(ParameterizedTypeTree node, Void p) {
List extends ExpressionTree> typeArgs = (List extends ExpressionTree>) resolveMultiParameters(node.getTypeArguments());
ParameterizedTypeTree nue = make.ParameterizedType(node.getType(), typeArgs);
rewrite(node, nue);
return super.visitParameterizedType(node, p);
}
@Override
public Number visitSwitch(SwitchTree node, Void p) {
List extends CaseTree> cases = (List extends CaseTree>) resolveMultiParameters(node.getCases());
SwitchTree nue = make.Switch(node.getExpression(), cases);
rewrite(node, nue);
return super.visitSwitch(node, p);
}
@Override
public Number visitTry(TryTree node, Void p) {
List extends Tree> resources = (List extends Tree>) resolveMultiParameters(node.getResources());
List extends CatchTree> catches = (List extends CatchTree>) resolveMultiParameters(node.getCatches());
TryTree nue = make.Try(resources, node.getBlock(), catches, node.getFinallyBlock());
rewrite(node, nue);
return super.visitTry(node, p);
}
@Override
public Number visitModifiers(ModifiersTree node, Void p) {
List annotations = new ArrayList<>(node.getAnnotations());
IdentifierTree ident = !annotations.isEmpty() && annotations.get(0).getAnnotationType().getKind() == Kind.IDENTIFIER ? (IdentifierTree) annotations.get(0).getAnnotationType() : null;
if (ident != null) {
annotations.remove(0);
String name = ident.getName().toString();
TreePath orig = parameters.get(name);
ModifiersTree nue;
if (orig != null && orig.getLeaf().getKind() == Kind.MODIFIERS) {
ModifiersTree origMods = (ModifiersTree) orig.getLeaf();
Object actualContent = extraParamsData.get(name);
Set actualFlags = EnumSet.noneOf(Modifier.class);
boolean[] actualAnnotationsMask = new boolean[0];
if (actualContent instanceof Object[] objects && objects[0] instanceof Set set) {
actualFlags.addAll(NbCollections.checkedSetByFilter(set, Modifier.class, false));
}
if (actualContent instanceof Object[] objects && objects[1] instanceof boolean[] booleans) {
actualAnnotationsMask = booleans;
}
nue = origMods;
for (Modifier m : origMods.getFlags()) {
if (actualFlags.contains(m)) continue;
nue = make.removeModifiersModifier(nue, m);
}
for (Modifier m : node.getFlags()) {
nue = make.addModifiersModifier(nue, m);
}
int ai = 0;
OUTER: for (AnnotationTree a : origMods.getAnnotations()) {
if (actualAnnotationsMask.length <= ai || actualAnnotationsMask[ai++]) {
continue;
}
for (Iterator it = annotations.iterator(); it.hasNext();) {
AnnotationTree toCheck = it.next();
Collection extends Occurrence> match = org.netbeans.api.java.source.matching.Matcher.create(info)
.setTreeTopSearch()
.setSearchRoot(new TreePath(getCurrentPath(), a))
.match(Pattern.createSimplePattern(new TreePath(getCurrentPath(), toCheck)));
if (!match.isEmpty()) {
//should be kept:
it.remove();
break OUTER;
}
}
nue = make.removeModifiersAnnotation(nue, a);
}
for (AnnotationTree a : annotations) {
nue = make.addModifiersAnnotation(nue, a);
scan(a, p);
}
} else {
nue = make.removeModifiersAnnotation(node, 0);
}
rewrite(node, nue);
return null;
}
return super.visitModifiers(node, p);
}
@Override
public Number visitNewArray(NewArrayTree node, Void p) {
List extends ExpressionTree> dimensions = (List extends ExpressionTree>) resolveMultiParameters(node.getDimensions());
List extends ExpressionTree> initializers = (List extends ExpressionTree>) resolveMultiParameters(node.getInitializers());
NewArrayTree nue = make.NewArray(node.getType(), dimensions, initializers);
rewrite(node, nue);
return super.visitNewArray(node, p);
}
@Override
public Number visitLambdaExpression(LambdaExpressionTree node, Void p) {
List extends VariableTree> args = resolveMultiParameters(node.getParameters());
LambdaExpressionTree nue = make.LambdaExpression(args, node.getBody());
Hacks.copyLambdaKind(node, nue);
rewrite(node, nue);
return super.visitLambdaExpression(node, p);
}
@Override
public Number visitAnnotation(AnnotationTree node, Void p) {
List extends ExpressionTree> args = resolveMultiParameters(node.getArguments());
AnnotationTree nue = make.Annotation(node.getAnnotationType(), args);
rewrite(node, nue);
return super.visitAnnotation(node, p);
}
@SuppressWarnings("unchecked")
private List resolveMultiParameters(List list) {
if (list == null) return null;
if (!Utilities.containsMultistatementTrees(list)) return list;
List result = new LinkedList<>();
for (T t : list) {
if (Utilities.isMultistatementWildcardTree(t)) {
Collection embedded = parametersMulti.get(Utilities.getWildcardTreeName(t).toString());
if (embedded != null) {
for (TreePath tp : embedded) {
if (tp != null) {
result.add((T) tp.getLeaf());
}
}
}
} else {
result.add(t);
}
}
return result;
}
@SuppressWarnings("unchecked")
private T resolveOptionalValue(T in) {
if (in != null && Utilities.isMultistatementWildcardTree(in)) {
TreePath out = parameters.get(Utilities.getWildcardTreeName(in).toString());
if (out != null) return (T) out.getLeaf();
return null;
}
return in;
}
private ExpressionTree negate(ExpressionTree original, Tree parent, boolean nullOnPlainNeg) {
ExpressionTree newTree;
switch (original.getKind()) {
case PARENTHESIZED -> {
ExpressionTree expr = ((ParenthesizedTree) original).getExpression();
ExpressionTree negatedOrNull = negate(expr, original, nullOnPlainNeg);
if (negatedOrNull != null) {
if (negatedOrNull.getKind() != Kind.PARENTHESIZED) {
negatedOrNull = make.Parenthesized(negatedOrNull);
}
}
return negatedOrNull;
/**
if (nullOnPlainNeg) {
return null;
} else {
return make.Unary(Kind.LOGICAL_COMPLEMENT, original);
}
*/
}
case INSTANCE_OF -> {
return make.Unary(Kind.LOGICAL_COMPLEMENT, make.Parenthesized(original));
}
case LOGICAL_COMPLEMENT -> {
newTree = ((UnaryTree) original).getExpression();
while (newTree.getKind() == Kind.PARENTHESIZED && !JavaFixUtilities.requiresParenthesis(((ParenthesizedTree) newTree).getExpression(), original, parent)) {
newTree = ((ParenthesizedTree) newTree).getExpression();
}
}
case NOT_EQUAL_TO -> newTree = negateBinaryOperator(original, Kind.EQUAL_TO, false);
case EQUAL_TO -> newTree = negateBinaryOperator(original, Kind.NOT_EQUAL_TO, false);
case BOOLEAN_LITERAL -> newTree = make.Literal(!(Boolean) ((LiteralTree) original).getValue());
case CONDITIONAL_AND -> newTree = negateBinaryOperator(original, Kind.CONDITIONAL_OR, true);
case CONDITIONAL_OR -> newTree = negateBinaryOperator(original, Kind.CONDITIONAL_AND, true);
case LESS_THAN -> newTree = negateBinaryOperator(original, Kind.GREATER_THAN_EQUAL, false);
case LESS_THAN_EQUAL -> newTree = negateBinaryOperator(original, Kind.GREATER_THAN, false);
case GREATER_THAN -> newTree = negateBinaryOperator(original, Kind.LESS_THAN_EQUAL, false);
case GREATER_THAN_EQUAL -> newTree = negateBinaryOperator(original, Kind.LESS_THAN, false);
default -> {
if (nullOnPlainNeg)
return null;
newTree = make.Unary(Kind.LOGICAL_COMPLEMENT, original);
}
}
if (JavaFixUtilities.requiresParenthesis(newTree, original, parent)) {
newTree = make.Parenthesized(newTree);
}
return newTree;
}
private ExpressionTree negateBinaryOperator(Tree original, Kind newKind, boolean negateOperands) {
BinaryTree bt = (BinaryTree) original;
BinaryTree nonNegated = make.Binary(newKind,
bt.getLeftOperand(),
bt.getRightOperand());
if (negateOperands) {
ExpressionTree lo = negate(bt.getLeftOperand(), nonNegated, false);
ExpressionTree ro = negate(bt.getRightOperand(), nonNegated, false);
return make.Binary(newKind,
lo != null ? lo : bt.getLeftOperand(),
ro != null ? ro : bt.getRightOperand());
}
return nonNegated;
}
private void rewrite(Tree from, Tree to) {
if (originalTrees.contains(from)) return ;
rewriteFromTo.put(from, to);
order.add(from);
}
}
private static final Map OPERATOR_PRIORITIES;
static {
OPERATOR_PRIORITIES = new EnumMap<>(Kind.class);
OPERATOR_PRIORITIES.put(Kind.IDENTIFIER, 0);
for (Kind k : Kind.values()) {
if (k.asInterface() == LiteralTree.class) {
OPERATOR_PRIORITIES.put(k, 0);
}
}
OPERATOR_PRIORITIES.put(Kind.ARRAY_ACCESS, 1);
OPERATOR_PRIORITIES.put(Kind.METHOD_INVOCATION, 1);
OPERATOR_PRIORITIES.put(Kind.MEMBER_REFERENCE, 1);
OPERATOR_PRIORITIES.put(Kind.MEMBER_SELECT, 1);
OPERATOR_PRIORITIES.put(Kind.POSTFIX_DECREMENT, 1);
OPERATOR_PRIORITIES.put(Kind.POSTFIX_INCREMENT, 1);
OPERATOR_PRIORITIES.put(Kind.NEW_ARRAY, 1);
OPERATOR_PRIORITIES.put(Kind.NEW_CLASS, 1);
OPERATOR_PRIORITIES.put(Kind.BITWISE_COMPLEMENT, 2);
OPERATOR_PRIORITIES.put(Kind.LOGICAL_COMPLEMENT, 2);
OPERATOR_PRIORITIES.put(Kind.PREFIX_DECREMENT, 2);
OPERATOR_PRIORITIES.put(Kind.PREFIX_INCREMENT, 2);
OPERATOR_PRIORITIES.put(Kind.UNARY_MINUS, 2);
OPERATOR_PRIORITIES.put(Kind.UNARY_PLUS, 2);
OPERATOR_PRIORITIES.put(Kind.TYPE_CAST, 3);
OPERATOR_PRIORITIES.put(Kind.DIVIDE, 4);
OPERATOR_PRIORITIES.put(Kind.MULTIPLY, 4);
OPERATOR_PRIORITIES.put(Kind.REMAINDER, 4);
OPERATOR_PRIORITIES.put(Kind.MINUS, 5);
OPERATOR_PRIORITIES.put(Kind.PLUS, 5);
OPERATOR_PRIORITIES.put(Kind.LEFT_SHIFT, 6);
OPERATOR_PRIORITIES.put(Kind.RIGHT_SHIFT, 6);
OPERATOR_PRIORITIES.put(Kind.UNSIGNED_RIGHT_SHIFT, 6);
OPERATOR_PRIORITIES.put(Kind.INSTANCE_OF, 7);
OPERATOR_PRIORITIES.put(Kind.GREATER_THAN, 7);
OPERATOR_PRIORITIES.put(Kind.GREATER_THAN_EQUAL, 7);
OPERATOR_PRIORITIES.put(Kind.LESS_THAN, 7);
OPERATOR_PRIORITIES.put(Kind.LESS_THAN_EQUAL, 7);
OPERATOR_PRIORITIES.put(Kind.EQUAL_TO, 8);
OPERATOR_PRIORITIES.put(Kind.NOT_EQUAL_TO, 8);
OPERATOR_PRIORITIES.put(Kind.AND, 9);
OPERATOR_PRIORITIES.put(Kind.OR, 11);
OPERATOR_PRIORITIES.put(Kind.XOR, 10);
OPERATOR_PRIORITIES.put(Kind.CONDITIONAL_AND, 12);
OPERATOR_PRIORITIES.put(Kind.CONDITIONAL_OR, 13);
OPERATOR_PRIORITIES.put(Kind.CONDITIONAL_EXPRESSION, 14);
OPERATOR_PRIORITIES.put(Kind.AND_ASSIGNMENT, 15);
OPERATOR_PRIORITIES.put(Kind.ASSIGNMENT, 15);
OPERATOR_PRIORITIES.put(Kind.DIVIDE_ASSIGNMENT, 15);
OPERATOR_PRIORITIES.put(Kind.LEFT_SHIFT_ASSIGNMENT, 15);
OPERATOR_PRIORITIES.put(Kind.MINUS_ASSIGNMENT, 15);
OPERATOR_PRIORITIES.put(Kind.MULTIPLY_ASSIGNMENT, 15);
OPERATOR_PRIORITIES.put(Kind.OR_ASSIGNMENT, 15);
OPERATOR_PRIORITIES.put(Kind.PLUS_ASSIGNMENT, 15);
OPERATOR_PRIORITIES.put(Kind.REMAINDER_ASSIGNMENT, 15);
OPERATOR_PRIORITIES.put(Kind.RIGHT_SHIFT_ASSIGNMENT, 15);
OPERATOR_PRIORITIES.put(Kind.UNSIGNED_RIGHT_SHIFT_ASSIGNMENT, 15);
OPERATOR_PRIORITIES.put(Kind.XOR_ASSIGNMENT, 15);
}
/**
* Checks whether {@code tree} can be used in places where a Primary is
* required (for instance, as the receiver expression of a method invocation).
* This is a friend API intended to be used by the java.hints module.
* Other modules should not use this API because it might not be stabilized.
* @param tree the tree to check
* @return {@code true} iff {@code tree} can be used where a Primary is
* required.
* @since 1.31
*/
public static boolean isPrimary(@NonNull Tree tree) {
final Integer treePriority = OPERATOR_PRIORITIES.get(tree.getKind());
return (treePriority != null && treePriority <= 1);
}
/**Checks whether putting {@code inner} tree into {@code outter} tree,
* when {@code original} is being replaced with {@code inner} requires parentheses.
*
* @param inner the new tree node that will be placed under {@code outter}
* @param original the tree node that is being replaced with {@code inner}
* @param outter the future parent node of {@code inner}
* @return true if and only if inner needs to be wrapped using {@link TreeMaker#Parenthesized(com.sun.source.tree.ExpressionTree) }
* to keep the original meaning.
*/
public static boolean requiresParenthesis(Tree inner, Tree original, Tree outter) {
if (!ExpressionTree.class.isAssignableFrom(inner.getKind().asInterface()) || outter == null) return false;
if (!ExpressionTree.class.isAssignableFrom(outter.getKind().asInterface())) {
boolean condition = switch (outter.getKind()) {
case IF -> original == ((IfTree)outter).getCondition();
case WHILE_LOOP -> original == ((WhileLoopTree)outter).getCondition();
case DO_WHILE_LOOP -> original == ((DoWhileLoopTree)outter).getCondition();
default -> false;
};
return condition && inner.getKind() != Tree.Kind.PARENTHESIZED;
}
if (outter.getKind() == Kind.PARENTHESIZED || inner.getKind() == Kind.PARENTHESIZED) return false;
if (outter.getKind() == Kind.METHOD_INVOCATION) {
if (((MethodInvocationTree) outter).getArguments().contains(original)) return false;
}
if (outter.getKind() == Kind.LAMBDA_EXPRESSION) {
LambdaExpressionTree lt = ((LambdaExpressionTree)outter);
if (lt.getParameters().contains(original)) {
return false;
}
if (lt.getBodyKind() == LambdaExpressionTree.BodyKind.STATEMENT) {
return false;
}
return original.getKind() == Tree.Kind.PARENTHESIZED;
}
if (outter.getKind() == Kind.NEW_CLASS) {
if (((NewClassTree) outter).getArguments().contains(original)) return false;
}
Integer innerPriority = OPERATOR_PRIORITIES.get(inner.getKind());
Integer outterPriority = OPERATOR_PRIORITIES.get(outter.getKind());
if (innerPriority == null || outterPriority == null) {
Logger.getLogger(JavaFix.class.getName()).log(Level.WARNING, "Unknown tree kind(s): {0}/{1}", new Object[] {inner.getKind(), outter.getKind()});
return true;
}
if (innerPriority > outterPriority) {
return true;
}
if (innerPriority < outterPriority) {
return false;
}
//associativity
if (BinaryTree.class.isAssignableFrom(outter.getKind().asInterface())) {
BinaryTree ot = (BinaryTree) outter;
//TODO: for + it might be possible to skip the parenthesis:
return ot.getRightOperand() == original;
}
if (CompoundAssignmentTree.class.isAssignableFrom(outter.getKind().asInterface())) {
CompoundAssignmentTree ot = (CompoundAssignmentTree) outter;
return ot.getVariable() == original;
}
if (AssignmentTree.class.isAssignableFrom(outter.getKind().asInterface())) {
AssignmentTree ot = (AssignmentTree) outter;
return ot.getVariable() == original;
}
return false;
}
private static final class RemoveFromParent extends JavaFix {
private final String displayName;
private final boolean safely;
public RemoveFromParent(String displayName, CompilationInfo info, TreePath toRemove, boolean safely) {
super(info, toRemove);
this.displayName = displayName;
this.safely = safely;
}
@Override
protected String getText() {
return displayName;
}
@Override
protected void performRewrite(TransformationContext ctx) {
WorkingCopy wc = ctx.getWorkingCopy();
TreePath tp = ctx.getPath();
doRemoveFromParent(wc, tp);
if (safely) {
Element el = wc.getTrees().getElement(tp);
if (el != null) {
new TreePathScanner() {
@Override
public Void scan(Tree tree, Void p) {
if (tree != null && tree != tp.getLeaf()) {
TreePath treePath = new TreePath(getCurrentPath(), tree);
Element e = wc.getTrees().getElement(treePath);
if (el == e) {
doRemoveFromParent(wc, treePath);
}
}
return super.scan(tree, p);
}
}.scan(new TreePath(wc.getCompilationUnit()), null);
}
}
}
private void doRemoveFromParent(WorkingCopy wc, TreePath what) {
TreeMaker make = wc.getTreeMaker();
Tree leaf = what.getLeaf();
Tree parentLeaf = what.getParentPath().getLeaf();
switch (parentLeaf.getKind()) {
case ANNOTATION -> {
AnnotationTree at = (AnnotationTree) parentLeaf;
AnnotationTree newAnnot = make.removeAnnotationAttrValue(at, (ExpressionTree) leaf);
wc.rewrite(at, newAnnot);
}
case BLOCK -> {
BlockTree bt = (BlockTree) parentLeaf;
wc.rewrite(bt, make.removeBlockStatement(bt, (StatementTree) leaf));
}
case CASE -> {
CaseTree caseTree = (CaseTree) parentLeaf;
wc.rewrite(caseTree, make.removeCaseStatement(caseTree, (StatementTree) leaf));
}
case CLASS -> {
ClassTree classTree = (ClassTree) parentLeaf;
ClassTree nueClassTree;
if (classTree.getTypeParameters().contains(leaf)) {
nueClassTree = make.removeClassTypeParameter(classTree, (TypeParameterTree) leaf);
} else if (classTree.getExtendsClause() == leaf) {
nueClassTree = make.Class(classTree.getModifiers(), classTree.getSimpleName(), classTree.getTypeParameters(), null, classTree.getImplementsClause(), classTree.getMembers());
} else if (classTree.getImplementsClause().contains(leaf)) {
nueClassTree = make.removeClassImplementsClause(classTree, leaf);
} else if (classTree.getMembers().contains(leaf)) {
nueClassTree = make.removeClassMember(classTree, leaf);
} else {
throw new UnsupportedOperationException();
}
wc.rewrite(classTree, nueClassTree);
}
case UNION_TYPE -> {
UnionTypeTree disjunct = (UnionTypeTree) parentLeaf;
List extends Tree> alternatives = new LinkedList(disjunct.getTypeAlternatives());
alternatives.remove(leaf);
wc.rewrite(disjunct, make.UnionType(alternatives));
}
case METHOD -> {
MethodTree mTree = (MethodTree) parentLeaf;
MethodTree newMethod;
if (mTree.getTypeParameters().contains(leaf)) {
newMethod = make.removeMethodTypeParameter(mTree, (TypeParameterTree) leaf);
} else if (mTree.getParameters().contains(leaf)) {
newMethod = make.removeMethodParameter(mTree, (VariableTree) leaf);
} else if (mTree.getThrows().contains(leaf)) {
newMethod = make.removeMethodThrows(mTree, (ExpressionTree) leaf);
} else {
throw new UnsupportedOperationException();
}
wc.rewrite(mTree, newMethod);
}
case METHOD_INVOCATION -> {
MethodInvocationTree iTree = (MethodInvocationTree) parentLeaf;
MethodInvocationTree newInvocation;
if (iTree.getTypeArguments().contains(leaf)) {
newInvocation = make.removeMethodInvocationTypeArgument(iTree, (ExpressionTree) leaf);
} else if (iTree.getArguments().contains(leaf)) {
newInvocation = make.removeMethodInvocationArgument(iTree, (ExpressionTree) leaf);
} else {
throw new UnsupportedOperationException();
}
wc.rewrite(iTree, newInvocation);
}
case MODIFIERS -> {
ModifiersTree modsTree = (ModifiersTree) parentLeaf;
wc.rewrite(modsTree, make.removeModifiersAnnotation(modsTree, (AnnotationTree) leaf));
}
case NEW_CLASS -> {
NewClassTree newCTree = (NewClassTree) parentLeaf;
NewClassTree newNCT;
if (newCTree.getTypeArguments().contains(leaf)) {
newNCT = make.removeNewClassTypeArgument(newCTree, (ExpressionTree) leaf);
} else if (newCTree.getArguments().contains(leaf)) {
newNCT = make.removeNewClassArgument(newCTree, (ExpressionTree) leaf);
} else {
throw new UnsupportedOperationException();
}
wc.rewrite(newCTree, newNCT);
}
case PARAMETERIZED_TYPE -> {
ParameterizedTypeTree parTree = (ParameterizedTypeTree) parentLeaf;
wc.rewrite(parTree, make.removeParameterizedTypeTypeArgument(parTree, (ExpressionTree) leaf));
}
case SWITCH -> {
SwitchTree switchTree = (SwitchTree) parentLeaf;
SwitchTree newSwitch;
if (switchTree.getCases().contains(leaf)) {
newSwitch = make.removeSwitchCase(switchTree, (CaseTree) leaf);
} else {
throw new UnsupportedOperationException();
}
wc.rewrite(switchTree, newSwitch);
}
case TRY -> {
TryTree tryTree = (TryTree) parentLeaf;
TryTree newTry;
if (tryTree.getResources().contains(leaf)) {
LinkedList resources = new LinkedList<>(tryTree.getResources());
resources.remove(leaf);
newTry = make.Try(resources, tryTree.getBlock(), tryTree.getCatches(), tryTree.getFinallyBlock());
} else if (tryTree.getCatches().contains(leaf)) {
newTry = make.removeTryCatch(tryTree, (CatchTree) leaf);
} else {
throw new UnsupportedOperationException();
}
wc.rewrite(tryTree, newTry);
}
case EXPRESSION_STATEMENT -> {
doRemoveFromParent(wc, what.getParentPath());
}
case ASSIGNMENT -> {
AssignmentTree assignmentTree = (AssignmentTree) parentLeaf;
if (leaf == assignmentTree.getVariable()) {
if (wc.getTreeUtilities().isExpressionStatement(assignmentTree.getExpression())) {
wc.rewrite(parentLeaf, assignmentTree.getExpression());
} else {
doRemoveFromParent(wc, what.getParentPath());
}
} else {
throw new UnsupportedOperationException();
}
}
case AND_ASSIGNMENT, DIVIDE_ASSIGNMENT, LEFT_SHIFT_ASSIGNMENT, MINUS_ASSIGNMENT,
MULTIPLY_ASSIGNMENT, OR_ASSIGNMENT, PLUS_ASSIGNMENT, REMAINDER_ASSIGNMENT,
RIGHT_SHIFT_ASSIGNMENT, UNSIGNED_RIGHT_SHIFT_ASSIGNMENT, XOR_ASSIGNMENT -> {
CompoundAssignmentTree compoundAssignmentTree = (CompoundAssignmentTree) parentLeaf;
if (leaf == compoundAssignmentTree.getVariable()) {
if (wc.getTreeUtilities().isExpressionStatement(compoundAssignmentTree.getExpression())) {
wc.rewrite(parentLeaf, compoundAssignmentTree.getExpression());
} else {
doRemoveFromParent(wc, what.getParentPath());
}
} else {
throw new UnsupportedOperationException();
}
}
default -> {
wc.rewrite(what.getLeaf(), make.Block(Collections.emptyList(), false));
}
}
}
private static boolean canSafelyRemove(CompilationInfo info, TreePath tp) {
AtomicBoolean ret = new AtomicBoolean(true);
Element el = info.getTrees().getElement(tp);
if (el != null) {
new TreePathScanner() {
@Override
public Void scan(Tree tree, Void p) {
if (tree != null && tree != tp.getLeaf()) {
TreePath treePath = new TreePath(getCurrentPath(), tree);
Element e = info.getTrees().getElement(treePath);
if (el == e) {
Tree parentLeaf = treePath.getParentPath().getLeaf();
switch (parentLeaf.getKind()) {
case ASSIGNMENT -> {
AssignmentTree assignmentTree = (AssignmentTree) parentLeaf;
if (tree == assignmentTree.getVariable()) {
if (!info.getTreeUtilities().isExpressionStatement(assignmentTree.getExpression()) && canHaveSideEffects(assignmentTree.getExpression())) {
ret.set(false);
return null;
}
} else {
ret.set(false);
return null;
}
}
case AND_ASSIGNMENT, DIVIDE_ASSIGNMENT, LEFT_SHIFT_ASSIGNMENT, MINUS_ASSIGNMENT,
MULTIPLY_ASSIGNMENT, OR_ASSIGNMENT, PLUS_ASSIGNMENT, REMAINDER_ASSIGNMENT,
RIGHT_SHIFT_ASSIGNMENT, UNSIGNED_RIGHT_SHIFT_ASSIGNMENT, XOR_ASSIGNMENT -> {
CompoundAssignmentTree compoundAssignmentTree = (CompoundAssignmentTree) parentLeaf;
if (tree == compoundAssignmentTree.getVariable()) {
if (!info.getTreeUtilities().isExpressionStatement(compoundAssignmentTree.getExpression()) && canHaveSideEffects(compoundAssignmentTree.getExpression())) {
ret.set(false);
return null;
}
} else {
ret.set(false);
return null;
}
}
default -> {
ret.set(false);
return null;
}
}
}
}
return super.scan(tree, p);
}
}.scan(new TreePath(info.getCompilationUnit()), null);
}
return ret.get();
}
private static boolean canHaveSideEffects(Tree tree) {
AtomicBoolean ret = new AtomicBoolean();
new TreeScanner() {
@Override
public Void scan(Tree tree, Void p) {
if (tree != null) {
switch (tree.getKind()) {
case METHOD_INVOCATION, NEW_CLASS, POSTFIX_DECREMENT, POSTFIX_INCREMENT,
PREFIX_DECREMENT, PREFIX_INCREMENT -> {
ret.set(true);
return null;
}
}
}
return super.scan(tree, p);
}
}.scan(tree, null);
return ret.get();
}
}
//TODO: from FileMovePlugin
private static class MoveFile extends SimpleRefactoringElementImplementation {
private FileObject toMove;
private final FileObject sourceRoot;
private final String targetFolderName;
public MoveFile(FileObject toMove, FileObject sourceRoot, String targetFolderName) {
this.toMove = toMove;
this.sourceRoot = sourceRoot;
this.targetFolderName = targetFolderName;
}
@Override
@Messages({"#{0} - original file name", "TXT_MoveFile=Move {0}"})
public String getText() {
return Bundle.TXT_MoveFile(toMove.getNameExt());
}
@Override
public String getDisplayText() {
return getText();
}
DataFolder sourceFolder;
DataObject source;
@Override
public void performChange() {
try {
FileObject target = FileUtil.createFolder(sourceRoot, targetFolderName);
DataFolder targetFolder = DataFolder.findFolder(target);
if (!toMove.isValid()) {
String path = FileUtil.getFileDisplayName(toMove);
Logger.getLogger(JavaFix.class.getName()).fine("Invalid FileObject " + path + "trying to recreate...");
toMove = FileUtil.toFileObject(FileUtil.toFile(toMove));
if (toMove==null) {
Logger.getLogger(JavaFix.class.getName()).severe("Invalid FileObject " + path + "\n. File not found.");
return;
}
}
source = DataObject.find(toMove);
sourceFolder = source.getFolder();
source.move(targetFolder);
} catch (IOException ex) {
ex.printStackTrace();
}
}
@Override
public void undoChange() {
try {
source.move(sourceFolder);
} catch (IOException ex) {
ex.printStackTrace();
}
}
@Override
public Lookup getLookup() {
return Lookup.EMPTY;
}
@Override
public FileObject getParentFile() {
return toMove;
}
@Override
public PositionBounds getPosition() {
return null;
}
}
}