| /* |
| * Copyright (C) 1999-2000 Harri Porten (porten@kde.org) |
| * Copyright (C) 2004-2019 Apple Inc. All rights reserved. |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
| */ |
| |
| #include "config.h" |
| #include "JSPluginElementFunctions.h" |
| |
| #include "BridgeJSC.h" |
| #include "HTMLNames.h" |
| #include "HTMLPlugInElement.h" |
| #include "JSHTMLElement.h" |
| #include "PluginViewBase.h" |
| |
| |
| namespace WebCore { |
| using namespace JSC; |
| |
| using namespace Bindings; |
| using namespace HTMLNames; |
| |
| // JavaScript access to plug-in-exported properties for JSHTMLAppletElement, JSHTMLEmbedElement and JSHTMLObjectElement. |
| |
| Instance* pluginInstance(HTMLElement& element) |
| { |
| // The plugin element holds an owning reference, so we don't have to. |
| if (!is<HTMLPlugInElement>(element)) |
| return nullptr; |
| auto* instance = downcast<HTMLPlugInElement>(element).bindingsInstance(); |
| if (!instance || !instance->rootObject()) |
| return nullptr; |
| return instance; |
| } |
| |
| static JSObject* pluginScriptObjectFromPluginViewBase(HTMLPlugInElement& pluginElement, JSGlobalObject* globalObject) |
| { |
| Widget* pluginWidget = pluginElement.pluginWidget(); |
| if (!is<PluginViewBase>(pluginWidget)) |
| return nullptr; |
| |
| return downcast<PluginViewBase>(*pluginWidget).scriptObject(globalObject); |
| } |
| |
| static JSObject* pluginScriptObjectFromPluginViewBase(JSHTMLElement* jsHTMLElement) |
| { |
| HTMLElement& element = jsHTMLElement->wrapped(); |
| if (!is<HTMLPlugInElement>(element)) |
| return nullptr; |
| |
| HTMLPlugInElement& pluginElement = downcast<HTMLPlugInElement>(element); |
| return pluginScriptObjectFromPluginViewBase(pluginElement, jsHTMLElement->globalObject()); |
| } |
| |
| JSObject* pluginScriptObject(ExecState* exec, JSHTMLElement* jsHTMLElement) |
| { |
| HTMLElement& element = jsHTMLElement->wrapped(); |
| if (!is<HTMLPlugInElement>(element)) |
| return nullptr; |
| |
| auto& pluginElement = downcast<HTMLPlugInElement>(element); |
| |
| // Choke point for script/plugin interaction; notify DOMTimer of the event. |
| DOMTimer::scriptDidInteractWithPlugin(pluginElement); |
| |
| // First, see if the element has a plug-in replacement with a script. |
| if (auto* scriptObject = pluginElement.scriptObjectForPluginReplacement()) |
| return scriptObject; |
| |
| // Next, see if we can ask the plug-in view for its script object. |
| if (auto* scriptObject = pluginScriptObjectFromPluginViewBase(pluginElement, jsHTMLElement->globalObject())) |
| return scriptObject; |
| |
| // Otherwise, fall back to getting the object from the instance. |
| |
| // The plugin element holds an owning reference, so we don't have to. |
| auto* instance = pluginElement.bindingsInstance(); |
| if (!instance || !instance->rootObject()) |
| return nullptr; |
| |
| return instance->createRuntimeObject(exec); |
| } |
| |
| static EncodedJSValue pluginElementPropertyGetter(ExecState* exec, EncodedJSValue thisValue, PropertyName propertyName) |
| { |
| VM& vm = exec->vm(); |
| auto scope = DECLARE_THROW_SCOPE(vm); |
| |
| JSHTMLElement* thisObject = jsDynamicCast<JSHTMLElement*>(vm, JSValue::decode(thisValue)); |
| if (!thisObject) |
| return throwVMTypeError(exec, scope); |
| JSObject* scriptObject = pluginScriptObject(exec, thisObject); |
| if (!scriptObject) |
| return JSValue::encode(jsUndefined()); |
| |
| return JSValue::encode(scriptObject->get(exec, propertyName)); |
| } |
| |
| bool pluginElementCustomGetOwnPropertySlot(JSHTMLElement* element, ExecState* exec, PropertyName propertyName, PropertySlot& slot) |
| { |
| if (!element->globalObject()->world().isNormal()) { |
| JSC::JSValue proto = element->getPrototypeDirect(exec->vm()); |
| if (proto.isObject() && JSC::jsCast<JSC::JSObject*>(asObject(proto))->hasProperty(exec, propertyName)) |
| return false; |
| } |
| |
| JSObject* scriptObject = pluginScriptObject(exec, element); |
| if (!scriptObject) |
| return false; |
| |
| if (!scriptObject->hasProperty(exec, propertyName)) |
| return false; |
| |
| slot.setCustom(element, JSC::PropertyAttribute::DontDelete | JSC::PropertyAttribute::DontEnum, pluginElementPropertyGetter); |
| return true; |
| } |
| |
| bool pluginElementCustomPut(JSHTMLElement* element, ExecState* exec, PropertyName propertyName, JSValue value, PutPropertySlot& slot, bool& putResult) |
| { |
| JSObject* scriptObject = pluginScriptObject(exec, element); |
| if (!scriptObject) |
| return false; |
| if (!scriptObject->hasProperty(exec, propertyName)) |
| return false; |
| putResult = scriptObject->methodTable(exec->vm())->put(scriptObject, exec, propertyName, value, slot); |
| return true; |
| } |
| |
| static EncodedJSValue JSC_HOST_CALL callPlugin(JSGlobalObject* globalObject, CallFrame* callFrame) |
| { |
| JSHTMLElement* element = jsCast<JSHTMLElement*>(callFrame->jsCallee()); |
| |
| // Get the plug-in script object. |
| JSObject* scriptObject = pluginScriptObject(callFrame, element); |
| ASSERT(scriptObject); |
| |
| size_t argumentCount = callFrame->argumentCount(); |
| MarkedArgumentBuffer argumentList; |
| for (size_t i = 0; i < argumentCount; i++) |
| argumentList.append(callFrame->argument(i)); |
| ASSERT(!argumentList.hasOverflowed()); |
| |
| CallData callData; |
| CallType callType = getCallData(globalObject->vm(), scriptObject, callData); |
| ASSERT(callType == CallType::Host); |
| |
| // Call the object. |
| JSValue result = call(callFrame, scriptObject, callType, callData, callFrame->thisValue(), argumentList); |
| return JSValue::encode(result); |
| } |
| |
| CallType pluginElementCustomGetCallData(JSHTMLElement* element, CallData& callData) |
| { |
| // First, ask the plug-in view base for its runtime object. |
| if (JSObject* scriptObject = pluginScriptObjectFromPluginViewBase(element)) { |
| CallData scriptObjectCallData; |
| |
| VM& vm = scriptObject->vm(); |
| if (scriptObject->methodTable(vm)->getCallData(scriptObject, scriptObjectCallData) == CallType::None) |
| return CallType::None; |
| |
| callData.native.function = callPlugin; |
| return CallType::Host; |
| } |
| |
| Instance* instance = pluginInstance(element->wrapped()); |
| if (!instance || !instance->supportsInvokeDefaultMethod()) |
| return CallType::None; |
| callData.native.function = callPlugin; |
| return CallType::Host; |
| } |
| |
| } // namespace WebCore |