package org.basex.query.expr;
import static org.basex.query.QueryText.*;
import java.io.IOException;
import org.basex.io.serial.Serializer;
import org.basex.query.QueryContext;
import org.basex.query.QueryException;
import org.basex.query.item.Value;
import org.basex.query.iter.Iter;
import org.basex.query.util.Var;
import org.basex.util.InputInfo;
import org.basex.util.TokenBuilder;
import org.basex.util.Util;
/**
* Typeswitch expression.
*
* @author BaseX Team 2005-12, BSD License
* @author Christian Gruen
*/
public final class TypeSwitch extends ParseExpr {
/** Cases. */
private final TypeCase[] cases;
/** Condition. */
private Expr ts;
/**
* Constructor.
* @param ii input info
* @param t typeswitch expression
* @param c case expressions
*/
public TypeSwitch(final InputInfo ii, final Expr t, final TypeCase[] c) {
super(ii);
ts = t;
cases = c;
}
@Override
public Expr comp(final QueryContext ctx) throws QueryException {
ts = checkUp(ts, ctx).comp(ctx);
final Expr[] tmp = new Expr[cases.length];
for(int i = 0; i < cases.length; ++i) tmp[i] = cases[i].expr;
checkUp(ctx, tmp);
// static condition: return branch in question
if(ts.isValue()) {
for(final TypeCase c : cases) {
if(c.var.type == null || c.var.type.instance(ts.value(ctx)))
return optPre(c.comp(ctx, (Value) ts).expr, ctx);
}
}
// compile branches
for(final TypeCase c : cases) c.comp(ctx);
// return result if all branches are equal (e.g., empty)
boolean eq = true;
for(int i = 1; i < cases.length; ++i) {
eq &= cases[i - 1].expr.sameAs(cases[i].expr);
}
if(eq) return optPre(null, ctx);
// evaluate return type
type = cases[0].type();
for(int c = 1; c < cases.length; ++c) {
type = type.intersect(cases[c].type());
}
return this;
}
@Override
public Iter iter(final QueryContext ctx) throws QueryException {
final Value seq = ctx.value(ts);
for(final TypeCase c : cases) {
final Iter iter = c.iter(ctx, seq);
if(iter != null) return iter;
}
// will never happen
throw Util.notexpected();
}
@Override
public boolean isVacuous() {
for(final TypeCase c : cases) if(!c.expr.isVacuous()) return false;
return true;
}
@Override
public boolean uses(final Use u) {
if(u == Use.VAR) return true;
for(final TypeCase c : cases) if(c.uses(u)) return true;
return ts.uses(u);
}
@Override
public int count(final Var v) {
int c = ts.count(v);
for(final TypeCase t : cases) c += t.count(v);
return c;
}
@Override
public boolean removable(final Var v) {
for(final TypeCase c : cases) if(!c.removable(v)) return false;
return ts.removable(v);
}
@Override
public Expr remove(final Var v) {
for(final TypeCase c : cases) c.remove(v);
ts = ts.remove(v);
return this;
}
@Override
public void plan(final Serializer ser) throws IOException {
ser.openElement(this);
for(final TypeCase c : cases) c.plan(ser);
ts.plan(ser);
ser.closeElement();
}
@Override
public String toString() {
return new TokenBuilder(TYPESWITCH + PAR1 + ts + PAR2 + ' ').addSep(
cases, SEP).toString();
}
@Override
public Expr markTailCalls() {
for(final TypeCase t : cases) t.markTailCalls();
return this;
}
}