1 /*******************************************************************************
2 * Copyright (c) 2000, 2011 IBM Corporation and others.
3 * All rights reserved. This program and the accompanying materials
4 * are made available under the terms of the Eclipse Public License v1.0
5 * which accompanies this distribution, and is available at
6 * http://www.eclipse.org/legal/epl-v10.html
9 * IBM Corporation - initial API and implementation
10 *******************************************************************************/
11 package org.eclipse.jdt.internal.corext.dom;
13 import java.util.ArrayList;
14 import java.util.Iterator;
15 import java.util.List;
17 import org.eclipse.text.edits.TextEditGroup;
19 import org.eclipse.jdt.core.dom.AST;
20 import org.eclipse.jdt.core.dom.ASTNode;
21 import org.eclipse.jdt.core.dom.Annotation;
22 import org.eclipse.jdt.core.dom.AnnotationTypeDeclaration;
23 import org.eclipse.jdt.core.dom.AnnotationTypeMemberDeclaration;
24 import org.eclipse.jdt.core.dom.EnumConstantDeclaration;
25 import org.eclipse.jdt.core.dom.EnumDeclaration;
26 import org.eclipse.jdt.core.dom.FieldDeclaration;
27 import org.eclipse.jdt.core.dom.IExtendedModifier;
28 import org.eclipse.jdt.core.dom.MethodDeclaration;
29 import org.eclipse.jdt.core.dom.Modifier;
30 import org.eclipse.jdt.core.dom.SingleVariableDeclaration;
31 import org.eclipse.jdt.core.dom.TypeDeclaration;
32 import org.eclipse.jdt.core.dom.VariableDeclarationExpression;
33 import org.eclipse.jdt.core.dom.VariableDeclarationStatement;
34 import org.eclipse.jdt.core.dom.rewrite.ASTRewrite;
35 import org.eclipse.jdt.core.dom.rewrite.ITrackedNodePosition;
36 import org.eclipse.jdt.core.dom.rewrite.ListRewrite;
38 import org.eclipse.jdt.internal.corext.fix.LinkedProposalPositionGroup;
39 import org.eclipse.jdt.internal.corext.fix.LinkedProposalPositionGroup.PositionInformation;
44 public class ModifierRewrite {
46 public static final int VISIBILITY_MODIFIERS= Modifier.PUBLIC | Modifier.PRIVATE | Modifier.PROTECTED;
48 private ListRewrite fModifierRewrite;
52 public static ModifierRewrite create(ASTRewrite rewrite, ASTNode declNode) {
53 return new ModifierRewrite(rewrite, declNode);
56 private ModifierRewrite(ASTRewrite rewrite, ASTNode declNode) {
57 fModifierRewrite= evaluateListRewrite(rewrite, declNode);
58 fAst= declNode.getAST();
61 private ListRewrite evaluateListRewrite(ASTRewrite rewrite, ASTNode declNode) {
62 switch (declNode.getNodeType()) {
63 case ASTNode.METHOD_DECLARATION:
64 return rewrite.getListRewrite(declNode, MethodDeclaration.MODIFIERS2_PROPERTY);
65 case ASTNode.FIELD_DECLARATION:
66 return rewrite.getListRewrite(declNode, FieldDeclaration.MODIFIERS2_PROPERTY);
67 case ASTNode.VARIABLE_DECLARATION_EXPRESSION:
68 return rewrite.getListRewrite(declNode, VariableDeclarationExpression.MODIFIERS2_PROPERTY);
69 case ASTNode.VARIABLE_DECLARATION_STATEMENT:
70 return rewrite.getListRewrite(declNode, VariableDeclarationStatement.MODIFIERS2_PROPERTY);
71 case ASTNode.SINGLE_VARIABLE_DECLARATION:
72 return rewrite.getListRewrite(declNode, SingleVariableDeclaration.MODIFIERS2_PROPERTY);
73 case ASTNode.TYPE_DECLARATION:
74 return rewrite.getListRewrite(declNode, TypeDeclaration.MODIFIERS2_PROPERTY);
75 case ASTNode.ENUM_DECLARATION:
76 return rewrite.getListRewrite(declNode, EnumDeclaration.MODIFIERS2_PROPERTY);
77 case ASTNode.ANNOTATION_TYPE_DECLARATION:
78 return rewrite.getListRewrite(declNode, AnnotationTypeDeclaration.MODIFIERS2_PROPERTY);
79 case ASTNode.ENUM_CONSTANT_DECLARATION:
80 return rewrite.getListRewrite(declNode, EnumConstantDeclaration.MODIFIERS2_PROPERTY);
81 case ASTNode.ANNOTATION_TYPE_MEMBER_DECLARATION:
82 return rewrite.getListRewrite(declNode, AnnotationTypeMemberDeclaration.MODIFIERS2_PROPERTY);
84 throw new IllegalArgumentException("node has no modifiers: " + declNode.getClass().getName()); //$NON-NLS-1$
88 public ListRewrite getModifierRewrite() {
89 return fModifierRewrite;
93 * Sets the given modifiers. Removes all other flags, but leaves annotations in place.
95 * @param modifiers the modifiers to set
96 * @param editGroup the edit group in which to collect the corresponding text edits, or
97 * <code>null</code> if ungrouped
98 * @return a tracked position that contains the changed modifiers
100 public PositionInformation setModifiers(int modifiers, TextEditGroup editGroup) {
101 return internalSetModifiers(modifiers, -1, editGroup);
105 * Sets the included modifiers and removes the excluded modifiers. Does not touch other flags
106 * and leaves annotations in place.
108 * @param included the modifiers to set
109 * @param excluded the modifiers to remove
110 * @param editGroup the edit group in which to collect the corresponding text edits, or
111 * <code>null</code> if ungrouped
112 * @return a tracked position that contains the changed modifiers
114 public PositionInformation setModifiers(int included, int excluded, TextEditGroup editGroup) {
115 return internalSetModifiers(included, included | excluded, editGroup);
119 * Sets the included visibility modifiers and removes existing visibility modifiers. Does not
120 * touch other flags and leaves annotations in place.
122 * @param visibilityFlags the new visibility modifiers
123 * @param editGroup the edit group in which to collect the corresponding text edits, or
124 * <code>null</code> if ungrouped
125 * @return a tracked position that contains the changed modifiers, or <code>null</code> iff <code>editGroup == null</code>
127 public PositionInformation setVisibility(int visibilityFlags, TextEditGroup editGroup) {
128 return internalSetModifiers(visibilityFlags, VISIBILITY_MODIFIERS, editGroup);
131 public void copyAllModifiers(ASTNode otherDecl, TextEditGroup editGroup) {
132 copyAllModifiers(otherDecl, editGroup, false);
135 public void copyAllModifiers(ASTNode otherDecl, TextEditGroup editGroup, boolean copyIndividually) {
136 ListRewrite modifierList= evaluateListRewrite(fModifierRewrite.getASTRewrite(), otherDecl);
137 List<IExtendedModifier> originalList= modifierList.getOriginalList();
138 if (originalList.isEmpty()) {
142 if (copyIndividually) {
143 for (Iterator<IExtendedModifier> iterator= originalList.iterator(); iterator.hasNext();) {
144 ASTNode modifier= (ASTNode) iterator.next();
145 ASTNode copy= fModifierRewrite.getASTRewrite().createCopyTarget(modifier);
146 if (copy != null) { //paranoia check (only left here because we're in RC1)
147 fModifierRewrite.insertLast(copy, editGroup);
151 ASTNode copy= modifierList.createCopyTarget((ASTNode) originalList.get(0), (ASTNode) originalList.get(originalList.size() - 1));
152 if (copy != null) { //paranoia check (only left here because we're in RC1)
153 fModifierRewrite.insertLast(copy, editGroup);
158 public void copyAllAnnotations(ASTNode otherDecl, TextEditGroup editGroup) {
159 ListRewrite modifierList= evaluateListRewrite(fModifierRewrite.getASTRewrite(), otherDecl);
160 List<IExtendedModifier> originalList= modifierList.getOriginalList();
162 for (Iterator<IExtendedModifier> iterator= originalList.iterator(); iterator.hasNext();) {
163 IExtendedModifier modifier= iterator.next();
164 if (modifier.isAnnotation()) {
165 fModifierRewrite.insertLast(fModifierRewrite.getASTRewrite().createCopyTarget((Annotation) modifier), editGroup);
171 * Sets the given modifiers and removes all other modifiers that match the consideredFlags mask.
172 * Does not touch other flags and leaves annotations in place.
174 * @param modifiers the modifiers to set
175 * @param consideredFlags mask of modifiers to consider
176 * @param editGroup the edit group in which to collect the corresponding text edits, or
177 * <code>null</code> if ungrouped
178 * @return a tracked position that contains the changed modifiers
180 private PositionInformation internalSetModifiers(int modifiers, int consideredFlags, TextEditGroup editGroup) {
181 int newModifiers= modifiers & consideredFlags;
183 ITrackedNodePosition trackedFallback= null;
184 List<ITrackedNodePosition> trackedNodes= new ArrayList<ITrackedNodePosition>();
187 List<IExtendedModifier> originalList= fModifierRewrite.getOriginalList();
188 for (int i= 0; i < originalList.size(); i++) {
189 ASTNode curr= (ASTNode) originalList.get(i);
190 if (curr instanceof Modifier) {
191 int flag= ((Modifier)curr).getKeyword().toFlagValue();
192 if ((consideredFlags & flag) != 0) {
193 if ((newModifiers & flag) == 0) {
194 fModifierRewrite.remove(curr, editGroup);
195 if (trackedFallback == null)
196 trackedFallback= fModifierRewrite.getASTRewrite().track(curr);
198 newModifiers &= ~flag;
203 // find last annotation
204 IExtendedModifier lastAnnotation= null;
205 List<IExtendedModifier> extendedList= fModifierRewrite.getRewrittenList();
206 for (int i= 0; i < extendedList.size(); i++) {
207 IExtendedModifier curr= extendedList.get(i);
208 if (curr.isAnnotation())
209 lastAnnotation= curr;
213 List<Modifier> newNodes= ASTNodeFactory.newModifiers(fAst, newModifiers);
214 for (int i= 0; i < newNodes.size(); i++) {
215 Modifier curr= newNodes.get(i);
216 if ((curr.getKeyword().toFlagValue() & VISIBILITY_MODIFIERS) != 0) {
217 if (lastAnnotation != null)
218 fModifierRewrite.insertAfter(curr, (ASTNode) lastAnnotation, editGroup);
220 fModifierRewrite.insertFirst(curr, editGroup);
222 fModifierRewrite.insertLast(curr, editGroup);
224 trackedNodes.add(fModifierRewrite.getASTRewrite().track(curr));
227 if (trackedNodes.isEmpty()) {
228 if (trackedFallback == null) {
230 trackedFallback= fModifierRewrite.getASTRewrite().track(fModifierRewrite.getParent());
232 return new LinkedProposalPositionGroup.StartPositionInformation(trackedFallback);
234 return new LinkedProposalPositionGroup.TrackedNodesPosition(trackedNodes);