/*******************************************************************************
* Copyright (c) 2008 CWI.
* 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:
* jurgen@vinju.org
*******************************************************************************/
package org.rascalmpl.value.visitors;
import org.rascalmpl.value.IConstructor;
import org.rascalmpl.value.IExternalValue;
import org.rascalmpl.value.IList;
import org.rascalmpl.value.IListWriter;
import org.rascalmpl.value.IMap;
import org.rascalmpl.value.IMapWriter;
import org.rascalmpl.value.INode;
import org.rascalmpl.value.ISet;
import org.rascalmpl.value.ISetWriter;
import org.rascalmpl.value.ITuple;
import org.rascalmpl.value.IValue;
import org.rascalmpl.value.IValueFactory;
import org.rascalmpl.value.visitors.IValueVisitor;
/**
* This visitor will apply another visitor in a bottom-up fashion to an IValue
*
*/
public class BottomUpVisitor<T, E extends Throwable> extends VisitorAdapter<T, E> {
protected IValueFactory fFactory;
public BottomUpVisitor(IValueVisitor<T, E> visitor, IValueFactory factory) {
super(visitor);
this.fFactory = factory;
}
@Override
public T visitNode(INode o) throws E {
for (int i = 0; i < o.arity(); i++) {
o.get(i).accept(this);
}
return fVisitor.visitNode(o);
}
public T visitConstructor(IConstructor o) throws E {
for (int i = 0; i < o.arity(); i++) {
o.get(i).accept(this);
}
return fVisitor.visitConstructor(o);
}
@Override
public T visitList(IList o) throws E {
IListWriter w = fFactory.listWriter();
for (IValue elem : o) {
elem.accept(this);
}
return fVisitor.visitList(w.done());
}
@Override
public T visitSet(ISet o) throws E {
ISetWriter w = fFactory.setWriter();
for (IValue elem : o) {
elem.accept(this);
}
return fVisitor.visitSet(w.done());
}
@Override
public T visitMap(IMap o) throws E {
IMapWriter w = fFactory.mapWriter();
for (IValue elem : o) {
elem.accept(this);
o.get(elem).accept(this);
}
return fVisitor.visitMap(w.done());
}
@Override
public T visitRelation(ISet o) throws E {
ISetWriter w = fFactory.setWriter();
for (IValue tuple : o) {
tuple.accept(this);
}
return fVisitor.visitRelation(w.done());
}
@Override
public T visitTuple(ITuple o) throws E {
for (int i = 0; i < o.arity(); i++) {
o.get(i).accept(this);
}
return fVisitor.visitTuple(o);
}
public T visitExternal(IExternalValue externalValue) throws E {
return fVisitor.visitExternal(externalValue);
}
public T visitListRelation(IList o) throws E {
IListWriter w = fFactory.listWriter();
for (IValue tuple : o) {
tuple.accept(this);
}
return fVisitor.visitListRelation(w.done());
}
}