blob: 17fc2685c6ba8f8f6b3f2dc0181110c3311c5b9c [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2005 IBM Corporation 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:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package org.eclipse.jdt.internal.corext.codemanipulation;
import java.text.Collator;
import java.util.Comparator;
import java.util.List;
import org.eclipse.core.resources.IWorkspaceRunnable;
import org.eclipse.core.resources.ResourcesPlugin;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.core.runtime.IProgressMonitor;
import org.eclipse.core.runtime.jobs.ISchedulingRule;
import org.eclipse.jdt.core.ICompilationUnit;
import org.eclipse.jdt.core.util.CompilationUnitSorter;
import org.eclipse.jdt.core.dom.*;
import org.eclipse.jdt.internal.corext.dom.ASTNodes;
import org.eclipse.jdt.internal.corext.util.JdtFlags;
import org.eclipse.jdt.internal.ui.JavaPlugin;
import org.eclipse.jdt.internal.ui.preferences.MembersOrderPreferenceCache;
/**
* Orders members is a compilation unit. A working copy must be passed.
*/
public class SortMembersOperation implements IWorkspaceRunnable {
/**
* Default comparator for body declarations.
*/
public class DefaultJavaElementComparator implements Comparator {
private Collator fCollator;
private MembersOrderPreferenceCache fMemberOrderCache;
public DefaultJavaElementComparator() {
this.fCollator= Collator.getInstance();
fMemberOrderCache= JavaPlugin.getDefault().getMemberOrderPreferenceCache();
}
private int category(BodyDeclaration bodyDeclaration) {
switch (bodyDeclaration.getNodeType()) {
case ASTNode.METHOD_DECLARATION:
{
MethodDeclaration method= (MethodDeclaration) bodyDeclaration;
if (method.isConstructor()) {
return getMemberCategory(MembersOrderPreferenceCache.CONSTRUCTORS_INDEX);
}
int flags= method.getModifiers();
if (Modifier.isStatic(flags))
return getMemberCategory(MembersOrderPreferenceCache.STATIC_METHODS_INDEX);
else
return getMemberCategory(MembersOrderPreferenceCache.METHOD_INDEX);
}
case ASTNode.FIELD_DECLARATION :
{
int flags= ((FieldDeclaration) bodyDeclaration).getModifiers();
if (Modifier.isStatic(flags))
return getMemberCategory(MembersOrderPreferenceCache.STATIC_FIELDS_INDEX);
else
return getMemberCategory(MembersOrderPreferenceCache.FIELDS_INDEX);
}
case ASTNode.INITIALIZER :
{
int flags= ((Initializer) bodyDeclaration).getModifiers();
if (Modifier.isStatic(flags))
return getMemberCategory(MembersOrderPreferenceCache.STATIC_INIT_INDEX);
else
return getMemberCategory(MembersOrderPreferenceCache.INIT_INDEX);
}
case ASTNode.TYPE_DECLARATION :
case ASTNode.ENUM_DECLARATION :
case ASTNode.ANNOTATION_TYPE_DECLARATION :
return getMemberCategory(MembersOrderPreferenceCache.TYPE_INDEX);
case ASTNode.ENUM_CONSTANT_DECLARATION :
return getMemberCategory(MembersOrderPreferenceCache.ENUM_CONSTANTS_INDEX);
case ASTNode.ANNOTATION_TYPE_MEMBER_DECLARATION:
return getMemberCategory(MembersOrderPreferenceCache.METHOD_INDEX); // reusing the method index
}
return 0; // should never happen
}
private int getMemberCategory(int kind) {
return fMemberOrderCache.getCategoryIndex(kind);
}
/**
* This comparator follows the contract defined in CompilationUnitSorter.sort.
* @see Comparator#compare(java.lang.Object, java.lang.Object)
* @see CompilationUnitSorter#sort(org.eclipse.jdt.core.ICompilationUnit, int[], java.util.Comparator, int, org.eclipse.core.runtime.IProgressMonitor)
*/
public int compare(Object e1, Object e2) {
BodyDeclaration bodyDeclaration1= (BodyDeclaration) e1;
BodyDeclaration bodyDeclaration2= (BodyDeclaration) e2;
int cat1= category(bodyDeclaration1);
int cat2= category(bodyDeclaration2);
if (cat1 != cat2) {
return cat1 - cat2;
}
if (fMemberOrderCache.isSortByVisibility()) {
int flags1= JdtFlags.getVisibilityCode(bodyDeclaration1);
int flags2= JdtFlags.getVisibilityCode(bodyDeclaration2);
int vis= fMemberOrderCache.getVisibilityIndex(flags1) - fMemberOrderCache.getVisibilityIndex(flags2);
if (vis != 0) {
return vis;
}
}
switch (bodyDeclaration1.getNodeType()) {
case ASTNode.METHOD_DECLARATION :
{
MethodDeclaration method1= (MethodDeclaration) bodyDeclaration1;
MethodDeclaration method2= (MethodDeclaration) bodyDeclaration2;
if (fMemberOrderCache.isSortByVisibility()) {
int vis= fMemberOrderCache.getVisibilityIndex(method1.getModifiers()) - fMemberOrderCache.getVisibilityIndex(method2.getModifiers());
if (vis != 0) {
return vis;
}
}
String name1= method1.getName().getIdentifier();
String name2= method2.getName().getIdentifier();
// method declarations (constructors) are sorted by name
int cmp= this.fCollator.compare(name1, name2);
if (cmp != 0) {
return cmp;
}
// if names equal, sort by parameter types
List parameters1= method1.parameters();
List parameters2= method2.parameters();
int length1= parameters1.size();
int length2= parameters2.size();
int len= Math.min(length1, length2);
for (int i= 0; i < len; i++) {
SingleVariableDeclaration param1= (SingleVariableDeclaration) parameters1.get(i);
SingleVariableDeclaration param2= (SingleVariableDeclaration) parameters2.get(i);
cmp= this.fCollator.compare(buildSignature(param1.getType()), buildSignature(param2.getType()));
if (cmp != 0) {
return cmp;
}
}
if (length1 != length2) {
return length1 - length2;
}
// preserve relative order
int value1= ((Integer) bodyDeclaration1.getProperty(CompilationUnitSorter.RELATIVE_ORDER)).intValue();
int value2= ((Integer) bodyDeclaration2.getProperty(CompilationUnitSorter.RELATIVE_ORDER)).intValue();
return value1 - value2;
}
case ASTNode.FIELD_DECLARATION :
{
FieldDeclaration field1= (FieldDeclaration) bodyDeclaration1;
FieldDeclaration field2= (FieldDeclaration) bodyDeclaration2;
String name1= ((VariableDeclarationFragment) field1.fragments().get(0)).getName().getIdentifier();
String name2= ((VariableDeclarationFragment) field2.fragments().get(0)).getName().getIdentifier();
// field declarations are sorted by name
return compareNames(bodyDeclaration1, bodyDeclaration2, name1, name2);
}
case ASTNode.INITIALIZER :
{
// preserve relative order
int value1= ((Integer) bodyDeclaration1.getProperty(CompilationUnitSorter.RELATIVE_ORDER)).intValue();
int value2= ((Integer) bodyDeclaration2.getProperty(CompilationUnitSorter.RELATIVE_ORDER)).intValue();
return value1 - value2;
}
case ASTNode.TYPE_DECLARATION :
case ASTNode.ENUM_DECLARATION :
case ASTNode.ANNOTATION_TYPE_DECLARATION :
{
AbstractTypeDeclaration type1= (AbstractTypeDeclaration) bodyDeclaration1;
AbstractTypeDeclaration type2= (AbstractTypeDeclaration) bodyDeclaration2;
String name1= type1.getName().getIdentifier();
String name2= type2.getName().getIdentifier();
// typedeclarations are sorted by name
return compareNames(bodyDeclaration1, bodyDeclaration2, name1, name2);
}
case ASTNode.ENUM_CONSTANT_DECLARATION :
{
EnumConstantDeclaration decl1= (EnumConstantDeclaration) bodyDeclaration1;
EnumConstantDeclaration decl2= (EnumConstantDeclaration) bodyDeclaration2;
String name1= decl1.getName().getIdentifier();
String name2= decl2.getName().getIdentifier();
// enum constants declarations are sorted by name
return compareNames(bodyDeclaration1, bodyDeclaration2, name1, name2);
}
case ASTNode.ANNOTATION_TYPE_MEMBER_DECLARATION :
{
AnnotationTypeMemberDeclaration decl1= (AnnotationTypeMemberDeclaration) bodyDeclaration1;
AnnotationTypeMemberDeclaration decl2= (AnnotationTypeMemberDeclaration) bodyDeclaration2;
String name1= decl1.getName().getIdentifier();
String name2= decl2.getName().getIdentifier();
// enum constants declarations are sorted by name
return compareNames(bodyDeclaration1, bodyDeclaration2, name1, name2);
}
}
return 0;
}
private int compareNames(BodyDeclaration bodyDeclaration1, BodyDeclaration bodyDeclaration2, String name1, String name2) {
int cmp= this.fCollator.compare(name1, name2);
if (cmp != 0) {
return cmp;
}
// preserve relative order
int value1= ((Integer) bodyDeclaration1.getProperty(CompilationUnitSorter.RELATIVE_ORDER)).intValue();
int value2= ((Integer) bodyDeclaration2.getProperty(CompilationUnitSorter.RELATIVE_ORDER)).intValue();
return value1 - value2;
}
private String buildSignature(Type type) {
return ASTNodes.asString(type);
}
}
private ICompilationUnit fCompilationUnit;
private int[] fPositions;
/**
* Creates the operation.
* @param cu The working copy of a compilation unit.
* @param positions Positions to track or <code>null</code> if no positions
* should be tracked.
*/
public SortMembersOperation(ICompilationUnit cu, int[] positions) {
fCompilationUnit= cu;
fPositions= positions;
}
/**
* Runs the operation.
*/
public void run(IProgressMonitor monitor) throws CoreException {
CompilationUnitSorter.sort(fCompilationUnit, fPositions, new DefaultJavaElementComparator(), 0, monitor);
}
/**
* @return Returns the scheduling rule for this operation
*/
public ISchedulingRule getScheduleRule() {
return ResourcesPlugin.getWorkspace().getRoot();
}
}