Mozilla/mozilla/xpfe/appshell/src/nsWebShellWindow.cpp
pinkerton%netscape.com 4859c3f41a fix the dom dredging code to handle the presense of a xul namespace in the xml file and clean up things with nsCOMPtr.
git-svn-id: svn://10.0.0.236/trunk@18834 18797224-902f-48f8-a5cc-f745e15eee43
1999-01-28 00:21:12 +00:00

843 lines
26 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.0 (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.
*/
#include "nsWebShellWindow.h"
#include "nsCOMPtr.h"
#include "nsRepository.h"
#include "nsIServiceManager.h"
#include "nsIURL.h"
#include "nsGUIEvent.h"
#include "nsWidgetsCID.h"
#include "nsIWidget.h"
#include "nsIAppShell.h"
#include "nsIAppShellService.h"
#include "nsIWidgetController.h"
#include "nsAppShellCIDs.h"
#include "nsXULCommand.h"
#include "nsIXULCommand.h"
#include "nsIDOMCharacterData.h"
#include "nsIMenuBar.h"
#include "nsIMenu.h"
#include "nsIMenuItem.h"
// XXX: Only needed for the creation of the widget controller...
#include "nsIDocumentViewer.h"
#include "nsIDocument.h"
#include "nsIDOMDocument.h"
#include "nsIDOMNode.h"
#include "nsIDOMElement.h"
#include "nsIDocumentLoader.h"
#include "nsIDOMHTMLInputElement.h"
#include "nsIDOMHTMLImageElement.h"
/* Define Class IDs */
static NS_DEFINE_IID(kWindowCID, NS_WINDOW_CID);
static NS_DEFINE_IID(kWebShellCID, NS_WEB_SHELL_CID);
static NS_DEFINE_IID(kAppShellServiceCID, NS_APPSHELL_SERVICE_CID);
static NS_DEFINE_IID(kMenuBarCID, NS_MENUBAR_CID);
static NS_DEFINE_IID(kMenuCID, NS_MENU_CID);
static NS_DEFINE_IID(kMenuItemCID, NS_MENUITEM_CID);
/* Define Interface IDs */
static NS_DEFINE_IID(kISupportsIID, NS_ISUPPORTS_IID);
static NS_DEFINE_IID(kIWidgetIID, NS_IWIDGET_IID);
static NS_DEFINE_IID(kIWebShellIID, NS_IWEB_SHELL_IID);
static NS_DEFINE_IID(kIWebShellContainerIID, NS_IWEB_SHELL_CONTAINER_IID);
static NS_DEFINE_IID(kIAppShellServiceIID, NS_IAPPSHELL_SERVICE_IID);
static NS_DEFINE_IID(kIWidgetControllerIID, NS_IWIDGETCONTROLLER_IID);
static NS_DEFINE_IID(kIDocumentLoaderObserverIID, NS_IDOCUMENT_LOADER_OBSERVER_IID);
static NS_DEFINE_IID(kIDocumentViewerIID, NS_IDOCUMENT_VIEWER_IID);
static NS_DEFINE_IID(kIDOMDocumentIID, NS_IDOMDOCUMENT_IID);
static NS_DEFINE_IID(kIDOMNodeIID, NS_IDOMNODE_IID);
static NS_DEFINE_IID(kIDOMElementIID, NS_IDOMELEMENT_IID);
static NS_DEFINE_IID(kIXULCommandIID, NS_IXULCOMMAND_IID);
static NS_DEFINE_IID(kIDOMCharacterDataIID, NS_IDOMCHARACTERDATA_IID);
static NS_DEFINE_IID(kIDOMHTMLInputElementIID, NS_IDOMHTMLINPUTELEMENT_IID);
static NS_DEFINE_IID(kIDOMHTMLImageElementIID, NS_IDOMHTMLIMAGEELEMENT_IID);
static NS_DEFINE_IID(kIMenuIID, NS_IMENU_IID);
static NS_DEFINE_IID(kIMenuBarIID, NS_IMENUBAR_IID);
static NS_DEFINE_IID(kIMenuItemIID, NS_IMENUITEM_IID);
#include "nsIWebShell.h"
const char * kThrobberOnStr = "resource:/res/throbber/anims07.gif";
const char * kThrobberOffStr = "resource:/res/throbber/anims00.gif";
nsWebShellWindow::nsWebShellWindow()
{
NS_INIT_REFCNT();
mWebShell = nsnull;
mWindow = nsnull;
mController = nsnull;
mStatusText = nsnull;
mURLBarText = nsnull;
mThrobber = nsnull;
}
nsWebShellWindow::~nsWebShellWindow()
{
if (nsnull != mWebShell) {
mWebShell->Destroy();
NS_RELEASE(mWebShell);
}
NS_IF_RELEASE(mWindow);
NS_IF_RELEASE(mController);
NS_IF_RELEASE(mStatusText);
NS_IF_RELEASE(mURLBarText);
NS_IF_RELEASE(mThrobber);
}
NS_IMPL_ADDREF(nsWebShellWindow);
NS_IMPL_RELEASE(nsWebShellWindow);
nsresult
nsWebShellWindow::QueryInterface(REFNSIID aIID, void** aInstancePtr)
{
nsresult rv = NS_NOINTERFACE;
if (NULL == aInstancePtr) {
return NS_ERROR_NULL_POINTER;
}
if (aIID.Equals(kIWebShellContainerIID)) {
*aInstancePtr = (void*)(nsIWebShellContainer*)this;
NS_ADDREF_THIS();
return NS_OK;
}
if (aIID.Equals(kIDocumentLoaderObserverIID)) {
*aInstancePtr = (void*) ((nsIDocumentLoaderObserver*)this);
NS_ADDREF_THIS();
return NS_OK;
}
if (aIID.Equals(kISupportsIID)) {
*aInstancePtr = (void*)(nsISupports*)(nsIWebShellContainer*)this;
NS_ADDREF_THIS();
return NS_OK;
}
return rv;
}
nsresult nsWebShellWindow::Initialize(nsIAppShell* aShell, nsIURL* aUrl,
nsString& aControllerIID)
{
nsresult rv;
nsString urlString;
const char *tmpStr = NULL;
nsIID iid;
char str[40];
aUrl->GetSpec(&tmpStr);
urlString = tmpStr;
// XXX: need to get the default window size from prefs...
nsRect r(0, 0, 650, 618);
nsWidgetInitData initData;
if (nsnull == aUrl) {
rv = NS_ERROR_NULL_POINTER;
goto done;
}
// Create top level window
rv = nsRepository::CreateInstance(kWindowCID, nsnull, kIWidgetIID,
(void**)&mWindow);
if (NS_OK != rv) {
goto done;
}
initData.mBorderStyle = eBorderStyle_dialog;
mWindow->SetClientData(this);
mWindow->Create((nsIWidget*)nsnull, // Parent nsIWidget
r, // Widget dimensions
nsWebShellWindow::HandleEvent, // Event handler function
nsnull, // Device context
aShell, // Application shell
nsnull, // nsIToolkit
&initData); // Widget initialization data
mWindow->GetClientBounds(r);
mWindow->SetBackgroundColor(NS_RGB(192,192,192));
// Create web shell
rv = nsRepository::CreateInstance(kWebShellCID, nsnull,
kIWebShellIID,
(void**)&mWebShell);
if (NS_OK != rv) {
goto done;
}
r.x = r.y = 0;
rv = mWebShell->Init(mWindow->GetNativeData(NS_NATIVE_WIDGET),
r.x, r.y, r.width, r.height,
nsScrollPreference_kNeverScroll,
PR_TRUE, // Allow Plugins
PR_TRUE);
mWebShell->SetContainer(this);
//mWebShell->SetObserver((nsIStreamObserver*)this);
nsIDocumentLoader * docLoader;
mWebShell->GetDocumentLoader(docLoader);
if (nsnull != docLoader) {
docLoader->AddObserver((nsIDocumentLoaderObserver *)this);
}
/// webShell->SetPrefs(aPrefs);
mWebShell->LoadURL(urlString);
mWebShell->Show();
mWindow->Show(PR_TRUE);
// Create the IWidgetController for the document...
mController = nsnull;
aControllerIID.ToCString(str, sizeof(str));
iid.Parse(str);
rv = nsRepository::CreateInstance(iid, nsnull,
kIWidgetControllerIID,
(void**)&mController);
done:
return rv;
}
/*
* Event handler function...
*
* This function is called to process events for the nsIWidget of the
* nsWebShellWindow...
*/
nsEventStatus PR_CALLBACK
nsWebShellWindow::HandleEvent(nsGUIEvent *aEvent)
{
nsresult rv;
nsEventStatus result = nsEventStatus_eIgnore;
nsIWebShell* webShell = nsnull;
// Get the WebShell instance...
if (nsnull != aEvent->widget) {
void* data;
aEvent->widget->GetClientData(data);
if (nsnull != data) {
webShell = ((nsWebShellWindow*)data)->mWebShell;
}
}
if (nsnull != webShell) {
switch(aEvent->message) {
/*
* For size events, the WebShell must be resized to fill the entire
* client area of the window...
*/
case NS_SIZE: {
nsSizeEvent* sizeEvent = (nsSizeEvent*)aEvent;
webShell->SetBounds(0, 0, sizeEvent->windowSize->width, sizeEvent->windowSize->height);
result = nsEventStatus_eConsumeNoDefault;
break;
}
/*
* Notify the ApplicationShellService that the window is being closed...
*/
case NS_DESTROY: {
nsIAppShellService* appShell;
rv = nsServiceManager::GetService(kAppShellServiceCID,
kIAppShellServiceIID,
(nsISupports**)&appShell);
if (NS_SUCCEEDED(rv)) {
appShell->CloseTopLevelWindow(aEvent->widget);
nsServiceManager::ReleaseService(kAppShellServiceCID, appShell);
}
break;
}
}
}
return nsEventStatus_eIgnore;
}
void
nsWebShellWindow::UpdateButtonStatus(PRBool aIsBusy)
{
if (aIsBusy) {
SetCommandEnabled(nsAutoString("nsCmd:BrowserBack"), PR_FALSE);
SetCommandEnabled(nsAutoString("nsCmd:BrowserForward"), PR_FALSE);
SetCommandEnabled(nsAutoString("nsCmd:BrowserReload"), PR_FALSE);
SetCommandEnabled(nsAutoString("nsCmd:BrowserStop"), PR_TRUE);
SetCommandEnabled(nsAutoString("nsCmd:BrowserHome"), PR_FALSE);
SetCommandEnabled(nsAutoString("nsCmd:BrowserPrint"), PR_FALSE);
} else {
nsIWebShell* contentWS = nsnull;
mWebShell->FindChildWithName(nsAutoString("browser.webwindow"), contentWS);
if(nsnull != contentWS){
SetCommandEnabled(nsAutoString("nsCmd:BrowserBack"), contentWS->CanBack() == NS_OK);
SetCommandEnabled(nsAutoString("nsCmd:BrowserForward"), contentWS->CanForward() == NS_OK);
SetCommandEnabled(nsAutoString("nsCmd:BrowserReload"), PR_TRUE);
SetCommandEnabled(nsAutoString("nsCmd:BrowserStop"), PR_FALSE);
SetCommandEnabled(nsAutoString("nsCmd:BrowserHome"), PR_TRUE);
SetCommandEnabled(nsAutoString("nsCmd:BrowserPrint"), PR_TRUE);
NS_RELEASE(contentWS);
}
}
}
NS_IMETHODIMP
nsWebShellWindow::WillLoadURL(nsIWebShell* aShell, const PRUnichar* aURL,
nsLoadType aReason)
{
if (nsnull != mThrobber) {
mThrobber->SetSrc(kThrobberOnStr);
}
nsAutoString url(aURL);
nsAutoString gecko("Gecko - ");
gecko.Append(url);
mWindow->SetTitle(gecko);
if (nsnull != mURLBarText) {
mURLBarText->SetValue(url);
}
UpdateButtonStatus(PR_TRUE);
if (nsnull != mStatusText) {
nsAutoString msg(aURL);
msg.Append(" :Start");
mStatusText->SetData(msg);
}
return NS_OK;
}
NS_IMETHODIMP
nsWebShellWindow::BeginLoadURL(nsIWebShell* aShell, const PRUnichar* aURL)
{
return NS_OK;
}
NS_IMETHODIMP
nsWebShellWindow::ProgressLoadURL(nsIWebShell* aShell, const PRUnichar* aURL,
PRInt32 aProgress, PRInt32 aProgressMax)
{
if (nsnull != mStatusText) {
nsAutoString url(aURL);
url.Append(": progress ");
url.Append(aProgress, 10);
if (0 != aProgressMax) {
url.Append(" (out of ");
url.Append(aProgressMax, 10);
url.Append(")");
}
mStatusText->SetData(url);
}
return NS_OK;
}
NS_IMETHODIMP
nsWebShellWindow::EndLoadURL(nsIWebShell* aShell, const PRUnichar* aURL,
PRInt32 aStatus)
{
if (nsnull != mThrobber) {
mThrobber->SetSrc(kThrobberOffStr);
}
UpdateButtonStatus(PR_FALSE);
if (nsnull != mStatusText) {
nsAutoString msg(aURL);
msg.Append(" :Stop");
mStatusText->SetData(msg);
}
return NS_OK;
}
nsIDOMNode * nsWebShellWindow::FindNamedParentFromDoc(nsIDOMDocument * aDomDoc, const nsString &aName)
{
nsIDOMElement * element;
aDomDoc->GetDocumentElement(&element);
if (nsnull != element) {
nsIDOMNode * parent;
if (NS_OK == element->QueryInterface(kIDOMNodeIID, (void**) &parent)) {
nsIDOMNode * node;
parent->GetFirstChild(&node);
while (nsnull != node) {
nsString name;
node->GetNodeName(name);
//printf("[%s]\n", name.ToNewCString());
if (name.Equals(aName)) {
NS_ADDREF(node);
NS_RELEASE(parent);
NS_RELEASE(element);
return node;
}
nsIDOMNode * oldNode = node;
oldNode->GetNextSibling(&node);
NS_RELEASE(oldNode);
}
NS_IF_RELEASE(node);
NS_RELEASE(parent);
}
NS_RELEASE(element);
}
return nsnull;
}
void nsWebShellWindow::LoadCommands(nsIWebShell * aWebShell, nsIDOMDocument * aDOMDoc)
{
// locate the window element which holds toolbars and menus and commands
nsCOMPtr<nsIDOMNode> window ( dont_AddRef(FindNamedParentFromDoc(aDOMDoc, nsAutoString("window"))) );
if ( !window )
return;
// load up commands
PRUint32 endCount = 0;
nsCOMPtr<nsIDOMNode> parentCmd ( dont_AddRef(FindNamedDOMNode(nsAutoString("commands"), window, endCount, 1)) );
if (nsnull != parentCmd) {
nsCOMPtr<nsIDOMNode> node;
parentCmd->GetFirstChild(getter_AddRefs(node));
while (nsnull != node) {
nsString value;
nsString nodeType;
nsString name;
nsCOMPtr<nsIDOMElement> element ( node );
if ( element ) {
element->GetNodeName(nodeType);
if (nodeType.Equals("command")) {
element->GetAttribute(nsAutoString("name"), name);
element->GetAttribute(nsAutoString("onCommand"), value);
nsXULCommand * xulCmd = new nsXULCommand();
xulCmd->SetName(name);
xulCmd->SetCommand(value);
xulCmd->SetWebShell(aWebShell);
xulCmd->SetDOMElement(element);
nsIXULCommand * icmd;
if (NS_OK == xulCmd->QueryInterface(kIXULCommandIID, (void**) &icmd)) {
mCommands.AppendElement(icmd);
}
//printf("Commands[%s] value[%s]\n", name.ToNewCString(), value.ToNewCString());
}
}
nsCOMPtr<nsIDOMNode> oldNode ( node );
oldNode->GetNextSibling(getter_AddRefs(node));
}
}
// Now install the commands onto the Toolbar GUI Nodes
nsCOMPtr<nsIDOMNode> toolbar ( dont_AddRef(FindNamedDOMNode(nsAutoString("toolbar"), window, endCount, 1)) );
if (nsnull != toolbar) {
nsAutoString cmdAtom("cmd");
nsCOMPtr<nsIDOMNode> node;
toolbar->GetFirstChild(getter_AddRefs(node));
while (nsnull != node) {
nsAutoString value;
nsString nodeCmd;
nsCOMPtr<nsIDOMElement> element ( node );
if ( element ) {
nsString name;
element->GetNodeName(name);
if (name.Equals(nsAutoString("BUTTON"))) {
element->GetAttribute(cmdAtom, nodeCmd);
PRInt32 i, n = mCommands.Count();
for (i = 0; i < n; i++) {
nsIXULCommand* cmd = (nsIXULCommand*) mCommands.ElementAt(i);
nsAutoString cmdName;
cmd->GetName(cmdName);
//printf("Cmd [%s] Node[%s]\n", cmdName.ToNewCString(), nodeCmd.ToNewCString());
if (nodeCmd.Equals(cmdName)) {
printf("Linking up cmd to button [%s]\n", cmdName.ToNewCString());
cmd->AddUINode(node);
}
}
} else if (name.Equals(nsAutoString("INPUT"))) {
nsXULCommand * xulCmd = new nsXULCommand();
xulCmd->SetName(name);
xulCmd->SetCommand(value);
xulCmd->SetWebShell(aWebShell);
xulCmd->SetDOMElement(element);
nsIXULCommand * icmd;
if (NS_OK == xulCmd->QueryInterface(kIXULCommandIID, (void**) &icmd)) {
mCommands.AppendElement(icmd);
}
xulCmd->AddUINode(node);
//printf("Linking up cmd to button [%s]\n", cmdName.ToNewCString());
}
}
nsCOMPtr<nsIDOMNode> oldNode ( node );
oldNode->GetNextSibling(getter_AddRefs(node));
}
}
/*PRInt32 i, n = mCommands.Count();
for (i = 0; i < n; i++) {
nsIXULCommand* cmd = (nsIXULCommand*) mCommands.ElementAt(i);
cmd->SetEnabled(PR_TRUE);
}
SetCommandEnabled(nsAutoString("nsCmd:BrowserStop"), PR_FALSE);
*/
UpdateButtonStatus(PR_FALSE);
}
// Load Menus
void nsWebShellWindow::LoadMenus(nsIDOMDocument * aDOMDoc, nsIWidget * aParentWindow)
{
// locate the window element which holds toolbars and menus and commands
nsCOMPtr<nsIDOMNode> window ( dont_AddRef(FindNamedParentFromDoc(aDOMDoc, nsAutoString("window"))) );
if ( !window )
return;
nsresult rv;
PRUint32 endCount = 0;
nsCOMPtr<nsIDOMNode> menubarNode ( dont_AddRef(FindNamedDOMNode(nsAutoString("menubar"), window, endCount, 1)) );
if (nsnull != menubarNode) {
nsIMenuBar * pnsMenuBar = nsnull;
rv = nsRepository::CreateInstance(kMenuBarCID, nsnull, kIMenuBarIID,
(void**)&pnsMenuBar);
if (NS_OK != rv) {
// Error
}
//pnsMenuBar = new nsMenuBar();
if (nsnull != pnsMenuBar) {
// TODO: set pnsMenuBar as a nsMenuListener on aParentWindow
//GetWidget()->AddMenuListener((nsIMenuListener*)pnsMenuBar);
mWindow->AddMenuListener((nsIMenuListener*)pnsMenuBar);
nsIDOMNode * menuNode = nsnull;
menubarNode->GetFirstChild(&menuNode);
while (nsnull != menuNode) {
nsIDOMElement * menuElement = nsnull;
nsString menuNodeType;
nsString menuName;
if (NS_OK == menuNode->QueryInterface(kIDOMNodeIID, (void**) &menuElement)) {
menuElement->GetNodeName(menuNodeType);
if (menuNodeType.Equals("menu")) {
menuElement->GetAttribute(nsAutoString("name"), menuName);
printf("Menu [%s]\n", menuName.ToNewCString() );
// Create nsMenu
nsIMenu * pnsMenu = nsnull;
rv = nsRepository::CreateInstance(kMenuCID, nsnull, kIMenuIID,
(void**)&pnsMenu);
if (NS_OK != rv) {
// Error
}
// TODO: Set nsMenu Name
pnsMenu->SetLabel(menuName);
// TODO: Make nsMenu a child of nsMenuBar
pnsMenuBar->AddMenu(pnsMenu);
// Begin menuitem inner loop
nsIDOMNode * menuitemNode = nsnull;
menuNode->GetFirstChild(&menuitemNode);
while(nsnull != menuitemNode){
nsIDOMElement * menuitemElement = nsnull;
nsString menuitemNodeType;
nsString menuitemName;
if (NS_OK == menuitemNode->QueryInterface(kIDOMNodeIID, (void**) &menuitemElement)) {
if(menuitemElement){
menuitemElement->GetNodeName(menuitemNodeType);
if (menuitemNodeType.Equals("menuitem")) {
menuitemElement->GetAttribute(nsAutoString("name"), menuitemName);
printf("MenuItem [%s]\n", menuitemName.ToNewCString() );
// Create nsMenuItem
nsIMenuItem * pnsMenuItem = nsnull;
rv = nsRepository::CreateInstance(kMenuItemCID, nsnull, kIMenuItemIID,
(void**)&pnsMenuItem);
if (NS_OK != rv) {
// Error
}
// TODO: Set nsMenuItem Name
pnsMenuItem->SetLabel(menuitemName);
// TODO: Make nsMenuItem a child of nsMenu
pnsMenu->AddMenuItem(pnsMenuItem);
}
NS_RELEASE(menuitemElement);
}
}
nsIDOMNode * oldmenuitemNode = menuitemNode;
oldmenuitemNode->GetNextSibling(&menuitemNode);
NS_RELEASE(oldmenuitemNode);
} // end menu item innner loop
NS_IF_RELEASE(menuitemNode);
}
NS_RELEASE(menuElement);
}
nsIDOMNode * oldmenuNode = menuNode;
oldmenuNode->GetNextSibling(&menuNode);
NS_RELEASE(oldmenuNode);
} // end while (nsnull != menuNode)
NS_IF_RELEASE(menuNode);
// TODO: Give the aParentWindow this nsMenuBar to hold onto.
pnsMenuBar->Paint();
} // end if ( nsnull != pnsMenuBar )
} // end if (nsnull != node)
}
nsIXULCommand *
nsWebShellWindow::FindCommandByName(const nsString & aCmdName)
{
PRInt32 i, n = mCommands.Count();
for (i = 0; i < n; i++) {
nsAutoString name;
nsIXULCommand* cmd = (nsIXULCommand*) mCommands.ElementAt(i);
cmd->GetName(name);
if (name.Equals(aCmdName)) {
NS_ADDREF(cmd);
return cmd;
}
}
return nsnull;
}
void
nsWebShellWindow::SetCommandEnabled(const nsString & aCmdName, PRBool aState)
{
nsIXULCommand * cmd = FindCommandByName(aCmdName);
if (nsnull != cmd) {
cmd->SetEnabled(aState);
NS_RELEASE(cmd);
}
}
NS_IMETHODIMP
nsWebShellWindow::NewWebShell(PRUint32 aChromeMask, PRBool aVisible,
nsIWebShell *&aNewWebShell)
{
return NS_ERROR_FAILURE;
}
NS_IMETHODIMP nsWebShellWindow::FindWebShellWithName(const PRUnichar* aName,
nsIWebShell*& aResult)
{
aResult = nsnull;
return NS_OK;
}
NS_IMETHODIMP
nsWebShellWindow::FocusAvailable(nsIWebShell* aFocusedWebShell)
{
return NS_OK;
}
//----------------------------------------
//----------------------------------------
//----------------------------------------
//----------------------------------------
// nsIDocumentLoaderObserver implementation
//----------------------------------------
NS_IMETHODIMP nsWebShellWindow::OnStartURLLoad(nsIURL* aURL, const char* aContentType, nsIContentViewer* aViewer)
{
return NS_OK;
}
//----------------------------------------
nsIDOMNode * nsWebShellWindow::FindNamedDOMNode(const nsString &aName, nsIDOMNode * aParent, PRInt32 & aCount, PRInt32 aEndCount)
{
nsIDOMNode * node;
aParent->GetFirstChild(&node);
while (nsnull != node) {
nsString name;
node->GetNodeName(name);
//printf("FindNamedDOMNode[%s] %d == %d\n", name.ToNewCString(), aCount, aEndCount);
if (name.Equals(aName)) {
aCount++;
if (aCount == aEndCount) {
NS_ADDREF(node);
return node;
}
}
PRBool hasChildren;
node->HasChildNodes(&hasChildren);
if (hasChildren) {
nsIDOMNode * found = FindNamedDOMNode(aName, node, aCount, aEndCount);
if (nsnull != found) {
return found;
}
}
nsIDOMNode * oldNode = node;
oldNode->GetNextSibling(&node);
NS_RELEASE(oldNode);
}
NS_IF_RELEASE(node);
return nsnull;
}
//----------------------------------------
nsIDOMNode * nsWebShellWindow::GetParentNodeFromDOMDoc(nsIDOMDocument * aDOMDoc)
{
if (nsnull == aDOMDoc) {
return nsnull;
}
nsIDOMElement * element;
nsIDOMNode * node = nsnull;
aDOMDoc->GetDocumentElement(&element);
if (nsnull != element) {
if (NS_OK == element->QueryInterface(kIDOMNodeIID, (void**) &node)) {
// no errors
}
NS_RELEASE(element);
}
return node;
}
//----------------------------------------
nsIDOMDocument * nsWebShellWindow::GetNamedDOMDoc(const nsString & aWebShellName)
{
nsIDOMDocument * domDoc = nsnull;
// first get the toolbar child WebShell
nsIWebShell* childWebShell = nsnull;
mWebShell->FindChildWithName(aWebShellName, childWebShell);
if (nsnull != childWebShell) {
nsIContentViewer* cv = nsnull;
childWebShell->GetContentViewer(&cv);
if (nsnull != cv) {
nsIDocumentViewer* docv = nsnull;
cv->QueryInterface(kIDocumentViewerIID, (void**) &docv);
if (nsnull != docv) {
nsIDocument * doc;
docv->GetDocument(doc);
if (nsnull != doc) {
if (NS_OK == doc->QueryInterface(kIDOMDocumentIID, (void**) &domDoc)) {
// no errors
}
NS_RELEASE(doc);
}
NS_RELEASE(docv);
}
NS_RELEASE(cv);
}
NS_IF_RELEASE(childWebShell);
}
return domDoc;
}
//----------------------------------------
NS_IMETHODIMP nsWebShellWindow::OnConnectionsComplete()
{
///////////////////////////////
// Find the Menubar DOM and Load the menus
///////////////////////////////
nsIDOMDocument * menubarDOMDoc = nsnull;
menubarDOMDoc = GetNamedDOMDoc(nsAutoString("browser.toolbar"));
if (nsnull != menubarDOMDoc) {
LoadMenus(menubarDOMDoc, mWindow);
}
///////////////////////////////
// Find the Toolbar DOM and Load all the commands
///////////////////////////////
nsIDOMDocument * toolbarDOMDoc = GetNamedDOMDoc(nsAutoString("browser.toolbar"));
if (nsnull != toolbarDOMDoc) {
nsIDOMNode * parent = GetParentNodeFromDOMDoc(toolbarDOMDoc);
if (nsnull != parent) {
nsIWebShell* contentWS = nsnull;
mWebShell->FindChildWithName(nsAutoString("browser.webwindow"), contentWS);
if (nsnull != contentWS) {
LoadCommands(contentWS, toolbarDOMDoc);
PRInt32 count = 0;
nsIDOMNode * imgNode = FindNamedDOMNode(nsAutoString("IMG"), parent, count, 7);
if (nsnull != imgNode) {
if (NS_OK == imgNode->QueryInterface(kIDOMHTMLImageElementIID, (void**) &mThrobber)) {
nsAutoString srcStr;
mThrobber->GetSrc(srcStr);
//printf("src: %s\n", srcStr.ToNewCString());
}
NS_RELEASE(imgNode);
}
count = 0;
nsIDOMNode * node = FindNamedDOMNode(nsAutoString("INPUT"), parent, count, 1);
if (nsnull != node) {
if (NS_OK == node->QueryInterface(kIDOMHTMLInputElementIID, (void**) &mURLBarText)) {
PRUnichar * urlStr = nsnull;
contentWS->GetURL(0, &urlStr);
if (nsnull != urlStr) {
nsString url(urlStr);
mURLBarText->SetValue(url);
}
}
NS_RELEASE(node);
}
NS_RELEASE(contentWS);
}
NS_RELEASE(parent);
}
NS_RELEASE(toolbarDOMDoc);
}
///////////////////////////////
// Find the Status Text DOM Node
///////////////////////////////
nsIDOMDocument * statusDOMDoc = GetNamedDOMDoc(nsAutoString("browser.status"));
if (nsnull != statusDOMDoc) {
nsIDOMNode * parent = GetParentNodeFromDOMDoc(statusDOMDoc);
if (nsnull != parent) {
PRInt32 count = 0;
nsIDOMNode * statusNode = FindNamedDOMNode(nsAutoString("#text"), parent, count, 7);
if (nsnull != statusNode) {
nsIDOMCharacterData * charData;
if (NS_OK == statusNode->QueryInterface(kIDOMCharacterDataIID, (void**) &charData)) {
mStatusText = charData;
mStatusText->SetData(nsAutoString("Ready....."));
}
NS_RELEASE(statusNode);
}
NS_RELEASE(parent);
}
NS_RELEASE(statusDOMDoc);
}
return NS_OK;
}