gecko-dev/content/html/document/src/nsImageDocument.cpp

464 lines
13 KiB
C++
Raw Normal View History

1998-07-27 17:50:58 +00:00
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
*
* The contents of this file are subject to the Netscape Public
* License Version 1.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/
1998-07-27 17:50:58 +00:00
*
* 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.
1998-07-27 17:50:58 +00:00
*
* The Original Code is Mozilla Communicator client code.
*
* The Initial Developer of the Original Code is Netscape Communications
* Corporation. Portions created by Netscape are
* Copyright (C) 1998 Netscape Communications Corporation. All
* Rights Reserved.
*
* Contributor(s):
1998-07-27 17:50:58 +00:00
*/
#include "nsCOMPtr.h"
#include "nsHTMLDocument.h"
1998-07-27 17:50:58 +00:00
#include "nsHTMLParts.h"
#include "nsHTMLAtoms.h"
#include "nsIHTMLContent.h"
#include "nsIImageGroup.h"
#include "nsIImageRequest.h"
#include "nsIStreamListener.h"
#include "nsIURL.h"
#include "nsHTMLValue.h"
#include "nsIHTMLStyleSheet.h"
#include "nsIHTMLCSSStyleSheet.h"
#include "nsIPresShell.h"
#include "nsIPresContext.h"
#include "nsIViewManager.h"
#include "nsIChannel.h"
#include "nsINameSpaceManager.h"
#include "nsINodeInfo.h"
// Needed for Localization
#include "nsXPIDLString.h"
#include "nsIStringBundle.h"
// Needed to fetch scrollbar prefs for image documents that are iframes/frameset frames
#include "nsIScrollable.h"
#include "nsWeakReference.h"
#define NSIMAGEDOCUMENT_PROPERTIES_URI "chrome://communicator/locale/layout/ImageDocument.properties"
static NS_DEFINE_CID(kStringBundleServiceCID, NS_STRINGBUNDLESERVICE_CID);
// done L10N
1998-07-27 17:50:58 +00:00
// XXX TODO:
// 1. hookup the original stream to the image library directly so that we
// don't have to load the image twice.
// 2. have a synthetic document url that we load that has patterns in it
// that we replace with the image url (so that we can customize the
// the presentation of the image): we could add script code to set the
// width/height to provide scale buttons, we could show the image
// attributes; etc.; should we have a seperate style sheet?
// 3. override the save-as methods so that we don't write out our synthetic
// html
class nsImageDocument : public nsHTMLDocument {
1998-07-27 17:50:58 +00:00
public:
nsImageDocument();
virtual ~nsImageDocument();
NS_IMETHOD StartDocumentLoad(const char* aCommand,
nsIChannel* aChannel,
nsILoadGroup* aLoadGroup,
nsISupports* aContainer,
nsIStreamListener **aDocListener,
PRBool aReset = PR_TRUE);
1998-07-27 17:50:58 +00:00
nsresult CreateSyntheticDocument();
nsresult StartImageLoad(nsIURI* aURL, nsIStreamListener*& aListener);
nsresult EndLayout(nsISupports *ctxt,
nsresult status,
const PRUnichar *errorMsg);
nsresult UpdateTitle( void );
1998-07-27 17:50:58 +00:00
void StartLayout();
nsIImageRequest* mImageRequest;
nscolor mBlack;
nsWeakPtr mContainer;
1998-07-27 17:50:58 +00:00
};
//----------------------------------------------------------------------
1998-07-27 17:50:58 +00:00
class ImageListener : public nsIStreamListener {
public:
ImageListener(nsImageDocument* aDoc);
virtual ~ImageListener();
1998-07-27 17:50:58 +00:00
NS_DECL_ISUPPORTS
1999-06-25 01:53:22 +00:00
// nsIStreamObserver methods:
NS_DECL_NSISTREAMOBSERVER
1999-06-25 01:53:22 +00:00
// nsIStreamListener methods:
NS_DECL_NSISTREAMLISTENER
1998-07-27 17:50:58 +00:00
nsImageDocument* mDocument;
nsIStreamListener* mNextStream;
1998-07-27 17:50:58 +00:00
};
ImageListener::ImageListener(nsImageDocument* aDoc)
{
NS_INIT_ISUPPORTS();
1998-07-27 17:50:58 +00:00
mDocument = aDoc;
NS_ADDREF(aDoc);
}
ImageListener::~ImageListener()
{
NS_RELEASE(mDocument);
NS_IF_RELEASE(mNextStream);
1998-07-27 17:50:58 +00:00
}
NS_IMPL_THREADSAFE_ISUPPORTS1(ImageListener, nsIStreamListener)
1998-07-27 17:50:58 +00:00
NS_IMETHODIMP
ImageListener::OnStartRequest(nsIRequest* request, nsISupports *ctxt)
1998-07-27 17:50:58 +00:00
{
nsresult rv;
1999-06-25 01:53:22 +00:00
nsIURI* uri;
nsCOMPtr<nsIChannel> channel = do_QueryInterface(request);
if (!channel) return NS_ERROR_NULL_POINTER;
rv = channel->GetURI(&uri);
if (NS_FAILED(rv)) return rv;
1999-06-25 01:53:22 +00:00
mDocument->StartImageLoad(uri, mNextStream);
NS_RELEASE(uri);
if (nsnull == mNextStream) {
return NS_ERROR_FAILURE;
}
return mNextStream->OnStartRequest(request, ctxt);
1998-07-27 17:50:58 +00:00
}
NS_IMETHODIMP
ImageListener::OnStopRequest(nsIRequest* request, nsISupports *ctxt,
nsresult status, const PRUnichar *errorMsg)
1998-07-27 17:50:58 +00:00
{
if(mDocument){
mDocument->EndLayout(ctxt, status, errorMsg);
}
if (nsnull == mNextStream) {
return NS_ERROR_FAILURE;
}
return mNextStream->OnStopRequest(request, ctxt, status, errorMsg);
1998-07-27 17:50:58 +00:00
}
NS_IMETHODIMP
ImageListener::OnDataAvailable(nsIRequest* request, nsISupports *ctxt,
nsIInputStream *inStr, PRUint32 sourceOffset, PRUint32 count)
1998-07-27 17:50:58 +00:00
{
if (nsnull == mNextStream) {
return NS_ERROR_FAILURE;
1998-07-27 17:50:58 +00:00
}
return mNextStream->OnDataAvailable(request, ctxt, inStr, sourceOffset, count);
1998-07-27 17:50:58 +00:00
}
//----------------------------------------------------------------------
NS_LAYOUT nsresult
NS_NewImageDocument(nsIDocument** aInstancePtrResult)
{
nsImageDocument* doc = new nsImageDocument();
1999-07-18 06:35:52 +00:00
if(doc)
return doc->QueryInterface(NS_GET_IID(nsIDocument), (void**) aInstancePtrResult);
1999-07-18 06:35:52 +00:00
return NS_ERROR_OUT_OF_MEMORY;
1998-07-27 17:50:58 +00:00
}
nsImageDocument::nsImageDocument()
{
mImageRequest = nsnull;
mBlack = NS_RGB(0, 0, 0);
1998-07-27 17:50:58 +00:00
}
nsImageDocument::~nsImageDocument()
{
NS_IF_RELEASE(mImageRequest);
1998-07-27 17:50:58 +00:00
}
NS_IMETHODIMP
nsImageDocument::StartDocumentLoad(const char* aCommand,
nsIChannel* aChannel,
nsILoadGroup* aLoadGroup,
nsISupports* aContainer,
nsIStreamListener **aDocListener,
PRBool aReset)
1998-07-27 17:50:58 +00:00
{
NS_ASSERTION(aDocListener, "null aDocListener");
NS_ENSURE_ARG_POINTER(aContainer);
mContainer = dont_AddRef(NS_GetWeakReference(aContainer));
nsresult rv = Init();
if (NS_FAILED(rv) && rv != NS_ERROR_ALREADY_INITIALIZED) {
return rv;
}
rv = nsDocument::StartDocumentLoad(aCommand, aChannel, aLoadGroup,
aContainer, aDocListener, aReset);
1998-12-11 02:47:25 +00:00
if (NS_FAILED(rv)) {
return rv;
}
// Create synthetic document
rv = CreateSyntheticDocument();
if (NS_OK != rv) {
return rv;
1998-07-27 17:50:58 +00:00
}
*aDocListener = new ImageListener(this);
if (!*aDocListener)
return NS_ERROR_OUT_OF_MEMORY;
NS_ADDREF(*aDocListener);
1998-07-27 17:50:58 +00:00
return NS_OK;
}
nsresult
nsImageDocument::StartImageLoad(nsIURI* aURL, nsIStreamListener*& aListener)
{
nsresult rv = NS_OK;
aListener = nsnull;
// Tell image group to load the stream now. This will get the image
// hooked up to the open stream and return the underlying listener
// so that we can pass it back upwards.
nsIPresShell* shell = GetShellAt(0);
if (nsnull != shell) {
nsCOMPtr<nsIPresContext> cx;
shell->GetPresContext(getter_AddRefs(cx));
if (cx) {
nsIImageGroup* group = nsnull;
cx->GetImageGroup(&group);
if (nsnull != group) {
1999-06-25 01:53:22 +00:00
char* spec;
(void)aURL->GetSpec(&spec);
nsIStreamListener* listener = nsnull;
rv = group->GetImageFromStream(spec, nsnull, nsnull,
0, 0, 0,
mImageRequest, listener);
//set flag to indicate view-image needs to use imgcache
group->SetImgLoadAttributes(nsImageLoadFlags_kSticky);
1999-06-25 01:53:22 +00:00
nsCRT::free(spec);
aListener = listener;
NS_RELEASE(group);
}
}
NS_RELEASE(shell);
}
// Finally, start the layout going
StartLayout();
return NS_OK;
}
1998-07-27 17:50:58 +00:00
nsresult
nsImageDocument::CreateSyntheticDocument()
{
// Synthesize an html document that refers to the image
nsresult rv;
nsCOMPtr<nsINodeInfo> nodeInfo;
rv = mNodeInfoManager->GetNodeInfo(nsHTMLAtoms::html, nsnull,
kNameSpaceID_None,
*getter_AddRefs(nodeInfo));
NS_ENSURE_SUCCESS(rv, rv);
1998-07-27 17:50:58 +00:00
nsIHTMLContent* root;
rv = NS_NewHTMLHtmlElement(&root, nodeInfo);
1998-07-27 17:50:58 +00:00
if (NS_OK != rv) {
return rv;
}
root->SetDocument(this, PR_FALSE, PR_TRUE);
SetRootContent(root);
1998-07-27 17:50:58 +00:00
rv = mNodeInfoManager->GetNodeInfo(nsHTMLAtoms::body, nsnull,
kNameSpaceID_None,
*getter_AddRefs(nodeInfo));
NS_ENSURE_SUCCESS(rv, rv);
1998-07-27 17:50:58 +00:00
nsIHTMLContent* body;
rv = NS_NewHTMLBodyElement(&body, nodeInfo);
1998-07-27 17:50:58 +00:00
if (NS_OK != rv) {
return rv;
}
body->SetDocument(this, PR_FALSE, PR_TRUE);
1998-07-27 17:50:58 +00:00
rv = mNodeInfoManager->GetNodeInfo(nsHTMLAtoms::p, nsnull, kNameSpaceID_None,
*getter_AddRefs(nodeInfo));
NS_ENSURE_SUCCESS(rv, rv);
1998-07-27 17:50:58 +00:00
nsIHTMLContent* center;
rv = NS_NewHTMLParagraphElement(&center, nodeInfo);
1998-07-27 17:50:58 +00:00
if (NS_OK != rv) {
return rv;
}
center->SetDocument(this, PR_FALSE, PR_TRUE);
1998-07-27 17:50:58 +00:00
rv = mNodeInfoManager->GetNodeInfo(nsHTMLAtoms::img, nsnull,
kNameSpaceID_None,
*getter_AddRefs(nodeInfo));
NS_ENSURE_SUCCESS(rv, rv);
1998-07-27 17:50:58 +00:00
nsIHTMLContent* image;
rv = NS_NewHTMLImageElement(&image, nodeInfo);
1998-07-27 17:50:58 +00:00
if (NS_OK != rv) {
return rv;
}
image->SetDocument(this, PR_FALSE, PR_TRUE);
1999-06-25 01:53:22 +00:00
char* src;
mDocumentURL->GetSpec(&src);
2000-04-16 11:19:26 +00:00
nsString src_string; src_string.AssignWithConversion(src);
nsHTMLValue val(src_string);
1999-05-26 22:53:12 +00:00
delete[] src;
1998-12-20 01:21:23 +00:00
image->SetHTMLAttribute(nsHTMLAtoms::src, val, PR_FALSE);
image->SetHTMLAttribute(nsHTMLAtoms::alt, val, PR_FALSE);
1998-07-27 17:50:58 +00:00
root->AppendChildTo(body, PR_FALSE);
center->AppendChildTo(image, PR_FALSE);
body->AppendChildTo(center, PR_FALSE);
1998-07-27 17:50:58 +00:00
NS_RELEASE(image);
NS_RELEASE(center);
NS_RELEASE(body);
NS_RELEASE(root);
return NS_OK;
}
void
nsImageDocument::StartLayout()
{
// Reset scrolling to default settings for this shell.
// This must happen before the initial reflow, when we create the root frame
nsCOMPtr<nsIScrollable> scrollableContainer(do_QueryReferent(mContainer));
if (scrollableContainer) {
scrollableContainer->ResetScrollbarPreferences();
}
1998-07-27 17:50:58 +00:00
PRInt32 i, ns = GetNumberOfShells();
for (i = 0; i < ns; i++) {
nsIPresShell* shell = GetShellAt(i);
if (nsnull != shell) {
// Make shell an observer for next time
shell->BeginObservingDocument();
// Resize-reflow this time
nsCOMPtr<nsIPresContext> cx;
shell->GetPresContext(getter_AddRefs(cx));
1998-07-27 17:50:58 +00:00
nsRect r;
cx->GetVisibleArea(r);
shell->InitialReflow(r.width, r.height);
// Now trigger a refresh
// XXX It's unfortunate that this has to be here
nsCOMPtr<nsIViewManager> vm;
shell->GetViewManager(getter_AddRefs(vm));
if (vm) {
vm->EnableRefresh(NS_VMREFRESH_IMMEDIATE);
}
1998-07-27 17:50:58 +00:00
NS_RELEASE(shell);
}
}
}
nsresult
nsImageDocument::EndLayout(nsISupports *ctxt,
nsresult status,
const PRUnichar *errorMsg)
{
// Layout has completed: now update the title
UpdateTitle();
return NS_OK;
}
// NOTE: call this AFTER the shell has been installed as an observer of the
// document so the update notification gets processed by the shell
// and it updates the titlebar
nsresult nsImageDocument::UpdateTitle( void )
{
#ifdef USE_EXTENSION_FOR_TYPE
// XXX TEMPORARY XXX
// We want to display the image type, however there is no way to right now
// so instead we just get the image-extension
// - get the URL interface, get the extension, convert to upper-case
// Unless the Imagerequest or Image can tell us the type this is the best we can do.
nsIURL *pURL=nsnull;
if(NS_SUCCEEDED(mDocumentURL->QueryInterface(NS_GET_IID(nsIURL),(void **)&pURL))){
char *pExtension=nsnull;
pURL->GetFileExtension(&pExtension);
if(pExtension){
2000-04-15 20:15:37 +00:00
nsString strExt; strExt.AssignWithConversion(pExtension);
strExt.ToUpperCase();
titleStr.Append(strExt);
nsCRT::free(pExtension);
pExtension=nsnull;
}
NS_IF_RELEASE(pURL);
}
#endif
nsCOMPtr<nsIStringBundle> bundle;
nsresult rv;
// Create a bundle for the localization
NS_WITH_SERVICE(nsIStringBundleService, stringService, kStringBundleServiceCID, &rv);
if (NS_SUCCEEDED(rv) && stringService) {
nsCOMPtr<nsILocale> locale = nsnull;
rv = stringService->CreateBundle(NSIMAGEDOCUMENT_PROPERTIES_URI, locale, getter_AddRefs(bundle));
}
if (NS_SUCCEEDED(rv) && bundle) {
nsAutoString key;
nsXPIDLString valUni;
if (mImageRequest) {
PRUint32 width, height;
mImageRequest->GetNaturalDimensions(&width, &height);
// if we got a valid size (sometimes we do not) then display it
if (width != 0 && height != 0){
key.AssignWithConversion("ImageTitleWithDimensions");
nsAutoString widthStr; widthStr.AppendInt(width);
nsAutoString heightStr; heightStr.AppendInt(height);
const PRUnichar *formatStrings[2] = {widthStr.GetUnicode(), heightStr.GetUnicode()};
rv = bundle->FormatStringFromName(key.GetUnicode(), formatStrings, 2, getter_Copies(valUni));
}
}
if (nsLiteralString(valUni).IsEmpty()) {
key.AssignWithConversion("ImageTitleWithoutDimensions");
rv = bundle->GetStringFromName(key.GetUnicode(), getter_Copies(valUni));
}
if (NS_SUCCEEDED(rv) && valUni) {
nsString titleStr;
titleStr.Assign(valUni);
// set it on the document
SetTitle(titleStr);
}
}
return NS_OK;
}