Files
Mozilla/mozilla/modules/libpr0n/src/imgRequestProxy.cpp
pavlov%netscape.com 0c94ba945f fixing bug 106199. r=ben sr=hyatt
git-svn-id: svn://10.0.0.236/trunk@106035 18797224-902f-48f8-a5cc-f745e15eee43
2001-10-23 08:07:03 +00:00

372 lines
8.6 KiB
C++

/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
*
* The contents of this file are subject to the Mozilla Public
* License Version 1.1 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at http://www.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
* implied. See the License for the specific language governing
* rights and limitations under the License.
*
* The Original Code is mozilla.org code.
*
* The Initial Developer of the Original Code is Netscape
* Communications Corporation. Portions created by Netscape are
* Copyright (C) 2001 Netscape Communications Corporation.
* All Rights Reserved.
*
* Contributor(s):
* Stuart Parmenter <pavlov@netscape.com>
*/
#include "imgRequestProxy.h"
#include "nsIInputStream.h"
#include "nsIComponentManager.h"
#include "nsIServiceManager.h"
#include "nsAutoLock.h"
#include "nsString.h"
#include "nsXPIDLString.h"
#include "ImageLogging.h"
#include "nspr.h"
NS_IMPL_THREADSAFE_ISUPPORTS2(imgRequestProxy, imgIRequest, nsIRequest)
imgRequestProxy::imgRequestProxy() :
mOwner(nsnull),
mLoadFlags(nsIRequest::LOAD_NORMAL),
mCanceled(PR_FALSE),
mLock(nsnull)
{
NS_INIT_ISUPPORTS();
/* member initializers and constructor code */
mLock = PR_NewLock();
}
imgRequestProxy::~imgRequestProxy()
{
/* destructor code */
if (mOwner) {
if (!mCanceled) {
PR_Lock(mLock);
mCanceled = PR_TRUE;
/* set mListener to null so that we don't forward any callbacks that
RemoveProxy might generate
*/
mListener = nsnull;
PR_Unlock(mLock);
/* Call RemoveProxy with a successful status. This will keep the
channel, if still downloading data, from being canceled if 'this' is
the last observer. This allows the image to continue to download and
be cached even if no one is using it currently.
*/
mOwner->RemoveProxy(this, NS_OK);
}
NS_RELEASE(mOwner);
}
PR_DestroyLock(mLock);
}
nsresult imgRequestProxy::Init(imgRequest *request, nsILoadGroup *aLoadGroup, imgIDecoderObserver *aObserver, nsISupports *cx)
{
NS_PRECONDITION(request, "no request");
if (!request)
return NS_ERROR_NULL_POINTER;
LOG_SCOPE_WITH_PARAM(gImgLog, "imgRequestProxy::Init", "request", request);
PR_Lock(mLock);
mOwner = request;
NS_ADDREF(mOwner);
mListener = aObserver;
mContext = cx;
if (aLoadGroup) {
PRUint32 imageStatus = mOwner->GetImageStatus();
if (!(imageStatus & imgIRequest::STATUS_LOAD_COMPLETE) &&
!(imageStatus & imgIRequest::STATUS_ERROR)) {
aLoadGroup->AddRequest(this, cx);
mLoadGroup = aLoadGroup;
}
}
PR_Unlock(mLock);
request->AddProxy(this);
return NS_OK;
}
/** nsIRequest / imgIRequest methods **/
/* readonly attribute wstring name; */
NS_IMETHODIMP imgRequestProxy::GetName(PRUnichar * *aName)
{
nsAutoString name;
if (mOwner) {
nsCOMPtr<nsIURI> uri;
mOwner->GetURI(getter_AddRefs(uri));
if (uri) {
nsXPIDLCString spec;
uri->GetSpec(getter_Copies(spec));
if (spec)
name.Append(NS_ConvertUTF8toUCS2(spec));
}
}
*aName = nsCRT::strdup(name.get());
return NS_OK;
}
/* boolean isPending (); */
NS_IMETHODIMP imgRequestProxy::IsPending(PRBool *_retval)
{
return NS_ERROR_NOT_IMPLEMENTED;
}
/* readonly attribute nsresult status; */
NS_IMETHODIMP imgRequestProxy::GetStatus(nsresult *aStatus)
{
return NS_ERROR_NOT_IMPLEMENTED;
}
/* void cancel (in nsresult status); */
NS_IMETHODIMP imgRequestProxy::Cancel(nsresult status)
{
if (mCanceled || !mOwner)
return NS_ERROR_FAILURE;
LOG_SCOPE(gImgLog, "imgRequestProxy::Cancel");
PR_Lock(mLock);
mCanceled = PR_TRUE;
mListener = nsnull;
PR_Unlock(mLock);
mOwner->RemoveProxy(this, status);
return NS_OK;
}
/* void suspend (); */
NS_IMETHODIMP imgRequestProxy::Suspend()
{
return NS_ERROR_NOT_IMPLEMENTED;
}
/* void resume (); */
NS_IMETHODIMP imgRequestProxy::Resume()
{
return NS_ERROR_NOT_IMPLEMENTED;
}
/* attribute nsILoadGroup loadGroup */
NS_IMETHODIMP imgRequestProxy::GetLoadGroup(nsILoadGroup **loadGroup)
{
nsAutoLock lock(mLock);
NS_IF_ADDREF(*loadGroup = mLoadGroup.get());
return NS_OK;
}
NS_IMETHODIMP imgRequestProxy::SetLoadGroup(nsILoadGroup *loadGroup)
{
nsAutoLock lock(mLock);
mLoadGroup = loadGroup;
return NS_OK;
}
/* attribute nsLoadFlags loadFlags */
NS_IMETHODIMP imgRequestProxy::GetLoadFlags(nsLoadFlags *flags)
{
*flags = mLoadFlags;
return NS_OK;
}
NS_IMETHODIMP imgRequestProxy::SetLoadFlags(nsLoadFlags flags)
{
mLoadFlags = flags;
return NS_OK;
}
/** imgIRequest methods **/
/* attribute imgIContainer image; */
NS_IMETHODIMP imgRequestProxy::GetImage(imgIContainer * *aImage)
{
if (!mOwner)
return NS_ERROR_FAILURE;
nsAutoLock lock(mLock);
mOwner->GetImage(aImage);
return NS_OK;
}
/* readonly attribute unsigned long imageStatus; */
NS_IMETHODIMP imgRequestProxy::GetImageStatus(PRUint32 *aStatus)
{
if (!mOwner) {
*aStatus = imgIRequest::STATUS_ERROR;
return NS_ERROR_FAILURE;
}
nsAutoLock lock(mLock);
*aStatus = mOwner->GetImageStatus();
return NS_OK;
}
/* readonly attribute nsIURI URI; */
NS_IMETHODIMP imgRequestProxy::GetURI(nsIURI **aURI)
{
if (!mOwner)
return NS_ERROR_FAILURE;
nsAutoLock lock(mLock);
return mOwner->GetURI(aURI);
}
/* readonly attribute imgIDecoderObserver decoderObserver; */
NS_IMETHODIMP imgRequestProxy::GetDecoderObserver(imgIDecoderObserver **aDecoderObserver)
{
*aDecoderObserver = mListener;
NS_IF_ADDREF(*aDecoderObserver);
return NS_OK;
}
/* readonly attribute string mimeType; */
NS_IMETHODIMP imgRequestProxy::GetMimeType(char **aMimeType)
{
if (!mOwner)
return NS_ERROR_FAILURE;
const char *type = mOwner->GetMimeType();
if (!type)
return NS_ERROR_FAILURE;
*aMimeType = nsCRT::strdup(type);
return NS_OK;
}
/** imgIContainerObserver methods **/
void imgRequestProxy::FrameChanged(imgIContainer *container, gfxIImageFrame *newframe, nsRect * dirtyRect)
{
LOG_FUNC(gImgLog, "imgRequestProxy::FrameChanged");
if (mListener)
mListener->FrameChanged(container, mContext, newframe, dirtyRect);
}
/** imgIDecoderObserver methods **/
void imgRequestProxy::OnStartDecode()
{
LOG_FUNC(gImgLog, "imgRequestProxy::OnStartDecode");
if (mListener)
mListener->OnStartDecode(this, mContext);
}
void imgRequestProxy::OnStartContainer(imgIContainer *image)
{
LOG_FUNC(gImgLog, "imgRequestProxy::OnStartContainer");
if (mListener)
mListener->OnStartContainer(this, mContext, image);
}
void imgRequestProxy::OnStartFrame(gfxIImageFrame *frame)
{
LOG_FUNC(gImgLog, "imgRequestProxy::OnStartFrame");
if (mListener)
mListener->OnStartFrame(this, mContext, frame);
}
void imgRequestProxy::OnDataAvailable(gfxIImageFrame *frame, const nsRect * rect)
{
LOG_FUNC(gImgLog, "imgRequestProxy::OnDataAvailable");
if (mListener)
mListener->OnDataAvailable(this, mContext, frame, rect);
}
void imgRequestProxy::OnStopFrame(gfxIImageFrame *frame)
{
LOG_FUNC(gImgLog, "imgRequestProxy::OnStopFrame");
if (mListener)
mListener->OnStopFrame(this, mContext, frame);
}
void imgRequestProxy::OnStopContainer(imgIContainer *image)
{
LOG_FUNC(gImgLog, "imgRequestProxy::OnStopContainer");
if (mListener)
mListener->OnStopContainer(this, mContext, image);
}
void imgRequestProxy::OnStopDecode(nsresult status, const PRUnichar *statusArg)
{
LOG_FUNC(gImgLog, "imgRequestProxy::OnStopDecode");
if (mListener)
mListener->OnStopDecode(this, mContext, status, statusArg);
}
void imgRequestProxy::OnStartRequest(nsIRequest *request, nsISupports *ctxt)
{
#ifdef PR_LOGGING
nsXPIDLString name;
GetName(getter_Copies(name));
LOG_FUNC_WITH_PARAM(gImgLog, "imgRequestProxy::OnStartRequest", "name", NS_ConvertUCS2toUTF8(name).get());
#endif
}
void imgRequestProxy::OnStopRequest(nsIRequest *request, nsISupports *ctxt, nsresult statusCode)
{
/* it is ok to get multiple OnStopRequest messages */
if (!mLoadGroup)
return;
#ifdef PR_LOGGING
nsXPIDLString name;
GetName(getter_Copies(name));
LOG_FUNC_WITH_PARAM(gImgLog, "imgRequestProxy::OnStopRequest", "name", NS_ConvertUCS2toUTF8(name).get());
#endif
/* calling RemoveRequest may cause the document to finish loading,
which could result in our death. We need to make sure that we stay
alive long enough to fight another battle... at least until we exit
this function.
*/
nsCOMPtr<imgIRequest> kungFuDeathGrip(this);
mLoadGroup->RemoveRequest(this, mContext, statusCode);
mLoadGroup = nsnull;
}