diff options
Diffstat (limited to 'src/lombok')
-rw-r--r-- | src/lombok/core/AST.java | 58 |
1 files changed, 53 insertions, 5 deletions
diff --git a/src/lombok/core/AST.java b/src/lombok/core/AST.java index 8e060f07..871f24be 100644 --- a/src/lombok/core/AST.java +++ b/src/lombok/core/AST.java @@ -71,6 +71,25 @@ public abstract class AST<N> { return nodeMap.get(node); } + @SuppressWarnings("unchecked") + private Node replaceNewWithExistingOld(Map<N, Node> oldNodes, Node newNode) { + Node oldNode = oldNodes.get(newNode.get()); + if ( oldNode == null ) return newNode; + + List<Object> oldChildren = new ArrayList<Object>(); + for ( Node child : newNode.children ) { + Node oldChild = replaceNewWithExistingOld(oldNodes, child); + if ( oldChild == null ) oldChildren.add(child); + else { + oldChildren.add(oldChild); + oldChild.parent = oldNode; + } + } + + oldNode.children.addAll((Collection) oldChildren); + return oldNode; + } + public abstract class Node { protected final Kind kind; protected final N node; @@ -87,6 +106,11 @@ public abstract class AST<N> { this.isStructurallySignificant = calculateIsStructurallySignificant(); } + @Override public String toString() { + return String.format("NODE %s (%s) %s%s", + kind, node == null ? "(NULL)" : node.getClass(), handled ? "[HANDLED]" : "", node == null ? "" : node); + } + public String getPackageDeclaration() { return AST.this.getPackageDeclaration(); } @@ -162,6 +186,29 @@ public abstract class AST<N> { return n; } + /** + * Reparses the AST node represented by this node. Any existing nodes that occupy a different space in the AST are rehomed, any + * nodes that no longer exist are removed, and new nodes are created. + * + * Careful - the node you call this on must not itself have been removed or rehomed - it rebuilds <i>all children</i>. + */ + public void rebuild() { + Map<N, Node> oldNodes = new HashMap<N, Node>(); + gatherAndRemoveChildren(oldNodes); + + Node newNode = buildTree(get(), kind); + + replaceNewWithExistingOld(oldNodes, newNode); + } + + private void gatherAndRemoveChildren(Map<N, Node> map) { + for ( Node child : children ) child.gatherAndRemoveChildren(map); + map.put(get(), this); + children.clear(); + identityDetector.remove(get()); + nodeMap.remove(get()); + } + public void removeChild(Node child) { children.remove(child); } @@ -254,12 +301,13 @@ public abstract class AST<N> { return list; } + @SuppressWarnings("unchecked") private <T extends Node> void buildWithField0(Class<T> nodeType, N child, FieldAccess fa, Collection<T> list) { try { Object o = fa.field.get(child); if ( o == null ) return; if ( fa.dim == 0 ) { - Node node = buildStatement(o); + Node node = buildTree((N)o, Kind.STATEMENT); if ( node != null ) list.add(nodeType.cast(node)); } else if ( o.getClass().isArray() ) buildWithArray(nodeType, o, list, fa.dim); else if ( Collection.class.isInstance(o) ) buildWithCollection(nodeType, o, list, fa.dim); @@ -268,10 +316,11 @@ public abstract class AST<N> { } } + @SuppressWarnings("unchecked") private <T extends Node> void buildWithArray(Class<T> nodeType, Object array, Collection<T> list, int dim) { if ( dim == 1 ) for ( Object v : (Object[])array ) { if ( v == null ) continue; - Node node = buildStatement(v); + Node node = buildTree((N)v, Kind.STATEMENT); if ( node != null ) list.add(nodeType.cast(node)); } else for ( Object v : (Object[])array ) { if ( v == null ) return; @@ -279,15 +328,14 @@ public abstract class AST<N> { } } + @SuppressWarnings("unchecked") private <T extends Node> void buildWithCollection(Class<T> nodeType, Object collection, Collection<T> list, int dim) { if ( dim == 1 ) for ( Object v : (Collection<?>)collection ) { if ( v == null ) continue; - Node node = buildStatement(v); + Node node = buildTree((N)v, Kind.STATEMENT); if ( node != null ) list.add(nodeType.cast(node)); } else for ( Object v : (Collection<?>)collection ) { buildWithCollection(nodeType, v, list, dim-1); } } - - protected abstract Node buildStatement(Object statement); } |