/*
Copyright (C) 2013 Raquel Pau and Albert Coroleu.
Walkmod is free software: you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
Walkmod is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with Walkmod. If not, see <http://www.gnu.org/licenses/>.*/
package org.walkmod.javalang.ast.type;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import org.walkmod.javalang.ast.Node;
import org.walkmod.javalang.ast.expr.AnnotationExpr;
import org.walkmod.javalang.visitors.GenericVisitor;
import org.walkmod.javalang.visitors.VoidVisitor;
/**
* @author Julio Vilmar Gesser
*/
public final class ReferenceType extends Type {
private Type type;
private int arrayCount;
private List<List<AnnotationExpr>> arraysAnnotations;
public ReferenceType() {
}
public ReferenceType(Type type) {
setType(type);
}
public ReferenceType(Type type, int arrayCount) {
setType(type);
this.arrayCount = arrayCount;
}
public ReferenceType(Type type, int arrayCount, List<AnnotationExpr> annotations) {
super(annotations);
setType(type);
this.arrayCount = arrayCount;
}
public ReferenceType(Type type, int arrayCount, List<AnnotationExpr> annotations,
List<List<AnnotationExpr>> arraysAnnotations) {
super(annotations);
setType(type);
this.arrayCount = arrayCount;
setArraysAnnotations(arraysAnnotations);
}
public ReferenceType(int beginLine, int beginColumn, int endLine, int endColumn, Type type, int arrayCount) {
super(beginLine, beginColumn, endLine, endColumn);
setType(type);
this.arrayCount = arrayCount;
}
public ReferenceType(int beginLine, int beginColumn, int endLine, int endColumn, Type type, int arrayCount,
List<AnnotationExpr> annotations, List<List<AnnotationExpr>> arraysAnnotations) {
super(beginLine, beginColumn, endLine, endColumn, annotations);
setType(type);
this.arrayCount = arrayCount;
setArraysAnnotations(arraysAnnotations);
}
@Override
public boolean removeChild(Node child) {
boolean result = false;
if (child != null) {
if (child == type) {
type = null;
result = true;
}
if (arraysAnnotations != null) {
if (child instanceof AnnotationExpr) {
List<List<AnnotationExpr>> arraysAnnotationsAux = new LinkedList<List<AnnotationExpr>>();
Iterator<List<AnnotationExpr>> it = arraysAnnotations.iterator();
while (it.hasNext()) {
List<AnnotationExpr> next = it.next();
List<AnnotationExpr> nextAux = new LinkedList<AnnotationExpr>(next);
result = result || nextAux.remove(child);
arraysAnnotationsAux.add(nextAux);
}
arraysAnnotations = arraysAnnotationsAux;
}
}
}
if(result){
updateReferences(child);
}
return result;
}
@Override
public List<Node> getChildren() {
List<Node> children = super.getChildren();
if (type != null) {
children.add(type);
}
if (arraysAnnotations != null) {
for (List<AnnotationExpr> annList : arraysAnnotations) {
children.addAll(annList);
}
}
return children;
}
@Override
public <R, A> R accept(GenericVisitor<R, A> v, A arg) {
if (!check()) {
return null;
}
return v.visit(this, arg);
}
@Override
public <A> void accept(VoidVisitor<A> v, A arg) {
if (check()) {
v.visit(this, arg);
}
}
public int getArrayCount() {
return arrayCount;
}
public Type getType() {
return type;
}
public void setArrayCount(int arrayCount) {
this.arrayCount = arrayCount;
}
public void setType(Type type) {
if (this.type != null) {
updateReferences(this.type);
}
this.type = type;
setAsParentNodeOf(type);
}
public List<List<AnnotationExpr>> getArraysAnnotations() {
return arraysAnnotations;
}
public void setArraysAnnotations(List<List<AnnotationExpr>> arraysAnnotations) {
this.arraysAnnotations = arraysAnnotations;
if (arraysAnnotations != null) {
for (List<AnnotationExpr> ann : arraysAnnotations) {
setAsParentNodeOf(ann);
}
}
}
@Override
public boolean replaceChildNode(Node oldChild, Node newChild) {
boolean updated = super.replaceChildNode(oldChild, newChild);
if (oldChild == type) {
setType((ClassOrInterfaceType) newChild);
updated = true;
}
if (!updated) {
if (arraysAnnotations != null) {
for (List<AnnotationExpr> list : arraysAnnotations) {
updated = updated || replaceChildNodeInList(oldChild, newChild, list);
}
}
}
return updated;
}
@Override
public ReferenceType clone() throws CloneNotSupportedException {
List<List<AnnotationExpr>> copy = null;
if (arraysAnnotations != null) {
copy = new LinkedList<List<AnnotationExpr>>();
for (List<AnnotationExpr> item : arraysAnnotations) {
copy.add(clone(item));
}
}
return new ReferenceType(clone(type), arrayCount, clone(getAnnotations()), copy);
}
}