package org.eclipse.jdt.internal.core;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Map;
import org.eclipse.core.resources.IResourceDelta;
import org.eclipse.jdt.core.IJavaElement;
import org.eclipse.jdt.core.IJavaElementDelta;
import org.eclipse.jdt.core.dom.CompilationUnit;
@SuppressWarnings({"rawtypes", "unchecked"})
public class JavaElementDelta extends SimpleDelta implements IJavaElementDelta {
IJavaElementDelta[] affectedChildren = EMPTY_DELTA;
CompilationUnit ast = null;
IJavaElement changedElement;
IResourceDelta[] resourceDeltas = null;
int resourceDeltasCounter;
IJavaElement movedFromHandle = null;
IJavaElement movedToHandle = null;
IJavaElementDelta[] annotationDeltas = EMPTY_DELTA;
static IJavaElementDelta[] EMPTY_DELTA= new IJavaElementDelta[] {};
static int NEED_CHILD_INDEX = 3;
Map<Key, Integer> childIndex;
public boolean ignoreFromTests = false;
protected static class Key {
public final IJavaElement element;
public Key(IJavaElement element) {
this.element = element;
}
@Override
public int hashCode() {
return this.element.hashCode();
}
@Override
public boolean equals(Object obj) {
if (!(obj instanceof Key))
return false;
return equalsAndSameParent(this.element, ((Key) obj).element);
}
}
public JavaElementDelta(IJavaElement element) {
this.changedElement = element;
}
protected void addAffectedChild(JavaElementDelta child) {
switch (this.kind) {
case ADDED:
case REMOVED:
return;
case CHANGED:
this.changeFlags |= F_CHILDREN;
break;
default:
this.kind = CHANGED;
this.changeFlags |= F_CHILDREN;
}
if (this.changedElement.getElementType() >= IJavaElement.COMPILATION_UNIT) {
fineGrained();
}
Key childKey = new Key(child.getElement());
Integer existingChildIndex = getChildIndex(childKey);
if (existingChildIndex == null) {
addNewChild(child);
} else {
JavaElementDelta existingChild = (JavaElementDelta) this.affectedChildren[existingChildIndex];
switch (existingChild.getKind()) {
case ADDED:
switch (child.getKind()) {
case ADDED:
case CHANGED:
return;
case REMOVED:
removeExistingChild(childKey, existingChildIndex);
return;
}
break;
case REMOVED:
switch (child.getKind()) {
case ADDED:
child.kind = CHANGED;
this.affectedChildren[existingChildIndex] = child;
return;
case CHANGED:
case REMOVED:
return;
}
break;
case CHANGED:
switch (child.getKind()) {
case ADDED:
case REMOVED:
this.affectedChildren[existingChildIndex] = child;
return;
case CHANGED:
IJavaElementDelta[] children = child.getAffectedChildren();
for (int i = 0; i < children.length; i++) {
JavaElementDelta childsChild = (JavaElementDelta) children[i];
existingChild.addAffectedChild(childsChild);
}
int flags = child.changeFlags;
if ((existingChild.changeFlags & F_FINE_GRAINED) != 0 && (flags & F_FINE_GRAINED) == 0) {
flags &= ~F_CONTENT;
}
existingChild.changeFlags |= flags;
IResourceDelta[] resDeltas = child.getResourceDeltas();
if (resDeltas != null) {
existingChild.resourceDeltas = resDeltas;
existingChild.resourceDeltasCounter = child.resourceDeltasCounter;
}
return;
}
break;
default:
int flags = existingChild.getFlags();
this.affectedChildren[existingChildIndex] = child;
child.changeFlags |= flags;
}
}
}
public void added(IJavaElement element) {
added(element, 0);
}
public void added(IJavaElement element, int flags) {
JavaElementDelta addedDelta = new JavaElementDelta(element);
addedDelta.added();
addedDelta.changeFlags |= flags;
insertDeltaTree(element, addedDelta);
}
protected void addNewChild(JavaElementDelta child) {
this.affectedChildren = growAndAddToArray(this.affectedChildren, child);
if (this.childIndex != null) {
this.childIndex.put(new Key(child.getElement()), this.affectedChildren.length - 1);
}
}
protected void addResourceDelta(IResourceDelta child) {
switch (this.kind) {
case ADDED:
case REMOVED:
return;
case CHANGED:
this.changeFlags |= F_CONTENT;
break;
default:
this.kind = CHANGED;
this.changeFlags |= F_CONTENT;
}
if (this.resourceDeltas == null) {
this.resourceDeltas = new IResourceDelta[5];
this.resourceDeltas[this.resourceDeltasCounter++] = child;
return;
}
if (this.resourceDeltas.length == this.resourceDeltasCounter) {
System.arraycopy(this.resourceDeltas, 0, (this.resourceDeltas = new IResourceDelta[this.resourceDeltasCounter * 2]), 0, this.resourceDeltasCounter);
}
this.resourceDeltas[this.resourceDeltasCounter++] = child;
}
public JavaElementDelta changed(IJavaElement element, int changeFlag) {
JavaElementDelta changedDelta = new JavaElementDelta(element);
changedDelta.changed(changeFlag);
insertDeltaTree(element, changedDelta);
return changedDelta;
}
public void changedAST(CompilationUnit changedAST) {
this.ast = changedAST;
changed(F_AST_AFFECTED);
}
protected void clearAffectedChildren() {
this.affectedChildren = EMPTY_DELTA;
this.childIndex = null;
}
public void contentChanged() {
this.changeFlags |= F_CONTENT;
}
public void closed(IJavaElement element) {
JavaElementDelta delta = new JavaElementDelta(element);
delta.changed(F_CLOSED);
insertDeltaTree(element, delta);
}
protected JavaElementDelta createDeltaTree(IJavaElement element, JavaElementDelta delta) {
JavaElementDelta childDelta = delta;
ArrayList ancestors= getAncestors(element);
if (ancestors == null) {
if (equalsAndSameParent(delta.getElement(), getElement())) {
this.kind= delta.kind;
this.changeFlags = delta.changeFlags;
this.movedToHandle = delta.movedToHandle;
this.movedFromHandle = delta.movedFromHandle;
}
} else {
for (int i = 0, size = ancestors.size(); i < size; i++) {
IJavaElement ancestor = (IJavaElement) ancestors.get(i);
JavaElementDelta ancestorDelta = new JavaElementDelta(ancestor);
ancestorDelta.addAffectedChild(childDelta);
childDelta = ancestorDelta;
}
}
return childDelta;
}
protected static boolean equalsAndSameParent(IJavaElement e1, IJavaElement e2) {
IJavaElement parent1;
return e1.equals(e2) && ((parent1 = e1.getParent()) != null) && parent1.equals(e2.getParent());
}
protected JavaElementDelta find(IJavaElement e) {
if (equalsAndSameParent(getElement(), e))
return this;
return findDescendant(new Key(e));
}
protected JavaElementDelta findDescendant(Key key) {
if (this.affectedChildren.length == 0)
return null;
Integer index = getChildIndex(key);
if (index != null)
return (JavaElementDelta) this.affectedChildren[index];
for (IJavaElementDelta child : this.affectedChildren) {
JavaElementDelta delta = ((JavaElementDelta) child).findDescendant(key);
if (delta != null)
return delta;
}
return null;
}
public void fineGrained() {
changed(F_FINE_GRAINED);
}
@Override
public IJavaElementDelta[] getAddedChildren() {
return getChildrenOfType(ADDED);
}
@Override
public IJavaElementDelta[] getAffectedChildren() {
return this.affectedChildren;
}
private ArrayList getAncestors(IJavaElement element) {
IJavaElement parent = element.getParent();
if (parent == null) {
return null;
}
ArrayList parents = new ArrayList();
while (!parent.equals(this.changedElement)) {
parents.add(parent);
parent = parent.getParent();
if (parent == null) {
return null;
}
}
parents.trimToSize();
return parents;
}
@Override
public CompilationUnit getCompilationUnitAST() {
return this.ast;
}
@Override
public IJavaElementDelta[] getAnnotationDeltas() {
return this.annotationDeltas;
}
@Override
public IJavaElementDelta[] getChangedChildren() {
return getChildrenOfType(CHANGED);
}
protected Integer getChildIndex(Key key) {
int length = this.affectedChildren.length;
if (length < NEED_CHILD_INDEX) {
for (int i = 0; i < length; i++) {
if (equalsAndSameParent(key.element, this.affectedChildren[i].getElement())) {
return i;
}
}
return null;
}
if (this.childIndex == null) {
this.childIndex = new HashMap<Key, Integer>();
for (int i = 0; i < length; i++) {
this.childIndex.put(new Key(this.affectedChildren[i].getElement()), i);
}
}
return this.childIndex.get(key);
}
protected IJavaElementDelta[] getChildrenOfType(int type) {
int length = this.affectedChildren.length;
if (length == 0) {
return new IJavaElementDelta[] {};
}
ArrayList children= new ArrayList(length);
for (int i = 0; i < length; i++) {
if (this.affectedChildren[i].getKind() == type) {
children.add(this.affectedChildren[i]);
}
}
IJavaElementDelta[] childrenOfType = new IJavaElementDelta[children.size()];
children.toArray(childrenOfType);
return childrenOfType;
}
protected JavaElementDelta getDeltaFor(IJavaElement element) {
return find(element);
}
@Override
public IJavaElement getElement() {
return this.changedElement;
}
@Override
public IJavaElement getMovedFromElement() {
return this.movedFromHandle;
}
@Override
public IJavaElement getMovedToElement() {
return this.movedToHandle;
}
@Override
public IJavaElementDelta[] getRemovedChildren() {
return getChildrenOfType(REMOVED);
}
@Override
public IResourceDelta[] getResourceDeltas() {
if (this.resourceDeltas == null) return null;
if (this.resourceDeltas.length != this.resourceDeltasCounter) {
System.arraycopy(this.resourceDeltas, 0, this.resourceDeltas = new IResourceDelta[this.resourceDeltasCounter], 0, this.resourceDeltasCounter);
}
return this.resourceDeltas;
}
protected IJavaElementDelta[] growAndAddToArray(IJavaElementDelta[] array, IJavaElementDelta addition) {
IJavaElementDelta[] old = array;
array = new IJavaElementDelta[old.length + 1];
System.arraycopy(old, 0, array, 0, old.length);
array[old.length] = addition;
return array;
}
protected void insertDeltaTree(IJavaElement element, JavaElementDelta delta) {
JavaElementDelta childDelta= createDeltaTree(element, delta);
if (!equalsAndSameParent(element, getElement())) {
addAffectedChild(childDelta);
}
}
public void movedFrom(IJavaElement movedFromElement, IJavaElement movedToElement) {
JavaElementDelta removedDelta = new JavaElementDelta(movedFromElement);
removedDelta.kind = REMOVED;
removedDelta.changeFlags |= F_MOVED_TO;
removedDelta.movedToHandle = movedToElement;
insertDeltaTree(movedFromElement, removedDelta);
}
public void movedTo(IJavaElement movedToElement, IJavaElement movedFromElement) {
JavaElementDelta addedDelta = new JavaElementDelta(movedToElement);
addedDelta.kind = ADDED;
addedDelta.changeFlags |= F_MOVED_FROM;
addedDelta.movedFromHandle = movedFromElement;
insertDeltaTree(movedToElement, addedDelta);
}
public void opened(IJavaElement element) {
JavaElementDelta delta = new JavaElementDelta(element);
delta.changed(F_OPENED);
insertDeltaTree(element, delta);
}
protected void removeAffectedChild(JavaElementDelta child) {
if (this.affectedChildren.length == 0)
return;
Key childKey = new Key(child.getElement());
Integer exisingChildIndex = getChildIndex(childKey);
if (exisingChildIndex != null) {
removeExistingChild(childKey, exisingChildIndex);
}
}
protected IJavaElementDelta[] removeAndShrinkArray(IJavaElementDelta[] old, int index) {
IJavaElementDelta[] array = new IJavaElementDelta[old.length - 1];
if (index > 0)
System.arraycopy(old, 0, array, 0, index);
int rest = old.length - index - 1;
if (rest > 0)
System.arraycopy(old, index + 1, array, index, rest);
return array;
}
public void removed(IJavaElement element) {
removed(element, 0);
}
public void removed(IJavaElement element, int flags) {
JavaElementDelta removedDelta= new JavaElementDelta(element);
insertDeltaTree(element, removedDelta);
JavaElementDelta actualDelta = getDeltaFor(element);
if (actualDelta != null) {
actualDelta.removed();
actualDelta.changeFlags |= flags;
actualDelta.clearAffectedChildren();
}
}
protected void removeExistingChild(Key key, int index) {
this.affectedChildren = removeAndShrinkArray(this.affectedChildren, index);
if (this.childIndex != null) {
int length = this.affectedChildren.length;
if (length < NEED_CHILD_INDEX)
this.childIndex = null;
else {
this.childIndex.remove(key);
for (int i = index; i < length; i++) {
this.childIndex.put(new Key(this.affectedChildren[i].getElement()), i);
}
}
}
}
public void sourceAttached(IJavaElement element) {
JavaElementDelta attachedDelta = new JavaElementDelta(element);
attachedDelta.changed(F_SOURCEATTACHED);
insertDeltaTree(element, attachedDelta);
}
public void sourceDetached(IJavaElement element) {
JavaElementDelta detachedDelta = new JavaElementDelta(element);
detachedDelta.changed(F_SOURCEDETACHED);
insertDeltaTree(element, detachedDelta);
}
public String toDebugString(int depth) {
StringBuffer buffer = new StringBuffer();
for (int i= 0; i < depth; i++) {
buffer.append('\t');
}
buffer.append(((JavaElement)getElement()).toDebugString());
toDebugString(buffer);
IJavaElementDelta[] children = getAffectedChildren();
if (children != null) {
for (int i = 0; i < children.length; ++i) {
buffer.append("\n");
buffer.append(((JavaElementDelta) children[i]).toDebugString(depth + 1));
}
}
for (int i = 0; i < this.resourceDeltasCounter; i++) {
buffer.append("\n");
for (int j = 0; j < depth+1; j++) {
buffer.append('\t');
}
IResourceDelta resourceDelta = this.resourceDeltas[i];
buffer.append(resourceDelta.toString());
buffer.append("[");
switch (resourceDelta.getKind()) {
case IResourceDelta.ADDED :
buffer.append('+');
break;
case IResourceDelta.REMOVED :
buffer.append('-');
break;
case IResourceDelta.CHANGED :
buffer.append('*');
break;
default :
buffer.append('?');
break;
}
buffer.append("]");
}
IJavaElementDelta[] annotations = getAnnotationDeltas();
if (annotations != null) {
for (int i = 0; i < annotations.length; ++i) {
buffer.append("\n");
buffer.append(((JavaElementDelta) annotations[i]).toDebugString(depth + 1));
}
}
return buffer.toString();
}
@Override
protected boolean toDebugString(StringBuffer buffer, int flags) {
boolean prev = super.toDebugString(buffer, flags);
if ((flags & IJavaElementDelta.F_CHILDREN) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("CHILDREN");
prev = true;
}
if ((flags & IJavaElementDelta.F_CONTENT) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("CONTENT");
prev = true;
}
if ((flags & IJavaElementDelta.F_MOVED_FROM) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("MOVED_FROM(" + ((JavaElement)getMovedFromElement()).toStringWithAncestors() + ")");
prev = true;
}
if ((flags & IJavaElementDelta.F_MOVED_TO) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("MOVED_TO(" + ((JavaElement)getMovedToElement()).toStringWithAncestors() + ")");
prev = true;
}
if ((flags & IJavaElementDelta.F_ADDED_TO_CLASSPATH) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("ADDED TO CLASSPATH");
prev = true;
}
if ((flags & IJavaElementDelta.F_REMOVED_FROM_CLASSPATH) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("REMOVED FROM CLASSPATH");
prev = true;
}
if ((flags & IJavaElementDelta.F_REORDER) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("REORDERED");
prev = true;
}
if ((flags & IJavaElementDelta.F_ARCHIVE_CONTENT_CHANGED) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("ARCHIVE CONTENT CHANGED");
prev = true;
}
if ((flags & IJavaElementDelta.F_SOURCEATTACHED) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("SOURCE ATTACHED");
prev = true;
}
if ((flags & IJavaElementDelta.F_SOURCEDETACHED) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("SOURCE DETACHED");
prev = true;
}
if ((flags & IJavaElementDelta.F_FINE_GRAINED) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("FINE GRAINED");
prev = true;
}
if ((flags & IJavaElementDelta.F_PRIMARY_WORKING_COPY) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("PRIMARY WORKING COPY");
prev = true;
}
if ((flags & IJavaElementDelta.F_CLASSPATH_CHANGED) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("RAW CLASSPATH CHANGED");
prev = true;
}
if ((flags & IJavaElementDelta.F_RESOLVED_CLASSPATH_CHANGED) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("RESOLVED CLASSPATH CHANGED");
prev = true;
}
if ((flags & IJavaElementDelta.F_PRIMARY_RESOURCE) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("PRIMARY RESOURCE");
prev = true;
}
if ((flags & IJavaElementDelta.F_OPENED) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("OPENED");
prev = true;
}
if ((flags & IJavaElementDelta.F_CLOSED) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("CLOSED");
prev = true;
}
if ((flags & IJavaElementDelta.F_AST_AFFECTED) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("AST AFFECTED");
prev = true;
}
if ((flags & IJavaElementDelta.F_CATEGORIES) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("CATEGORIES");
prev = true;
}
if ((flags & IJavaElementDelta.F_ANNOTATIONS) != 0) {
if (prev)
buffer.append(" | ");
buffer.append("ANNOTATIONS");
prev = true;
}
return prev;
}
@Override
public String toString() {
return toDebugString(0);
}
}