514 lines
16 KiB
C++
514 lines
16 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
*
|
|
* The contents of this file are subject to the Netscape Public
|
|
* License Version 1.1 (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.mozilla.org/NPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS
|
|
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
|
|
* implied. See the License for the specific language governing
|
|
* rights and limitations under the License.
|
|
*
|
|
* The Original Code is Mozilla Communicator client code.
|
|
*
|
|
* The Initial Developer of the Original Code is Netscape Communications
|
|
* Corporation. Portions created by Netscape are
|
|
* Copyright (C) 1998 Netscape Communications Corporation. All
|
|
* Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
*/
|
|
#include "nsIDOMHTMLElement.h"
|
|
#include "nsIScriptObjectOwner.h"
|
|
#include "nsIDOMEventReceiver.h"
|
|
#include "nsIHTMLContent.h"
|
|
#include "nsGenericHTMLElement.h"
|
|
#include "nsHTMLAtoms.h"
|
|
#include "nsHTMLIIDs.h"
|
|
#include "nsIStyleContext.h"
|
|
#include "nsIMutableStyleContext.h"
|
|
#include "nsStyleConsts.h"
|
|
#include "nsIPresContext.h"
|
|
#include "nsIPresShell.h"
|
|
#include "nsIDocument.h"
|
|
#include "nsIFrame.h"
|
|
#include "nsObjectFrame.h"
|
|
#include "nsLayoutAtoms.h"
|
|
#include "nsObjectFrame.h"
|
|
#include "nsCOMPtr.h"
|
|
#include "nsMemory.h"
|
|
#include "xptinfo.h"
|
|
#include "nsIInterfaceInfoManager.h"
|
|
#include "nsIPluginInstance.h"
|
|
#include "nsIScriptablePlugin.h"
|
|
#include "nsIXPConnect.h"
|
|
#include "nsIServiceManager.h"
|
|
#include "nsIDOMHTMLEmbedElement.h"
|
|
|
|
class nsHTMLEmbedElement : public nsIDOMHTMLEmbedElement,
|
|
public nsIJSScriptObject,
|
|
public nsIHTMLContent
|
|
{
|
|
public:
|
|
nsHTMLEmbedElement(nsINodeInfo *aNodeInfo);
|
|
virtual ~nsHTMLEmbedElement();
|
|
|
|
// nsISupports
|
|
NS_DECL_ISUPPORTS
|
|
|
|
// nsIDOMNode
|
|
NS_IMPL_IDOMNODE_USING_GENERIC(mInner)
|
|
|
|
// nsIDOMElement
|
|
NS_IMPL_IDOMELEMENT_USING_GENERIC(mInner)
|
|
|
|
// nsIDOMHTMLElement
|
|
NS_IMPL_IDOMHTMLELEMENT_USING_GENERIC(mInner)
|
|
|
|
// nsIDOMHTMLEmbedElement
|
|
NS_DECL_IDOMHTMLEMBEDELEMENT
|
|
|
|
// nsIJSScriptObject
|
|
NS_IMETHOD GetScriptObject(nsIScriptContext* aContext,
|
|
void** aScriptObject);
|
|
NS_IMETHOD SetScriptObject(void *aScriptObject);
|
|
|
|
virtual PRBool AddProperty(JSContext *aContext, JSObject *aObj,
|
|
jsval aID, jsval *aVp);
|
|
virtual PRBool DeleteProperty(JSContext *aContext, JSObject *aObj,
|
|
jsval aID, jsval *aVp);
|
|
virtual PRBool GetProperty(JSContext *aContext, JSObject *aObj,
|
|
jsval aID, jsval *aVp);
|
|
virtual PRBool SetProperty(JSContext *aContext, JSObject *aObj,
|
|
jsval aID, jsval *aVp);
|
|
virtual PRBool EnumerateProperty(JSContext *aContext, JSObject *aObj);
|
|
virtual PRBool Resolve(JSContext *aContext, JSObject *aObj, jsval aID);
|
|
virtual PRBool Convert(JSContext *aContext, JSObject *aObj, jsval aID);
|
|
virtual void Finalize(JSContext *aContext, JSObject *aObj);
|
|
|
|
// nsIContent
|
|
NS_IMPL_ICONTENT_USING_GENERIC(mInner)
|
|
|
|
// nsIHTMLContent
|
|
NS_IMPL_IHTMLCONTENT_USING_GENERIC(mInner)
|
|
|
|
protected:
|
|
nsresult GetPluginInstance(nsIPluginInstance** aPluginInstance);
|
|
|
|
protected:
|
|
nsGenericHTMLLeafElement mInner;
|
|
};
|
|
|
|
nsresult
|
|
NS_NewHTMLEmbedElement(nsIHTMLContent** aInstancePtrResult,
|
|
nsINodeInfo *aNodeInfo)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aInstancePtrResult);
|
|
NS_ENSURE_ARG_POINTER(aNodeInfo);
|
|
|
|
nsIHTMLContent* it = new nsHTMLEmbedElement(aNodeInfo);
|
|
if (nsnull == it) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
return it->QueryInterface(kIHTMLContentIID, (void**) aInstancePtrResult);
|
|
}
|
|
|
|
nsHTMLEmbedElement::nsHTMLEmbedElement(nsINodeInfo *aNodeInfo)
|
|
{
|
|
NS_INIT_REFCNT();
|
|
mInner.Init(this, aNodeInfo);
|
|
}
|
|
|
|
nsHTMLEmbedElement::~nsHTMLEmbedElement()
|
|
{
|
|
}
|
|
|
|
NS_IMPL_ADDREF(nsHTMLEmbedElement)
|
|
NS_IMPL_RELEASE(nsHTMLEmbedElement)
|
|
|
|
nsresult
|
|
nsHTMLEmbedElement::QueryInterface(REFNSIID aIID, void** aInstancePtr)
|
|
{
|
|
NS_IMPL_HTML_CONTENT_QUERY_INTERFACE(aIID, aInstancePtr, this)
|
|
if (aIID.Equals(NS_GET_IID(nsIDOMHTMLEmbedElement))) {
|
|
nsIDOMHTMLEmbedElement* tmp = this;
|
|
*aInstancePtr = (void*) tmp;
|
|
NS_ADDREF_THIS();
|
|
return NS_OK;
|
|
}
|
|
|
|
return NS_NOINTERFACE;
|
|
}
|
|
|
|
nsresult
|
|
nsHTMLEmbedElement::CloneNode(PRBool aDeep, nsIDOMNode** aReturn)
|
|
{
|
|
nsHTMLEmbedElement* it = new nsHTMLEmbedElement(mInner.mNodeInfo);
|
|
if (nsnull == it) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
mInner.CopyInnerTo(this, &it->mInner, aDeep);
|
|
return it->QueryInterface(kIDOMNodeIID, (void**) aReturn);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEmbedElement::StringToAttribute(nsIAtom* aAttribute,
|
|
const nsString& aValue,
|
|
nsHTMLValue& aResult)
|
|
{
|
|
if (aAttribute == nsHTMLAtoms::align) {
|
|
if (nsGenericHTMLElement::ParseAlignValue(aValue, aResult)) {
|
|
return NS_CONTENT_ATTR_HAS_VALUE;
|
|
}
|
|
}
|
|
else if (nsGenericHTMLElement::ParseImageAttribute(aAttribute,
|
|
aValue, aResult)) {
|
|
return NS_CONTENT_ATTR_HAS_VALUE;
|
|
}
|
|
return NS_CONTENT_ATTR_NOT_THERE;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEmbedElement::AttributeToString(nsIAtom* aAttribute,
|
|
const nsHTMLValue& aValue,
|
|
nsString& aResult) const
|
|
{
|
|
if (aAttribute == nsHTMLAtoms::align) {
|
|
if (eHTMLUnit_Enumerated == aValue.GetUnit()) {
|
|
nsGenericHTMLElement::AlignValueToString(aValue, aResult);
|
|
return NS_CONTENT_ATTR_HAS_VALUE;
|
|
}
|
|
}
|
|
else if (nsGenericHTMLElement::ImageAttributeToString(aAttribute,
|
|
aValue, aResult)) {
|
|
return NS_CONTENT_ATTR_HAS_VALUE;
|
|
}
|
|
return mInner.AttributeToString(aAttribute, aValue, aResult);
|
|
}
|
|
|
|
static void
|
|
MapAttributesInto(const nsIHTMLMappedAttributes* aAttributes,
|
|
nsIMutableStyleContext* aContext,
|
|
nsIPresContext* aPresContext)
|
|
{
|
|
nsGenericHTMLElement::MapImageAlignAttributeInto(aAttributes, aContext, aPresContext);
|
|
nsGenericHTMLElement::MapImageAttributesInto(aAttributes, aContext, aPresContext);
|
|
nsGenericHTMLElement::MapImageBorderAttributeInto(aAttributes, aContext, aPresContext, nsnull);
|
|
nsGenericHTMLElement::MapCommonAttributesInto(aAttributes, aContext, aPresContext);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEmbedElement::GetMappedAttributeImpact(const nsIAtom* aAttribute,
|
|
PRInt32& aHint) const
|
|
{
|
|
if (! nsGenericHTMLElement::GetCommonMappedAttributesImpact(aAttribute, aHint)) {
|
|
if (! nsGenericHTMLElement::GetImageMappedAttributesImpact(aAttribute, aHint)) {
|
|
if (! nsGenericHTMLElement::GetImageAlignAttributeImpact(aAttribute, aHint)) {
|
|
if (! nsGenericHTMLElement::GetImageBorderAttributeImpact(aAttribute, aHint)) {
|
|
aHint = NS_STYLE_HINT_CONTENT;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEmbedElement::GetAttributeMappingFunctions(nsMapAttributesFunc& aFontMapFunc,
|
|
nsMapAttributesFunc& aMapFunc) const
|
|
{
|
|
aFontMapFunc = nsnull;
|
|
aMapFunc = &MapAttributesInto;
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEmbedElement::HandleDOMEvent(nsIPresContext* aPresContext,
|
|
nsEvent* aEvent,
|
|
nsIDOMEvent** aDOMEvent,
|
|
PRUint32 aFlags,
|
|
nsEventStatus* aEventStatus)
|
|
{
|
|
return mInner.HandleDOMEvent(aPresContext, aEvent, aDOMEvent,
|
|
aFlags, aEventStatus);
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLEmbedElement::SizeOf(nsISizeOfHandler* aSizer, PRUint32* aResult) const
|
|
{
|
|
return mInner.SizeOf(aSizer, aResult, sizeof(*this));
|
|
}
|
|
|
|
/***************************************************************************/
|
|
|
|
// This is cribbed from nsHTMLImageElement::GetImageFrame.
|
|
|
|
nsresult
|
|
nsHTMLEmbedElement::GetPluginInstance(nsIPluginInstance** aPluginInstance)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aPluginInstance);
|
|
*aPluginInstance = nsnull;
|
|
|
|
nsresult result;
|
|
nsCOMPtr<nsIPresContext> context;
|
|
nsCOMPtr<nsIPresShell> shell;
|
|
|
|
if (mInner.mDocument) {
|
|
// Make sure the presentation is up-to-date
|
|
result = mInner.mDocument->FlushPendingNotifications();
|
|
if (NS_FAILED(result)) {
|
|
return result;
|
|
}
|
|
}
|
|
|
|
result = nsGenericHTMLElement::GetPresContext(this,
|
|
getter_AddRefs(context));
|
|
if (NS_FAILED(result)) {
|
|
return result;
|
|
}
|
|
|
|
result = context->GetShell(getter_AddRefs(shell));
|
|
if (NS_FAILED(result)) {
|
|
return result;
|
|
}
|
|
|
|
nsIFrame* frame;
|
|
result = shell->GetPrimaryFrameFor(this, &frame);
|
|
if (NS_FAILED(result)) {
|
|
return result;
|
|
}
|
|
|
|
if (frame) {
|
|
nsCOMPtr<nsIAtom> type;
|
|
|
|
frame->GetFrameType(getter_AddRefs(type));
|
|
|
|
if (type.get() == nsLayoutAtoms::objectFrame) {
|
|
// XXX We could have created an interface for this, but Troy
|
|
// preferred the ugliness of a static cast to the weight of
|
|
// a new interface.
|
|
|
|
nsObjectFrame* objectFrame = NS_STATIC_CAST(nsObjectFrame*, frame);
|
|
|
|
return objectFrame->GetPluginInstance(*aPluginInstance);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
/***************************************************************************/
|
|
|
|
/*
|
|
* For plugins, we want to expose both attributes of the plugin tag
|
|
* and any scriptable methods that the plugin itself exposes. To do
|
|
* this, we get the plugin object itself (the XPCOM object) and wrap
|
|
* it as a scriptable object via xpconnect. We then set the original
|
|
* node element, which exposes the DOM node methods, as the javascript
|
|
* prototype object of that object. Then we get both sets of methods, and
|
|
* plugin methods can potentially override DOM methods.
|
|
*/
|
|
NS_IMETHODIMP
|
|
nsHTMLEmbedElement::GetScriptObject(nsIScriptContext* aContext,
|
|
void** aScriptObject)
|
|
{
|
|
if (mInner.mDOMSlots && mInner.mDOMSlots->mScriptObject)
|
|
return mInner.GetScriptObject(aContext, aScriptObject);
|
|
|
|
nsresult rv;
|
|
*aScriptObject = nsnull;
|
|
|
|
// Get the JS object corresponding to this dom node. This will become
|
|
// the javascript prototype object of the object we eventually reflect to the
|
|
// DOM.
|
|
JSObject* elementObject = nsnull;
|
|
rv = mInner.GetScriptObject(aContext, (void**)&elementObject);
|
|
if (NS_FAILED(rv) || !elementObject)
|
|
return rv;
|
|
|
|
// Flush pending reflows to ensure the plugin is instansiated, assuming
|
|
// it's visible
|
|
if (mInner.mDocument) {
|
|
mInner.mDocument->FlushPendingNotifications();
|
|
}
|
|
|
|
nsCOMPtr<nsIPluginInstance> pi;
|
|
rv = GetPluginInstance(getter_AddRefs(pi));
|
|
|
|
// If GetPluginInstance() fails it means there's no frame for this element
|
|
// yet, in that case we return the script object for the element but we
|
|
// don't cache it so that the next call can get the correct script object
|
|
// if the plugin instance is available at the next call.
|
|
if (NS_FAILED(rv)) {
|
|
mInner.SetScriptObject(nsnull);
|
|
|
|
*aScriptObject = elementObject;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
// Check if the plugin object has the nsIScriptablePlugin
|
|
// interface, describing how to expose it to JavaScript. Given this
|
|
// interface, use it to get the scriptable peer object (possibly the
|
|
// plugin object itself) and the scriptable interface to expose it
|
|
// with
|
|
nsIID scriptableInterface;
|
|
nsCOMPtr<nsISupports> scriptablePeer;
|
|
if (NS_SUCCEEDED(rv) && pi) {
|
|
nsCOMPtr<nsIScriptablePlugin> spi(do_QueryInterface(pi, &rv));
|
|
if (NS_SUCCEEDED(rv) && spi) {
|
|
nsIID *scriptableInterfacePtr = nsnull;
|
|
rv = spi->GetScriptableInterface(&scriptableInterfacePtr);
|
|
|
|
if (NS_SUCCEEDED(rv) && scriptableInterfacePtr) {
|
|
rv = spi->GetScriptablePeer(getter_AddRefs(scriptablePeer));
|
|
|
|
scriptableInterface = *scriptableInterfacePtr;
|
|
|
|
nsMemory::Free(scriptableInterfacePtr);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (NS_FAILED(rv) || !scriptablePeer) {
|
|
// Fall back to returning the element object.
|
|
*aScriptObject = elementObject;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
// Wrap it.
|
|
JSObject* interfaceObject; // XPConnect-wrapped peer object, when we get it.
|
|
JSContext *cx = (JSContext *)aContext->GetNativeContext();
|
|
nsCOMPtr<nsIXPConnect> xpc =
|
|
do_GetService(nsIXPConnect::GetCID());
|
|
if (cx && xpc) {
|
|
JSObject* parentObject = JS_GetParent(cx, elementObject);
|
|
nsCOMPtr<nsIXPConnectJSObjectHolder> holder;
|
|
if (NS_SUCCEEDED(xpc->WrapNative(cx, parentObject,
|
|
scriptablePeer, scriptableInterface,
|
|
getter_AddRefs(holder))) && holder &&
|
|
NS_SUCCEEDED(holder->GetJSObject(&interfaceObject)) && interfaceObject) {
|
|
*aScriptObject = interfaceObject;
|
|
}
|
|
}
|
|
|
|
// If we got an xpconnect-wrapped plugin object, set its' prototype to the
|
|
// element object.
|
|
if (!*aScriptObject || !JS_SetPrototype(cx, interfaceObject, elementObject)) {
|
|
*aScriptObject = elementObject; // fall back
|
|
return NS_OK;
|
|
}
|
|
|
|
// Cache it.
|
|
mInner.SetScriptObject(*aScriptObject);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
// TODO: if this method ever gets called, it will destroy the
|
|
// prototype chain.
|
|
NS_IMETHODIMP
|
|
nsHTMLEmbedElement::SetScriptObject(void *aScriptObject)
|
|
{
|
|
return mInner.SetScriptObject(aScriptObject);
|
|
}
|
|
|
|
// nsIJSScriptObject
|
|
|
|
PRBool
|
|
nsHTMLEmbedElement::AddProperty(JSContext *aContext, JSObject *aObj, jsval aID, jsval *aVp)
|
|
{
|
|
return mInner.AddProperty(aContext, aObj, aID, aVp);
|
|
}
|
|
|
|
PRBool
|
|
nsHTMLEmbedElement::DeleteProperty(JSContext *aContext, JSObject *aObj, jsval aID, jsval *aVp)
|
|
{
|
|
return mInner.DeleteProperty(aContext, aObj, aID, aVp);
|
|
}
|
|
|
|
// Allow access to arbitrary XPCOM interfaces supported by the plugin
|
|
// via a pluginObject.nsISomeInterface notation.
|
|
PRBool
|
|
nsHTMLEmbedElement::GetProperty(JSContext *aContext, JSObject *aObj, jsval aID, jsval *aVp)
|
|
{
|
|
if (JSVAL_IS_STRING(aID)) {
|
|
PRBool retval = PR_FALSE;
|
|
char* cString = JS_GetStringBytes(JS_ValueToString(aContext, aID));
|
|
|
|
nsCOMPtr<nsIInterfaceInfoManager> iim =
|
|
dont_AddRef(XPTI_GetInterfaceInfoManager());
|
|
nsCOMPtr<nsIXPConnect> xpc =
|
|
do_GetService(nsIXPConnect::GetCID());
|
|
|
|
if (iim && xpc) {
|
|
nsIID* iid;
|
|
if (NS_SUCCEEDED(iim->GetIIDForName(cString, &iid)) && iid) {
|
|
nsCOMPtr<nsIPluginInstance> pi;
|
|
if (NS_SUCCEEDED(GetPluginInstance(getter_AddRefs(pi))) && pi) {
|
|
nsCOMPtr<nsIXPConnectJSObjectHolder> holder;
|
|
JSObject* ifaceObj;
|
|
|
|
if (NS_SUCCEEDED(xpc->WrapNative(aContext, aObj, pi, *iid,
|
|
getter_AddRefs(holder))) && holder &&
|
|
NS_SUCCEEDED(holder->GetJSObject(&ifaceObj)) && ifaceObj) {
|
|
*aVp = OBJECT_TO_JSVAL(ifaceObj);
|
|
retval = PR_TRUE;
|
|
}
|
|
}
|
|
nsMemory::Free(iid);
|
|
return retval;
|
|
}
|
|
}
|
|
}
|
|
|
|
return mInner.GetProperty(aContext, aObj, aID, aVp);
|
|
}
|
|
|
|
PRBool
|
|
nsHTMLEmbedElement::SetProperty(JSContext *aContext, JSObject *aObj, jsval aID, jsval *aVp)
|
|
{
|
|
return mInner.SetProperty(aContext, aObj, aID, aVp);
|
|
}
|
|
|
|
PRBool
|
|
nsHTMLEmbedElement::EnumerateProperty(JSContext *aContext, JSObject *aObj)
|
|
{
|
|
return mInner.EnumerateProperty(aContext, aObj);
|
|
}
|
|
|
|
PRBool
|
|
nsHTMLEmbedElement::Resolve(JSContext *aContext, JSObject *aObj, jsval aID)
|
|
{
|
|
return mInner.Resolve(aContext, aObj, aID);
|
|
}
|
|
|
|
PRBool
|
|
nsHTMLEmbedElement::Convert(JSContext *aContext, JSObject *aObj, jsval aID)
|
|
{
|
|
return mInner.Convert(aContext, aObj, aID);
|
|
}
|
|
|
|
void
|
|
nsHTMLEmbedElement::Finalize(JSContext *aContext, JSObject *aObj)
|
|
{
|
|
mInner.Finalize(aContext, aObj);
|
|
}
|
|
|
|
/////////////////////////////////////////////
|
|
// Implement nsIDOMHTMLEmbedElement interface
|
|
NS_IMPL_STRING_ATTR(nsHTMLEmbedElement, Align, align)
|
|
NS_IMPL_STRING_ATTR(nsHTMLEmbedElement, Height, height)
|
|
NS_IMPL_STRING_ATTR(nsHTMLEmbedElement, Width, width)
|
|
NS_IMPL_STRING_ATTR(nsHTMLEmbedElement, Name, name)
|
|
NS_IMPL_STRING_ATTR(nsHTMLEmbedElement, Type, type)
|
|
NS_IMPL_STRING_ATTR(nsHTMLEmbedElement, Src, src)
|
|
|