package com.ashardalon.pratt.commands.impls; import bjc.data.Tree; import com.ashardalon.pratt.ParserContext; import com.ashardalon.pratt.blocks.ParseBlock; import com.ashardalon.pratt.commands.BinaryPostCommand; import com.ashardalon.pratt.commands.CommandResult; import com.ashardalon.pratt.commands.CommandResult.Status; import com.ashardalon.pratt.tokens.Token; import bjc.data.SimpleTree; import bjc.utils.parserutils.ParserException; /** * A ternary command, like C's ?: * * @author bjculkin * * @param The key type of the tokens. * * @param The value type of the tokens. * * @param The state type of the parser. */ public class TernaryCommand extends BinaryPostCommand { private final ParseBlock innerBlck; private final Token mark; private final boolean nonassoc; /** * Create a new ternary command. * * @param precedence The precedence of this operator. * * @param innerBlock The representation of the inner block of the expression. * * @param marker The token to use as the root of the AST node. * * @param isNonassoc Whether or not the conditional is associative. */ public TernaryCommand(final int precedence, final ParseBlock innerBlock, final Token marker, final boolean isNonassoc) { super(precedence); if (innerBlock == null) throw new NullPointerException("Inner block must not be null"); else if (marker == null) throw new NullPointerException("Marker must not be null"); innerBlck = innerBlock; mark = marker; nonassoc = isNonassoc; } @Override public CommandResult denote(final Tree> operand, final Token operator, final ParserContext ctx) throws ParserException { final CommandResult innerRes = innerBlck.parse(ctx); if (innerRes.status != Status.SUCCESS) return innerRes; Tree> inner = innerRes.success(); final CommandResult outerRes = ctx.parse.parseExpression(1 + leftBinding(), ctx.tokens, ctx.state, false); if (outerRes.status != Status.SUCCESS) return outerRes; Tree> outer = outerRes.success(); return CommandResult.success(new SimpleTree<>(mark, inner, operand, outer)); } @Override public int nextBinding() { if (nonassoc) return leftBinding() - 1; return leftBinding(); } }