blob: b5600222144aa190dd8b7825ad4f2d512102cbed [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2018 IBM Corporation and others.
*
* This program and the accompanying materials
* are made available under the terms of the Eclipse Public License 2.0
* which accompanies this distribution, and is available at
* https://www.eclipse.org/legal/epl-2.0/
*
* SPDX-License-Identifier: EPL-2.0
*
* Contributors:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package org.eclipse.jdt.internal.corext.refactoring;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Comparator;
import java.util.List;
import java.util.Map;
import org.eclipse.core.runtime.Assert;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.jdt.core.Flags;
import org.eclipse.jdt.core.IBuffer;
import org.eclipse.jdt.core.ICompilationUnit;
import org.eclipse.jdt.core.IField;
import org.eclipse.jdt.core.IImportContainer;
import org.eclipse.jdt.core.IJavaElement;
import org.eclipse.jdt.core.ISourceReference;
import org.eclipse.jdt.core.JavaModelException;
import org.eclipse.jdt.core.dom.ASTNode;
import org.eclipse.jdt.core.dom.ASTParser;
import org.eclipse.jdt.core.dom.CompilationUnit;
import org.eclipse.jdt.core.dom.FieldDeclaration;
import org.eclipse.jdt.core.dom.VariableDeclarationFragment;
import org.eclipse.jdt.internal.corext.refactoring.reorg.ReorgUtils;
import org.eclipse.jdt.internal.corext.refactoring.structure.ASTNodeSearchUtil;
import org.eclipse.jdt.internal.core.manipulation.util.Strings;
import org.eclipse.jdt.internal.corext.dom.IASTSharedValues;
/**
* A tuple used to keep source of an element and its type.
* @see IJavaElement
* @see org.eclipse.jdt.core.ISourceReference
*/
public class TypedSource {
private static class SourceTuple {
private SourceTuple(ICompilationUnit unit) {
this.unit= unit;
}
private ICompilationUnit unit;
private CompilationUnit node;
}
private final String fSource;
private final int fType;
private TypedSource(String source, int type){
Assert.isNotNull(source);
Assert.isTrue(canCreateForType(type));
fSource= source;
fType= type;
}
public static TypedSource create(String source, int type) {
if (source == null || ! canCreateForType(type))
return null;
return new TypedSource(source, type);
}
public String getSource() {
return fSource;
}
public int getType() {
return fType;
}
@Override
public boolean equals(Object other) {
if (! (other instanceof TypedSource))
return false;
TypedSource ts= (TypedSource)other;
return ts.getSource().equals(getSource()) && ts.getType() == getType();
}
@Override
public int hashCode() {
return getSource().hashCode() ^ (97 * getType());
}
private static boolean canCreateForType(int type){
return type == IJavaElement.FIELD
|| type == IJavaElement.TYPE
|| type == IJavaElement.IMPORT_CONTAINER
|| type == IJavaElement.IMPORT_DECLARATION
|| type == IJavaElement.INITIALIZER
|| type == IJavaElement.METHOD
|| type == IJavaElement.PACKAGE_DECLARATION
|| type == IJavaElement.JAVA_MODULE;
}
public static void sortByType(TypedSource[] typedSources){
Arrays.sort(typedSources, createTypeComparator());
}
public static Comparator<TypedSource> createTypeComparator() {
return (arg0, arg1) -> arg0.getType() - arg1.getType();
}
public static TypedSource[] createTypedSources(IJavaElement[] javaElements) throws CoreException {
//Map<ICompilationUnit, List<IJavaElement>>
Map<ICompilationUnit, List<IJavaElement>> grouped= ReorgUtils.groupByCompilationUnit(Arrays.asList(javaElements));
List<TypedSource> result= new ArrayList<>(javaElements.length);
for (Map.Entry<ICompilationUnit, List<IJavaElement>> entry : grouped.entrySet()) {
ICompilationUnit cu= entry.getKey();
for (IJavaElement javaElement : entry.getValue()) {
SourceTuple tuple= new SourceTuple(cu);
TypedSource[] ts= createTypedSources(javaElement, tuple);
if (ts != null)
result.addAll(Arrays.asList(ts));
}
}
return result.toArray(new TypedSource[result.size()]);
}
private static TypedSource[] createTypedSources(IJavaElement elem, SourceTuple tuple) throws CoreException {
if (! ReorgUtils.isInsideCompilationUnit(elem))
return null;
if (elem.getElementType() == IJavaElement.IMPORT_CONTAINER)
return createTypedSourcesForImportContainer(tuple, (IImportContainer)elem);
else if (elem.getElementType() == IJavaElement.FIELD)
return new TypedSource[] {create(getFieldSource((IField)elem, tuple), elem.getElementType())};
return new TypedSource[] {create(getSourceOfDeclararationNode(elem, tuple.unit), elem.getElementType())};
}
private static TypedSource[] createTypedSourcesForImportContainer(SourceTuple tuple, IImportContainer container) throws JavaModelException, CoreException {
IJavaElement[] imports= container.getChildren();
List<TypedSource> result= new ArrayList<>(imports.length);
for (IJavaElement importedElement : imports) {
result.addAll(Arrays.asList(createTypedSources(importedElement, tuple)));
}
return result.toArray(new TypedSource[result.size()]);
}
private static String getFieldSource(IField field, SourceTuple tuple) throws CoreException {
if (Flags.isEnum(field.getFlags())) {
String source= field.getSource();
if (source != null)
return source;
} else {
if (tuple.node == null) {
ASTParser parser= ASTParser.newParser(IASTSharedValues.SHARED_AST_LEVEL);
parser.setSource(tuple.unit);
tuple.node= (CompilationUnit) parser.createAST(null);
}
FieldDeclaration declaration= ASTNodeSearchUtil.getFieldDeclarationNode(field, tuple.node);
if (declaration.fragments().size() == 1)
return getSourceOfDeclararationNode(field, tuple.unit);
VariableDeclarationFragment declarationFragment= ASTNodeSearchUtil.getFieldDeclarationFragmentNode(field, tuple.node);
IBuffer buffer= tuple.unit.getBuffer();
StringBuilder buff= new StringBuilder();
buff.append(buffer.getText(declaration.getStartPosition(), ((ASTNode) declaration.fragments().get(0)).getStartPosition() - declaration.getStartPosition()));
buff.append(buffer.getText(declarationFragment.getStartPosition(), declarationFragment.getLength()));
buff.append(";"); //$NON-NLS-1$
return buff.toString();
}
return ""; //$NON-NLS-1$
}
private static String getSourceOfDeclararationNode(IJavaElement elem, ICompilationUnit cu) throws JavaModelException, CoreException {
Assert.isTrue(elem.getElementType() != IJavaElement.IMPORT_CONTAINER);
if (elem instanceof ISourceReference) {
ISourceReference reference= (ISourceReference) elem;
String source= reference.getSource();
if (source != null)
return Strings.trimIndentation(source, cu.getJavaProject(), false);
}
return ""; //$NON-NLS-1$
}
}