forked from mirrors/gecko-dev
Bug 753 - Remove nsIImage, gfxIImageFrame, and their implementations, and expose an equivalent api on imgIContainer. r=roc,josh,bz,longsonr,vlad,karlt,jimm,bsmedberg,mfinkle,peterw,peterv sr=vlad,roc
--HG-- rename : gfx/src/shared/gfxImageFrame.cpp => modules/libpr0n/src/imgFrame.cpp rename : gfx/src/shared/gfxImageFrame.h => modules/libpr0n/src/imgFrame.h
This commit is contained in:
parent
a6e367551a
commit
2aa59235f2
98 changed files with 2290 additions and 3404 deletions
|
|
@ -115,9 +115,9 @@ nsHTMLImageAccessible::GetStateInternal(PRUint32 *aState, PRUint32 *aExtraState)
|
|||
imageRequest->GetImage(getter_AddRefs(imgContainer));
|
||||
|
||||
if (imgContainer) {
|
||||
PRUint32 numFrames;
|
||||
imgContainer->GetNumFrames(&numFrames);
|
||||
if (numFrames > 1)
|
||||
PRBool animated;
|
||||
imgContainer->GetAnimated(&animated);
|
||||
if (animated)
|
||||
*aState |= nsIAccessibleStates::STATE_ANIMATED;
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -91,4 +91,12 @@ EXTRA_DSO_LDOPTS += \
|
|||
$(NULL)
|
||||
endif
|
||||
|
||||
ifndef MOZ_ENABLE_LIBXUL
|
||||
ifeq ($(MOZ_WIDGET_TOOLKIT),windows)
|
||||
# Doesn't work, couldn't figure out why
|
||||
#EXTRA_DSO_LIBS += thebes
|
||||
EXTRA_DSO_LDOPTS += $(LIBXUL_DIST)/lib/$(LIB_PREFIX)thebes.$(LIB_SUFFIX)
|
||||
endif
|
||||
endif
|
||||
|
||||
include $(topsrcdir)/config/rules.mk
|
||||
|
|
|
|||
|
|
@ -47,7 +47,6 @@
|
|||
#include "nsIGConfService.h"
|
||||
#include "nsIGnomeVFSService.h"
|
||||
#include "nsIStringBundle.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsIOutputStream.h"
|
||||
#include "nsIProcess.h"
|
||||
#include "nsNetUtil.h"
|
||||
|
|
@ -55,7 +54,6 @@
|
|||
#include "nsIImageLoadingContent.h"
|
||||
#include "imgIRequest.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "nsIImage.h"
|
||||
#include "prprf.h"
|
||||
#ifdef MOZ_WIDGET_GTK2
|
||||
#include "nsIImageToPixbuf.h"
|
||||
|
|
@ -349,12 +347,8 @@ nsGNOMEShellService::SetShouldCheckDefaultBrowser(PRBool aShouldCheck)
|
|||
}
|
||||
|
||||
static nsresult
|
||||
WriteImage(const nsCString& aPath, gfxIImageFrame* aImage)
|
||||
WriteImage(const nsCString& aPath, imgIContainer* aImage)
|
||||
{
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(aImage));
|
||||
if (!img)
|
||||
return NS_ERROR_NOT_AVAILABLE;
|
||||
|
||||
#ifndef MOZ_WIDGET_GTK2
|
||||
return NS_ERROR_NOT_AVAILABLE;
|
||||
#else
|
||||
|
|
@ -363,7 +357,7 @@ WriteImage(const nsCString& aPath, gfxIImageFrame* aImage)
|
|||
if (!imgToPixbuf)
|
||||
return NS_ERROR_NOT_AVAILABLE;
|
||||
|
||||
GdkPixbuf* pixbuf = imgToPixbuf->ConvertImageToPixbuf(img);
|
||||
GdkPixbuf* pixbuf = imgToPixbuf->ConvertImageToPixbuf(aImage);
|
||||
if (!pixbuf)
|
||||
return NS_ERROR_NOT_AVAILABLE;
|
||||
|
||||
|
|
@ -379,8 +373,6 @@ nsGNOMEShellService::SetDesktopBackground(nsIDOMElement* aElement,
|
|||
PRInt32 aPosition)
|
||||
{
|
||||
nsresult rv;
|
||||
nsCOMPtr<gfxIImageFrame> gfxFrame;
|
||||
|
||||
nsCOMPtr<nsIImageLoadingContent> imageContent = do_QueryInterface(aElement, &rv);
|
||||
if (!imageContent) return rv;
|
||||
|
||||
|
|
@ -393,12 +385,6 @@ nsGNOMEShellService::SetDesktopBackground(nsIDOMElement* aElement,
|
|||
rv = request->GetImage(getter_AddRefs(container));
|
||||
if (!container) return rv;
|
||||
|
||||
// get the current frame, which holds the image data
|
||||
container->GetCurrentFrame(getter_AddRefs(gfxFrame));
|
||||
|
||||
if (!gfxFrame)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
// Write the background file to the home directory.
|
||||
nsCAutoString filePath(PR_GetEnv("HOME"));
|
||||
|
||||
|
|
@ -423,7 +409,7 @@ nsGNOMEShellService::SetDesktopBackground(nsIDOMElement* aElement,
|
|||
filePath.Append("_wallpaper.png");
|
||||
|
||||
// write the image to a file in the home dir
|
||||
rv = WriteImage(filePath, gfxFrame);
|
||||
rv = WriteImage(filePath, container);
|
||||
|
||||
// if the file was written successfully, set it as the system wallpaper
|
||||
nsCOMPtr<nsIGConfService> gconf = do_GetService(NS_GCONFSERVICE_CONTRACTID);
|
||||
|
|
|
|||
|
|
@ -41,7 +41,6 @@
|
|||
*
|
||||
* ***** END LICENSE BLOCK ***** */
|
||||
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "imgIRequest.h"
|
||||
#include "nsIDOMDocument.h"
|
||||
|
|
@ -519,23 +518,18 @@ nsWindowsShellService::SetShouldCheckDefaultBrowser(PRBool aShouldCheck)
|
|||
}
|
||||
|
||||
static nsresult
|
||||
WriteBitmap(nsIFile* aFile, gfxIImageFrame* aImage)
|
||||
WriteBitmap(nsIFile* aFile, imgIContainer* aImage)
|
||||
{
|
||||
PRInt32 width, height;
|
||||
aImage->GetWidth(&width);
|
||||
aImage->GetHeight(&height);
|
||||
nsRefPtr<gfxImageSurface> image;
|
||||
nsresult rv = aImage->CopyCurrentFrame(getter_AddRefs(image));
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
PRUint8* bits;
|
||||
PRUint32 length;
|
||||
aImage->LockImageData();
|
||||
aImage->GetImageData(&bits, &length);
|
||||
if (!bits) {
|
||||
aImage->UnlockImageData();
|
||||
return NS_ERROR_FAILURE;
|
||||
}
|
||||
PRInt32 width = image->Width();
|
||||
PRInt32 height = image->Height();
|
||||
|
||||
PRUint32 bpr;
|
||||
aImage->GetImageBytesPerRow(&bpr);
|
||||
PRUint8* bits = image->Data();
|
||||
PRUint32 length = image->GetDataSize();
|
||||
PRUint32 bpr = PRUint32(image->Stride());
|
||||
PRInt32 bitCount = bpr/width;
|
||||
|
||||
// initialize these bitmap structs which we will later
|
||||
|
|
@ -562,7 +556,7 @@ WriteBitmap(nsIFile* aFile, gfxIImageFrame* aImage)
|
|||
|
||||
// get a file output stream
|
||||
nsCOMPtr<nsIOutputStream> stream;
|
||||
nsresult rv = NS_NewLocalFileOutputStream(getter_AddRefs(stream), aFile);
|
||||
rv = NS_NewLocalFileOutputStream(getter_AddRefs(stream), aFile);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
// write the bitmap headers and rgb pixel data to the file
|
||||
|
|
@ -592,7 +586,6 @@ WriteBitmap(nsIFile* aFile, gfxIImageFrame* aImage)
|
|||
stream->Close();
|
||||
}
|
||||
|
||||
aImage->UnlockImageData();
|
||||
return rv;
|
||||
}
|
||||
|
||||
|
|
@ -602,8 +595,7 @@ nsWindowsShellService::SetDesktopBackground(nsIDOMElement* aElement,
|
|||
{
|
||||
nsresult rv;
|
||||
|
||||
nsCOMPtr<gfxIImageFrame> gfxFrame;
|
||||
|
||||
nsCOMPtr<imgIContainer> container;
|
||||
nsCOMPtr<nsIDOMHTMLImageElement> imgElement(do_QueryInterface(aElement));
|
||||
if (!imgElement) {
|
||||
// XXX write background loading stuff!
|
||||
|
|
@ -620,18 +612,11 @@ nsWindowsShellService::SetDesktopBackground(nsIDOMElement* aElement,
|
|||
getter_AddRefs(request));
|
||||
if (!request)
|
||||
return rv;
|
||||
nsCOMPtr<imgIContainer> container;
|
||||
rv = request->GetImage(getter_AddRefs(container));
|
||||
if (!container)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
// get the current frame, which holds the image data
|
||||
container->GetCurrentFrame(getter_AddRefs(gfxFrame));
|
||||
}
|
||||
|
||||
if (!gfxFrame)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
// get the file name from localized strings
|
||||
nsCOMPtr<nsIStringBundleService>
|
||||
bundleService(do_GetService(NS_STRINGBUNDLE_CONTRACTID, &rv));
|
||||
|
|
@ -664,7 +649,7 @@ nsWindowsShellService::SetDesktopBackground(nsIDOMElement* aElement,
|
|||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
// write the bitmap to a file in the profile directory
|
||||
rv = WriteBitmap(file, gfxFrame);
|
||||
rv = WriteBitmap(file, container);
|
||||
|
||||
// if the file was written successfully, set it as the system wallpaper
|
||||
if (NS_SUCCEEDED(rv)) {
|
||||
|
|
|
|||
|
|
@ -78,12 +78,12 @@ class nsIThreadJSContextStack;
|
|||
class nsIParserService;
|
||||
class nsIIOService;
|
||||
class nsIURI;
|
||||
class imgIContainer;
|
||||
class imgIDecoderObserver;
|
||||
class imgIRequest;
|
||||
class imgILoader;
|
||||
class imgICache;
|
||||
class nsIPrefBranch;
|
||||
class nsIImage;
|
||||
class nsIImageLoadingContent;
|
||||
class nsIDOMHTMLFormElement;
|
||||
class nsIDOMDocument;
|
||||
|
|
@ -654,13 +654,13 @@ public:
|
|||
static PRBool IsImageInCache(nsIURI* aURI);
|
||||
|
||||
/**
|
||||
* Method to get an nsIImage from an image loading content
|
||||
* Method to get an imgIContainer from an image loading content
|
||||
*
|
||||
* @param aContent The image loading content. Must not be null.
|
||||
* @param aRequest The image request [out]
|
||||
* @return the nsIImage corresponding to the first frame of the image
|
||||
* @return the imgIContainer corresponding to the first frame of the image
|
||||
*/
|
||||
static already_AddRefed<nsIImage> GetImageFromContent(nsIImageLoadingContent* aContent, imgIRequest **aRequest = nsnull);
|
||||
static already_AddRefed<imgIContainer> GetImageFromContent(nsIImageLoadingContent* aContent, imgIRequest **aRequest = nsnull);
|
||||
|
||||
/**
|
||||
* Method that decides whether a content node is draggable
|
||||
|
|
|
|||
|
|
@ -80,7 +80,6 @@
|
|||
#include "nsINameSpaceManager.h"
|
||||
#include "nsUnicharUtils.h"
|
||||
#include "nsIURL.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsIDocument.h"
|
||||
#include "nsIScriptSecurityManager.h"
|
||||
#include "nsIPrincipal.h"
|
||||
|
|
@ -93,6 +92,7 @@
|
|||
#include "nsEscape.h"
|
||||
#include "nsContentUtils.h"
|
||||
#include "nsIMIMEService.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "imgIRequest.h"
|
||||
#include "nsContentCID.h"
|
||||
#include "nsDOMDataTransfer.h"
|
||||
|
|
@ -171,7 +171,7 @@ private:
|
|||
nsString mInfoString;
|
||||
|
||||
PRBool mIsAnchor;
|
||||
nsCOMPtr<nsIImage> mImage;
|
||||
nsCOMPtr<imgIContainer> mImage;
|
||||
};
|
||||
|
||||
|
||||
|
|
@ -972,7 +972,7 @@ nsTransferableFactory::Produce(nsDOMDataTransfer* aDataTransfer,
|
|||
nsCOMPtr<imgIRequest> imgRequest;
|
||||
|
||||
// grab the image data, and its request.
|
||||
nsCOMPtr<nsIImage> img =
|
||||
nsCOMPtr<imgIContainer> img =
|
||||
nsContentUtils::GetImageFromContent(image,
|
||||
getter_AddRefs(imgRequest));
|
||||
|
||||
|
|
|
|||
|
|
@ -53,7 +53,6 @@ class nsIDOMDocument;
|
|||
class nsIDOMDragEvent;
|
||||
class nsISelection;
|
||||
class nsITransferable;
|
||||
class nsIImage;
|
||||
class nsIPresShell;
|
||||
class nsPresContext;
|
||||
class nsIContent;
|
||||
|
|
|
|||
|
|
@ -98,8 +98,6 @@
|
|||
#include "imgIRequest.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "imgILoader.h"
|
||||
#include "nsIImage.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsIImageLoadingContent.h"
|
||||
#include "nsIInterfaceRequestor.h"
|
||||
#include "nsIInterfaceRequestorUtils.h"
|
||||
|
|
@ -2446,7 +2444,7 @@ nsContentUtils::LoadImage(nsIURI* aURI, nsIDocument* aLoadingDocument,
|
|||
}
|
||||
|
||||
// static
|
||||
already_AddRefed<nsIImage>
|
||||
already_AddRefed<imgIContainer>
|
||||
nsContentUtils::GetImageFromContent(nsIImageLoadingContent* aContent,
|
||||
imgIRequest **aRequest)
|
||||
{
|
||||
|
|
@ -2470,26 +2468,11 @@ nsContentUtils::GetImageFromContent(nsIImageLoadingContent* aContent,
|
|||
return nsnull;
|
||||
}
|
||||
|
||||
nsCOMPtr<gfxIImageFrame> imgFrame;
|
||||
imgContainer->GetFrameAt(0, getter_AddRefs(imgFrame));
|
||||
|
||||
if (!imgFrame) {
|
||||
return nsnull;
|
||||
}
|
||||
|
||||
nsCOMPtr<nsIInterfaceRequestor> ir = do_QueryInterface(imgFrame);
|
||||
|
||||
if (!ir) {
|
||||
return nsnull;
|
||||
}
|
||||
|
||||
if (aRequest) {
|
||||
imgRequest.swap(*aRequest);
|
||||
}
|
||||
|
||||
nsIImage* image = nsnull;
|
||||
CallGetInterface(ir.get(), &image);
|
||||
return image;
|
||||
return imgContainer.forget();
|
||||
}
|
||||
|
||||
// static
|
||||
|
|
|
|||
|
|
@ -49,6 +49,7 @@
|
|||
#include "nsXPCOM.h"
|
||||
#include "nsISupportsPrimitives.h"
|
||||
#include "nsIDOMRange.h"
|
||||
#include "imgIContainer.h"
|
||||
|
||||
#include "nsIDocShell.h"
|
||||
#include "nsIContentViewerEdit.h"
|
||||
|
|
@ -65,7 +66,6 @@
|
|||
// image copy stuff
|
||||
#include "nsIImageLoadingContent.h"
|
||||
#include "nsIInterfaceRequestorUtils.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsContentUtils.h"
|
||||
#include "nsContentCID.h"
|
||||
|
||||
|
|
@ -463,7 +463,7 @@ nsCopySupport::ImageCopy(nsIImageLoadingContent* aImageElement,
|
|||
|
||||
if (aCopyFlags & nsIContentViewerEdit::COPY_IMAGE_DATA) {
|
||||
// get the image data from the element
|
||||
nsCOMPtr<nsIImage> image =
|
||||
nsCOMPtr<imgIContainer> image =
|
||||
nsContentUtils::GetImageFromContent(aImageElement);
|
||||
NS_ENSURE_TRUE(image, NS_ERROR_FAILURE);
|
||||
|
||||
|
|
|
|||
|
|
@ -55,7 +55,6 @@
|
|||
#include "nsIURI.h"
|
||||
#include "nsILoadGroup.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "imgILoader.h"
|
||||
#include "nsThreadUtils.h"
|
||||
#include "nsNetUtil.h"
|
||||
|
|
@ -158,10 +157,9 @@ nsImageLoadingContent::~nsImageLoadingContent()
|
|||
*/
|
||||
NS_IMETHODIMP
|
||||
nsImageLoadingContent::FrameChanged(imgIContainer* aContainer,
|
||||
gfxIImageFrame* aFrame,
|
||||
nsIntRect* aDirtyRect)
|
||||
{
|
||||
LOOP_OVER_OBSERVERS(FrameChanged(aContainer, aFrame, aDirtyRect));
|
||||
LOOP_OVER_OBSERVERS(FrameChanged(aContainer, aDirtyRect));
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
|
@ -196,7 +194,7 @@ nsImageLoadingContent::OnStartContainer(imgIRequest* aRequest,
|
|||
|
||||
NS_IMETHODIMP
|
||||
nsImageLoadingContent::OnStartFrame(imgIRequest* aRequest,
|
||||
gfxIImageFrame* aFrame)
|
||||
PRUint32 aFrame)
|
||||
{
|
||||
LOOP_OVER_OBSERVERS(OnStartFrame(aRequest, aFrame));
|
||||
return NS_OK;
|
||||
|
|
@ -204,16 +202,16 @@ nsImageLoadingContent::OnStartFrame(imgIRequest* aRequest,
|
|||
|
||||
NS_IMETHODIMP
|
||||
nsImageLoadingContent::OnDataAvailable(imgIRequest* aRequest,
|
||||
gfxIImageFrame* aFrame,
|
||||
PRBool aCurrentFrame,
|
||||
const nsIntRect* aRect)
|
||||
{
|
||||
LOOP_OVER_OBSERVERS(OnDataAvailable(aRequest, aFrame, aRect));
|
||||
LOOP_OVER_OBSERVERS(OnDataAvailable(aRequest, aCurrentFrame, aRect));
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
NS_IMETHODIMP
|
||||
nsImageLoadingContent::OnStopFrame(imgIRequest* aRequest,
|
||||
gfxIImageFrame* aFrame)
|
||||
PRUint32 aFrame)
|
||||
{
|
||||
LOOP_OVER_OBSERVERS(OnStopFrame(aRequest, aFrame));
|
||||
return NS_OK;
|
||||
|
|
|
|||
|
|
@ -58,14 +58,14 @@ nsStubImageDecoderObserver::OnStartContainer(imgIRequest *aRequest,
|
|||
|
||||
NS_IMETHODIMP
|
||||
nsStubImageDecoderObserver::OnStartFrame(imgIRequest *aRequest,
|
||||
gfxIImageFrame *aFrame)
|
||||
PRUint32 aFrame)
|
||||
{
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
NS_IMETHODIMP
|
||||
nsStubImageDecoderObserver::OnDataAvailable(imgIRequest *aRequest,
|
||||
gfxIImageFrame *aFrame,
|
||||
PRBool aCurrentFrame,
|
||||
const nsIntRect * aRect)
|
||||
{
|
||||
return NS_OK;
|
||||
|
|
@ -73,7 +73,7 @@ nsStubImageDecoderObserver::OnDataAvailable(imgIRequest *aRequest,
|
|||
|
||||
NS_IMETHODIMP
|
||||
nsStubImageDecoderObserver::OnStopFrame(imgIRequest *aRequest,
|
||||
gfxIImageFrame *aFrame)
|
||||
PRUint32 aFrame)
|
||||
{
|
||||
return NS_OK;
|
||||
}
|
||||
|
|
@ -102,7 +102,6 @@ nsStubImageDecoderObserver::OnStopRequest(imgIRequest *aRequest,
|
|||
|
||||
NS_IMETHODIMP
|
||||
nsStubImageDecoderObserver::FrameChanged(imgIContainer *aContainer,
|
||||
gfxIImageFrame *aFrame,
|
||||
nsIntRect * aDirtyRect)
|
||||
{
|
||||
return NS_OK;
|
||||
|
|
|
|||
|
|
@ -61,8 +61,6 @@
|
|||
#include "nsIDOMSVGURIReference.h"
|
||||
#include "nsImageLoadingContent.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsNetUtil.h"
|
||||
#include "nsSVGPreserveAspectRatio.h"
|
||||
#include "nsIInterfaceRequestorUtils.h"
|
||||
|
|
@ -5191,8 +5189,7 @@ public:
|
|||
NS_IMETHOD OnStopDecode(imgIRequest *aRequest, nsresult status,
|
||||
const PRUnichar *statusArg);
|
||||
// imgIContainerObserver
|
||||
NS_IMETHOD FrameChanged(imgIContainer *aContainer, gfxIImageFrame *newframe,
|
||||
nsIntRect *dirtyRect);
|
||||
NS_IMETHOD FrameChanged(imgIContainer *aContainer, nsIntRect *dirtyRect);
|
||||
// imgIContainerObserver
|
||||
NS_IMETHOD OnStartContainer(imgIRequest *aRequest,
|
||||
imgIContainer *aContainer);
|
||||
|
|
@ -5378,31 +5375,27 @@ nsSVGFEImageElement::Filter(nsSVGFilterInstance *instance,
|
|||
if (currentRequest)
|
||||
currentRequest->GetImage(getter_AddRefs(imageContainer));
|
||||
|
||||
nsCOMPtr<gfxIImageFrame> currentFrame;
|
||||
nsRefPtr<gfxASurface> currentFrame;
|
||||
if (imageContainer)
|
||||
imageContainer->GetCurrentFrame(getter_AddRefs(currentFrame));
|
||||
|
||||
nsRefPtr<gfxPattern> thebesPattern = nsnull;
|
||||
if (currentFrame) {
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(currentFrame));
|
||||
|
||||
img->GetPattern(getter_AddRefs(thebesPattern));
|
||||
}
|
||||
// We need to wrap the surface in a pattern to have somewhere to set the
|
||||
// graphics filter.
|
||||
nsRefPtr<gfxPattern> thebesPattern;
|
||||
if (currentFrame)
|
||||
thebesPattern = new gfxPattern(currentFrame);
|
||||
|
||||
if (thebesPattern) {
|
||||
thebesPattern->SetFilter(nsLayoutUtils::GetGraphicsFilterForFrame(frame));
|
||||
|
||||
PRInt32 x, y, nativeWidth, nativeHeight;
|
||||
currentFrame->GetX(&x);
|
||||
currentFrame->GetY(&y);
|
||||
currentFrame->GetWidth(&nativeWidth);
|
||||
currentFrame->GetHeight(&nativeHeight);
|
||||
PRInt32 nativeWidth, nativeHeight;
|
||||
imageContainer->GetWidth(&nativeWidth);
|
||||
imageContainer->GetHeight(&nativeHeight);
|
||||
|
||||
nsCOMPtr<nsIDOMSVGMatrix> trans;
|
||||
const gfxRect& filterSubregion = aTarget->mFilterPrimitiveSubregion;
|
||||
trans = nsSVGUtils::GetViewBoxTransform(filterSubregion.Width(), filterSubregion.Height(),
|
||||
x, y,
|
||||
nativeWidth, nativeHeight,
|
||||
0, 0, nativeWidth, nativeHeight,
|
||||
mPreserveAspectRatio);
|
||||
nsCOMPtr<nsIDOMSVGMatrix> xy, fini;
|
||||
NS_NewSVGMatrix(getter_AddRefs(xy), 1, 0, 0, 1, filterSubregion.X(), filterSubregion.Y());
|
||||
|
|
@ -5457,11 +5450,10 @@ nsSVGFEImageElement::OnStopDecode(imgIRequest *aRequest,
|
|||
|
||||
NS_IMETHODIMP
|
||||
nsSVGFEImageElement::FrameChanged(imgIContainer *aContainer,
|
||||
gfxIImageFrame *newframe,
|
||||
nsIntRect *dirtyRect)
|
||||
{
|
||||
nsresult rv =
|
||||
nsImageLoadingContent::FrameChanged(aContainer, newframe, dirtyRect);
|
||||
nsImageLoadingContent::FrameChanged(aContainer, dirtyRect);
|
||||
Invalidate();
|
||||
return rv;
|
||||
}
|
||||
|
|
|
|||
|
|
@ -85,7 +85,6 @@
|
|||
#include "nsPresContext.h"
|
||||
#include "nsIParser.h"
|
||||
#include "nsParserCIID.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsXPCOM.h"
|
||||
#include "nsISupportsPrimitives.h"
|
||||
#include "nsLinebreakConverter.h"
|
||||
|
|
|
|||
|
|
@ -93,7 +93,6 @@
|
|||
#include "nsIDOMDocumentFragment.h"
|
||||
#include "nsIPresShell.h"
|
||||
#include "nsPresContext.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsXPCOM.h"
|
||||
#include "nsISupportsPrimitives.h"
|
||||
#include "SetDocTitleTxn.h"
|
||||
|
|
|
|||
|
|
@ -50,7 +50,6 @@ XPIDLSRCS = \
|
|||
nsIFontEnumerator.idl \
|
||||
nsIScriptableRegion.idl \
|
||||
gfxIFormats.idl \
|
||||
gfxIImageFrame.idl \
|
||||
gfxidltypes.idl \
|
||||
$(NULL)
|
||||
|
||||
|
|
|
|||
|
|
@ -1,168 +0,0 @@
|
|||
/** -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
||||
*
|
||||
* ***** BEGIN LICENSE BLOCK *****
|
||||
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
||||
*
|
||||
* 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 the Initial Developer are Copyright (C) 2001
|
||||
* the Initial Developer. All Rights Reserved.
|
||||
*
|
||||
* Contributor(s):
|
||||
* Stuart Parmenter <pavlov@netscape.com>
|
||||
* Chris Saari <saari@netscape.com>
|
||||
*
|
||||
* Alternatively, the contents of this file may be used under the terms of
|
||||
* either of 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 MPL, 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 MPL, the GPL or the LGPL.
|
||||
*
|
||||
* ***** END LICENSE BLOCK ***** */
|
||||
|
||||
#include "nsISupports.idl"
|
||||
#include "gfxidltypes.idl"
|
||||
#include "gfxIFormats.idl"
|
||||
|
||||
%{C++
|
||||
#include "nsRect.h"
|
||||
%}
|
||||
|
||||
native nsRectRef(nsIntRect &);
|
||||
|
||||
/**
|
||||
* gfxIImageFrame interface
|
||||
*
|
||||
* All x, y, width, height values are in pixels.
|
||||
*
|
||||
* @author Tim Rowley <tor@cs.brown.edu>
|
||||
* @author Stuart Parmenter <pavlov@netscape.com>
|
||||
* @version 0.1
|
||||
*/
|
||||
[scriptable, uuid(9c37930b-cadd-453c-89e1-9ed456715b9c)]
|
||||
interface gfxIImageFrame : nsISupports
|
||||
{
|
||||
/**
|
||||
* Create a new \a aWidth x \a aHeight sized image.
|
||||
*
|
||||
* @param aX The x-offset from the origin of the gfxIImageContainer parent.
|
||||
* @param aY The y-offset from the origin of the gfxIImageContainer parent.
|
||||
* @param aWidth The width of the image to create.
|
||||
* @param aHeight The height of the image to create.
|
||||
* @param aFormat the width of the image to create.
|
||||
*
|
||||
* @note The data in a new image is uninitialized.
|
||||
*/
|
||||
void init(in PRInt32 aX,
|
||||
in PRInt32 aY,
|
||||
in PRInt32 aWidth,
|
||||
in PRInt32 aHeight,
|
||||
in gfx_format aFormat,
|
||||
in gfx_depth aDepth);
|
||||
|
||||
/**
|
||||
* TRUE by default. When set to FALSE, you will no longer be able to make any modifications
|
||||
* to the data of the image. Any attempts will fail.
|
||||
*/
|
||||
attribute boolean mutable;
|
||||
|
||||
/**
|
||||
* The x-offset of the image.
|
||||
*/
|
||||
readonly attribute PRInt32 x;
|
||||
|
||||
/**
|
||||
* The y-offset of the image.
|
||||
*/
|
||||
readonly attribute PRInt32 y;
|
||||
|
||||
/**
|
||||
* The width of the image.
|
||||
*/
|
||||
readonly attribute PRInt32 width;
|
||||
|
||||
/**
|
||||
* The height of the image.
|
||||
*/
|
||||
readonly attribute PRInt32 height;
|
||||
|
||||
/**
|
||||
* The rectangle this frame ocupies.
|
||||
* @param rect this is really an out parameter.
|
||||
*/
|
||||
[noscript] void getRect(in nsRectRef rect);
|
||||
|
||||
/**
|
||||
* The image data format the image was created with.
|
||||
* @see gfxIFormats
|
||||
*/
|
||||
readonly attribute gfx_format format;
|
||||
|
||||
/**
|
||||
* returns whether the image requires the background to be painted
|
||||
*/
|
||||
readonly attribute boolean needsBackground;
|
||||
|
||||
/* data accessors */
|
||||
readonly attribute unsigned long imageBytesPerRow;
|
||||
|
||||
/**
|
||||
* returns the number of bytes allocated for the image
|
||||
*/
|
||||
readonly attribute unsigned long imageDataLength;
|
||||
|
||||
// XXX do we copy here? lets not...
|
||||
void getImageData([array, size_is(length)] out PRUint8 bits, out unsigned long length);
|
||||
|
||||
/**
|
||||
* Get Palette data pointer
|
||||
*/
|
||||
void getPaletteData([array, size_is(length)] out gfx_color palette, out unsigned long length);
|
||||
|
||||
/**
|
||||
* Lock image pixels before addressing the data directly
|
||||
*/
|
||||
void lockImageData();
|
||||
|
||||
/**
|
||||
* Unlock image pixels
|
||||
*/
|
||||
void unlockImageData();
|
||||
|
||||
|
||||
/* GIF Specific methods. These should be in a different class or interface. */
|
||||
|
||||
/**
|
||||
* Represents the number of milliseconds until the next frame should be displayed.
|
||||
* @note -1 means that this frame should be displayed forever.
|
||||
*/
|
||||
attribute long timeout;
|
||||
|
||||
/* frame disposal method, used only by GIFs. Should probably subclass image frame
|
||||
* and image container for GIFs special needs, but for simplicity it is here for the
|
||||
* moment
|
||||
*/
|
||||
attribute long frameDisposalMethod;
|
||||
|
||||
/* PNG specific methods */
|
||||
|
||||
attribute long blendMethod;
|
||||
};
|
||||
|
|
@ -60,7 +60,6 @@ EXPORTS = \
|
|||
nsIRenderingContext.h \
|
||||
nsIFontMetrics.h \
|
||||
nsIDeviceContext.h \
|
||||
nsIImage.h \
|
||||
nsGfxCIID.h \
|
||||
nsIRegion.h \
|
||||
nsITheme.h \
|
||||
|
|
|
|||
|
|
@ -1,307 +0,0 @@
|
|||
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
||||
/* ***** BEGIN LICENSE BLOCK *****
|
||||
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
||||
*
|
||||
* 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 the Initial Developer are Copyright (C) 1998
|
||||
* the Initial Developer. All Rights Reserved.
|
||||
*
|
||||
* Contributor(s):
|
||||
*
|
||||
* Alternatively, the contents of this file may be used under the terms of
|
||||
* either of 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 MPL, 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 MPL, the GPL or the LGPL.
|
||||
*
|
||||
* ***** END LICENSE BLOCK ***** */
|
||||
|
||||
#ifndef nsIImage_h___
|
||||
#define nsIImage_h___
|
||||
|
||||
#include "nsISupports.h"
|
||||
#include "nsMargin.h"
|
||||
#include "nsRect.h"
|
||||
#include "gfxPattern.h"
|
||||
|
||||
class gfxASurface;
|
||||
struct gfxMatrix;
|
||||
struct gfxRect;
|
||||
class gfxContext;
|
||||
|
||||
class nsIDeviceContext;
|
||||
|
||||
struct nsColorMap
|
||||
{
|
||||
//I lifted this from the image lib. The difference is that
|
||||
//this uses nscolor instead of NI_RGB. Multiple color pollution
|
||||
//is a bad thing. MMP
|
||||
PRInt32 NumColors; // Number of colors in the colormap.
|
||||
// A negative value can be used to denote a
|
||||
// possibly non-unique set.
|
||||
//nscolor *Map; // Colormap colors.
|
||||
PRUint8 *Index; // NULL, if map is in index order, otherwise
|
||||
// specifies the indices of the map entries. */
|
||||
};
|
||||
|
||||
typedef enum {
|
||||
nsMaskRequirements_kNoMask,
|
||||
nsMaskRequirements_kNeeds1Bit,
|
||||
nsMaskRequirements_kNeeds8Bit
|
||||
} nsMaskRequirements;
|
||||
|
||||
|
||||
#define nsImageUpdateFlags_kColorMapChanged 0x1
|
||||
#define nsImageUpdateFlags_kBitsChanged 0x2
|
||||
|
||||
// IID for the nsIImage interface
|
||||
// 0358ce68-b076-43b0-8f5c-36ed4592822c
|
||||
#define NS_IIMAGE_IID \
|
||||
{ 0x358ce68, 0xb076, 0x43b0, \
|
||||
{ 0x8f, 0x5c, 0x36, 0xed, 0x45, 0x92, 0x82, 0x2c } }
|
||||
|
||||
// Interface to Images
|
||||
class nsIImage : public nsISupports
|
||||
{
|
||||
|
||||
public:
|
||||
NS_DECLARE_STATIC_IID_ACCESSOR(NS_IIMAGE_IID)
|
||||
|
||||
/**
|
||||
* Build and initialize the nsIImage
|
||||
* @param aWidth The width in pixels of the desired pixelmap
|
||||
* @param aHeight The height in pixels of the desired pixelmap
|
||||
* @param aDepth The number of bits per pixel for the pixelmap
|
||||
* @param aMaskRequirements A flag indicating if a alpha mask should be allocated
|
||||
*/
|
||||
virtual nsresult Init(PRInt32 aWidth, PRInt32 aHeight, PRInt32 aDepth, nsMaskRequirements aMaskRequirements) = 0;
|
||||
|
||||
/**
|
||||
* Get the number of bytes per pixel for this image
|
||||
* @update - dwc 2/3/99
|
||||
* @return - the number of bytes per pixel
|
||||
*/
|
||||
virtual PRInt32 GetBytesPix() = 0;
|
||||
|
||||
/**
|
||||
* Get whether rows are organized top to bottom, or bottom to top
|
||||
* @update - syd 3/29/99
|
||||
* @return PR_TRUE if top to bottom, else PR_FALSE
|
||||
*/
|
||||
virtual PRBool GetIsRowOrderTopToBottom() = 0;
|
||||
|
||||
/**
|
||||
* Get the width for the pixelmap
|
||||
* @update - dwc 2/1/99
|
||||
* @return The width in pixels for the pixelmap
|
||||
*/
|
||||
virtual PRInt32 GetWidth() = 0;
|
||||
|
||||
/**
|
||||
* Get the height for the pixelmap
|
||||
* @update - dwc 2/1/99
|
||||
* @return The height in pixels for the pixelmap
|
||||
*/
|
||||
virtual PRInt32 GetHeight() = 0;
|
||||
|
||||
/**
|
||||
* Get a pointer to the bits for the pixelmap, only if it is not optimized
|
||||
* @update - dwc 2/1/99
|
||||
* @return address of the DIB pixel array
|
||||
*/
|
||||
virtual PRUint8 * GetBits() = 0;
|
||||
|
||||
/**
|
||||
* Get the number of bytes needed to get to the next scanline for the pixelmap
|
||||
* @update - dwc 2/1/99
|
||||
* @return The number of bytes in each scanline
|
||||
*/
|
||||
virtual PRInt32 GetLineStride() = 0;
|
||||
|
||||
/**
|
||||
* Get whether this image has an alpha mask. Preferable to testing
|
||||
* if GetAlphaBits() is non-null.
|
||||
* @update - sfraser 10/19/99
|
||||
* @return PR_TRUE if the image has an alpha mask, PR_FALSE otherwise
|
||||
*/
|
||||
virtual PRBool GetHasAlphaMask() = 0;
|
||||
|
||||
/**
|
||||
* Get a pointer to the bits for the alpha mask
|
||||
* @update - dwc 2/1/99
|
||||
* @return address of the alpha mask pixel array
|
||||
*/
|
||||
virtual PRUint8 * GetAlphaBits() = 0;
|
||||
|
||||
/**
|
||||
* Get the number of bytes per scanline for the alpha mask
|
||||
* @update - dwc 2/1/99
|
||||
* @return The number of bytes in each scanline
|
||||
*/
|
||||
virtual PRInt32 GetAlphaLineStride() = 0;
|
||||
|
||||
/**
|
||||
* Update the nsIImage color table
|
||||
* @update - dougt 9/9/08
|
||||
* @param aFlags Used to pass in parameters for the update
|
||||
* @param aUpdateRect The rectangle to update
|
||||
* @return success code. failure means stop decoding
|
||||
*/
|
||||
virtual nsresult ImageUpdated(nsIDeviceContext *aContext, PRUint8 aFlags, nsIntRect *aUpdateRect) = 0;
|
||||
|
||||
/**
|
||||
* Get whether this image's region is completely filled with data.
|
||||
* @return PR_TRUE if image is complete, PR_FALSE if image is not yet
|
||||
* complete or broken
|
||||
*/
|
||||
virtual PRBool GetIsImageComplete() = 0;
|
||||
|
||||
/**
|
||||
* Converted this pixelmap to an optimized pixelmap for the device
|
||||
* @update - dwc 2/1/99
|
||||
* @param aContext The device to optimize for
|
||||
* @return the result of the operation, if NS_OK, then the pixelmap is optimized
|
||||
*/
|
||||
virtual nsresult Optimize(nsIDeviceContext* aContext) = 0;
|
||||
|
||||
/**
|
||||
* Get the colormap for the nsIImage
|
||||
* @update - dwc 2/1/99
|
||||
* @return if non null, the colormap for the pixelmap,otherwise the image is not color mapped
|
||||
*/
|
||||
virtual nsColorMap * GetColorMap() = 0;
|
||||
|
||||
/**
|
||||
* BitBlit the nsIImage to a device, the source and dest can be scaled.
|
||||
* @param aContext the destination
|
||||
* @param aFilter the filter for the image
|
||||
* @param aUserSpaceToImageSpace the transform that maps user-space
|
||||
* coordinates to coordinates in (tiled, post-padding) image pixels
|
||||
* @param aFill the area to fill with tiled images
|
||||
* @param aPadding the padding to be added to this image before tiling,
|
||||
* in image pixels
|
||||
* @param aSubimage the subimage in padded+tiled image space that we're
|
||||
* extracting the contents from. Pixels outside this rectangle must not
|
||||
* be sampled.
|
||||
*
|
||||
* So this is supposed to
|
||||
* -- add aPadding transparent pixels around the image
|
||||
* -- use that image to tile the plane
|
||||
* -- replace everything outside the aSubimage region with the nearest
|
||||
* border pixel of that region (like EXTEND_PAD)
|
||||
* -- fill aFill with the image, using aImageSpaceToDeviceSpace as the
|
||||
* image-space-to-device-space transform
|
||||
*/
|
||||
virtual void Draw(gfxContext* aContext,
|
||||
gfxPattern::GraphicsFilter aFilter,
|
||||
const gfxMatrix& aUserSpaceToImageSpace,
|
||||
const gfxRect& aFill,
|
||||
const nsIntMargin& aPadding,
|
||||
const nsIntRect& aSubimage) = 0;
|
||||
|
||||
/**
|
||||
* Get the alpha depth for the image mask
|
||||
* @update - lordpixel 2001/05/16
|
||||
* @return the alpha mask depth for the image, ie, 0, 1 or 8
|
||||
*/
|
||||
virtual PRInt8 GetAlphaDepth() = 0;
|
||||
|
||||
/**
|
||||
* Return information about the bits for this structure
|
||||
* @update - dwc 2/1/99
|
||||
* @return a bitmap info structure for the Device Dependent Bits
|
||||
*/
|
||||
virtual void* GetBitInfo() = 0;
|
||||
|
||||
|
||||
/**
|
||||
* LockImagePixels
|
||||
* Lock the image pixels so that we can access them directly,
|
||||
* with safety. May be a noop on some platforms.
|
||||
*
|
||||
* If you want to be able to call GetSurface(), wrap the call in
|
||||
* LockImagePixels()/UnlockImagePixels(). This also allows you to write to
|
||||
* the surface returned by GetSurface().
|
||||
*
|
||||
* aMaskPixels = PR_TRUE for the mask, PR_FALSE for the image
|
||||
*
|
||||
* Must be balanced by a call to UnlockImagePixels().
|
||||
*
|
||||
* @update - sfraser 10/18/99
|
||||
* @return error result
|
||||
*/
|
||||
NS_IMETHOD LockImagePixels(PRBool aMaskPixels) = 0;
|
||||
|
||||
/**
|
||||
* UnlockImagePixels
|
||||
* Unlock the image pixels. May be a noop on some platforms.
|
||||
*
|
||||
* Should balance an earlier call to LockImagePixels().
|
||||
*
|
||||
* aMaskPixels = PR_TRUE for the mask, PR_FALSE for the image
|
||||
*
|
||||
* @update - sfraser 10/18/99
|
||||
* @return error result
|
||||
*/
|
||||
NS_IMETHOD UnlockImagePixels(PRBool aMaskPixels) = 0;
|
||||
|
||||
/**
|
||||
* GetSurface
|
||||
* Return the Thebes gfxASurface in aSurface, if there is one. Should be
|
||||
* wrapped by LockImagePixels()/UnlockImagePixels().
|
||||
*
|
||||
* aSurface will be AddRef'd (as with most getters), so
|
||||
* getter_AddRefs should be used.
|
||||
*/
|
||||
NS_IMETHOD GetSurface(gfxASurface **aSurface) = 0;
|
||||
|
||||
/**
|
||||
* GetSurface
|
||||
* Return the Thebes gfxPattern in aPattern. It is always possible to get a
|
||||
* gfxPattern (unlike the gfxASurface from GetSurface()).
|
||||
*
|
||||
* aPattern will be AddRef'd (as with most getters), so
|
||||
* getter_AddRefs should be used.
|
||||
*/
|
||||
NS_IMETHOD GetPattern(gfxPattern **aPattern) = 0;
|
||||
|
||||
/**
|
||||
* SetHasNoAlpha
|
||||
*
|
||||
* Hint to the image that all the pixels are fully opaque, even if
|
||||
* the original format requested a 1-bit or 8-bit alpha mask
|
||||
*/
|
||||
virtual void SetHasNoAlpha() = 0;
|
||||
|
||||
/**
|
||||
* Extract a rectangular region of the nsIImage and return it as a new
|
||||
* nsIImage.
|
||||
* @param aSubimage the region to extract
|
||||
* @param aResult the extracted image
|
||||
*/
|
||||
NS_IMETHOD Extract(const nsIntRect& aSubimage,
|
||||
nsIImage** aResult NS_OUTPARAM) = 0;
|
||||
};
|
||||
|
||||
NS_DEFINE_STATIC_IID_ACCESSOR(nsIImage, NS_IIMAGE_IID)
|
||||
|
||||
#endif
|
||||
|
|
@ -59,7 +59,7 @@ REQUIRES = xpcom \
|
|||
xpconnect \
|
||||
$(NULL)
|
||||
|
||||
DIRS = shared thebes
|
||||
DIRS = thebes
|
||||
|
||||
ifdef MOZ_ENABLE_POSTSCRIPT
|
||||
DIRS += psshared
|
||||
|
|
|
|||
|
|
@ -1,67 +0,0 @@
|
|||
#
|
||||
# ***** BEGIN LICENSE BLOCK *****
|
||||
# Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
||||
#
|
||||
# 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 the Initial Developer are Copyright (C) 2001
|
||||
# the Initial Developer. All Rights Reserved.
|
||||
#
|
||||
# Contributor(s):
|
||||
#
|
||||
# Alternatively, the contents of this file may be used under the terms of
|
||||
# either of 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 MPL, 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 MPL, the GPL or the LGPL.
|
||||
#
|
||||
# ***** END LICENSE BLOCK *****
|
||||
|
||||
DEPTH = ../../..
|
||||
topsrcdir = @top_srcdir@
|
||||
srcdir = @srcdir@
|
||||
VPATH = @srcdir@
|
||||
|
||||
include $(DEPTH)/config/autoconf.mk
|
||||
|
||||
MODULE = gfx
|
||||
LIBRARY_NAME = gfxshared_s
|
||||
FORCE_STATIC_LIB=1
|
||||
EXPORT_LIBRARY=1
|
||||
LIBXUL_LIBRARY = 1
|
||||
|
||||
REQUIRES= \
|
||||
xpcom \
|
||||
string \
|
||||
imglib2 \
|
||||
widget \
|
||||
thebes \
|
||||
$(NULL)
|
||||
|
||||
CPPSRCS = \
|
||||
gfxImageFrame.cpp \
|
||||
$(NULL)
|
||||
|
||||
LOCAL_INCLUDES = -I$(srcdir)/.. \
|
||||
$(NULL)
|
||||
|
||||
include $(topsrcdir)/config/rules.mk
|
||||
|
||||
|
|
@ -1,428 +0,0 @@
|
|||
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
||||
*
|
||||
* ***** BEGIN LICENSE BLOCK *****
|
||||
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
||||
*
|
||||
* 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 the Initial Developer are Copyright (C) 2001
|
||||
* the Initial Developer. All Rights Reserved.
|
||||
*
|
||||
* Contributor(s):
|
||||
* Stuart Parmenter <pavlov@netscape.com>
|
||||
*
|
||||
* Alternatively, the contents of this file may be used under the terms of
|
||||
* either of 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 MPL, 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 MPL, the GPL or the LGPL.
|
||||
*
|
||||
* ***** END LICENSE BLOCK ***** */
|
||||
|
||||
#include "gfxImageFrame.h"
|
||||
#include "nsIServiceManager.h"
|
||||
#include <limits.h>
|
||||
#include "prmem.h"
|
||||
|
||||
NS_IMPL_ISUPPORTS2(gfxImageFrame, gfxIImageFrame, nsIInterfaceRequestor)
|
||||
|
||||
gfxImageFrame::gfxImageFrame() :
|
||||
mImageData(nsnull),
|
||||
mTimeout(100),
|
||||
mDisposalMethod(0), /* imgIContainer::kDisposeNotSpecified */
|
||||
mBlendMethod(1), /* imgIContainer::kBlendOver */
|
||||
mInitialized(PR_FALSE),
|
||||
mMutable(PR_TRUE)
|
||||
{
|
||||
/* member initializers and constructor code */
|
||||
}
|
||||
|
||||
gfxImageFrame::~gfxImageFrame()
|
||||
{
|
||||
/* destructor code */
|
||||
PR_FREEIF(mImageData);
|
||||
mInitialized = PR_FALSE;
|
||||
}
|
||||
|
||||
/* void init (in PRInt32 aX, in PRInt32 aY, in PRInt32 aWidth, in PRInt32 aHeight, in gfx_format aFormat); */
|
||||
NS_IMETHODIMP gfxImageFrame::Init(PRInt32 aX, PRInt32 aY, PRInt32 aWidth, PRInt32 aHeight, gfx_format aFormat,gfx_depth aDepth)
|
||||
{
|
||||
if (mInitialized)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
// assert for properties that should be verified by decoders, warn for properties related to bad content
|
||||
|
||||
if (aWidth <= 0 || aHeight <= 0) {
|
||||
NS_ASSERTION(0, "error - negative image size\n");
|
||||
return NS_ERROR_FAILURE;
|
||||
}
|
||||
|
||||
/* check to make sure we don't overflow a 32-bit */
|
||||
PRInt32 tmp = aWidth * aHeight;
|
||||
if (tmp / aHeight != aWidth) {
|
||||
NS_WARNING("width or height too large");
|
||||
return NS_ERROR_FAILURE;
|
||||
}
|
||||
tmp = tmp * 4;
|
||||
if (tmp / 4 != aWidth * aHeight) {
|
||||
NS_WARNING("width or height too large");
|
||||
return NS_ERROR_FAILURE;
|
||||
}
|
||||
|
||||
/* reject over-wide or over-tall images */
|
||||
const PRInt32 k64KLimit = 0x0000FFFF;
|
||||
if ( aWidth > k64KLimit || aHeight > k64KLimit ){
|
||||
NS_WARNING("image too big");
|
||||
return NS_ERROR_FAILURE;
|
||||
}
|
||||
|
||||
#if defined(XP_MACOSX)
|
||||
// CoreGraphics is limited to images < 32K in *height*, so clamp all surfaces on the Mac to that height
|
||||
if (aHeight > SHRT_MAX) {
|
||||
NS_WARNING("image too big");
|
||||
return NS_ERROR_FAILURE;
|
||||
}
|
||||
#endif
|
||||
|
||||
mOffset.MoveTo(aX, aY);
|
||||
mSize.SizeTo(aWidth, aHeight);
|
||||
|
||||
mFormat = aFormat;
|
||||
mDepth = aDepth;
|
||||
|
||||
PRBool needImage = PR_TRUE;
|
||||
nsMaskRequirements maskReq;
|
||||
|
||||
switch (aFormat) {
|
||||
case gfxIFormats::BGR:
|
||||
case gfxIFormats::RGB:
|
||||
maskReq = nsMaskRequirements_kNoMask;
|
||||
break;
|
||||
|
||||
case gfxIFormats::BGRA:
|
||||
case gfxIFormats::RGBA:
|
||||
maskReq = nsMaskRequirements_kNeeds8Bit;
|
||||
break;
|
||||
|
||||
case gfxIFormats::BGR_A1:
|
||||
case gfxIFormats::RGB_A1:
|
||||
maskReq = nsMaskRequirements_kNeeds1Bit;
|
||||
break;
|
||||
|
||||
case gfxIFormats::BGR_A8:
|
||||
case gfxIFormats::RGB_A8:
|
||||
maskReq = nsMaskRequirements_kNeeds8Bit;
|
||||
break;
|
||||
|
||||
case gfxIFormats::PAL:
|
||||
case gfxIFormats::PAL_A1:
|
||||
needImage = PR_FALSE;
|
||||
break;
|
||||
|
||||
default:
|
||||
NS_ERROR("unsupported gfx_format\n");
|
||||
return NS_ERROR_FAILURE;
|
||||
}
|
||||
|
||||
if (needImage) {
|
||||
if (aDepth != 24) {
|
||||
NS_ERROR("This Depth is not supported");
|
||||
return NS_ERROR_FAILURE;
|
||||
}
|
||||
|
||||
nsresult rv;
|
||||
mImage = do_CreateInstance("@mozilla.org/gfx/image;1", &rv);
|
||||
NS_ASSERTION(mImage, "creation of image failed");
|
||||
if (NS_FAILED(rv)) return rv;
|
||||
|
||||
rv = mImage->Init(aWidth, aHeight, aDepth, maskReq);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
} else {
|
||||
if ((aDepth < 1) || (aDepth > 8)) {
|
||||
NS_ERROR("This Depth is not supported\n");
|
||||
return NS_ERROR_FAILURE;
|
||||
}
|
||||
mImageData = (PRUint8*)PR_MALLOC(PaletteDataLength() + ImageDataLength());
|
||||
NS_ENSURE_TRUE(mImageData, NS_ERROR_OUT_OF_MEMORY);
|
||||
}
|
||||
|
||||
mInitialized = PR_TRUE;
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
||||
/* attribute boolean mutable */
|
||||
NS_IMETHODIMP gfxImageFrame::GetMutable(PRBool *aMutable)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
NS_ASSERTION(mInitialized, "gfxImageFrame::GetMutable called on non-inited gfxImageFrame");
|
||||
*aMutable = mMutable;
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
NS_IMETHODIMP gfxImageFrame::SetMutable(PRBool aMutable)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
mMutable = aMutable;
|
||||
|
||||
if (!aMutable && mImage)
|
||||
mImage->Optimize(nsnull);
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/* readonly attribute PRInt32 x; */
|
||||
NS_IMETHODIMP gfxImageFrame::GetX(PRInt32 *aX)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
*aX = mOffset.x;
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/* readonly attribute PRInt32 y; */
|
||||
NS_IMETHODIMP gfxImageFrame::GetY(PRInt32 *aY)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
*aY = mOffset.y;
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
||||
/* readonly attribute PRInt32 width; */
|
||||
NS_IMETHODIMP gfxImageFrame::GetWidth(PRInt32 *aWidth)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
*aWidth = mSize.width;
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/* readonly attribute PRInt32 height; */
|
||||
NS_IMETHODIMP gfxImageFrame::GetHeight(PRInt32 *aHeight)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
*aHeight = mSize.height;
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/* void getRect(in nsRectRef rect); */
|
||||
NS_IMETHODIMP gfxImageFrame::GetRect(nsIntRect &aRect)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
aRect.SetRect(mOffset.x, mOffset.y, mSize.width, mSize.height);
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/* readonly attribute gfx_format format; */
|
||||
NS_IMETHODIMP gfxImageFrame::GetFormat(gfx_format *aFormat)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
*aFormat = mFormat;
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/* readonly attribute boolean needsBackground; */
|
||||
NS_IMETHODIMP gfxImageFrame::GetNeedsBackground(PRBool *aNeedsBackground)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
*aNeedsBackground = (mFormat != gfxIFormats::RGB &&
|
||||
mFormat != gfxIFormats::PAL &&
|
||||
mFormat != gfxIFormats::BGR) ||
|
||||
!mImage->GetIsImageComplete();
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
||||
/* readonly attribute unsigned long imageBytesPerRow; */
|
||||
NS_IMETHODIMP gfxImageFrame::GetImageBytesPerRow(PRUint32 *aBytesPerRow)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
*aBytesPerRow = mImage ? mImage->GetLineStride(): mSize.width;
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/* readonly attribute unsigned long imageDataLength; */
|
||||
NS_IMETHODIMP gfxImageFrame::GetImageDataLength(PRUint32 *aBitsLength)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
*aBitsLength = ImageDataLength();
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/* void getImageData([array, size_is(length)] out PRUint8 bits, out unsigned long length); */
|
||||
NS_IMETHODIMP gfxImageFrame::GetImageData(PRUint8 **aData, PRUint32 *length)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
NS_ASSERTION(mMutable, "trying to get data on an immutable frame");
|
||||
|
||||
*aData = mImage ? mImage->GetBits() : (mImageData + PaletteDataLength());
|
||||
*length = ImageDataLength();
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/* void getPaletteData ([array, size_is (length)] out PRUint32 palette, out unsigned long length); */
|
||||
NS_IMETHODIMP gfxImageFrame::GetPaletteData(gfx_color **aPalette, PRUint32 *length)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
if (!mImageData)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
*aPalette = (gfx_color*)mImageData;
|
||||
*length = PaletteDataLength();
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/* void lockImageData (); */
|
||||
NS_IMETHODIMP gfxImageFrame::LockImageData()
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
if (!mImage)
|
||||
return NS_OK;
|
||||
return mImage->LockImagePixels(PR_FALSE);
|
||||
}
|
||||
|
||||
/* void unlockImageData (); */
|
||||
NS_IMETHODIMP gfxImageFrame::UnlockImageData()
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
if (!mImage)
|
||||
return NS_OK;
|
||||
return mImage->UnlockImagePixels(PR_FALSE);
|
||||
}
|
||||
|
||||
/* attribute long timeout; */
|
||||
NS_IMETHODIMP gfxImageFrame::GetTimeout(PRInt32 *aTimeout)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
// Ensure a minimal time between updates so we don't throttle the UI thread.
|
||||
// consider 0 == unspecified and make it fast but not too fast. See bug
|
||||
// 125137, bug 139677, and bug 207059. The behavior of recent IE and Opera
|
||||
// versions seems to be:
|
||||
// IE 6/Win:
|
||||
// 10 - 50ms go 100ms
|
||||
// >50ms go correct speed
|
||||
// Opera 7 final/Win:
|
||||
// 10ms goes 100ms
|
||||
// >10ms go correct speed
|
||||
// It seems that there are broken tools out there that set a 0ms or 10ms
|
||||
// timeout when they really want a "default" one. So munge values in that
|
||||
// range.
|
||||
if (mTimeout >= 0 && mTimeout <= 10)
|
||||
*aTimeout = 100;
|
||||
else
|
||||
*aTimeout = mTimeout;
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
NS_IMETHODIMP gfxImageFrame::SetTimeout(PRInt32 aTimeout)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
mTimeout = aTimeout;
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/* attribute long frameDisposalMethod; */
|
||||
NS_IMETHODIMP gfxImageFrame::GetFrameDisposalMethod(PRInt32 *aFrameDisposalMethod)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
*aFrameDisposalMethod = mDisposalMethod;
|
||||
return NS_OK;
|
||||
}
|
||||
NS_IMETHODIMP gfxImageFrame::SetFrameDisposalMethod(PRInt32 aFrameDisposalMethod)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
mDisposalMethod = aFrameDisposalMethod;
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/* attribute long blendMethod; */
|
||||
NS_IMETHODIMP gfxImageFrame::GetBlendMethod(PRInt32 *aBlendMethod)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
*aBlendMethod = mBlendMethod;
|
||||
return NS_OK;
|
||||
}
|
||||
NS_IMETHODIMP gfxImageFrame::SetBlendMethod(PRInt32 aBlendMethod)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
mBlendMethod = (PRInt8)aBlendMethod;
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
NS_IMETHODIMP gfxImageFrame::GetInterface(const nsIID & aIID, void * *result)
|
||||
{
|
||||
if (!mInitialized)
|
||||
return NS_ERROR_NOT_INITIALIZED;
|
||||
|
||||
NS_ENSURE_ARG_POINTER(result);
|
||||
|
||||
if (NS_SUCCEEDED(QueryInterface(aIID, result)))
|
||||
return NS_OK;
|
||||
if (mImage && aIID.Equals(NS_GET_IID(nsIImage)))
|
||||
return mImage->QueryInterface(aIID, result);
|
||||
|
||||
return NS_NOINTERFACE;
|
||||
}
|
||||
|
|
@ -1,94 +0,0 @@
|
|||
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
||||
*
|
||||
* ***** BEGIN LICENSE BLOCK *****
|
||||
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
||||
*
|
||||
* 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 the Initial Developer are Copyright (C) 2000-2001
|
||||
* the Initial Developer. All Rights Reserved.
|
||||
*
|
||||
* Contributor(s):
|
||||
* Stuart Parmenter <pavlov@netscape.com>
|
||||
*
|
||||
* Alternatively, the contents of this file may be used under the terms of
|
||||
* either of 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 MPL, 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 MPL, the GPL or the LGPL.
|
||||
*
|
||||
* ***** END LICENSE BLOCK ***** */
|
||||
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsIInterfaceRequestor.h"
|
||||
#include "nsIInterfaceRequestorUtils.h"
|
||||
|
||||
#include "nsIImage.h"
|
||||
|
||||
#include "nsPoint.h"
|
||||
#include "nsSize.h"
|
||||
|
||||
#include "nsCOMPtr.h"
|
||||
|
||||
#define GFX_IMAGEFRAME_CID \
|
||||
{ /* aa699204-1dd1-11b2-84a9-a280c268e4fb */ \
|
||||
0xaa699204, \
|
||||
0x1dd1, \
|
||||
0x11b2, \
|
||||
{0x84, 0xa9, 0xa2, 0x80, 0xc2, 0x68, 0xe4, 0xfb} \
|
||||
}
|
||||
|
||||
class gfxImageFrame : public gfxIImageFrame,
|
||||
public nsIInterfaceRequestor
|
||||
{
|
||||
public:
|
||||
NS_DECL_ISUPPORTS
|
||||
NS_DECL_GFXIIMAGEFRAME
|
||||
NS_DECL_NSIINTERFACEREQUESTOR
|
||||
|
||||
gfxImageFrame();
|
||||
virtual ~gfxImageFrame();
|
||||
|
||||
protected:
|
||||
nsIntSize mSize;
|
||||
|
||||
private:
|
||||
PRUint32 PaletteDataLength() const {
|
||||
return ((1 << mDepth) * sizeof(gfx_color));
|
||||
}
|
||||
|
||||
PRUint32 ImageDataLength() const {
|
||||
return (mImage ? mImage->GetLineStride() : mSize.width) * mSize.height;
|
||||
}
|
||||
|
||||
nsCOMPtr<nsIImage> mImage;
|
||||
PRUint8* mImageData;
|
||||
|
||||
PRInt32 mTimeout; // -1 means display forever
|
||||
nsIntPoint mOffset;
|
||||
PRInt32 mDisposalMethod;
|
||||
|
||||
gfx_format mFormat;
|
||||
gfx_depth mDepth;
|
||||
PRInt8 mBlendMethod;
|
||||
PRPackedBool mInitialized;
|
||||
PRPackedBool mMutable;
|
||||
};
|
||||
|
|
@ -72,7 +72,6 @@ endif
|
|||
|
||||
CPPSRCS = \
|
||||
nsThebesDeviceContext.cpp \
|
||||
nsThebesImage.cpp \
|
||||
nsThebesRegion.cpp \
|
||||
nsThebesGfxFactory.cpp \
|
||||
nsThebesRenderingContext.cpp \
|
||||
|
|
@ -128,7 +127,6 @@ LOCAL_INCLUDES = \
|
|||
$(NULL)
|
||||
|
||||
EXTRA_DSO_LDOPTS += \
|
||||
../shared/$(LIB_PREFIX)gfxshared_s.$(LIB_SUFFIX) \
|
||||
$(LIBS_DIR) \
|
||||
$(EXTRA_DSO_LIBS) \
|
||||
$(DEPTH)/modules/libutil/src/$(LIB_PREFIX)mozutil_s.$(LIB_SUFFIX) \
|
||||
|
|
|
|||
|
|
@ -42,21 +42,17 @@
|
|||
#include "nsGfxCIID.h"
|
||||
|
||||
#include "nsScriptableRegion.h"
|
||||
#include "gfxImageFrame.h"
|
||||
|
||||
#include "nsThebesDeviceContext.h"
|
||||
#include "nsThebesRenderingContext.h"
|
||||
#include "nsThebesImage.h"
|
||||
#include "nsThebesRegion.h"
|
||||
#include "nsThebesFontMetrics.h"
|
||||
#include "nsThebesFontEnumerator.h"
|
||||
#include "gfxPlatform.h"
|
||||
|
||||
NS_GENERIC_FACTORY_CONSTRUCTOR(nsThebesFontMetrics)
|
||||
NS_GENERIC_FACTORY_CONSTRUCTOR(gfxImageFrame)
|
||||
NS_GENERIC_FACTORY_CONSTRUCTOR(nsThebesDeviceContext)
|
||||
NS_GENERIC_FACTORY_CONSTRUCTOR(nsThebesRenderingContext)
|
||||
NS_GENERIC_FACTORY_CONSTRUCTOR(nsThebesImage)
|
||||
NS_GENERIC_FACTORY_CONSTRUCTOR(nsThebesRegion)
|
||||
NS_GENERIC_FACTORY_CONSTRUCTOR(nsThebesFontEnumerator)
|
||||
|
||||
|
|
@ -119,10 +115,6 @@ static const nsModuleComponentInfo components[] =
|
|||
NS_RENDERING_CONTEXT_CID,
|
||||
"@mozilla.org/gfx/renderingcontext;1",
|
||||
nsThebesRenderingContextConstructor },
|
||||
{ "Thebes nsImage",
|
||||
NS_IMAGE_CID,
|
||||
"@mozilla.org/gfx/image;1",
|
||||
nsThebesImageConstructor },
|
||||
{ "Thebes Region",
|
||||
NS_REGION_CID,
|
||||
"@mozilla.org/gfx/region/nsThebes;1",
|
||||
|
|
@ -131,10 +123,6 @@ static const nsModuleComponentInfo components[] =
|
|||
NS_SCRIPTABLE_REGION_CID,
|
||||
"@mozilla.org/gfx/region;1",
|
||||
nsScriptableRegionConstructor },
|
||||
{ "image frame",
|
||||
GFX_IMAGEFRAME_CID,
|
||||
"@mozilla.org/gfx/image/frame;2",
|
||||
gfxImageFrameConstructor },
|
||||
};
|
||||
|
||||
static nsresult
|
||||
|
|
|
|||
|
|
@ -1,857 +0,0 @@
|
|||
/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
||||
/* ***** BEGIN LICENSE BLOCK *****
|
||||
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
||||
*
|
||||
* 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 thebes gfx
|
||||
*
|
||||
* The Initial Developer of the Original Code is
|
||||
* mozilla.org.
|
||||
* Portions created by the Initial Developer are Copyright (C) 2005
|
||||
* the Initial Developer. All Rights Reserved.
|
||||
*
|
||||
* Contributor(s):
|
||||
* Vladimir Vukicevic <vladimir@pobox.com>
|
||||
*
|
||||
* 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 MPL, 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 MPL, the GPL or the LGPL.
|
||||
*
|
||||
* ***** END LICENSE BLOCK ***** */
|
||||
|
||||
#include "nsThebesImage.h"
|
||||
#include "nsThebesRenderingContext.h"
|
||||
|
||||
#include "gfxContext.h"
|
||||
#include "gfxPattern.h"
|
||||
|
||||
#include "gfxPlatform.h"
|
||||
|
||||
#include "prenv.h"
|
||||
|
||||
static PRBool gDisableOptimize = PR_FALSE;
|
||||
|
||||
/*XXX get CAIRO_HAS_DDRAW_SURFACE */
|
||||
#include "cairo.h"
|
||||
|
||||
#ifdef CAIRO_HAS_DDRAW_SURFACE
|
||||
#include "gfxDDrawSurface.h"
|
||||
#endif
|
||||
|
||||
#if defined(XP_WIN) || defined(WINCE)
|
||||
#include "gfxWindowsPlatform.h"
|
||||
#endif
|
||||
|
||||
#if defined(XP_WIN) && !defined(WINCE)
|
||||
|
||||
/* Whether to use the windows surface; only for desktop win32 */
|
||||
#define USE_WIN_SURFACE 1
|
||||
|
||||
static PRUint32 gTotalDDBs = 0;
|
||||
static PRUint32 gTotalDDBSize = 0;
|
||||
// only use up a maximum of 64MB in DDBs
|
||||
#define kMaxDDBSize (64*1024*1024)
|
||||
// and don't let anything in that's bigger than 4MB
|
||||
#define kMaxSingleDDBSize (4*1024*1024)
|
||||
|
||||
#endif
|
||||
|
||||
NS_IMPL_ISUPPORTS1(nsThebesImage, nsIImage)
|
||||
|
||||
nsThebesImage::nsThebesImage()
|
||||
: mFormat(gfxImageSurface::ImageFormatRGB24),
|
||||
mWidth(0),
|
||||
mHeight(0),
|
||||
mDecoded(0,0,0,0),
|
||||
mImageComplete(PR_FALSE),
|
||||
mSinglePixel(PR_FALSE),
|
||||
mFormatChanged(PR_FALSE),
|
||||
mNeverUseDeviceSurface(PR_FALSE),
|
||||
mSinglePixelColor(0),
|
||||
mAlphaDepth(0)
|
||||
{
|
||||
static PRBool hasCheckedOptimize = PR_FALSE;
|
||||
if (!hasCheckedOptimize) {
|
||||
if (PR_GetEnv("MOZ_DISABLE_IMAGE_OPTIMIZE")) {
|
||||
gDisableOptimize = PR_TRUE;
|
||||
}
|
||||
hasCheckedOptimize = PR_TRUE;
|
||||
}
|
||||
|
||||
#ifdef USE_WIN_SURFACE
|
||||
mIsDDBSurface = PR_FALSE;
|
||||
#endif
|
||||
}
|
||||
|
||||
nsresult
|
||||
nsThebesImage::Init(PRInt32 aWidth, PRInt32 aHeight, PRInt32 aDepth, nsMaskRequirements aMaskRequirements)
|
||||
{
|
||||
mWidth = aWidth;
|
||||
mHeight = aHeight;
|
||||
|
||||
// Reject over-wide or over-tall images.
|
||||
if (!AllowedImageSize(aWidth, aHeight))
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
// Check to see if we are running OOM
|
||||
nsCOMPtr<nsIMemory> mem;
|
||||
NS_GetMemoryManager(getter_AddRefs(mem));
|
||||
if (!mem)
|
||||
return NS_ERROR_UNEXPECTED;
|
||||
|
||||
PRBool lowMemory;
|
||||
mem->IsLowMemory(&lowMemory);
|
||||
if (lowMemory)
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
|
||||
gfxImageSurface::gfxImageFormat format;
|
||||
switch(aMaskRequirements)
|
||||
{
|
||||
case nsMaskRequirements_kNeeds1Bit:
|
||||
format = gfxImageSurface::ImageFormatARGB32;
|
||||
mAlphaDepth = 1;
|
||||
break;
|
||||
case nsMaskRequirements_kNeeds8Bit:
|
||||
format = gfxImageSurface::ImageFormatARGB32;
|
||||
mAlphaDepth = 8;
|
||||
break;
|
||||
default:
|
||||
format = gfxImageSurface::ImageFormatRGB24;
|
||||
mAlphaDepth = 0;
|
||||
break;
|
||||
}
|
||||
|
||||
mFormat = format;
|
||||
|
||||
// For Windows, we must create the device surface first (if we're
|
||||
// going to) so that the image surface can wrap it. Can't be done
|
||||
// the other way around.
|
||||
#ifdef USE_WIN_SURFACE
|
||||
if (!mNeverUseDeviceSurface && !ShouldUseImageSurfaces()) {
|
||||
mWinSurface = new gfxWindowsSurface(gfxIntSize(mWidth, mHeight), format);
|
||||
if (mWinSurface && mWinSurface->CairoStatus() == 0) {
|
||||
// no error
|
||||
mImageSurface = mWinSurface->GetImageSurface();
|
||||
} else {
|
||||
mWinSurface = nsnull;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
// For other platforms we create the image surface first and then
|
||||
// possibly wrap it in a device surface. This branch is also used
|
||||
// on Windows if we're not using device surfaces or if we couldn't
|
||||
// create one.
|
||||
if (!mImageSurface)
|
||||
mImageSurface = new gfxImageSurface(gfxIntSize(mWidth, mHeight), format);
|
||||
|
||||
if (!mImageSurface || mImageSurface->CairoStatus()) {
|
||||
mImageSurface = nsnull;
|
||||
// guess
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
}
|
||||
|
||||
#ifdef XP_MACOSX
|
||||
if (!mNeverUseDeviceSurface && !ShouldUseImageSurfaces()) {
|
||||
mQuartzSurface = new gfxQuartzImageSurface(mImageSurface);
|
||||
}
|
||||
#endif
|
||||
|
||||
mStride = mImageSurface->Stride();
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
nsThebesImage::~nsThebesImage()
|
||||
{
|
||||
#ifdef USE_WIN_SURFACE
|
||||
if (mIsDDBSurface) {
|
||||
gTotalDDBs--;
|
||||
gTotalDDBSize -= mWidth*mHeight*4;
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
PRInt32
|
||||
nsThebesImage::GetBytesPix()
|
||||
{
|
||||
return 4;
|
||||
}
|
||||
|
||||
PRBool
|
||||
nsThebesImage::GetIsRowOrderTopToBottom()
|
||||
{
|
||||
return PR_TRUE;
|
||||
}
|
||||
|
||||
PRInt32
|
||||
nsThebesImage::GetWidth()
|
||||
{
|
||||
return mWidth;
|
||||
}
|
||||
|
||||
PRInt32
|
||||
nsThebesImage::GetHeight()
|
||||
{
|
||||
return mHeight;
|
||||
}
|
||||
|
||||
PRUint8 *
|
||||
nsThebesImage::GetBits()
|
||||
{
|
||||
if (mImageSurface)
|
||||
return mImageSurface->Data();
|
||||
return nsnull;
|
||||
}
|
||||
|
||||
PRInt32
|
||||
nsThebesImage::GetLineStride()
|
||||
{
|
||||
return mStride;
|
||||
}
|
||||
|
||||
PRBool
|
||||
nsThebesImage::GetHasAlphaMask()
|
||||
{
|
||||
return mAlphaDepth > 0;
|
||||
}
|
||||
|
||||
PRUint8 *
|
||||
nsThebesImage::GetAlphaBits()
|
||||
{
|
||||
return nsnull;
|
||||
}
|
||||
|
||||
PRInt32
|
||||
nsThebesImage::GetAlphaLineStride()
|
||||
{
|
||||
return (mAlphaDepth > 0) ? mStride : 0;
|
||||
}
|
||||
|
||||
nsresult
|
||||
nsThebesImage::ImageUpdated(nsIDeviceContext *aContext, PRUint8 aFlags, nsIntRect *aUpdateRect)
|
||||
{
|
||||
// Check to see if we are running OOM
|
||||
nsCOMPtr<nsIMemory> mem;
|
||||
NS_GetMemoryManager(getter_AddRefs(mem));
|
||||
if (!mem)
|
||||
return NS_ERROR_UNEXPECTED;
|
||||
|
||||
PRBool lowMemory;
|
||||
mem->IsLowMemory(&lowMemory);
|
||||
if (lowMemory)
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
|
||||
mDecoded.UnionRect(mDecoded, *aUpdateRect);
|
||||
|
||||
// clamp to bounds, in case someone sends a bogus
|
||||
// updateRect (I'm looking at you, gif decoder)
|
||||
nsIntRect boundsRect(0, 0, mWidth, mHeight);
|
||||
mDecoded.IntersectRect(mDecoded, boundsRect);
|
||||
|
||||
#ifdef XP_MACOSX
|
||||
if (mQuartzSurface)
|
||||
mQuartzSurface->Flush();
|
||||
#endif
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
PRBool
|
||||
nsThebesImage::GetIsImageComplete()
|
||||
{
|
||||
if (!mImageComplete)
|
||||
mImageComplete = (mDecoded == nsIntRect(0, 0, mWidth, mHeight));
|
||||
return mImageComplete;
|
||||
}
|
||||
|
||||
nsresult
|
||||
nsThebesImage::Optimize(nsIDeviceContext* aContext)
|
||||
{
|
||||
if (gDisableOptimize)
|
||||
return NS_OK;
|
||||
|
||||
if (mOptSurface || mSinglePixel)
|
||||
return NS_OK;
|
||||
|
||||
/* Figure out if the entire image is a constant color */
|
||||
|
||||
// this should always be true
|
||||
if (mStride == mWidth * 4) {
|
||||
PRUint32 *imgData = (PRUint32*) mImageSurface->Data();
|
||||
PRUint32 firstPixel = * (PRUint32*) imgData;
|
||||
PRUint32 pixelCount = mWidth * mHeight + 1;
|
||||
|
||||
while (--pixelCount && *imgData++ == firstPixel)
|
||||
;
|
||||
|
||||
if (pixelCount == 0) {
|
||||
// all pixels were the same
|
||||
if (mFormat == gfxImageSurface::ImageFormatARGB32 ||
|
||||
mFormat == gfxImageSurface::ImageFormatRGB24)
|
||||
{
|
||||
mSinglePixelColor = gfxRGBA
|
||||
(firstPixel,
|
||||
(mFormat == gfxImageSurface::ImageFormatRGB24 ?
|
||||
gfxRGBA::PACKED_XRGB :
|
||||
gfxRGBA::PACKED_ARGB_PREMULTIPLIED));
|
||||
|
||||
mSinglePixel = PR_TRUE;
|
||||
|
||||
// blow away the older surfaces, to release data
|
||||
|
||||
mImageSurface = nsnull;
|
||||
mOptSurface = nsnull;
|
||||
#ifdef USE_WIN_SURFACE
|
||||
mWinSurface = nsnull;
|
||||
#endif
|
||||
#ifdef XP_MACOSX
|
||||
mQuartzSurface = nsnull;
|
||||
#endif
|
||||
return NS_OK;
|
||||
}
|
||||
}
|
||||
|
||||
// if it's not RGB24/ARGB32, don't optimize, but we never hit this at the moment
|
||||
}
|
||||
|
||||
// if we're being forced to use image surfaces due to
|
||||
// resource constraints, don't try to optimize beyond same-pixel.
|
||||
if (mNeverUseDeviceSurface || ShouldUseImageSurfaces())
|
||||
return NS_OK;
|
||||
|
||||
mOptSurface = nsnull;
|
||||
|
||||
#ifdef USE_WIN_SURFACE
|
||||
// we need to special-case windows here, because windows has
|
||||
// a distinction between DIB and DDB and we want to use DDBs as much
|
||||
// as we can.
|
||||
if (mWinSurface) {
|
||||
// Don't do DDBs for large images; see bug 359147
|
||||
// Note that we bother with DDBs at all because they are much faster
|
||||
// on some systems; on others there isn't much of a speed difference
|
||||
// between DIBs and DDBs.
|
||||
//
|
||||
// Originally this just limited to 1024x1024; but that still
|
||||
// had us hitting overall total memory usage limits (which was
|
||||
// around 220MB on my intel shared memory system with 2GB RAM
|
||||
// and 16-128mb in use by the video card, so I can't make
|
||||
// heads or tails out of this limit).
|
||||
//
|
||||
// So instead, we clamp the max size to 64MB (this limit shuld
|
||||
// be made dynamic based on.. something.. as soon a we figure
|
||||
// out that something) and also limit each individual image to
|
||||
// be less than 4MB to keep very large images out of DDBs.
|
||||
|
||||
// assume (almost -- we don't quadword-align) worst-case size
|
||||
PRUint32 ddbSize = mWidth * mHeight * 4;
|
||||
if (ddbSize <= kMaxSingleDDBSize &&
|
||||
ddbSize + gTotalDDBSize <= kMaxDDBSize)
|
||||
{
|
||||
nsRefPtr<gfxWindowsSurface> wsurf = mWinSurface->OptimizeToDDB(nsnull, gfxIntSize(mWidth, mHeight), mFormat);
|
||||
if (wsurf) {
|
||||
gTotalDDBs++;
|
||||
gTotalDDBSize += ddbSize;
|
||||
mIsDDBSurface = PR_TRUE;
|
||||
mOptSurface = wsurf;
|
||||
}
|
||||
}
|
||||
if (!mOptSurface && !mFormatChanged) {
|
||||
// just use the DIB if the format has not changed
|
||||
mOptSurface = mWinSurface;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef XP_MACOSX
|
||||
if (mQuartzSurface) {
|
||||
mQuartzSurface->Flush();
|
||||
mOptSurface = mQuartzSurface;
|
||||
}
|
||||
#endif
|
||||
|
||||
if (mOptSurface == nsnull)
|
||||
mOptSurface = gfxPlatform::GetPlatform()->OptimizeImage(mImageSurface, mFormat);
|
||||
|
||||
if (mOptSurface) {
|
||||
mImageSurface = nsnull;
|
||||
#ifdef USE_WIN_SURFACE
|
||||
mWinSurface = nsnull;
|
||||
#endif
|
||||
#ifdef XP_MACOSX
|
||||
mQuartzSurface = nsnull;
|
||||
#endif
|
||||
}
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
nsColorMap *
|
||||
nsThebesImage::GetColorMap()
|
||||
{
|
||||
return NULL;
|
||||
}
|
||||
|
||||
PRInt8
|
||||
nsThebesImage::GetAlphaDepth()
|
||||
{
|
||||
return mAlphaDepth;
|
||||
}
|
||||
|
||||
void *
|
||||
nsThebesImage::GetBitInfo()
|
||||
{
|
||||
return NULL;
|
||||
}
|
||||
|
||||
NS_IMETHODIMP
|
||||
nsThebesImage::LockImagePixels(PRBool aMaskPixels)
|
||||
{
|
||||
if (aMaskPixels)
|
||||
return NS_ERROR_NOT_IMPLEMENTED;
|
||||
if ((mOptSurface || mSinglePixel) && !mImageSurface) {
|
||||
// Recover the pixels
|
||||
mImageSurface = new gfxImageSurface(gfxIntSize(mWidth, mHeight),
|
||||
gfxImageSurface::ImageFormatARGB32);
|
||||
if (!mImageSurface || mImageSurface->CairoStatus())
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
gfxContext context(mImageSurface);
|
||||
context.SetOperator(gfxContext::OPERATOR_SOURCE);
|
||||
if (mSinglePixel)
|
||||
context.SetDeviceColor(mSinglePixelColor);
|
||||
else
|
||||
context.SetSource(mOptSurface);
|
||||
context.Paint();
|
||||
|
||||
#ifdef USE_WIN_SURFACE
|
||||
mWinSurface = nsnull;
|
||||
#endif
|
||||
#ifdef XP_MACOSX
|
||||
mQuartzSurface = nsnull;
|
||||
#endif
|
||||
}
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
NS_IMETHODIMP
|
||||
nsThebesImage::UnlockImagePixels(PRBool aMaskPixels)
|
||||
{
|
||||
if (aMaskPixels)
|
||||
return NS_ERROR_NOT_IMPLEMENTED;
|
||||
mOptSurface = nsnull;
|
||||
#ifdef XP_MACOSX
|
||||
if (mQuartzSurface)
|
||||
mQuartzSurface->Flush();
|
||||
#endif
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
static PRBool
|
||||
IsSafeImageTransformComponent(gfxFloat aValue)
|
||||
{
|
||||
return aValue >= -32768 && aValue <= 32767;
|
||||
}
|
||||
|
||||
void
|
||||
nsThebesImage::Draw(gfxContext* aContext,
|
||||
gfxPattern::GraphicsFilter aFilter,
|
||||
const gfxMatrix& aUserSpaceToImageSpace,
|
||||
const gfxRect& aFill,
|
||||
const nsIntMargin& aPadding,
|
||||
const nsIntRect& aSubimage)
|
||||
{
|
||||
NS_ASSERTION(!aFill.IsEmpty(), "zero dest size --- fix caller");
|
||||
NS_ASSERTION(!aSubimage.IsEmpty(), "zero source size --- fix caller");
|
||||
|
||||
PRBool doPadding = aPadding != nsIntMargin(0,0,0,0);
|
||||
PRBool doPartialDecode = !GetIsImageComplete();
|
||||
gfxContext::GraphicsOperator op = aContext->CurrentOperator();
|
||||
|
||||
if (mSinglePixel && !doPadding && !doPartialDecode) {
|
||||
// Single-color fast path
|
||||
// if a == 0, it's a noop
|
||||
if (mSinglePixelColor.a == 0.0)
|
||||
return;
|
||||
|
||||
if (op == gfxContext::OPERATOR_OVER && mSinglePixelColor.a == 1.0)
|
||||
aContext->SetOperator(gfxContext::OPERATOR_SOURCE);
|
||||
|
||||
aContext->SetDeviceColor(mSinglePixelColor);
|
||||
aContext->NewPath();
|
||||
aContext->Rectangle(aFill);
|
||||
aContext->Fill();
|
||||
aContext->SetOperator(op);
|
||||
aContext->SetDeviceColor(gfxRGBA(0,0,0,0));
|
||||
return;
|
||||
}
|
||||
|
||||
gfxMatrix userSpaceToImageSpace = aUserSpaceToImageSpace;
|
||||
gfxRect sourceRect = userSpaceToImageSpace.Transform(aFill);
|
||||
gfxRect imageRect(0, 0, mWidth + aPadding.LeftRight(), mHeight + aPadding.TopBottom());
|
||||
gfxRect subimage(aSubimage.x, aSubimage.y, aSubimage.width, aSubimage.height);
|
||||
gfxRect fill = aFill;
|
||||
nsRefPtr<gfxASurface> surface;
|
||||
gfxImageSurface::gfxImageFormat format;
|
||||
|
||||
NS_ASSERTION(!sourceRect.Intersect(subimage).IsEmpty(),
|
||||
"We must be allowed to sample *some* source pixels!");
|
||||
|
||||
PRBool doTile = !imageRect.Contains(sourceRect);
|
||||
if (doPadding || doPartialDecode) {
|
||||
gfxRect available = gfxRect(mDecoded.x, mDecoded.y, mDecoded.width, mDecoded.height) +
|
||||
gfxPoint(aPadding.left, aPadding.top);
|
||||
|
||||
if (!doTile && !mSinglePixel) {
|
||||
// Not tiling, and we have a surface, so we can account for
|
||||
// padding and/or a partial decode just by twiddling parameters.
|
||||
// First, update our user-space fill rect.
|
||||
sourceRect = sourceRect.Intersect(available);
|
||||
gfxMatrix imageSpaceToUserSpace = userSpaceToImageSpace;
|
||||
imageSpaceToUserSpace.Invert();
|
||||
fill = imageSpaceToUserSpace.Transform(sourceRect);
|
||||
|
||||
surface = ThebesSurface();
|
||||
format = mFormat;
|
||||
subimage = subimage.Intersect(available) - gfxPoint(aPadding.left, aPadding.top);
|
||||
userSpaceToImageSpace.Multiply(
|
||||
gfxMatrix().Translate(-gfxPoint(aPadding.left, aPadding.top)));
|
||||
sourceRect = sourceRect - gfxPoint(aPadding.left, aPadding.top);
|
||||
imageRect = gfxRect(0, 0, mWidth, mHeight);
|
||||
} else {
|
||||
// Create a temporary surface
|
||||
gfxIntSize size(PRInt32(imageRect.Width()),
|
||||
PRInt32(imageRect.Height()));
|
||||
// Give this surface an alpha channel because there are
|
||||
// transparent pixels in the padding or undecoded area
|
||||
format = gfxASurface::ImageFormatARGB32;
|
||||
surface = gfxPlatform::GetPlatform()->CreateOffscreenSurface(size,
|
||||
format);
|
||||
if (!surface || surface->CairoStatus() != 0)
|
||||
return;
|
||||
|
||||
// Fill 'available' with whatever we've got
|
||||
gfxContext tmpCtx(surface);
|
||||
tmpCtx.SetOperator(gfxContext::OPERATOR_SOURCE);
|
||||
if (mSinglePixel) {
|
||||
tmpCtx.SetDeviceColor(mSinglePixelColor);
|
||||
} else {
|
||||
tmpCtx.SetSource(ThebesSurface(), gfxPoint(aPadding.left, aPadding.top));
|
||||
}
|
||||
tmpCtx.Rectangle(available);
|
||||
tmpCtx.Fill();
|
||||
}
|
||||
} else {
|
||||
NS_ASSERTION(!mSinglePixel, "This should already have been handled");
|
||||
surface = ThebesSurface();
|
||||
format = mFormat;
|
||||
}
|
||||
// At this point, we've taken care of mSinglePixel images, images with
|
||||
// aPadding, and partially-decoded images.
|
||||
|
||||
if (!AllowedImageSize(fill.size.width + 1, fill.size.height + 1)) {
|
||||
NS_WARNING("Destination area too large, bailing out");
|
||||
return;
|
||||
}
|
||||
// Compute device-space-to-image-space transform. We need to sanity-
|
||||
// check it to work around a pixman bug :-(
|
||||
// XXX should we only do this for certain surface types?
|
||||
gfxFloat deviceX, deviceY;
|
||||
nsRefPtr<gfxASurface> currentTarget =
|
||||
aContext->CurrentSurface(&deviceX, &deviceY);
|
||||
gfxMatrix currentMatrix = aContext->CurrentMatrix();
|
||||
gfxMatrix deviceToUser = currentMatrix;
|
||||
deviceToUser.Invert();
|
||||
deviceToUser.Translate(-gfxPoint(-deviceX, -deviceY));
|
||||
gfxMatrix deviceToImage = deviceToUser;
|
||||
deviceToImage.Multiply(userSpaceToImageSpace);
|
||||
|
||||
PRBool pushedGroup = PR_FALSE;
|
||||
if (currentTarget->GetType() != gfxASurface::SurfaceTypeQuartz) {
|
||||
// BEGIN working around cairo/pixman bug (bug 364968)
|
||||
// Quartz's limits for matrix are much larger than pixman
|
||||
|
||||
// Our device-space-to-image-space transform may not be acceptable to pixman.
|
||||
if (!IsSafeImageTransformComponent(deviceToImage.xx) ||
|
||||
!IsSafeImageTransformComponent(deviceToImage.xy) ||
|
||||
!IsSafeImageTransformComponent(deviceToImage.yx) ||
|
||||
!IsSafeImageTransformComponent(deviceToImage.yy)) {
|
||||
NS_WARNING("Scaling up too much, bailing out");
|
||||
return;
|
||||
}
|
||||
|
||||
if (!IsSafeImageTransformComponent(deviceToImage.x0) ||
|
||||
!IsSafeImageTransformComponent(deviceToImage.y0)) {
|
||||
// We'll push a group, which will hopefully reduce our transform's
|
||||
// translation so it's in bounds
|
||||
aContext->Save();
|
||||
|
||||
// Clip the rounded-out-to-device-pixels bounds of the
|
||||
// transformed fill area. This is the area for the group we
|
||||
// want to push.
|
||||
aContext->IdentityMatrix();
|
||||
gfxRect bounds = currentMatrix.TransformBounds(fill);
|
||||
bounds.RoundOut();
|
||||
aContext->Clip(bounds);
|
||||
aContext->SetMatrix(currentMatrix);
|
||||
|
||||
aContext->PushGroup(gfxASurface::CONTENT_COLOR_ALPHA);
|
||||
aContext->SetOperator(gfxContext::OPERATOR_OVER);
|
||||
pushedGroup = PR_TRUE;
|
||||
}
|
||||
// END working around cairo/pixman bug (bug 364968)
|
||||
}
|
||||
|
||||
nsRefPtr<gfxPattern> pattern = new gfxPattern(surface);
|
||||
pattern->SetMatrix(userSpaceToImageSpace);
|
||||
|
||||
// OK now, the hard part left is to account for the subimage sampling
|
||||
// restriction. If all the transforms involved are just integer
|
||||
// translations, then we assume no resampling will occur so there's
|
||||
// nothing to do.
|
||||
// XXX if only we had source-clipping in cairo!
|
||||
if (!currentMatrix.HasNonIntegerTranslation() &&
|
||||
!userSpaceToImageSpace.HasNonIntegerTranslation()) {
|
||||
if (doTile) {
|
||||
pattern->SetExtend(gfxPattern::EXTEND_REPEAT);
|
||||
}
|
||||
} else {
|
||||
if (doTile || !subimage.Contains(imageRect)) {
|
||||
// EXTEND_PAD won't help us here; we have to create a temporary
|
||||
// surface to hold the subimage of pixels we're allowed to
|
||||
// sample
|
||||
|
||||
gfxRect userSpaceClipExtents = aContext->GetClipExtents();
|
||||
// This isn't optimal --- if aContext has a rotation then
|
||||
// GetClipExtents will have to do a bounding-box computation,
|
||||
// and TransformBounds might too, so we could get a better
|
||||
// result if we computed image space clip extents in one go
|
||||
// --- but it doesn't really matter and this is easier to
|
||||
// understand.
|
||||
gfxRect imageSpaceClipExtents =
|
||||
userSpaceToImageSpace.TransformBounds(userSpaceClipExtents);
|
||||
// Inflate by one pixel because bilinear filtering will sample
|
||||
// at most one pixel beyond the computed image pixel coordinate.
|
||||
imageSpaceClipExtents.Outset(1.0);
|
||||
|
||||
gfxRect needed =
|
||||
imageSpaceClipExtents.Intersect(sourceRect).Intersect(subimage);
|
||||
needed.RoundOut();
|
||||
// if 'needed' is empty, nothing will be drawn since aFill
|
||||
// must be entirely outside the clip region, so it doesn't
|
||||
// matter what we do here, but we should avoid trying to
|
||||
// create a zero-size surface.
|
||||
if (!needed.IsEmpty()) {
|
||||
gfxIntSize size(PRInt32(needed.Width()), PRInt32(needed.Height()));
|
||||
nsRefPtr<gfxASurface> temp =
|
||||
gfxPlatform::GetPlatform()->CreateOffscreenSurface(size, format);
|
||||
if (temp && temp->CairoStatus() == 0) {
|
||||
gfxContext tmpCtx(temp);
|
||||
tmpCtx.SetOperator(gfxContext::OPERATOR_SOURCE);
|
||||
nsRefPtr<gfxPattern> tmpPattern = new gfxPattern(surface);
|
||||
if (tmpPattern) {
|
||||
tmpPattern->SetExtend(gfxPattern::EXTEND_REPEAT);
|
||||
tmpPattern->SetMatrix(gfxMatrix().Translate(needed.pos));
|
||||
tmpCtx.SetPattern(tmpPattern);
|
||||
tmpCtx.Paint();
|
||||
tmpPattern = new gfxPattern(temp);
|
||||
if (tmpPattern) {
|
||||
pattern.swap(tmpPattern);
|
||||
pattern->SetMatrix(
|
||||
gfxMatrix(userSpaceToImageSpace).Multiply(gfxMatrix().Translate(-needed.pos)));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// In theory we can handle this using cairo's EXTEND_PAD,
|
||||
// but implementation limitations mean we have to consult
|
||||
// the surface type.
|
||||
switch (currentTarget->GetType()) {
|
||||
case gfxASurface::SurfaceTypeXlib:
|
||||
case gfxASurface::SurfaceTypeXcb: {
|
||||
// See bug 324698. This is a workaround for EXTEND_PAD not being
|
||||
// implemented correctly on linux in the X server.
|
||||
//
|
||||
// Set the filter to CAIRO_FILTER_FAST --- otherwise,
|
||||
// pixman's sampling will sample transparency for the outside edges and we'll
|
||||
// get blurry edges. CAIRO_EXTEND_PAD would also work here, if
|
||||
// available
|
||||
//
|
||||
// But don't do this for simple downscales because it's horrible.
|
||||
// Downscaling means that device-space coordinates are
|
||||
// scaled *up* to find the image pixel coordinates.
|
||||
//
|
||||
// deviceToImage is slightly stale because up above we may
|
||||
// have adjusted the pattern's matrix ... but the adjustment
|
||||
// is only a translation so the scale factors in deviceToImage
|
||||
// are still valid.
|
||||
PRBool isDownscale =
|
||||
deviceToImage.xx >= 1.0 && deviceToImage.yy >= 1.0 &&
|
||||
deviceToImage.xy == 0.0 && deviceToImage.yx == 0.0;
|
||||
if (!isDownscale) {
|
||||
pattern->SetFilter(gfxPattern::FILTER_FAST);
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
case gfxASurface::SurfaceTypeQuartz:
|
||||
case gfxASurface::SurfaceTypeQuartzImage:
|
||||
// Don't set EXTEND_PAD, Mac seems to be OK. Really?
|
||||
pattern->SetFilter(aFilter);
|
||||
break;
|
||||
|
||||
default:
|
||||
// turn on EXTEND_PAD.
|
||||
// This is what we really want for all surface types, if the
|
||||
// implementation was universally good.
|
||||
pattern->SetExtend(gfxPattern::EXTEND_PAD);
|
||||
pattern->SetFilter(aFilter);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if ((op == gfxContext::OPERATOR_OVER || pushedGroup) &&
|
||||
format == gfxASurface::ImageFormatRGB24) {
|
||||
aContext->SetOperator(gfxContext::OPERATOR_SOURCE);
|
||||
}
|
||||
|
||||
// Phew! Now we can actually draw this image
|
||||
aContext->NewPath();
|
||||
#ifdef MOZ_GFX_OPTIMIZE_MOBILE
|
||||
pattern->SetFilter(gfxPattern::FILTER_FAST);
|
||||
#endif
|
||||
aContext->SetPattern(pattern);
|
||||
aContext->Rectangle(fill);
|
||||
aContext->Fill();
|
||||
|
||||
aContext->SetOperator(op);
|
||||
if (pushedGroup) {
|
||||
aContext->PopGroupToSource();
|
||||
aContext->Paint();
|
||||
aContext->Restore();
|
||||
}
|
||||
}
|
||||
|
||||
nsresult
|
||||
nsThebesImage::Extract(const nsIntRect& aRegion,
|
||||
nsIImage** aResult)
|
||||
{
|
||||
nsRefPtr<nsThebesImage> subImage(new nsThebesImage());
|
||||
if (!subImage)
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
|
||||
// The scaling problems described in bug 468496 are especially
|
||||
// likely to be visible for the sub-image, as at present the only
|
||||
// user is the border-image code and border-images tend to get
|
||||
// stretched a lot. At the same time, the performance concerns
|
||||
// that prevent us from just using Cairo's fallback scaler when
|
||||
// accelerated graphics won't cut it are less relevant to such
|
||||
// images, since they also tend to be small. Thus, we forcibly
|
||||
// disable the use of anything other than a client-side image
|
||||
// surface for the sub-image; this ensures that the correct
|
||||
// (albeit slower) Cairo fallback scaler will be used.
|
||||
subImage->mNeverUseDeviceSurface = PR_TRUE;
|
||||
|
||||
// ->Init() is just going to convert this back, bleah.
|
||||
nsMaskRequirements maskReq;
|
||||
switch (mAlphaDepth) {
|
||||
case 0: maskReq = nsMaskRequirements_kNoMask; break;
|
||||
case 1: maskReq = nsMaskRequirements_kNeeds1Bit; break;
|
||||
case 8: maskReq = nsMaskRequirements_kNeeds8Bit; break;
|
||||
default:
|
||||
NS_NOTREACHED("impossible alpha depth");
|
||||
maskReq = nsMaskRequirements_kNeeds8Bit; // safe
|
||||
}
|
||||
|
||||
nsresult rv = subImage->Init(aRegion.width, aRegion.height,
|
||||
8 /* ignored */, maskReq);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
{ // scope to destroy ctx
|
||||
gfxContext ctx(subImage->ThebesSurface());
|
||||
ctx.SetOperator(gfxContext::OPERATOR_SOURCE);
|
||||
if (mSinglePixel) {
|
||||
ctx.SetDeviceColor(mSinglePixelColor);
|
||||
} else {
|
||||
// SetSource() places point (0,0) of its first argument at
|
||||
// the coordinages given by its second argument. We want
|
||||
// (x,y) of the image to be (0,0) of source space, so we
|
||||
// put (0,0) of the image at (-x,-y).
|
||||
ctx.SetSource(this->ThebesSurface(),
|
||||
gfxPoint(-aRegion.x, -aRegion.y));
|
||||
}
|
||||
ctx.Rectangle(gfxRect(0, 0, aRegion.width, aRegion.height));
|
||||
ctx.Fill();
|
||||
}
|
||||
|
||||
nsIntRect filled(0, 0, aRegion.width, aRegion.height);
|
||||
subImage->ImageUpdated(nsnull, nsImageUpdateFlags_kBitsChanged, &filled);
|
||||
subImage->Optimize(nsnull);
|
||||
|
||||
NS_ADDREF(*aResult = subImage);
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
PRBool
|
||||
nsThebesImage::ShouldUseImageSurfaces()
|
||||
{
|
||||
#if defined(WINCE)
|
||||
// There is no test on windows mobile to check for Gui resources.
|
||||
// Allocate, until we run out of memory.
|
||||
gfxWindowsPlatform::RenderMode rmode = gfxWindowsPlatform::GetPlatform()->GetRenderMode();
|
||||
return rmode != gfxWindowsPlatform::RENDER_DDRAW &&
|
||||
rmode != gfxWindowsPlatform::RENDER_DDRAW_GL;
|
||||
|
||||
#elif defined(USE_WIN_SURFACE)
|
||||
static const DWORD kGDIObjectsHighWaterMark = 7000;
|
||||
|
||||
// at 7000 GDI objects, stop allocating normal images to make sure
|
||||
// we never hit the 10k hard limit.
|
||||
// GetCurrentProcess() just returns (HANDLE)-1, it's inlined afaik
|
||||
DWORD count = GetGuiResources(GetCurrentProcess(), GR_GDIOBJECTS);
|
||||
if (count == 0 ||
|
||||
count > kGDIObjectsHighWaterMark)
|
||||
{
|
||||
// either something's broken (count == 0),
|
||||
// or we hit our high water mark; disable
|
||||
// image allocations for a bit.
|
||||
return PR_TRUE;
|
||||
}
|
||||
#endif
|
||||
|
||||
return PR_FALSE;
|
||||
}
|
||||
|
||||
// A hint from the image decoders that this image has no alpha, even
|
||||
// though we created is ARGB32. This changes our format to RGB24,
|
||||
// which in turn will cause us to Optimize() to RGB24. Has no effect
|
||||
// after Optimize() is called, though in all cases it will be just a
|
||||
// performance win -- the pixels are still correct and have the A byte
|
||||
// set to 0xff.
|
||||
void
|
||||
nsThebesImage::SetHasNoAlpha()
|
||||
{
|
||||
if (mFormat == gfxASurface::ImageFormatARGB32) {
|
||||
mFormat = gfxASurface::ImageFormatRGB24;
|
||||
mFormatChanged = PR_TRUE;
|
||||
mAlphaDepth = 0;
|
||||
}
|
||||
}
|
||||
|
|
@ -1,187 +0,0 @@
|
|||
/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
||||
/* ***** BEGIN LICENSE BLOCK *****
|
||||
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
||||
*
|
||||
* 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 thebes gfx
|
||||
*
|
||||
* The Initial Developer of the Original Code is
|
||||
* mozilla.org.
|
||||
* Portions created by the Initial Developer are Copyright (C) 2005
|
||||
* the Initial Developer. All Rights Reserved.
|
||||
*
|
||||
* Contributor(s):
|
||||
* Vladimir Vukicevic <vladimir@pobox.com>
|
||||
*
|
||||
* 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 MPL, 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 MPL, the GPL or the LGPL.
|
||||
*
|
||||
* ***** END LICENSE BLOCK ***** */
|
||||
|
||||
#ifndef _NSTHEBESIMAGE_H_
|
||||
#define _NSTHEBESIMAGE_H_
|
||||
|
||||
#include "nsIImage.h"
|
||||
|
||||
#include "gfxColor.h"
|
||||
#include "gfxASurface.h"
|
||||
#include "gfxImageSurface.h"
|
||||
#include "gfxPattern.h"
|
||||
#if defined(XP_WIN)
|
||||
#include "gfxWindowsSurface.h"
|
||||
#elif defined(XP_MACOSX)
|
||||
#include "gfxQuartzImageSurface.h"
|
||||
#endif
|
||||
|
||||
class nsThebesImage : public nsIImage
|
||||
{
|
||||
public:
|
||||
nsThebesImage();
|
||||
~nsThebesImage();
|
||||
|
||||
NS_DECL_ISUPPORTS
|
||||
|
||||
virtual nsresult Init(PRInt32 aWidth, PRInt32 aHeight,
|
||||
PRInt32 aDepth, nsMaskRequirements aMaskRequirements);
|
||||
virtual PRInt32 GetBytesPix();
|
||||
virtual PRBool GetIsRowOrderTopToBottom();
|
||||
virtual PRInt32 GetWidth();
|
||||
virtual PRInt32 GetHeight();
|
||||
virtual PRUint8 *GetBits();
|
||||
virtual PRInt32 GetLineStride();
|
||||
virtual PRBool GetHasAlphaMask();
|
||||
virtual PRUint8 *GetAlphaBits();
|
||||
virtual PRInt32 GetAlphaLineStride();
|
||||
virtual PRBool GetIsImageComplete();
|
||||
virtual nsresult ImageUpdated(nsIDeviceContext *aContext, PRUint8 aFlags, nsIntRect *aUpdateRect);
|
||||
virtual nsresult Optimize(nsIDeviceContext* aContext);
|
||||
virtual nsColorMap *GetColorMap();
|
||||
|
||||
virtual void Draw(gfxContext* aContext,
|
||||
gfxPattern::GraphicsFilter aFilter,
|
||||
const gfxMatrix& aUserSpaceToImageSpace,
|
||||
const gfxRect& aFill,
|
||||
const nsIntMargin& aPadding,
|
||||
const nsIntRect& aSubimage);
|
||||
|
||||
virtual PRInt8 GetAlphaDepth();
|
||||
virtual void* GetBitInfo();
|
||||
NS_IMETHOD LockImagePixels(PRBool aMaskPixels);
|
||||
NS_IMETHOD UnlockImagePixels(PRBool aMaskPixels);
|
||||
|
||||
NS_IMETHOD GetSurface(gfxASurface **aSurface) {
|
||||
*aSurface = ThebesSurface();
|
||||
NS_ADDREF(*aSurface);
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
NS_IMETHOD GetPattern(gfxPattern **aPattern) {
|
||||
if (mSinglePixel)
|
||||
*aPattern = new gfxPattern(mSinglePixelColor);
|
||||
else
|
||||
*aPattern = new gfxPattern(ThebesSurface());
|
||||
NS_ADDREF(*aPattern);
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
gfxASurface* ThebesSurface() {
|
||||
if (mOptSurface)
|
||||
return mOptSurface;
|
||||
#if defined(XP_WIN) && !defined(WINCE)
|
||||
if (mWinSurface)
|
||||
return mWinSurface;
|
||||
#elif defined(XP_MACOSX)
|
||||
if (mQuartzSurface)
|
||||
return mQuartzSurface;
|
||||
#endif
|
||||
return mImageSurface;
|
||||
}
|
||||
|
||||
void SetHasNoAlpha();
|
||||
|
||||
NS_IMETHOD Extract(const nsIntRect& aSubimage,
|
||||
nsIImage** aResult NS_OUTPARAM);
|
||||
|
||||
protected:
|
||||
static PRBool AllowedImageSize(PRInt32 aWidth, PRInt32 aHeight) {
|
||||
NS_ASSERTION(aWidth > 0, "invalid image width");
|
||||
NS_ASSERTION(aHeight > 0, "invalid image height");
|
||||
|
||||
// reject over-wide or over-tall images
|
||||
const PRInt32 k64KLimit = 0x0000FFFF;
|
||||
if (NS_UNLIKELY(aWidth > k64KLimit || aHeight > k64KLimit )) {
|
||||
NS_WARNING("image too big");
|
||||
return PR_FALSE;
|
||||
}
|
||||
|
||||
// protect against division by zero - this really shouldn't happen
|
||||
// if our consumers were well behaved, but they aren't (bug 368427)
|
||||
if (NS_UNLIKELY(aHeight == 0)) {
|
||||
return PR_FALSE;
|
||||
}
|
||||
|
||||
// check to make sure we don't overflow a 32-bit
|
||||
PRInt32 tmp = aWidth * aHeight;
|
||||
if (NS_UNLIKELY(tmp / aHeight != aWidth)) {
|
||||
NS_WARNING("width or height too large");
|
||||
return PR_FALSE;
|
||||
}
|
||||
tmp = tmp * 4;
|
||||
if (NS_UNLIKELY(tmp / 4 != aWidth * aHeight)) {
|
||||
NS_WARNING("width or height too large");
|
||||
return PR_FALSE;
|
||||
}
|
||||
return PR_TRUE;
|
||||
}
|
||||
|
||||
gfxImageSurface::gfxImageFormat mFormat;
|
||||
PRInt32 mWidth;
|
||||
PRInt32 mHeight;
|
||||
PRInt32 mStride;
|
||||
nsIntRect mDecoded;
|
||||
PRPackedBool mImageComplete;
|
||||
PRPackedBool mSinglePixel;
|
||||
PRPackedBool mFormatChanged;
|
||||
PRPackedBool mNeverUseDeviceSurface;
|
||||
#if defined(XP_WIN) && !defined(WINCE)
|
||||
PRPackedBool mIsDDBSurface;
|
||||
#endif
|
||||
|
||||
gfxRGBA mSinglePixelColor;
|
||||
|
||||
nsRefPtr<gfxImageSurface> mImageSurface;
|
||||
nsRefPtr<gfxASurface> mOptSurface;
|
||||
#if defined(XP_WIN) && !defined(WINCE)
|
||||
nsRefPtr<gfxWindowsSurface> mWinSurface;
|
||||
#elif defined(XP_MACOSX)
|
||||
nsRefPtr<gfxQuartzImageSurface> mQuartzSurface;
|
||||
#endif
|
||||
|
||||
PRUint8 mAlphaDepth;
|
||||
|
||||
// this function should return true if
|
||||
// we should (temporarily) not allocate any
|
||||
// platform native surfaces and instead use
|
||||
// image surfaces for everything.
|
||||
static PRBool ShouldUseImageSurfaces();
|
||||
};
|
||||
|
||||
#endif /* _NSTHEBESIMAGE_H_ */
|
||||
|
|
@ -46,12 +46,7 @@
|
|||
#include "nsIInterfaceRequestorUtils.h"
|
||||
#include "nsGfxCIID.h"
|
||||
|
||||
#include "imgIContainer.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsIImage.h"
|
||||
|
||||
#include "nsThebesRegion.h"
|
||||
#include "nsThebesImage.h"
|
||||
|
||||
#include <stdlib.h>
|
||||
#include <sys/types.h>
|
||||
|
|
|
|||
|
|
@ -54,8 +54,6 @@
|
|||
#include "nsIThebesFontMetrics.h"
|
||||
#include "gfxContext.h"
|
||||
|
||||
class nsIImage;
|
||||
|
||||
class nsThebesRenderingContext : public nsIRenderingContext
|
||||
{
|
||||
public:
|
||||
|
|
|
|||
|
|
@ -46,7 +46,6 @@
|
|||
|
||||
#include "nsStyleConsts.h"
|
||||
#include "nsPresContext.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsIFrame.h"
|
||||
#include "nsPoint.h"
|
||||
#include "nsRect.h"
|
||||
|
|
@ -63,7 +62,6 @@
|
|||
#include "nsIScrollableFrame.h"
|
||||
#include "imgIRequest.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsCSSRendering.h"
|
||||
#include "nsCSSColorUtils.h"
|
||||
#include "nsITheme.h"
|
||||
|
|
@ -321,7 +319,7 @@ static void DrawBorderImage(nsPresContext* aPresContext,
|
|||
|
||||
static void DrawBorderImageComponent(nsIRenderingContext& aRenderingContext,
|
||||
nsIFrame* aForFrame,
|
||||
nsIImage* aImage,
|
||||
imgIContainer* aImage,
|
||||
const nsRect& aDirtyRect,
|
||||
const nsRect& aFill,
|
||||
const nsIntRect& aSrc,
|
||||
|
|
@ -1687,27 +1685,9 @@ nsCSSRendering::PaintBackgroundWithSC(nsPresContext* aPresContext,
|
|||
drawBackgroundColor) {
|
||||
nsCOMPtr<imgIContainer> image;
|
||||
bottomImage->GetImage(getter_AddRefs(image));
|
||||
// If the image is completely opaque, we may not need to paint
|
||||
// the background color.
|
||||
nsCOMPtr<gfxIImageFrame> gfxImgFrame;
|
||||
image->GetCurrentFrame(getter_AddRefs(gfxImgFrame));
|
||||
if (gfxImgFrame) {
|
||||
gfxImgFrame->GetNeedsBackground(&drawBackgroundColor);
|
||||
if (!drawBackgroundColor) {
|
||||
// If the current frame is smaller than its container, we
|
||||
// need to paint the background color even if the frame
|
||||
// itself is opaque.
|
||||
nsIntSize iSize;
|
||||
image->GetWidth(&iSize.width);
|
||||
image->GetHeight(&iSize.height);
|
||||
nsIntRect iframeRect;
|
||||
gfxImgFrame->GetRect(iframeRect);
|
||||
if (iSize.width != iframeRect.width ||
|
||||
iSize.height != iframeRect.height) {
|
||||
drawBackgroundColor = PR_TRUE;
|
||||
}
|
||||
}
|
||||
}
|
||||
PRBool isOpaque;
|
||||
if (NS_SUCCEEDED(image->GetCurrentFrameIsOpaque(&isOpaque)) && isOpaque)
|
||||
drawBackgroundColor = PR_FALSE;
|
||||
}
|
||||
|
||||
// The background color is rendered over the entire dirty area,
|
||||
|
|
@ -1938,18 +1918,6 @@ DrawBorderImage(nsPresContext* aPresContext,
|
|||
imgContainer->GetWidth(&imageSize.width);
|
||||
imgContainer->GetHeight(&imageSize.height);
|
||||
|
||||
nsCOMPtr<gfxIImageFrame> imgFrame;
|
||||
imgContainer->GetCurrentFrame(getter_AddRefs(imgFrame));
|
||||
|
||||
nsIntRect innerRect;
|
||||
imgFrame->GetRect(innerRect);
|
||||
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(imgFrame));
|
||||
// The inner rectangle should precisely enclose the image pixels for
|
||||
// this frame.
|
||||
NS_ASSERTION(innerRect.width == img->GetWidth(), "img inner width off");
|
||||
NS_ASSERTION(innerRect.height == img->GetHeight(), "img inner height off");
|
||||
|
||||
// Convert percentages and clamp values to the image size.
|
||||
nsIntMargin split;
|
||||
NS_FOR_CSS_SIDES(s) {
|
||||
|
|
@ -1983,7 +1951,7 @@ DrawBorderImage(nsPresContext* aPresContext,
|
|||
// These helper tables recharacterize the 'split' and 'border' margins
|
||||
// in a more convenient form: they are the x/y/width/height coords
|
||||
// required for various bands of the border, and they have been transformed
|
||||
// to be relative to the innerRect (for 'split') or the page (for 'border').
|
||||
// to be relative to the image (for 'split') or the page (for 'border').
|
||||
enum {
|
||||
LEFT, MIDDLE, RIGHT,
|
||||
TOP = LEFT, BOTTOM = RIGHT
|
||||
|
|
@ -2010,14 +1978,14 @@ DrawBorderImage(nsPresContext* aPresContext,
|
|||
};
|
||||
|
||||
const PRInt32 splitX[3] = {
|
||||
-innerRect.x + 0,
|
||||
-innerRect.x + split.left,
|
||||
-innerRect.x + imageSize.width - split.right,
|
||||
0,
|
||||
split.left,
|
||||
imageSize.width - split.right,
|
||||
};
|
||||
const PRInt32 splitY[3] = {
|
||||
-innerRect.y + 0,
|
||||
-innerRect.y + split.top,
|
||||
-innerRect.y + imageSize.height - split.bottom,
|
||||
0,
|
||||
split.top,
|
||||
imageSize.height - split.bottom,
|
||||
};
|
||||
const PRInt32 splitWidth[3] = {
|
||||
split.left,
|
||||
|
|
@ -2108,7 +2076,7 @@ DrawBorderImage(nsPresContext* aPresContext,
|
|||
}
|
||||
|
||||
DrawBorderImageComponent(aRenderingContext, aForFrame,
|
||||
img, aDirtyRect,
|
||||
imgContainer, aDirtyRect,
|
||||
destArea, subArea,
|
||||
fillStyleH, fillStyleV, unitSize);
|
||||
}
|
||||
|
|
@ -2118,7 +2086,7 @@ DrawBorderImage(nsPresContext* aPresContext,
|
|||
static void
|
||||
DrawBorderImageComponent(nsIRenderingContext& aRenderingContext,
|
||||
nsIFrame* aForFrame,
|
||||
nsIImage* aImage,
|
||||
imgIContainer* aImage,
|
||||
const nsRect& aDirtyRect,
|
||||
const nsRect& aFill,
|
||||
const nsIntRect& aSrc,
|
||||
|
|
@ -2129,8 +2097,8 @@ DrawBorderImageComponent(nsIRenderingContext& aRenderingContext,
|
|||
if (aFill.IsEmpty() || aSrc.IsEmpty())
|
||||
return;
|
||||
|
||||
nsCOMPtr<nsIImage> subImage;
|
||||
if (NS_FAILED(aImage->Extract(aSrc, getter_AddRefs(subImage))))
|
||||
nsCOMPtr<imgIContainer> subImage;
|
||||
if (NS_FAILED(aImage->ExtractCurrentFrame(aSrc, getter_AddRefs(subImage))))
|
||||
return;
|
||||
|
||||
gfxPattern::GraphicsFilter graphicsFilter =
|
||||
|
|
|
|||
|
|
@ -39,7 +39,6 @@
|
|||
|
||||
#include "nsStyleConsts.h"
|
||||
#include "nsPresContext.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsIFrame.h"
|
||||
#include "nsPoint.h"
|
||||
#include "nsRect.h"
|
||||
|
|
@ -56,7 +55,6 @@
|
|||
#include "nsIScrollableFrame.h"
|
||||
#include "imgIRequest.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsCSSRendering.h"
|
||||
#include "nsCSSColorUtils.h"
|
||||
#include "nsITheme.h"
|
||||
|
|
|
|||
|
|
@ -58,9 +58,7 @@
|
|||
#endif
|
||||
|
||||
#include "imgIContainer.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsIInterfaceRequestorUtils.h"
|
||||
#include "nsIImage.h"
|
||||
|
||||
nsDisplayListBuilder::nsDisplayListBuilder(nsIFrame* aReferenceFrame,
|
||||
PRBool aIsForEvents, PRBool aBuildCaret)
|
||||
|
|
@ -555,18 +553,12 @@ nsDisplayBackground::IsOpaque(nsDisplayListBuilder* aBuilder) {
|
|||
nsCOMPtr<imgIContainer> container;
|
||||
bottomLayer.mImage->GetImage(getter_AddRefs(container));
|
||||
if (container) {
|
||||
PRUint32 nframes;
|
||||
container->GetNumFrames(&nframes);
|
||||
if (nframes == 1) {
|
||||
nsCOMPtr<gfxIImageFrame> imgFrame;
|
||||
container->GetCurrentFrame(getter_AddRefs(imgFrame));
|
||||
if (imgFrame) {
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(imgFrame));
|
||||
|
||||
PRBool hasMask = img->GetHasAlphaMask();
|
||||
|
||||
return !hasMask;
|
||||
}
|
||||
PRBool animated;
|
||||
container->GetAnimated(&animated);
|
||||
if (!animated) {
|
||||
PRBool isOpaque;
|
||||
if (NS_SUCCEEDED(container->GetCurrentFrameIsOpaque(&isOpaque)))
|
||||
return isOpaque;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -154,7 +154,7 @@ NS_IMETHODIMP nsImageLoader::OnStartContainer(imgIRequest *aRequest,
|
|||
}
|
||||
|
||||
NS_IMETHODIMP nsImageLoader::OnStopFrame(imgIRequest *aRequest,
|
||||
gfxIImageFrame *aFrame)
|
||||
PRUint32 aFrame)
|
||||
{
|
||||
if (!mFrame)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
|
@ -182,7 +182,6 @@ NS_IMETHODIMP nsImageLoader::OnStopFrame(imgIRequest *aRequest,
|
|||
}
|
||||
|
||||
NS_IMETHODIMP nsImageLoader::FrameChanged(imgIContainer *aContainer,
|
||||
gfxIImageFrame *newframe,
|
||||
nsIntRect *dirtyRect)
|
||||
{
|
||||
if (!mFrame)
|
||||
|
|
|
|||
|
|
@ -70,7 +70,7 @@ public:
|
|||
|
||||
// imgIDecoderObserver (override nsStubImageDecoderObserver)
|
||||
NS_IMETHOD OnStartContainer(imgIRequest *aRequest, imgIContainer *aImage);
|
||||
NS_IMETHOD OnStopFrame(imgIRequest *aRequest, gfxIImageFrame *aFrame);
|
||||
NS_IMETHOD OnStopFrame(imgIRequest *aRequest, PRUint32 aFrame);
|
||||
// Do not override OnDataAvailable since background images are not
|
||||
// displayed incrementally; they are displayed after the entire image
|
||||
// has been loaded.
|
||||
|
|
@ -78,8 +78,7 @@ public:
|
|||
// incrementally in nsImageFrame.cpp.
|
||||
|
||||
// imgIContainerObserver (override nsStubImageDecoderObserver)
|
||||
NS_IMETHOD FrameChanged(imgIContainer *aContainer, gfxIImageFrame *newframe,
|
||||
nsIntRect *dirtyRect);
|
||||
NS_IMETHOD FrameChanged(imgIContainer *aContainer, nsIntRect *dirtyRect);
|
||||
|
||||
void Destroy();
|
||||
|
||||
|
|
|
|||
|
|
@ -61,12 +61,10 @@
|
|||
#include "nsFrameManager.h"
|
||||
#include "nsBlockFrame.h"
|
||||
#include "nsBidiPresUtils.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "gfxRect.h"
|
||||
#include "gfxContext.h"
|
||||
#include "gfxFont.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsIInterfaceRequestorUtils.h"
|
||||
#include "nsCSSRendering.h"
|
||||
#include "nsContentUtils.h"
|
||||
|
|
@ -2732,14 +2730,13 @@ MapToFloatUserPixels(const gfxSize& aSize,
|
|||
|
||||
static nsresult
|
||||
DrawImageInternal(nsIRenderingContext* aRenderingContext,
|
||||
nsIImage* aImage,
|
||||
imgIContainer* aImage,
|
||||
gfxPattern::GraphicsFilter aGraphicsFilter,
|
||||
const nsRect& aDest,
|
||||
const nsRect& aFill,
|
||||
const nsPoint& aAnchor,
|
||||
const nsRect& aDirty,
|
||||
const nsIntSize& aImageSize,
|
||||
const nsIntRect& aInnerRect)
|
||||
const nsIntSize& aImageSize)
|
||||
{
|
||||
if (aDest.IsEmpty() || aFill.IsEmpty())
|
||||
return NS_OK;
|
||||
|
|
@ -2839,22 +2836,18 @@ DrawImageInternal(nsIRenderingContext* aRenderingContext,
|
|||
if (finalFillRect.IsEmpty())
|
||||
return NS_OK;
|
||||
|
||||
nsIntMargin padding(aInnerRect.x, aInnerRect.y,
|
||||
imageSize.width - aInnerRect.XMost(),
|
||||
imageSize.height - aInnerRect.YMost());
|
||||
aImage->Draw(ctx, aGraphicsFilter, transform, finalFillRect, padding, intSubimage);
|
||||
aImage->Draw(ctx, aGraphicsFilter, transform, finalFillRect, intSubimage);
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/* Workhorse for DrawSingleUnscaledImage. */
|
||||
static nsresult
|
||||
DrawSingleUnscaledImageInternal(nsIRenderingContext* aRenderingContext,
|
||||
nsIImage* aImage,
|
||||
imgIContainer* aImage,
|
||||
const nsPoint& aDest,
|
||||
const nsRect& aDirty,
|
||||
const nsRect* aSourceArea,
|
||||
const nsIntSize& aImageSize,
|
||||
const nsIntRect& aInnerRect)
|
||||
const nsIntSize& aImageSize)
|
||||
{
|
||||
if (aImageSize.width == 0 || aImageSize.height == 0)
|
||||
return NS_OK;
|
||||
|
|
@ -2877,19 +2870,18 @@ DrawSingleUnscaledImageInternal(nsIRenderingContext* aRenderingContext,
|
|||
// translation but we don't want to actually tile the image.
|
||||
fill.IntersectRect(fill, dest);
|
||||
return DrawImageInternal(aRenderingContext, aImage, gfxPattern::FILTER_NEAREST,
|
||||
dest, fill, aDest, aDirty, aImageSize, aInnerRect);
|
||||
dest, fill, aDest, aDirty, aImageSize);
|
||||
}
|
||||
|
||||
/* Workhorse for DrawSingleImage. */
|
||||
static nsresult
|
||||
DrawSingleImageInternal(nsIRenderingContext* aRenderingContext,
|
||||
nsIImage* aImage,
|
||||
imgIContainer* aImage,
|
||||
gfxPattern::GraphicsFilter aGraphicsFilter,
|
||||
const nsRect& aDest,
|
||||
const nsRect& aDirty,
|
||||
const nsRect* aSourceArea,
|
||||
const nsIntSize& aImageSize,
|
||||
const nsIntRect& aInnerRect)
|
||||
const nsIntSize& aImageSize)
|
||||
{
|
||||
if (aImageSize.width == 0 || aImageSize.height == 0)
|
||||
return NS_OK;
|
||||
|
|
@ -2911,7 +2903,7 @@ DrawSingleImageInternal(nsIRenderingContext* aRenderingContext,
|
|||
nsRect fill;
|
||||
fill.IntersectRect(aDest, dest);
|
||||
return DrawImageInternal(aRenderingContext, aImage, aGraphicsFilter, dest, fill,
|
||||
fill.TopLeft(), aDirty, aImageSize, aInnerRect);
|
||||
fill.TopLeft(), aDirty, aImageSize);
|
||||
}
|
||||
|
||||
/* The exposed Draw*Image functions just do interface conversion and call the
|
||||
|
|
@ -2926,38 +2918,13 @@ nsLayoutUtils::DrawImage(nsIRenderingContext* aRenderingContext,
|
|||
const nsPoint& aAnchor,
|
||||
const nsRect& aDirty)
|
||||
{
|
||||
nsCOMPtr<gfxIImageFrame> imgFrame;
|
||||
aImage->GetCurrentFrame(getter_AddRefs(imgFrame));
|
||||
if (!imgFrame) return NS_ERROR_FAILURE;
|
||||
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(imgFrame));
|
||||
if (!img) return NS_ERROR_FAILURE;
|
||||
|
||||
nsIntRect innerRect;
|
||||
imgFrame->GetRect(innerRect);
|
||||
|
||||
nsIntSize imageSize;
|
||||
aImage->GetWidth(&imageSize.width);
|
||||
aImage->GetHeight(&imageSize.height);
|
||||
|
||||
return DrawImageInternal(aRenderingContext, img, aGraphicsFilter,
|
||||
aDest, aFill, aAnchor, aDirty,
|
||||
imageSize, innerRect);
|
||||
}
|
||||
|
||||
/* static */ nsresult
|
||||
nsLayoutUtils::DrawImage(nsIRenderingContext* aRenderingContext,
|
||||
nsIImage* aImage,
|
||||
gfxPattern::GraphicsFilter aGraphicsFilter,
|
||||
const nsRect& aDest,
|
||||
const nsRect& aFill,
|
||||
const nsPoint& aAnchor,
|
||||
const nsRect& aDirty)
|
||||
{
|
||||
nsIntSize imageSize(aImage->GetWidth(), aImage->GetHeight());
|
||||
return DrawImageInternal(aRenderingContext, aImage, aGraphicsFilter,
|
||||
aDest, aFill, aAnchor, aDirty,
|
||||
imageSize, nsIntRect(nsIntPoint(0,0), imageSize));
|
||||
imageSize);
|
||||
}
|
||||
|
||||
/* static */ nsresult
|
||||
|
|
@ -2967,23 +2934,13 @@ nsLayoutUtils::DrawSingleUnscaledImage(nsIRenderingContext* aRenderingContext,
|
|||
const nsRect& aDirty,
|
||||
const nsRect* aSourceArea)
|
||||
{
|
||||
nsCOMPtr<gfxIImageFrame> imgFrame;
|
||||
aImage->GetCurrentFrame(getter_AddRefs(imgFrame));
|
||||
if (!imgFrame) return NS_ERROR_FAILURE;
|
||||
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(imgFrame));
|
||||
if (!img) return NS_ERROR_FAILURE;
|
||||
|
||||
nsIntRect innerRect;
|
||||
imgFrame->GetRect(innerRect);
|
||||
|
||||
nsIntSize imageSize;
|
||||
aImage->GetWidth(&imageSize.width);
|
||||
aImage->GetHeight(&imageSize.height);
|
||||
|
||||
return DrawSingleUnscaledImageInternal(aRenderingContext, img,
|
||||
return DrawSingleUnscaledImageInternal(aRenderingContext, aImage,
|
||||
aDest, aDirty, aSourceArea,
|
||||
imageSize, innerRect);
|
||||
imageSize);
|
||||
}
|
||||
|
||||
/* static */ nsresult
|
||||
|
|
@ -2994,41 +2951,15 @@ nsLayoutUtils::DrawSingleImage(nsIRenderingContext* aRenderingContext,
|
|||
const nsRect& aDirty,
|
||||
const nsRect* aSourceArea)
|
||||
{
|
||||
nsCOMPtr<gfxIImageFrame> imgFrame;
|
||||
aImage->GetCurrentFrame(getter_AddRefs(imgFrame));
|
||||
if (!imgFrame) return NS_ERROR_FAILURE;
|
||||
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(imgFrame));
|
||||
if (!img) return NS_ERROR_FAILURE;
|
||||
|
||||
nsIntRect innerRect;
|
||||
imgFrame->GetRect(innerRect);
|
||||
|
||||
nsIntSize imageSize;
|
||||
aImage->GetWidth(&imageSize.width);
|
||||
aImage->GetHeight(&imageSize.height);
|
||||
|
||||
return DrawSingleImageInternal(aRenderingContext, img, aGraphicsFilter,
|
||||
aDest, aDirty, aSourceArea,
|
||||
imageSize, innerRect);
|
||||
}
|
||||
|
||||
/* static */ nsresult
|
||||
nsLayoutUtils::DrawSingleImage(nsIRenderingContext* aRenderingContext,
|
||||
nsIImage* aImage,
|
||||
gfxPattern::GraphicsFilter aGraphicsFilter,
|
||||
const nsRect& aDest,
|
||||
const nsRect& aDirty,
|
||||
const nsRect* aSourceArea)
|
||||
{
|
||||
nsIntSize imageSize(aImage->GetWidth(), aImage->GetHeight());
|
||||
return DrawSingleImageInternal(aRenderingContext, aImage, aGraphicsFilter,
|
||||
aDest, aDirty, aSourceArea,
|
||||
imageSize,
|
||||
nsIntRect(nsIntPoint(0, 0), imageSize));
|
||||
imageSize);
|
||||
}
|
||||
|
||||
|
||||
/* static */ nsRect
|
||||
nsLayoutUtils::GetWholeImageDestination(const nsIntSize& aWholeImageSize,
|
||||
const nsRect& aImageSourceArea,
|
||||
|
|
@ -3386,7 +3317,6 @@ nsLayoutUtils::SurfaceFromElement(nsIDOMElement *aElement,
|
|||
#endif
|
||||
|
||||
// Finally, check if it's a normal image
|
||||
nsCOMPtr<imgIContainer> imgContainer;
|
||||
nsCOMPtr<nsIImageLoadingContent> imageLoader = do_QueryInterface(aElement);
|
||||
|
||||
if (!imageLoader)
|
||||
|
|
@ -3425,34 +3355,28 @@ nsLayoutUtils::SurfaceFromElement(nsIDOMElement *aElement,
|
|||
return result;
|
||||
}
|
||||
|
||||
nsCOMPtr<imgIContainer> imgContainer;
|
||||
rv = imgRequest->GetImage(getter_AddRefs(imgContainer));
|
||||
if (NS_FAILED(rv) || !imgContainer)
|
||||
return result;
|
||||
|
||||
nsCOMPtr<gfxIImageFrame> frame;
|
||||
rv = imgContainer->GetCurrentFrame(getter_AddRefs(frame));
|
||||
nsRefPtr<gfxASurface> framesurf;
|
||||
rv = imgContainer->GetCurrentFrame(getter_AddRefs(framesurf));
|
||||
if (NS_FAILED(rv))
|
||||
return result;
|
||||
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(frame));
|
||||
if (!img)
|
||||
return result;
|
||||
|
||||
PRInt32 imgWidth, imgHeight;
|
||||
rv = frame->GetWidth(&imgWidth);
|
||||
rv |= frame->GetHeight(&imgHeight);
|
||||
rv = imgContainer->GetWidth(&imgWidth);
|
||||
rv |= imgContainer->GetHeight(&imgHeight);
|
||||
if (NS_FAILED(rv))
|
||||
return result;
|
||||
|
||||
nsRefPtr<gfxPattern> gfxpattern;
|
||||
img->GetPattern(getter_AddRefs(gfxpattern));
|
||||
nsRefPtr<gfxASurface> gfxsurf = gfxpattern->GetSurface();
|
||||
|
||||
if (wantImageSurface && gfxsurf->GetType() != gfxASurface::SurfaceTypeImage) {
|
||||
if (wantImageSurface && framesurf->GetType() != gfxASurface::SurfaceTypeImage) {
|
||||
forceCopy = PR_TRUE;
|
||||
}
|
||||
|
||||
if (forceCopy || !gfxsurf) {
|
||||
nsRefPtr<gfxASurface> gfxsurf = framesurf;
|
||||
if (forceCopy) {
|
||||
if (wantImageSurface) {
|
||||
gfxsurf = new gfxImageSurface (gfxIntSize(imgWidth, imgHeight), gfxASurface::ImageFormatARGB32);
|
||||
} else {
|
||||
|
|
@ -3463,7 +3387,7 @@ nsLayoutUtils::SurfaceFromElement(nsIDOMElement *aElement,
|
|||
nsRefPtr<gfxContext> ctx = new gfxContext(gfxsurf);
|
||||
|
||||
ctx->SetOperator(gfxContext::OPERATOR_SOURCE);
|
||||
ctx->SetPattern(gfxpattern);
|
||||
ctx->SetSource(framesurf);
|
||||
ctx->Paint();
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -862,14 +862,6 @@ public:
|
|||
const nsPoint& aAnchor,
|
||||
const nsRect& aDirty);
|
||||
|
||||
static nsresult DrawImage(nsIRenderingContext* aRenderingContext,
|
||||
nsIImage* aImage,
|
||||
gfxPattern::GraphicsFilter aGraphicsFilter,
|
||||
const nsRect& aDest,
|
||||
const nsRect& aFill,
|
||||
const nsPoint& aAnchor,
|
||||
const nsRect& aDirty);
|
||||
|
||||
/**
|
||||
* Draw a whole image without scaling or tiling.
|
||||
*
|
||||
|
|
@ -911,13 +903,6 @@ public:
|
|||
const nsRect& aDirty,
|
||||
const nsRect* aSourceArea = nsnull);
|
||||
|
||||
static nsresult DrawSingleImage(nsIRenderingContext* aRenderingContext,
|
||||
nsIImage* aImage,
|
||||
gfxPattern::GraphicsFilter aGraphicsFilter,
|
||||
const nsRect& aDest,
|
||||
const nsRect& aDirty,
|
||||
const nsRect* aSourceArea = nsnull);
|
||||
|
||||
/**
|
||||
* Given a source area of an image (in appunits) and a destination area
|
||||
* that we want to map that source area too, computes the area that
|
||||
|
|
|
|||
|
|
@ -84,7 +84,6 @@ class nsIContent;
|
|||
class nsIFontMetrics;
|
||||
class nsIFrame;
|
||||
class nsFrameManager;
|
||||
class nsIImage;
|
||||
class nsILinkHandler;
|
||||
class nsStyleContext;
|
||||
class nsIAtom;
|
||||
|
|
|
|||
|
|
@ -52,7 +52,6 @@
|
|||
#include "nsISupports.h"
|
||||
#include "nsGkAtoms.h"
|
||||
#include "nsCSSAnonBoxes.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsStyleConsts.h"
|
||||
#include "nsIComponentManager.h"
|
||||
#include "nsIDocument.h"
|
||||
|
|
|
|||
|
|
@ -49,7 +49,6 @@
|
|||
#include "nsGkAtoms.h"
|
||||
#include "nsIDeviceContext.h"
|
||||
#include "nsIFontMetrics.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsStyleConsts.h"
|
||||
#include "nsFormControlFrame.h"
|
||||
#include "nsGUIEvent.h"
|
||||
|
|
|
|||
|
|
@ -71,13 +71,12 @@ public:
|
|||
NS_DECL_ISUPPORTS
|
||||
// imgIDecoderObserver (override nsStubImageDecoderObserver)
|
||||
NS_IMETHOD OnStartContainer(imgIRequest *aRequest, imgIContainer *aImage);
|
||||
NS_IMETHOD OnDataAvailable(imgIRequest *aRequest, gfxIImageFrame *aFrame,
|
||||
NS_IMETHOD OnDataAvailable(imgIRequest *aRequest, PRBool aCurrentFrame,
|
||||
const nsIntRect *aRect);
|
||||
NS_IMETHOD OnStopDecode(imgIRequest *aRequest, nsresult status,
|
||||
const PRUnichar *statusArg);
|
||||
// imgIContainerObserver (override nsStubImageDecoderObserver)
|
||||
NS_IMETHOD FrameChanged(imgIContainer *aContainer, gfxIImageFrame *newframe,
|
||||
nsIntRect *dirtyRect);
|
||||
NS_IMETHOD FrameChanged(imgIContainer *aContainer, nsIntRect *dirtyRect);
|
||||
|
||||
void SetFrame(nsBulletFrame *frame) { mFrame = frame; }
|
||||
|
||||
|
|
@ -1458,7 +1457,7 @@ NS_IMETHODIMP nsBulletFrame::OnStartContainer(imgIRequest *aRequest,
|
|||
}
|
||||
|
||||
NS_IMETHODIMP nsBulletFrame::OnDataAvailable(imgIRequest *aRequest,
|
||||
gfxIImageFrame *aFrame,
|
||||
PRBool aCurrentFrame,
|
||||
const nsIntRect *aRect)
|
||||
{
|
||||
// The image has changed.
|
||||
|
|
@ -1489,7 +1488,6 @@ NS_IMETHODIMP nsBulletFrame::OnStopDecode(imgIRequest *aRequest,
|
|||
}
|
||||
|
||||
NS_IMETHODIMP nsBulletFrame::FrameChanged(imgIContainer *aContainer,
|
||||
gfxIImageFrame *aNewFrame,
|
||||
nsIntRect *aDirtyRect)
|
||||
{
|
||||
// Invalidate the entire content area. Maybe it's not optimal but it's simple and
|
||||
|
|
@ -1547,13 +1545,13 @@ NS_IMETHODIMP nsBulletListener::OnStartContainer(imgIRequest *aRequest,
|
|||
}
|
||||
|
||||
NS_IMETHODIMP nsBulletListener::OnDataAvailable(imgIRequest *aRequest,
|
||||
gfxIImageFrame *aFrame,
|
||||
PRBool aCurrentFrame,
|
||||
const nsIntRect *aRect)
|
||||
{
|
||||
if (!mFrame)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
return mFrame->OnDataAvailable(aRequest, aFrame, aRect);
|
||||
return mFrame->OnDataAvailable(aRequest, aCurrentFrame, aRect);
|
||||
}
|
||||
|
||||
NS_IMETHODIMP nsBulletListener::OnStopDecode(imgIRequest *aRequest,
|
||||
|
|
@ -1567,11 +1565,10 @@ NS_IMETHODIMP nsBulletListener::OnStopDecode(imgIRequest *aRequest,
|
|||
}
|
||||
|
||||
NS_IMETHODIMP nsBulletListener::FrameChanged(imgIContainer *aContainer,
|
||||
gfxIImageFrame *newframe,
|
||||
nsIntRect *dirtyRect)
|
||||
{
|
||||
if (!mFrame)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
return mFrame->FrameChanged(aContainer, newframe, dirtyRect);
|
||||
return mFrame->FrameChanged(aContainer, dirtyRect);
|
||||
}
|
||||
|
|
|
|||
|
|
@ -45,7 +45,6 @@
|
|||
|
||||
#include "imgIRequest.h"
|
||||
#include "imgIDecoderObserver.h"
|
||||
class gfxIImageFrame;
|
||||
|
||||
/**
|
||||
* A simple class that manages the layout and rendering of html bullets.
|
||||
|
|
@ -81,13 +80,12 @@ public:
|
|||
|
||||
NS_IMETHOD OnStartContainer(imgIRequest *aRequest, imgIContainer *aImage);
|
||||
NS_IMETHOD OnDataAvailable(imgIRequest *aRequest,
|
||||
gfxIImageFrame *aFrame,
|
||||
PRBool aCurrentFrame,
|
||||
const nsIntRect *aRect);
|
||||
NS_IMETHOD OnStopDecode(imgIRequest *aRequest,
|
||||
nsresult aStatus,
|
||||
const PRUnichar *aStatusArg);
|
||||
NS_IMETHOD FrameChanged(imgIContainer *aContainer,
|
||||
gfxIImageFrame *aNewframe,
|
||||
nsIntRect *aDirtyRect);
|
||||
|
||||
/* get list item text, without '.' */
|
||||
|
|
|
|||
|
|
@ -109,7 +109,6 @@
|
|||
#include "nsIServiceManager.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "imgIRequest.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsILookAndFeel.h"
|
||||
#include "nsLayoutCID.h"
|
||||
#include "nsWidgetsCID.h" // for NS_LOOKANDFEEL_CID
|
||||
|
|
|
|||
|
|
@ -47,7 +47,6 @@
|
|||
#include "nsPresContext.h"
|
||||
#include "nsIRenderingContext.h"
|
||||
#include "nsIPresShell.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsGkAtoms.h"
|
||||
#include "nsIDocument.h"
|
||||
#include "nsINodeInfo.h"
|
||||
|
|
@ -93,7 +92,6 @@
|
|||
#include "nsCSSFrameConstructor.h"
|
||||
#include "nsIPrefBranch2.h"
|
||||
#include "nsIPrefService.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsIDOMRange.h"
|
||||
|
||||
#include "nsIContentPolicy.h"
|
||||
|
|
@ -523,7 +521,7 @@ nsImageFrame::OnStartContainer(imgIRequest *aRequest, imgIContainer *aImage)
|
|||
|
||||
nsresult
|
||||
nsImageFrame::OnDataAvailable(imgIRequest *aRequest,
|
||||
gfxIImageFrame *aFrame,
|
||||
PRBool aCurrentFrame,
|
||||
const nsIntRect *aRect)
|
||||
{
|
||||
// XXX do we need to make sure that the reflow from the
|
||||
|
|
@ -555,16 +553,8 @@ nsImageFrame::OnDataAvailable(imgIRequest *aRequest,
|
|||
|
||||
// Don't invalidate if the current visible frame isn't the one the data is
|
||||
// from
|
||||
nsCOMPtr<imgIContainer> container;
|
||||
aRequest->GetImage(getter_AddRefs(container));
|
||||
if (container) {
|
||||
nsCOMPtr<gfxIImageFrame> currentFrame;
|
||||
container->GetCurrentFrame(getter_AddRefs(currentFrame));
|
||||
if (aFrame != currentFrame) {
|
||||
// just bail
|
||||
return NS_OK;
|
||||
}
|
||||
}
|
||||
if (!aCurrentFrame)
|
||||
return NS_OK;
|
||||
|
||||
#ifdef DEBUG_decode
|
||||
printf("Source rect (%d,%d,%d,%d) -> invalidate dest rect (%d,%d,%d,%d)\n",
|
||||
|
|
@ -634,9 +624,7 @@ nsImageFrame::OnStopDecode(imgIRequest *aRequest,
|
|||
}
|
||||
|
||||
nsresult
|
||||
nsImageFrame::FrameChanged(imgIContainer *aContainer,
|
||||
gfxIImageFrame *aNewFrame,
|
||||
nsIntRect *aDirtyRect)
|
||||
nsImageFrame::FrameChanged(imgIContainer *aContainer, nsIntRect *aDirtyRect)
|
||||
{
|
||||
if (!GetStyleVisibility()->IsVisible()) {
|
||||
return NS_OK;
|
||||
|
|
@ -1833,13 +1821,13 @@ NS_IMETHODIMP nsImageListener::OnStartContainer(imgIRequest *aRequest,
|
|||
}
|
||||
|
||||
NS_IMETHODIMP nsImageListener::OnDataAvailable(imgIRequest *aRequest,
|
||||
gfxIImageFrame *aFrame,
|
||||
PRBool aCurrentFrame,
|
||||
const nsIntRect *aRect)
|
||||
{
|
||||
if (!mFrame)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
return mFrame->OnDataAvailable(aRequest, aFrame, aRect);
|
||||
return mFrame->OnDataAvailable(aRequest, aCurrentFrame, aRect);
|
||||
}
|
||||
|
||||
NS_IMETHODIMP nsImageListener::OnStopDecode(imgIRequest *aRequest,
|
||||
|
|
@ -1853,13 +1841,12 @@ NS_IMETHODIMP nsImageListener::OnStopDecode(imgIRequest *aRequest,
|
|||
}
|
||||
|
||||
NS_IMETHODIMP nsImageListener::FrameChanged(imgIContainer *aContainer,
|
||||
gfxIImageFrame *newframe,
|
||||
nsIntRect * dirtyRect)
|
||||
{
|
||||
if (!mFrame)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
return mFrame->FrameChanged(aContainer, newframe, dirtyRect);
|
||||
return mFrame->FrameChanged(aContainer, dirtyRect);
|
||||
}
|
||||
|
||||
static PRBool
|
||||
|
|
|
|||
|
|
@ -73,13 +73,12 @@ public:
|
|||
NS_DECL_ISUPPORTS
|
||||
// imgIDecoderObserver (override nsStubImageDecoderObserver)
|
||||
NS_IMETHOD OnStartContainer(imgIRequest *aRequest, imgIContainer *aImage);
|
||||
NS_IMETHOD OnDataAvailable(imgIRequest *aRequest, gfxIImageFrame *aFrame,
|
||||
NS_IMETHOD OnDataAvailable(imgIRequest *aRequest, PRBool aCurrentFrame,
|
||||
const nsIntRect *aRect);
|
||||
NS_IMETHOD OnStopDecode(imgIRequest *aRequest, nsresult status,
|
||||
const PRUnichar *statusArg);
|
||||
// imgIContainerObserver (override nsStubImageDecoderObserver)
|
||||
NS_IMETHOD FrameChanged(imgIContainer *aContainer, gfxIImageFrame *newframe,
|
||||
nsIntRect * dirtyRect);
|
||||
NS_IMETHOD FrameChanged(imgIContainer *aContainer, nsIntRect * dirtyRect);
|
||||
|
||||
void SetFrame(nsImageFrame *frame) { mFrame = frame; }
|
||||
|
||||
|
|
@ -218,15 +217,12 @@ protected:
|
|||
protected:
|
||||
friend class nsImageListener;
|
||||
nsresult OnStartContainer(imgIRequest *aRequest, imgIContainer *aImage);
|
||||
nsresult OnDataAvailable(imgIRequest *aRequest,
|
||||
gfxIImageFrame *aFrame,
|
||||
nsresult OnDataAvailable(imgIRequest *aRequest, PRBool aCurrentFrame,
|
||||
const nsIntRect *rect);
|
||||
nsresult OnStopDecode(imgIRequest *aRequest,
|
||||
nsresult aStatus,
|
||||
const PRUnichar *aStatusArg);
|
||||
nsresult FrameChanged(imgIContainer *aContainer,
|
||||
gfxIImageFrame *aNewframe,
|
||||
nsIntRect *aDirtyRect);
|
||||
nsresult FrameChanged(imgIContainer *aContainer, nsIntRect *aDirtyRect);
|
||||
|
||||
private:
|
||||
// random helpers
|
||||
|
|
|
|||
|
|
@ -21,6 +21,6 @@
|
|||
|
||||
<rect width="100%" height="100%" fill="lime"/>
|
||||
<rect width="183" height="183" fill="red"/>
|
||||
<image width="183" height="183" xlink:href="%2FxhBQAAAAxJREFUGFdjYPjPAAACAgEA0dNGRAAAAABJRU5ErkJggg%3D%3D"/>
|
||||
<image style="image-rendering: -moz-crisp-edges" width="183" height="183" xlink:href="%2FxhBQAAAAxJREFUGFdjYPjPAAACAgEA0dNGRAAAAABJRU5ErkJggg%3D%3D"/>
|
||||
|
||||
</svg>
|
||||
|
|
|
|||
|
Before Width: | Height: | Size: 1 KiB After Width: | Height: | Size: 1.1 KiB |
|
|
@ -37,7 +37,6 @@
|
|||
#include "nsSVGPathGeometryFrame.h"
|
||||
#include "nsIDOMSVGMatrix.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsStubImageDecoderObserver.h"
|
||||
#include "nsImageLoadingContent.h"
|
||||
#include "nsIDOMSVGImageElement.h"
|
||||
|
|
@ -47,7 +46,6 @@
|
|||
#include "nsSVGMatrix.h"
|
||||
#include "gfxContext.h"
|
||||
#include "nsIInterfaceRequestorUtils.h"
|
||||
#include "nsIImage.h"
|
||||
|
||||
class nsSVGImageFrame;
|
||||
|
||||
|
|
@ -61,8 +59,7 @@ public:
|
|||
NS_IMETHOD OnStopDecode(imgIRequest *aRequest, nsresult status,
|
||||
const PRUnichar *statusArg);
|
||||
// imgIContainerObserver (override nsStubImageDecoderObserver)
|
||||
NS_IMETHOD FrameChanged(imgIContainer *aContainer, gfxIImageFrame *newframe,
|
||||
nsRect * dirtyRect);
|
||||
NS_IMETHOD FrameChanged(imgIContainer *aContainer, nsIntRect * dirtyRect);
|
||||
// imgIContainerObserver (override nsStubImageDecoderObserver)
|
||||
NS_IMETHOD OnStartContainer(imgIRequest *aRequest,
|
||||
imgIContainer *aContainer);
|
||||
|
|
@ -243,16 +240,15 @@ nsSVGImageFrame::PaintSVG(nsSVGRenderState *aContext,
|
|||
currentRequest->GetImage(getter_AddRefs(mImageContainer));
|
||||
}
|
||||
|
||||
nsCOMPtr<gfxIImageFrame> currentFrame;
|
||||
nsRefPtr<gfxASurface> currentFrame;
|
||||
if (mImageContainer)
|
||||
mImageContainer->GetCurrentFrame(getter_AddRefs(currentFrame));
|
||||
|
||||
nsRefPtr<gfxPattern> thebesPattern = nsnull;
|
||||
if (currentFrame) {
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(currentFrame));
|
||||
|
||||
img->GetPattern(getter_AddRefs(thebesPattern));
|
||||
}
|
||||
// We need to wrap the surface in a pattern to have somewhere to set the
|
||||
// graphics filter.
|
||||
nsRefPtr<gfxPattern> thebesPattern;
|
||||
if (currentFrame)
|
||||
thebesPattern = new gfxPattern(currentFrame);
|
||||
|
||||
if (thebesPattern) {
|
||||
|
||||
|
|
@ -279,8 +275,8 @@ nsSVGImageFrame::PaintSVG(nsSVGRenderState *aContext,
|
|||
}
|
||||
|
||||
PRInt32 nativeWidth, nativeHeight;
|
||||
currentFrame->GetWidth(&nativeWidth);
|
||||
currentFrame->GetHeight(&nativeHeight);
|
||||
mImageContainer->GetWidth(&nativeWidth);
|
||||
mImageContainer->GetHeight(&nativeHeight);
|
||||
|
||||
nsSVGUtils::CompositePatternMatrix(gfx, thebesPattern, fini, nativeWidth, nativeHeight, opacity);
|
||||
|
||||
|
|
@ -384,8 +380,7 @@ NS_IMETHODIMP nsSVGImageListener::OnStopDecode(imgIRequest *aRequest,
|
|||
}
|
||||
|
||||
NS_IMETHODIMP nsSVGImageListener::FrameChanged(imgIContainer *aContainer,
|
||||
gfxIImageFrame *newframe,
|
||||
nsRect * dirtyRect)
|
||||
nsIntRect * dirtyRect)
|
||||
{
|
||||
if (!mFrame)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
|
|
|||
|
|
@ -58,7 +58,6 @@
|
|||
#include "nsPresContext.h"
|
||||
#include "nsIRenderingContext.h"
|
||||
#include "nsIPresShell.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsIDocument.h"
|
||||
#include "nsIHTMLDocument.h"
|
||||
#include "nsStyleConsts.h"
|
||||
|
|
@ -543,7 +542,6 @@ NS_IMETHODIMP nsImageBoxFrame::OnStopDecode(imgIRequest *request,
|
|||
}
|
||||
|
||||
NS_IMETHODIMP nsImageBoxFrame::FrameChanged(imgIContainer *container,
|
||||
gfxIImageFrame *newframe,
|
||||
nsIntRect *dirtyRect)
|
||||
{
|
||||
nsBoxLayoutState state(PresContext());
|
||||
|
|
@ -591,12 +589,11 @@ NS_IMETHODIMP nsImageBoxListener::OnStopDecode(imgIRequest *request,
|
|||
}
|
||||
|
||||
NS_IMETHODIMP nsImageBoxListener::FrameChanged(imgIContainer *container,
|
||||
gfxIImageFrame *newframe,
|
||||
nsIntRect *dirtyRect)
|
||||
{
|
||||
if (!mFrame)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
return mFrame->FrameChanged(container, newframe, dirtyRect);
|
||||
return mFrame->FrameChanged(container, dirtyRect);
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -59,8 +59,7 @@ public:
|
|||
NS_IMETHOD OnStopDecode(imgIRequest *request, nsresult status,
|
||||
const PRUnichar *statusArg);
|
||||
// imgIContainerObserver (override nsStubImageDecoderObserver)
|
||||
NS_IMETHOD FrameChanged(imgIContainer *container, gfxIImageFrame *newframe,
|
||||
nsIntRect *dirtyRect);
|
||||
NS_IMETHOD FrameChanged(imgIContainer *container, nsIntRect *dirtyRect);
|
||||
|
||||
void SetFrame(nsImageBoxFrame *frame) { mFrame = frame; }
|
||||
|
||||
|
|
@ -121,9 +120,7 @@ public:
|
|||
NS_IMETHOD OnStopDecode(imgIRequest *request,
|
||||
nsresult status,
|
||||
const PRUnichar *statusArg);
|
||||
NS_IMETHOD FrameChanged(imgIContainer *container,
|
||||
gfxIImageFrame *newframe,
|
||||
nsIntRect *dirtyRect);
|
||||
NS_IMETHOD FrameChanged(imgIContainer *container, nsIntRect *dirtyRect);
|
||||
|
||||
virtual ~nsImageBoxFrame();
|
||||
|
||||
|
|
|
|||
|
|
@ -2173,11 +2173,11 @@ nsTreeBodyFrame::GetImage(PRInt32 aRowIndex, nsTreeColumn* aCol, PRBool aUseCont
|
|||
imgIRequest *imgReq = entry.request;
|
||||
imgReq->GetImageStatus(&status);
|
||||
imgReq->GetImage(aResult); // We hand back the image here. The GetImage call addrefs *aResult.
|
||||
PRUint32 numFrames = 1;
|
||||
PRBool animated = PR_FALSE;
|
||||
if (*aResult)
|
||||
(*aResult)->GetNumFrames(&numFrames);
|
||||
(*aResult)->GetAnimated(&animated);
|
||||
|
||||
if ((!(status & imgIRequest::STATUS_LOAD_COMPLETE)) || numFrames > 1) {
|
||||
if ((!(status & imgIRequest::STATUS_LOAD_COMPLETE)) || animated) {
|
||||
// We either aren't done loading, or we're animating. Add our row as a listener for invalidations.
|
||||
nsCOMPtr<imgIDecoderObserver> obs;
|
||||
imgReq->GetDecoderObserver(getter_AddRefs(obs));
|
||||
|
|
|
|||
|
|
@ -64,7 +64,7 @@ NS_IMETHODIMP nsTreeImageListener::OnStartContainer(imgIRequest *aRequest,
|
|||
}
|
||||
|
||||
NS_IMETHODIMP nsTreeImageListener::OnDataAvailable(imgIRequest *aRequest,
|
||||
gfxIImageFrame *aFrame,
|
||||
PRBool aCurrentFrame,
|
||||
const nsIntRect *aRect)
|
||||
{
|
||||
Invalidate();
|
||||
|
|
@ -72,7 +72,6 @@ NS_IMETHODIMP nsTreeImageListener::OnDataAvailable(imgIRequest *aRequest,
|
|||
}
|
||||
|
||||
NS_IMETHODIMP nsTreeImageListener::FrameChanged(imgIContainer *aContainer,
|
||||
gfxIImageFrame *newframe,
|
||||
nsIntRect *dirtyRect)
|
||||
{
|
||||
Invalidate();
|
||||
|
|
|
|||
|
|
@ -72,11 +72,10 @@ public:
|
|||
NS_DECL_ISUPPORTS
|
||||
// imgIDecoderObserver (override nsStubImageDecoderObserver)
|
||||
NS_IMETHOD OnStartContainer(imgIRequest *aRequest, imgIContainer *aImage);
|
||||
NS_IMETHOD OnDataAvailable(imgIRequest *aRequest, gfxIImageFrame *aFrame,
|
||||
NS_IMETHOD OnDataAvailable(imgIRequest *aRequest, PRBool aCurrentFrame,
|
||||
const nsIntRect *aRect);
|
||||
// imgIContainerObserver (override nsStubImageDecoderObserver)
|
||||
NS_IMETHOD FrameChanged(imgIContainer *aContainer, gfxIImageFrame *newframe,
|
||||
nsIntRect *dirtyRect);
|
||||
NS_IMETHOD FrameChanged(imgIContainer *aContainer, nsIntRect *dirtyRect);
|
||||
|
||||
NS_IMETHOD AddCell(PRInt32 aIndex, nsITreeColumn* aCol);
|
||||
|
||||
|
|
|
|||
|
|
@ -47,6 +47,7 @@
|
|||
#define IMG_BUILD_xbm 1
|
||||
#endif
|
||||
|
||||
#include "nsIDeviceContext.h"
|
||||
#include "nsIGenericFactory.h"
|
||||
#include "nsIModule.h"
|
||||
#include "nsICategoryManager.h"
|
||||
|
|
@ -206,7 +207,7 @@ static const nsModuleComponentInfo components[] =
|
|||
imgLoaderConstructor, },
|
||||
{ "image container",
|
||||
NS_IMGCONTAINER_CID,
|
||||
"@mozilla.org/image/container;1",
|
||||
"@mozilla.org/image/container;2",
|
||||
imgContainerConstructor, },
|
||||
{ "image loader",
|
||||
NS_IMGLOADER_CID,
|
||||
|
|
@ -313,6 +314,12 @@ static const nsModuleComponentInfo components[] =
|
|||
static nsresult
|
||||
imglib_Initialize(nsIModule* aSelf)
|
||||
{
|
||||
// Hack: We need the gfx module to be initialized because we use gfxPlatform
|
||||
// in imgFrame. Request something from the gfx module to ensure that
|
||||
// everything's set up for us.
|
||||
nsCOMPtr<nsIDeviceContext> devctx =
|
||||
do_CreateInstance("@mozilla.org/gfx/devicecontext;1");
|
||||
|
||||
imgLoader::InitCache();
|
||||
return NS_OK;
|
||||
}
|
||||
|
|
|
|||
|
|
@ -51,7 +51,6 @@
|
|||
#include "nsIInterfaceRequestorUtils.h"
|
||||
|
||||
#include "imgILoad.h"
|
||||
#include "nsIImage.h"
|
||||
|
||||
#include "prlog.h"
|
||||
|
||||
|
|
@ -89,11 +88,7 @@ NS_IMETHODIMP nsBMPDecoder::Init(imgILoad *aLoad)
|
|||
mObserver = do_QueryInterface(aLoad);
|
||||
|
||||
nsresult rv;
|
||||
mImage = do_CreateInstance("@mozilla.org/image/container;1", &rv);
|
||||
if (NS_FAILED(rv))
|
||||
return rv;
|
||||
|
||||
mFrame = do_CreateInstance("@mozilla.org/gfx/image/frame;2", &rv);
|
||||
mImage = do_CreateInstance("@mozilla.org/image/container;2", &rv);
|
||||
if (NS_FAILED(rv))
|
||||
return rv;
|
||||
|
||||
|
|
@ -106,13 +101,12 @@ NS_IMETHODIMP nsBMPDecoder::Close()
|
|||
|
||||
mImage->DecodingComplete();
|
||||
if (mObserver) {
|
||||
mObserver->OnStopFrame(nsnull, mFrame);
|
||||
mObserver->OnStopFrame(nsnull, 0);
|
||||
mObserver->OnStopContainer(nsnull, mImage);
|
||||
mObserver->OnStopDecode(nsnull, NS_OK, nsnull);
|
||||
mObserver = nsnull;
|
||||
}
|
||||
mImage = nsnull;
|
||||
mFrame = nsnull;
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
|
@ -266,8 +260,10 @@ NS_METHOD nsBMPDecoder::ProcessData(const char* aBuffer, PRUint32 aCount)
|
|||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
mOldLine = mCurLine = real_height;
|
||||
|
||||
PRUint32 imageLength;
|
||||
if ((mBIH.compression == BI_RLE8) || (mBIH.compression == BI_RLE4)) {
|
||||
rv = mFrame->Init(0, 0, mBIH.width, real_height, RLE_GFXFORMAT_ALPHA, 24);
|
||||
rv = mImage->AppendFrame(0, 0, mBIH.width, real_height, gfxASurface::ImageFormatARGB32,
|
||||
(PRUint8**)&mImageData, &imageLength);
|
||||
} else {
|
||||
// mRow is not used for RLE encoded images
|
||||
mRow = (PRUint8*)malloc((mBIH.width * mBIH.bpp)/8 + 4);
|
||||
|
|
@ -277,12 +273,10 @@ NS_METHOD nsBMPDecoder::ProcessData(const char* aBuffer, PRUint32 aCount)
|
|||
if (!mRow) {
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
}
|
||||
rv = mFrame->Init(0, 0, mBIH.width, real_height, BMP_GFXFORMAT, 24);
|
||||
rv = mImage->AppendFrame(0, 0, mBIH.width, real_height, gfxASurface::ImageFormatRGB24,
|
||||
(PRUint8**)&mImageData, &imageLength);
|
||||
}
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
PRUint32 imageLength;
|
||||
mFrame->GetImageData((PRUint8**)&mImageData, &imageLength);
|
||||
if (!mImageData)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
|
|
@ -297,9 +291,7 @@ NS_METHOD nsBMPDecoder::ProcessData(const char* aBuffer, PRUint32 aCount)
|
|||
memset(mImageData, 0, imageLength);
|
||||
}
|
||||
|
||||
rv = mImage->AppendFrame(mFrame);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
mObserver->OnStartFrame(nsnull, mFrame);
|
||||
mObserver->OnStartFrame(nsnull, 0);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
}
|
||||
PRUint8 bpc; // bytes per color
|
||||
|
|
@ -603,14 +595,11 @@ NS_METHOD nsBMPDecoder::ProcessData(const char* aBuffer, PRUint32 aCount)
|
|||
nsIntRect r(0, mBIH.height < 0 ? -mBIH.height - mOldLine : mCurLine,
|
||||
mBIH.width, rows);
|
||||
|
||||
// Tell the image that it's data has been updated
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(mFrame, &rv));
|
||||
// Tell the image that its data has been updated
|
||||
rv = mImage->FrameUpdated(0, r);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
rv = img->ImageUpdated(nsnull, nsImageUpdateFlags_kBitsChanged, &r);
|
||||
if (NS_FAILED(rv))
|
||||
return rv;
|
||||
|
||||
mObserver->OnDataAvailable(nsnull, mFrame, &r);
|
||||
mObserver->OnDataAvailable(nsnull, PR_TRUE, &r);
|
||||
mOldLine = mCurLine;
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -43,7 +43,6 @@
|
|||
#include "imgIDecoder.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "imgIDecoderObserver.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "gfxColor.h"
|
||||
|
||||
#define NS_BMPDECODER_CID \
|
||||
|
|
@ -118,9 +117,6 @@ struct bitFields {
|
|||
#endif
|
||||
|
||||
#define USE_RGB
|
||||
#define BMP_GFXFORMAT gfxIFormats::RGB
|
||||
#define RLE_GFXFORMAT_ALPHA gfxIFormats::RGB_A1
|
||||
#define GFXBYTESPERPIXEL 4
|
||||
|
||||
// BMPINFOHEADER.compression defines
|
||||
#define BI_RLE8 1
|
||||
|
|
@ -173,7 +169,6 @@ private:
|
|||
nsCOMPtr<imgIDecoderObserver> mObserver;
|
||||
|
||||
nsCOMPtr<imgIContainer> mImage;
|
||||
nsCOMPtr<gfxIImageFrame> mFrame;
|
||||
|
||||
PRUint32 mPos;
|
||||
|
||||
|
|
|
|||
|
|
@ -51,7 +51,6 @@
|
|||
#include "imgILoad.h"
|
||||
#include "nsIInterfaceRequestor.h"
|
||||
#include "nsIInterfaceRequestorUtils.h"
|
||||
#include "nsIImage.h"
|
||||
|
||||
#include "nsIProperties.h"
|
||||
#include "nsISupportsPrimitives.h"
|
||||
|
|
@ -93,39 +92,30 @@ NS_IMETHODIMP nsICODecoder::Init(imgILoad *aLoad)
|
|||
{
|
||||
mObserver = do_QueryInterface(aLoad);
|
||||
|
||||
mImage = do_CreateInstance("@mozilla.org/image/container;1");
|
||||
mImage = do_CreateInstance("@mozilla.org/image/container;2");
|
||||
if (!mImage)
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
|
||||
mFrame = do_CreateInstance("@mozilla.org/gfx/image/frame;2");
|
||||
if (!mFrame)
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
|
||||
return aLoad->SetImage(mImage);
|
||||
}
|
||||
|
||||
NS_IMETHODIMP nsICODecoder::Close()
|
||||
{
|
||||
// Tell the image that it's data has been updated
|
||||
// This should be a mFrame function, so that we don't have to query for interface...
|
||||
nsIntRect r(0, 0, mDirEntry.mWidth, mDirEntry.mHeight);
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(mFrame));
|
||||
nsresult rv = NS_OK;
|
||||
if (img)
|
||||
rv = img->ImageUpdated(nsnull, nsImageUpdateFlags_kBitsChanged, &r);
|
||||
nsresult rv = mImage->FrameUpdated(0, r);
|
||||
|
||||
mImage->DecodingComplete();
|
||||
|
||||
if (mObserver) {
|
||||
mObserver->OnDataAvailable(nsnull, mFrame, &r);
|
||||
mObserver->OnStopFrame(nsnull, mFrame);
|
||||
mObserver->OnStopContainer(nsnull, mImage);
|
||||
mObserver->OnDataAvailable(nsnull, PR_TRUE, &r);
|
||||
mObserver->OnStopFrame(nsnull, 0);
|
||||
mObserver->OnStopContainer(nsnull, 0);
|
||||
mObserver->OnStopDecode(nsnull, NS_OK, nsnull);
|
||||
mObserver = nsnull;
|
||||
}
|
||||
|
||||
mImage = nsnull;
|
||||
mFrame = nsnull;
|
||||
|
||||
mPos = 0;
|
||||
|
||||
|
|
@ -304,15 +294,13 @@ nsresult nsICODecoder::ProcessData(const char* aBuffer, PRUint32 aCount) {
|
|||
if (!mRow)
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
|
||||
rv = mFrame->Init(0, 0, mDirEntry.mWidth, mDirEntry.mHeight, GFXFORMATALPHA8, 24);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
rv = mImage->AppendFrame(mFrame);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
mObserver->OnStartFrame(nsnull, mFrame);
|
||||
PRUint32 imageLength;
|
||||
rv = mImage->AppendFrame(0, 0, mDirEntry.mWidth, mDirEntry.mHeight,
|
||||
gfxASurface::ImageFormatARGB32, (PRUint8**)&mImageData, &imageLength);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
PRUint32 imageLength;
|
||||
mFrame->GetImageData((PRUint8**)&mImageData, &imageLength);
|
||||
mObserver->OnStartFrame(nsnull, 0);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
}
|
||||
|
||||
if (mColors && (mPos >= mImageOffset + BITMAPINFOSIZE) &&
|
||||
|
|
|
|||
|
|
@ -44,20 +44,12 @@
|
|||
#include "imgIDecoder.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "imgIDecoderObserver.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsBMPDecoder.h"
|
||||
|
||||
// {CB3EDE1A-0FA5-4e27-AAFE-0F7801E5A1F1}
|
||||
#define NS_ICODECODER_CID \
|
||||
{ 0xcb3ede1a, 0xfa5, 0x4e27, { 0xaa, 0xfe, 0xf, 0x78, 0x1, 0xe5, 0xa1, 0xf1 } }
|
||||
|
||||
#if defined(XP_WIN) || defined(XP_OS2) || defined(XP_BEOS) || defined(MOZ_WIDGET_PHOTON)
|
||||
#define GFXFORMATALPHA8 gfxIFormats::BGR_A8
|
||||
#else
|
||||
#define USE_RGBA1
|
||||
#define GFXFORMATALPHA8 gfxIFormats::RGB_A8
|
||||
#endif
|
||||
|
||||
struct IconDirEntry
|
||||
{
|
||||
PRUint8 mWidth;
|
||||
|
|
@ -103,7 +95,6 @@ private:
|
|||
private:
|
||||
nsCOMPtr<imgIDecoderObserver> mObserver;
|
||||
nsCOMPtr<imgIContainer> mImage;
|
||||
nsCOMPtr<gfxIImageFrame> mFrame;
|
||||
|
||||
PRUint32 mPos;
|
||||
PRUint16 mNumIcons;
|
||||
|
|
|
|||
|
|
@ -74,7 +74,6 @@ mailing address.
|
|||
#include <stddef.h>
|
||||
#include "prmem.h"
|
||||
|
||||
#include "nsIImage.h"
|
||||
#include "nsIInterfaceRequestorUtils.h"
|
||||
|
||||
#include "nsGIFDecoder2.h"
|
||||
|
|
@ -142,7 +141,7 @@ NS_IMETHODIMP nsGIFDecoder2::Init(imgILoad *aLoad)
|
|||
{
|
||||
mObserver = do_QueryInterface(aLoad);
|
||||
|
||||
mImageContainer = do_CreateInstance("@mozilla.org/image/container;1");
|
||||
mImageContainer = do_CreateInstance("@mozilla.org/image/container;2");
|
||||
aLoad->SetImage(mImageContainer);
|
||||
|
||||
// Start with the version (GIF89a|GIF87a)
|
||||
|
|
@ -161,8 +160,7 @@ NS_IMETHODIMP nsGIFDecoder2::Init(imgILoad *aLoad)
|
|||
/* void close (); */
|
||||
NS_IMETHODIMP nsGIFDecoder2::Close()
|
||||
{
|
||||
if (mImageFrame)
|
||||
EndImageFrame();
|
||||
EndImageFrame();
|
||||
EndGIF();
|
||||
|
||||
PR_FREEIF(mGIFStruct.local_colormap);
|
||||
|
|
@ -205,8 +203,7 @@ nsGIFDecoder2::FlushImageData(PRUint32 fromRow, PRUint32 rows)
|
|||
nsIntRect r(0, fromRow, mGIFStruct.width, rows);
|
||||
|
||||
// Update image
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(mImageFrame));
|
||||
nsresult rv = img->ImageUpdated(nsnull, nsImageUpdateFlags_kBitsChanged, &r);
|
||||
nsresult rv = mImageContainer->FrameUpdated(mGIFStruct.images_decoded, r);
|
||||
if (NS_FAILED(rv)) {
|
||||
return rv;
|
||||
}
|
||||
|
|
@ -214,8 +211,10 @@ nsGIFDecoder2::FlushImageData(PRUint32 fromRow, PRUint32 rows)
|
|||
// Offset to the frame position
|
||||
// Only notify observer(s) for first frame
|
||||
if (!mGIFStruct.images_decoded && mObserver) {
|
||||
PRUint32 imgCurFrame;
|
||||
mImageContainer->GetCurrentFrameIndex(&imgCurFrame);
|
||||
r.y += mGIFStruct.y_offset;
|
||||
mObserver->OnDataAvailable(nsnull, mImageFrame, &r);
|
||||
mObserver->OnDataAvailable(nsnull, imgCurFrame == PRUint32(mGIFStruct.images_decoded), &r);
|
||||
}
|
||||
return NS_OK;
|
||||
}
|
||||
|
|
@ -251,7 +250,7 @@ nsresult nsGIFDecoder2::ProcessData(unsigned char *data, PRUint32 count, PRUint3
|
|||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
// Flushing is only needed for first frame
|
||||
if (!mGIFStruct.images_decoded && mImageFrame) {
|
||||
if (!mGIFStruct.images_decoded) {
|
||||
rv = FlushImageData();
|
||||
mLastFlushedRow = mCurrentRow;
|
||||
mLastFlushedPass = mCurrentPass;
|
||||
|
|
@ -324,8 +323,6 @@ void nsGIFDecoder2::EndGIF()
|
|||
//******************************************************************************
|
||||
void nsGIFDecoder2::BeginImageFrame(gfx_depth aDepth)
|
||||
{
|
||||
mImageFrame = nsnull; // clear out our current frame reference
|
||||
|
||||
if (!mGIFStruct.images_decoded) {
|
||||
// Send a onetime OnDataAvailable (Display Refresh) for the first frame
|
||||
// if it has a y-axis offset. Otherwise, the area may never be refreshed
|
||||
|
|
@ -333,41 +330,46 @@ void nsGIFDecoder2::BeginImageFrame(gfx_depth aDepth)
|
|||
if (mGIFStruct.y_offset > 0) {
|
||||
PRInt32 imgWidth;
|
||||
mImageContainer->GetWidth(&imgWidth);
|
||||
PRUint32 imgCurFrame;
|
||||
mImageContainer->GetCurrentFrameIndex(&imgCurFrame);
|
||||
nsIntRect r(0, 0, imgWidth, mGIFStruct.y_offset);
|
||||
mObserver->OnDataAvailable(nsnull, mImageFrame, &r);
|
||||
mObserver->OnDataAvailable(nsnull, imgCurFrame == PRUint32(mGIFStruct.images_decoded), &r);
|
||||
}
|
||||
}
|
||||
|
||||
PRUint32 imageDataLength;
|
||||
nsresult rv;
|
||||
gfxASurface::gfxImageFormat format;
|
||||
if (mGIFStruct.is_transparent)
|
||||
format = gfxASurface::ImageFormatARGB32;
|
||||
else
|
||||
format = gfxASurface::ImageFormatRGB24;
|
||||
|
||||
// Use correct format, RGB for first frame, PAL for following frames
|
||||
// and include transparency to allow for optimization of opaque images
|
||||
gfx_format format;
|
||||
if (mGIFStruct.images_decoded) {
|
||||
// Image data is stored with original depth and palette
|
||||
format = mGIFStruct.is_transparent ? gfxIFormats::PAL_A1 : gfxIFormats::PAL;
|
||||
rv = mImageContainer->AppendPalettedFrame(mGIFStruct.x_offset, mGIFStruct.y_offset,
|
||||
mGIFStruct.width, mGIFStruct.height,
|
||||
format, aDepth, &mImageData, &imageDataLength,
|
||||
&mColormap, &mColormapSize);
|
||||
|
||||
|
||||
} else {
|
||||
// Regardless of depth of input, image is decoded into 24bit RGB
|
||||
format = mGIFStruct.is_transparent ? gfxIFormats::RGB_A1 : gfxIFormats::RGB;
|
||||
aDepth = 24;
|
||||
rv = mImageContainer->AppendFrame(mGIFStruct.x_offset, mGIFStruct.y_offset,
|
||||
mGIFStruct.width, mGIFStruct.height,
|
||||
format, &mImageData, &imageDataLength);
|
||||
}
|
||||
|
||||
// initialize the frame and append it to the container
|
||||
mImageFrame = do_CreateInstance("@mozilla.org/gfx/image/frame;2");
|
||||
if (!mImageFrame || NS_FAILED(mImageFrame->Init(
|
||||
mGIFStruct.x_offset, mGIFStruct.y_offset,
|
||||
mGIFStruct.width, mGIFStruct.height, format, aDepth))) {
|
||||
mImageFrame = 0;
|
||||
if (NS_FAILED(rv))
|
||||
return;
|
||||
}
|
||||
|
||||
mImageFrame->SetFrameDisposalMethod(mGIFStruct.disposal_method);
|
||||
if (!mGIFStruct.images_decoded)
|
||||
mImageContainer->AppendFrame(mImageFrame);
|
||||
mImageContainer->SetFrameDisposalMethod(mGIFStruct.images_decoded,
|
||||
mGIFStruct.disposal_method);
|
||||
|
||||
if (mObserver)
|
||||
mObserver->OnStartFrame(nsnull, mImageFrame);
|
||||
|
||||
PRUint32 imageDataLength;
|
||||
mImageFrame->GetImageData(&mImageData, &imageDataLength);
|
||||
mObserver->OnStartFrame(nsnull, mGIFStruct.images_decoded);
|
||||
}
|
||||
|
||||
|
||||
|
|
@ -384,20 +386,23 @@ void nsGIFDecoder2::EndImageFrame()
|
|||
// This will clear the remaining bits of the placeholder. (Bug 37589)
|
||||
const PRUint32 realFrameHeight = mGIFStruct.height + mGIFStruct.y_offset;
|
||||
if (realFrameHeight < mGIFStruct.screen_height) {
|
||||
PRUint32 imgCurFrame;
|
||||
mImageContainer->GetCurrentFrameIndex(&imgCurFrame);
|
||||
nsIntRect r(0, realFrameHeight,
|
||||
mGIFStruct.screen_width,
|
||||
mGIFStruct.screen_height - realFrameHeight);
|
||||
mObserver->OnDataAvailable(nsnull, mImageFrame, &r);
|
||||
mObserver->OnDataAvailable(nsnull, imgCurFrame == PRUint32(mGIFStruct.images_decoded), &r);
|
||||
}
|
||||
// This transparency check is only valid for first frame
|
||||
if (mGIFStruct.is_transparent && !mSawTransparency) {
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(mImageFrame));
|
||||
img->SetHasNoAlpha();
|
||||
mImageContainer->SetFrameHasNoAlpha(mGIFStruct.images_decoded);
|
||||
}
|
||||
}
|
||||
mCurrentRow = mLastFlushedRow = -1;
|
||||
mCurrentPass = mLastFlushedPass = 0;
|
||||
|
||||
PRUint32 curframe = mGIFStruct.images_decoded;
|
||||
|
||||
// Only add frame if we have any rows at all
|
||||
if (mGIFStruct.rows_remaining != mGIFStruct.height) {
|
||||
if (mGIFStruct.rows_remaining && mGIFStruct.images_decoded) {
|
||||
|
|
@ -410,18 +415,14 @@ void nsGIFDecoder2::EndImageFrame()
|
|||
// image data, at least according to the spec, but we delay in setting the
|
||||
// timeout for the image until here to help ensure that we have the whole
|
||||
// image frame decoded before we go off and try to display another frame.
|
||||
mImageFrame->SetTimeout(mGIFStruct.delay_time);
|
||||
if (mGIFStruct.images_decoded)
|
||||
mImageContainer->AppendFrame(mImageFrame);
|
||||
mImageContainer->SetFrameTimeout(mGIFStruct.images_decoded, mGIFStruct.delay_time);
|
||||
mImageContainer->EndFrameDecode(mGIFStruct.images_decoded);
|
||||
mGIFStruct.images_decoded++;
|
||||
|
||||
mGIFStruct.images_decoded++;
|
||||
}
|
||||
|
||||
if (mObserver)
|
||||
mObserver->OnStopFrame(nsnull, mImageFrame);
|
||||
|
||||
// Release reference to this frame
|
||||
mImageFrame = nsnull;
|
||||
mObserver->OnStopFrame(nsnull, curframe);
|
||||
|
||||
// Reset the transparent pixel
|
||||
if (mOldColor) {
|
||||
|
|
@ -497,7 +498,7 @@ PRUint32 nsGIFDecoder2::OutputRow()
|
|||
if (drow_end > drow_start) {
|
||||
// irow is the current row filled
|
||||
for (int r = drow_start; r <= drow_end; r++) {
|
||||
if (r != mGIFStruct.irow) {
|
||||
if (r != int(mGIFStruct.irow)) {
|
||||
memcpy(mImageData + (r * bpr), rowp, bpr);
|
||||
}
|
||||
}
|
||||
|
|
@ -1060,7 +1061,7 @@ nsresult nsGIFDecoder2::GifWrite(const PRUint8 *buf, PRUint32 len)
|
|||
BeginImageFrame(realDepth);
|
||||
|
||||
// handle allocation error
|
||||
if (!mImageFrame) {
|
||||
if (!mImageData) {
|
||||
mGIFStruct.state = gif_error;
|
||||
break;
|
||||
}
|
||||
|
|
@ -1086,17 +1087,12 @@ nsresult nsGIFDecoder2::GifWrite(const PRUint8 *buf, PRUint32 len)
|
|||
if (q[8] & 0x80) /* has a local colormap? */
|
||||
{
|
||||
mGIFStruct.local_colormap_size = 1 << depth;
|
||||
PRUint32 paletteSize;
|
||||
if (mGIFStruct.images_decoded) {
|
||||
// Copy directly into the palette of current frame,
|
||||
// by pointing mColormap to that palette.
|
||||
mImageFrame->GetPaletteData(&mColormap, &paletteSize);
|
||||
} else {
|
||||
if (!mGIFStruct.images_decoded) {
|
||||
// First frame has local colormap, allocate space for it
|
||||
// as the image frame doesn't have its own palette
|
||||
paletteSize = sizeof(PRUint32) << realDepth;
|
||||
mColormapSize = sizeof(PRUint32) << realDepth;
|
||||
if (!mGIFStruct.local_colormap) {
|
||||
mGIFStruct.local_colormap = (PRUint32*)PR_MALLOC(paletteSize);
|
||||
mGIFStruct.local_colormap = (PRUint32*)PR_MALLOC(mColormapSize);
|
||||
if (!mGIFStruct.local_colormap) {
|
||||
mGIFStruct.state = gif_oom;
|
||||
break;
|
||||
|
|
@ -1105,9 +1101,9 @@ nsresult nsGIFDecoder2::GifWrite(const PRUint8 *buf, PRUint32 len)
|
|||
mColormap = mGIFStruct.local_colormap;
|
||||
}
|
||||
const PRUint32 size = 3 << depth;
|
||||
if (paletteSize > size) {
|
||||
if (mColormapSize > size) {
|
||||
// Clear the notfilled part of the colormap
|
||||
memset(((PRUint8*)mColormap) + size, 0, paletteSize - size);
|
||||
memset(((PRUint8*)mColormap) + size, 0, mColormapSize - size);
|
||||
}
|
||||
if (len < size) {
|
||||
// Use 'hold' pattern to get the image colormap
|
||||
|
|
@ -1122,12 +1118,11 @@ nsresult nsGIFDecoder2::GifWrite(const PRUint8 *buf, PRUint32 len)
|
|||
break;
|
||||
} else {
|
||||
/* Switch back to the global palette */
|
||||
mColormap = mGIFStruct.global_colormap;
|
||||
if (mGIFStruct.images_decoded) {
|
||||
// Copy global colormap into the palette of current frame
|
||||
PRUint32 size;
|
||||
mImageFrame->GetPaletteData(&mColormap, &size);
|
||||
memcpy(mColormap, mGIFStruct.global_colormap, size);
|
||||
memcpy(mColormap, mGIFStruct.global_colormap, mColormapSize);
|
||||
} else {
|
||||
mColormap = mGIFStruct.global_colormap;
|
||||
}
|
||||
}
|
||||
GETN(1, gif_lzw_start);
|
||||
|
|
|
|||
|
|
@ -44,7 +44,6 @@
|
|||
#include "imgIDecoder.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "imgIDecoderObserver.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
|
||||
#include "GIF2.h"
|
||||
|
||||
|
|
@ -88,13 +87,13 @@ private:
|
|||
inline int ClearCode() const { return 1 << mGIFStruct.datasize; }
|
||||
|
||||
nsCOMPtr<imgIContainer> mImageContainer;
|
||||
nsCOMPtr<gfxIImageFrame> mImageFrame;
|
||||
nsCOMPtr<imgIDecoderObserver> mObserver; // this is just qi'd from mRequest for speed
|
||||
PRInt32 mCurrentRow;
|
||||
PRInt32 mLastFlushedRow;
|
||||
|
||||
PRUint8 *mImageData; // Pointer to image data in either Cairo or 8bit format
|
||||
PRUint32 *mColormap; // Current colormap to be used in Cairo format
|
||||
PRUint32 mColormapSize;
|
||||
PRUint32 mOldColor; // The old value of the transparent pixel
|
||||
PRUint8 mCurrentPass;
|
||||
PRUint8 mLastFlushedPass;
|
||||
|
|
|
|||
|
|
@ -47,7 +47,6 @@
|
|||
#include "nsRect.h"
|
||||
#include "nsComponentManagerUtils.h"
|
||||
|
||||
#include "nsIImage.h"
|
||||
#include "nsIInterfaceRequestorUtils.h"
|
||||
|
||||
NS_IMPL_THREADSAFE_ADDREF(nsIconDecoder)
|
||||
|
|
@ -71,14 +70,11 @@ NS_IMETHODIMP nsIconDecoder::Init(imgILoad *aLoad)
|
|||
{
|
||||
mObserver = do_QueryInterface(aLoad); // we're holding 2 strong refs to the request.
|
||||
|
||||
mImage = do_CreateInstance("@mozilla.org/image/container;1");
|
||||
mImage = do_CreateInstance("@mozilla.org/image/container;2");
|
||||
if (!mImage) return NS_ERROR_OUT_OF_MEMORY;
|
||||
|
||||
aLoad->SetImage(mImage);
|
||||
|
||||
mFrame = do_CreateInstance("@mozilla.org/gfx/image/frame;2");
|
||||
if (!mFrame) return NS_ERROR_OUT_OF_MEMORY;
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
|
@ -88,7 +84,7 @@ NS_IMETHODIMP nsIconDecoder::Close()
|
|||
|
||||
if (mObserver)
|
||||
{
|
||||
mObserver->OnStopFrame(nsnull, mFrame);
|
||||
mObserver->OnStopFrame(nsnull, 0);
|
||||
mObserver->OnStopContainer(nsnull, mImage);
|
||||
mObserver->OnStopDecode(nsnull, NS_OK, nsnull);
|
||||
}
|
||||
|
|
@ -120,17 +116,15 @@ NS_IMETHODIMP nsIconDecoder::WriteFrom(nsIInputStream *inStr, PRUint32 count, PR
|
|||
if (mObserver)
|
||||
mObserver->OnStartContainer(nsnull, mImage);
|
||||
|
||||
rv = mFrame->Init(0, 0, w, h, gfxIFormats::BGRA, 24);
|
||||
PRUint32 imageLen;
|
||||
PRUint8 *imageData;
|
||||
|
||||
rv = mImage->AppendFrame(0, 0, w, h, gfxASurface::ImageFormatARGB32, &imageData, &imageLen);
|
||||
if (NS_FAILED(rv))
|
||||
return rv;
|
||||
|
||||
mImage->AppendFrame(mFrame);
|
||||
if (mObserver)
|
||||
mObserver->OnStartFrame(nsnull, mFrame);
|
||||
|
||||
PRUint32 imageLen;
|
||||
PRUint8 *imageData;
|
||||
mFrame->GetImageData(&imageData, &imageLen);
|
||||
mObserver->OnStartFrame(nsnull, 0);
|
||||
|
||||
// Ensure that there enough in the inputStream
|
||||
NS_ENSURE_TRUE(count >= imageLen, NS_ERROR_UNEXPECTED);
|
||||
|
|
@ -142,12 +136,11 @@ NS_IMETHODIMP nsIconDecoder::WriteFrom(nsIInputStream *inStr, PRUint32 count, PR
|
|||
|
||||
// Notify the image...
|
||||
nsIntRect r(0, 0, w, h);
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(mFrame));
|
||||
rv = img->ImageUpdated(nsnull, nsImageUpdateFlags_kBitsChanged, &r);
|
||||
rv = mImage->FrameUpdated(0, r);
|
||||
if (NS_FAILED(rv))
|
||||
return rv;
|
||||
|
||||
mObserver->OnDataAvailable(nsnull, mFrame, &r);
|
||||
mObserver->OnDataAvailable(nsnull, PR_TRUE, &r);
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
|
|
|||
|
|
@ -46,7 +46,6 @@
|
|||
|
||||
#include "imgIContainer.h"
|
||||
#include "imgIDecoderObserver.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
|
||||
#define NS_ICONDECODER_CID \
|
||||
{ /* FFC08380-256C-11d5-9905-001083010E9B */ \
|
||||
|
|
@ -87,7 +86,6 @@ public:
|
|||
|
||||
private:
|
||||
nsCOMPtr<imgIContainer> mImage;
|
||||
nsCOMPtr<gfxIImageFrame> mFrame;
|
||||
nsCOMPtr<imgIDecoderObserver> mObserver; // this is just qi'd from mRequest for speed
|
||||
};
|
||||
|
||||
|
|
|
|||
|
|
@ -48,7 +48,6 @@
|
|||
#include "nspr.h"
|
||||
#include "nsCRT.h"
|
||||
#include "ImageLogging.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsIInterfaceRequestorUtils.h"
|
||||
#include "gfxColor.h"
|
||||
|
||||
|
|
@ -101,6 +100,7 @@ nsJPEGDecoder::nsJPEGDecoder()
|
|||
mState = JPEG_HEADER;
|
||||
mReading = PR_TRUE;
|
||||
mError = NS_OK;
|
||||
mImageData = nsnull;
|
||||
|
||||
mBytesToSkip = 0;
|
||||
memset(&mInfo, 0, sizeof(jpeg_decompress_struct));
|
||||
|
|
@ -187,7 +187,7 @@ NS_IMETHODIMP nsJPEGDecoder::Init(imgILoad *aLoad)
|
|||
mImageLoad->GetImage(getter_AddRefs(mImage));
|
||||
|
||||
if (!mImage) {
|
||||
mImage = do_CreateInstance("@mozilla.org/image/container;1");
|
||||
mImage = do_CreateInstance("@mozilla.org/image/container;2");
|
||||
if (!mImage)
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
|
||||
|
|
@ -473,49 +473,23 @@ nsresult nsJPEGDecoder::ProcessData(const char *data, PRUint32 count, PRUint32 *
|
|||
|
||||
mObserver->OnStartContainer(nsnull, mImage);
|
||||
|
||||
mImage->GetFrameAt(0, getter_AddRefs(mFrame));
|
||||
|
||||
if (mFrame) {
|
||||
PRInt32 width, height;
|
||||
mFrame->GetWidth(&width);
|
||||
mFrame->GetHeight(&height);
|
||||
|
||||
if ((width != (PRInt32)mInfo.image_width) ||
|
||||
(height != (PRInt32)mInfo.image_height)) {
|
||||
// Can't reuse frame, create a new one with correct size
|
||||
mFrame = nsnull;
|
||||
}
|
||||
}
|
||||
|
||||
if (!mFrame) {
|
||||
mFrame = do_CreateInstance("@mozilla.org/gfx/image/frame;2");
|
||||
if (!mFrame) {
|
||||
mState = JPEG_ERROR;
|
||||
PR_LOG(gJPEGDecoderAccountingLog, PR_LOG_DEBUG,
|
||||
("} (could not create image frame)"));
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
}
|
||||
|
||||
gfx_format format = gfxIFormats::RGB;
|
||||
#if defined(XP_WIN) || defined(XP_OS2) || defined(XP_BEOS)
|
||||
format = gfxIFormats::BGR;
|
||||
#endif
|
||||
|
||||
if (NS_FAILED(mFrame->Init(0, 0, mInfo.image_width, mInfo.image_height, format, 24))) {
|
||||
mState = JPEG_ERROR;
|
||||
PR_LOG(gJPEGDecoderAccountingLog, PR_LOG_DEBUG,
|
||||
("} (could not initialize image frame)"));
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
}
|
||||
|
||||
mImage->AppendFrame(mFrame);
|
||||
|
||||
// Use EnsureCleanFrame so we don't create a new frame if we're being
|
||||
// reused for e.g. multipart/x-replace
|
||||
PRUint32 imagelength;
|
||||
if (NS_FAILED(mImage->EnsureCleanFrame(0, 0, 0, mInfo.image_width, mInfo.image_height,
|
||||
gfxASurface::ImageFormatRGB24,
|
||||
&mImageData, &imagelength))) {
|
||||
mState = JPEG_ERROR;
|
||||
PR_LOG(gJPEGDecoderAccountingLog, PR_LOG_DEBUG,
|
||||
(" JPEGDecoderAccounting: nsJPEGDecoder::ProcessData -- created image frame with %ux%u pixels",
|
||||
mInfo.image_width, mInfo.image_height));
|
||||
("} (could not initialize image frame)"));
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
}
|
||||
|
||||
mObserver->OnStartFrame(nsnull, mFrame);
|
||||
PR_LOG(gJPEGDecoderAccountingLog, PR_LOG_DEBUG,
|
||||
(" JPEGDecoderAccounting: nsJPEGDecoder::ProcessData -- created image frame with %ux%u pixels",
|
||||
mInfo.image_width, mInfo.image_height));
|
||||
|
||||
mObserver->OnStartFrame(nsnull, 0);
|
||||
mState = JPEG_START_DECOMPRESS;
|
||||
}
|
||||
|
||||
|
|
@ -699,16 +673,9 @@ nsJPEGDecoder::OutputScanlines(PRBool* suspend)
|
|||
const PRUint32 top = mInfo.output_scanline;
|
||||
nsresult rv = NS_OK;
|
||||
|
||||
mFrame->LockImageData();
|
||||
|
||||
// we're thebes. we can write stuff directly to the data
|
||||
PRUint8 *imageData;
|
||||
PRUint32 imageDataLength;
|
||||
mFrame->GetImageData(&imageData, &imageDataLength);
|
||||
|
||||
while ((mInfo.output_scanline < mInfo.output_height)) {
|
||||
/* Use the Cairo image buffer as scanline buffer */
|
||||
PRUint32 *imageRow = ((PRUint32*)imageData) +
|
||||
PRUint32 *imageRow = ((PRUint32*)mImageData) +
|
||||
(mInfo.output_scanline * mInfo.output_width);
|
||||
|
||||
if (mInfo.cconvert->color_convert == ycc_rgb_convert_argb) {
|
||||
|
|
@ -791,13 +758,10 @@ nsJPEGDecoder::OutputScanlines(PRBool* suspend)
|
|||
|
||||
if (top != mInfo.output_scanline) {
|
||||
nsIntRect r(0, top, mInfo.output_width, mInfo.output_scanline-top);
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(mFrame));
|
||||
rv = img->ImageUpdated(nsnull, nsImageUpdateFlags_kBitsChanged, &r);
|
||||
mObserver->OnDataAvailable(nsnull, mFrame, &r);
|
||||
rv = mImage->FrameUpdated(0, r);
|
||||
mObserver->OnDataAvailable(nsnull, PR_TRUE, &r);
|
||||
}
|
||||
|
||||
mFrame->UnlockImageData();
|
||||
|
||||
|
||||
return rv;
|
||||
}
|
||||
|
||||
|
|
@ -1013,15 +977,16 @@ term_source (j_decompress_ptr jd)
|
|||
nsJPEGDecoder *decoder = (nsJPEGDecoder *)(jd->client_data);
|
||||
|
||||
if (decoder->mObserver) {
|
||||
decoder->mObserver->OnStopFrame(nsnull, decoder->mFrame);
|
||||
decoder->mObserver->OnStopFrame(nsnull, 0);
|
||||
decoder->mObserver->OnStopContainer(nsnull, decoder->mImage);
|
||||
decoder->mObserver->OnStopDecode(nsnull, NS_OK, nsnull);
|
||||
}
|
||||
|
||||
PRBool isMutable = PR_FALSE;
|
||||
PRBool multipart = PR_FALSE;
|
||||
if (decoder->mImageLoad)
|
||||
decoder->mImageLoad->GetIsMultiPartChannel(&isMutable);
|
||||
decoder->mFrame->SetMutable(isMutable);
|
||||
decoder->mImageLoad->GetIsMultiPartChannel(&multipart);
|
||||
if (!multipart)
|
||||
decoder->mImage->DecodingComplete();
|
||||
}
|
||||
|
||||
|
||||
|
|
|
|||
|
|
@ -45,7 +45,6 @@
|
|||
#include "nsCOMPtr.h"
|
||||
|
||||
#include "imgIContainer.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "imgIDecoderObserver.h"
|
||||
#include "imgILoad.h"
|
||||
#include "nsIInputStream.h"
|
||||
|
|
@ -99,9 +98,9 @@ protected:
|
|||
public:
|
||||
nsCOMPtr<imgIContainer> mImage;
|
||||
nsCOMPtr<imgILoad> mImageLoad;
|
||||
nsCOMPtr<gfxIImageFrame> mFrame;
|
||||
|
||||
nsCOMPtr<imgIDecoderObserver> mObserver;
|
||||
PRUint8 *mImageData;
|
||||
|
||||
struct jpeg_decompress_struct mInfo;
|
||||
struct jpeg_source_mgr mSourceMgr;
|
||||
|
|
|
|||
|
|
@ -48,7 +48,6 @@
|
|||
#include "nsIInputStream.h"
|
||||
|
||||
#include "imgIContainerObserver.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsIInterfaceRequestorUtils.h"
|
||||
|
||||
#include "gfxColor.h"
|
||||
|
|
@ -99,25 +98,29 @@ nsPNGDecoder::~nsPNGDecoder()
|
|||
|
||||
// CreateFrame() is used for both simple and animated images
|
||||
void nsPNGDecoder::CreateFrame(png_uint_32 x_offset, png_uint_32 y_offset,
|
||||
PRInt32 width, PRInt32 height, gfx_format format)
|
||||
PRInt32 width, PRInt32 height,
|
||||
gfxASurface::gfxImageFormat format)
|
||||
{
|
||||
mFrame = do_CreateInstance("@mozilla.org/gfx/image/frame;2");
|
||||
if (!mFrame)
|
||||
longjmp(mPNG->jmpbuf, 5); // NS_ERROR_OUT_OF_MEMORY
|
||||
|
||||
nsresult rv = mFrame->Init(x_offset, y_offset, width, height, format, 24);
|
||||
PRUint32 imageDataLength;
|
||||
nsresult rv = mImage->AppendFrame(x_offset, y_offset, width, height, format,
|
||||
&mImageData, &imageDataLength);
|
||||
if (NS_FAILED(rv))
|
||||
longjmp(mPNG->jmpbuf, 5); // NS_ERROR_OUT_OF_MEMORY
|
||||
|
||||
mFrameRect.x = x_offset;
|
||||
mFrameRect.y = y_offset;
|
||||
mFrameRect.width = width;
|
||||
mFrameRect.height = height;
|
||||
|
||||
if (png_get_valid(mPNG, mInfo, PNG_INFO_acTL))
|
||||
SetAnimFrameInfo();
|
||||
|
||||
mImage->AppendFrame(mFrame);
|
||||
|
||||
if (mObserver)
|
||||
mObserver->OnStartFrame(nsnull, mFrame);
|
||||
|
||||
|
||||
PRUint32 numFrames = 0;
|
||||
mImage->GetNumFrames(&numFrames);
|
||||
|
||||
if (mObserver)
|
||||
mObserver->OnStartFrame(nsnull, numFrames - 1);
|
||||
|
||||
PR_LOG(gPNGDecoderAccountingLog, PR_LOG_DEBUG,
|
||||
("PNGDecoderAccounting: nsPNGDecoder::CreateFrame -- created image frame with %dx%d pixels in container %p",
|
||||
width, height,
|
||||
|
|
@ -141,7 +144,7 @@ void nsPNGDecoder::SetAnimFrameInfo()
|
|||
blend_op = png_get_next_frame_blend_op(mPNG, mInfo);
|
||||
|
||||
if (delay_num == 0) {
|
||||
timeout = 0; // gfxImageFrame::SetTimeout() will set to a minimum
|
||||
timeout = 0; // SetFrameTimeout() will set to a minimum
|
||||
} else {
|
||||
if (delay_den == 0)
|
||||
delay_den = 100; // so says the APNG spec
|
||||
|
|
@ -151,54 +154,49 @@ void nsPNGDecoder::SetAnimFrameInfo()
|
|||
timeout = static_cast<PRInt32>
|
||||
(static_cast<PRFloat64>(delay_num) * 1000 / delay_den);
|
||||
}
|
||||
mFrame->SetTimeout(timeout);
|
||||
|
||||
PRUint32 numFrames = 0;
|
||||
mImage->GetNumFrames(&numFrames);
|
||||
|
||||
mImage->SetFrameTimeout(numFrames - 1, timeout);
|
||||
|
||||
if (dispose_op == PNG_DISPOSE_OP_PREVIOUS)
|
||||
mFrame->SetFrameDisposalMethod(imgIContainer::kDisposeRestorePrevious);
|
||||
mImage->SetFrameDisposalMethod(numFrames - 1, imgIContainer::kDisposeRestorePrevious);
|
||||
else if (dispose_op == PNG_DISPOSE_OP_BACKGROUND)
|
||||
mFrame->SetFrameDisposalMethod(imgIContainer::kDisposeClear);
|
||||
mImage->SetFrameDisposalMethod(numFrames - 1, imgIContainer::kDisposeClear);
|
||||
else
|
||||
mFrame->SetFrameDisposalMethod(imgIContainer::kDisposeKeep);
|
||||
mImage->SetFrameDisposalMethod(numFrames - 1, imgIContainer::kDisposeKeep);
|
||||
|
||||
if (blend_op == PNG_BLEND_OP_SOURCE)
|
||||
mFrame->SetBlendMethod(imgIContainer::kBlendSource);
|
||||
/*else // 'over' is the default for a gfxImageFrame
|
||||
mFrame->SetBlendMethod(imgIContainer::kBlendOver); */
|
||||
mImage->SetFrameBlendMethod(numFrames - 1, imgIContainer::kBlendSource);
|
||||
/*else // 'over' is the default
|
||||
mImage->SetFrameBlendMethod(numFrames - 1, imgIContainer::kBlendOver); */
|
||||
}
|
||||
|
||||
// set timeout and frame disposal method for the current frame
|
||||
void nsPNGDecoder::EndImageFrame()
|
||||
{
|
||||
if (mFrameHasNoAlpha) {
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(mFrame));
|
||||
img->SetHasNoAlpha();
|
||||
}
|
||||
|
||||
// First tell the container that this frame is complete
|
||||
PRInt32 timeout = 100;
|
||||
PRUint32 numFrames = 0;
|
||||
mFrame->GetTimeout(&timeout);
|
||||
mImage->GetNumFrames(&numFrames);
|
||||
|
||||
// We can't use mPNG->num_frames_read as it may be one ahead.
|
||||
if (numFrames > 1) {
|
||||
// Tell the image renderer that the frame is complete
|
||||
PRInt32 width, height;
|
||||
mFrame->GetWidth(&width);
|
||||
mFrame->GetHeight(&height);
|
||||
if (mFrameHasNoAlpha)
|
||||
mImage->SetFrameHasNoAlpha(numFrames - 1);
|
||||
|
||||
nsIntRect r(0, 0, width, height);
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(mFrame));
|
||||
if (NS_FAILED(img->ImageUpdated(nsnull, nsImageUpdateFlags_kBitsChanged, &r))) {
|
||||
if (NS_FAILED(mImage->FrameUpdated(numFrames - 1, mFrameRect))) {
|
||||
mError = PR_TRUE;
|
||||
// allow the call out to the observers.
|
||||
}
|
||||
mObserver->OnDataAvailable(nsnull, mFrame, &r);
|
||||
PRUint32 curFrame;
|
||||
mImage->GetCurrentFrameIndex(&curFrame);
|
||||
mObserver->OnDataAvailable(nsnull, curFrame == numFrames - 1, &mFrameRect);
|
||||
}
|
||||
|
||||
mImage->EndFrameDecode(numFrames);
|
||||
mImage->EndFrameDecode(numFrames - 1);
|
||||
if (mObserver)
|
||||
mObserver->OnStopFrame(nsnull, mFrame);
|
||||
mObserver->OnStopFrame(nsnull, numFrames - 1);
|
||||
}
|
||||
|
||||
|
||||
|
|
@ -265,7 +263,7 @@ NS_IMETHODIMP nsPNGDecoder::Init(imgILoad *aLoad)
|
|||
*/
|
||||
mImageLoad->GetImage(getter_AddRefs(mImage));
|
||||
if (!mImage) {
|
||||
mImage = do_CreateInstance("@mozilla.org/image/container;1");
|
||||
mImage = do_CreateInstance("@mozilla.org/image/container;2");
|
||||
if (!mImage)
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
|
||||
|
|
@ -556,12 +554,11 @@ info_callback(png_structp png_ptr, png_infop info_ptr)
|
|||
decoder->mInProfile = PNGGetColorProfile(png_ptr, info_ptr,
|
||||
color_type, &inType, &pIntent);
|
||||
/* If we're not mandating an intent, use the one from the image. */
|
||||
if (intent == -1)
|
||||
if (intent == PRUint32(-1))
|
||||
intent = pIntent;
|
||||
}
|
||||
if (decoder->mInProfile && gfxPlatform::GetCMSOutputProfile()) {
|
||||
qcms_data_type outType;
|
||||
PRUint32 dwFlags = 0;
|
||||
|
||||
if (color_type & PNG_COLOR_MASK_ALPHA || num_trans)
|
||||
outType = QCMS_DATA_RGBA_8;
|
||||
|
|
@ -631,22 +628,17 @@ info_callback(png_structp png_ptr, png_infop info_ptr)
|
|||
if (containerWidth == 0 && containerHeight == 0) {
|
||||
// the image hasn't been inited yet
|
||||
decoder->mImage->Init(width, height, decoder->mObserver);
|
||||
} else if (containerWidth != width || containerHeight != height) {
|
||||
} else if (containerWidth != PRInt32(width) || containerHeight != PRInt32(height)) {
|
||||
longjmp(decoder->mPNG->jmpbuf, 5); // NS_ERROR_UNEXPECTED
|
||||
}
|
||||
|
||||
if (decoder->mObserver)
|
||||
decoder->mObserver->OnStartContainer(nsnull, decoder->mImage);
|
||||
|
||||
if (channels == 1 || channels == 3) {
|
||||
decoder->format = gfxIFormats::RGB;
|
||||
} else if (channels == 2 || channels == 4) {
|
||||
if (alpha_bits == 8) {
|
||||
decoder->format = gfxIFormats::RGB_A8;
|
||||
} else if (alpha_bits == 1) {
|
||||
decoder->format = gfxIFormats::RGB_A1;
|
||||
}
|
||||
}
|
||||
if (channels == 1 || channels == 3)
|
||||
decoder->format = gfxASurface::ImageFormatRGB24;
|
||||
else if (channels == 2 || channels == 4)
|
||||
decoder->format = gfxASurface::ImageFormatARGB32;
|
||||
|
||||
if (png_get_valid(png_ptr, info_ptr, PNG_INFO_acTL))
|
||||
png_set_progressive_frame_fn(png_ptr, frame_info_callback, NULL);
|
||||
|
|
@ -674,9 +666,6 @@ info_callback(png_structp png_ptr, png_infop info_ptr)
|
|||
}
|
||||
}
|
||||
|
||||
if (png_get_first_frame_is_hidden(png_ptr, info_ptr))
|
||||
decoder->mFrame = nsnull;
|
||||
|
||||
/* Reject any ancillary chunk after IDAT with a bad CRC (bug #397593).
|
||||
* It would be better to show the default frame (if one has already been
|
||||
* successfully decoded) before bailing, but it's simpler to just bail
|
||||
|
|
@ -725,9 +714,8 @@ row_callback(png_structp png_ptr, png_bytep new_row,
|
|||
return;
|
||||
|
||||
if (new_row) {
|
||||
PRInt32 width;
|
||||
decoder->mFrame->GetWidth(&width);
|
||||
PRUint32 iwidth = width;
|
||||
PRInt32 width = decoder->mFrameRect.width;
|
||||
PRUint32 iwidth = decoder->mFrameRect.width;
|
||||
|
||||
png_bytep line = new_row;
|
||||
if (decoder->interlacebuf) {
|
||||
|
|
@ -735,11 +723,8 @@ row_callback(png_structp png_ptr, png_bytep new_row,
|
|||
png_progressive_combine_row(png_ptr, line, new_row);
|
||||
}
|
||||
|
||||
// we're thebes. we can write stuff directly to the data
|
||||
PRUint8 *imageData;
|
||||
PRUint32 imageDataLength, bpr = width * sizeof(PRUint32);
|
||||
decoder->mFrame->GetImageData(&imageData, &imageDataLength);
|
||||
PRUint32 *cptr32 = (PRUint32*)(imageData + (row_num*bpr));
|
||||
PRUint32 bpr = width * sizeof(PRUint32);
|
||||
PRUint32 *cptr32 = (PRUint32*)(decoder->mImageData + (row_num*bpr));
|
||||
PRBool rowHasNoAlpha = PR_TRUE;
|
||||
|
||||
if (decoder->mTransform) {
|
||||
|
|
@ -758,7 +743,7 @@ row_callback(png_structp png_ptr, png_bytep new_row,
|
|||
}
|
||||
|
||||
switch (decoder->format) {
|
||||
case gfxIFormats::RGB:
|
||||
case gfxASurface::ImageFormatRGB24:
|
||||
{
|
||||
// counter for while() loops below
|
||||
PRUint32 idx = iwidth;
|
||||
|
|
@ -785,17 +770,7 @@ row_callback(png_structp png_ptr, png_bytep new_row,
|
|||
}
|
||||
}
|
||||
break;
|
||||
case gfxIFormats::RGB_A1:
|
||||
{
|
||||
for (PRUint32 x=iwidth; x>0; --x) {
|
||||
*cptr32++ = GFX_PACKED_PIXEL(line[3]?0xFF:0x00, line[0], line[1], line[2]);
|
||||
if (line[3] == 0)
|
||||
rowHasNoAlpha = PR_FALSE;
|
||||
line += 4;
|
||||
}
|
||||
}
|
||||
break;
|
||||
case gfxIFormats::RGB_A8:
|
||||
case gfxASurface::ImageFormatARGB32:
|
||||
{
|
||||
for (PRUint32 x=width; x>0; --x) {
|
||||
*cptr32++ = GFX_PACKED_PIXEL(line[3], line[0], line[1], line[2]);
|
||||
|
|
@ -805,6 +780,10 @@ row_callback(png_structp png_ptr, png_bytep new_row,
|
|||
}
|
||||
}
|
||||
break;
|
||||
default:
|
||||
NS_ERROR("Unknown PNG format!");
|
||||
NS_ABORT();
|
||||
break;
|
||||
}
|
||||
|
||||
if (!rowHasNoAlpha)
|
||||
|
|
@ -815,12 +794,13 @@ row_callback(png_structp png_ptr, png_bytep new_row,
|
|||
if (numFrames <= 1) {
|
||||
// Only do incremental image display for the first frame
|
||||
nsIntRect r(0, row_num, width, 1);
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(decoder->mFrame));
|
||||
if (NS_FAILED(img->ImageUpdated(nsnull, nsImageUpdateFlags_kBitsChanged, &r))) {
|
||||
if (NS_FAILED(decoder->mImage->FrameUpdated(numFrames - 1, r))) {
|
||||
decoder->mError = PR_TRUE; /* bail */
|
||||
return;
|
||||
}
|
||||
decoder->mObserver->OnDataAvailable(nsnull, decoder->mFrame, &r);
|
||||
PRUint32 curFrame;
|
||||
decoder->mImage->GetCurrentFrameIndex(&curFrame);
|
||||
decoder->mObserver->OnDataAvailable(nsnull, curFrame == numFrames - 1, &r);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -44,9 +44,8 @@
|
|||
|
||||
#include "imgIContainer.h"
|
||||
#include "imgIDecoderObserver.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "imgILoad.h"
|
||||
|
||||
#include "gfxASurface.h"
|
||||
|
||||
#include "nsCOMPtr.h"
|
||||
|
||||
|
|
@ -72,25 +71,27 @@ public:
|
|||
virtual ~nsPNGDecoder();
|
||||
|
||||
void CreateFrame(png_uint_32 x_offset, png_uint_32 y_offset,
|
||||
PRInt32 width, PRInt32 height, gfx_format format);
|
||||
PRInt32 width, PRInt32 height,
|
||||
gfxASurface::gfxImageFormat format);
|
||||
void SetAnimFrameInfo();
|
||||
|
||||
void EndImageFrame();
|
||||
|
||||
public:
|
||||
nsCOMPtr<imgIContainer> mImage;
|
||||
nsCOMPtr<gfxIImageFrame> mFrame;
|
||||
nsCOMPtr<imgILoad> mImageLoad;
|
||||
nsCOMPtr<imgIDecoderObserver> mObserver; // this is just qi'd from mRequest for speed
|
||||
|
||||
png_structp mPNG;
|
||||
png_infop mInfo;
|
||||
nsIntRect mFrameRect;
|
||||
PRUint8 *mCMSLine;
|
||||
PRUint8 *interlacebuf;
|
||||
PRUint8 *mImageData;
|
||||
qcms_profile *mInProfile;
|
||||
qcms_transform *mTransform;
|
||||
|
||||
gfx_format format;
|
||||
gfxASurface::gfxImageFormat format;
|
||||
PRUint8 mChannels;
|
||||
PRPackedBool mError;
|
||||
PRPackedBool mFrameHasNoAlpha;
|
||||
|
|
|
|||
|
|
@ -47,7 +47,6 @@
|
|||
|
||||
#include "nsIInputStream.h"
|
||||
#include "nsIComponentManager.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsIInterfaceRequestorUtils.h"
|
||||
|
||||
#include "imgILoad.h"
|
||||
|
|
@ -56,7 +55,6 @@
|
|||
#include "nsISupportsPrimitives.h"
|
||||
|
||||
#include "gfxColor.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsIInterfaceRequestorUtils.h"
|
||||
|
||||
// Static colormap
|
||||
|
|
@ -82,11 +80,7 @@ NS_IMETHODIMP nsXBMDecoder::Init(imgILoad *aLoad)
|
|||
nsresult rv;
|
||||
mObserver = do_QueryInterface(aLoad);
|
||||
|
||||
mImage = do_CreateInstance("@mozilla.org/image/container;1", &rv);
|
||||
if (NS_FAILED(rv))
|
||||
return rv;
|
||||
|
||||
mFrame = do_CreateInstance("@mozilla.org/gfx/image/frame;2", &rv);
|
||||
mImage = do_CreateInstance("@mozilla.org/image/container;2", &rv);
|
||||
if (NS_FAILED(rv))
|
||||
return rv;
|
||||
|
||||
|
|
@ -106,7 +100,6 @@ NS_IMETHODIMP nsXBMDecoder::Close()
|
|||
mObserver->OnStopDecode(nsnull, NS_OK, nsnull);
|
||||
mObserver = nsnull;
|
||||
mImage = nsnull;
|
||||
mFrame = nsnull;
|
||||
mImageData = nsnull;
|
||||
|
||||
return NS_OK;
|
||||
|
|
@ -198,7 +191,9 @@ nsresult nsXBMDecoder::ProcessData(const char* aData, PRUint32 aCount) {
|
|||
mImage->Init(mWidth, mHeight, mObserver);
|
||||
mObserver->OnStartContainer(nsnull, mImage);
|
||||
|
||||
nsresult rv = mFrame->Init(0, 0, mWidth, mHeight, gfxIFormats::RGB_A1, 24);
|
||||
PRUint32 imageLen;
|
||||
nsresult rv = mImage->AppendFrame(0, 0, mWidth, mHeight, gfxASurface::ImageFormatARGB32,
|
||||
(PRUint8**)&mImageData, &imageLen);
|
||||
if (NS_FAILED(rv))
|
||||
return rv;
|
||||
|
||||
|
|
@ -218,11 +213,7 @@ nsresult nsXBMDecoder::ProcessData(const char* aData, PRUint32 aCount) {
|
|||
}
|
||||
}
|
||||
|
||||
mImage->AppendFrame(mFrame);
|
||||
mObserver->OnStartFrame(nsnull, mFrame);
|
||||
|
||||
PRUint32 imageLen;
|
||||
mFrame->GetImageData((PRUint8**)&mImageData, &imageLen);
|
||||
mObserver->OnStartFrame(nsnull, 0);
|
||||
|
||||
mState = RECV_SEEK;
|
||||
|
||||
|
|
@ -240,7 +231,6 @@ nsresult nsXBMDecoder::ProcessData(const char* aData, PRUint32 aCount) {
|
|||
}
|
||||
}
|
||||
if (mState == RECV_DATA) {
|
||||
nsCOMPtr<nsIImage> img = do_GetInterface(mFrame);
|
||||
PRUint32 *ar = mImageData + mCurRow * mWidth + mCurCol;
|
||||
|
||||
do {
|
||||
|
|
@ -279,17 +269,17 @@ nsresult nsXBMDecoder::ProcessData(const char* aData, PRUint32 aCount) {
|
|||
mCurCol += numPixels;
|
||||
if (mCurCol == mWidth || mState == RECV_DONE) {
|
||||
nsIntRect r(0, mCurRow, mWidth, 1);
|
||||
nsresult rv = img->ImageUpdated(nsnull, nsImageUpdateFlags_kBitsChanged, &r);
|
||||
nsresult rv = mImage->FrameUpdated(0, r);
|
||||
if (NS_FAILED(rv)) {
|
||||
return rv;
|
||||
}
|
||||
|
||||
mObserver->OnDataAvailable(nsnull, mFrame, &r);
|
||||
mObserver->OnDataAvailable(nsnull, PR_TRUE, &r);
|
||||
|
||||
mCurRow++;
|
||||
if (mCurRow == mHeight) {
|
||||
mState = RECV_DONE;
|
||||
return mObserver->OnStopFrame(nsnull, mFrame);
|
||||
return mObserver->OnStopFrame(nsnull, 0);
|
||||
}
|
||||
mCurCol = 0;
|
||||
}
|
||||
|
|
|
|||
|
|
@ -44,7 +44,6 @@
|
|||
#include "imgIDecoder.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "imgIDecoderObserver.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
|
||||
#define NS_XBMDECODER_CID \
|
||||
{ /* {dbfd145d-3298-4f3c-902f-2c5e1a1494ce} */ \
|
||||
|
|
@ -72,7 +71,6 @@ private:
|
|||
nsCOMPtr<imgIDecoderObserver> mObserver;
|
||||
|
||||
nsCOMPtr<imgIContainer> mImage;
|
||||
nsCOMPtr<gfxIImageFrame> mFrame;
|
||||
|
||||
PRUint32 mCurRow;
|
||||
PRUint32 mCurCol;
|
||||
|
|
|
|||
|
|
@ -23,6 +23,7 @@
|
|||
* Contributor(s):
|
||||
* Stuart Parmenter <pavlov@netscape.com>
|
||||
* Federico Mena-Quintero <federico@novell.com>
|
||||
* Joe Drew <joe@drew.ca>
|
||||
*
|
||||
* 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
|
||||
|
|
@ -39,22 +40,113 @@
|
|||
* ***** END LICENSE BLOCK ***** */
|
||||
|
||||
#include "nsISupports.idl"
|
||||
#include "gfxidltypes.idl"
|
||||
#include "gfxIFormats.idl"
|
||||
|
||||
interface gfxIImageFrame;
|
||||
interface imgIContainerObserver;
|
||||
|
||||
%{C++
|
||||
#include "gfxImageSurface.h"
|
||||
#include "gfxContext.h"
|
||||
#include "gfxMatrix.h"
|
||||
#include "gfxRect.h"
|
||||
#include "gfxPattern.h"
|
||||
#include "gfxASurface.h"
|
||||
#include "nsRect.h"
|
||||
%}
|
||||
|
||||
[ptr] native gfxImageSurface(gfxImageSurface);
|
||||
[ptr] native gfxASurface(gfxASurface);
|
||||
native gfxImageFormat(gfxASurface::gfxImageFormat);
|
||||
[ptr] native gfxContext(gfxContext);
|
||||
[ref] native gfxMatrix(gfxMatrix);
|
||||
[ref] native gfxRect(gfxRect);
|
||||
native gfxGraphicsFilter(gfxPattern::GraphicsFilter);
|
||||
[ref] native nsIntRect(nsIntRect);
|
||||
|
||||
/**
|
||||
* gfxIImageContainer interface
|
||||
* imgIContainer is the interface that represents an image. It allows
|
||||
* access to frames as Thebes surfaces, and permits users to extract subregions
|
||||
* as other imgIContainers. It also allows drawing of images on to Thebes
|
||||
* contexts.
|
||||
*
|
||||
* @author Stuart Parmenter <pavlov@netscape.com>
|
||||
* @version 0.2
|
||||
* @see "gfx2"
|
||||
* Internally, imgIContainer also manages animation of images.
|
||||
*/
|
||||
[scriptable, uuid(a4e9df32-f5d7-479f-9082-70e1fead23d5)]
|
||||
[scriptable, uuid(1bcf7a25-1356-47a8-bf80-e284989ea38f)]
|
||||
interface imgIContainer : nsISupports
|
||||
{
|
||||
/**
|
||||
* The width of the container rectangle.
|
||||
*/
|
||||
readonly attribute PRInt32 width;
|
||||
|
||||
/**
|
||||
* The height of the container rectangle.
|
||||
*/
|
||||
readonly attribute PRInt32 height;
|
||||
|
||||
/**
|
||||
* Whether this image is animated.
|
||||
*/
|
||||
readonly attribute boolean animated;
|
||||
|
||||
/**
|
||||
* Whether the current frame is opaque; that is, needs the background painted
|
||||
* behind it.
|
||||
*/
|
||||
readonly attribute boolean currentFrameIsOpaque;
|
||||
|
||||
/**
|
||||
* Get a surface for the current frame. This may be a platform-native,
|
||||
* optimized frame, so you cannot inspect its pixel data.
|
||||
*/
|
||||
[noscript] readonly attribute gfxASurface currentFrame;
|
||||
|
||||
/**
|
||||
* Create and return a new copy of the current frame that you can write to
|
||||
* and otherwise inspect the pixels of.
|
||||
*/
|
||||
[noscript] gfxImageSurface copyCurrentFrame();
|
||||
|
||||
/**
|
||||
* Create a new imgContainer that contains only a single frame, which itself
|
||||
* contains a subregion of the current frame.
|
||||
*
|
||||
* @param aRect the area of the current frame to be duplicated in the
|
||||
* returned imgContainer's frame.
|
||||
*/
|
||||
[noscript] imgIContainer extractCurrentFrame([const] in nsIntRect aRect);
|
||||
|
||||
/**
|
||||
* Draw the current frame on to the context specified.
|
||||
*
|
||||
* @param aContext The Thebex context to draw the image to.
|
||||
* @param aFilter The filter to be used if we're scaling the image.
|
||||
* @param aUserSpaceToImageSpace The transformation from user space (e.g.,
|
||||
* appunits) to image space.
|
||||
* @param aFill The area in the context to draw pixels to. Image will be
|
||||
* automatically tiled as necessary.
|
||||
* @param @aSubimage The area of the image, in pixels, that we are allowed to
|
||||
* sample from.
|
||||
*/
|
||||
[noscript] void draw(in gfxContext aContext, in gfxGraphicsFilter aFilter,
|
||||
in gfxMatrix aUserSpaceToImageSpace, in gfxRect aFill,
|
||||
in nsIntRect aSubimage);
|
||||
|
||||
|
||||
/************ Internal libpr0n use only below here. *****************/
|
||||
|
||||
/**
|
||||
* Create a new \a aWidth x \a aHeight sized image container.
|
||||
*
|
||||
* @param aWidth The width of the container in which all the
|
||||
* frames will fit.
|
||||
* @param aHeight The height of the container in which all the
|
||||
* frames will fit.
|
||||
* @param aObserver Observer to send animation notifications to.
|
||||
*/
|
||||
void init(in PRInt32 aWidth,
|
||||
in PRInt32 aHeight,
|
||||
in imgIContainerObserver aObserver);
|
||||
|
||||
/**
|
||||
* "Disposal" method indicates how the image should be handled before the
|
||||
* subsequent image is displayed.
|
||||
|
|
@ -78,42 +170,6 @@ interface imgIContainer : nsISupports
|
|||
const long kBlendOver = 1; // The frame should be composited onto the output buffer
|
||||
// based on its alpha, using a simple OVER operation
|
||||
|
||||
/**
|
||||
* Create a new \a aWidth x \a aHeight sized image container.
|
||||
*
|
||||
* @param aWidth The width of the container in which all the
|
||||
* gfxIImageFrame children will fit.
|
||||
* @param aHeight The height of the container in which all the
|
||||
* gfxIImageFrame children will fit.
|
||||
* @param aObserver Observer to send animation notifications to.
|
||||
*/
|
||||
void init(in PRInt32 aWidth,
|
||||
in PRInt32 aHeight,
|
||||
in imgIContainerObserver aObserver);
|
||||
|
||||
|
||||
/* this should probably be on the device context (or equiv) */
|
||||
readonly attribute gfx_format preferredAlphaChannelFormat;
|
||||
|
||||
/**
|
||||
* The width of the container rectangle.
|
||||
*/
|
||||
readonly attribute PRInt32 width;
|
||||
|
||||
/**
|
||||
* The height of the container rectangle.
|
||||
*/
|
||||
readonly attribute PRInt32 height;
|
||||
|
||||
|
||||
/**
|
||||
* Get the current frame that would be drawn if the image was to be drawn now
|
||||
*/
|
||||
readonly attribute gfxIImageFrame currentFrame;
|
||||
|
||||
|
||||
readonly attribute unsigned long numFrames;
|
||||
|
||||
/**
|
||||
* Animation mode Constants
|
||||
* 0 = normal
|
||||
|
|
@ -126,13 +182,54 @@ interface imgIContainer : nsISupports
|
|||
|
||||
attribute unsigned short animationMode;
|
||||
|
||||
gfxIImageFrame getFrameAt(in unsigned long index);
|
||||
/**
|
||||
* The rectangle defining the location and size of the currently displayed frame.
|
||||
* Should be an attribute, but can't be because of reference/pointer
|
||||
* conflicts with native types in xpidl.
|
||||
*/
|
||||
[noscript] void getCurrentFrameRect(in nsIntRect aFrameRect);
|
||||
|
||||
/**
|
||||
* Adds \a item to the end of the list of frames.
|
||||
* @param item frame to add.
|
||||
* The index of the current frame that would be drawn if the image was to be
|
||||
* drawn now.
|
||||
*/
|
||||
void appendFrame(in gfxIImageFrame item);
|
||||
readonly attribute unsigned long currentFrameIndex;
|
||||
|
||||
/**
|
||||
* The total number of frames in this image.
|
||||
*/
|
||||
readonly attribute unsigned long numFrames;
|
||||
|
||||
/**
|
||||
* Get the size, in bytes, of a particular frame's image data.
|
||||
*/
|
||||
unsigned long getFrameImageDataLength(in unsigned long framenumber);
|
||||
|
||||
void getFrameColormap(in unsigned long framenumber,
|
||||
[array, size_is(paletteLength)] out PRUint32 paletteData,
|
||||
out unsigned long paletteLength);
|
||||
|
||||
void setFrameDisposalMethod(in unsigned long framenumber, in PRInt32 aDisposalMethod);
|
||||
void setFrameBlendMethod(in unsigned long framenumber, in PRInt32 aBlendMethod);
|
||||
void setFrameTimeout(in unsigned long framenumber, in PRInt32 aTimeout);
|
||||
void setFrameHasNoAlpha(in unsigned long framenumber);
|
||||
|
||||
/**
|
||||
* Create or re-use a frame at index aFrameNum. It is an error to call this with aFrameNum not in the range [0, numFrames].
|
||||
*/
|
||||
[noscript] void ensureCleanFrame(in unsigned long aFramenum, in PRInt32 aX, in PRInt32 aY, in PRInt32 aWidth, in PRInt32 aHeight, in gfxImageFormat aFormat,
|
||||
[array, size_is(imageLength)] out PRUint8 imageData, out unsigned long imageLength);
|
||||
|
||||
/**
|
||||
* Adds to the end of the list of frames.
|
||||
*/
|
||||
[noscript] void appendFrame(in PRInt32 aX, in PRInt32 aY, in PRInt32 aWidth, in PRInt32 aHeight, in gfxImageFormat aFormat,
|
||||
[array, size_is(imageLength)] out PRUint8 imageData, out unsigned long imageLength);
|
||||
[noscript] void appendPalettedFrame(in PRInt32 aX, in PRInt32 aY, in PRInt32 aWidth, in PRInt32 aHeight, in gfxImageFormat aFormat, in PRUint8 aPaletteDepth,
|
||||
[array, size_is(imageLength)] out PRUint8 imageData, out unsigned long imageLength,
|
||||
[array, size_is(paletteLength)] out PRUint32 paletteData, out unsigned long paletteLength);
|
||||
|
||||
[noscript] void frameUpdated(in unsigned long framenum, in nsIntRect aNewRect);
|
||||
|
||||
/* notification when the current frame is done decoding */
|
||||
void endFrameDecode(in unsigned long framenumber);
|
||||
|
|
|
|||
|
|
@ -46,18 +46,14 @@
|
|||
|
||||
interface imgIContainer;
|
||||
|
||||
interface gfxIImageFrame;
|
||||
|
||||
/**
|
||||
* imgIContainerObserver interface
|
||||
*
|
||||
* @author Stuart Parmenter <pavlov@netscape.com>
|
||||
* @version 0.1
|
||||
*/
|
||||
[scriptable, uuid(53102f15-0f53-4939-957e-aea353ad2700)]
|
||||
[scriptable, uuid(e214c295-4b8e-4aa9-9907-45289e57295b)]
|
||||
interface imgIContainerObserver : nsISupports
|
||||
{
|
||||
[noscript] void frameChanged(in imgIContainer aContainer,
|
||||
in gfxIImageFrame aFrame,
|
||||
in nsIntRect aDirtyRect);
|
||||
[noscript] void frameChanged(in imgIContainer aContainer, in nsIntRect aDirtyRect);
|
||||
};
|
||||
|
|
|
|||
|
|
@ -41,7 +41,6 @@
|
|||
|
||||
interface imgIRequest;
|
||||
interface imgIContainer;
|
||||
interface gfxIImageFrame;
|
||||
|
||||
%{C++
|
||||
#include "nsRect.h"
|
||||
|
|
@ -59,7 +58,7 @@ interface gfxIImageFrame;
|
|||
* @version 0.1
|
||||
* @see imagelib2
|
||||
*/
|
||||
[scriptable, uuid(876f14ee-f27c-41cd-b6fb-9efda3ebc7b5)]
|
||||
[scriptable, uuid(1dfc9189-6421-4281-83b2-d9c1c9ba4d1b)]
|
||||
interface imgIDecoderObserver : imgIContainerObserver
|
||||
{
|
||||
/**
|
||||
|
|
@ -85,17 +84,17 @@ interface imgIDecoderObserver : imgIContainerObserver
|
|||
/**
|
||||
* called when each frame is created
|
||||
*/
|
||||
void onStartFrame(in imgIRequest aRequest, in gfxIImageFrame aFrame);
|
||||
void onStartFrame(in imgIRequest aRequest, in unsigned long aFrame);
|
||||
|
||||
/**
|
||||
* called when some part of the frame has new data in it
|
||||
*/
|
||||
[noscript] void onDataAvailable(in imgIRequest aRequest, in gfxIImageFrame aFrame, [const] in nsIntRect aRect);
|
||||
[noscript] void onDataAvailable(in imgIRequest aRequest, in boolean aCurrentFrame, [const] in nsIntRect aRect);
|
||||
|
||||
/**
|
||||
* called when a frame is finished decoding
|
||||
*/
|
||||
void onStopFrame(in imgIRequest aRequest, in gfxIImageFrame aFrame);
|
||||
void onStopFrame(in imgIRequest aRequest, in unsigned long aFrame);
|
||||
|
||||
/**
|
||||
* probably not needed. called right before onStopDecode
|
||||
|
|
|
|||
|
|
@ -59,10 +59,12 @@ REQUIRES = xpcom \
|
|||
xpconnect \
|
||||
js \
|
||||
uriloader \
|
||||
qcms \
|
||||
$(NULL)
|
||||
|
||||
CPPSRCS = \
|
||||
imgContainer.cpp \
|
||||
imgFrame.cpp \
|
||||
imgLoader.cpp \
|
||||
imgRequest.cpp \
|
||||
imgRequestProxy.cpp \
|
||||
|
|
@ -70,3 +72,7 @@ CPPSRCS = \
|
|||
|
||||
include $(topsrcdir)/config/rules.mk
|
||||
|
||||
# Because imgFrame.cpp includes "cairo.h"
|
||||
CXXFLAGS += $(MOZ_CAIRO_CFLAGS)
|
||||
|
||||
|
||||
|
|
|
|||
File diff suppressed because it is too large
Load diff
|
|
@ -55,18 +55,18 @@
|
|||
#include "nsCOMArray.h"
|
||||
#include "nsCOMPtr.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsIProperties.h"
|
||||
#include "nsITimer.h"
|
||||
#include "nsWeakReference.h"
|
||||
#include "nsTArray.h"
|
||||
#include "imgFrame.h"
|
||||
|
||||
#define NS_IMGCONTAINER_CID \
|
||||
{ /* 27f0682c-ff64-4dd2-ae7a-668e59f2fd38 */ \
|
||||
0x27f0682c, \
|
||||
0xff64, \
|
||||
0x4dd2, \
|
||||
{0xae, 0x7a, 0x66, 0x8e, 0x59, 0xf2, 0xfd, 0x38} \
|
||||
{ /* c76ff2c1-9bf6-418a-b143-3340c00112f7 */ \
|
||||
0x376ff2c1, \
|
||||
0x9bf6, \
|
||||
0x418a, \
|
||||
{0xb1, 0x43, 0x33, 0x40, 0xc0, 0x01, 0x12, 0xf7} \
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
@ -167,14 +167,14 @@ private:
|
|||
* lastCompositedFrameIndex to -1. Code assume that if
|
||||
* lastCompositedFrameIndex >= 0 then compositingFrame exists.
|
||||
*/
|
||||
nsCOMPtr<gfxIImageFrame> compositingFrame;
|
||||
nsAutoPtr<imgFrame> compositingFrame;
|
||||
/** the previous composited frame, for DISPOSE_RESTORE_PREVIOUS
|
||||
*
|
||||
* The Previous Frame (all frames composited up to the current) needs to be
|
||||
* stored in cases where the image specifies it wants the last frame back
|
||||
* when it's done with the current frame.
|
||||
*/
|
||||
nsCOMPtr<gfxIImageFrame> compositingPrevFrame;
|
||||
nsAutoPtr<imgFrame> compositingPrevFrame;
|
||||
//! Timer to animate multiframed images
|
||||
nsCOMPtr<nsITimer> timer;
|
||||
|
||||
|
|
@ -195,7 +195,9 @@ private:
|
|||
}
|
||||
};
|
||||
|
||||
gfxIImageFrame* GetCurrentFrameNoRef();
|
||||
imgFrame* GetImgFrame(PRUint32 framenum);
|
||||
imgFrame* GetCurrentImgFrame();
|
||||
PRInt32 GetCurrentImgFrameIndex() const;
|
||||
|
||||
inline Anim* ensureAnimExists() {
|
||||
if (!mAnim)
|
||||
|
|
@ -212,76 +214,53 @@ private:
|
|||
* @param aNextFrame Frame we need to incorperate/display
|
||||
* @param aNextFrameIndex Position of aNextFrame in mFrames list
|
||||
*/
|
||||
nsresult DoComposite(gfxIImageFrame** aFrameToUse, nsIntRect* aDirtyRect,
|
||||
gfxIImageFrame* aPrevFrame,
|
||||
gfxIImageFrame* aNextFrame,
|
||||
nsresult DoComposite(imgFrame** aFrameToUse, nsIntRect* aDirtyRect,
|
||||
imgFrame* aPrevFrame,
|
||||
imgFrame* aNextFrame,
|
||||
PRInt32 aNextFrameIndex);
|
||||
|
||||
/**
|
||||
* Combine aOverlayFrame's mask into aCompositingFrame's mask.
|
||||
*
|
||||
* This takes the mask information from the passed in aOverlayFrame and
|
||||
* inserts that information into the aCompositingFrame's mask at the proper
|
||||
* offsets. It does *not* rebuild the entire mask.
|
||||
*
|
||||
* @param aCompositingFrame Target frame
|
||||
* @param aOverlayFrame This frame's mask is being copied
|
||||
*/
|
||||
void BuildCompositeMask(gfxIImageFrame* aCompositingFrame,
|
||||
gfxIImageFrame* aOverlayFrame);
|
||||
|
||||
/** Sets an area of the frame's mask.
|
||||
*
|
||||
* @param aFrame Target Frame
|
||||
* @param aVisible Turn on (PR_TRUE) or off (PR_FALSE) visibility
|
||||
*
|
||||
* @note Invisible area of frame's image will need to be set to 0
|
||||
*/
|
||||
void SetMaskVisibility(gfxIImageFrame *aFrame, PRBool aVisible);
|
||||
//! @overload
|
||||
void SetMaskVisibility(gfxIImageFrame *aFrame,
|
||||
PRInt32 aX, PRInt32 aY,
|
||||
PRInt32 aWidth, PRInt32 aHeight,
|
||||
PRBool aVisible);
|
||||
//! @overload
|
||||
void SetMaskVisibility(gfxIImageFrame *aFrame,
|
||||
nsIntRect &aRect, PRBool aVisible) {
|
||||
SetMaskVisibility(aFrame, aRect.x, aRect.y,
|
||||
aRect.width, aRect.height, aVisible);
|
||||
}
|
||||
|
||||
/** Clears an area of <aFrame> with transparent black.
|
||||
*
|
||||
* @param aFrame Target Frame
|
||||
*
|
||||
* @note Does also clears the transparancy mask
|
||||
*/
|
||||
static void ClearFrame(gfxIImageFrame* aFrame);
|
||||
static void ClearFrame(imgFrame* aFrame);
|
||||
|
||||
//! @overload
|
||||
static void ClearFrame(gfxIImageFrame* aFrame, nsIntRect &aRect);
|
||||
static void ClearFrame(imgFrame* aFrame, nsIntRect &aRect);
|
||||
|
||||
//! Copy one gfxIImageFrame's image and mask into another
|
||||
static PRBool CopyFrameImage(gfxIImageFrame *aSrcFrame,
|
||||
gfxIImageFrame *aDstFrame);
|
||||
//! Copy one frames's image and mask into another
|
||||
static PRBool CopyFrameImage(imgFrame *aSrcFrame,
|
||||
imgFrame *aDstFrame);
|
||||
|
||||
/** Draws one gfxIImageFrame's image to into another,
|
||||
/** Draws one frames's image to into another,
|
||||
* at the position specified by aRect
|
||||
*
|
||||
* @param aSrcFrame Frame providing the source image
|
||||
* @param aDstFrame Frame where the image is drawn into
|
||||
* @param aRect The position and size to draw the image
|
||||
*/
|
||||
static nsresult DrawFrameTo(gfxIImageFrame *aSrcFrame,
|
||||
gfxIImageFrame *aDstFrame,
|
||||
static nsresult DrawFrameTo(imgFrame *aSrcFrame,
|
||||
imgFrame *aDstFrame,
|
||||
nsIntRect& aRect);
|
||||
|
||||
nsresult InternalAddFrameHelper(PRUint32 framenum, imgFrame *frame,
|
||||
PRUint8 **imageData, PRUint32 *imageLength,
|
||||
PRUint32 **paletteData, PRUint32 *paletteLength);
|
||||
nsresult InternalAddFrame(PRUint32 framenum, PRInt32 aX, PRInt32 aY, PRInt32 aWidth, PRInt32 aHeight,
|
||||
gfxASurface::gfxImageFormat aFormat, PRUint8 aPaletteDepth,
|
||||
PRUint8 **imageData, PRUint32 *imageLength,
|
||||
PRUint32 **paletteData, PRUint32 *paletteLength);
|
||||
|
||||
private: // data
|
||||
|
||||
nsIntSize mSize;
|
||||
|
||||
//! All the <gfxIImageFrame>s of the PNG
|
||||
//! All the frames of the image
|
||||
// *** IMPORTANT: if you use mFrames in a method, call RestoreDiscardedData() first to ensure
|
||||
// that the frames actually exist (they may have been discarded to save memory).
|
||||
nsCOMArray<gfxIImageFrame> mFrames;
|
||||
nsTArray<imgFrame *> mFrames;
|
||||
int mNumFrames; /* stored separately from mFrames.Count() to support discarded images */
|
||||
|
||||
nsCOMPtr<nsIProperties> mProperties;
|
||||
|
|
|
|||
922
modules/libpr0n/src/imgFrame.cpp
Normal file
922
modules/libpr0n/src/imgFrame.cpp
Normal file
|
|
@ -0,0 +1,922 @@
|
|||
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
||||
*
|
||||
* ***** BEGIN LICENSE BLOCK *****
|
||||
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
||||
*
|
||||
* 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 Mozilla Corporation.
|
||||
* Portions created by the Initial Developer are Copyright (C) 2009
|
||||
* the Initial Developer. All Rights Reserved.
|
||||
*
|
||||
* Contributor(s):
|
||||
* Joe Drew <joe@drew.ca> (original author)
|
||||
*
|
||||
* Alternatively, the contents of this file may be used under the terms of
|
||||
* either of 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 MPL, 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 MPL, the GPL or the LGPL.
|
||||
*
|
||||
* ***** END LICENSE BLOCK ***** */
|
||||
|
||||
#include "imgFrame.h"
|
||||
|
||||
#include <limits.h>
|
||||
|
||||
#include "prmem.h"
|
||||
#include "prenv.h"
|
||||
|
||||
#include "gfxPlatform.h"
|
||||
|
||||
static PRBool gDisableOptimize = PR_FALSE;
|
||||
|
||||
/*XXX get CAIRO_HAS_DDRAW_SURFACE */
|
||||
#include "cairo.h"
|
||||
|
||||
#ifdef CAIRO_HAS_DDRAW_SURFACE
|
||||
#include "gfxDDrawSurface.h"
|
||||
#endif
|
||||
|
||||
#if defined(XP_WIN) || defined(WINCE)
|
||||
#include "gfxWindowsPlatform.h"
|
||||
#endif
|
||||
|
||||
#if defined(XP_WIN) && !defined(WINCE)
|
||||
|
||||
/* Whether to use the windows surface; only for desktop win32 */
|
||||
#define USE_WIN_SURFACE 1
|
||||
|
||||
static PRUint32 gTotalDDBs = 0;
|
||||
static PRUint32 gTotalDDBSize = 0;
|
||||
// only use up a maximum of 64MB in DDBs
|
||||
#define kMaxDDBSize (64*1024*1024)
|
||||
// and don't let anything in that's bigger than 4MB
|
||||
#define kMaxSingleDDBSize (4*1024*1024)
|
||||
|
||||
#endif
|
||||
|
||||
// Returns true if an image of aWidth x aHeight is allowed and legal.
|
||||
static PRBool AllowedImageSize(PRInt32 aWidth, PRInt32 aHeight)
|
||||
{
|
||||
NS_ASSERTION(aWidth > 0, "invalid image width");
|
||||
NS_ASSERTION(aHeight > 0, "invalid image height");
|
||||
|
||||
// reject over-wide or over-tall images
|
||||
const PRInt32 k64KLimit = 0x0000FFFF;
|
||||
if (NS_UNLIKELY(aWidth > k64KLimit || aHeight > k64KLimit )) {
|
||||
NS_WARNING("image too big");
|
||||
return PR_FALSE;
|
||||
}
|
||||
|
||||
// protect against division by zero - this really shouldn't happen
|
||||
// if our consumers were well behaved, but they aren't (bug 368427)
|
||||
if (NS_UNLIKELY(aHeight == 0)) {
|
||||
return PR_FALSE;
|
||||
}
|
||||
|
||||
// check to make sure we don't overflow a 32-bit
|
||||
PRInt32 tmp = aWidth * aHeight;
|
||||
if (NS_UNLIKELY(tmp / aHeight != aWidth)) {
|
||||
NS_WARNING("width or height too large");
|
||||
return PR_FALSE;
|
||||
}
|
||||
tmp = tmp * 4;
|
||||
if (NS_UNLIKELY(tmp / 4 != aWidth * aHeight)) {
|
||||
NS_WARNING("width or height too large");
|
||||
return PR_FALSE;
|
||||
}
|
||||
#if defined(XP_MACOSX)
|
||||
// CoreGraphics is limited to images < 32K in *height*, so clamp all surfaces on the Mac to that height
|
||||
if (NS_UNLIKELY(aHeight > SHRT_MAX)) {
|
||||
NS_WARNING("image too big");
|
||||
return PR_FALSE;
|
||||
}
|
||||
#endif
|
||||
return PR_TRUE;
|
||||
}
|
||||
|
||||
// Returns whether we should, at this time, use image surfaces instead of
|
||||
// optimized platform-specific surfaces.
|
||||
static PRBool ShouldUseImageSurfaces()
|
||||
{
|
||||
#if defined(WINCE)
|
||||
// There is no test on windows mobile to check for Gui resources.
|
||||
// Allocate, until we run out of memory.
|
||||
gfxWindowsPlatform::RenderMode rmode = gfxWindowsPlatform::GetPlatform()->GetRenderMode();
|
||||
return rmode != gfxWindowsPlatform::RENDER_DDRAW &&
|
||||
rmode != gfxWindowsPlatform::RENDER_DDRAW_GL;
|
||||
|
||||
#elif defined(USE_WIN_SURFACE)
|
||||
static const DWORD kGDIObjectsHighWaterMark = 7000;
|
||||
|
||||
// at 7000 GDI objects, stop allocating normal images to make sure
|
||||
// we never hit the 10k hard limit.
|
||||
// GetCurrentProcess() just returns (HANDLE)-1, it's inlined afaik
|
||||
DWORD count = GetGuiResources(GetCurrentProcess(), GR_GDIOBJECTS);
|
||||
if (count == 0 ||
|
||||
count > kGDIObjectsHighWaterMark)
|
||||
{
|
||||
// either something's broken (count == 0),
|
||||
// or we hit our high water mark; disable
|
||||
// image allocations for a bit.
|
||||
return PR_TRUE;
|
||||
}
|
||||
#endif
|
||||
|
||||
return PR_FALSE;
|
||||
}
|
||||
|
||||
imgFrame::imgFrame() :
|
||||
mDecoded(0, 0, 0, 0),
|
||||
mPalettedImageData(nsnull),
|
||||
mSinglePixelColor(0),
|
||||
mTimeout(100),
|
||||
mDisposalMethod(0), /* imgIContainer::kDisposeNotSpecified */
|
||||
mBlendMethod(1), /* imgIContainer::kBlendOver */
|
||||
mSinglePixel(PR_FALSE),
|
||||
mNeverUseDeviceSurface(PR_FALSE),
|
||||
mFormatChanged(PR_FALSE)
|
||||
#ifdef USE_WIN_SURFACE
|
||||
, mIsDDBSurface(PR_FALSE)
|
||||
#endif
|
||||
{
|
||||
static PRBool hasCheckedOptimize = PR_FALSE;
|
||||
if (!hasCheckedOptimize) {
|
||||
if (PR_GetEnv("MOZ_DISABLE_IMAGE_OPTIMIZE")) {
|
||||
gDisableOptimize = PR_TRUE;
|
||||
}
|
||||
hasCheckedOptimize = PR_TRUE;
|
||||
}
|
||||
}
|
||||
|
||||
imgFrame::~imgFrame()
|
||||
{
|
||||
PR_FREEIF(mPalettedImageData);
|
||||
#ifdef USE_WIN_SURFACE
|
||||
if (mIsDDBSurface) {
|
||||
gTotalDDBs--;
|
||||
gTotalDDBSize -= mSize.width * mSize.height * 4;
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
nsresult imgFrame::Init(PRInt32 aX, PRInt32 aY, PRInt32 aWidth, PRInt32 aHeight,
|
||||
gfxASurface::gfxImageFormat aFormat, PRInt8 aPaletteDepth /* = 0 */)
|
||||
{
|
||||
// assert for properties that should be verified by decoders, warn for properties related to bad content
|
||||
if (!AllowedImageSize(aWidth, aHeight))
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
// Check to see if we are running OOM
|
||||
nsCOMPtr<nsIMemory> mem;
|
||||
NS_GetMemoryManager(getter_AddRefs(mem));
|
||||
if (!mem)
|
||||
return NS_ERROR_UNEXPECTED;
|
||||
|
||||
PRBool lowMemory;
|
||||
mem->IsLowMemory(&lowMemory);
|
||||
if (lowMemory)
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
|
||||
mOffset.MoveTo(aX, aY);
|
||||
mSize.SizeTo(aWidth, aHeight);
|
||||
|
||||
mFormat = aFormat;
|
||||
mPaletteDepth = aPaletteDepth;
|
||||
|
||||
if (aPaletteDepth != 0) {
|
||||
// We're creating for a paletted image.
|
||||
if (aPaletteDepth > 8) {
|
||||
NS_ERROR("This Depth is not supported\n");
|
||||
return NS_ERROR_FAILURE;
|
||||
}
|
||||
|
||||
mPalettedImageData = (PRUint8*)PR_MALLOC(PaletteDataLength() + GetImageDataLength());
|
||||
NS_ENSURE_TRUE(mPalettedImageData, NS_ERROR_OUT_OF_MEMORY);
|
||||
} else {
|
||||
// For Windows, we must create the device surface first (if we're
|
||||
// going to) so that the image surface can wrap it. Can't be done
|
||||
// the other way around.
|
||||
#ifdef USE_WIN_SURFACE
|
||||
if (!mNeverUseDeviceSurface && !ShouldUseImageSurfaces()) {
|
||||
mWinSurface = new gfxWindowsSurface(gfxIntSize(mSize.width, mSize.height), mFormat);
|
||||
if (mWinSurface && mWinSurface->CairoStatus() == 0) {
|
||||
// no error
|
||||
mImageSurface = mWinSurface->GetImageSurface();
|
||||
} else {
|
||||
mWinSurface = nsnull;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
// For other platforms we create the image surface first and then
|
||||
// possibly wrap it in a device surface. This branch is also used
|
||||
// on Windows if we're not using device surfaces or if we couldn't
|
||||
// create one.
|
||||
if (!mImageSurface)
|
||||
mImageSurface = new gfxImageSurface(gfxIntSize(mSize.width, mSize.height), mFormat);
|
||||
|
||||
if (!mImageSurface || mImageSurface->CairoStatus()) {
|
||||
mImageSurface = nsnull;
|
||||
// guess
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
}
|
||||
|
||||
#ifdef XP_MACOSX
|
||||
if (!mNeverUseDeviceSurface && !ShouldUseImageSurfaces()) {
|
||||
mQuartzSurface = new gfxQuartzImageSurface(mImageSurface);
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
nsresult imgFrame::Optimize()
|
||||
{
|
||||
if (gDisableOptimize)
|
||||
return NS_OK;
|
||||
|
||||
if (mPalettedImageData || mOptSurface || mSinglePixel)
|
||||
return NS_OK;
|
||||
|
||||
/* Figure out if the entire image is a constant color */
|
||||
|
||||
// this should always be true
|
||||
if (mImageSurface->Stride() == mSize.width * 4) {
|
||||
PRUint32 *imgData = (PRUint32*) mImageSurface->Data();
|
||||
PRUint32 firstPixel = * (PRUint32*) imgData;
|
||||
PRUint32 pixelCount = mSize.width * mSize.height + 1;
|
||||
|
||||
while (--pixelCount && *imgData++ == firstPixel)
|
||||
;
|
||||
|
||||
if (pixelCount == 0) {
|
||||
// all pixels were the same
|
||||
if (mFormat == gfxASurface::ImageFormatARGB32 ||
|
||||
mFormat == gfxASurface::ImageFormatRGB24)
|
||||
{
|
||||
mSinglePixelColor = gfxRGBA
|
||||
(firstPixel,
|
||||
(mFormat == gfxImageSurface::ImageFormatRGB24 ?
|
||||
gfxRGBA::PACKED_XRGB :
|
||||
gfxRGBA::PACKED_ARGB_PREMULTIPLIED));
|
||||
|
||||
mSinglePixel = PR_TRUE;
|
||||
|
||||
// blow away the older surfaces (if they exist), to release their memory
|
||||
mImageSurface = nsnull;
|
||||
mOptSurface = nsnull;
|
||||
#ifdef USE_WIN_SURFACE
|
||||
mWinSurface = nsnull;
|
||||
#endif
|
||||
#ifdef XP_MACOSX
|
||||
mQuartzSurface = nsnull;
|
||||
#endif
|
||||
return NS_OK;
|
||||
}
|
||||
}
|
||||
|
||||
// if it's not RGB24/ARGB32, don't optimize, but we never hit this at the moment
|
||||
}
|
||||
|
||||
// if we're being forced to use image surfaces due to
|
||||
// resource constraints, don't try to optimize beyond same-pixel.
|
||||
if (mNeverUseDeviceSurface || ShouldUseImageSurfaces())
|
||||
return NS_OK;
|
||||
|
||||
mOptSurface = nsnull;
|
||||
|
||||
#ifdef USE_WIN_SURFACE
|
||||
// we need to special-case windows here, because windows has
|
||||
// a distinction between DIB and DDB and we want to use DDBs as much
|
||||
// as we can.
|
||||
if (mWinSurface) {
|
||||
// Don't do DDBs for large images; see bug 359147
|
||||
// Note that we bother with DDBs at all because they are much faster
|
||||
// on some systems; on others there isn't much of a speed difference
|
||||
// between DIBs and DDBs.
|
||||
//
|
||||
// Originally this just limited to 1024x1024; but that still
|
||||
// had us hitting overall total memory usage limits (which was
|
||||
// around 220MB on my intel shared memory system with 2GB RAM
|
||||
// and 16-128mb in use by the video card, so I can't make
|
||||
// heads or tails out of this limit).
|
||||
//
|
||||
// So instead, we clamp the max size to 64MB (this limit shuld
|
||||
// be made dynamic based on.. something.. as soon a we figure
|
||||
// out that something) and also limit each individual image to
|
||||
// be less than 4MB to keep very large images out of DDBs.
|
||||
|
||||
// assume (almost -- we don't quadword-align) worst-case size
|
||||
PRUint32 ddbSize = mSize.width * mSize.height * 4;
|
||||
if (ddbSize <= kMaxSingleDDBSize &&
|
||||
ddbSize + gTotalDDBSize <= kMaxDDBSize)
|
||||
{
|
||||
nsRefPtr<gfxWindowsSurface> wsurf = mWinSurface->OptimizeToDDB(nsnull, gfxIntSize(mSize.width, mSize.height), mFormat);
|
||||
if (wsurf) {
|
||||
gTotalDDBs++;
|
||||
gTotalDDBSize += ddbSize;
|
||||
mIsDDBSurface = PR_TRUE;
|
||||
mOptSurface = wsurf;
|
||||
}
|
||||
}
|
||||
if (!mOptSurface && !mFormatChanged) {
|
||||
// just use the DIB if the format has not changed
|
||||
mOptSurface = mWinSurface;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef XP_MACOSX
|
||||
if (mQuartzSurface) {
|
||||
mQuartzSurface->Flush();
|
||||
mOptSurface = mQuartzSurface;
|
||||
}
|
||||
#endif
|
||||
|
||||
if (mOptSurface == nsnull)
|
||||
mOptSurface = gfxPlatform::GetPlatform()->OptimizeImage(mImageSurface, mFormat);
|
||||
|
||||
if (mOptSurface) {
|
||||
mImageSurface = nsnull;
|
||||
#ifdef USE_WIN_SURFACE
|
||||
mWinSurface = nsnull;
|
||||
#endif
|
||||
#ifdef XP_MACOSX
|
||||
mQuartzSurface = nsnull;
|
||||
#endif
|
||||
}
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
static PRBool
|
||||
IsSafeImageTransformComponent(gfxFloat aValue)
|
||||
{
|
||||
return aValue >= -32768 && aValue <= 32767;
|
||||
}
|
||||
|
||||
void imgFrame::Draw(gfxContext *aContext, gfxPattern::GraphicsFilter aFilter,
|
||||
const gfxMatrix &aUserSpaceToImageSpace, const gfxRect& aFill,
|
||||
const nsIntMargin &aPadding, const nsIntRect &aSubimage)
|
||||
{
|
||||
NS_ASSERTION(!aFill.IsEmpty(), "zero dest size --- fix caller");
|
||||
NS_ASSERTION(!aSubimage.IsEmpty(), "zero source size --- fix caller");
|
||||
NS_ASSERTION(!mPalettedImageData, "Directly drawing a paletted image!");
|
||||
|
||||
PRBool doPadding = aPadding != nsIntMargin(0,0,0,0);
|
||||
PRBool doPartialDecode = !ImageComplete();
|
||||
gfxContext::GraphicsOperator op = aContext->CurrentOperator();
|
||||
|
||||
if (mSinglePixel && !doPadding && ImageComplete()) {
|
||||
// Single-color fast path
|
||||
// if a == 0, it's a noop
|
||||
if (mSinglePixelColor.a == 0.0)
|
||||
return;
|
||||
|
||||
if (op == gfxContext::OPERATOR_OVER && mSinglePixelColor.a == 1.0)
|
||||
aContext->SetOperator(gfxContext::OPERATOR_SOURCE);
|
||||
|
||||
aContext->SetDeviceColor(mSinglePixelColor);
|
||||
aContext->NewPath();
|
||||
aContext->Rectangle(aFill);
|
||||
aContext->Fill();
|
||||
aContext->SetOperator(op);
|
||||
aContext->SetDeviceColor(gfxRGBA(0,0,0,0));
|
||||
return;
|
||||
}
|
||||
|
||||
gfxMatrix userSpaceToImageSpace = aUserSpaceToImageSpace;
|
||||
gfxRect sourceRect = userSpaceToImageSpace.Transform(aFill);
|
||||
gfxRect imageRect(0, 0, mSize.width + aPadding.LeftRight(), mSize.height + aPadding.TopBottom());
|
||||
gfxRect subimage(aSubimage.x, aSubimage.y, aSubimage.width, aSubimage.height);
|
||||
gfxRect fill = aFill;
|
||||
nsRefPtr<gfxASurface> surface;
|
||||
gfxImageSurface::gfxImageFormat format;
|
||||
|
||||
NS_ASSERTION(!sourceRect.Intersect(subimage).IsEmpty(),
|
||||
"We must be allowed to sample *some* source pixels!");
|
||||
|
||||
PRBool doTile = !imageRect.Contains(sourceRect);
|
||||
if (doPadding || doPartialDecode) {
|
||||
gfxRect available = gfxRect(mDecoded.x, mDecoded.y, mDecoded.width, mDecoded.height) +
|
||||
gfxPoint(aPadding.left, aPadding.top);
|
||||
|
||||
if (!doTile && !mSinglePixel) {
|
||||
// Not tiling, and we have a surface, so we can account for
|
||||
// padding and/or a partial decode just by twiddling parameters.
|
||||
// First, update our user-space fill rect.
|
||||
sourceRect = sourceRect.Intersect(available);
|
||||
gfxMatrix imageSpaceToUserSpace = userSpaceToImageSpace;
|
||||
imageSpaceToUserSpace.Invert();
|
||||
fill = imageSpaceToUserSpace.Transform(sourceRect);
|
||||
|
||||
surface = ThebesSurface();
|
||||
format = mFormat;
|
||||
subimage = subimage.Intersect(available) - gfxPoint(aPadding.left, aPadding.top);
|
||||
userSpaceToImageSpace.Multiply(gfxMatrix().Translate(-gfxPoint(aPadding.left, aPadding.top)));
|
||||
sourceRect = sourceRect - gfxPoint(aPadding.left, aPadding.top);
|
||||
imageRect = gfxRect(0, 0, mSize.width, mSize.height);
|
||||
} else {
|
||||
// Create a temporary surface
|
||||
gfxIntSize size(PRInt32(imageRect.Width()), PRInt32(imageRect.Height()));
|
||||
// Give this surface an alpha channel because there are
|
||||
// transparent pixels in the padding or undecoded area
|
||||
format = gfxASurface::ImageFormatARGB32;
|
||||
surface = gfxPlatform::GetPlatform()->CreateOffscreenSurface(size, format);
|
||||
if (!surface || surface->CairoStatus() != 0)
|
||||
return;
|
||||
|
||||
// Fill 'available' with whatever we've got
|
||||
gfxContext tmpCtx(surface);
|
||||
tmpCtx.SetOperator(gfxContext::OPERATOR_SOURCE);
|
||||
if (mSinglePixel) {
|
||||
tmpCtx.SetDeviceColor(mSinglePixelColor);
|
||||
} else {
|
||||
tmpCtx.SetSource(ThebesSurface(), gfxPoint(aPadding.left, aPadding.top));
|
||||
}
|
||||
tmpCtx.Rectangle(available);
|
||||
tmpCtx.Fill();
|
||||
}
|
||||
} else {
|
||||
NS_ASSERTION(!mSinglePixel, "This should already have been handled");
|
||||
surface = ThebesSurface();
|
||||
format = mFormat;
|
||||
}
|
||||
// At this point, we've taken care of mSinglePixel images, images with
|
||||
// aPadding, and partially-decoded images.
|
||||
|
||||
if (!AllowedImageSize(fill.size.width + 1, fill.size.height + 1)) {
|
||||
NS_WARNING("Destination area too large, bailing out");
|
||||
return;
|
||||
}
|
||||
|
||||
// Compute device-space-to-image-space transform. We need to sanity-
|
||||
// check it to work around a pixman bug :-(
|
||||
// XXX should we only do this for certain surface types?
|
||||
gfxFloat deviceX, deviceY;
|
||||
nsRefPtr<gfxASurface> currentTarget =
|
||||
aContext->CurrentSurface(&deviceX, &deviceY);
|
||||
gfxMatrix currentMatrix = aContext->CurrentMatrix();
|
||||
gfxMatrix deviceToUser = currentMatrix;
|
||||
deviceToUser.Invert();
|
||||
deviceToUser.Translate(-gfxPoint(-deviceX, -deviceY));
|
||||
gfxMatrix deviceToImage = deviceToUser;
|
||||
deviceToImage.Multiply(userSpaceToImageSpace);
|
||||
|
||||
PRBool pushedGroup = PR_FALSE;
|
||||
if (currentTarget->GetType() != gfxASurface::SurfaceTypeQuartz) {
|
||||
// BEGIN working around cairo/pixman bug (bug 364968)
|
||||
// Quartz's limits for matrix are much larger than pixman
|
||||
|
||||
// Our device-space-to-image-space transform may not be acceptable to pixman.
|
||||
if (!IsSafeImageTransformComponent(deviceToImage.xx) ||
|
||||
!IsSafeImageTransformComponent(deviceToImage.xy) ||
|
||||
!IsSafeImageTransformComponent(deviceToImage.yx) ||
|
||||
!IsSafeImageTransformComponent(deviceToImage.yy)) {
|
||||
NS_WARNING("Scaling up too much, bailing out");
|
||||
return;
|
||||
}
|
||||
|
||||
if (!IsSafeImageTransformComponent(deviceToImage.x0) ||
|
||||
!IsSafeImageTransformComponent(deviceToImage.y0)) {
|
||||
// We'll push a group, which will hopefully reduce our transform's
|
||||
// translation so it's in bounds
|
||||
aContext->Save();
|
||||
|
||||
// Clip the rounded-out-to-device-pixels bounds of the
|
||||
// transformed fill area. This is the area for the group we
|
||||
// want to push.
|
||||
aContext->IdentityMatrix();
|
||||
gfxRect bounds = currentMatrix.TransformBounds(fill);
|
||||
bounds.RoundOut();
|
||||
aContext->Clip(bounds);
|
||||
aContext->SetMatrix(currentMatrix);
|
||||
|
||||
aContext->PushGroup(gfxASurface::CONTENT_COLOR_ALPHA);
|
||||
aContext->SetOperator(gfxContext::OPERATOR_OVER);
|
||||
pushedGroup = PR_TRUE;
|
||||
}
|
||||
// END working around cairo/pixman bug (bug 364968)
|
||||
}
|
||||
|
||||
nsRefPtr<gfxPattern> pattern = new gfxPattern(surface);
|
||||
pattern->SetMatrix(userSpaceToImageSpace);
|
||||
|
||||
// OK now, the hard part left is to account for the subimage sampling
|
||||
// restriction. If all the transforms involved are just integer
|
||||
// translations, then we assume no resampling will occur so there's
|
||||
// nothing to do.
|
||||
// XXX if only we had source-clipping in cairo!
|
||||
if (!currentMatrix.HasNonIntegerTranslation() &&
|
||||
!userSpaceToImageSpace.HasNonIntegerTranslation()) {
|
||||
if (doTile) {
|
||||
pattern->SetExtend(gfxPattern::EXTEND_REPEAT);
|
||||
}
|
||||
} else {
|
||||
if (doTile || !subimage.Contains(imageRect)) {
|
||||
// EXTEND_PAD won't help us here; we have to create a temporary
|
||||
// surface to hold the subimage of pixels we're allowed to
|
||||
// sample
|
||||
|
||||
gfxRect userSpaceClipExtents = aContext->GetClipExtents();
|
||||
// This isn't optimal --- if aContext has a rotation then GetClipExtents
|
||||
// will have to do a bounding-box computation, and TransformBounds might
|
||||
// too, so we could get a better result if we computed image space clip
|
||||
// extents in one go --- but it doesn't really matter and this is easier
|
||||
// to understand.
|
||||
gfxRect imageSpaceClipExtents = userSpaceToImageSpace.TransformBounds(userSpaceClipExtents);
|
||||
// Inflate by one pixel because bilinear filtering will sample at most
|
||||
// one pixel beyond the computed image pixel coordinate.
|
||||
imageSpaceClipExtents.Outset(1.0);
|
||||
|
||||
gfxRect needed = imageSpaceClipExtents.Intersect(sourceRect).Intersect(subimage);
|
||||
needed.RoundOut();
|
||||
|
||||
// if 'needed' is empty, nothing will be drawn since aFill
|
||||
// must be entirely outside the clip region, so it doesn't
|
||||
// matter what we do here, but we should avoid trying to
|
||||
// create a zero-size surface.
|
||||
if (!needed.IsEmpty()) {
|
||||
gfxIntSize size(PRInt32(needed.Width()), PRInt32(needed.Height()));
|
||||
nsRefPtr<gfxASurface> temp =
|
||||
gfxPlatform::GetPlatform()->CreateOffscreenSurface(size, format);
|
||||
if (temp && temp->CairoStatus() == 0) {
|
||||
gfxContext tmpCtx(temp);
|
||||
tmpCtx.SetOperator(gfxContext::OPERATOR_SOURCE);
|
||||
nsRefPtr<gfxPattern> tmpPattern = new gfxPattern(surface);
|
||||
if (tmpPattern) {
|
||||
tmpPattern->SetExtend(gfxPattern::EXTEND_REPEAT);
|
||||
tmpPattern->SetMatrix(gfxMatrix().Translate(needed.pos));
|
||||
tmpCtx.SetPattern(tmpPattern);
|
||||
tmpCtx.Paint();
|
||||
tmpPattern = new gfxPattern(temp);
|
||||
if (tmpPattern) {
|
||||
pattern.swap(tmpPattern);
|
||||
pattern->SetMatrix(
|
||||
gfxMatrix(userSpaceToImageSpace).Multiply(gfxMatrix().Translate(-needed.pos)));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// In theory we can handle this using cairo's EXTEND_PAD,
|
||||
// but implementation limitations mean we have to consult
|
||||
// the surface type.
|
||||
switch (currentTarget->GetType()) {
|
||||
case gfxASurface::SurfaceTypeXlib:
|
||||
case gfxASurface::SurfaceTypeXcb:
|
||||
{
|
||||
// See bug 324698. This is a workaround for EXTEND_PAD not being
|
||||
// implemented correctly on linux in the X server.
|
||||
//
|
||||
// Set the filter to CAIRO_FILTER_FAST --- otherwise,
|
||||
// pixman's sampling will sample transparency for the outside edges and we'll
|
||||
// get blurry edges. CAIRO_EXTEND_PAD would also work here, if
|
||||
// available
|
||||
//
|
||||
// But don't do this for simple downscales because it's horrible.
|
||||
// Downscaling means that device-space coordinates are
|
||||
// scaled *up* to find the image pixel coordinates.
|
||||
//
|
||||
// deviceToImage is slightly stale because up above we may
|
||||
// have adjusted the pattern's matrix ... but the adjustment
|
||||
// is only a translation so the scale factors in deviceToImage
|
||||
// are still valid.
|
||||
PRBool isDownscale =
|
||||
deviceToImage.xx >= 1.0 && deviceToImage.yy >= 1.0 &&
|
||||
deviceToImage.xy == 0.0 && deviceToImage.yx == 0.0;
|
||||
if (!isDownscale) {
|
||||
pattern->SetFilter(gfxPattern::FILTER_FAST);
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
case gfxASurface::SurfaceTypeQuartz:
|
||||
case gfxASurface::SurfaceTypeQuartzImage:
|
||||
// Don't set EXTEND_PAD, Mac seems to be OK. Really?
|
||||
pattern->SetFilter(aFilter);
|
||||
break;
|
||||
|
||||
default:
|
||||
// turn on EXTEND_PAD.
|
||||
// This is what we really want for all surface types, if the
|
||||
// implementation was universally good.
|
||||
pattern->SetExtend(gfxPattern::EXTEND_PAD);
|
||||
pattern->SetFilter(aFilter);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if ((op == gfxContext::OPERATOR_OVER || pushedGroup) &&
|
||||
format == gfxASurface::ImageFormatRGB24) {
|
||||
aContext->SetOperator(gfxContext::OPERATOR_SOURCE);
|
||||
}
|
||||
|
||||
// Phew! Now we can actually draw this image
|
||||
aContext->NewPath();
|
||||
#ifdef MOZ_GFX_OPTIMIZE_MOBILE
|
||||
pattern->SetFilter(gfxPattern::FILTER_FAST);
|
||||
#endif
|
||||
aContext->SetPattern(pattern);
|
||||
aContext->Rectangle(fill);
|
||||
aContext->Fill();
|
||||
|
||||
aContext->SetOperator(op);
|
||||
if (pushedGroup) {
|
||||
aContext->PopGroupToSource();
|
||||
aContext->Paint();
|
||||
aContext->Restore();
|
||||
}
|
||||
}
|
||||
|
||||
nsresult imgFrame::Extract(const nsIntRect& aRegion, imgFrame** aResult)
|
||||
{
|
||||
nsAutoPtr<imgFrame> subImage(new imgFrame());
|
||||
if (!subImage)
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
|
||||
// The scaling problems described in bug 468496 are especially
|
||||
// likely to be visible for the sub-image, as at present the only
|
||||
// user is the border-image code and border-images tend to get
|
||||
// stretched a lot. At the same time, the performance concerns
|
||||
// that prevent us from just using Cairo's fallback scaler when
|
||||
// accelerated graphics won't cut it are less relevant to such
|
||||
// images, since they also tend to be small. Thus, we forcibly
|
||||
// disable the use of anything other than a client-side image
|
||||
// surface for the sub-image; this ensures that the correct
|
||||
// (albeit slower) Cairo fallback scaler will be used.
|
||||
subImage->mNeverUseDeviceSurface = PR_TRUE;
|
||||
|
||||
nsresult rv = subImage->Init(0, 0, aRegion.width, aRegion.height,
|
||||
mFormat, mPaletteDepth);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
// scope to destroy ctx
|
||||
{
|
||||
gfxContext ctx(subImage->ThebesSurface());
|
||||
ctx.SetOperator(gfxContext::OPERATOR_SOURCE);
|
||||
if (mSinglePixel) {
|
||||
ctx.SetDeviceColor(mSinglePixelColor);
|
||||
} else {
|
||||
// SetSource() places point (0,0) of its first argument at
|
||||
// the coordinages given by its second argument. We want
|
||||
// (x,y) of the image to be (0,0) of source space, so we
|
||||
// put (0,0) of the image at (-x,-y).
|
||||
ctx.SetSource(this->ThebesSurface(), gfxPoint(-aRegion.x, -aRegion.y));
|
||||
}
|
||||
ctx.Rectangle(gfxRect(0, 0, aRegion.width, aRegion.height));
|
||||
ctx.Fill();
|
||||
}
|
||||
|
||||
nsIntRect filled(0, 0, aRegion.width, aRegion.height);
|
||||
|
||||
rv = subImage->ImageUpdated(filled);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
subImage->Optimize();
|
||||
|
||||
*aResult = subImage.forget();
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
nsresult imgFrame::ImageUpdated(const nsIntRect &aUpdateRect)
|
||||
{
|
||||
// Check to see if we are running OOM
|
||||
nsCOMPtr<nsIMemory> mem;
|
||||
NS_GetMemoryManager(getter_AddRefs(mem));
|
||||
if (!mem)
|
||||
return NS_ERROR_UNEXPECTED;
|
||||
|
||||
PRBool lowMemory;
|
||||
mem->IsLowMemory(&lowMemory);
|
||||
if (lowMemory)
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
|
||||
mDecoded.UnionRect(mDecoded, aUpdateRect);
|
||||
|
||||
// clamp to bounds, in case someone sends a bogus updateRect (I'm looking at
|
||||
// you, gif decoder)
|
||||
nsIntRect boundsRect(0, 0, mSize.width, mSize.height);
|
||||
mDecoded.IntersectRect(mDecoded, boundsRect);
|
||||
|
||||
#ifdef XP_MACOSX
|
||||
if (mQuartzSurface)
|
||||
mQuartzSurface->Flush();
|
||||
#endif
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
PRInt32 imgFrame::GetX() const
|
||||
{
|
||||
return mOffset.x;
|
||||
}
|
||||
|
||||
PRInt32 imgFrame::GetY() const
|
||||
{
|
||||
return mOffset.y;
|
||||
}
|
||||
|
||||
PRInt32 imgFrame::GetWidth() const
|
||||
{
|
||||
return mSize.width;
|
||||
}
|
||||
|
||||
PRInt32 imgFrame::GetHeight() const
|
||||
{
|
||||
return mSize.height;
|
||||
}
|
||||
|
||||
nsIntRect imgFrame::GetRect() const
|
||||
{
|
||||
return nsIntRect(mOffset, mSize);
|
||||
}
|
||||
|
||||
gfxASurface::gfxImageFormat imgFrame::GetFormat() const
|
||||
{
|
||||
return mFormat;
|
||||
}
|
||||
|
||||
PRBool imgFrame::GetNeedsBackground() const
|
||||
{
|
||||
// We need a background painted if we have alpha or we're incomplete.
|
||||
return (mFormat == gfxASurface::ImageFormatARGB32 || !ImageComplete());
|
||||
}
|
||||
|
||||
PRUint32 imgFrame::GetImageBytesPerRow() const
|
||||
{
|
||||
if (mImageSurface)
|
||||
return mImageSurface->Stride();
|
||||
else
|
||||
return mSize.width;
|
||||
}
|
||||
|
||||
PRUint32 imgFrame::GetImageDataLength() const
|
||||
{
|
||||
if (mImageSurface)
|
||||
return mImageSurface->Stride() * mSize.height;
|
||||
else
|
||||
return mSize.width * mSize.height;
|
||||
}
|
||||
|
||||
void imgFrame::GetImageData(PRUint8 **aData, PRUint32 *length) const
|
||||
{
|
||||
if (mImageSurface)
|
||||
*aData = mImageSurface->Data();
|
||||
else if (mPalettedImageData)
|
||||
*aData = mPalettedImageData + PaletteDataLength();
|
||||
else
|
||||
*aData = nsnull;
|
||||
|
||||
*length = GetImageDataLength();
|
||||
}
|
||||
|
||||
PRBool imgFrame::GetIsPaletted() const
|
||||
{
|
||||
return mPalettedImageData != nsnull;
|
||||
}
|
||||
|
||||
PRBool imgFrame::GetHasAlpha() const
|
||||
{
|
||||
return mFormat != gfxASurface::ImageFormatARGB32;
|
||||
}
|
||||
|
||||
void imgFrame::GetPaletteData(PRUint32 **aPalette, PRUint32 *length) const
|
||||
{
|
||||
if (!mPalettedImageData) {
|
||||
*aPalette = nsnull;
|
||||
*length = 0;
|
||||
} else {
|
||||
*aPalette = (PRUint32 *) mPalettedImageData;
|
||||
*length = PaletteDataLength();
|
||||
}
|
||||
}
|
||||
|
||||
nsresult imgFrame::LockImageData()
|
||||
{
|
||||
if (mPalettedImageData)
|
||||
return NS_OK;
|
||||
|
||||
if ((mOptSurface || mSinglePixel) && !mImageSurface) {
|
||||
// Recover the pixels
|
||||
mImageSurface = new gfxImageSurface(gfxIntSize(mSize.width, mSize.height),
|
||||
gfxImageSurface::ImageFormatARGB32);
|
||||
if (!mImageSurface || mImageSurface->CairoStatus())
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
|
||||
gfxContext context(mImageSurface);
|
||||
context.SetOperator(gfxContext::OPERATOR_SOURCE);
|
||||
if (mSinglePixel)
|
||||
context.SetDeviceColor(mSinglePixelColor);
|
||||
else
|
||||
context.SetSource(mOptSurface);
|
||||
context.Paint();
|
||||
|
||||
mOptSurface = nsnull;
|
||||
#ifdef USE_WIN_SURFACE
|
||||
mWinSurface = nsnull;
|
||||
#endif
|
||||
#ifdef XP_MACOSX
|
||||
mQuartzSurface = nsnull;
|
||||
#endif
|
||||
}
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
nsresult imgFrame::UnlockImageData()
|
||||
{
|
||||
if (mPalettedImageData)
|
||||
return NS_OK;
|
||||
|
||||
#ifdef XP_MACOSX
|
||||
if (mQuartzSurface)
|
||||
mQuartzSurface->Flush();
|
||||
#endif
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
PRInt32 imgFrame::GetTimeout() const
|
||||
{
|
||||
// Ensure a minimal time between updates so we don't throttle the UI thread.
|
||||
// consider 0 == unspecified and make it fast but not too fast. See bug
|
||||
// 125137, bug 139677, and bug 207059. The behavior of recent IE and Opera
|
||||
// versions seems to be:
|
||||
// IE 6/Win:
|
||||
// 10 - 50ms go 100ms
|
||||
// >50ms go correct speed
|
||||
// Opera 7 final/Win:
|
||||
// 10ms goes 100ms
|
||||
// >10ms go correct speed
|
||||
// It seems that there are broken tools out there that set a 0ms or 10ms
|
||||
// timeout when they really want a "default" one. So munge values in that
|
||||
// range.
|
||||
if (mTimeout >= 0 && mTimeout <= 10)
|
||||
return 100;
|
||||
else
|
||||
return mTimeout;
|
||||
}
|
||||
|
||||
void imgFrame::SetTimeout(PRInt32 aTimeout)
|
||||
{
|
||||
mTimeout = aTimeout;
|
||||
}
|
||||
|
||||
PRInt32 imgFrame::GetFrameDisposalMethod() const
|
||||
{
|
||||
return mDisposalMethod;
|
||||
}
|
||||
|
||||
void imgFrame::SetFrameDisposalMethod(PRInt32 aFrameDisposalMethod)
|
||||
{
|
||||
mDisposalMethod = aFrameDisposalMethod;
|
||||
}
|
||||
|
||||
PRInt32 imgFrame::GetBlendMethod() const
|
||||
{
|
||||
return mBlendMethod;
|
||||
}
|
||||
|
||||
void imgFrame::SetBlendMethod(PRInt32 aBlendMethod)
|
||||
{
|
||||
mBlendMethod = (PRInt8)aBlendMethod;
|
||||
}
|
||||
|
||||
PRBool imgFrame::ImageComplete() const
|
||||
{
|
||||
return mDecoded == nsIntRect(0, 0, mSize.width, mSize.height);
|
||||
}
|
||||
|
||||
// A hint from the image decoders that this image has no alpha, even
|
||||
// though we created is ARGB32. This changes our format to RGB24,
|
||||
// which in turn will cause us to Optimize() to RGB24. Has no effect
|
||||
// after Optimize() is called, though in all cases it will be just a
|
||||
// performance win -- the pixels are still correct and have the A byte
|
||||
// set to 0xff.
|
||||
void imgFrame::SetHasNoAlpha()
|
||||
{
|
||||
if (mFormat == gfxASurface::ImageFormatARGB32) {
|
||||
mFormat = gfxASurface::ImageFormatRGB24;
|
||||
mFormatChanged = PR_TRUE;
|
||||
}
|
||||
}
|
||||
178
modules/libpr0n/src/imgFrame.h
Normal file
178
modules/libpr0n/src/imgFrame.h
Normal file
|
|
@ -0,0 +1,178 @@
|
|||
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
||||
*
|
||||
* ***** BEGIN LICENSE BLOCK *****
|
||||
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
||||
*
|
||||
* 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 Mozilla Corporation.
|
||||
* Portions created by the Initial Developer are Copyright (C) 2009
|
||||
* the Initial Developer. All Rights Reserved.
|
||||
*
|
||||
* Contributor(s):
|
||||
* Joe Drew <joe@drew.ca> (original author)
|
||||
*
|
||||
* 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 MPL, 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 MPL, the GPL or the LGPL.
|
||||
*
|
||||
* ***** END LICENSE BLOCK ***** */
|
||||
|
||||
#ifndef imgFrame_h
|
||||
#define imgFrame_h
|
||||
|
||||
#include "nsRect.h"
|
||||
#include "nsPoint.h"
|
||||
#include "nsSize.h"
|
||||
#include "gfxTypes.h"
|
||||
#include "nsID.h"
|
||||
#include "gfxIFormats.h"
|
||||
#include "gfxContext.h"
|
||||
#include "gfxPattern.h"
|
||||
#include "gfxImageSurface.h"
|
||||
#if defined(XP_WIN)
|
||||
#include "gfxWindowsSurface.h"
|
||||
#elif defined(XP_MACOSX)
|
||||
#include "gfxQuartzImageSurface.h"
|
||||
#endif
|
||||
#include "nsAutoPtr.h"
|
||||
|
||||
class imgFrame
|
||||
{
|
||||
public:
|
||||
imgFrame();
|
||||
~imgFrame();
|
||||
|
||||
nsresult Init(PRInt32 aX, PRInt32 aY, PRInt32 aWidth, PRInt32 aHeight, gfxASurface::gfxImageFormat aFormat, PRInt8 aPaletteDepth = 0);
|
||||
nsresult Optimize();
|
||||
|
||||
void Draw(gfxContext *aContext, gfxPattern::GraphicsFilter aFilter,
|
||||
const gfxMatrix &aUserSpaceToImageSpace, const gfxRect& aFill,
|
||||
const nsIntMargin &aPadding, const nsIntRect &aSubimage);
|
||||
|
||||
nsresult Extract(const nsIntRect& aRegion, imgFrame** aResult);
|
||||
|
||||
nsresult ImageUpdated(const nsIntRect &aUpdateRect);
|
||||
|
||||
PRInt32 GetX() const;
|
||||
PRInt32 GetY() const;
|
||||
PRInt32 GetWidth() const;
|
||||
PRInt32 GetHeight() const;
|
||||
nsIntRect GetRect() const;
|
||||
gfxASurface::gfxImageFormat GetFormat() const;
|
||||
PRBool GetNeedsBackground() const;
|
||||
PRUint32 GetImageBytesPerRow() const;
|
||||
PRUint32 GetImageDataLength() const;
|
||||
PRBool GetIsPaletted() const;
|
||||
PRBool GetHasAlpha() const;
|
||||
void GetImageData(PRUint8 **aData, PRUint32 *length) const;
|
||||
void GetPaletteData(PRUint32 **aPalette, PRUint32 *length) const;
|
||||
|
||||
PRInt32 GetTimeout() const;
|
||||
void SetTimeout(PRInt32 aTimeout);
|
||||
|
||||
PRInt32 GetFrameDisposalMethod() const;
|
||||
void SetFrameDisposalMethod(PRInt32 aFrameDisposalMethod);
|
||||
PRInt32 GetBlendMethod() const;
|
||||
void SetBlendMethod(PRInt32 aBlendMethod);
|
||||
PRBool ImageComplete() const;
|
||||
|
||||
void SetHasNoAlpha();
|
||||
|
||||
nsresult LockImageData();
|
||||
nsresult UnlockImageData();
|
||||
|
||||
nsresult GetSurface(gfxASurface **aSurface) const
|
||||
{
|
||||
*aSurface = ThebesSurface();
|
||||
NS_IF_ADDREF(*aSurface);
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
nsresult GetPattern(gfxPattern **aPattern) const
|
||||
{
|
||||
if (mSinglePixel)
|
||||
*aPattern = new gfxPattern(mSinglePixelColor);
|
||||
else
|
||||
*aPattern = new gfxPattern(ThebesSurface());
|
||||
NS_ADDREF(*aPattern);
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
gfxASurface* ThebesSurface() const
|
||||
{
|
||||
if (mOptSurface)
|
||||
return mOptSurface;
|
||||
#if defined(XP_WIN) && !defined(WINCE)
|
||||
if (mWinSurface)
|
||||
return mWinSurface;
|
||||
#elif defined(XP_MACOSX)
|
||||
if (mQuartzSurface)
|
||||
return mQuartzSurface;
|
||||
#endif
|
||||
return mImageSurface;
|
||||
}
|
||||
|
||||
|
||||
private: // methods
|
||||
PRUint32 PaletteDataLength() const {
|
||||
return ((1 << mPaletteDepth) * sizeof(PRUint32));
|
||||
}
|
||||
|
||||
private: // data
|
||||
nsRefPtr<gfxImageSurface> mImageSurface;
|
||||
nsRefPtr<gfxASurface> mOptSurface;
|
||||
#if defined(XP_WIN) && !defined(WINCE)
|
||||
nsRefPtr<gfxWindowsSurface> mWinSurface;
|
||||
#elif defined(XP_MACOSX)
|
||||
nsRefPtr<gfxQuartzImageSurface> mQuartzSurface;
|
||||
#endif
|
||||
|
||||
nsIntSize mSize;
|
||||
nsIntPoint mOffset;
|
||||
|
||||
nsIntRect mDecoded;
|
||||
|
||||
// The palette and image data for images that are paletted, since Cairo
|
||||
// doesn't support these images.
|
||||
// The paletted data comes first, then the image data itself.
|
||||
// Total length is PaletteDataLength() + GetImageDataLength().
|
||||
PRUint8* mPalettedImageData;
|
||||
|
||||
gfxRGBA mSinglePixelColor;
|
||||
|
||||
PRInt32 mTimeout; // -1 means display forever
|
||||
PRInt32 mDisposalMethod;
|
||||
|
||||
gfxASurface::gfxImageFormat mFormat;
|
||||
PRInt8 mPaletteDepth;
|
||||
PRInt8 mBlendMethod;
|
||||
PRPackedBool mSinglePixel;
|
||||
PRPackedBool mNeverUseDeviceSurface;
|
||||
PRPackedBool mFormatChanged;
|
||||
|
||||
#ifdef XP_WIN
|
||||
PRPackedBool mIsDDBSurface;
|
||||
#endif
|
||||
|
||||
};
|
||||
|
||||
#endif /* imgFrame_h */
|
||||
|
|
@ -46,8 +46,6 @@
|
|||
#include "ImageErrors.h"
|
||||
#include "ImageLogging.h"
|
||||
|
||||
#include "gfxIImageFrame.h"
|
||||
|
||||
#include "netCore.h"
|
||||
|
||||
#include "nsIChannel.h"
|
||||
|
|
@ -269,24 +267,19 @@ nsresult imgRequest::NotifyProxyListener(imgRequestProxy *proxy)
|
|||
mImage->GetNumFrames(&nframes);
|
||||
|
||||
if (nframes > 0) {
|
||||
nsCOMPtr<gfxIImageFrame> frame;
|
||||
|
||||
// get the current frame or only frame
|
||||
mImage->GetCurrentFrame(getter_AddRefs(frame));
|
||||
NS_ENSURE_TRUE(frame, NS_ERROR_OUT_OF_MEMORY);
|
||||
|
||||
// OnStartFrame
|
||||
PRUint32 frame;
|
||||
mImage->GetCurrentFrameIndex(&frame);
|
||||
proxy->OnStartFrame(frame);
|
||||
|
||||
if (!(mState & onStopContainer)) {
|
||||
// OnDataAvailable
|
||||
nsIntRect r;
|
||||
frame->GetRect(r); // XXX we should only send the currently decoded rectangle here.
|
||||
mImage->GetCurrentFrameRect(r); // XXX we should only send the currently decoded rectangle here.
|
||||
proxy->OnDataAvailable(frame, &r);
|
||||
} else {
|
||||
// OnDataAvailable
|
||||
nsIntRect r;
|
||||
frame->GetRect(r); // We're done loading this image, send the the whole rect
|
||||
mImage->GetCurrentFrameRect(r); // We're done loading this image, send the the whole rect
|
||||
proxy->OnDataAvailable(frame, &r);
|
||||
|
||||
// OnStopFrame
|
||||
|
|
@ -511,16 +504,15 @@ NS_IMETHODIMP imgRequest::GetIsMultiPartChannel(PRBool *aIsMultiPartChannel)
|
|||
|
||||
/** imgIContainerObserver methods **/
|
||||
|
||||
/* [noscript] void frameChanged (in imgIContainer container, in gfxIImageFrame newframe, in nsIntRect dirtyRect); */
|
||||
/* [noscript] void frameChanged (in imgIContainer container, in nsIntRect dirtyRect); */
|
||||
NS_IMETHODIMP imgRequest::FrameChanged(imgIContainer *container,
|
||||
gfxIImageFrame *newframe,
|
||||
nsIntRect * dirtyRect)
|
||||
{
|
||||
LOG_SCOPE(gImgLog, "imgRequest::FrameChanged");
|
||||
|
||||
nsTObserverArray<imgRequestProxy*>::ForwardIterator iter(mObservers);
|
||||
while (iter.HasMore()) {
|
||||
iter.GetNext()->FrameChanged(container, newframe, dirtyRect);
|
||||
iter.GetNext()->FrameChanged(container, dirtyRect);
|
||||
}
|
||||
|
||||
return NS_OK;
|
||||
|
|
@ -577,9 +569,9 @@ NS_IMETHODIMP imgRequest::OnStartContainer(imgIRequest *request, imgIContainer *
|
|||
return NS_OK;
|
||||
}
|
||||
|
||||
/* void onStartFrame (in imgIRequest request, in gfxIImageFrame frame); */
|
||||
/* void onStartFrame (in imgIRequest request, in unsigned long frame); */
|
||||
NS_IMETHODIMP imgRequest::OnStartFrame(imgIRequest *request,
|
||||
gfxIImageFrame *frame)
|
||||
PRUint32 frame)
|
||||
{
|
||||
LOG_SCOPE(gImgLog, "imgRequest::OnStartFrame");
|
||||
|
||||
|
|
@ -591,28 +583,25 @@ NS_IMETHODIMP imgRequest::OnStartFrame(imgIRequest *request,
|
|||
return NS_OK;
|
||||
}
|
||||
|
||||
/* [noscript] void onDataAvailable (in imgIRequest request, in gfxIImageFrame frame, [const] in nsIntRect rect); */
|
||||
/* [noscript] void onDataAvailable (in imgIRequest request, in boolean aCurrentFrame, [const] in nsIntRect rect); */
|
||||
NS_IMETHODIMP imgRequest::OnDataAvailable(imgIRequest *request,
|
||||
gfxIImageFrame *frame,
|
||||
PRBool aCurrentFrame,
|
||||
const nsIntRect * rect)
|
||||
{
|
||||
LOG_SCOPE(gImgLog, "imgRequest::OnDataAvailable");
|
||||
|
||||
nsTObserverArray<imgRequestProxy*>::ForwardIterator iter(mObservers);
|
||||
while (iter.HasMore()) {
|
||||
iter.GetNext()->OnDataAvailable(frame, rect);
|
||||
iter.GetNext()->OnDataAvailable(aCurrentFrame, rect);
|
||||
}
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/* void onStopFrame (in imgIRequest request, in gfxIImageFrame frame); */
|
||||
/* void onStopFrame (in imgIRequest request, in unsigned long frame); */
|
||||
NS_IMETHODIMP imgRequest::OnStopFrame(imgIRequest *request,
|
||||
gfxIImageFrame *frame)
|
||||
PRUint32 frame)
|
||||
{
|
||||
NS_ASSERTION(frame, "imgRequest::OnStopFrame called with NULL frame");
|
||||
if (!frame) return NS_ERROR_UNEXPECTED;
|
||||
|
||||
LOG_SCOPE(gImgLog, "imgRequest::OnStopFrame");
|
||||
|
||||
mImageStatus |= imgIRequest::STATUS_FRAME_COMPLETE;
|
||||
|
|
@ -621,7 +610,8 @@ NS_IMETHODIMP imgRequest::OnStopFrame(imgIRequest *request,
|
|||
PRUint32 cacheSize = mCacheEntry->GetDataSize();
|
||||
|
||||
PRUint32 imageSize = 0;
|
||||
frame->GetImageDataLength(&imageSize);
|
||||
if (mImage)
|
||||
mImage->GetFrameImageDataLength(frame, &imageSize);
|
||||
|
||||
mCacheEntry->SetDataSize(cacheSize + imageSize);
|
||||
|
||||
|
|
|
|||
|
|
@ -425,14 +425,14 @@ NS_IMETHODIMP imgRequestProxy::GetHasTransferredData(PRBool* hasData)
|
|||
|
||||
/** imgIContainerObserver methods **/
|
||||
|
||||
void imgRequestProxy::FrameChanged(imgIContainer *container, gfxIImageFrame *newframe, nsIntRect * dirtyRect)
|
||||
void imgRequestProxy::FrameChanged(imgIContainer *container, nsIntRect * dirtyRect)
|
||||
{
|
||||
LOG_FUNC(gImgLog, "imgRequestProxy::FrameChanged");
|
||||
|
||||
if (mListener && !mCanceled) {
|
||||
// Hold a ref to the listener while we call it, just in case.
|
||||
nsCOMPtr<imgIDecoderObserver> kungFuDeathGrip(mListener);
|
||||
mListener->FrameChanged(container, newframe, dirtyRect);
|
||||
mListener->FrameChanged(container, dirtyRect);
|
||||
}
|
||||
}
|
||||
|
||||
|
|
@ -460,7 +460,7 @@ void imgRequestProxy::OnStartContainer(imgIContainer *image)
|
|||
}
|
||||
}
|
||||
|
||||
void imgRequestProxy::OnStartFrame(gfxIImageFrame *frame)
|
||||
void imgRequestProxy::OnStartFrame(PRUint32 frame)
|
||||
{
|
||||
LOG_FUNC(gImgLog, "imgRequestProxy::OnStartFrame");
|
||||
|
||||
|
|
@ -471,18 +471,18 @@ void imgRequestProxy::OnStartFrame(gfxIImageFrame *frame)
|
|||
}
|
||||
}
|
||||
|
||||
void imgRequestProxy::OnDataAvailable(gfxIImageFrame *frame, const nsIntRect * rect)
|
||||
void imgRequestProxy::OnDataAvailable(PRBool aCurrentFrame, const nsIntRect * rect)
|
||||
{
|
||||
LOG_FUNC(gImgLog, "imgRequestProxy::OnDataAvailable");
|
||||
|
||||
if (mListener && !mCanceled) {
|
||||
// Hold a ref to the listener while we call it, just in case.
|
||||
nsCOMPtr<imgIDecoderObserver> kungFuDeathGrip(mListener);
|
||||
mListener->OnDataAvailable(this, frame, rect);
|
||||
mListener->OnDataAvailable(this, aCurrentFrame, rect);
|
||||
}
|
||||
}
|
||||
|
||||
void imgRequestProxy::OnStopFrame(gfxIImageFrame *frame)
|
||||
void imgRequestProxy::OnStopFrame(PRUint32 frame)
|
||||
{
|
||||
LOG_FUNC(gImgLog, "imgRequestProxy::OnStopFrame");
|
||||
|
||||
|
|
|
|||
|
|
@ -111,14 +111,14 @@ protected:
|
|||
/* non-virtual imgIDecoderObserver methods */
|
||||
void OnStartDecode ();
|
||||
void OnStartContainer(imgIContainer *aContainer);
|
||||
void OnStartFrame (gfxIImageFrame *aFrame);
|
||||
void OnDataAvailable (gfxIImageFrame *aFrame, const nsIntRect * aRect);
|
||||
void OnStopFrame (gfxIImageFrame *aFrame);
|
||||
void OnStartFrame (PRUint32 aFrame);
|
||||
void OnDataAvailable (PRBool aCurrentFrame, const nsIntRect * aRect);
|
||||
void OnStopFrame (PRUint32 aFrame);
|
||||
void OnStopContainer (imgIContainer *aContainer);
|
||||
void OnStopDecode (nsresult status, const PRUnichar *statusArg);
|
||||
|
||||
/* non-virtual imgIContainerObserver methods */
|
||||
void FrameChanged(imgIContainer *aContainer, gfxIImageFrame *aFrame, nsIntRect * aDirtyRect);
|
||||
void FrameChanged(imgIContainer *aContainer, nsIntRect * aDirtyRect);
|
||||
|
||||
/* non-virtual nsIRequestObserver (plus some) methods */
|
||||
void OnStartRequest(nsIRequest *request, nsISupports *ctxt);
|
||||
|
|
|
|||
|
|
@ -46,9 +46,6 @@
|
|||
#include "imgIEncoder.h"
|
||||
#include "imgIDecoderObserver.h"
|
||||
#include "imgIContainerObserver.h"
|
||||
#include "nsIImage.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "gfxImageSurface.h"
|
||||
#include "gfxContext.h"
|
||||
#include "nsStringStream.h"
|
||||
#include "nsComponentManagerUtils.h"
|
||||
|
|
@ -129,22 +126,21 @@ HelperLoader::OnStartContainer(imgIRequest *aRequest, imgIContainer
|
|||
|
||||
/* Implement imgIDecoderObserver::onStartFrame() */
|
||||
NS_IMETHODIMP
|
||||
HelperLoader::OnStartFrame(imgIRequest *aRequest, gfxIImageFrame *aFrame)
|
||||
HelperLoader::OnStartFrame(imgIRequest *aRequest, PRUint32 aFrame)
|
||||
{
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/* Implement imgIDecoderObserver::onDataAvailable() */
|
||||
NS_IMETHODIMP
|
||||
HelperLoader::OnDataAvailable(imgIRequest *aRequest, gfxIImageFrame
|
||||
*aFrame, const nsIntRect * aRect)
|
||||
HelperLoader::OnDataAvailable(imgIRequest *aRequest, PRBool aCurrentFrame, const nsIntRect * aRect)
|
||||
{
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/* Implement imgIDecoderObserver::onStopFrame() */
|
||||
NS_IMETHODIMP
|
||||
HelperLoader::OnStopFrame(imgIRequest *aRequest, gfxIImageFrame *aFrame)
|
||||
HelperLoader::OnStopFrame(imgIRequest *aRequest, PRUint32 aFrame)
|
||||
{
|
||||
return NS_OK;
|
||||
}
|
||||
|
|
@ -174,8 +170,7 @@ HelperLoader::OnStopRequest(imgIRequest *aRequest, PRBool aIsLastPart)
|
|||
|
||||
/* implement imgIContainerObserver::frameChanged() */
|
||||
NS_IMETHODIMP
|
||||
HelperLoader::FrameChanged(imgIContainer *aContainer,
|
||||
gfxIImageFrame *aFrame, nsIntRect * aDirtyRect)
|
||||
HelperLoader::FrameChanged(imgIContainer *aContainer, nsIntRect * aDirtyRect)
|
||||
{
|
||||
return NS_OK;
|
||||
}
|
||||
|
|
@ -230,7 +225,6 @@ NS_IMETHODIMP imgTools::DecodeImageData(nsIInputStream* aInStr,
|
|||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
PRUint32 written;
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
rv = decoder->WriteFrom(aInStr, length, &written);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
if (written != length)
|
||||
|
|
@ -285,19 +279,16 @@ NS_IMETHODIMP imgTools::EncodeScaledImage(imgIContainer *aContainer,
|
|||
return NS_IMAGELIB_ERROR_NO_ENCODER;
|
||||
|
||||
// Use frame 0 from the image container.
|
||||
nsCOMPtr<gfxIImageFrame> frame;
|
||||
rv = aContainer->GetFrameAt(0, getter_AddRefs(frame));
|
||||
nsRefPtr<gfxImageSurface> frame;
|
||||
rv = aContainer->CopyCurrentFrame(getter_AddRefs(frame));
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
if (!frame)
|
||||
return NS_ERROR_NOT_AVAILABLE;
|
||||
|
||||
PRInt32 w,h;
|
||||
frame->GetWidth(&w);
|
||||
frame->GetHeight(&h);
|
||||
PRInt32 w = frame->Width(), h = frame->Height();
|
||||
if (!w || !h)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(frame));
|
||||
nsRefPtr<gfxImageSurface> dest;
|
||||
|
||||
if (!doScaling) {
|
||||
|
|
@ -305,23 +296,16 @@ NS_IMETHODIMP imgTools::EncodeScaledImage(imgIContainer *aContainer,
|
|||
aScaledWidth = w;
|
||||
aScaledHeight = h;
|
||||
|
||||
img->LockImagePixels(PR_FALSE);
|
||||
bitmapData = img->GetBits();
|
||||
if (!bitmapData) {
|
||||
img->UnlockImagePixels(PR_FALSE);
|
||||
bitmapData = frame->Data();
|
||||
if (!bitmapData)
|
||||
return NS_ERROR_FAILURE;
|
||||
}
|
||||
|
||||
frame->GetImageBytesPerRow(&strideSize);
|
||||
strideSize = frame->Stride();
|
||||
bitmapDataLength = aScaledHeight * strideSize;
|
||||
|
||||
} else {
|
||||
// Prepare to draw a scaled version of the image to a temporary surface...
|
||||
|
||||
// Get the source image surface
|
||||
nsRefPtr<gfxPattern> gfxpat;
|
||||
img->GetPattern(getter_AddRefs(gfxpat));
|
||||
|
||||
// Create a temporary image surface
|
||||
dest = new gfxImageSurface(gfxIntSize(aScaledWidth, aScaledHeight),
|
||||
gfxASurface::ImageFormatARGB32);
|
||||
|
|
@ -337,7 +321,7 @@ NS_IMETHODIMP imgTools::EncodeScaledImage(imgIContainer *aContainer,
|
|||
|
||||
// Paint a scaled image
|
||||
ctx.SetOperator(gfxContext::OPERATOR_SOURCE);
|
||||
ctx.SetPattern(gfxpat);
|
||||
ctx.SetSource(frame);
|
||||
ctx.Paint();
|
||||
|
||||
bitmapData = dest->Data();
|
||||
|
|
@ -349,8 +333,6 @@ NS_IMETHODIMP imgTools::EncodeScaledImage(imgIContainer *aContainer,
|
|||
rv = encoder->InitFromData(bitmapData, bitmapDataLength,
|
||||
aScaledWidth, aScaledHeight, strideSize,
|
||||
imgIEncoder::INPUT_FORMAT_HOSTARGB, EmptyString());
|
||||
if (!doScaling)
|
||||
img->UnlockImagePixels(PR_FALSE);
|
||||
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
|
|
|
|||
|
|
@ -383,19 +383,23 @@ imgFile = do_get_file(imgName);
|
|||
istream = getFileInputStream(imgFile);
|
||||
do_check_eq(istream.available(), 17759);
|
||||
|
||||
// You'd think the decoder would fail, but it doesn't. The decoders use
|
||||
// stream->ReadSegments with a callback, and buffered streams ignore errors
|
||||
// from the callback. :-( See bug 413595.
|
||||
outParam = { value: null };
|
||||
imgTools.decodeImageData(istream, inMimeType, outParam);
|
||||
container = outParam.value;
|
||||
|
||||
try {
|
||||
istream = imgTools.encodeImage(container, "image/png");
|
||||
outParam = { value: null };
|
||||
imgTools.decodeImageData(istream, inMimeType, outParam);
|
||||
container = outParam.value;
|
||||
|
||||
// We should never hit this - decodeImageData throws an assertion because the
|
||||
// image decoded doesn't have enough frames.
|
||||
try {
|
||||
istream = imgTools.encodeImage(container, "image/png");
|
||||
} catch (e) {
|
||||
err = e;
|
||||
}
|
||||
} catch (e) {
|
||||
err = e;
|
||||
err = e;
|
||||
}
|
||||
checkExpectedError(/NS_ERROR_NOT_AVAILABLE/, err);
|
||||
|
||||
checkExpectedError(/NS_ERROR_ILLEGAL_VALUE/, err);
|
||||
|
||||
|
||||
/* ========== end ========== */
|
||||
|
|
|
|||
|
|
@ -95,7 +95,6 @@ STATIC_LIBS += \
|
|||
xpcom_core \
|
||||
ucvutil_s \
|
||||
gkgfx \
|
||||
gfxshared_s \
|
||||
$(NULL)
|
||||
|
||||
#ifndef MOZ_EMBEDDING_LEVEL_DEFAULT
|
||||
|
|
|
|||
|
|
@ -36,10 +36,9 @@
|
|||
* ***** END LICENSE BLOCK ***** */
|
||||
|
||||
#include "nsAlertsIconListener.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "imgILoader.h"
|
||||
#include "imgIRequest.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsNetUtil.h"
|
||||
#include "nsIImageToPixbuf.h"
|
||||
#include "nsIStringBundle.h"
|
||||
|
|
@ -106,7 +105,7 @@ nsAlertsIconListener::OnStartContainer(imgIRequest* aRequest,
|
|||
|
||||
NS_IMETHODIMP
|
||||
nsAlertsIconListener::OnStartFrame(imgIRequest* aRequest,
|
||||
gfxIImageFrame* aFrame)
|
||||
PRUint32 aFrame)
|
||||
{
|
||||
return NS_OK;
|
||||
}
|
||||
|
|
@ -114,7 +113,7 @@ nsAlertsIconListener::OnStartFrame(imgIRequest* aRequest,
|
|||
|
||||
NS_IMETHODIMP
|
||||
nsAlertsIconListener::OnDataAvailable(imgIRequest* aRequest,
|
||||
gfxIImageFrame* aFrame,
|
||||
PRBool aCurrentFrame,
|
||||
const nsIntRect* aRect)
|
||||
{
|
||||
return NS_OK;
|
||||
|
|
@ -138,7 +137,6 @@ nsAlertsIconListener::OnStopDecode(imgIRequest* aRequest,
|
|||
|
||||
NS_IMETHODIMP
|
||||
nsAlertsIconListener::FrameChanged(imgIContainer* aContainer,
|
||||
gfxIImageFrame* aFrame,
|
||||
nsIntRect* aDirtyRect)
|
||||
{
|
||||
return NS_OK;
|
||||
|
|
@ -165,7 +163,7 @@ nsAlertsIconListener::OnStopRequest(imgIRequest* aRequest,
|
|||
|
||||
NS_IMETHODIMP
|
||||
nsAlertsIconListener::OnStopFrame(imgIRequest* aRequest,
|
||||
gfxIImageFrame* aFrame)
|
||||
PRUint32 aFrame)
|
||||
{
|
||||
if (aRequest != mIconRequest)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
|
@ -173,12 +171,8 @@ nsAlertsIconListener::OnStopFrame(imgIRequest* aRequest,
|
|||
if (mLoadedFrame)
|
||||
return NS_OK; // only use one frame
|
||||
|
||||
nsCOMPtr<gfxIImageFrame> frame = aFrame;
|
||||
nsCOMPtr<nsIImage> image = do_GetInterface(frame);
|
||||
if (!image)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
nsresult rv = image->LockImagePixels(PR_FALSE);
|
||||
nsCOMPtr<imgIContainer> image;
|
||||
nsresult rv = aRequest->GetImage(getter_AddRefs(image));
|
||||
if (NS_FAILED(rv))
|
||||
return rv;
|
||||
|
||||
|
|
@ -189,10 +183,6 @@ nsAlertsIconListener::OnStopFrame(imgIRequest* aRequest,
|
|||
if (!imagePixbuf)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
rv = image->UnlockImagePixels(PR_FALSE);
|
||||
if (NS_FAILED(rv))
|
||||
return rv;
|
||||
|
||||
ShowAlert(imagePixbuf);
|
||||
|
||||
g_object_unref(imagePixbuf);
|
||||
|
|
|
|||
|
|
@ -94,9 +94,6 @@ JS_Init
|
|||
<atoms>
|
||||
NS_NewPermanentAtom
|
||||
|
||||
<images>
|
||||
gfxImageFrame::Init
|
||||
|
||||
<images>
|
||||
gif_write
|
||||
|
||||
|
|
|
|||
|
|
@ -43,13 +43,13 @@
|
|||
#include "nsXPIDLString.h"
|
||||
#include "nsPrimitiveHelpers.h"
|
||||
#include "nsMemory.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsILocalFile.h"
|
||||
#include "nsStringStream.h"
|
||||
#include "nsDragService.h"
|
||||
#include "nsEscape.h"
|
||||
#include "nsPrintfCString.h"
|
||||
#include "nsObjCExceptions.h"
|
||||
#include "imgIContainer.h"
|
||||
|
||||
// Screenshots use the (undocumented) png pasteboard type.
|
||||
#define IMAGE_PASTEBOARD_TYPES NSTIFFPboardType, @"Apple PNG pasteboard type", nil
|
||||
|
|
@ -425,18 +425,19 @@ nsClipboard::PasteboardDictFromTransferable(nsITransferable* aTransferable)
|
|||
nsCOMPtr<nsISupports> primitiveData;
|
||||
ptrPrimitive->GetData(getter_AddRefs(primitiveData));
|
||||
|
||||
nsCOMPtr<nsIImage> image(do_QueryInterface(primitiveData));
|
||||
nsCOMPtr<imgIContainer> image(do_QueryInterface(primitiveData));
|
||||
if (!image) {
|
||||
NS_WARNING("Image isn't an nsIImage in transferable");
|
||||
NS_WARNING("Image isn't an imgIContainer in transferable");
|
||||
continue;
|
||||
}
|
||||
|
||||
if (NS_FAILED(image->LockImagePixels(PR_FALSE)))
|
||||
nsRefPtr<gfxImageSurface> currentFrame;
|
||||
if (NS_FAILED(image->CopyCurrentFrame(getter_AddRefs(currentFrame))))
|
||||
continue;
|
||||
|
||||
PRInt32 height = image->GetHeight();
|
||||
PRInt32 stride = image->GetLineStride();
|
||||
PRInt32 width = image->GetWidth();
|
||||
PRInt32 height = currentFrame->Height();
|
||||
PRInt32 stride = currentFrame->Stride();
|
||||
PRInt32 width = currentFrame->Width();
|
||||
if ((stride % 4 != 0) || (height < 1) || (width < 1))
|
||||
continue;
|
||||
|
||||
|
|
@ -444,7 +445,7 @@ nsClipboard::PasteboardDictFromTransferable(nsITransferable* aTransferable)
|
|||
// the alpha ordering and endianness of the machine so we don't have to
|
||||
// touch the bits ourselves.
|
||||
CGDataProviderRef dataProvider = CGDataProviderCreateWithData(NULL,
|
||||
image->GetBits(),
|
||||
currentFrame->Data(),
|
||||
stride * height,
|
||||
NULL);
|
||||
CGColorSpaceRef colorSpace = CGColorSpaceCreateWithName(kCGColorSpaceGenericRGB);
|
||||
|
|
@ -475,7 +476,7 @@ nsClipboard::PasteboardDictFromTransferable(nsITransferable* aTransferable)
|
|||
if (destRef)
|
||||
CFRelease(destRef);
|
||||
|
||||
if (NS_FAILED(image->UnlockImagePixels(PR_FALSE)) || !successfullyConverted) {
|
||||
if (!successfullyConverted) {
|
||||
if (tiffData)
|
||||
CFRelease(tiffData);
|
||||
continue;
|
||||
|
|
|
|||
|
|
@ -50,7 +50,6 @@
|
|||
#include "nsIDOMNode.h"
|
||||
#include "nsRect.h"
|
||||
#include "nsPoint.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsICharsetConverterManager.h"
|
||||
#include "nsIIOService.h"
|
||||
#include "nsNetUtil.h"
|
||||
|
|
|
|||
|
|
@ -59,9 +59,9 @@
|
|||
#include "nsNetUtil.h"
|
||||
#include "imgILoader.h"
|
||||
#include "imgIRequest.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsMenuItemX.h"
|
||||
#include "gfxImageSurface.h"
|
||||
#include "imgIContainer.h"
|
||||
|
||||
static const PRUint32 kIconWidth = 16;
|
||||
static const PRUint32 kIconHeight = 16;
|
||||
|
|
@ -280,9 +280,8 @@ nsMenuItemIconX::LoadIcon(nsIURI* aIconURI)
|
|||
//
|
||||
|
||||
NS_IMETHODIMP
|
||||
nsMenuItemIconX::FrameChanged(imgIContainer* aContainer,
|
||||
gfxIImageFrame* aFrame,
|
||||
nsIntRect* aDirtyRect)
|
||||
nsMenuItemIconX::FrameChanged(imgIContainer* aContainer,
|
||||
nsIntRect* aDirtyRect)
|
||||
{
|
||||
return NS_OK;
|
||||
}
|
||||
|
|
@ -305,32 +304,33 @@ nsMenuItemIconX::OnStartDecode(imgIRequest* aRequest)
|
|||
|
||||
NS_IMETHODIMP
|
||||
nsMenuItemIconX::OnStartContainer(imgIRequest* aRequest,
|
||||
imgIContainer* aContainer)
|
||||
imgIContainer* aContainer)
|
||||
{
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
NS_IMETHODIMP
|
||||
nsMenuItemIconX::OnStartFrame(imgIRequest* aRequest, gfxIImageFrame* aFrame)
|
||||
nsMenuItemIconX::OnStartFrame(imgIRequest* aRequest, PRUint32 aFrame)
|
||||
{
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
NS_IMETHODIMP
|
||||
nsMenuItemIconX::OnDataAvailable(imgIRequest* aRequest,
|
||||
gfxIImageFrame* aFrame,
|
||||
const nsIntRect* aRect)
|
||||
PRBool aCurrentFrame,
|
||||
const nsIntRect* aRect)
|
||||
{
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
NS_IMETHODIMP
|
||||
nsMenuItemIconX::OnStopFrame(imgIRequest* aRequest,
|
||||
gfxIImageFrame* aFrame)
|
||||
PRUint32 aFrame)
|
||||
{
|
||||
NS_OBJC_BEGIN_TRY_ABORT_BLOCK_NSRESULT;
|
||||
|
||||
if (aRequest != mIconRequest) return NS_ERROR_FAILURE;
|
||||
if (aRequest != mIconRequest)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
// Only support one frame.
|
||||
if (mLoadedIcon)
|
||||
|
|
@ -338,22 +338,22 @@ nsMenuItemIconX::OnStopFrame(imgIRequest* aRequest,
|
|||
|
||||
if (!mNativeMenuItem) return NS_ERROR_FAILURE;
|
||||
|
||||
nsCOMPtr<gfxIImageFrame> frame = aFrame;
|
||||
nsCOMPtr<nsIImage> image = do_GetInterface(frame);
|
||||
if (!image) return NS_ERROR_FAILURE;
|
||||
nsCOMPtr<imgIContainer> imageContainer;
|
||||
aRequest->GetImage(getter_AddRefs(imageContainer));
|
||||
if (!imageContainer)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
nsresult rv = image->LockImagePixels(PR_FALSE);
|
||||
if (NS_FAILED(rv))
|
||||
return rv;
|
||||
nsRefPtr<gfxImageSurface> image;
|
||||
imageContainer->CopyCurrentFrame(getter_AddRefs(image));
|
||||
|
||||
PRInt32 height = image->GetHeight();
|
||||
PRInt32 stride = image->GetLineStride();
|
||||
PRInt32 width = image->GetWidth();
|
||||
PRInt32 height = image->Height();
|
||||
PRInt32 stride = image->Stride();
|
||||
PRInt32 width = image->Width();
|
||||
PRUint32 imageLength = ((stride * height) / 4);
|
||||
if ((stride % 4 != 0) || (height < 1) || (width < 1))
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
PRUint32* imageData = (PRUint32*)image->GetBits();
|
||||
PRUint32* imageData = (PRUint32*)image->Data();
|
||||
|
||||
PRUint32* reorderedData = (PRUint32*)malloc(height * stride);
|
||||
if (!reorderedData)
|
||||
|
|
@ -378,12 +378,6 @@ nsMenuItemIconX::OnStopFrame(imgIRequest* aRequest,
|
|||
NULL, true, kCGRenderingIntentDefault);
|
||||
::CGDataProviderRelease(provider);
|
||||
|
||||
rv = image->UnlockImagePixels(PR_FALSE);
|
||||
if (NS_FAILED(rv)) {
|
||||
::CGColorSpaceRelease(colorSpace);
|
||||
return rv;
|
||||
}
|
||||
|
||||
// The image may not be the right size for a menu icon (16x16).
|
||||
// Create a new CGImage for the menu item.
|
||||
PRUint8* bitmap = (PRUint8*)malloc(kIconBytes);
|
||||
|
|
|
|||
|
|
@ -44,10 +44,11 @@
|
|||
#include "nsPrimitiveHelpers.h"
|
||||
#include "nsICharsetConverterManager.h"
|
||||
#include "nsIServiceManager.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsImageToPixbuf.h"
|
||||
#include "nsStringStream.h"
|
||||
|
||||
#include "imgIContainer.h"
|
||||
|
||||
#include <gtk/gtk.h>
|
||||
|
||||
// For manipulation of the X event queue
|
||||
|
|
@ -239,22 +240,17 @@ nsClipboard::SetData(nsITransferable *aTransferable,
|
|||
|
||||
nsCOMPtr<nsISupports> primitiveData;
|
||||
ptrPrimitive->GetData(getter_AddRefs(primitiveData));
|
||||
nsCOMPtr<nsIImage> image(do_QueryInterface(primitiveData));
|
||||
nsCOMPtr<imgIContainer> image(do_QueryInterface(primitiveData));
|
||||
if (!image) // Not getting an image for an image mime type!?
|
||||
continue;
|
||||
|
||||
if (NS_FAILED(image->LockImagePixels(PR_FALSE)))
|
||||
continue;
|
||||
GdkPixbuf* pixbuf = nsImageToPixbuf::ImageToPixbuf(image);
|
||||
if (!pixbuf) {
|
||||
image->UnlockImagePixels(PR_FALSE);
|
||||
if (!pixbuf)
|
||||
continue;
|
||||
}
|
||||
|
||||
GtkClipboard *aClipboard = gtk_clipboard_get(GetSelectionAtom(aWhichClipboard));
|
||||
gtk_clipboard_set_image(aClipboard, pixbuf);
|
||||
g_object_unref(pixbuf);
|
||||
image->UnlockImagePixels(PR_FALSE);
|
||||
continue;
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -44,17 +44,17 @@
|
|||
{ 0xdfa4ac93, 0x83f2, 0x4ab8, \
|
||||
{ 0x9b, 0x2a, 0x0f, 0xf7, 0x02, 0x2a, 0xeb, 0xe2 } }
|
||||
|
||||
class nsIImage;
|
||||
class imgIContainer;
|
||||
typedef struct _GdkPixbuf GdkPixbuf;
|
||||
|
||||
/**
|
||||
* An interface that allows converting an nsIImage to a GdkPixbuf*.
|
||||
* An interface that allows converting the current frame of an imgIContainer to a GdkPixbuf*.
|
||||
*/
|
||||
class nsIImageToPixbuf : public nsISupports {
|
||||
public:
|
||||
NS_DECLARE_STATIC_IID_ACCESSOR(NSIIMAGETOPIXBUF_IID)
|
||||
|
||||
NS_IMETHOD_(GdkPixbuf*) ConvertImageToPixbuf(nsIImage* aImage) = 0;
|
||||
NS_IMETHOD_(GdkPixbuf*) ConvertImageToPixbuf(imgIContainer* aImage) = 0;
|
||||
};
|
||||
|
||||
NS_DEFINE_STATIC_IID_ACCESSOR(nsIImageToPixbuf, NSIIMAGETOPIXBUF_IID)
|
||||
|
|
|
|||
|
|
@ -41,7 +41,7 @@
|
|||
#include "gfxImageSurface.h"
|
||||
#include "gfxContext.h"
|
||||
|
||||
#include "nsIImage.h"
|
||||
#include "imgIContainer.h"
|
||||
|
||||
#include "nsAutoPtr.h"
|
||||
|
||||
|
|
@ -60,21 +60,18 @@ unpremultiply (unsigned char color,
|
|||
}
|
||||
|
||||
NS_IMETHODIMP_(GdkPixbuf*)
|
||||
nsImageToPixbuf::ConvertImageToPixbuf(nsIImage* aImage)
|
||||
nsImageToPixbuf::ConvertImageToPixbuf(imgIContainer* aImage)
|
||||
{
|
||||
return ImageToPixbuf(aImage);
|
||||
}
|
||||
|
||||
GdkPixbuf*
|
||||
nsImageToPixbuf::ImageToPixbuf(nsIImage* aImage)
|
||||
nsImageToPixbuf::ImageToPixbuf(imgIContainer* aImage)
|
||||
{
|
||||
PRInt32 width = aImage->GetWidth(),
|
||||
height = aImage->GetHeight();
|
||||
nsRefPtr<gfxImageSurface> frame;
|
||||
aImage->CopyCurrentFrame(getter_AddRefs(frame));
|
||||
|
||||
nsRefPtr<gfxPattern> pattern;
|
||||
aImage->GetPattern(getter_AddRefs(pattern));
|
||||
|
||||
return PatternToPixbuf(pattern, width, height);
|
||||
return ImgSurfaceToPixbuf(frame, frame->Width(), frame->Height());
|
||||
}
|
||||
|
||||
GdkPixbuf*
|
||||
|
|
@ -158,39 +155,3 @@ nsImageToPixbuf::SurfaceToPixbuf(gfxASurface* aSurface, PRInt32 aWidth, PRInt32
|
|||
|
||||
return ImgSurfaceToPixbuf(imgSurface, aWidth, aHeight);
|
||||
}
|
||||
|
||||
GdkPixbuf*
|
||||
nsImageToPixbuf::PatternToPixbuf(gfxPattern* aPattern, PRInt32 aWidth, PRInt32 aHeight)
|
||||
{
|
||||
if (aPattern->CairoStatus()) {
|
||||
NS_ERROR("invalid pattern");
|
||||
return nsnull;
|
||||
}
|
||||
|
||||
nsRefPtr<gfxImageSurface> imgSurface;
|
||||
if (aPattern->GetType() == gfxPattern::PATTERN_SURFACE) {
|
||||
nsRefPtr<gfxASurface> surface = aPattern->GetSurface();
|
||||
if (surface->GetType() == gfxASurface::SurfaceTypeImage) {
|
||||
imgSurface = static_cast<gfxImageSurface*>
|
||||
(static_cast<gfxASurface*>(surface.get()));
|
||||
}
|
||||
}
|
||||
|
||||
if (!imgSurface) {
|
||||
imgSurface = new gfxImageSurface(gfxIntSize(aWidth, aHeight),
|
||||
gfxImageSurface::ImageFormatARGB32);
|
||||
|
||||
if (!imgSurface)
|
||||
return nsnull;
|
||||
|
||||
nsRefPtr<gfxContext> context = new gfxContext(imgSurface);
|
||||
if (!context)
|
||||
return nsnull;
|
||||
|
||||
context->SetOperator(gfxContext::OPERATOR_SOURCE);
|
||||
context->SetPattern(aPattern);
|
||||
context->Paint();
|
||||
}
|
||||
|
||||
return ImgSurfaceToPixbuf(imgSurface, aWidth, aHeight);
|
||||
}
|
||||
|
|
|
|||
|
|
@ -47,15 +47,13 @@ class gfxImageSurface;
|
|||
class nsImageToPixbuf : public nsIImageToPixbuf {
|
||||
public:
|
||||
NS_DECL_ISUPPORTS
|
||||
NS_IMETHOD_(GdkPixbuf*) ConvertImageToPixbuf(nsIImage* aImage);
|
||||
NS_IMETHOD_(GdkPixbuf*) ConvertImageToPixbuf(imgIContainer* aImage);
|
||||
|
||||
// Friendlier version of ConvertImageToPixbuf for callers inside of
|
||||
// widget
|
||||
static GdkPixbuf* ImageToPixbuf(nsIImage* aImage);
|
||||
static GdkPixbuf* ImageToPixbuf(imgIContainer * aImage);
|
||||
static GdkPixbuf* SurfaceToPixbuf(gfxASurface* aSurface,
|
||||
PRInt32 aWidth, PRInt32 aHeight);
|
||||
static GdkPixbuf* PatternToPixbuf(gfxPattern* aPattern,
|
||||
PRInt32 aWidth, PRInt32 aHeight);
|
||||
private:
|
||||
static GdkPixbuf* ImgSurfaceToPixbuf(gfxImageSurface* aImgSurface,
|
||||
PRInt32 aWidth, PRInt32 aHeight);
|
||||
|
|
|
|||
|
|
@ -104,9 +104,7 @@ static const char sAccessibilityKey [] = "config.use_system_prefs.accessibility"
|
|||
/* SetCursor(imgIContainer*) */
|
||||
#include <gdk/gdk.h>
|
||||
#include "imgIContainer.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsGfxCIID.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsImageToPixbuf.h"
|
||||
#include "nsIInterfaceRequestorUtils.h"
|
||||
#include "nsAutoPtr.h"
|
||||
|
|
@ -1572,17 +1570,8 @@ nsWindow::SetCursor(imgIContainer* aCursor,
|
|||
}
|
||||
mCursor = nsCursor(-1);
|
||||
|
||||
// Get first image frame
|
||||
nsCOMPtr<gfxIImageFrame> frame;
|
||||
aCursor->GetFrameAt(0, getter_AddRefs(frame));
|
||||
if (!frame)
|
||||
return NS_ERROR_NOT_AVAILABLE;
|
||||
|
||||
nsCOMPtr<nsIImage> img(do_GetInterface(frame));
|
||||
if (!img)
|
||||
return NS_ERROR_NOT_AVAILABLE;
|
||||
|
||||
GdkPixbuf* pixbuf = nsImageToPixbuf::ImageToPixbuf(img);
|
||||
// Get the image's current frame
|
||||
GdkPixbuf* pixbuf = nsImageToPixbuf::ImageToPixbuf(aCursor);
|
||||
if (!pixbuf)
|
||||
return NS_ERROR_NOT_AVAILABLE;
|
||||
|
||||
|
|
|
|||
|
|
@ -78,7 +78,6 @@
|
|||
#include "nsOS2Uni.h"
|
||||
|
||||
#include "imgIContainer.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
|
||||
#include <stdlib.h>
|
||||
#include <ctype.h>
|
||||
|
|
@ -1711,39 +1710,19 @@ NS_IMETHODIMP nsWindow::SetCursor(imgIContainer* aCursor,
|
|||
return NS_OK;
|
||||
}
|
||||
|
||||
nsCOMPtr<gfxIImageFrame> frame;
|
||||
aCursor->GetFrameAt(0, getter_AddRefs(frame));
|
||||
nsRefPtr<gfxImageFrame> frame;
|
||||
aCursor->CopyCurrentFrame(getter_AddRefs(frame));
|
||||
if (!frame)
|
||||
return NS_ERROR_NOT_AVAILABLE;
|
||||
|
||||
// if the image is ridiculously large, exit because
|
||||
// it will be unrecognizable when shrunk to 32x32
|
||||
PRInt32 width, height;
|
||||
frame->GetWidth(&width);
|
||||
frame->GetHeight(&height);
|
||||
PRInt32 width = frame->Width();
|
||||
PRInt32 height = frame->Height();
|
||||
if (width > 128 || height > 128)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
gfx_format format;
|
||||
nsresult rv = frame->GetFormat(&format);
|
||||
if (NS_FAILED(rv))
|
||||
return rv;
|
||||
|
||||
// only 24-bit images with 0, 1, or 8-bit alpha data are supported.
|
||||
// These are all the formats used in Cairo, and all map to the RGB24 resp. ARGB32 Cairo formats.
|
||||
if (format != gfxIFormats::BGR_A1 && format != gfxIFormats::RGB_A1 &&
|
||||
format != gfxIFormats::BGR_A8 && format != gfxIFormats::RGB_A8 &&
|
||||
format != gfxIFormats::BGR && format != gfxIFormats::RGB)
|
||||
return NS_ERROR_UNEXPECTED;
|
||||
|
||||
frame->LockImageData();
|
||||
PRUint32 dataLen;
|
||||
PRUint8* data;
|
||||
rv = frame->GetImageData(&data, &dataLen);
|
||||
if (NS_FAILED(rv)) {
|
||||
frame->UnlockImageData();
|
||||
return rv;
|
||||
}
|
||||
PRUint8* data = frame->Data();
|
||||
|
||||
// create the color bitmap
|
||||
HBITMAP hBmp = CreateBitmapRGB(data, width, height);
|
||||
|
|
@ -1751,15 +1730,12 @@ NS_IMETHODIMP nsWindow::SetCursor(imgIContainer* aCursor,
|
|||
return NS_ERROR_FAILURE;
|
||||
|
||||
// create a transparency mask from the alpha bytes
|
||||
HBITMAP hAlpha = CreateTransparencyMask(format, data, width, height);
|
||||
HBITMAP hAlpha = CreateTransparencyMask(frame->Format(), data, width, height);
|
||||
if (!hAlpha) {
|
||||
GpiDeleteBitmap(hBmp);
|
||||
return NS_ERROR_FAILURE;
|
||||
}
|
||||
|
||||
// Unlock image data after both processing colors and alpha data
|
||||
frame->UnlockImageData();
|
||||
|
||||
POINTERINFO info = {0};
|
||||
info.fPointer = TRUE;
|
||||
info.xHotspot = aHotspotX;
|
||||
|
|
@ -1875,7 +1851,7 @@ HBITMAP nsWindow::CreateBitmapRGB(PRUint8* aImageData,
|
|||
|
||||
// create a monochrome AND/XOR bitmap from 0, 1, or 8-bit alpha data
|
||||
|
||||
HBITMAP nsWindow::CreateTransparencyMask(gfx_format format,
|
||||
HBITMAP nsWindow::CreateTransparencyMask(gfxASurface::gfxImageFormat format,
|
||||
PRUint8* aImageData,
|
||||
PRUint32 aWidth,
|
||||
PRUint32 aHeight)
|
||||
|
|
@ -1889,36 +1865,29 @@ HBITMAP nsWindow::CreateTransparencyMask(gfx_format format,
|
|||
if (!mono)
|
||||
return NULL;
|
||||
|
||||
switch (format) {
|
||||
// gfxIFormats::BGR and case gfxIFormats::RGB are already
|
||||
// taken care of by initializing XOR and AND masks to zero
|
||||
if (format == gfxASurface::ImageFormatARGB32) {
|
||||
// Non-alpha formats are already taken care of by initializing the XOR and
|
||||
// AND masks to zero
|
||||
|
||||
// make the AND mask the inverse of the 8-bit alpha data
|
||||
case gfxIFormats::BGR_A1:
|
||||
case gfxIFormats::RGB_A1:
|
||||
case gfxIFormats::RGB_A8:
|
||||
case gfxIFormats::BGR_A8: {
|
||||
PRInt32* pSrc = (PRInt32*)aImageData;
|
||||
for (PRUint32 row = aHeight; row > 0; --row) {
|
||||
// Point to the right row in the AND mask
|
||||
PRUint8* pDst = mono + cbData + abpr * (row - 1);
|
||||
PRUint8 mask = 0x80;
|
||||
for (PRUint32 col = aWidth; col > 0; --col) {
|
||||
// Use sign bit to test for transparency, as alpha byte is highest byte
|
||||
// Positive means, alpha < 128, so consider as transparent and set the AND mask
|
||||
if (*pSrc++ >= 0) {
|
||||
*pDst |= mask;
|
||||
}
|
||||
PRInt32* pSrc = (PRInt32*)aImageData;
|
||||
for (PRUint32 row = aHeight; row > 0; --row) {
|
||||
// Point to the right row in the AND mask
|
||||
PRUint8* pDst = mono + cbData + abpr * (row - 1);
|
||||
PRUint8 mask = 0x80;
|
||||
for (PRUint32 col = aWidth; col > 0; --col) {
|
||||
// Use sign bit to test for transparency, as alpha byte is highest byte
|
||||
// Positive means, alpha < 128, so consider as transparent and set the AND mask
|
||||
if (*pSrc++ >= 0) {
|
||||
*pDst |= mask;
|
||||
}
|
||||
|
||||
mask >>= 1;
|
||||
if (!mask) {
|
||||
pDst++;
|
||||
mask = 0x80;
|
||||
}
|
||||
mask >>= 1;
|
||||
if (!mask) {
|
||||
pDst++;
|
||||
mask = 0x80;
|
||||
}
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -313,9 +313,8 @@ protected:
|
|||
HBITMAP DataToBitmap(PRUint8* aImageData, PRUint32 aWidth,
|
||||
PRUint32 aHeight, PRUint32 aDepth);
|
||||
HBITMAP CreateBitmapRGB(PRUint8* aImageData, PRUint32 aWidth, PRUint32 aHeight);
|
||||
// 'format' should be 'gfx_format' which is a PRInt32
|
||||
HBITMAP CreateTransparencyMask(PRInt32 format, PRUint8* aImageData,
|
||||
PRUint32 aWidth, PRUint32 aHeight);
|
||||
HBITMAP CreateTransparencyMask(gfxASurface::gfxImageFormat format,
|
||||
PRUint8* aImageData, PRUint32 aWidth, PRUint32 aHeight);
|
||||
|
||||
BOOL NotifyForeignChildWindows(HWND aWnd);
|
||||
void ScrollChildWindows(PRInt32 aX, PRInt32 aY);
|
||||
|
|
|
|||
|
|
@ -76,9 +76,7 @@
|
|||
|
||||
/* SetCursor(imgIContainer*) */
|
||||
#include "imgIContainer.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsGfxCIID.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsIInterfaceRequestorUtils.h"
|
||||
#include "nsAutoPtr.h"
|
||||
|
||||
|
|
|
|||
|
|
@ -70,8 +70,6 @@
|
|||
#include "nsNetUtil.h"
|
||||
#include "nsEscape.h"
|
||||
|
||||
#include "nsIImage.h"
|
||||
|
||||
|
||||
// oddly, this isn't in the MSVC headers anywhere.
|
||||
UINT nsClipboard::CF_HTML = ::RegisterClipboardFormatW(L"HTML Format");
|
||||
|
|
|
|||
|
|
@ -51,7 +51,6 @@
|
|||
#include "IEnumFE.h"
|
||||
#include "nsPrimitiveHelpers.h"
|
||||
#include "nsXPIDLString.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsImageClipboard.h"
|
||||
#include "nsCRT.h"
|
||||
#include "nsPrintfCString.h"
|
||||
|
|
@ -859,7 +858,7 @@ nsDataObj::GetBitmap ( const nsACString& , FORMATETC&, STGMEDIUM& )
|
|||
// GetDIB
|
||||
//
|
||||
// Someone is asking for a bitmap. The data in the transferable will be a straight
|
||||
// nsIImage, so just QI it.
|
||||
// imgIContainer, so just QI it.
|
||||
//
|
||||
HRESULT
|
||||
nsDataObj :: GetDib ( const nsACString& inFlavor, FORMATETC &, STGMEDIUM & aSTG )
|
||||
|
|
@ -869,12 +868,11 @@ nsDataObj :: GetDib ( const nsACString& inFlavor, FORMATETC &, STGMEDIUM & aSTG
|
|||
PRUint32 len = 0;
|
||||
nsCOMPtr<nsISupports> genericDataWrapper;
|
||||
mTransferable->GetTransferData(PromiseFlatCString(inFlavor).get(), getter_AddRefs(genericDataWrapper), &len);
|
||||
nsCOMPtr<nsIImage> image ( do_QueryInterface(genericDataWrapper) );
|
||||
nsCOMPtr<imgIContainer> image ( do_QueryInterface(genericDataWrapper) );
|
||||
if ( !image ) {
|
||||
// In the 0.9.4 timeframe, I had some embedding clients put the nsIImage directly into the
|
||||
// transferable. Newer code, however, wraps the nsIImage in a nsISupportsInterfacePointer.
|
||||
// We should be backwards compatibile with code already out in the field. If we can't find
|
||||
// the image directly out of the transferable, unwrap the image from its wrapper.
|
||||
// Check if the image was put in an nsISupportsInterfacePointer wrapper.
|
||||
// This might not be necessary any more, but could be useful for backwards
|
||||
// compatibility.
|
||||
nsCOMPtr<nsISupportsInterfacePointer> ptr(do_QueryInterface(genericDataWrapper));
|
||||
if ( ptr )
|
||||
ptr->GetData(getter_AddRefs(image));
|
||||
|
|
@ -1384,13 +1382,12 @@ HRESULT nsDataObj::GetFile(FORMATETC& aFE, STGMEDIUM& aSTG)
|
|||
nsCOMPtr<nsISupports> genericDataWrapper;
|
||||
|
||||
mTransferable->GetTransferData(kNativeImageMime, getter_AddRefs(genericDataWrapper), &len);
|
||||
nsCOMPtr<nsIImage> image ( do_QueryInterface(genericDataWrapper) );
|
||||
nsCOMPtr<imgIContainer> image ( do_QueryInterface(genericDataWrapper) );
|
||||
|
||||
if (!image) {
|
||||
// In the 0.9.4 timeframe, I had some embedding clients put the nsIImage directly into the
|
||||
// transferable. Newer code, however, wraps the nsIImage in a nsISupportsInterfacePointer.
|
||||
// We should be backwards compatibile with code already out in the field. If we can't find
|
||||
// the image directly out of the transferable, unwrap the image from its wrapper.
|
||||
// Check if the image was put in an nsISupportsInterfacePointer wrapper.
|
||||
// This might not be necessary any more, but could be useful for backwards
|
||||
// compatibility.
|
||||
nsCOMPtr<nsISupportsInterfacePointer> ptr(do_QueryInterface(genericDataWrapper));
|
||||
if (ptr)
|
||||
ptr->GetData(getter_AddRefs(image));
|
||||
|
|
|
|||
|
|
@ -59,9 +59,9 @@ Any other render format? HTML?
|
|||
//
|
||||
// nsImageToClipboard ctor
|
||||
//
|
||||
// Given an nsIImage, convert it to a DIB that is ready to go on the win32 clipboard
|
||||
// Given an imgIContainer, convert it to a DIB that is ready to go on the win32 clipboard
|
||||
//
|
||||
nsImageToClipboard :: nsImageToClipboard ( nsIImage* inImage )
|
||||
nsImageToClipboard :: nsImageToClipboard ( imgIContainer* inImage )
|
||||
: mImage(inImage)
|
||||
{
|
||||
// nothing to do here
|
||||
|
|
@ -143,19 +143,21 @@ nsImageToClipboard::CalcSpanLength(PRUint32 aWidth, PRUint32 aBitCount)
|
|||
// image.
|
||||
//
|
||||
nsresult
|
||||
nsImageToClipboard::CreateFromImage ( nsIImage* inImage, HANDLE* outBitmap )
|
||||
nsImageToClipboard::CreateFromImage ( imgIContainer* inImage, HANDLE* outBitmap )
|
||||
{
|
||||
nsresult result = NS_OK;
|
||||
*outBitmap = nsnull;
|
||||
|
||||
inImage->LockImagePixels(PR_FALSE);
|
||||
nsRefPtr<gfxImageSurface> frame;
|
||||
nsresult rv = inImage->CopyCurrentFrame(getter_AddRefs(frame));
|
||||
if (NS_FAILED(rv))
|
||||
return rv;
|
||||
|
||||
const PRUint32 imageSize = inImage->GetLineStride() * inImage->GetHeight();
|
||||
const PRUint32 imageSize = frame->GetDataSize();
|
||||
const PRInt32 bitmapSize = sizeof(BITMAPINFOHEADER) + imageSize;
|
||||
|
||||
HGLOBAL glob = ::GlobalAlloc(GMEM_MOVEABLE | GMEM_DDESHARE | GMEM_ZEROINIT, bitmapSize);
|
||||
if (!glob) {
|
||||
inImage->UnlockImagePixels(PR_FALSE);
|
||||
return NS_ERROR_OUT_OF_MEMORY;
|
||||
}
|
||||
|
||||
|
|
@ -164,18 +166,21 @@ nsImageToClipboard::CreateFromImage ( nsIImage* inImage, HANDLE* outBitmap )
|
|||
|
||||
BITMAPINFOHEADER *header = (BITMAPINFOHEADER*)data;
|
||||
header->biSize = sizeof(BITMAPINFOHEADER);
|
||||
header->biWidth = inImage->GetWidth();
|
||||
header->biHeight = inImage->GetHeight();
|
||||
header->biWidth = frame->Width();
|
||||
header->biHeight = frame->Height();
|
||||
|
||||
header->biPlanes = 1;
|
||||
header->biBitCount = inImage->GetBytesPix() * 8;
|
||||
if (frame->Format() == gfxASurface::ImageFormatARGB32)
|
||||
header->biBitCount = 32;
|
||||
else if (frame->Format() == gfxASurface::ImageFormatRGB24)
|
||||
header->biBitCount = 24;
|
||||
header->biCompression = BI_RGB;
|
||||
header->biSizeImage = imageSize;
|
||||
|
||||
const PRUint32 bpr = inImage->GetLineStride();
|
||||
const PRUint32 bpr = frame->Stride();
|
||||
|
||||
BYTE *dstBits = (BYTE*)data + sizeof(BITMAPINFOHEADER);
|
||||
BYTE *srcBits = inImage->GetBits();
|
||||
BYTE *srcBits = frame->Data();
|
||||
for (PRInt32 i = 0; i < header->biHeight; ++i) {
|
||||
PRUint32 srcOffset = imageSize - (bpr * (i + 1));
|
||||
PRUint32 dstOffset = i * bpr;
|
||||
|
|
@ -186,11 +191,7 @@ nsImageToClipboard::CreateFromImage ( nsIImage* inImage, HANDLE* outBitmap )
|
|||
|
||||
*outBitmap = (HANDLE)glob;
|
||||
|
||||
inImage->UnlockImagePixels(PR_FALSE);
|
||||
return NS_OK;
|
||||
|
||||
// Wow the old code is broken. I'm not touching it.
|
||||
// It should probably lock/unlock the same object....
|
||||
}
|
||||
|
||||
nsImageFromClipboard :: nsImageFromClipboard ()
|
||||
|
|
|
|||
|
|
@ -50,20 +50,20 @@ Any other render format? HTML?
|
|||
#include <windows.h>
|
||||
|
||||
#include "nsCOMPtr.h"
|
||||
#include "nsIImage.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "nsIInputStream.h"
|
||||
|
||||
|
||||
//
|
||||
// nsImageToClipboard
|
||||
//
|
||||
// A utility class that takes an nsIImage and does all the bitmap magic
|
||||
// A utility class that takes an imgIContainer and does all the bitmap magic
|
||||
// to allow us to put it on the clipboard
|
||||
//
|
||||
class nsImageToClipboard
|
||||
{
|
||||
public:
|
||||
nsImageToClipboard ( nsIImage* inImage );
|
||||
nsImageToClipboard ( imgIContainer* inImage );
|
||||
~nsImageToClipboard();
|
||||
|
||||
// Call to get the actual bits that go on the clipboard. If |nsnull|, the
|
||||
|
|
@ -79,9 +79,9 @@ private:
|
|||
PRInt32 CalcSpanLength(PRUint32 aWidth, PRUint32 aBitCount);
|
||||
|
||||
// Do the work
|
||||
nsresult CreateFromImage ( nsIImage* inImage, HANDLE* outBitmap );
|
||||
nsresult CreateFromImage ( imgIContainer* inImage, HANDLE* outBitmap );
|
||||
|
||||
nsCOMPtr<nsIImage> mImage; // the image we're working with
|
||||
nsCOMPtr<imgIContainer> mImage; // the image we're working with
|
||||
|
||||
}; // class nsImageToClipboard
|
||||
|
||||
|
|
|
|||
|
|
@ -119,13 +119,11 @@
|
|||
#include "nsISupportsPrimitives.h"
|
||||
#include "nsIDOMNSUIEvent.h"
|
||||
#include "nsITheme.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsIPrefBranch.h"
|
||||
#include "nsIPrefService.h"
|
||||
#include "nsIObserverService.h"
|
||||
#include "nsIScreenManager.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "nsIFile.h"
|
||||
#include "nsIRollupListener.h"
|
||||
#include "nsIMenuRollup.h"
|
||||
|
|
@ -2058,14 +2056,13 @@ NS_IMETHODIMP nsWindow::SetCursor(imgIContainer* aCursor,
|
|||
}
|
||||
|
||||
// Get the image data
|
||||
nsCOMPtr<gfxIImageFrame> frame;
|
||||
aCursor->GetFrameAt(0, getter_AddRefs(frame));
|
||||
nsRefPtr<gfxImageSurface> frame;
|
||||
aCursor->CopyCurrentFrame(getter_AddRefs(frame));
|
||||
if (!frame)
|
||||
return NS_ERROR_NOT_AVAILABLE;
|
||||
|
||||
PRInt32 width, height;
|
||||
frame->GetWidth(&width);
|
||||
frame->GetHeight(&height);
|
||||
PRInt32 width = frame->Width();
|
||||
PRInt32 height = frame->Height();
|
||||
|
||||
// Reject cursors greater than 128 pixels in either direction, to prevent
|
||||
// spoofing.
|
||||
|
|
@ -2074,19 +2071,10 @@ NS_IMETHODIMP nsWindow::SetCursor(imgIContainer* aCursor,
|
|||
if (width > 128 || height > 128)
|
||||
return NS_ERROR_NOT_AVAILABLE;
|
||||
|
||||
frame->LockImageData();
|
||||
|
||||
PRUint32 dataLen;
|
||||
PRUint8 *data;
|
||||
nsresult rv = frame->GetImageData(&data, &dataLen);
|
||||
if (NS_FAILED(rv)) {
|
||||
frame->UnlockImageData();
|
||||
return rv;
|
||||
}
|
||||
PRUint8 *data = frame->Data();
|
||||
|
||||
HBITMAP bmp = DataToBitmap(data, width, -height, 32);
|
||||
PRUint8* a1data = Data32BitTo1Bit(data, width, height);
|
||||
frame->UnlockImageData();
|
||||
if (!a1data) {
|
||||
return NS_ERROR_FAILURE;
|
||||
}
|
||||
|
|
|
|||
|
|
@ -62,9 +62,7 @@
|
|||
#include "nsIDOMDataTransfer.h"
|
||||
#include "nsIEventStateManager.h"
|
||||
#include "nsICanvasElement.h"
|
||||
#include "nsIImage.h"
|
||||
#include "nsIImageLoadingContent.h"
|
||||
#include "gfxIImageFrame.h"
|
||||
#include "imgIContainer.h"
|
||||
#include "imgIRequest.h"
|
||||
#include "nsIViewObserver.h"
|
||||
|
|
@ -530,7 +528,7 @@ nsBaseDragService::DrawDragForImage(nsPresContext* aPresContext,
|
|||
nsIntRect* aScreenDragRect,
|
||||
gfxASurface** aSurface)
|
||||
{
|
||||
nsCOMPtr<nsIImage> img;
|
||||
nsCOMPtr<imgIContainer> imgContainer;
|
||||
if (aImageLoader) {
|
||||
nsCOMPtr<imgIRequest> imgRequest;
|
||||
nsresult rv = aImageLoader->GetRequest(nsIImageLoadingContent::CURRENT_REQUEST,
|
||||
|
|
@ -539,21 +537,11 @@ nsBaseDragService::DrawDragForImage(nsPresContext* aPresContext,
|
|||
if (!imgRequest)
|
||||
return NS_ERROR_NOT_AVAILABLE;
|
||||
|
||||
nsCOMPtr<imgIContainer> imgContainer;
|
||||
rv = imgRequest->GetImage(getter_AddRefs(imgContainer));
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
if (!imgContainer)
|
||||
return NS_ERROR_NOT_AVAILABLE;
|
||||
|
||||
nsCOMPtr<gfxIImageFrame> iframe;
|
||||
imgContainer->GetCurrentFrame(getter_AddRefs(iframe));
|
||||
if (!iframe)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
img = do_GetInterface(iframe);
|
||||
if (!img)
|
||||
return NS_ERROR_FAILURE;
|
||||
|
||||
// use the size of the image as the size of the drag image
|
||||
imgContainer->GetWidth(&aScreenDragRect->width);
|
||||
imgContainer->GetHeight(&aScreenDragRect->height);
|
||||
|
|
@ -612,8 +600,8 @@ nsBaseDragService::DrawDragForImage(nsPresContext* aPresContext,
|
|||
gfxRect outRect(0, 0, destSize.width, destSize.height);
|
||||
gfxMatrix scale =
|
||||
gfxMatrix().Scale(srcSize.width/outRect.Width(), srcSize.height/outRect.Height());
|
||||
img->Draw(ctx, gfxPattern::FILTER_GOOD, scale, outRect, nsIntMargin(0,0,0,0),
|
||||
nsIntRect(0, 0, srcSize.width, srcSize.height));
|
||||
nsIntRect imgSize(0, 0, srcSize.width, srcSize.height);
|
||||
imgContainer->Draw(ctx, gfxPattern::FILTER_GOOD, scale, outRect, imgSize);
|
||||
return NS_OK;
|
||||
} else {
|
||||
return aCanvas->RenderContexts(ctx, gfxPattern::FILTER_GOOD);
|
||||
|
|
|
|||
Loading…
Reference in a new issue