mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-06 09:05:45 +00:00
390 lines
12 KiB
C++
390 lines
12 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla 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/MPL/
|
|
*
|
|
* 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 XForms support.
|
|
*
|
|
* The Initial Developer of the Original Code is IBM Corporation.
|
|
* Portions created by the Initial Developer are Copyright (C) 2004
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* Darin Fisher <darin@meer.net>
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
#include "nsIXTFXMLVisual.h"
|
|
#include "nsIDOM3Node.h"
|
|
#include "nsIDOMElement.h"
|
|
#include "nsIDOMUIEvent.h"
|
|
#include "nsIDOMEventTarget.h"
|
|
#include "nsIDOMDocumentEvent.h"
|
|
#include "nsIDOMDocument.h"
|
|
#include "nsIDOMHTMLElement.h"
|
|
#include "nsIDOMHTMLButtonElement.h"
|
|
#include "nsIDOMDocumentView.h"
|
|
#include "nsIDOMAbstractView.h"
|
|
#include "nsCOMPtr.h"
|
|
#include "nsString.h"
|
|
#include "nsIXTFXMLVisualWrapper.h"
|
|
#include "nsXFormsAtoms.h"
|
|
#include "nsXFormsUtils.h"
|
|
#include "nsXFormsControlStub.h"
|
|
#include "nsIXFormsSubmitElement.h"
|
|
#include "nsIXFormsSubmissionElement.h"
|
|
|
|
class nsXFormsTriggerElement : public nsXFormsControlStub
|
|
{
|
|
public:
|
|
// nsIXTFXMLVisual overrides
|
|
NS_IMETHOD OnCreated(nsIXTFXMLVisualWrapper *aWrapper);
|
|
|
|
// nsIXTFVisual overrides
|
|
NS_IMETHOD GetVisualContent(nsIDOMElement **aElement);
|
|
NS_IMETHOD GetInsertionPoint(nsIDOMElement **aElement);
|
|
|
|
// nsIXTFElement overrides
|
|
NS_IMETHOD BeginAddingChildren();
|
|
NS_IMETHOD DoneAddingChildren();
|
|
NS_IMETHOD AttributeRemoved(nsIAtom *aName);
|
|
NS_IMETHOD AttributeSet(nsIAtom *aName, const nsAString &aValue);
|
|
|
|
// nsIXFormsControl
|
|
NS_IMETHOD HandleDefault(nsIDOMEvent *aEvent, PRBool *aHandled);
|
|
NS_IMETHOD Refresh();
|
|
NS_IMETHOD TryFocus(PRBool* aOK);
|
|
|
|
// nsXFormsTriggerElement
|
|
nsXFormsTriggerElement() :
|
|
mIsMinimal(PR_FALSE), mDoneAddingChildren(PR_TRUE) {}
|
|
|
|
protected:
|
|
nsCOMPtr<nsIDOMElement> mVisualContent;
|
|
nsCOMPtr<nsIDOMHTMLElement> mHTMLElement;
|
|
nsCOMPtr<nsIDOMHTMLElement> mInsertionPoint;
|
|
PRPackedBool mIsMinimal;
|
|
PRPackedBool mDoneAddingChildren;
|
|
};
|
|
|
|
// nsIXTFXMLVisual
|
|
|
|
NS_IMETHODIMP
|
|
nsXFormsTriggerElement::OnCreated(nsIXTFXMLVisualWrapper *aWrapper)
|
|
{
|
|
nsresult rv = nsXFormsControlStub::OnCreated(aWrapper);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
aWrapper->SetNotificationMask(kStandardNotificationMask |
|
|
nsIXTFElement::NOTIFY_HANDLE_DEFAULT |
|
|
nsIXTFElement::NOTIFY_BEGIN_ADDING_CHILDREN |
|
|
nsIXTFElement::NOTIFY_DONE_ADDING_CHILDREN |
|
|
nsIXTFElement::NOTIFY_ATTRIBUTE_REMOVED);
|
|
|
|
ResetHelpAndHint(PR_TRUE);
|
|
|
|
nsCOMPtr<nsIDOMDocument> domDoc;
|
|
mElement->GetOwnerDocument(getter_AddRefs(domDoc));
|
|
NS_ENSURE_STATE(domDoc);
|
|
|
|
// Create visual (span + button)
|
|
domDoc->CreateElementNS(NS_LITERAL_STRING(NS_NAMESPACE_XHTML),
|
|
NS_LITERAL_STRING("span"),
|
|
getter_AddRefs(mVisualContent));
|
|
|
|
nsCOMPtr<nsIDOMElement> button;
|
|
domDoc->CreateElementNS(NS_LITERAL_STRING(NS_NAMESPACE_XHTML),
|
|
NS_LITERAL_STRING("button"),
|
|
getter_AddRefs(button));
|
|
|
|
nsCOMPtr<nsIDOMNode> insertedButton;
|
|
mVisualContent->AppendChild(button, getter_AddRefs(insertedButton));
|
|
mHTMLElement = do_QueryInterface(insertedButton);
|
|
NS_ENSURE_STATE(mHTMLElement);
|
|
|
|
// Create insertion point
|
|
nsCOMPtr<nsIDOMElement> span;
|
|
domDoc->CreateElementNS(NS_LITERAL_STRING(NS_NAMESPACE_XHTML),
|
|
NS_LITERAL_STRING("span"),
|
|
getter_AddRefs(span));
|
|
NS_ENSURE_STATE(span);
|
|
|
|
nsCOMPtr<nsIDOMNode> insertedSpan;
|
|
rv = mHTMLElement->AppendChild(span, getter_AddRefs(insertedSpan));
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
mInsertionPoint = do_QueryInterface(insertedSpan);
|
|
NS_ENSURE_STATE(mInsertionPoint);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
// nsIXTFVisual
|
|
|
|
NS_IMETHODIMP
|
|
nsXFormsTriggerElement::GetVisualContent(nsIDOMElement **aElement)
|
|
{
|
|
NS_ADDREF(*aElement = mVisualContent);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsXFormsTriggerElement::GetInsertionPoint(nsIDOMElement **aElement)
|
|
{
|
|
NS_ADDREF(*aElement = mInsertionPoint);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsXFormsTriggerElement::BeginAddingChildren()
|
|
{
|
|
mDoneAddingChildren = PR_FALSE;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsXFormsTriggerElement::DoneAddingChildren()
|
|
{
|
|
mDoneAddingChildren = PR_TRUE;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsXFormsTriggerElement::AttributeRemoved(nsIAtom *aName)
|
|
{
|
|
return aName == nsXFormsAtoms::appearance
|
|
? AttributeSet(aName, EmptyString())
|
|
: nsXFormsControlStub::AttributeRemoved(aName);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsXFormsTriggerElement::AttributeSet(nsIAtom *aName, const nsAString &aValue)
|
|
{
|
|
if (aName == nsXFormsAtoms::appearance) {
|
|
if (mIsMinimal != aValue.EqualsLiteral("minimal")) {
|
|
nsCOMPtr<nsIDOMNode> parent, next, tmp;
|
|
if (mDoneAddingChildren) {
|
|
mElement->GetParentNode(getter_AddRefs(parent));
|
|
if (parent) {
|
|
mElement->GetNextSibling(getter_AddRefs(next));
|
|
// Removing this element temporarily from the document so that
|
|
// the UI will be re-created properly later.
|
|
parent->RemoveChild(mElement, getter_AddRefs(tmp));
|
|
}
|
|
}
|
|
// Switch state
|
|
mIsMinimal = !mIsMinimal;
|
|
|
|
// Create new element
|
|
nsCOMPtr<nsIDOMDocument> domDoc;
|
|
mVisualContent->GetOwnerDocument(getter_AddRefs(domDoc));
|
|
NS_ENSURE_STATE(domDoc);
|
|
nsCOMPtr<nsIDOMElement> htmlElement;
|
|
if (mIsMinimal) {
|
|
domDoc->CreateElementNS(NS_LITERAL_STRING(NS_NAMESPACE_XHTML),
|
|
NS_LITERAL_STRING("span"),
|
|
getter_AddRefs(htmlElement));
|
|
} else {
|
|
domDoc->CreateElementNS(NS_LITERAL_STRING(NS_NAMESPACE_XHTML),
|
|
NS_LITERAL_STRING("button"),
|
|
getter_AddRefs(htmlElement));
|
|
}
|
|
NS_ENSURE_STATE(htmlElement);
|
|
|
|
// Append inner content (insertion point)
|
|
nsCOMPtr<nsIDOMNode> newnode;
|
|
nsresult rv = htmlElement->AppendChild(mInsertionPoint, getter_AddRefs(newnode));
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
// Replace current visual
|
|
rv = mVisualContent->ReplaceChild(htmlElement, mHTMLElement, getter_AddRefs(tmp));
|
|
mHTMLElement = do_QueryInterface(htmlElement);
|
|
|
|
if (mDoneAddingChildren && parent)
|
|
parent->InsertBefore(mElement, next, getter_AddRefs(tmp));
|
|
}
|
|
} else {
|
|
nsXFormsControlStub::AttributeSet(aName, aValue);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsXFormsTriggerElement::HandleDefault(nsIDOMEvent *aEvent, PRBool *aHandled)
|
|
{
|
|
nsresult rv;
|
|
|
|
rv = nsXFormsControlStub::HandleDefault(aEvent, aHandled);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
if (*aHandled || !mIsMinimal) {
|
|
return NS_OK;
|
|
}
|
|
|
|
nsAutoString type;
|
|
aEvent->GetType(type);
|
|
|
|
// Check for click on minimal trigger
|
|
if (!(*aHandled = type.EqualsLiteral("click")))
|
|
return NS_OK;
|
|
|
|
// We need to dend DOMActivate
|
|
nsCOMPtr<nsIDOMEventTarget> target = do_QueryInterface(mElement);
|
|
NS_ENSURE_STATE(target);
|
|
|
|
nsCOMPtr<nsIDOMDocument> domDoc;
|
|
mElement->GetOwnerDocument(getter_AddRefs(domDoc));
|
|
|
|
nsCOMPtr<nsIDOMDocumentEvent> doc = do_QueryInterface(domDoc);
|
|
NS_ENSURE_STATE(doc);
|
|
|
|
nsCOMPtr<nsIDOMDocumentView> dView = do_QueryInterface(domDoc);
|
|
NS_ENSURE_STATE(dView);
|
|
nsCOMPtr<nsIDOMAbstractView> aView;
|
|
dView->GetDefaultView(getter_AddRefs(aView));
|
|
NS_ENSURE_STATE(aView);
|
|
|
|
nsCOMPtr<nsIDOMEvent> event;
|
|
doc->CreateEvent(NS_LITERAL_STRING("UIEvents"), getter_AddRefs(event));
|
|
nsCOMPtr<nsIDOMUIEvent> uiEvent = do_QueryInterface(event);
|
|
NS_ENSURE_TRUE(uiEvent, NS_ERROR_OUT_OF_MEMORY);
|
|
|
|
uiEvent->InitUIEvent(NS_LITERAL_STRING("DOMActivate"),
|
|
PR_TRUE,
|
|
PR_TRUE,
|
|
aView,
|
|
1); // Simple click
|
|
|
|
PRBool cancelled;
|
|
return target->DispatchEvent(uiEvent, &cancelled);
|
|
}
|
|
|
|
// nsIXFormsControl
|
|
|
|
NS_IMETHODIMP
|
|
nsXFormsTriggerElement::Refresh()
|
|
{
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsXFormsTriggerElement::TryFocus(PRBool *aOK)
|
|
{
|
|
*aOK = GetRelevantState() && nsXFormsUtils::FocusControl(mHTMLElement);
|
|
return NS_OK;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
class nsXFormsSubmitElement : public nsXFormsTriggerElement,
|
|
public nsIXFormsSubmitElement
|
|
{
|
|
public:
|
|
NS_DECL_ISUPPORTS_INHERITED
|
|
NS_DECL_NSIXFORMSSUBMITELEMENT
|
|
|
|
// nsIXTFElement overrides
|
|
NS_IMETHOD HandleDefault(nsIDOMEvent *aEvent, PRBool *aHandled);
|
|
};
|
|
|
|
NS_IMPL_ISUPPORTS_INHERITED1(nsXFormsSubmitElement,
|
|
nsXFormsTriggerElement,
|
|
nsIXFormsSubmitElement)
|
|
|
|
// nsIXTFElement
|
|
|
|
NS_IMETHODIMP
|
|
nsXFormsSubmitElement::HandleDefault(nsIDOMEvent *aEvent, PRBool *aHandled)
|
|
{
|
|
nsresult rv;
|
|
|
|
rv = nsXFormsTriggerElement::HandleDefault(aEvent, aHandled);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
if (*aHandled) {
|
|
return NS_OK;
|
|
}
|
|
|
|
nsAutoString type;
|
|
aEvent->GetType(type);
|
|
if (!(*aHandled = type.EqualsLiteral("DOMActivate")))
|
|
return NS_OK;
|
|
|
|
NS_NAMED_LITERAL_STRING(submission, "submission");
|
|
nsAutoString submissionID;
|
|
mElement->GetAttribute(submission, submissionID);
|
|
|
|
nsCOMPtr<nsIDOMDocument> ownerDoc;
|
|
mElement->GetOwnerDocument(getter_AddRefs(ownerDoc));
|
|
NS_ENSURE_STATE(ownerDoc);
|
|
|
|
nsCOMPtr<nsIDOMElement> submissionElement;
|
|
ownerDoc->GetElementById(submissionID, getter_AddRefs(submissionElement));
|
|
nsCOMPtr<nsIXFormsSubmissionElement> xfSubmission(do_QueryInterface(submissionElement));
|
|
|
|
if (!xfSubmission) {
|
|
return nsXFormsUtils::DispatchEvent(mElement, eEvent_BindingException);
|
|
}
|
|
|
|
xfSubmission->SetActivator(this);
|
|
nsXFormsUtils::DispatchEvent(submissionElement, eEvent_Submit);
|
|
|
|
*aHandled = PR_TRUE;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsXFormsSubmitElement::SetDisabled(PRBool aDisable)
|
|
{
|
|
///
|
|
/// @todo Fix SetDisabled() behaviour when mIsMinimal == PR_TRUE (XXX)
|
|
nsCOMPtr<nsIDOMHTMLButtonElement> button = do_QueryInterface(mHTMLElement);
|
|
if (button)
|
|
button->SetDisabled(aDisable);
|
|
return NS_OK;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
NS_HIDDEN_(nsresult)
|
|
NS_NewXFormsTriggerElement(nsIXTFElement **aResult)
|
|
{
|
|
*aResult = new nsXFormsTriggerElement();
|
|
if (!*aResult)
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
NS_ADDREF(*aResult);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_HIDDEN_(nsresult)
|
|
NS_NewXFormsSubmitElement(nsIXTFElement **aResult)
|
|
{
|
|
*aResult = new nsXFormsSubmitElement();
|
|
if (!*aResult)
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
NS_ADDREF(*aResult);
|
|
return NS_OK;
|
|
}
|