/* -*- 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 "nsIDOMHTMLAreaElement.h" #include "nsIDOMNSHTMLAreaElement.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 "nsIEventStateManager.h" #include "nsIURL.h" #include "nsNetUtil.h" static NS_DEFINE_IID(kIDOMHTMLAreaElementIID, NS_IDOMHTMLAREAELEMENT_IID); class nsHTMLAreaElement : public nsIDOMHTMLAreaElement, public nsIDOMNSHTMLAreaElement, public nsIJSScriptObject, public nsIHTMLContent { public: nsHTMLAreaElement(nsINodeInfo *aNodeInfo); virtual ~nsHTMLAreaElement(); // 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) // nsIDOMHTMLAreaElement NS_IMETHOD GetAccessKey(nsString& aAccessKey); NS_IMETHOD SetAccessKey(const nsString& aAccessKey); NS_IMETHOD GetAlt(nsString& aAlt); NS_IMETHOD SetAlt(const nsString& aAlt); NS_IMETHOD GetCoords(nsString& aCoords); NS_IMETHOD SetCoords(const nsString& aCoords); NS_IMETHOD GetHref(nsString& aHref); NS_IMETHOD SetHref(const nsString& aHref); NS_IMETHOD GetNoHref(PRBool* aNoHref); NS_IMETHOD SetNoHref(PRBool aNoHref); NS_IMETHOD GetShape(nsString& aShape); NS_IMETHOD SetShape(const nsString& aShape); NS_IMETHOD GetTabIndex(PRInt32* aTabIndex); NS_IMETHOD SetTabIndex(PRInt32 aTabIndex); NS_IMETHOD GetTarget(nsString& aTarget); NS_IMETHOD SetTarget(const nsString& aTarget); // nsIDOMNSHTMLAreaElement NS_IMETHOD GetProtocol(nsString& aProtocol); NS_IMETHOD GetHost(nsString& aHost); NS_IMETHOD GetHostname(nsString& aHostname); NS_IMETHOD GetPathname(nsString& aPathname); NS_IMETHOD GetSearch(nsString& aSearch); NS_IMETHOD GetPort(nsString& aPort); NS_IMETHOD GetHash(nsString& aHash); // nsIJSScriptObject NS_IMPL_IJSSCRIPTOBJECT_USING_GENERIC(mInner) // nsIContent NS_IMPL_ICONTENT_NO_FOCUS_USING_GENERIC(mInner) // nsIHTMLContent NS_IMPL_IHTMLCONTENT_USING_GENERIC(mInner) protected: nsGenericHTMLLeafElement mInner; }; nsresult NS_NewHTMLAreaElement(nsIHTMLContent** aInstancePtrResult, nsINodeInfo *aNodeInfo) { NS_ENSURE_ARG_POINTER(aInstancePtrResult); NS_ENSURE_ARG_POINTER(aNodeInfo); nsIHTMLContent* it = new nsHTMLAreaElement(aNodeInfo); if (nsnull == it) { return NS_ERROR_OUT_OF_MEMORY; } return it->QueryInterface(kIHTMLContentIID, (void**) aInstancePtrResult); } nsHTMLAreaElement::nsHTMLAreaElement(nsINodeInfo *aNodeInfo) { NS_INIT_REFCNT(); mInner.Init(this, aNodeInfo); } nsHTMLAreaElement::~nsHTMLAreaElement() { } NS_IMPL_ADDREF(nsHTMLAreaElement) NS_IMPL_RELEASE(nsHTMLAreaElement) nsresult nsHTMLAreaElement::QueryInterface(REFNSIID aIID, void** aInstancePtr) { NS_IMPL_HTML_CONTENT_QUERY_INTERFACE(aIID, aInstancePtr, this) if (aIID.Equals(kIDOMHTMLAreaElementIID)) { nsIDOMHTMLAreaElement* tmp = this; *aInstancePtr = (void*) tmp; NS_ADDREF_THIS(); return NS_OK; } else if (aIID.Equals(NS_GET_IID(nsIDOMNSHTMLAreaElement))) { nsIDOMNSHTMLAreaElement* tmp = this; *aInstancePtr = (void*) tmp; NS_ADDREF_THIS(); return NS_OK; } return NS_NOINTERFACE; } nsresult nsHTMLAreaElement::CloneNode(PRBool aDeep, nsIDOMNode** aReturn) { nsHTMLAreaElement* it = new nsHTMLAreaElement(mInner.mNodeInfo); if (nsnull == it) { return NS_ERROR_OUT_OF_MEMORY; } mInner.CopyInnerTo(this, &it->mInner, aDeep); return it->QueryInterface(kIDOMNodeIID, (void**) aReturn); } NS_IMPL_STRING_ATTR(nsHTMLAreaElement, AccessKey, accesskey) NS_IMPL_STRING_ATTR(nsHTMLAreaElement, Alt, alt) NS_IMPL_STRING_ATTR(nsHTMLAreaElement, Coords, coords) NS_IMPL_STRING_ATTR(nsHTMLAreaElement, Href, href) NS_IMPL_BOOL_ATTR(nsHTMLAreaElement, NoHref, nohref) NS_IMPL_STRING_ATTR(nsHTMLAreaElement, Shape, shape) NS_IMPL_INT_ATTR(nsHTMLAreaElement, TabIndex, tabindex) NS_IMPL_STRING_ATTR(nsHTMLAreaElement, Target, target) NS_IMETHODIMP nsHTMLAreaElement::StringToAttribute(nsIAtom* aAttribute, const nsString& aValue, nsHTMLValue& aResult) { if (aAttribute == nsHTMLAtoms::nohref) { aResult.SetEmptyValue(); return NS_CONTENT_ATTR_HAS_VALUE; } else if (aAttribute == nsHTMLAtoms::tabindex) { if (nsGenericHTMLElement::ParseValue(aValue, 0, aResult, eHTMLUnit_Integer)) { return NS_CONTENT_ATTR_HAS_VALUE; } } return NS_CONTENT_ATTR_NOT_THERE; } NS_IMETHODIMP nsHTMLAreaElement::AttributeToString(nsIAtom* aAttribute, const nsHTMLValue& aValue, nsString& aResult) const { return mInner.AttributeToString(aAttribute, aValue, aResult); } static void MapAttributesInto(const nsIHTMLMappedAttributes* aAttributes, nsIMutableStyleContext* aContext, nsIPresContext* aPresContext) { nsGenericHTMLElement::MapCommonAttributesInto(aAttributes, aContext, aPresContext); } NS_IMETHODIMP nsHTMLAreaElement::GetMappedAttributeImpact(const nsIAtom* aAttribute, PRInt32& aHint) const { if (! nsGenericHTMLElement::GetCommonMappedAttributesImpact(aAttribute, aHint)) { aHint = NS_STYLE_HINT_CONTENT; } return NS_OK; } NS_IMETHODIMP nsHTMLAreaElement::GetAttributeMappingFunctions(nsMapAttributesFunc& aFontMapFunc, nsMapAttributesFunc& aMapFunc) const { aFontMapFunc = nsnull; aMapFunc = &MapAttributesInto; return NS_OK; } NS_IMETHODIMP nsHTMLAreaElement::HandleDOMEvent(nsIPresContext* aPresContext, nsEvent* aEvent, nsIDOMEvent** aDOMEvent, PRUint32 aFlags, nsEventStatus* aEventStatus) { return mInner.HandleDOMEventForAnchors(this, aPresContext, aEvent, aDOMEvent, aFlags, aEventStatus); } NS_IMETHODIMP nsHTMLAreaElement::SetFocus(nsIPresContext* aPresContext) { nsIEventStateManager* esm; if (NS_OK == aPresContext->GetEventStateManager(&esm)) { esm->SetContentState(this, NS_EVENT_STATE_FOCUS); NS_RELEASE(esm); } // XXX write me return NS_OK; } NS_IMETHODIMP nsHTMLAreaElement::RemoveFocus(nsIPresContext* aPresContext) { // XXX write me return NS_OK; } NS_IMETHODIMP nsHTMLAreaElement::SizeOf(nsISizeOfHandler* aSizer, PRUint32* aResult) const { return mInner.SizeOf(aSizer, aResult, sizeof(*this)); } NS_IMETHODIMP nsHTMLAreaElement::GetProtocol(nsString& aProtocol) { nsAutoString href; nsIURI *url; nsresult result = NS_OK; result = GetHref(href); if (NS_OK == result) { result = NS_NewURI(&url, href); if (NS_OK == result) { char* protocol; result = url->GetScheme(&protocol); if (result == NS_OK) { aProtocol.AssignWithConversion(protocol); aProtocol.AppendWithConversion(":"); nsCRT::free(protocol); } NS_RELEASE(url); } } return result; } NS_IMETHODIMP nsHTMLAreaElement::GetHost(nsString& aHost) { nsAutoString href; nsIURI *url; nsresult result = NS_OK; result = GetHref(href); if (NS_OK == result) { result = NS_NewURI(&url, href); if (NS_OK == result) { char* host; result = url->GetHost(&host); if (result == NS_OK) { aHost.AssignWithConversion(host); nsCRT::free(host); PRInt32 port; (void)url->GetPort(&port); if (-1 != port) { aHost.AppendWithConversion(":"); aHost.AppendInt(port, 10); } } NS_RELEASE(url); } } return result; } NS_IMETHODIMP nsHTMLAreaElement::GetHostname(nsString& aHostname) { nsAutoString href; nsIURI *url; nsresult result = NS_OK; result = GetHref(href); if (NS_OK == result) { result = NS_NewURI(&url, href); if (NS_OK == result) { char* host; result = url->GetHost(&host); if (result == NS_OK) { aHostname.AssignWithConversion(host); nsCRT::free(host); } NS_RELEASE(url); } } return result; } NS_IMETHODIMP nsHTMLAreaElement::GetPathname(nsString& aPathname) { nsAutoString href; nsIURI *url; nsresult result = NS_OK; result = GetHref(href); if (NS_OK == result) { result = NS_NewURI(&url, href); if (NS_OK == result) { char* file; result = url->GetPath(&file); if (result == NS_OK) { aPathname.AssignWithConversion(file); nsCRT::free(file); } NS_IF_RELEASE(url); } } return result; } NS_IMETHODIMP nsHTMLAreaElement::GetSearch(nsString& aSearch) { nsAutoString href; nsIURI *uri; nsresult result = NS_OK; result = GetHref(href); if (NS_OK == result) { result = NS_NewURI(&uri, href); if (NS_OK == result) { char *search; nsIURL* url; result = uri->QueryInterface(NS_GET_IID(nsIURL), (void**)&url); if (NS_SUCCEEDED(result)) { result = url->GetQuery(&search); NS_RELEASE(url); } if (result == NS_OK && (nsnull != search) && ('\0' != *search)) { aSearch.AssignWithConversion("?"); aSearch.AppendWithConversion(search); nsCRT::free(search); } else { aSearch.SetLength(0); } NS_RELEASE(uri); } } return result; } NS_IMETHODIMP nsHTMLAreaElement::GetPort(nsString& aPort) { nsAutoString href; nsIURI *url; nsresult result = NS_OK; result = GetHref(href); if (NS_OK == result) { result = NS_NewURI(&url, href); if (NS_OK == result) { aPort.SetLength(0); PRInt32 port; (void)url->GetPort(&port); if (-1 != port) { aPort.AppendInt(port, 10); } NS_RELEASE(url); } } return result; } NS_IMETHODIMP nsHTMLAreaElement::GetHash(nsString& aHash) { nsAutoString href; nsIURI *uri; nsresult result = NS_OK; result = GetHref(href); if (NS_OK == result) { result = NS_NewURI(&uri, href); if (NS_OK == result) { char *ref; nsIURL* url; result = uri->QueryInterface(NS_GET_IID(nsIURL), (void**)&url); if (NS_SUCCEEDED(result)) { result = url->GetRef(&ref); NS_RELEASE(url); } if (result == NS_OK && (nsnull != ref) && ('\0' != *ref)) { aHash.AssignWithConversion("#"); aHash.AppendWithConversion(ref); nsCRT::free(ref); } else { aHash.SetLength(0); } NS_RELEASE(uri); } } return result; }