pjs/extensions/xforms/nsXFormsModelElement.h

444 строки
14 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):
* Brian Ryner <bryner@brianryner.com>
* Olli Pettay <Olli.Pettay@helsinki.fi>
*
* 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 ***** */
#ifndef nsXFormsModelElement_h_
#define nsXFormsModelElement_h_
#include "nsXFormsStubElement.h"
#include "nsIModelElementPrivate.h"
#include "nsIXFormsNSModelElement.h"
#include "nsIDOMEventListener.h"
#include "nsISchema.h"
#include "nsCOMArray.h"
#include "nsVoidArray.h"
#include "nsCOMPtr.h"
#include "nsIDOMDocument.h"
#include "nsIDOMNodeList.h"
#include "nsXFormsMDGEngine.h"
#include "nsXFormsUtils.h"
#include "nsIClassInfo.h"
#include "nsISchemaLoader.h"
#include "nsISchema.h"
#include "nsIXFormsContextControl.h"
class nsIDOMElement;
class nsIDOMNode;
class nsIXFormsXPathEvaluator;
class nsIDOMXPathResult;
class nsXFormsControl;
class nsXFormsModelInstanceDocuments;
/**
* Implementation of the instance node list returned by
* nsIXFormsModel::getInstanceDocuments.
*
* Manages the list of all instance elements that belong to a given
* nsXFormsModelElement.
*/
class nsXFormsModelInstanceDocuments : public nsIDOMNodeList,
public nsIClassInfo
{
public:
nsXFormsModelInstanceDocuments();
NS_DECL_ISUPPORTS
NS_DECL_NSICLASSINFO
NS_DECL_NSIDOMNODELIST
/**
* Add an instance element
*
* @param aInstance The new instance element
*/
void AddInstance(nsIInstanceElementPrivate *aInstance);
/**
* Remove an instance element
*
* @param aInstance The instance element
*/
void RemoveInstance(nsIInstanceElementPrivate *aInstance);
/**
* Get the instance document at a given index
*
* @note Does NOT addref the returned element!
*
* @param aIndex The index
* @return The instance element (or nsnull if not found)
*/
nsIInstanceElementPrivate* GetInstanceAt(PRUint32 aIndex);
/**
* Instructs the class to drop references to all instance elements
*/
void DropReferences();
protected:
/** The array holding the instance elements */
nsCOMArray<nsIInstanceElementPrivate> mInstanceList;
};
/**
* A class for storing pointers to XForms controls added to an XForms
* model. Organized as a tree, with pointers to first child and next sibling.
*
* Notes:
* 1) The root node is special; only has children and a nsnull mNode.
* 2) All functions operate on the node they are called on, and its
* subtree. Functions will never go up (as in: nearer the root) in the tree.
*/
class nsXFormsControlListItem
{
/** The XForms control itself */
nsCOMPtr<nsIXFormsControl> mNode;
/** The next sibling of the node */
nsXFormsControlListItem *mNextSibling;
/** The first child of the node */
nsXFormsControlListItem *mFirstChild;
public:
nsXFormsControlListItem(nsIXFormsControl* aControl);
~nsXFormsControlListItem();
nsXFormsControlListItem(const nsXFormsControlListItem& aCopy);
/** Clear contents of current node, all siblings, and all children */
void Clear();
/**
* Remove a control from the current (sub-) tree. Will search from that node
* and down in the tree for the node.
*
* @param aControl The control to remove
* @param aRemoved Was the control found and removed?
*/
nsresult RemoveControl(nsIXFormsControl *aControl, PRBool &aRemoved);
/**
* Add a control to the (sub-) tree as a child to the given |aParent|. If
* there is no |aParent|, it will insert it as a sibling to |this|. If
* |aParent| is not found, it will bail.
*
* @param aControl The control to insert
* @param aParent The (eventual) parent to insert it under
*/
nsresult AddControl(nsIXFormsControl *aControl,
nsIXFormsControl *aParent);
/**
* Find a control in the (sub-) tree.
*
* @param aControl The control to find
*/
nsXFormsControlListItem* FindControl(nsIXFormsControl *aControl);
/**
* Return the nsIXFormsControl that this node contains.
*/
already_AddRefed<nsIXFormsControl> Control();
/** Return the first child of the node */
nsXFormsControlListItem* FirstChild() { return mFirstChild; };
/** Return the next sibling of the node */
nsXFormsControlListItem* NextSibling() { return mNextSibling; };
/**
* An iterator implementation for the class.
*/
class iterator
{
private:
/** The control the iterator is currently pointing at */
nsXFormsControlListItem *mCur;
/** A stack of non-visited nodes */
nsVoidArray mStack;
public:
iterator();
iterator(const iterator&);
iterator operator=(nsXFormsControlListItem*);
bool operator!=(const nsXFormsControlListItem*);
iterator operator++();
nsXFormsControlListItem* operator*();
};
friend class nsXFormsControlListItem::iterator;
/** The begining position for the node (itself) */
nsXFormsControlListItem* begin();
/** The end position for the node (nsnull) */
nsXFormsControlListItem* end();
};
/**
* Implementation of the XForms \<model\> element.
*
* This includes all of the code for loading the model's external resources and
* initializing the model and its controls.
*
* @see http://www.w3.org/TR/xforms/slice3.html#structure-model
*/
class nsXFormsModelElement : public nsXFormsStubElement,
public nsIModelElementPrivate,
public nsIXFormsNSModelElement,
public nsISchemaLoadListener,
public nsIDOMEventListener,
public nsIXFormsContextControl
{
public:
nsXFormsModelElement() NS_HIDDEN;
NS_DECL_ISUPPORTS_INHERITED
NS_DECL_NSIXFORMSMODELELEMENT
NS_DECL_NSIMODELELEMENTPRIVATE
NS_DECL_NSIXFORMSNSMODELELEMENT
NS_DECL_NSISCHEMALOADLISTENER
NS_DECL_NSIWEBSERVICEERRORHANDLER
NS_DECL_NSIDOMEVENTLISTENER
NS_DECL_NSIXFORMSCONTEXTCONTROL
// nsIXTFGenericElement overrides
NS_IMETHOD OnDestroyed();
NS_IMETHOD GetScriptingInterfaces(PRUint32 *aCount, nsIID ***aArray);
NS_IMETHOD WillChangeDocument(nsIDOMDocument *aNewDocument);
NS_IMETHOD DocumentChanged(nsIDOMDocument *aNewDocument);
NS_IMETHOD DoneAddingChildren();
NS_IMETHOD HandleDefault(nsIDOMEvent *aEvent, PRBool *aHandled);
NS_IMETHOD OnCreated(nsIXTFGenericElementWrapper *aWrapper);
// nsIXFormsControlBase overrides
NS_IMETHOD Bind() {
// dummy method, so does nothing
return NS_OK;
};
NS_IMETHOD GetOnDeferredBindList(PRBool *onList) {
// dummy method, so does nothing
return NS_OK;
};
NS_IMETHOD SetOnDeferredBindList(PRBool putOnList) {
// dummy method, so does nothing
return NS_OK;
};
// Called after nsXFormsAtoms is registered
static NS_HIDDEN_(void) Startup();
/**
* The models are not ready for binding, so defer the binding of the control
* by storing it as a property on the document. The models will run through
* this list when they are ready for binding.
*
* @param aDoc Document that contains aElement
* @param aControl XForms control waiting to be bound
*/
static NS_HIDDEN_(nsresult) DeferElementBind(nsIDOMDocument *aDoc,
nsIXFormsControlBase *aControl);
static nsresult NeedsPostRefresh(nsIXFormsControl* aControl);
// Sometimes a child that is on the post refresh list will want to force
// its containing xforms control to refresh. Like if an xf:item's label
// refreshes, it wants to reflect any of its changes in the xf:select or
// xf:select1 that contains it. This function will try to minimize that by
// allowing xforms controls that function as containers (like select/select1)
// to defer their refresh until most or all of its children have refreshed.
// Well, at least until every control on the sPostRefreshList has been
// processed. This will return PR_TRUE if the refresh will be deferred.
static PRBool ContainerNeedsPostRefresh(nsIXFormsControl* aControl);
static void CancelPostRefresh(nsIXFormsControl* aControl);
/**
* Returns the DOM element for the model.
*/
already_AddRefed<nsIDOMElement> GetDOMElement();
private:
NS_HIDDEN_(already_AddRefed<nsIDOMDocument>)
FindInstanceDocument(const nsAString &aID);
NS_HIDDEN_(void) Reset();
NS_HIDDEN_(void) BackupOrRestoreInstanceData(PRBool restore);
/** Initializes the MIPs on all form controls */
NS_HIDDEN_(nsresult) InitializeControls();
NS_HIDDEN_(nsresult) InitializeInstances();
NS_HIDDEN_(nsresult) ProcessBindElements();
NS_HIDDEN_(nsresult) FinishConstruction();
NS_HIDDEN_(nsresult) ConstructDone();
NS_HIDDEN_(void) MaybeNotifyCompletion();
NS_HIDDEN_(nsresult) ProcessBind(nsIXFormsXPathEvaluator *aEvaluator,
nsIDOMNode *aContextNode,
PRInt32 aContextPosition,
PRInt32 aContextSize,
nsIDOMElement *aBindElement,
PRBool aIsOuter = PR_FALSE);
NS_HIDDEN_(void) RemoveModelFromDocument();
/**
* Sets the state of a specific state.
*
* @param aElement The element to dispatch events to and set states on
* @param aState The state value
* @param aOnEvent The on event for the state.
*/
NS_HIDDEN_(void) SetSingleState(nsIDOMElement *aElement,
PRBool aState,
nsXFormsEvent aOnEvent);
/**
* Call the Bind() and Refresh() on controls which was deferred because
* the model was not ready.
*
* @param aDoc Document that contains the XForms control
*/
static NS_HIDDEN_(void) ProcessDeferredBinds(nsIDOMDocument *aDoc);
// Returns true when all external documents have been loaded
PRBool IsComplete() const { return (mSchemaTotal == mSchemaCount
&& mPendingInstanceCount == 0); }
/**
* Called by HandleEvent. Event handler for the 'DOMContentLoaded' event.
*/
NS_HIDDEN_(nsresult) HandleLoad(nsIDOMEvent *aEvent);
/**
* Called by HandleEvent. Event handler for the 'unload' event.
*/
NS_HIDDEN_(nsresult) HandleUnload(nsIDOMEvent *aEvent);
NS_HIDDEN_(nsresult) RefreshSubTree(nsXFormsControlListItem *aCurrent,
PRBool aForceRebind);
NS_HIDDEN_(nsresult) ValidateDocument(nsIDOMDocument *aInstanceDocument,
PRBool *aResult);
nsIDOMElement *mElement;
nsCOMPtr<nsISchemaLoader> mSchemas;
nsStringArray mPendingInlineSchemas;
nsXFormsControlListItem mFormControls;
PRInt32 mSchemaCount;
PRInt32 mSchemaTotal;
PRInt32 mPendingInstanceCount;
/** The MDG for this model */
nsXFormsMDGEngine mMDG;
/**
* List of changed nodes, ie. nodes that have not been informed about
* changes yet
*/
nsCOMArray<nsIDOMNode> mChangedNodes;
// This flag indicates whether or not the document fired DOMContentLoaded
PRBool mDocumentLoaded;
// This flag indicates whether a xforms-rebuild has been called, but no
// xforms-revalidate yet
PRBool mNeedsRefresh;
// This flag indicates whether instance elements have been initialized
PRBool mInstancesInitialized;
/**
* Indicates whether the model has handled the xforms-ready event
*/
PRBool mReadyHandled;
/**
* All instance documents contained by this model, including lazy-authored
* instance documents.
*/
nsRefPtr<nsXFormsModelInstanceDocuments> mInstanceDocuments;
// Indicates whether the model's instance was built by lazy authoring
PRBool mLazyModel;
/**
* Type information for nodes, with their type set through \<xforms:bind\>.
*
* @see http://www.w3.org/TR/xforms/slice6.html#model-prop-type
*/
nsClassHashtable<nsISupportsHashKey, nsString> mNodeToType;
/**
* P3P type information for nodes, with their type set through
* \<xforms:bind\>.
* @see http://www.w3.org/TR/xforms/slice6.html#model-prop-p3ptype
*/
nsClassHashtable<nsISupportsHashKey, nsString> mNodeToP3PType;
/**
* Indicates whether the model has handled the xforms-model-construct-done
* event
*/
PRBool mConstructDoneHandled;
};
/**
* nsPostRefresh is needed by the UI Controls, which are implemented in
* XBL and used inside \<repeat\>. It is needed to refresh the controls,
* because XBL bindings are attached to XForms controls *after* refreshing
* the \<repeat\>.
*/
class nsPostRefresh {
public:
nsPostRefresh();
~nsPostRefresh();
static const nsVoidArray* PostRefreshList();
};
NS_HIDDEN_(nsresult) NS_NewXFormsModelElement(nsIXTFElement **aResult);
#endif