/* -*- Mode: java; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ package org.mozilla.javascript; import java.io.Serializable; /** * This class implements the object lookup required for the * <code>with</code> statement. * It simply delegates every action to its prototype except * for operations on its parent. */ public class NativeWith implements Scriptable, IdFunctionCall, Serializable { private static final long serialVersionUID = 1L; static void init(Scriptable scope, boolean sealed) { NativeWith obj = new NativeWith(); obj.setParentScope(scope); obj.setPrototype(ScriptableObject.getObjectPrototype(scope)); IdFunctionObject ctor = new IdFunctionObject(obj, FTAG, Id_constructor, "With", 0, scope); ctor.markAsConstructor(obj); if (sealed) { ctor.sealObject(); } ctor.exportAsScopeProperty(); } private NativeWith() { } protected NativeWith(Scriptable parent, Scriptable prototype) { this.parent = parent; this.prototype = prototype; } public String getClassName() { return "With"; } public boolean has(String id, Scriptable start) { return prototype.has(id, prototype); } public boolean has(int index, Scriptable start) { return prototype.has(index, prototype); } public Object get(String id, Scriptable start) { if (start == this) start = prototype; return prototype.get(id, start); } public Object get(int index, Scriptable start) { if (start == this) start = prototype; return prototype.get(index, start); } public void put(String id, Scriptable start, Object value) { if (start == this) start = prototype; prototype.put(id, start, value); } public void put(int index, Scriptable start, Object value) { if (start == this) start = prototype; prototype.put(index, start, value); } public void delete(String id) { prototype.delete(id); } public void delete(int index) { prototype.delete(index); } public Scriptable getPrototype() { return prototype; } public void setPrototype(Scriptable prototype) { this.prototype = prototype; } public Scriptable getParentScope() { return parent; } public void setParentScope(Scriptable parent) { this.parent = parent; } public Object[] getIds() { return prototype.getIds(); } public Object getDefaultValue(Class<?> typeHint) { return prototype.getDefaultValue(typeHint); } public boolean hasInstance(Scriptable value) { return prototype.hasInstance(value); } /** * Must return null to continue looping or the final collection result. */ protected Object updateDotQuery(boolean value) { // NativeWith itself does not support it throw new IllegalStateException(); } public Object execIdCall(IdFunctionObject f, Context cx, Scriptable scope, Scriptable thisObj, Object[] args) { if (f.hasTag(FTAG)) { if (f.methodId() == Id_constructor) { throw Context.reportRuntimeError1("msg.cant.call.indirect", "With"); } } throw f.unknown(); } static boolean isWithFunction(Object functionObj) { if (functionObj instanceof IdFunctionObject) { IdFunctionObject f = (IdFunctionObject)functionObj; return f.hasTag(FTAG) && f.methodId() == Id_constructor; } return false; } static Object newWithSpecial(Context cx, Scriptable scope, Object[] args) { ScriptRuntime.checkDeprecated(cx, "With"); scope = ScriptableObject.getTopLevelScope(scope); NativeWith thisObj = new NativeWith(); thisObj.setPrototype(args.length == 0 ? ScriptableObject.getObjectPrototype(scope) : ScriptRuntime.toObject(cx, scope, args[0])); thisObj.setParentScope(scope); return thisObj; } private static final Object FTAG = "With"; private static final int Id_constructor = 1; protected Scriptable prototype; protected Scriptable parent; }