gecko/content/xul/document/src/nsXULContentSink.h

147 lines
4.5 KiB
C
Raw Normal View History

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2012-05-21 04:12:37 -07:00
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
#ifndef nsXULContentSink_h__
#define nsXULContentSink_h__
#include "nsIExpatSink.h"
#include "nsIXMLContentSink.h"
#include "nsAutoPtr.h"
#include "nsNodeInfoManager.h"
#include "nsWeakPtr.h"
#include "nsXULElement.h"
#include "nsIDTD.h"
class nsIDocument;
class nsIScriptSecurityManager;
class nsAttrName;
class nsXULPrototypeDocument;
class nsXULPrototypeElement;
class nsXULPrototypeNode;
class XULContentSinkImpl : public nsIXMLContentSink,
public nsIExpatSink
{
public:
XULContentSinkImpl();
virtual ~XULContentSinkImpl();
// nsISupports
NS_DECL_ISUPPORTS
NS_DECL_NSIEXPATSINK
// nsIContentSink
NS_IMETHOD WillParse(void) { return NS_OK; }
NS_IMETHOD WillBuildModel(nsDTDMode aDTDMode);
NS_IMETHOD DidBuildModel(bool aTerminated);
NS_IMETHOD WillInterrupt(void);
NS_IMETHOD WillResume(void);
NS_IMETHOD SetParser(nsParserBase* aParser);
virtual void FlushPendingNotifications(mozFlushType aType) { }
NS_IMETHOD SetDocumentCharset(nsACString& aCharset);
virtual nsISupports *GetTarget();
/**
* Initialize the content sink, giving it an nsIDocument object
* with which to communicate with the outside world, and an
* nsXULPrototypeDocument to build.
*/
nsresult Init(nsIDocument* aDocument, nsXULPrototypeDocument* aPrototype);
protected:
// pseudo-constants
PRUnichar* mText;
PRInt32 mTextLength;
PRInt32 mTextSize;
bool mConstrainSize;
nsresult AddAttributes(const PRUnichar** aAttributes,
const PRUint32 aAttrLen,
nsXULPrototypeElement* aElement);
nsresult OpenRoot(const PRUnichar** aAttributes,
const PRUint32 aAttrLen,
nsINodeInfo *aNodeInfo);
nsresult OpenTag(const PRUnichar** aAttributes,
const PRUint32 aAttrLen,
const PRUint32 aLineNumber,
nsINodeInfo *aNodeInfo);
// If OpenScript returns NS_OK and after it returns our state is eInScript,
// that means that we created a prototype script and stuck it on
// mContextStack. If NS_OK is returned but the state is still
// eInDocumentElement then we didn't create a prototype script (e.g. the
// script had an unknown type), and the caller should create a prototype
// element.
nsresult OpenScript(const PRUnichar** aAttributes,
const PRUint32 aLineNumber);
static bool IsDataInBuffer(PRUnichar* aBuffer, PRInt32 aLength);
// Text management
nsresult FlushText(bool aCreateTextNode = true);
nsresult AddText(const PRUnichar* aText, PRInt32 aLength);
nsRefPtr<nsNodeInfoManager> mNodeInfoManager;
nsresult NormalizeAttributeString(const PRUnichar *aExpatName,
nsAttrName &aName);
nsresult CreateElement(nsINodeInfo *aNodeInfo,
nsXULPrototypeElement** aResult);
public:
enum State { eInProlog, eInDocumentElement, eInScript, eInEpilog };
protected:
State mState;
// content stack management
class ContextStack {
protected:
struct Entry {
nsRefPtr<nsXULPrototypeNode> mNode;
// a LOT of nodes have children; preallocate for 8
nsPrototypeArray mChildren;
State mState;
Entry* mNext;
Entry() : mChildren(8) {}
};
Entry* mTop;
PRInt32 mDepth;
public:
ContextStack();
~ContextStack();
PRInt32 Depth() { return mDepth; }
nsresult Push(nsXULPrototypeNode* aNode, State aState);
nsresult Pop(State* aState);
nsresult GetTopNode(nsRefPtr<nsXULPrototypeNode>& aNode);
nsresult GetTopChildren(nsPrototypeArray** aChildren);
void Clear();
};
friend class ContextStack;
ContextStack mContextStack;
nsWeakPtr mDocument; // [OWNER]
nsCOMPtr<nsIURI> mDocumentURL; // [OWNER]
nsRefPtr<nsXULPrototypeDocument> mPrototype; // [OWNER]
// We use regular pointer b/c of funky exports on nsIParser:
nsParserBase* mParser; // [OWNER]
nsCOMPtr<nsIScriptSecurityManager> mSecMan;
};
#endif /* nsXULContentSink_h__ */