mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
1a68eab98d
The nsCSSValue in nsGenericHTMLElement::MapBackgroundInto is a temporary. This causes a problem after Bug 697230 landed, because the nsCSSValue::Image we put into that value is destroyed once we're done doing style stuff. Previously the nsImageLoader would grab the request off the nsCSSValue::Image and hold it alive. Bug 697230 changed the behavior here; now when the nsCSSValue::Image is destroyed it tells the image loader to drop the request. The result is that all the references to the request are dropped and the frame is never told it has a background. The solution is to keep the nsCSSValue::Image alive longer. This patch adds two new types of nsAttrValue. The first is an nsCSSValue::URL. A ParseBackgroundAttribute method is added on nsGenericHTMLElement that the relevant elements (body/td/th/table/tr/tbody/thead/tfoot) call that parses background into an nsCSSValue::URL. The second is an nsCSSValue::Image. nsGenericHTMLElement::MapBackgroundInto attempts to convert the nsCSSValue::URL into an nsCSSValue::Image by kicking off the image load. The result is that image loads are only started when the element is actually visible. This also mirrors the way background-image works. This also allows us to fix two longstanding bugs in this code. Since MapBackgroundInto doesn't have a pointer to the actual element, it relied on grabbing the principal of the document. Now we can grab the principal of the node in ParseBackgroundAttribute. MapBackgroundInto also has no way to get at the element's base URI (to honor xml:base), which is now possible in ParseBackgroundAttribute. nsCSSValue::[Image|URL] have also been moved to be mozilla::css::[Image|URL]Value. nsAttrValue.h is included in external linkage code, so it can't include nsCSSValue.h to get the declarations of nsCSSValue::[Image|URL], and nested classes can't be forward declared. Moving the classes to a namespace solves the problem. Finally some old inoperative quirks mode code was removed. This code has done nothing since Bug 273078 was landed in 2004.
462 lines
11 KiB
C++
462 lines
11 KiB
C++
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
* You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
/* A class that handles style system image loads (other image loads are handled
|
|
* by the nodes in the content tree).
|
|
*/
|
|
|
|
#include "mozilla/css/ImageLoader.h"
|
|
#include "nsContentUtils.h"
|
|
#include "nsLayoutUtils.h"
|
|
#include "nsError.h"
|
|
|
|
namespace mozilla {
|
|
namespace css {
|
|
|
|
/* static */ PLDHashOperator
|
|
ImageLoader::SetAnimationModeEnumerator(nsISupports* aKey, FrameSet* aValue,
|
|
void* aClosure)
|
|
{
|
|
imgIRequest* request = static_cast<imgIRequest*>(aKey);
|
|
|
|
uint16_t* mode = static_cast<uint16_t*>(aClosure);
|
|
|
|
#ifdef DEBUG
|
|
{
|
|
nsCOMPtr<imgIRequest> debugRequest = do_QueryInterface(aKey);
|
|
NS_ASSERTION(debugRequest == request, "This is bad");
|
|
}
|
|
#endif
|
|
|
|
nsCOMPtr<imgIContainer> container;
|
|
request->GetImage(getter_AddRefs(container));
|
|
if (!container) {
|
|
return PL_DHASH_NEXT;
|
|
}
|
|
|
|
// This can fail if the image is in error, and we don't care.
|
|
container->SetAnimationMode(*mode);
|
|
|
|
return PL_DHASH_NEXT;
|
|
}
|
|
|
|
void
|
|
ImageLoader::DropDocumentReference()
|
|
{
|
|
ClearAll();
|
|
mDocument = nullptr;
|
|
}
|
|
|
|
void
|
|
ImageLoader::AssociateRequestToFrame(imgIRequest* aRequest,
|
|
nsIFrame* aFrame)
|
|
{
|
|
MOZ_ASSERT(mRequestToFrameMap.IsInitialized() &&
|
|
mFrameToRequestMap.IsInitialized() &&
|
|
mImages.IsInitialized());
|
|
|
|
nsCOMPtr<imgIDecoderObserver> observer;
|
|
aRequest->GetDecoderObserver(getter_AddRefs(observer));
|
|
if (!observer) {
|
|
// The request has already been canceled, so ignore it. This is ok because
|
|
// we're not going to get any more notifications from a canceled request.
|
|
return;
|
|
}
|
|
|
|
MOZ_ASSERT(observer == this);
|
|
|
|
FrameSet* frameSet = nullptr;
|
|
if (mRequestToFrameMap.Get(aRequest, &frameSet)) {
|
|
NS_ASSERTION(frameSet, "This should never be null!");
|
|
}
|
|
|
|
if (!frameSet) {
|
|
nsAutoPtr<FrameSet> newFrameSet(new FrameSet());
|
|
|
|
mRequestToFrameMap.Put(aRequest, newFrameSet);
|
|
frameSet = newFrameSet.forget();
|
|
|
|
nsPresContext* presContext = GetPresContext();
|
|
if (presContext) {
|
|
nsLayoutUtils::RegisterImageRequestIfAnimated(presContext,
|
|
aRequest,
|
|
nullptr);
|
|
}
|
|
}
|
|
|
|
RequestSet* requestSet = nullptr;
|
|
if (mFrameToRequestMap.Get(aFrame, &requestSet)) {
|
|
NS_ASSERTION(requestSet, "This should never be null");
|
|
}
|
|
|
|
if (!requestSet) {
|
|
nsAutoPtr<RequestSet> newRequestSet(new RequestSet());
|
|
|
|
mFrameToRequestMap.Put(aFrame, newRequestSet);
|
|
requestSet = newRequestSet.forget();
|
|
}
|
|
|
|
// Add these to the sets, but only if they're not already there.
|
|
uint32_t i;
|
|
if (!frameSet->GreatestIndexLtEq(aFrame, i)) {
|
|
frameSet->InsertElementAt(i, aFrame);
|
|
}
|
|
if (!requestSet->GreatestIndexLtEq(aRequest, i)) {
|
|
requestSet->InsertElementAt(i, aRequest);
|
|
}
|
|
}
|
|
|
|
void
|
|
ImageLoader::MaybeRegisterCSSImage(ImageLoader::Image* aImage)
|
|
{
|
|
NS_ASSERTION(aImage, "This should never be null!");
|
|
|
|
bool found = false;
|
|
aImage->mRequests.GetWeak(mDocument, &found);
|
|
if (found) {
|
|
// This document already has a request.
|
|
return;
|
|
}
|
|
|
|
imgIRequest* canonicalRequest = aImage->mRequests.GetWeak(nullptr);
|
|
if (!canonicalRequest) {
|
|
// The image was blocked or something.
|
|
return;
|
|
}
|
|
|
|
nsCOMPtr<imgIRequest> request;
|
|
|
|
// Ignore errors here. If cloning fails for some reason we'll put a null
|
|
// entry in the hash and we won't keep trying to clone.
|
|
mInClone = true;
|
|
canonicalRequest->Clone(this, getter_AddRefs(request));
|
|
mInClone = false;
|
|
|
|
aImage->mRequests.Put(mDocument, request);
|
|
|
|
AddImage(aImage);
|
|
}
|
|
|
|
void
|
|
ImageLoader::DeregisterCSSImage(ImageLoader::Image* aImage)
|
|
{
|
|
RemoveImage(aImage);
|
|
}
|
|
|
|
void
|
|
ImageLoader::DisassociateRequestFromFrame(imgIRequest* aRequest,
|
|
nsIFrame* aFrame)
|
|
{
|
|
FrameSet* frameSet = nullptr;
|
|
RequestSet* requestSet = nullptr;
|
|
|
|
MOZ_ASSERT(mRequestToFrameMap.IsInitialized() &&
|
|
mFrameToRequestMap.IsInitialized() &&
|
|
mImages.IsInitialized());
|
|
|
|
#ifdef DEBUG
|
|
{
|
|
nsCOMPtr<imgIDecoderObserver> observer;
|
|
aRequest->GetDecoderObserver(getter_AddRefs(observer));
|
|
MOZ_ASSERT(!observer || observer == this);
|
|
}
|
|
#endif
|
|
|
|
mRequestToFrameMap.Get(aRequest, &frameSet);
|
|
mFrameToRequestMap.Get(aFrame, &requestSet);
|
|
|
|
if (frameSet) {
|
|
frameSet->RemoveElementSorted(aFrame);
|
|
}
|
|
if (requestSet) {
|
|
requestSet->RemoveElementSorted(aRequest);
|
|
}
|
|
|
|
if (frameSet && !frameSet->Length()) {
|
|
mRequestToFrameMap.Remove(aRequest);
|
|
|
|
nsPresContext* presContext = GetPresContext();
|
|
if (presContext) {
|
|
nsLayoutUtils::DeregisterImageRequest(presContext,
|
|
aRequest,
|
|
nullptr);
|
|
}
|
|
}
|
|
|
|
if (requestSet && !requestSet->Length()) {
|
|
mFrameToRequestMap.Remove(aFrame);
|
|
}
|
|
}
|
|
|
|
void
|
|
ImageLoader::DropRequestsForFrame(nsIFrame* aFrame)
|
|
{
|
|
RequestSet* requestSet = nullptr;
|
|
if (!mFrameToRequestMap.Get(aFrame, &requestSet)) {
|
|
return;
|
|
}
|
|
|
|
NS_ASSERTION(requestSet, "This should never be null");
|
|
|
|
RequestSet frozenRequestSet(*requestSet);
|
|
for (RequestSet::size_type i = frozenRequestSet.Length(); i != 0; --i) {
|
|
imgIRequest* request = frozenRequestSet.ElementAt(i - 1);
|
|
|
|
DisassociateRequestFromFrame(request, aFrame);
|
|
}
|
|
}
|
|
|
|
void
|
|
ImageLoader::SetAnimationMode(uint16_t aMode)
|
|
{
|
|
NS_ASSERTION(aMode == imgIContainer::kNormalAnimMode ||
|
|
aMode == imgIContainer::kDontAnimMode ||
|
|
aMode == imgIContainer::kLoopOnceAnimMode,
|
|
"Wrong Animation Mode is being set!");
|
|
|
|
mRequestToFrameMap.EnumerateRead(SetAnimationModeEnumerator, &aMode);
|
|
}
|
|
|
|
static PLDHashOperator
|
|
ClearImageHashSet(nsPtrHashKey<ImageLoader::Image>* aKey, void* aClosure)
|
|
{
|
|
nsIDocument* doc = static_cast<nsIDocument*>(aClosure);
|
|
ImageLoader::Image* image = aKey->GetKey();
|
|
|
|
imgIRequest* request = image->mRequests.GetWeak(doc);
|
|
if (request) {
|
|
request->CancelAndForgetObserver(NS_BINDING_ABORTED);
|
|
}
|
|
|
|
image->mRequests.Remove(doc);
|
|
|
|
return PL_DHASH_REMOVE;
|
|
}
|
|
|
|
void
|
|
ImageLoader::ClearAll()
|
|
{
|
|
mRequestToFrameMap.Clear();
|
|
mFrameToRequestMap.Clear();
|
|
mImages.EnumerateEntries(&ClearImageHashSet, mDocument);
|
|
}
|
|
|
|
void
|
|
ImageLoader::LoadImage(nsIURI* aURI, nsIPrincipal* aOriginPrincipal,
|
|
nsIURI* aReferrer, ImageLoader::Image* aImage)
|
|
{
|
|
NS_ASSERTION(aImage->mRequests.Count() == 0, "Huh?");
|
|
|
|
aImage->mRequests.Put(nullptr, nullptr);
|
|
|
|
if (!aURI) {
|
|
return;
|
|
}
|
|
|
|
if (!nsContentUtils::CanLoadImage(aURI, mDocument, mDocument,
|
|
aOriginPrincipal)) {
|
|
return;
|
|
}
|
|
|
|
nsCOMPtr<imgIRequest> request;
|
|
nsContentUtils::LoadImage(aURI, mDocument, aOriginPrincipal, aReferrer,
|
|
nullptr, nsIRequest::LOAD_NORMAL,
|
|
getter_AddRefs(request));
|
|
|
|
if (!request) {
|
|
return;
|
|
}
|
|
|
|
nsCOMPtr<imgIRequest> clonedRequest;
|
|
mInClone = true;
|
|
nsresult rv = request->Clone(this, getter_AddRefs(clonedRequest));
|
|
mInClone = false;
|
|
|
|
if (NS_FAILED(rv)) {
|
|
return;
|
|
}
|
|
|
|
aImage->mRequests.Put(nullptr, request);
|
|
aImage->mRequests.Put(mDocument, clonedRequest);
|
|
|
|
AddImage(aImage);
|
|
}
|
|
|
|
void
|
|
ImageLoader::AddImage(ImageLoader::Image* aImage)
|
|
{
|
|
NS_ASSERTION(!mImages.Contains(aImage), "Huh?");
|
|
if (!mImages.PutEntry(aImage)) {
|
|
NS_RUNTIMEABORT("OOM");
|
|
}
|
|
}
|
|
|
|
void
|
|
ImageLoader::RemoveImage(ImageLoader::Image* aImage)
|
|
{
|
|
NS_ASSERTION(mImages.Contains(aImage), "Huh?");
|
|
mImages.RemoveEntry(aImage);
|
|
}
|
|
|
|
nsPresContext*
|
|
ImageLoader::GetPresContext()
|
|
{
|
|
if (!mDocument) {
|
|
return nullptr;
|
|
}
|
|
|
|
nsIPresShell* shell = mDocument->GetShell();
|
|
if (!shell) {
|
|
return nullptr;
|
|
}
|
|
|
|
return shell->GetPresContext();
|
|
}
|
|
|
|
void
|
|
ImageLoader::DoRedraw(FrameSet* aFrameSet)
|
|
{
|
|
NS_ASSERTION(aFrameSet, "Must have a frame set");
|
|
NS_ASSERTION(mDocument, "Should have returned earlier!");
|
|
|
|
FrameSet::size_type length = aFrameSet->Length();
|
|
for (FrameSet::size_type i = 0; i < length; i++) {
|
|
nsIFrame* frame = aFrameSet->ElementAt(i);
|
|
|
|
// NOTE: It is not sufficient to invalidate only the size of the image:
|
|
// the image may be tiled!
|
|
// The best option is to call into the frame, however lacking this
|
|
// we have to at least invalidate the frame's bounds, hence
|
|
// as long as we have a frame we'll use its size.
|
|
//
|
|
|
|
// Invalidate the entire frame
|
|
// XXX We really only need to invalidate the client area of the frame...
|
|
|
|
nsRect bounds(nsPoint(0, 0), frame->GetSize());
|
|
|
|
if (frame->GetType() == nsGkAtoms::canvasFrame) {
|
|
// The canvas's background covers the whole viewport.
|
|
bounds = frame->GetVisualOverflowRect();
|
|
}
|
|
|
|
if (frame->GetStyleVisibility()->IsVisible()) {
|
|
frame->Invalidate(bounds);
|
|
}
|
|
}
|
|
}
|
|
|
|
NS_IMPL_ADDREF(ImageLoader)
|
|
NS_IMPL_RELEASE(ImageLoader)
|
|
|
|
NS_INTERFACE_MAP_BEGIN(ImageLoader)
|
|
NS_INTERFACE_MAP_ENTRY(imgIDecoderObserver)
|
|
NS_INTERFACE_MAP_ENTRY(imgIContainerObserver)
|
|
NS_INTERFACE_MAP_ENTRY(imgIOnloadBlocker)
|
|
NS_INTERFACE_MAP_END
|
|
|
|
NS_IMETHODIMP
|
|
ImageLoader::OnStartContainer(imgIRequest* aRequest, imgIContainer* aImage)
|
|
{
|
|
nsPresContext* presContext = GetPresContext();
|
|
if (!presContext) {
|
|
return NS_OK;
|
|
}
|
|
|
|
aImage->SetAnimationMode(presContext->ImageAnimationMode());
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
ImageLoader::OnImageIsAnimated(imgIRequest* aRequest)
|
|
{
|
|
if (!mDocument) {
|
|
return NS_OK;
|
|
}
|
|
|
|
FrameSet* frameSet = nullptr;
|
|
if (!mRequestToFrameMap.Get(aRequest, &frameSet)) {
|
|
return NS_OK;
|
|
}
|
|
|
|
// Register with the refresh driver now that we are aware that
|
|
// we are animated.
|
|
nsPresContext* presContext = GetPresContext();
|
|
if (presContext) {
|
|
nsLayoutUtils::RegisterImageRequest(presContext,
|
|
aRequest,
|
|
nullptr);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
ImageLoader::OnStopFrame(imgIRequest *aRequest, uint32_t aFrame)
|
|
{
|
|
if (!mDocument || mInClone) {
|
|
return NS_OK;
|
|
}
|
|
|
|
FrameSet* frameSet = nullptr;
|
|
if (!mRequestToFrameMap.Get(aRequest, &frameSet)) {
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_ASSERTION(frameSet, "This should never be null!");
|
|
|
|
DoRedraw(frameSet);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
ImageLoader::FrameChanged(imgIRequest *aRequest,
|
|
imgIContainer *aContainer,
|
|
const nsIntRect *aDirtyRect)
|
|
{
|
|
if (!mDocument || mInClone) {
|
|
return NS_OK;
|
|
}
|
|
|
|
FrameSet* frameSet = nullptr;
|
|
if (!mRequestToFrameMap.Get(aRequest, &frameSet)) {
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_ASSERTION(frameSet, "This should never be null!");
|
|
|
|
DoRedraw(frameSet);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
ImageLoader::BlockOnload(imgIRequest* aRequest)
|
|
{
|
|
if (!mDocument) {
|
|
return NS_OK;
|
|
}
|
|
|
|
mDocument->BlockOnload();
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
ImageLoader::UnblockOnload(imgIRequest* aRequest)
|
|
{
|
|
if (!mDocument) {
|
|
return NS_OK;
|
|
}
|
|
|
|
mDocument->UnblockOnload(false);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
} // namespace css
|
|
} // namespace mozilla
|