/** * Copyright 2013 Douglas Campos, and individual contributors * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package org.dynjs.parser.ast; import java.util.ArrayList; import java.util.List; import org.dynjs.exception.ThrowException; import org.dynjs.parser.CodeVisitor; import org.dynjs.parser.js.Position; import org.dynjs.runtime.EnvironmentRecord; import org.dynjs.runtime.ExecutionContext; import org.dynjs.runtime.Reference; public class CompoundAssignmentExpression extends AbstractExpression { private final AbstractBinaryExpression rootExpr; public CompoundAssignmentExpression(AbstractBinaryExpression rootExpr) { this.rootExpr = rootExpr; } public Position getPosition() { return this.rootExpr.getPosition(); } public List<FunctionDeclaration> getFunctionDeclarations() { return this.rootExpr.getFunctionDeclarations(); } @Override public Object interpret(ExecutionContext context, boolean debug) { Object r = getRootExpr().interpret(context, debug); Object lref = getRootExpr().getLhs().interpret(context, debug); if (lref instanceof Reference) { if (((Reference) lref).isStrictReference()) { if (((Reference) lref).getBase() instanceof EnvironmentRecord) { if (((Reference) lref).getReferencedName().equals("arguments") || ((Reference) lref).getReferencedName().equals("eval")) { throw new ThrowException(context, context.createSyntaxError("invalid assignment: " + ((Reference) lref).getReferencedName())); } } } ((Reference) lref).putValue(context, r); return(r); } throw new ThrowException(context, context.createReferenceError("cannot assign to non-reference")); } public AbstractBinaryExpression getRootExpr() { return this.rootExpr; } public String toString() { return rootExpr.getLhs() + " " + rootExpr.getOp() + "=" + rootExpr.getRhs(); } public int getSizeMetric() { return this.rootExpr.getSizeMetric() + 5; } @Override public Object accept(Object context, CodeVisitor visitor, boolean strict) { return visitor.visit( context, this, strict); } public String dump(String indent) { StringBuilder buf = new StringBuilder(super.dump(indent)); buf.append(rootExpr.dump(indent + " ")); return buf.toString(); } }