/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* ***** BEGIN LICENSE BLOCK ***** * Version: NPL 1.1/GPL 2.0/LGPL 2.1 * * 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 the Initial Developer are Copyright (C) 1998 * the Initial Developer. All Rights Reserved. * * Contributor(s): * Dan Rosen * Roland Mainz * * * 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 NPL, 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 NPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ #include "nscore.h" #include "nsCOMPtr.h" #include "nsCRT.h" #include "nsString.h" #include "nsReadableUtils.h" #include "nsISupports.h" #include "nsIContent.h" #include "nsIContentViewerContainer.h" #include "nsIDocumentViewer.h" #include "nsIDOMWindowInternal.h" #include "nsIDocumentViewerPrint.h" #include "nsIDocument.h" #include "nsIPresContext.h" #include "nsIPresShell.h" #include "nsIStyleSet.h" #include "nsIStyleSheet.h" #include "nsICSSStyleSheet.h" #include "nsIFrame.h" #include "nsIScriptGlobalObjectOwner.h" #include "nsIScriptGlobalObject.h" #include "nsILinkHandler.h" #include "nsIDOMDocument.h" #include "nsISelectionListener.h" #include "nsISelectionPrivate.h" #include "nsIDOMHTMLDocument.h" #include "nsIDOMNSHTMLDocument.h" #include "nsIDOMHTMLCollection.h" #include "nsIDOMHTMLElement.h" #include "nsIDOMRange.h" #include "nsContentCID.h" #include "nsLayoutCID.h" #include "nsGenericHTMLElement.h" #include "nsViewsCID.h" #include "nsWidgetsCID.h" #include "nsIDeviceContext.h" #include "nsIDeviceContextSpec.h" #include "nsIDeviceContextSpecFactory.h" #include "nsIViewManager.h" #include "nsIView.h" #include "nsIPrefBranch.h" #include "nsIPrefService.h" #include "nsIPrefLocalizedString.h" #include "nsIPageSequenceFrame.h" #include "nsIURL.h" #include "nsIWebShell.h" #include "nsIContentViewerEdit.h" #include "nsIContentViewerFile.h" #include "nsIMarkupDocumentViewer.h" #include "nsIInterfaceRequestor.h" #include "nsIInterfaceRequestorUtils.h" #include "nsIDocShellTreeItem.h" #include "nsIDocShellTreeNode.h" #include "nsIDocShellTreeOwner.h" #include "nsIDocShell.h" #include "nsIBaseWindow.h" #include "nsIFrameDebug.h" #include "nsILayoutHistoryState.h" #include "nsLayoutAtoms.h" #include "nsIFrameManager.h" #include "nsIParser.h" #include "nsIPrintContext.h" #include "nsGUIEvent.h" #include "nsHTMLReflowState.h" #include "nsIDOMHTMLAnchorElement.h" #include "nsIDOMHTMLAreaElement.h" #include "nsIDOMHTMLLinkElement.h" #include "nsIDOMHTMLImageElement.h" #include "nsIDOMHTMLFrameSetElement.h" #include "nsIXULDocument.h" // Temporary code for Bug 136185 #include "nsIChromeRegistry.h" #include "nsIClipboardHelper.h" #include "nsIEventQueueService.h" #include "nsIEventQueue.h" #include "nsPIDOMWindow.h" #include "nsIFocusController.h" #include "nsIScrollableView.h" #include "nsIScrollable.h" #include "nsITimelineService.h" #include "nsGfxCIID.h" // Printing #include "nsIWebBrowserPrint.h" //-------------------------- // Printing Include //--------------------------- #ifdef NS_PRINTING #include "nsPrintEngine.h" // Print Options #include "nsIPrintSettings.h" #include "nsIPrintSettingsService.h" #include "nsIPrintOptions.h" #include "nsIServiceManager.h" #include "nsHTMLAtoms.h" // XXX until atoms get factored into nsLayoutAtoms #include "nsISimpleEnumerator.h" #include "nsXPCOM.h" #include "nsISupportsPrimitives.h" static NS_DEFINE_IID(kPrinterEnumeratorCID, NS_PRINTER_ENUMERATOR_CID); // PrintOptions is now implemented by PrintSettingsService static const char sPrintSettingsServiceContractID[] = "@mozilla.org/gfx/printsettings-service;1"; static const char sPrintOptionsContractID[] = "@mozilla.org/gfx/printsettings-service;1"; // Printing Events #include "nsIEventQueue.h" #include "nsIEventQueueService.h" #include "nsPrintPreviewListener.h" #include "nsIDOMHTMLFrameElement.h" #include "nsIDOMHTMLIFrameElement.h" #include "nsIDOMHTMLObjectElement.h" // Print Preview #include "nsIPrintPreviewContext.h" #include "imgIContainer.h" // image animation mode constants // Print Progress #include "nsIPrintProgress.h" #include "nsIPrintProgressParams.h" // Print error dialog #include "nsIPrompt.h" #include "nsIWindowWatcher.h" #include "nsIStringBundle.h" // Printing #include "nsPrintEngine.h" #include "nsPagePrintTimer.h" #endif // NS_PRINTING // FrameSet #include "nsINodeInfo.h" #include "nsIDocument.h" #include "nsHTMLAtoms.h" #include "nsIHTMLContent.h" #include "nsIWebShell.h" //focus #include "nsIDOMEventReceiver.h" #include "nsIDOMFocusListener.h" #include "nsISelectionController.h" #include "nsBidiUtils.h" static NS_DEFINE_CID(kPresShellCID, NS_PRESSHELL_CID); static NS_DEFINE_CID(kGalleyContextCID, NS_GALLEYCONTEXT_CID); static NS_DEFINE_CID(kPrintContextCID, NS_PRINTCONTEXT_CID); static NS_DEFINE_CID(kStyleSetCID, NS_STYLESET_CID); #ifdef NS_DEBUG #undef NOISY_VIEWER #else #undef NOISY_VIEWER #endif //----------------------------------------------------- // PR LOGGING #ifdef MOZ_LOGGING #define FORCE_PR_LOG /* Allow logging in the release build */ #endif #include "prlog.h" #ifdef PR_LOGGING #ifdef NS_DEBUG // PR_LOGGING is force to always be on (even in release builds) // but we only want some of it on, #define EXTENDED_DEBUG_PRINTING #endif #define DUMP_LAYOUT_LEVEL 9 // this turns on the dumping of each doucment's layout info static PRLogModuleInfo * kPrintingLogMod = PR_NewLogModule("printing"); #define PR_PL(_p1) PR_LOG(kPrintingLogMod, PR_LOG_DEBUG, _p1); #ifdef EXTENDED_DEBUG_PRINTING static PRUint32 gDumpFileNameCnt = 0; static PRUint32 gDumpLOFileNameCnt = 0; #endif #define PRT_YESNO(_p) ((_p)?"YES":"NO") static const char * gFrameTypesStr[] = {"eDoc", "eFrame", "eIFrame", "eFrameSet"}; static const char * gPrintFrameTypeStr[] = {"kNoFrames", "kFramesAsIs", "kSelectedFrame", "kEachFrameSep"}; static const char * gFrameHowToEnableStr[] = {"kFrameEnableNone", "kFrameEnableAll", "kFrameEnableAsIsAndEach"}; static const char * gPrintRangeStr[] = {"kRangeAllPages", "kRangeSpecifiedPageRange", "kRangeSelection", "kRangeFocusFrame"}; #else #define PRT_YESNO(_p) #define PR_PL(_p1) #endif //----------------------------------------------------- class DocumentViewerImpl; // New PrintPreview static NS_DEFINE_CID(kPrintPreviewContextCID, NS_PRINT_PREVIEW_CONTEXT_CID); // a small delegate class used to avoid circular references #ifdef XP_MAC #pragma mark ** nsDocViewerSelectionListener ** #endif class nsDocViewerSelectionListener : public nsISelectionListener { public: // nsISupports interface... NS_DECL_ISUPPORTS // nsISelectionListerner interface NS_DECL_NSISELECTIONLISTENER nsDocViewerSelectionListener() : mDocViewer(NULL) , mGotSelectionState(PR_FALSE) , mSelectionWasCollapsed(PR_FALSE) { } virtual ~nsDocViewerSelectionListener() {} nsresult Init(DocumentViewerImpl *aDocViewer); protected: DocumentViewerImpl* mDocViewer; PRPackedBool mGotSelectionState; PRPackedBool mSelectionWasCollapsed; }; /** editor Implementation of the FocusListener interface */ class nsDocViewerFocusListener : public nsIDOMFocusListener { public: /** default constructor */ nsDocViewerFocusListener(); /** default destructor */ virtual ~nsDocViewerFocusListener(); /*interfaces for addref and release and queryinterface*/ NS_DECL_ISUPPORTS /*BEGIN implementations of focus event handler interface*/ NS_IMETHOD HandleEvent(nsIDOMEvent* aEvent); NS_IMETHOD Focus(nsIDOMEvent* aEvent); NS_IMETHOD Blur(nsIDOMEvent* aEvent); /*END implementations of focus event handler interface*/ nsresult Init(DocumentViewerImpl *aDocViewer); private: DocumentViewerImpl* mDocViewer; }; #ifdef XP_MAC #pragma mark ** DocumentViewerImpl ** #endif //------------------------------------------------------------- class DocumentViewerImpl : public nsIDocumentViewer, public nsIContentViewerEdit, public nsIContentViewerFile, public nsIMarkupDocumentViewer, public nsIWebBrowserPrint, public nsIDocumentViewerPrint { friend class nsDocViewerSelectionListener; friend class nsPagePrintTimer; friend class nsPrintEngine; public: DocumentViewerImpl(); DocumentViewerImpl(nsIPresContext* aPresContext); NS_DECL_AND_IMPL_ZEROING_OPERATOR_NEW // nsISupports interface... NS_DECL_ISUPPORTS // nsIContentViewer interface... NS_DECL_NSICONTENTVIEWER // nsIDocumentViewer interface... NS_IMETHOD SetUAStyleSheet(nsIStyleSheet* aUAStyleSheet); NS_IMETHOD GetDocument(nsIDocument*& aResult); NS_IMETHOD GetPresShell(nsIPresShell*& aResult); NS_IMETHOD GetPresContext(nsIPresContext*& aResult); NS_IMETHOD CreateDocumentViewerUsing(nsIPresContext* aPresContext, nsIDocumentViewer*& aResult); // nsIContentViewerEdit NS_DECL_NSICONTENTVIEWEREDIT // nsIContentViewerFile NS_DECL_NSICONTENTVIEWERFILE // nsIMarkupDocumentViewer NS_DECL_NSIMARKUPDOCUMENTVIEWER // nsIWebBrowserPrint NS_DECL_NSIWEBBROWSERPRINT typedef void (*CallChildFunc)(nsIMarkupDocumentViewer* aViewer, void* aClosure); nsresult CallChildren(CallChildFunc aFunc, void* aClosure); // nsIDocumentViewerPrint Printing Methods NS_DECL_NSIDOCUMENTVIEWERPRINT // Helpers (also used by nsPrintEngine) PRBool IsWebShellAFrameSet(nsIWebShell * aParent); protected: virtual ~DocumentViewerImpl(); private: void ForceRefresh(void); nsresult MakeWindow(nsIWidget* aParentWidget, const nsRect& aBounds); nsresult InitInternal(nsIWidget* aParentWidget, nsIDeviceContext* aDeviceContext, const nsRect& aBounds, PRBool aDoCreation, PRBool aInPrintPreview); nsresult InitPresentationStuff(PRBool aDoInitialReflow); nsresult GetPopupNode(nsIDOMNode** aNode); nsresult GetPopupLinkNode(nsIDOMNode** aNode); nsresult GetPopupImageNode(nsIDOMNode** aNode); void PrepareToStartLoad(void); nsresult SyncParentSubDocMap(); #ifdef NS_PRINTING // Called when the DocViewer is notified that the state // of Printing or PP has changed void SetIsPrintingInDocShellTree(nsIDocShellTreeNode* aParentNode, PRBool aIsPrintingOrPP, PRBool aStartAtTop); #endif // NS_PRINTING protected: // IMPORTANT: The ownership implicit in the following member // variables has been explicitly checked and set using nsCOMPtr // for owning pointers and raw COM interface pointers for weak // (ie, non owning) references. If you add any members to this // class, please make the ownership explicit (pinkerton, scc). nsISupports* mContainer; // [WEAK] it owns me! nsCOMPtr mDeviceContext; // ??? can't hurt, but... // the following six items are explicitly in this order // so they will be destroyed in the reverse order (pinkerton, scc) nsCOMPtr mDocument; nsCOMPtr mWindow; // ??? should we really own it? nsCOMPtr mViewManager; nsCOMPtr mPresContext; nsCOMPtr mPresShell; nsCOMPtr mUAStyleSheet; nsCOMPtr mSelectionListener; nsCOMPtr mFocusListener; nsCOMPtr mPreviousViewer; PRBool mEnableRendering; PRBool mStopped; PRBool mLoaded; PRInt16 mNumURLStarts; PRInt16 mDestroyRefCount; // a second "refcount" for the document viewer's "destroy" nsIWidget* mParentWidget; // purposely won't be ref counted #ifdef NS_PRINTING nsPrintEngine* mPrintEngine; PRBool mClosingWhilePrinting; nsCOMPtr mDialogParentWin; #if NS_PRINT_PREVIEW // These data member support delayed printing when the document is loading nsCOMPtr mCachedPrintSettings; nsCOMPtr mCachedPrintWebProgressListner; PRPackedBool mPrintIsPending; PRPackedBool mPrintDocIsFullyLoaded; #endif // NS_PRINT_PREVIEW #ifdef NS_DEBUG FILE* mDebugFile; #endif // NS_DEBUG #endif // NS_PRINTING // document management data // these items are specific to markup documents (html and xml) // may consider splitting these out into a subclass PRPackedBool mAllowPlugins; PRPackedBool mIsSticky; /* character set member data */ nsString mDefaultCharacterSet; nsString mHintCharset; PRInt32 mHintCharsetSource; nsString mForceCharacterSet; nsString mPrevDocCharacterSet; }; //------------------------------------------------------------------ // DocumentViewerImpl //------------------------------------------------------------------ // Class IDs static NS_DEFINE_CID(kViewManagerCID, NS_VIEW_MANAGER_CID); static NS_DEFINE_CID(kWidgetCID, NS_CHILD_CID); static NS_DEFINE_CID(kViewCID, NS_VIEW_CID); //------------------------------------------------------------------ nsresult NS_NewDocumentViewer(nsIDocumentViewer** aResult) { *aResult = new DocumentViewerImpl(); if (!*aResult) { return NS_ERROR_OUT_OF_MEMORY; } NS_ADDREF(*aResult); return NS_OK; } // Note: operator new zeros our memory DocumentViewerImpl::DocumentViewerImpl() : mIsSticky(PR_TRUE) { PrepareToStartLoad(); mParentWidget = nsnull; } void DocumentViewerImpl::PrepareToStartLoad() { mEnableRendering = PR_TRUE; mStopped = PR_FALSE; mLoaded = PR_FALSE; #ifdef NS_PRINTING mPrintIsPending = PR_FALSE; mPrintDocIsFullyLoaded = PR_FALSE; mClosingWhilePrinting = PR_FALSE; // Make sure we have destroyed it and cleared the data member if (mPrintEngine) { mPrintEngine->Destroy(); NS_RELEASE(mPrintEngine); } #ifdef NS_PRINT_PREVIEW SetIsPrintPreview(PR_FALSE); #endif #ifdef NS_DEBUG mDebugFile = nsnull; #endif #endif // NS_PRINTING } DocumentViewerImpl::DocumentViewerImpl(nsIPresContext* aPresContext) : mPresContext(aPresContext), mAllowPlugins(PR_TRUE), mIsSticky(PR_TRUE) { mHintCharsetSource = kCharsetUninitialized; PrepareToStartLoad(); } NS_IMPL_ISUPPORTS7(DocumentViewerImpl, nsIContentViewer, nsIDocumentViewer, nsIMarkupDocumentViewer, nsIContentViewerFile, nsIContentViewerEdit, nsIWebBrowserPrint, nsIDocumentViewerPrint) DocumentViewerImpl::~DocumentViewerImpl() { NS_ASSERTION(!mDocument, "User did not call nsIContentViewer::Close"); if (mDocument) { Close(); } NS_ASSERTION(!mPresShell, "User did not call nsIContentViewer::Destroy"); if (mPresShell) { Destroy(); } // XXX(?) Revoke pending invalidate events // clear weak references before we go away if (mPresContext) { mPresContext->SetContainer(nsnull); mPresContext->SetLinkHandler(nsnull); } } /* * This method is called by the Document Loader once a document has * been created for a particular data stream... The content viewer * must cache this document for later use when Init(...) is called. * * This method is also called when an out of band document.write() happens. * In that case, the document passed in is the same as the previous document. */ NS_IMETHODIMP DocumentViewerImpl::LoadStart(nsISupports *aDoc) { #ifdef NOISY_VIEWER printf("DocumentViewerImpl::LoadStart\n"); #endif nsresult rv; if (!mDocument) { mDocument = do_QueryInterface(aDoc, &rv); } else if (mDocument == aDoc) { // Reset the document viewer's state back to what it was // when the document load started. PrepareToStartLoad(); } return rv; } nsresult DocumentViewerImpl::SyncParentSubDocMap() { nsCOMPtr item(do_QueryInterface(mContainer)); nsCOMPtr win(do_GetInterface(item)); nsCOMPtr pwin(do_QueryInterface(win)); nsCOMPtr content; if (mDocument && pwin) { nsCOMPtr frame_element; pwin->GetFrameElementInternal(getter_AddRefs(frame_element)); content = do_QueryInterface(frame_element); } if (content) { nsCOMPtr parent; item->GetParent(getter_AddRefs(parent)); nsCOMPtr parent_win(do_GetInterface(parent)); if (parent_win) { nsCOMPtr dom_doc; parent_win->GetDocument(getter_AddRefs(dom_doc)); nsCOMPtr parent_doc(do_QueryInterface(dom_doc)); if (parent_doc) { return parent_doc->SetSubDocumentFor(content, mDocument); } } } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::SetContainer(nsISupports* aContainer) { mContainer = aContainer; if (mPresContext) { mPresContext->SetContainer(aContainer); } // We're loading a new document into the window where this document // viewer lives, sync the parent document's frame element -> sub // document map return SyncParentSubDocMap(); } NS_IMETHODIMP DocumentViewerImpl::GetContainer(nsISupports** aResult) { NS_ENSURE_ARG_POINTER(aResult); *aResult = mContainer; NS_IF_ADDREF(*aResult); return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::Init(nsIWidget* aParentWidget, nsIDeviceContext* aDeviceContext, const nsRect& aBounds) { return InitInternal(aParentWidget, aDeviceContext, aBounds, PR_TRUE, PR_FALSE); } nsresult DocumentViewerImpl::InitPresentationStuff(PRBool aDoInitialReflow) { // Create the style set... nsCOMPtr styleSet; nsresult rv = CreateStyleSet(mDocument, getter_AddRefs(styleSet)); NS_ENSURE_SUCCESS(rv, rv); // Now make the shell for the document rv = mDocument->CreateShell(mPresContext, mViewManager, styleSet, getter_AddRefs(mPresShell)); NS_ENSURE_SUCCESS(rv, rv); if (aDoInitialReflow) { // Since InitialReflow() will create frames for *all* items // that are currently in the document tree, we need to flush // any pending notifications to prevent the content sink from // duplicating layout frames for content it has added to the tree // but hasn't notified the document about. (Bug 154018) // // Note that we are flushing before we add mPresShell as an observer // to avoid bogus notifications. mDocument->FlushPendingNotifications(PR_FALSE); } mPresShell->BeginObservingDocument(); // Initialize our view manager nsRect bounds; mWindow->GetBounds(bounds); float p2t; mPresContext->GetPixelsToTwips(&p2t); nscoord width = NSIntPixelsToTwips(bounds.width, p2t); nscoord height = NSIntPixelsToTwips(bounds.height, p2t); mViewManager->DisableRefresh(); mViewManager->SetWindowDimensions(width, height); // Setup default view manager background color // This may be overridden by the docshell with the background color // for the last document loaded into the docshell nscolor bgcolor = NS_RGB(0, 0, 0); mPresContext->GetDefaultBackgroundColor(&bgcolor); mViewManager->SetDefaultBackgroundColor(bgcolor); if (aDoInitialReflow) { nsCOMPtr sc = do_QueryInterface(mContainer); if (sc) { nsCOMPtr root; mDocument->GetRootContent(getter_AddRefs(root)); nsCOMPtr frameset(do_QueryInterface(root)); if (frameset) { // If this is a frameset (i.e. not a frame) then we never want // scrollbars on it, the scrollbars go inside the frames // inside the frameset... sc->SetCurrentScrollbarPreferences(nsIScrollable::ScrollOrientation_Y, NS_STYLE_OVERFLOW_HIDDEN); sc->SetCurrentScrollbarPreferences(nsIScrollable::ScrollOrientation_X, NS_STYLE_OVERFLOW_HIDDEN); } else { sc->ResetScrollbarPreferences(); } } // Initial reflow mPresShell->InitialReflow(width, height); // Now trigger a refresh if (mEnableRendering) { mViewManager->EnableRefresh(NS_VMREFRESH_IMMEDIATE); } } // now register ourselves as a selection listener, so that we get // called when the selection changes in the window nsDocViewerSelectionListener *selectionListener = new nsDocViewerSelectionListener(); NS_ENSURE_TRUE(selectionListener, NS_ERROR_OUT_OF_MEMORY); selectionListener->Init(this); // mSelectionListener is a owning reference mSelectionListener = selectionListener; nsCOMPtr selection; rv = GetDocumentSelection(getter_AddRefs(selection)); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr selPrivate(do_QueryInterface(selection)); rv = selPrivate->AddSelectionListener(mSelectionListener); if (NS_FAILED(rv)) return rv; // Save old listener so we can unregister it nsCOMPtr mOldFocusListener = mFocusListener; // focus listener // // now register ourselves as a focus listener, so that we get called // when the focus changes in the window nsDocViewerFocusListener *focusListener; NS_NEWXPCOM(focusListener, nsDocViewerFocusListener); NS_ENSURE_TRUE(focusListener, NS_ERROR_OUT_OF_MEMORY); focusListener->Init(this); // mFocusListener is a strong reference mFocusListener = focusListener; // get the DOM event receiver nsCOMPtr erP(do_QueryInterface(mDocument)); NS_WARN_IF_FALSE(erP, "No event receiver in document!"); if (erP) { rv = erP->AddEventListenerByIID(mFocusListener, NS_GET_IID(nsIDOMFocusListener)); NS_ASSERTION(NS_SUCCEEDED(rv), "failed to register focus listener"); if (mOldFocusListener) { rv = erP->RemoveEventListenerByIID(mOldFocusListener, NS_GET_IID(nsIDOMFocusListener)); NS_ASSERTION(NS_SUCCEEDED(rv), "failed to remove focus listener"); } } return NS_OK; } //----------------------------------------------- // This method can be used to initial the "presentation" // The aDoCreation indicates whether it should create // all the new objects or just initialize the existing ones nsresult DocumentViewerImpl::InitInternal(nsIWidget* aParentWidget, nsIDeviceContext* aDeviceContext, const nsRect& aBounds, PRBool aDoCreation, PRBool aInPrintPreview) { mParentWidget = aParentWidget; // not ref counted nsresult rv = NS_OK; NS_ENSURE_TRUE(mDocument, NS_ERROR_NULL_POINTER); mDeviceContext = dont_QueryInterface(aDeviceContext); #if defined(NS_PRINTING) && defined(NS_PRINT_PREVIEW) // Clear PrintPreview Alternate Device if (mDeviceContext) { mDeviceContext->SetAltDevice(nsnull); mDeviceContext->SetCanonicalPixelScale(1.0); } #endif PRBool makeCX = PR_FALSE; if (aDoCreation) { if (aParentWidget && !mPresContext) { // Create presentation context if (GetIsCreatingPrintPreview()) { mPresContext = do_CreateInstance(kPrintPreviewContextCID, &rv); } else { mPresContext = do_CreateInstance(kGalleyContextCID, &rv); } if (NS_FAILED(rv)) return rv; mPresContext->Init(aDeviceContext); #if defined(NS_PRINTING) && defined(NS_PRINT_PREVIEW) makeCX = !GetIsPrintPreview(); // needs to be true except when we are already in PP #else makeCX = PR_TRUE; #endif } } if (aDoCreation && mPresContext) { // Create the ViewManager and Root View... // We must do this before we tell the script global object about // this new document since doing that will cause us to re-enter // into nsHTMLFrameInnerFrame code through reflows caused by // FlushPendingNotifications() calls down the road... rv = MakeWindow(aParentWidget, aBounds); NS_ENSURE_SUCCESS(rv, rv); Hide(); } nsCOMPtr requestor(do_QueryInterface(mContainer)); if (requestor) { if (mPresContext) { nsCOMPtr linkHandler; requestor->GetInterface(NS_GET_IID(nsILinkHandler), getter_AddRefs(linkHandler)); mPresContext->SetContainer(mContainer); mPresContext->SetLinkHandler(linkHandler); } if (!aInPrintPreview) { // Set script-context-owner in the document nsCOMPtr global; requestor->GetInterface(NS_GET_IID(nsIScriptGlobalObject), getter_AddRefs(global)); if (global) { mDocument->SetScriptGlobalObject(global); nsCOMPtr domdoc(do_QueryInterface(mDocument)); if (domdoc) { global->SetNewDocument(domdoc, PR_TRUE, PR_TRUE); } } } } if (aDoCreation && mPresContext) { // The ViewManager and Root View was created above (in // MakeWindow())... rv = InitPresentationStuff(!makeCX); } return rv; } // // LoadComplete(aStatus) // // aStatus - The status returned from loading the document. // // This method is called by the container when the document has been // completely loaded. // NS_IMETHODIMP DocumentViewerImpl::LoadComplete(nsresult aStatus) { nsresult rv = NS_OK; NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); nsCOMPtr global; // First, get the script global object from the document... rv = mDocument->GetScriptGlobalObject(getter_AddRefs(global)); // Fail if no ScriptGlobalObject is available... NS_ENSURE_TRUE(global, NS_ERROR_NULL_POINTER); mLoaded = PR_TRUE; /* We need to protect ourself against auto-destruction in case the window is closed while processing the OnLoad event. See bug http://bugzilla.mozilla.org/show_bug.cgi?id=78445 for more explanation. */ nsCOMPtr kungFuDeathGrip(this); // Now, fire either an OnLoad or OnError event to the document... if(NS_SUCCEEDED(aStatus)) { nsEventStatus status = nsEventStatus_eIgnore; nsEvent event; event.eventStructType = NS_EVENT; event.message = NS_PAGE_LOAD; rv = global->HandleDOMEvent(mPresContext, &event, nsnull, NS_EVENT_FLAG_INIT, &status); #ifdef MOZ_TIMELINE // if navigator.xul's load is complete, the main nav window is visible // mark that point. if (mDocument) { //printf("DEBUG: getting uri from document (%p)\n", mDocument.get()); nsCOMPtr uri; mDocument->GetDocumentURL(getter_AddRefs(uri)); if (uri) { //printf("DEBUG: getting spec fro uri (%p)\n", uri.get()); nsCAutoString spec; uri->GetSpec(spec); if (!strcmp(spec.get(), "chrome://navigator/content/navigator.xul")) { NS_TIMELINE_MARK("Navigator Window visible now"); } } } #endif /* MOZ_TIMELINE */ } else { // XXX: Should fire error event to the document... } // Now that the document has loaded, we can tell the presshell // to unsuppress painting. if (mPresShell && !mStopped) { mPresShell->UnsuppressPainting(); } #ifdef NS_PRINTING // Check to see if someone tried to print during the load if (mPrintIsPending) { mPrintIsPending = PR_FALSE; mPrintDocIsFullyLoaded = PR_TRUE; Print(mCachedPrintSettings, mCachedPrintWebProgressListner); mCachedPrintSettings = nsnull; mCachedPrintWebProgressListner = nsnull; } #endif return rv; } NS_IMETHODIMP DocumentViewerImpl::Unload() { nsresult rv; if (!mDocument) { return NS_ERROR_NULL_POINTER; } // First, get the script global object from the document... nsCOMPtr global; rv = mDocument->GetScriptGlobalObject(getter_AddRefs(global)); if (!global) { // Fail if no ScriptGlobalObject is available... NS_ASSERTION(0, "nsIScriptGlobalObject not set for document!"); return NS_ERROR_NULL_POINTER; } // Now, fire an Unload event to the document... nsEventStatus status = nsEventStatus_eIgnore; nsEvent event; event.eventStructType = NS_EVENT; event.message = NS_PAGE_UNLOAD; rv = global->HandleDOMEvent(mPresContext, &event, nsnull, NS_EVENT_FLAG_INIT, &status); return rv; } NS_IMETHODIMP DocumentViewerImpl::Close() { // All callers are supposed to call close to break circular // references. If we do this stuff in the destructor, the // destructor might never be called (especially if we're being // used from JS. // Close is also needed to disable scripts during paint suppression, // since we transfer the existing global object to the new document // that is loaded. In the future, the global object may become a proxy // for an object that can be switched in and out so that we don't need // to disable scripts during paint suppression. if (mDocument) { #if defined(NS_PRINTING) && defined(NS_PRINT_PREVIEW) // Turn scripting back on // after PrintPreview had turned it off if (GetIsPrintPreview() && mPrintEngine) { mPrintEngine->TurnScriptingOn(PR_TRUE); } #endif // Before we clear the script global object, clear the undisplayed // content map, since XBL content can be destroyed by the // |SetDocument(null, ...)| triggered by calling // |SetScriptGlobalObject(null)|. if (mPresShell) { nsCOMPtr frameManager; mPresShell->GetFrameManager(getter_AddRefs(frameManager)); if (frameManager) frameManager->ClearUndisplayedContentMap(); } // Break global object circular reference on the document created // in the DocViewer Init nsCOMPtr globalObject; mDocument->GetScriptGlobalObject(getter_AddRefs(globalObject)); if (globalObject) { globalObject->SetNewDocument(nsnull, PR_TRUE, PR_TRUE); } #ifdef NS_PRINTING // A Close was called while we were printing // so don't clear the ScriptGlobalObject // or clear the mDocument below // Also, do an extra addref to keep the viewer from going away. if (mPrintEngine && !mClosingWhilePrinting) { mClosingWhilePrinting = PR_TRUE; NS_ADDREF_THIS(); } else { // out of band cleanup of webshell mDocument->SetScriptGlobalObject(nsnull); } #else mDocument->SetScriptGlobalObject(nsnull); #endif if (mFocusListener) { // get the DOM event receiver nsCOMPtr erP(do_QueryInterface(mDocument)); NS_WARN_IF_FALSE(erP, "No event receiver in document!"); if (erP) { erP->RemoveEventListenerByIID(mFocusListener, NS_GET_IID(nsIDOMFocusListener)); } } } #ifdef NS_PRINTING // Don't clear the document if we are printing. if (!mClosingWhilePrinting) { mDocument = nsnull; } #else mDocument = nsnull; #endif return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::Destroy() { #ifdef NS_PRINTING // Here is where we check to see if the docment was still being prepared // for printing when it was asked to be destroy from someone externally // This usually happens if the document is unloaded while the user is in the Print Dialog // // So we flip the bool to remember that the document is going away // and we can clean up and abort later after returning from the Print Dialog if (mPrintEngine) { if (mPrintEngine->CheckBeforeDestroy()) { return NS_OK; } } #endif // Don't let the document get unloaded while we are printing // this could happen if we hit the back button during printing if (mDestroyRefCount != 0) { --mDestroyRefCount; return NS_OK; } // All callers are supposed to call destroy to break circular // references. If we do this stuff in the destructor, the // destructor might never be called (especially if we're being // used from JS. #ifdef NS_PRINTING if (mPrintEngine) { mPrintEngine->Destroy(); NS_RELEASE(mPrintEngine); } #endif // Avoid leaking the old viewer. if (mPreviousViewer) { mPreviousViewer->Destroy(); mPreviousViewer = nsnull; } if (mDeviceContext) { mDeviceContext->FlushFontCache(); mDeviceContext = nsnull; } if (mPresShell) { // Break circular reference (or something) mPresShell->EndObservingDocument(); nsCOMPtr selection; GetDocumentSelection(getter_AddRefs(selection)); nsCOMPtr selPrivate(do_QueryInterface(selection)); if (selPrivate && mSelectionListener) selPrivate->RemoveSelectionListener(mSelectionListener); mPresShell->Destroy(); mPresShell = nsnull; } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::Stop(void) { NS_ASSERTION(mDocument, "Stop called too early or too late"); if (mDocument) { mDocument->StopDocumentLoad(); } mStopped = PR_TRUE; if (!mLoaded && mPresShell) { // Well, we might as well paint what we have so far. mPresShell->UnsuppressPainting(); } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetDOMDocument(nsIDOMDocument **aResult) { NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); return CallQueryInterface(mDocument.get(), aResult); } NS_IMETHODIMP DocumentViewerImpl::SetDOMDocument(nsIDOMDocument *aDocument) { // Assumptions: // // 1) this document viewer has been initialized with a call to Init(). // 2) the stylesheets associated with the document have been added // to the document. // XXX Right now, this method assumes that the layout of the current // document hasn't started yet. More cleanup will probably be // necessary to make this method work for the case when layout *has* // occurred for the current document. // That work can happen when and if it is needed. nsresult rv; if (!aDocument) return NS_ERROR_NULL_POINTER; nsCOMPtr newDoc = do_QueryInterface(aDocument, &rv); if (NS_FAILED(rv)) return rv; // 0) Replace the old document with the new one mDocument = newDoc; // 1) Set the script global object on the new document nsCOMPtr global(do_GetInterface(mContainer)); if (global) { mDocument->SetScriptGlobalObject(global); global->SetNewDocument(aDocument, PR_TRUE, PR_TRUE); rv = SyncParentSubDocMap(); NS_ENSURE_SUCCESS(rv, rv); } // 2) Replace the current pres shell with a new shell for the new document if (mPresShell) { mPresShell->EndObservingDocument(); mPresShell->Destroy(); mPresShell = nsnull; } // And if we're already given a prescontext... if (mPresContext) { // 3) Create a new style set for the document nsCOMPtr styleSet; rv = CreateStyleSet(mDocument, getter_AddRefs(styleSet)); if (NS_FAILED(rv)) return rv; rv = newDoc->CreateShell(mPresContext, mViewManager, styleSet, getter_AddRefs(mPresShell)); NS_ENSURE_SUCCESS(rv, rv); mPresShell->BeginObservingDocument(); // 4) Register the focus listener on the new document nsCOMPtr erP = do_QueryInterface(mDocument, &rv); NS_WARN_IF_FALSE(erP, "No event receiver in document!"); if (erP) { rv = erP->AddEventListenerByIID(mFocusListener, NS_GET_IID(nsIDOMFocusListener)); NS_ASSERTION(NS_SUCCEEDED(rv), "failed to register focus listener"); } } return rv; } NS_IMETHODIMP DocumentViewerImpl::SetUAStyleSheet(nsIStyleSheet* aUAStyleSheet) { NS_ASSERTION(aUAStyleSheet, "unexpected null pointer"); if (aUAStyleSheet) { nsCOMPtr sheet(do_QueryInterface(aUAStyleSheet)); nsCOMPtr newSheet; sheet->Clone(*getter_AddRefs(newSheet)); mUAStyleSheet = newSheet; } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetDocument(nsIDocument*& aResult) { aResult = mDocument; NS_IF_ADDREF(aResult); return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetPresShell(nsIPresShell*& aResult) { aResult = mPresShell; NS_IF_ADDREF(aResult); return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetPresContext(nsIPresContext*& aResult) { aResult = mPresContext; NS_IF_ADDREF(aResult); return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetBounds(nsRect& aResult) { NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); NS_PRECONDITION(mWindow, "null window"); if (mWindow) { mWindow->GetBounds(aResult); } else { aResult.SetRect(0, 0, 0, 0); } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetPreviousViewer(nsIContentViewer** aViewer) { *aViewer = mPreviousViewer; NS_IF_ADDREF(*aViewer); return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::SetPreviousViewer(nsIContentViewer* aViewer) { // NOTE: |Show| sets |mPreviousViewer| to null without calling this // function. if (aViewer) { NS_ASSERTION(!mPreviousViewer, "can't set previous viewer when there already is one"); // In a multiple chaining situation (which occurs when running a thrashing // test like i-bench or jrgm's tests with no delay), we can build up a // whole chain of viewers. In order to avoid this, we always set our previous // viewer to the MOST previous viewer in the chain, and then dump the intermediate // link from the chain. This ensures that at most only 2 documents are alive // and undestroyed at any given time (the one that is showing and the one that // is loading with painting suppressed). nsCOMPtr prevViewer; aViewer->GetPreviousViewer(getter_AddRefs(prevViewer)); if (prevViewer) { aViewer->SetPreviousViewer(nsnull); aViewer->Destroy(); return SetPreviousViewer(prevViewer); } } mPreviousViewer = aViewer; return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::SetBounds(const nsRect& aBounds) { NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); if (mWindow) { // Don't have the widget repaint. Layout will generate repaint requests // during reflow mWindow->Resize(aBounds.x, aBounds.y, aBounds.width, aBounds.height, PR_FALSE); } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::Move(PRInt32 aX, PRInt32 aY) { NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); NS_PRECONDITION(mWindow, "null window"); if (mWindow) { mWindow->Move(aX, aY); } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::Show(void) { NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); // We don't need the previous viewer anymore since we're not // displaying it. if (mPreviousViewer) { // This little dance *may* only be to keep // PresShell::EndObservingDocument happy, but I'm not sure. nsCOMPtr prevViewer(mPreviousViewer); mPreviousViewer = nsnull; prevViewer->Destroy(); } if (mWindow) { mWindow->Show(PR_TRUE); } if (mDocument && !mPresShell && !mWindow) { nsresult rv; nsCOMPtr base_win(do_QueryInterface(mContainer)); NS_ENSURE_TRUE(base_win, NS_ERROR_UNEXPECTED); base_win->GetParentWidget(&mParentWidget); NS_ENSURE_TRUE(mParentWidget, NS_ERROR_UNEXPECTED); mDeviceContext = dont_AddRef(mParentWidget->GetDeviceContext()); #if defined(NS_PRINTING) && defined(NS_PRINT_PREVIEW) // Clear PrintPreview Alternate Device if (mDeviceContext) { mDeviceContext->SetAltDevice(nsnull); } #endif // Create presentation context if (GetIsCreatingPrintPreview()) { NS_ERROR("Whoa, we should not get here!"); return NS_ERROR_UNEXPECTED; } mPresContext = do_CreateInstance(kGalleyContextCID, &rv); NS_ENSURE_SUCCESS(rv, rv); mPresContext->Init(mDeviceContext); nsRect tbounds; mParentWidget->GetBounds(tbounds); float p2t; mPresContext->GetPixelsToTwips(&p2t); tbounds *= p2t; // Create the view manager mViewManager = do_CreateInstance(kViewManagerCID); NS_ENSURE_TRUE(mViewManager, NS_ERROR_NOT_AVAILABLE); // Initialize the view manager with an offset. This allows the // viewmanager to manage a coordinate space offset from (0,0) rv = mViewManager->Init(mDeviceContext); if (NS_FAILED(rv)) return rv; rv = mViewManager->SetWindowOffset(tbounds.x, tbounds.y); if (NS_FAILED(rv)) return rv; // Reset the bounds offset so the root view is set to 0,0. The // offset is specified in nsIViewManager::Init above. // Besides, layout will reset the root view to (0,0) during reflow, // so changing it to 0,0 eliminates placing the root view in the // wrong place initially. tbounds.x = 0; tbounds.y = 0; // Create a child window of the parent that is our "root // view/window" Create a view nsIView *view = nsnull; rv = CallCreateInstance(kViewCID, &view); if (NS_FAILED(rv)) return rv; rv = view->Init(mViewManager, tbounds, nsnull); if (NS_FAILED(rv)) return rv; rv = view->CreateWidget(kWidgetCID, nsnull, mParentWidget->GetNativeData(NS_NATIVE_WIDGET), PR_TRUE, PR_FALSE); if (NS_FAILED(rv)) return rv; // Setup hierarchical relationship in view manager mViewManager->SetRootView(view); view->GetWidget(*getter_AddRefs(mWindow)); if (mPresContext && mContainer) { nsCOMPtr linkHandler(do_GetInterface(mContainer)); if (linkHandler) { mPresContext->SetLinkHandler(linkHandler); } mPresContext->SetContainer(mContainer); } if (mPresContext) { Hide(); rv = InitPresentationStuff(PR_TRUE); } // If we get here the document load has already started and the // window is shown because some JS on the page caused it to be // shown... mPresShell->UnsuppressPainting(); } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::Hide(void) { PRBool is_in_print_mode = PR_FALSE; GetDoingPrint(&is_in_print_mode); if (is_in_print_mode) { // If we, or one of our parents, is in print mode it means we're // right now returning from print and the layout frame that was // created for this document is being destroyed. In such a case we // ignore the Hide() call. return NS_OK; } GetDoingPrintPreview(&is_in_print_mode); if (is_in_print_mode) { // If we, or one of our parents, is in print preview mode it means // we're right now returning from print preview and the layout // frame that was created for this document is being destroyed. In // such a case we ignore the Hide() call. return NS_OK; } NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); NS_PRECONDITION(mWindow, "null window"); if (mWindow) { mWindow->Show(PR_FALSE); } if (!mPresShell || GetIsPrintPreview()) { return NS_OK; } // Avoid leaking the old viewer. if (mPreviousViewer) { mPreviousViewer->Destroy(); mPreviousViewer = nsnull; } if (mIsSticky) { // This window is sticky, that means that it might be shown again // and we don't want the presshell n' all that to be thrown away // just because the window is hidden. return NS_OK; } if (mDeviceContext) { mDeviceContext->FlushFontCache(); } // Break circular reference (or something) mPresShell->EndObservingDocument(); nsCOMPtr selection; GetDocumentSelection(getter_AddRefs(selection)); nsCOMPtr selPrivate(do_QueryInterface(selection)); if (selPrivate && mSelectionListener) { selPrivate->RemoveSelectionListener(mSelectionListener); } nsCOMPtr xul_doc(do_QueryInterface(mDocument)); if (xul_doc) { xul_doc->OnHide(); } mPresShell->Destroy(); mPresShell = nsnull; mPresContext = nsnull; mViewManager = nsnull; mWindow = nsnull; mDeviceContext = nsnull; mParentWidget = nsnull; nsCOMPtr base_win(do_QueryInterface(mContainer)); if (base_win) { base_win->SetParentWidget(nsnull); } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::SetEnableRendering(PRBool aOn) { NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); mEnableRendering = aOn; if (mViewManager) { if (aOn) { mViewManager->EnableRefresh(NS_VMREFRESH_IMMEDIATE); nsIView* view; mViewManager->GetRootView(view); // views are not refCounted if (view) { mViewManager->UpdateView(view, NS_VMREFRESH_IMMEDIATE); } } else { mViewManager->DisableRefresh(); } } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetSticky(PRBool *aSticky) { *aSticky = mIsSticky; return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::SetSticky(PRBool aSticky) { mIsSticky = aSticky; return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetEnableRendering(PRBool* aResult) { NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); NS_PRECONDITION(nsnull != aResult, "null OUT ptr"); if (aResult) { *aResult = mEnableRendering; } return NS_OK; } void DocumentViewerImpl::ForceRefresh() { mWindow->Invalidate(PR_TRUE); } nsresult DocumentViewerImpl::CreateStyleSet(nsIDocument* aDocument, nsIStyleSet** aStyleSet) { // this should eventually get expanded to allow for creating // different sets for different media nsresult rv; if (!mUAStyleSheet) { NS_WARNING("unable to load UA style sheet"); } rv = CallCreateInstance(kStyleSetCID, aStyleSet); if (NS_OK == rv) { PRInt32 index = 0; aDocument->GetNumberOfStyleSheets(PR_TRUE, &index); while (0 < index--) { nsCOMPtr sheet; aDocument->GetStyleSheetAt(index, PR_TRUE, getter_AddRefs(sheet)); /* * GetStyleSheetAt will return all style sheets in the document but * we're only interested in the ones that are enabled. */ PRBool styleApplicable; sheet->GetApplicable(styleApplicable); if (styleApplicable) { (*aStyleSet)->AddDocStyleSheet(sheet, aDocument); } } nsCOMPtr chromeRegistry = do_GetService("@mozilla.org/chrome/chrome-registry;1"); if (chromeRegistry) { nsCOMPtr sheets; // Now handle the user sheets. nsCOMPtr docShell(do_QueryInterface(mContainer)); PRInt32 shellType; docShell->GetItemType(&shellType); PRBool isChrome = (shellType == nsIDocShellTreeItem::typeChrome); sheets = nsnull; chromeRegistry->GetUserSheets(isChrome, getter_AddRefs(sheets)); if(sheets){ nsCOMPtr sheet; PRUint32 count; sheets->Count(&count); for(PRUint32 i=0; iGetElementAt(i, getter_AddRefs(sheet)); (*aStyleSet)->AppendUserStyleSheet(sheet); } } // Append chrome sheets (scrollbars + forms). nsCOMPtr ds(do_QueryInterface(mContainer)); chromeRegistry->GetAgentSheets(ds, getter_AddRefs(sheets)); if(sheets){ nsCOMPtr sheet; PRUint32 count; sheets->Count(&count); for(PRUint32 i=0; iGetElementAt(i, getter_AddRefs(sheet)); (*aStyleSet)->AppendAgentStyleSheet(sheet); } } } if (mUAStyleSheet) { (*aStyleSet)->AppendAgentStyleSheet(mUAStyleSheet); } } return NS_OK; } //--------------------------------------------------------------------- // Note this is also defined in nsPrintEngine // They can't share it because nsPrintEngine may not be available // when printing isn't turned on static nsIPresShell* GetPresShellFor(nsIDocShell* aDocShell) { nsCOMPtr domDoc(do_GetInterface(aDocShell)); if (!domDoc) return nsnull; nsCOMPtr doc(do_QueryInterface(domDoc)); if (!doc) return nsnull; nsIPresShell* shell = nsnull; doc->GetShellAt(0, &shell); return shell; } //--------------------------------------------------------------------- // This gets ref counted copies of the PresShell and Root Content // for a given nsIWebShell void DocumentViewerImpl::GetPresShellAndRootContent(nsIWebShell * aWebShell, nsIPresShell** aPresShell, nsIContent** aContent) { NS_ASSERTION(aWebShell, "Pointer is null!"); NS_ASSERTION(aPresShell, "Pointer is null!"); NS_ASSERTION(aContent, "Pointer is null!"); *aContent = nsnull; *aPresShell = nsnull; nsCOMPtr docShell(do_QueryInterface(aWebShell)); nsCOMPtr presShell(getter_AddRefs(GetPresShellFor(docShell))); if (!presShell) return; nsCOMPtr doc; presShell->GetDocument(getter_AddRefs(doc)); if (!doc) return; doc->GetRootContent(aContent); // this addrefs *aPresShell = presShell.get(); NS_ADDREF(*aPresShell); } //--------------------------------------------------------------------- nsresult DocumentViewerImpl::FindFrameSetWithIID(nsIContent * aParentContent, const nsIID& aIID) { PRInt32 numChildren; aParentContent->ChildCount(numChildren); // do a breadth search across all siblings PRInt32 inx; for (inx = 0; inx < numChildren; ++inx) { nsCOMPtr child; if (NS_SUCCEEDED(aParentContent->ChildAt(inx, *getter_AddRefs(child))) && child) { nsCOMPtr temp; if (NS_SUCCEEDED(child->QueryInterface(aIID, (void**)getter_AddRefs(temp)))) { return NS_OK; } } } return NS_ERROR_FAILURE; } //------------------------------------------------------- PRBool DocumentViewerImpl::IsWebShellAFrameSet(nsIWebShell * aWebShell) { NS_ASSERTION(aWebShell, "Pointer is null!"); PRBool doesContainFrameSet = PR_FALSE; nsCOMPtr presShell; nsCOMPtr rootContent; GetPresShellAndRootContent(aWebShell, getter_AddRefs(presShell), getter_AddRefs(rootContent)); if (rootContent) { if (NS_SUCCEEDED(FindFrameSetWithIID(rootContent, NS_GET_IID(nsIDOMHTMLFrameSetElement)))) { doesContainFrameSet = PR_TRUE; } } return doesContainFrameSet; } nsresult DocumentViewerImpl::MakeWindow(nsIWidget* aParentWidget, const nsRect& aBounds) { nsresult rv; mViewManager = do_CreateInstance(kViewManagerCID, &rv); if (NS_FAILED(rv)) return rv; nsCOMPtr dx; mPresContext->GetDeviceContext(getter_AddRefs(dx)); nsRect tbounds = aBounds; float p2t; mPresContext->GetPixelsToTwips(&p2t); tbounds *= p2t; // Initialize the view manager with an offset. This allows the viewmanager // to manage a coordinate space offset from (0,0) rv = mViewManager->Init(dx); if (NS_FAILED(rv)) return rv; rv = mViewManager->SetWindowOffset(tbounds.x, tbounds.y); if (NS_FAILED(rv)) return rv; // Reset the bounds offset so the root view is set to 0,0. The // offset is specified in nsIViewManager::Init above. // Besides, layout will reset the root view to (0,0) during reflow, // so changing it to 0,0 eliminates placing the root view in the // wrong place initially. tbounds.x = 0; tbounds.y = 0; // Create a child window of the parent that is our "root view/window" // Create a view nsIView *view = nsnull; rv = CallCreateInstance(kViewCID, &view); if (NS_FAILED(rv)) return rv; // if aParentWidget has a view, we'll hook our view manager up to its view tree void* clientData; nsIView* containerView = nsnull; if (NS_SUCCEEDED(aParentWidget->GetClientData(clientData))) { nsISupports* data = (nsISupports*)clientData; if (data) { CallQueryInterface(data, &containerView); } } if (containerView) { // see if the containerView has already been hooked into a foreign view manager hierarchy // if it has, then we have to hook into the hierarchy too otherwise bad things will happen. nsCOMPtr containerVM; containerView->GetViewManager(*getter_AddRefs(containerVM)); nsCOMPtr checkVM; nsIView* pView = containerView; do { pView->GetParent(pView); } while (pView != nsnull && NS_SUCCEEDED(pView->GetViewManager(*getter_AddRefs(checkVM))) && checkVM == containerVM); if (!pView) { // OK, so the container is not already hooked up into a foreign view manager hierarchy. // That means we can choose not to hook ourselves up. // // If the parent container is a chrome shell, or a frameset, then we won't hook into its view // tree. This will improve performance a little bit (especially given scrolling/painting perf bugs) // but is really just for peace of mind. This check can be removed if we want to support fancy // chrome effects like transparent controls floating over content, transparent Web browsers, and // things like that, and the perf bugs are fixed. nsCOMPtr container(do_QueryInterface(mContainer)); nsCOMPtr parentContainer; PRInt32 itemType; if (nsnull == container || NS_FAILED(container->GetParent(getter_AddRefs(parentContainer))) || nsnull == parentContainer || NS_FAILED(parentContainer->GetItemType(&itemType)) || itemType != nsIDocShellTreeItem::typeContent) { containerView = nsnull; } else { nsCOMPtr webShell(do_QueryInterface(parentContainer)); if (nsnull == webShell || IsWebShellAFrameSet(webShell)) { containerView = nsnull; } } } } rv = view->Init(mViewManager, tbounds, containerView); if (NS_FAILED(rv)) return rv; // pass in a native widget to be the parent widget ONLY if the view hierarchy will stand alone. // otherwise the view will find its own parent widget and "do the right thing" to // establish a parent/child widget relationship rv = view->CreateWidget(kWidgetCID, nsnull, containerView != nsnull ? nsnull : aParentWidget->GetNativeData(NS_NATIVE_WIDGET), PR_TRUE, PR_FALSE); if (rv != NS_OK) return rv; // Setup hierarchical relationship in view manager mViewManager->SetRootView(view); view->GetWidget(*getter_AddRefs(mWindow)); // This SetFocus is necessary so the Arrow Key and Page Key events // go to the scrolled view as soon as the Window is created instead of going to // the browser window (this enables keyboard scrolling of the document) // mWindow->SetFocus(); return rv; } nsresult DocumentViewerImpl::GetDocumentSelection(nsISelection **aSelection, nsIPresShell *aPresShell) { if (!aPresShell) { if (!mPresShell) { return NS_ERROR_NOT_INITIALIZED; } aPresShell = mPresShell; } if (!aSelection) return NS_ERROR_NULL_POINTER; if (!aPresShell) return NS_ERROR_NULL_POINTER; nsCOMPtr selcon; selcon = do_QueryInterface(aPresShell); if (selcon) return selcon->GetSelection(nsISelectionController::SELECTION_NORMAL, aSelection); return NS_ERROR_FAILURE; } NS_IMETHODIMP DocumentViewerImpl::CreateDocumentViewerUsing(nsIPresContext* aPresContext, nsIDocumentViewer*& aResult) { if (!mDocument) { // XXX better error return NS_ERROR_NULL_POINTER; } if (!aPresContext) { return NS_ERROR_NULL_POINTER; } // Create new viewer DocumentViewerImpl* viewer = new DocumentViewerImpl(aPresContext); if (!viewer) { return NS_ERROR_OUT_OF_MEMORY; } NS_ADDREF(viewer); // XXX make sure the ua style sheet is used (for now; need to be // able to specify an alternate) viewer->SetUAStyleSheet(mUAStyleSheet); // Bind the new viewer to the old document nsresult rv = viewer->LoadStart(mDocument); aResult = viewer; return rv; } #ifdef XP_MAC #pragma mark - #endif /* ======================================================================================== * nsIContentViewerEdit * ======================================================================================== */ NS_IMETHODIMP DocumentViewerImpl::Search() { NS_ASSERTION(0, "NOT IMPLEMENTED"); return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP DocumentViewerImpl::GetSearchable(PRBool *aSearchable) { NS_ASSERTION(0, "NOT IMPLEMENTED"); return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP DocumentViewerImpl::ClearSelection() { nsresult rv; nsCOMPtr selection; rv = GetDocumentSelection(getter_AddRefs(selection)); if (NS_FAILED(rv)) return rv; return selection->CollapseToStart(); } NS_IMETHODIMP DocumentViewerImpl::SelectAll() { // XXX this is a temporary implementation copied from nsWebShell // for now. I think nsDocument and friends should have some helper // functions to make this easier. nsCOMPtr selection; nsresult rv; rv = GetDocumentSelection(getter_AddRefs(selection)); if (NS_FAILED(rv)) return rv; nsCOMPtr htmldoc = do_QueryInterface(mDocument); nsCOMPtr bodyNode; if (htmldoc) { nsCOMPtrbodyElement; rv = htmldoc->GetBody(getter_AddRefs(bodyElement)); if (NS_FAILED(rv) || !bodyElement) return rv; bodyNode = do_QueryInterface(bodyElement); } else if (mDocument) { nsCOMPtr rootContent; mDocument->GetRootContent(getter_AddRefs(rootContent)); bodyNode = do_QueryInterface(rootContent); } if (!bodyNode) return NS_ERROR_FAILURE; rv = selection->RemoveAllRanges(); if (NS_FAILED(rv)) return rv; rv = selection->SelectAllChildren(bodyNode); return rv; } NS_IMETHODIMP DocumentViewerImpl::CopySelection() { NS_ENSURE_TRUE(mPresShell, NS_ERROR_NOT_INITIALIZED); return mPresShell->DoCopy(); } NS_IMETHODIMP DocumentViewerImpl::CopyLinkLocation() { NS_ENSURE_TRUE(mPresShell, NS_ERROR_NOT_INITIALIZED); nsCOMPtr node; nsresult rv = GetPopupLinkNode(getter_AddRefs(node)); // make noise if we're not in a link NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(node, NS_ERROR_FAILURE); nsAutoString locationText; rv = mPresShell->GetLinkLocation(node, locationText); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr clipboard(do_GetService("@mozilla.org/widget/clipboardhelper;1", &rv)); NS_ENSURE_SUCCESS(rv, rv); // copy the href onto the clipboard return clipboard->CopyString(locationText); } NS_IMETHODIMP DocumentViewerImpl::CopyImageLocation() { NS_ENSURE_TRUE(mPresShell, NS_ERROR_NOT_INITIALIZED); nsCOMPtr node; nsresult rv = GetPopupImageNode(getter_AddRefs(node)); // make noise if we're not in an image NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(node, NS_ERROR_FAILURE); nsAutoString locationText; rv = mPresShell->GetImageLocation(node, locationText); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr clipboard(do_GetService("@mozilla.org/widget/clipboardhelper;1", &rv)); NS_ENSURE_SUCCESS(rv, rv); // copy the href onto the clipboard return clipboard->CopyString(locationText); } NS_IMETHODIMP DocumentViewerImpl::CopyImageContents() { NS_ENSURE_TRUE(mPresShell, NS_ERROR_NOT_INITIALIZED); nsCOMPtr node; nsresult rv = GetPopupImageNode(getter_AddRefs(node)); // make noise if we're not in an image NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(node, NS_ERROR_FAILURE); return mPresShell->DoCopyImageContents(node); } NS_IMETHODIMP DocumentViewerImpl::GetCopyable(PRBool *aCopyable) { nsCOMPtr selection; nsresult rv; rv = GetDocumentSelection(getter_AddRefs(selection)); if (NS_FAILED(rv)) return rv; PRBool isCollapsed; selection->GetIsCollapsed(&isCollapsed); *aCopyable = !isCollapsed; return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::CutSelection() { NS_ASSERTION(0, "NOT IMPLEMENTED"); return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP DocumentViewerImpl::GetCutable(PRBool *aCutable) { *aCutable = PR_FALSE; // mm, will this ever be called for an editable document? return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::Paste() { NS_ASSERTION(0, "NOT IMPLEMENTED"); return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP DocumentViewerImpl::GetPasteable(PRBool *aPasteable) { *aPasteable = PR_FALSE; return NS_OK; } #ifdef XP_MAC #pragma mark - #endif static NS_DEFINE_IID(kDeviceContextSpecFactoryCID, NS_DEVICE_CONTEXT_SPEC_FACTORY_CID); /* ======================================================================================== * nsIContentViewerFile * ======================================================================================== */ /** --------------------------------------------------- * See documentation above in the nsIContentViewerfile class definition * @update 01/24/00 dwc */ NS_IMETHODIMP DocumentViewerImpl::Print(PRBool aSilent, FILE * aDebugFile, nsIPrintSettings* aPrintSettings) { #ifdef NS_PRINTING nsCOMPtr printSettings; #ifdef NS_DEBUG nsresult rv = NS_ERROR_FAILURE; mDebugFile = aDebugFile; // if they don't pass in a PrintSettings, then make one // it will have all the default values printSettings = aPrintSettings; nsCOMPtr printOptions = do_GetService(sPrintOptionsContractID, &rv); if (NS_SUCCEEDED(rv)) { // if they don't pass in a PrintSettings, then make one if (printSettings == nsnull) { printOptions->CreatePrintSettings(getter_AddRefs(printSettings)); } NS_ASSERTION(printSettings, "You can't PrintPreview without a PrintSettings!"); } if (printSettings) printSettings->SetPrintSilent(aSilent); if (printSettings) printSettings->SetShowPrintProgress(PR_FALSE); #endif return Print(printSettings, nsnull); #else return NS_ERROR_FAILURE; #endif } /* [noscript] void printWithParent (in nsIDOMWindowInternal aParentWin, in nsIPrintSettings aThePrintSettings, in nsIWebProgressListener aWPListener); */ NS_IMETHODIMP DocumentViewerImpl::PrintWithParent(nsIDOMWindowInternal *aParentWin, nsIPrintSettings *aThePrintSettings, nsIWebProgressListener *aWPListener) { mDialogParentWin = aParentWin; return Print(aThePrintSettings, aWPListener); } // nsIContentViewerFile interface NS_IMETHODIMP DocumentViewerImpl::GetPrintable(PRBool *aPrintable) { NS_ENSURE_ARG_POINTER(aPrintable); *aPrintable = !GetIsPrinting(); return NS_OK; } #ifdef XP_MAC #pragma mark - #endif //***************************************************************************** // nsIMarkupDocumentViewer //***************************************************************************** NS_IMETHODIMP DocumentViewerImpl::ScrollToNode(nsIDOMNode* aNode) { NS_ENSURE_ARG(aNode); NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); nsCOMPtr presShell; NS_ENSURE_SUCCESS(GetPresShell(*(getter_AddRefs(presShell))), NS_ERROR_FAILURE); // Get the nsIContent interface, because that's what we need to // get the primary frame nsCOMPtr content(do_QueryInterface(aNode)); NS_ENSURE_TRUE(content, NS_ERROR_FAILURE); // Get the primary frame nsIFrame* frame; // Remember Frames aren't ref-counted. They are in their // own special little world. NS_ENSURE_SUCCESS(presShell->GetPrimaryFrameFor(content, &frame), NS_ERROR_FAILURE); // tell the pres shell to scroll to the frame NS_ENSURE_SUCCESS(presShell->ScrollFrameIntoView(frame, NS_PRESSHELL_SCROLL_TOP, NS_PRESSHELL_SCROLL_ANYWHERE), NS_ERROR_FAILURE); return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetAllowPlugins(PRBool* aAllowPlugins) { NS_ENSURE_ARG_POINTER(aAllowPlugins); *aAllowPlugins = mAllowPlugins; return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::SetAllowPlugins(PRBool aAllowPlugins) { mAllowPlugins = aAllowPlugins; return NS_OK; } nsresult DocumentViewerImpl::CallChildren(CallChildFunc aFunc, void* aClosure) { nsCOMPtr docShellNode(do_QueryInterface(mContainer)); if (docShellNode) { PRInt32 i; PRInt32 n; docShellNode->GetChildCount(&n); for (i=0; i < n; i++) { nsCOMPtr child; docShellNode->GetChildAt(i, getter_AddRefs(child)); nsCOMPtr childAsShell(do_QueryInterface(child)); NS_ASSERTION(childAsShell, "null child in docshell"); if (childAsShell) { nsCOMPtr childCV; childAsShell->GetContentViewer(getter_AddRefs(childCV)); if (childCV) { nsCOMPtr markupCV = do_QueryInterface(childCV); if (markupCV) { (*aFunc)(markupCV, aClosure); } } } } } return NS_OK; } struct TextZoomInfo { float mTextZoom; }; static void SetChildTextZoom(nsIMarkupDocumentViewer* aChild, void* aClosure) { struct TextZoomInfo* textZoomInfo = (struct TextZoomInfo*) aClosure; aChild->SetTextZoom(textZoomInfo->mTextZoom); } NS_IMETHODIMP DocumentViewerImpl::SetTextZoom(float aTextZoom) { if (mDeviceContext) { float oldTextZoom = 1.0; // just in case mDeviceContext doesn't implement // Don't reflow if there's no change in the textZoom. mDeviceContext->GetTextZoom(oldTextZoom); mDeviceContext->SetTextZoom(aTextZoom); if (oldTextZoom != aTextZoom && mPresContext) { mPresContext->ClearStyleDataAndReflow(); } } // now set the text zoom on all children of mContainer (even if our zoom // didn't change, our children's zoom may be different, though it would // be unusual). struct TextZoomInfo textZoomInfo = { aTextZoom }; return CallChildren(SetChildTextZoom, &textZoomInfo); } NS_IMETHODIMP DocumentViewerImpl::GetTextZoom(float* aTextZoom) { NS_ENSURE_ARG_POINTER(aTextZoom); if (mDeviceContext) { return mDeviceContext->GetTextZoom(*aTextZoom); } *aTextZoom = 1.0; return NS_OK; } // XXX: SEMANTIC CHANGE! // returns a copy of the string. Caller is responsible for freeing result // using Recycle(aDefaultCharacterSet) NS_IMETHODIMP DocumentViewerImpl::GetDefaultCharacterSet(PRUnichar** aDefaultCharacterSet) { NS_ENSURE_ARG_POINTER(aDefaultCharacterSet); NS_ENSURE_STATE(mContainer); if (mDefaultCharacterSet.IsEmpty()) { nsXPIDLString defCharset; nsCOMPtr webShell; webShell = do_QueryInterface(mContainer); if (webShell) { nsCOMPtr prefBranch(do_GetService(NS_PREFSERVICE_CONTRACTID)); if (prefBranch) { nsCOMPtr prefLocalString; prefBranch->GetComplexValue("intl.charset.default", NS_GET_IID(nsIPrefLocalizedString), getter_AddRefs(prefLocalString)); if (prefLocalString) { prefLocalString->GetData(getter_Copies(defCharset)); } } } if (!defCharset.IsEmpty()) mDefaultCharacterSet.Assign(defCharset.get()); else mDefaultCharacterSet.Assign(NS_LITERAL_STRING("ISO-8859-1")); } *aDefaultCharacterSet = ToNewUnicode(mDefaultCharacterSet); return NS_OK; } static void SetChildDefaultCharacterSet(nsIMarkupDocumentViewer* aChild, void* aClosure) { aChild->SetDefaultCharacterSet((PRUnichar*) aClosure); } NS_IMETHODIMP DocumentViewerImpl::SetDefaultCharacterSet(const PRUnichar* aDefaultCharacterSet) { mDefaultCharacterSet = aDefaultCharacterSet; // this does a copy of aDefaultCharacterSet // now set the default char set on all children of mContainer return CallChildren(SetChildDefaultCharacterSet, (void*) aDefaultCharacterSet); } // XXX: SEMANTIC CHANGE! // returns a copy of the string. Caller is responsible for freeing result // using Recycle(aForceCharacterSet) NS_IMETHODIMP DocumentViewerImpl::GetForceCharacterSet(PRUnichar** aForceCharacterSet) { NS_ENSURE_ARG_POINTER(aForceCharacterSet); nsAutoString emptyStr; if (mForceCharacterSet.Equals(emptyStr)) { *aForceCharacterSet = nsnull; } else { *aForceCharacterSet = ToNewUnicode(mForceCharacterSet); } return NS_OK; } static void SetChildForceCharacterSet(nsIMarkupDocumentViewer* aChild, void* aClosure) { aChild->SetForceCharacterSet((PRUnichar*) aClosure); } NS_IMETHODIMP DocumentViewerImpl::SetForceCharacterSet(const PRUnichar* aForceCharacterSet) { mForceCharacterSet = aForceCharacterSet; // now set the force char set on all children of mContainer return CallChildren(SetChildForceCharacterSet, (void*) aForceCharacterSet); } // XXX: SEMANTIC CHANGE! // returns a copy of the string. Caller is responsible for freeing result // using Recycle(aHintCharacterSet) NS_IMETHODIMP DocumentViewerImpl::GetHintCharacterSet(PRUnichar * *aHintCharacterSet) { NS_ENSURE_ARG_POINTER(aHintCharacterSet); if(kCharsetUninitialized == mHintCharsetSource) { *aHintCharacterSet = nsnull; } else { *aHintCharacterSet = ToNewUnicode(mHintCharset); // this can't possibly be right. we can't set a value just because somebody got a related value! //mHintCharsetSource = kCharsetUninitialized; } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetHintCharacterSetSource(PRInt32 *aHintCharacterSetSource) { NS_ENSURE_ARG_POINTER(aHintCharacterSetSource); *aHintCharacterSetSource = mHintCharsetSource; return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetPrevDocCharacterSet(PRUnichar * *aPrevDocCharacterSet) { NS_ENSURE_ARG_POINTER(aPrevDocCharacterSet); *aPrevDocCharacterSet = ToNewUnicode(mPrevDocCharacterSet); return NS_OK; } static void SetChildPrevDocCharacterSet(nsIMarkupDocumentViewer* aChild, void* aClosure) { aChild->SetPrevDocCharacterSet((PRUnichar*) aClosure); } NS_IMETHODIMP DocumentViewerImpl::SetPrevDocCharacterSet(const PRUnichar* aPrevDocCharacterSet) { mPrevDocCharacterSet = aPrevDocCharacterSet; return CallChildren(SetChildPrevDocCharacterSet, (void*) aPrevDocCharacterSet); } static void SetChildHintCharacterSetSource(nsIMarkupDocumentViewer* aChild, void* aClosure) { aChild->SetHintCharacterSetSource(NS_PTR_TO_INT32(aClosure)); } NS_IMETHODIMP DocumentViewerImpl::SetHintCharacterSetSource(PRInt32 aHintCharacterSetSource) { mHintCharsetSource = aHintCharacterSetSource; // now set the hint char set source on all children of mContainer return CallChildren(SetChildHintCharacterSetSource, (void*) aHintCharacterSetSource); } static void SetChildHintCharacterSet(nsIMarkupDocumentViewer* aChild, void* aClosure) { aChild->SetHintCharacterSet((PRUnichar*) aClosure); } NS_IMETHODIMP DocumentViewerImpl::SetHintCharacterSet(const PRUnichar* aHintCharacterSet) { mHintCharset = aHintCharacterSet; // now set the hint char set on all children of mContainer return CallChildren(SetChildHintCharacterSet, (void*) aHintCharacterSet); } static void SetChildBidiOptions(nsIMarkupDocumentViewer* aChild, void* aClosure) { aChild->SetBidiOptions(NS_PTR_TO_INT32(aClosure)); } NS_IMETHODIMP DocumentViewerImpl::SetBidiTextDirection(PRUint8 aTextDirection) { PRUint32 bidiOptions; GetBidiOptions(&bidiOptions); SET_BIDI_OPTION_DIRECTION(bidiOptions, aTextDirection); SetBidiOptions(bidiOptions); return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetBidiTextDirection(PRUint8* aTextDirection) { PRUint32 bidiOptions; if (aTextDirection) { GetBidiOptions(&bidiOptions); *aTextDirection = GET_BIDI_OPTION_DIRECTION(bidiOptions); } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::SetBidiTextType(PRUint8 aTextType) { PRUint32 bidiOptions; GetBidiOptions(&bidiOptions); SET_BIDI_OPTION_TEXTTYPE(bidiOptions, aTextType); SetBidiOptions(bidiOptions); return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetBidiTextType(PRUint8* aTextType) { PRUint32 bidiOptions; if (aTextType) { GetBidiOptions(&bidiOptions); *aTextType = GET_BIDI_OPTION_TEXTTYPE(bidiOptions); } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::SetBidiControlsTextMode(PRUint8 aControlsTextMode) { PRUint32 bidiOptions; GetBidiOptions(&bidiOptions); SET_BIDI_OPTION_CONTROLSTEXTMODE(bidiOptions, aControlsTextMode); SetBidiOptions(bidiOptions); return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetBidiControlsTextMode(PRUint8* aControlsTextMode) { PRUint32 bidiOptions; if (aControlsTextMode) { GetBidiOptions(&bidiOptions); *aControlsTextMode = GET_BIDI_OPTION_CONTROLSTEXTMODE(bidiOptions); } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::SetBidiClipboardTextMode(PRUint8 aClipboardTextMode) { PRUint32 bidiOptions; GetBidiOptions(&bidiOptions); SET_BIDI_OPTION_CLIPBOARDTEXTMODE(bidiOptions, aClipboardTextMode); SetBidiOptions(bidiOptions); return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetBidiClipboardTextMode(PRUint8* aClipboardTextMode) { PRUint32 bidiOptions; if (aClipboardTextMode) { GetBidiOptions(&bidiOptions); *aClipboardTextMode = GET_BIDI_OPTION_CLIPBOARDTEXTMODE(bidiOptions); } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::SetBidiNumeral(PRUint8 aNumeral) { PRUint32 bidiOptions; GetBidiOptions(&bidiOptions); SET_BIDI_OPTION_NUMERAL(bidiOptions, aNumeral); SetBidiOptions(bidiOptions); return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetBidiNumeral(PRUint8* aNumeral) { PRUint32 bidiOptions; if (aNumeral) { GetBidiOptions(&bidiOptions); *aNumeral = GET_BIDI_OPTION_NUMERAL(bidiOptions); } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::SetBidiSupport(PRUint8 aSupport) { PRUint32 bidiOptions; GetBidiOptions(&bidiOptions); SET_BIDI_OPTION_SUPPORT(bidiOptions, aSupport); SetBidiOptions(bidiOptions); return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetBidiSupport(PRUint8* aSupport) { PRUint32 bidiOptions; if (aSupport) { GetBidiOptions(&bidiOptions); *aSupport = GET_BIDI_OPTION_SUPPORT(bidiOptions); } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::SetBidiCharacterSet(PRUint8 aCharacterSet) { PRUint32 bidiOptions; GetBidiOptions(&bidiOptions); SET_BIDI_OPTION_CHARACTERSET(bidiOptions, aCharacterSet); SetBidiOptions(bidiOptions); return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetBidiCharacterSet(PRUint8* aCharacterSet) { PRUint32 bidiOptions; if (aCharacterSet) { GetBidiOptions(&bidiOptions); *aCharacterSet = GET_BIDI_OPTION_CHARACTERSET(bidiOptions); } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::SetBidiOptions(PRUint32 aBidiOptions) { if (mPresContext) { #if 1 // forcing reflow will cause bug 80352. Temp turn off force reflow and // wait for simon@softel.co.il to find the real solution mPresContext->SetBidi(aBidiOptions, PR_FALSE); #else mPresContext->SetBidi(aBidiOptions, PR_TRUE); // force reflow #endif } // now set bidi on all children of mContainer CallChildren(SetChildBidiOptions, (void*) aBidiOptions); return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetBidiOptions(PRUint32* aBidiOptions) { if (aBidiOptions) { if (mPresContext) { mPresContext->GetBidi(aBidiOptions); } else *aBidiOptions = IBMBIDI_DEFAULT_BIDI_OPTIONS; } return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::SizeToContent() { NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); nsCOMPtr docShellAsItem(do_QueryInterface(mContainer)); NS_ENSURE_TRUE(docShellAsItem, NS_ERROR_FAILURE); nsCOMPtr docShellParent; docShellAsItem->GetSameTypeParent(getter_AddRefs(docShellParent)); // It's only valid to access this from a top frame. Doesn't work from // sub-frames. NS_ENSURE_TRUE(!docShellParent, NS_ERROR_FAILURE); nsCOMPtr presShell; GetPresShell(*getter_AddRefs(presShell)); NS_ENSURE_TRUE(presShell, NS_ERROR_FAILURE); NS_ENSURE_SUCCESS(presShell->ResizeReflow(NS_UNCONSTRAINEDSIZE, NS_UNCONSTRAINEDSIZE), NS_ERROR_FAILURE); nsCOMPtr presContext; GetPresContext(*getter_AddRefs(presContext)); NS_ENSURE_TRUE(presContext, NS_ERROR_FAILURE); nsRect shellArea; PRInt32 width, height; float pixelScale; // so how big is it? presContext->GetVisibleArea(shellArea); presContext->GetTwipsToPixels(&pixelScale); width = PRInt32((float)shellArea.width*pixelScale); height = PRInt32((float)shellArea.height*pixelScale); nsCOMPtr treeOwner; docShellAsItem->GetTreeOwner(getter_AddRefs(treeOwner)); NS_ENSURE_TRUE(treeOwner, NS_ERROR_FAILURE); /* presContext's size was calculated in twips and has already been rounded to the equivalent pixels (so the width/height calculation we just performed was probably exact, though it was based on values already rounded during ResizeReflow). In a surprising number of instances, this rounding makes a window which for want of one extra pixel's width ends up wrapping the longest line of text during actual window layout. This makes the window too short, generally clipping the OK/Cancel buttons. Here we add one pixel to the calculated width, to circumvent this problem. */ NS_ENSURE_SUCCESS(treeOwner->SizeShellTo(docShellAsItem, width+1, height), NS_ERROR_FAILURE); return NS_OK; } #ifdef XP_MAC #pragma mark - #endif NS_IMPL_ISUPPORTS1(nsDocViewerSelectionListener, nsISelectionListener); nsresult nsDocViewerSelectionListener::Init(DocumentViewerImpl *aDocViewer) { mDocViewer = aDocViewer; return NS_OK; } /* * GetPopupNode, GetPopupLinkNode and GetPopupImageNode are helpers * for the cmd_copyLink / cmd_copyImageLocation / cmd_copyImageContents family * of commands. The focus controller stores the popup node, these retrieve * them and munge appropriately. Note that we have to store the popup node * rather than retrieving it from EventStateManager::GetFocusedContent because * not all content (images included) can receive focus. */ nsresult DocumentViewerImpl::GetPopupNode(nsIDOMNode** aNode) { NS_ENSURE_ARG_POINTER(aNode); nsresult rv; // get the document nsCOMPtr document; rv = GetDocument(*getter_AddRefs(document)); NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(document, NS_ERROR_FAILURE); // get the script global object nsCOMPtr global; rv = document->GetScriptGlobalObject(getter_AddRefs(global)); NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(global, NS_ERROR_FAILURE); // get the internal dom window nsCOMPtr internalWin(do_QueryInterface(global, &rv)); NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(internalWin, NS_ERROR_FAILURE); // get the private dom window nsCOMPtr privateWin(do_QueryInterface(internalWin, &rv)); NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(privateWin, NS_ERROR_FAILURE); // get the focus controller nsCOMPtr focusController; rv = privateWin->GetRootFocusController(getter_AddRefs(focusController)); NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(focusController, NS_ERROR_FAILURE); // get the popup node rv = focusController->GetPopupNode(aNode); // addref happens here NS_ENSURE_SUCCESS(rv, rv); return rv; } // GetPopupLinkNode: return popup link node or fail nsresult DocumentViewerImpl::GetPopupLinkNode(nsIDOMNode** aNode) { NS_ENSURE_ARG_POINTER(aNode); // you get null unless i say so *aNode = nsnull; // find popup node nsCOMPtr node; nsresult rv = GetPopupNode(getter_AddRefs(node)); NS_ENSURE_SUCCESS(rv, rv); // find out if we have a link in our ancestry while (node) { // are we an anchor? nsCOMPtr anchor(do_QueryInterface(node)); nsCOMPtr area; nsCOMPtr link; nsAutoString xlinkType; if (!anchor) { // area? area = do_QueryInterface(node); if (!area) { // link? link = do_QueryInterface(node); if (!link) { // XLink? nsCOMPtr element(do_QueryInterface(node)); if (element) { element->GetAttributeNS(NS_LITERAL_STRING("http://www.w3.org/1999/xlink"),NS_LITERAL_STRING("type"),xlinkType); } } } } if (anchor || area || link || xlinkType.Equals(NS_LITERAL_STRING("simple"))) { *aNode = node; NS_IF_ADDREF(*aNode); // addref return NS_OK; } else { // if not, get our parent and keep trying... nsCOMPtr parentNode; node->GetParentNode(getter_AddRefs(parentNode)); node = parentNode; } } // if we have no node, fail return NS_ERROR_FAILURE; } // GetPopupLinkNode: return popup image node or fail nsresult DocumentViewerImpl::GetPopupImageNode(nsIDOMNode** aNode) { NS_ENSURE_ARG_POINTER(aNode); // you get null unless i say so *aNode = nsnull; // find popup node nsCOMPtr node; nsresult rv = GetPopupNode(getter_AddRefs(node)); NS_ENSURE_SUCCESS(rv, rv); // XXX find out if we're an image. this really ought to look for objects // XXX with type "image/...", but this is good enough for now. nsCOMPtr img(do_QueryInterface(node, &rv)); if (NS_FAILED(rv)) return rv; NS_ENSURE_TRUE(img, NS_ERROR_FAILURE); // if we made it here, we're an image. *aNode = node; NS_IF_ADDREF(*aNode); // addref return NS_OK; } /* * XXX dr * ------ * These two functions -- GetInLink and GetInImage -- are kind of annoying * in that they only get called from the controller (in * nsDOMWindowController::IsCommandEnabled). The actual construction of the * context menus in communicator (nsContextMenu.js) has its own, redundant * tests. No big deal, but good to keep in mind if we ever clean context * menus. */ NS_IMETHODIMP DocumentViewerImpl::GetInLink(PRBool* aInLink) { #ifdef DEBUG_dr printf("dr :: DocumentViewerImpl::GetInLink\n"); #endif NS_ENSURE_ARG_POINTER(aInLink); // we're not in a link unless i say so *aInLink = PR_FALSE; // get the popup link nsCOMPtr node; nsresult rv = GetPopupLinkNode(getter_AddRefs(node)); if (NS_FAILED(rv)) return rv; NS_ENSURE_TRUE(node, NS_ERROR_FAILURE); // if we made it here, we're in a link *aInLink = PR_TRUE; return NS_OK; } NS_IMETHODIMP DocumentViewerImpl::GetInImage(PRBool* aInImage) { #ifdef DEBUG_dr printf("dr :: DocumentViewerImpl::GetInImage\n"); #endif NS_ENSURE_ARG_POINTER(aInImage); // we're not in an image unless i say so *aInImage = PR_FALSE; // get the popup image nsCOMPtr node; nsresult rv = GetPopupImageNode(getter_AddRefs(node)); if (NS_FAILED(rv)) return rv; NS_ENSURE_TRUE(node, NS_ERROR_FAILURE); // if we made it here, we're in an image *aInImage = PR_TRUE; return NS_OK; } NS_IMETHODIMP nsDocViewerSelectionListener::NotifySelectionChanged(nsIDOMDocument *, nsISelection *, short) { NS_ASSERTION(mDocViewer, "Should have doc viewer!"); // get the selection state nsCOMPtr selection; nsresult rv = mDocViewer->GetDocumentSelection(getter_AddRefs(selection)); if (NS_FAILED(rv)) return rv; PRBool selectionCollapsed; selection->GetIsCollapsed(&selectionCollapsed); // we only call UpdateCommands when the selection changes from collapsed // to non-collapsed or vice versa. We might need another update string // for simple selection changes, but that would be expenseive. if (!mGotSelectionState || mSelectionWasCollapsed != selectionCollapsed) { nsCOMPtr theDoc; mDocViewer->GetDocument(*getter_AddRefs(theDoc)); if (!theDoc) return NS_ERROR_FAILURE; nsCOMPtr scriptGlobalObject; theDoc->GetScriptGlobalObject(getter_AddRefs(scriptGlobalObject)); nsCOMPtr domWindow = do_QueryInterface(scriptGlobalObject); if (!domWindow) return NS_ERROR_FAILURE; domWindow->UpdateCommands(NS_LITERAL_STRING("select")); mGotSelectionState = PR_TRUE; mSelectionWasCollapsed = selectionCollapsed; } return NS_OK; } //nsDocViewerFocusListener NS_IMPL_ISUPPORTS1(nsDocViewerFocusListener, nsIDOMFocusListener); nsDocViewerFocusListener::nsDocViewerFocusListener() :mDocViewer(nsnull) { } nsDocViewerFocusListener::~nsDocViewerFocusListener(){} nsresult nsDocViewerFocusListener::HandleEvent(nsIDOMEvent* aEvent) { return NS_OK; } NS_IMETHODIMP nsDocViewerFocusListener::Focus(nsIDOMEvent* aEvent) { nsCOMPtr shell; if(!mDocViewer) return NS_ERROR_FAILURE; nsresult result = mDocViewer->GetPresShell(*getter_AddRefs(shell));//deref once cause it take a ptr ref if(NS_FAILED(result) || !shell) return result?result:NS_ERROR_FAILURE; nsCOMPtr selCon; selCon = do_QueryInterface(shell); PRInt16 selectionStatus; selCon->GetDisplaySelection( &selectionStatus); //if selection was nsISelectionController::SELECTION_OFF, do nothing //otherwise re-enable it. if(selectionStatus == nsISelectionController::SELECTION_DISABLED || selectionStatus == nsISelectionController::SELECTION_HIDDEN) { selCon->SetDisplaySelection(nsISelectionController::SELECTION_ON); selCon->RepaintSelection(nsISelectionController::SELECTION_NORMAL); } return result; } NS_IMETHODIMP nsDocViewerFocusListener::Blur(nsIDOMEvent* aEvent) { nsCOMPtr shell; if(!mDocViewer) return NS_ERROR_FAILURE; nsresult result = mDocViewer->GetPresShell(*getter_AddRefs(shell));//deref once cause it take a ptr ref if(NS_FAILED(result) || !shell) return result?result:NS_ERROR_FAILURE; nsCOMPtr selCon; selCon = do_QueryInterface(shell); PRInt16 selectionStatus; selCon->GetDisplaySelection(&selectionStatus); //if selection was nsISelectionController::SELECTION_OFF, do nothing //otherwise re-enable it. if(selectionStatus == nsISelectionController::SELECTION_ON) { selCon->SetDisplaySelection(nsISelectionController::SELECTION_DISABLED); selCon->RepaintSelection(nsISelectionController::SELECTION_NORMAL); } return result; } nsresult nsDocViewerFocusListener::Init(DocumentViewerImpl *aDocViewer) { mDocViewer = aDocViewer; return NS_OK; } /** --------------------------------------------------- * From nsIWebBrowserPrint */ NS_IMETHODIMP DocumentViewerImpl::Print(nsIPrintSettings* aPrintSettings, nsIWebProgressListener* aWebProgressListener) { #ifdef NS_PRINTING INIT_RUNTIME_ERROR_CHECKING(); nsCOMPtr docShell(do_QueryInterface(mContainer)); NS_ASSERTION(docShell, "This has to be a docshell"); // Check to see if this document is still busy // If it is busy and we aren't already "queued" up to print then // Indicate there is a print pending and cache the args for later PRUint32 busyFlags = nsIDocShell::BUSY_FLAGS_NONE; if ((NS_FAILED(docShell->GetBusyFlags(&busyFlags)) || (busyFlags != nsIDocShell::BUSY_FLAGS_NONE && busyFlags & nsIDocShell::BUSY_FLAGS_PAGE_LOADING)) && !mPrintDocIsFullyLoaded) { if (!mPrintIsPending) { mCachedPrintSettings = aPrintSettings; mCachedPrintWebProgressListner = aWebProgressListener; mPrintIsPending = PR_TRUE; } return NS_OK; } nsCOMPtr presShell; docShell->GetPresShell(getter_AddRefs(presShell)); if (!presShell) { // A frame that's not displayed can't be printed! PR_PL(("Printing Stopped - PreShell was NULL!")); return NS_OK; } nsresult rv = NS_ERROR_FAILURE; // if we are printing another URL, then exit // the reason we check here is because this method can be called while // another is still in here (the printing dialog is a good example). // the only time we can print more than one job at a time is the regression tests if (GetIsPrinting()) { // Let the user know we are not ready to print. rv = NS_ERROR_NOT_AVAILABLE; nsPrintEngine::ShowPrintErrorDialog(rv); return rv; } if (!mPrintEngine) { mPrintEngine = new nsPrintEngine(); NS_ENSURE_TRUE(mPrintEngine, NS_ERROR_OUT_OF_MEMORY); NS_ADDREF(mPrintEngine); #ifdef NS_DEBUG mPrintEngine->Initialize(this, this, mContainer, mDocument, mDeviceContext, mPresContext, mWindow, mParentWidget, mDebugFile); #else mPrintEngine->Initialize(this, this, mContainer, mDocument, mDeviceContext, mPresContext, mWindow, mParentWidget, nsnull); #endif } rv = mPrintEngine->Print(aPrintSettings, aWebProgressListener); if (NS_FAILED(rv)) { OnDonePrinting(); } return rv; #else return NS_ERROR_FAILURE; #endif } /** --------------------------------------------------- * See documentation above in the nsIContentViewerfile class definition * @update 11/01/01 rods * * For a full and detailed understanding of the issues with * PrintPreview: See the design spec that is attached to Bug 107562 */ NS_IMETHODIMP DocumentViewerImpl::PrintPreview(nsIPrintSettings* aPrintSettings, nsIDOMWindow *aChildDOMWin, nsIWebProgressListener* aWebProgressListener) { #if defined(NS_PRINTING) && defined(NS_PRINT_PREVIEW) nsresult rv = NS_OK; if (GetIsPrinting()) { nsPrintEngine::CloseProgressDialog(aWebProgressListener); return NS_ERROR_FAILURE; } if (!mPrintEngine) { mPrintEngine = new nsPrintEngine(); NS_ENSURE_TRUE(mPrintEngine, NS_ERROR_OUT_OF_MEMORY); NS_ADDREF(mPrintEngine); #ifdef NS_DEBUG mPrintEngine->Initialize(this, this, mContainer, mDocument, mDeviceContext, mPresContext, mWindow, mParentWidget, mDebugFile); #else mPrintEngine->Initialize(this, this, mContainer, mDocument, mDeviceContext, mPresContext, mWindow, mParentWidget, nsnull); #endif } rv = mPrintEngine->PrintPreview(aPrintSettings, aChildDOMWin, aWebProgressListener); if (NS_FAILED(rv)) { OnDonePrinting(); } return rv; #else return NS_ERROR_FAILURE; #endif } //---------------------------------------------------------------------- NS_IMETHODIMP DocumentViewerImpl::PrintPreviewNavigate(PRInt16 aType, PRInt32 aPageNum) { #if defined(NS_PRINTING) && defined(NS_PRINT_PREVIEW) if (GetIsPrinting()) return NS_ERROR_FAILURE; if (!mPrintEngine) return NS_ERROR_FAILURE; nsIScrollableView* scrollableView; mViewManager->GetRootScrollableView(&scrollableView); if (scrollableView == nsnull) return NS_OK; // Check to see if we can short circut scrolling to the top if (aType == nsIWebBrowserPrint::PRINTPREVIEW_HOME || (aType == nsIWebBrowserPrint::PRINTPREVIEW_GOTO_PAGENUM && aPageNum == 1)) { scrollableView->ScrollTo(0, 0, PR_TRUE); return NS_OK; } // Finds the SimplePageSequencer frame // in PP mPrtPreview->mPrintObject->mSeqFrame is null nsIFrame* seqFrame = nsnull; PRInt32 pageCount = 0; if (NS_FAILED(mPrintEngine->GetSeqFrameAndCountPages(seqFrame, pageCount))) { return NS_ERROR_FAILURE; } // Figure where we are currently scrolled to const nsIView * clippedView; scrollableView->GetClipView(&clippedView); nscoord x; nscoord y; scrollableView->GetScrollPosition(x, y); PRInt32 pageNum = 1; nsIFrame * fndPageFrame = nsnull; nsIFrame * currentPage = nsnull; // If it is "End" then just do a "goto" to the last page if (aType == nsIWebBrowserPrint::PRINTPREVIEW_END) { aType = nsIWebBrowserPrint::PRINTPREVIEW_GOTO_PAGENUM; aPageNum = pageCount; } // Now, locate the current page we are on and // and the page of the page number nscoord gap = 0; nsIFrame * pageFrame; seqFrame->FirstChild(mPresContext, nsnull, &pageFrame); while (pageFrame != nsnull) { nsRect pageRect; pageFrame->GetRect(pageRect); if (pageNum == 1) { gap = pageRect.y; } pageRect.y -= gap; if (pageRect.Contains(pageRect.x, y)) { currentPage = pageFrame; } if (pageNum == aPageNum) { fndPageFrame = pageFrame; break; } pageNum++; pageFrame->GetNextSibling(&pageFrame); } if (aType == nsIWebBrowserPrint::PRINTPREVIEW_PREV_PAGE) { if (currentPage) { currentPage->GetPrevInFlow(&fndPageFrame); if (!fndPageFrame) { return NS_OK; } } else { return NS_OK; } } else if (aType == nsIWebBrowserPrint::PRINTPREVIEW_NEXT_PAGE) { if (currentPage) { currentPage->GetNextInFlow(&fndPageFrame); if (!fndPageFrame) { return NS_OK; } } else { return NS_OK; } } else { // If we get here we are doing "GoTo" if (aPageNum < 0 || aPageNum > pageCount) { return NS_OK; } } if (fndPageFrame && scrollableView) { // get the child rect nsRect fRect; fndPageFrame->GetRect(fRect); // find offset from view nsPoint pnt; nsIView * view; fndPageFrame->GetOffsetFromView(mPresContext, pnt, &view); nscoord deadSpaceGap = 0; nsIPageSequenceFrame * sqf; if (NS_SUCCEEDED(CallQueryInterface(seqFrame, &sqf))) { sqf->GetDeadSpaceValue(&deadSpaceGap); } // scroll so that top of page (plus the gray area) is at the top of the scroll area scrollableView->ScrollTo(0, fRect.y-deadSpaceGap, PR_TRUE); } return NS_OK; #else return NS_ERROR_FAILURE; #endif // NS_PRINT_PREVIEW } /* readonly attribute nsIPrintSettings globalPrintSettings; */ NS_IMETHODIMP DocumentViewerImpl::GetGlobalPrintSettings(nsIPrintSettings * *aGlobalPrintSettings) { #ifdef NS_PRINTING NS_ENSURE_ARG_POINTER(aGlobalPrintSettings); nsPrintEngine printEngine; return printEngine.GetGlobalPrintSettings(aGlobalPrintSettings); #else return NS_ERROR_FAILURE; #endif } /* readonly attribute boolean doingPrint; */ NS_IMETHODIMP DocumentViewerImpl::GetDoingPrint(PRBool *aDoingPrint) { #ifdef NS_PRINTING NS_ENSURE_ARG_POINTER(aDoingPrint); *aDoingPrint = PR_FALSE; if (mPrintEngine) { return mPrintEngine->GetDoingPrintPreview(aDoingPrint); } return NS_OK; #else return NS_ERROR_FAILURE; #endif } /* readonly attribute boolean doingPrintPreview; */ NS_IMETHODIMP DocumentViewerImpl::GetDoingPrintPreview(PRBool *aDoingPrintPreview) { #ifdef NS_PRINTING NS_ENSURE_ARG_POINTER(aDoingPrintPreview); *aDoingPrintPreview = PR_FALSE; if (mPrintEngine) { return mPrintEngine->GetDoingPrintPreview(aDoingPrintPreview); } return NS_OK; #else return NS_ERROR_FAILURE; #endif } /* readonly attribute nsIPrintSettings currentPrintSettings; */ NS_IMETHODIMP DocumentViewerImpl::GetCurrentPrintSettings(nsIPrintSettings * *aCurrentPrintSettings) { #ifdef NS_PRINTING NS_ENSURE_ARG_POINTER(aCurrentPrintSettings); *aCurrentPrintSettings = nsnull; NS_ENSURE_TRUE(mPrintEngine, NS_ERROR_FAILURE); return mPrintEngine->GetCurrentPrintSettings(aCurrentPrintSettings); #else return NS_ERROR_FAILURE; #endif } /* readonly attribute nsIDOMWindow currentChildDOMWindow; */ NS_IMETHODIMP DocumentViewerImpl::GetCurrentChildDOMWindow(nsIDOMWindow * *aCurrentChildDOMWindow) { NS_ENSURE_ARG_POINTER(aCurrentChildDOMWindow); *aCurrentChildDOMWindow = nsnull; return NS_ERROR_NOT_IMPLEMENTED; } /* void cancel (); */ NS_IMETHODIMP DocumentViewerImpl::Cancel() { #ifdef NS_PRINTING NS_ENSURE_TRUE(mPrintEngine, NS_ERROR_FAILURE); return mPrintEngine->Cancelled(); #else return NS_ERROR_FAILURE; #endif } /* void exitPrintPreview (); */ NS_IMETHODIMP DocumentViewerImpl::ExitPrintPreview() { #ifdef NS_PRINTING if (GetIsPrinting()) return NS_ERROR_FAILURE; NS_ENSURE_TRUE(mPrintEngine, NS_ERROR_FAILURE); if (GetIsPrintPreview()) { ReturnToGalleyPresentation(); } return NS_OK; #else return NS_ERROR_FAILURE; #endif } //---------------------------------------------------------------------------------- // Enumerate all the documents for their titles NS_IMETHODIMP DocumentViewerImpl::EnumerateDocumentNames(PRUint32* aCount, PRUnichar*** aResult) { #ifdef NS_PRINTING NS_ENSURE_ARG(aCount); NS_ENSURE_ARG_POINTER(aResult); NS_ENSURE_TRUE(mPrintEngine, NS_ERROR_FAILURE); return mPrintEngine->EnumerateDocumentNames(aCount, aResult); #else return NS_ERROR_FAILURE; #endif } /* readonly attribute boolean isFramesetFrameSelected; */ NS_IMETHODIMP DocumentViewerImpl::GetIsFramesetFrameSelected(PRBool *aIsFramesetFrameSelected) { #ifdef NS_PRINTING *aIsFramesetFrameSelected = PR_FALSE; NS_ENSURE_TRUE(mPrintEngine, NS_ERROR_FAILURE); return mPrintEngine->GetIsFramesetFrameSelected(aIsFramesetFrameSelected); #else return NS_ERROR_FAILURE; #endif } /* readonly attribute long printPreviewNumPages; */ NS_IMETHODIMP DocumentViewerImpl::GetPrintPreviewNumPages(PRInt32 *aPrintPreviewNumPages) { #ifdef NS_PRINTING NS_ENSURE_ARG_POINTER(aPrintPreviewNumPages); NS_ENSURE_TRUE(mPrintEngine, NS_ERROR_FAILURE); return mPrintEngine->GetPrintPreviewNumPages(aPrintPreviewNumPages); #else return NS_ERROR_FAILURE; #endif } /* readonly attribute boolean isFramesetDocument; */ NS_IMETHODIMP DocumentViewerImpl::GetIsFramesetDocument(PRBool *aIsFramesetDocument) { #ifdef NS_PRINTING *aIsFramesetDocument = PR_FALSE; NS_ENSURE_TRUE(mPrintEngine, NS_ERROR_FAILURE); return mPrintEngine->GetIsFramesetDocument(aIsFramesetDocument); #else return NS_ERROR_FAILURE; #endif } /* readonly attribute boolean isIFrameSelected; */ NS_IMETHODIMP DocumentViewerImpl::GetIsIFrameSelected(PRBool *aIsIFrameSelected) { #ifdef NS_PRINTING *aIsIFrameSelected = PR_FALSE; NS_ENSURE_TRUE(mPrintEngine, NS_ERROR_FAILURE); return mPrintEngine->GetIsIFrameSelected(aIsIFrameSelected); #else return NS_ERROR_FAILURE; #endif } /* readonly attribute boolean isRangeSelection; */ NS_IMETHODIMP DocumentViewerImpl::GetIsRangeSelection(PRBool *aIsRangeSelection) { #ifdef NS_PRINTING *aIsRangeSelection = PR_FALSE; NS_ENSURE_TRUE(mPrintEngine, NS_ERROR_FAILURE); return mPrintEngine->GetIsRangeSelection(aIsRangeSelection); #else return NS_ERROR_FAILURE; #endif } #ifdef NS_PRINTING //---------------------------------------------------------------------------------- // Printing/Print Preview Helpers //---------------------------------------------------------------------------------- //---------------------------------------------------------------------------------- // Walks the document tree and tells each DocShell whether Printing/PP is happening void DocumentViewerImpl::SetIsPrintingInDocShellTree(nsIDocShellTreeNode* aParentNode, PRBool aIsPrintingOrPP, PRBool aStartAtTop) { NS_ASSERTION(aParentNode, "Parent can't be NULL!"); nsCOMPtr parentItem(do_QueryInterface(aParentNode)); // find top of "same parent" tree if (aStartAtTop) { while (parentItem) { nsCOMPtr parent; parentItem->GetSameTypeParent(getter_AddRefs(parent)); if (!parent) { break; } parentItem = do_QueryInterface(parent); } } NS_ASSERTION(parentItem, "parentItem can't be null"); // Check to see if the DocShell's ContentViewer is printing/PP nsCOMPtr viewerContainer(do_QueryInterface(parentItem)); if (viewerContainer) { viewerContainer->SetIsPrinting(aIsPrintingOrPP); } // Traverse children to see if any of them are printing. PRInt32 n; aParentNode->GetChildCount(&n); for (PRInt32 i=0; i < n; i++) { nsCOMPtr child; aParentNode->GetChildAt(i, getter_AddRefs(child)); nsCOMPtr childAsNode(do_QueryInterface(child)); NS_ASSERTION(childAsNode, "child isn't nsIDocShellTreeNode"); if (childAsNode) { SetIsPrintingInDocShellTree(childAsNode, aIsPrintingOrPP, PR_FALSE); } } } #endif // NS_PRINTING //------------------------------------------------------------ PRBool DocumentViewerImpl::GetIsPrinting() { #ifdef NS_PRINTING if (mPrintEngine) { return mPrintEngine->GetIsPrinting(); } #endif return PR_FALSE; } //------------------------------------------------------------ // Notification from the PrintEngine of the current Printing status void DocumentViewerImpl::SetIsPrinting(PRBool aIsPrinting) { #ifdef NS_PRINTING // Set all the docShells in the docshell tree to be printing. // that way if anyone of them tries to "navigate" it can't if (mContainer) { nsCOMPtr docShellTreeNode(do_QueryInterface(mContainer)); NS_ASSERTION(docShellTreeNode, "mContainer has to be a nsIDocShellTreeNode"); SetIsPrintingInDocShellTree(docShellTreeNode, aIsPrinting, PR_TRUE); } #endif } //------------------------------------------------------------ // The PrintEngine holds the current value // this called from inside the DocViewer PRBool DocumentViewerImpl::GetIsPrintPreview() { #ifdef NS_PRINTING if (mPrintEngine) { return mPrintEngine->GetIsPrintPreview(); } #endif return PR_FALSE; } //------------------------------------------------------------ // Notification from the PrintEngine of the current PP status void DocumentViewerImpl::SetIsPrintPreview(PRBool aIsPrintPreview) { #ifdef NS_PRINTING // Set all the docShells in the docshell tree to be printing. // that way if anyone of them tries to "navigate" it can't if (mContainer) { nsCOMPtr docShellTreeNode(do_QueryInterface(mContainer)); NS_ASSERTION(docShellTreeNode, "mContainer has to be a nsIDocShellTreeNode"); SetIsPrintingInDocShellTree(docShellTreeNode, aIsPrintPreview, PR_TRUE); } #endif } //------------------------------------------------------------ // The PrintEngine holds the current value // this called from inside the DocViewer PRBool DocumentViewerImpl::GetIsCreatingPrintPreview() { #if defined(NS_PRINTING) && defined(NS_PRINT_PREVIEW) if (mPrintEngine) { return mPrintEngine->GetIsCreatingPrintPreview(); } #endif return PR_FALSE; } //---------------------------------------------------------------------------------- // nsIDocumentViewerPrint IFace //---------------------------------------------------------------------------------- //------------------------------------------------------------ void DocumentViewerImpl::IncrementDestroyRefCount() { ++mDestroyRefCount; } //------------------------------------------------------------ void DocumentViewerImpl::ReturnToGalleyPresentation() { #if defined(NS_PRINTING) && defined(NS_PRINT_PREVIEW) if (!GetIsPrintPreview()) { NS_ASSERTION(0, "Wow, we should never get here!"); return; } // Get the current size of what is being viewed nsRect area; mPresContext->GetVisibleArea(area); nsRect bounds; mWindow->GetBounds(bounds); // In case we have focus focus the parent DocShell // which in this case should always be chrome nsCOMPtr dstParentItem; nsCOMPtr dstItem(do_QueryInterface(mContainer)); if (dstItem) { dstItem->GetParent(getter_AddRefs(dstParentItem)); nsCOMPtr docShell(do_QueryInterface(dstParentItem)); if (docShell) { docShell->SetHasFocus(PR_TRUE); } } // Start to kill off the old Presentation // by cleaning up the PresShell if (mPresShell) { // Break circular reference (or something) mPresShell->EndObservingDocument(); nsCOMPtr selection; nsresult rv = GetDocumentSelection(getter_AddRefs(selection)); nsCOMPtr selPrivate(do_QueryInterface(selection)); if (NS_SUCCEEDED(rv) && selPrivate && mSelectionListener) selPrivate->RemoveSelectionListener(mSelectionListener); mPresShell->Destroy(); } // clear weak references before we go away if (mPresContext) { mPresContext->SetContainer(nsnull); mPresContext->SetLinkHandler(nsnull); } //------------------------------------------------ // NOTE: // Here is why the code below is a little confusing: // 1) Scripting needs to be turned back on before // the print engine is destroyed // 2) The PrintEngine must be destroyed BEFORE // calling InitInternal when caching documents (framesets) // BUT the PrintEngine must be destroyed AFTER // calling InitInternal when NOT caching documents (no framesets) //------------------------------------------------ // wasCached will be used below to indicate whether the // InitInternal should create all new objects or just // initialize the existing ones PRBool wasCached = PR_FALSE; if (mPrintEngine && mPrintEngine->HasCachedPres()) { mPrintEngine->GetCachedPresentation(mPresShell, mPresContext, mViewManager, mWindow); // Tell the "real" presshell to start observing the document // again. mPresShell->BeginObservingDocument(); mWindow->Show(PR_TRUE); wasCached = PR_TRUE; } else { // Destroy the old Presentation mPresShell = nsnull; mPresContext = nsnull; mViewManager = nsnull; mWindow = nsnull; } if (mPrintEngine) { // Very important! Turn On scripting mPrintEngine->TurnScriptingOn(PR_TRUE); if (wasCached) { mPrintEngine->Destroy(); NS_RELEASE(mPrintEngine); } } InitInternal(mParentWidget, mDeviceContext, bounds, !wasCached, PR_TRUE); if (mPrintEngine && !wasCached) { mPrintEngine->Destroy(); NS_RELEASE(mPrintEngine); } // this needs to be set here not earlier, // because it is needing when re-constructing the Galley Mode) SetIsPrintPreview(PR_FALSE); mViewManager->EnableRefresh(NS_VMREFRESH_NO_SYNC); Show(); #endif // NS_PRINTING && NS_PRINT_PREVIEW } //------------------------------------------------------------ void DocumentViewerImpl::InstallNewPresentation() { #if defined(NS_PRINTING) && defined(NS_PRINT_PREVIEW) // Get the current size of what is being viewed nsRect area; mPresContext->GetVisibleArea(area); nsRect bounds; mWindow->GetBounds(bounds); // In case we have focus focus the parent DocShell // which in this case should always be chrome nsCOMPtr dstParentItem; nsCOMPtr dstItem(do_QueryInterface(mContainer)); if (dstItem) { dstItem->GetParent(getter_AddRefs(dstParentItem)); nsCOMPtr docShell(do_QueryInterface(dstParentItem)); if (docShell) { docShell->SetHasFocus(PR_TRUE); } } // turn off selection painting nsCOMPtr selectionController = do_QueryInterface(mPresShell); if (selectionController) { selectionController->SetDisplaySelection(nsISelectionController::SELECTION_OFF); } // Start to kill off the old Presentation // by cleaning up the PresShell if (mPresShell) { // Break circular reference (or something) mPresShell->EndObservingDocument(); nsCOMPtr selection; nsresult rv = GetDocumentSelection(getter_AddRefs(selection)); nsCOMPtr selPrivate(do_QueryInterface(selection)); if (NS_SUCCEEDED(rv) && selPrivate && mSelectionListener) selPrivate->RemoveSelectionListener(mSelectionListener); // We need to destroy the PreShell if there is an existing PP // or we are not caching the original Presentation if (!mPrintEngine->IsCachingPres() || mPrintEngine->IsOldPrintPreviewPres()) { mPresShell->Destroy(); } } // clear weak references before we go away if (mPresContext) { mPresContext->SetContainer(nsnull); mPresContext->SetLinkHandler(nsnull); } // See if we are suppose to be caching the old Presentation // and then check to see if we already have. if (mPrintEngine->IsCachingPres() && !mPrintEngine->HasCachedPres()) { // Cach old presentation mPrintEngine->CachePresentation(mPresShell, mPresContext, mViewManager, mWindow); mWindow->Show(PR_FALSE); } else { // Destroy the old Presentation mPresShell = nsnull; mPresContext = nsnull; mViewManager = nsnull; mWindow = nsnull; } // XXX InstallPrintPreviewListener(); mPrintEngine->GetNewPresentation(mPresShell, mPresContext, mViewManager, mWindow); mPresShell->BeginObservingDocument(); nscoord width = bounds.width; nscoord height = bounds.height; float p2t; mPresContext->GetPixelsToTwips(&p2t); width = NSIntPixelsToTwips(width, p2t); height = NSIntPixelsToTwips(height, p2t); mViewManager->DisableRefresh(); mViewManager->SetWindowDimensions(width, height); mDeviceContext->SetUseAltDC(kUseAltDCFor_FONTMETRICS, PR_FALSE); mDeviceContext->SetUseAltDC(kUseAltDCFor_CREATERC_PAINT, PR_TRUE); mViewManager->EnableRefresh(NS_VMREFRESH_NO_SYNC); Show(); mPrintEngine->ShowDocList(PR_TRUE); #endif // NS_PRINTING && NS_PRINT_PREVIEW } //------------------------------------------------------------ // This called ONLY when printing has completed and the DV // is being notified that it should get rid of the PrintEngine. // // BUT, if we are in Print Preview then we want to ignore the // notification (we do not get rid of the PrintEngine) // // One small caveat: // This IS called from two places in this module for cleaning // up when an error occurred during the start up printing // and print preview // void DocumentViewerImpl::OnDonePrinting() { #if defined(NS_PRINTING) && defined(NS_PRINT_PREVIEW) if (mPrintEngine) { if (GetIsPrintPreview()) { mPrintEngine->DestroyPrintingData(); } else { mPrintEngine->Destroy(); NS_RELEASE(mPrintEngine); } // We are done printing, now cleanup if (mClosingWhilePrinting) { if (mDocument) { mDocument->SetScriptGlobalObject(nsnull); mDocument = nsnull; } mClosingWhilePrinting = PR_FALSE; NS_RELEASE_THIS(); } } #endif // NS_PRINTING && NS_PRINT_PREVIEW }