mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:30:27 +00:00
cbc917768b
- Bug 1266578 - OOM crash if malloc fails in ProcessIncomingMessages(). r=billm (5eca51be20) - Bug 1261340: Don't refer to |ScopedFreePtrTraits| in |RawDBusConnection|, r=shuang (61996db57c) - Bug 1246931: Move |DBusWatcher| into its own files, r=shuang (a7fc4a68c3) - Bug 1246931: Add support for |RefPtr<DBusConnection>| and convert callers, r=shuang (6c07d3e8d3) - Bug 1246931: Add support for |RefPtr<DBusMessage>|, r=shuang (c0ed14cc2b) - Bug 1246931: Add support for |RefPtr<DBusPendingCall>|, r=shuang (a007a15699) - Bug 1246931: Use |DBusConnection| in |DBusWatcher|, r=shuang (11b5eaf535) - Bug 1246931: Add DBus I/O helpers, r=shuang (be249fa65b) - Bug 1264398 - Avoid extra assign() on windows in IPC code (r=jld) (0f31b798a3) - Bug 1261231: Fix shutdown leak in imgLoader::GetInstance. r=gabor (96e5143c41) - Bug 1256999 - Use nsIDocument for ImageCacheKey. r=bz r=seth (cd4765ce34) - Bug 1257101. imgFrame::IsImageComplete says whether we've had pixels decoded to the whole image rect, but it's used to check if the frame is finished decoding. These are different things when the image has more than one progress pass. r=seth (06d619410f) - Bug 1222596. If RasterImage::LookupFrame does (some) sync decoding and encouters an error we don't want to return the surface with an error. r=seth (c4dbc8e0d5) - Bug 763784 - Make VectorImage::GetAnimated check for CSS animations. r=dholbert (4d0e5b88eb) - Bug 1210745 - Update CheckProgressConsistency() to match current ImageLib behavior. r=tn (2317b24fcc) - Bug 1249576. Add crashtest. (485f03120b) - Bug 1251091. Add crashtest. (7e1682a1e6) - Bug 1253362. SVGDocumentWrapper::IsAnimated can be called after SVGDocumentWrapper::DestroyViewer so null check mViewer. r=dholbert (2b4a1c4619) - Bug 1210745. Change image progress asserts to allow transparency to be posted after the size is posted. r=seth (699f3c5496) - Bug 1209780. Use the DrawResult return value of imgIContainer::Draw in the cocoa code. r=seth (abaea789e3) - No Bug - Remove some unnecessary SVGImageContext.h includes and add comments. r=sparky (8232661a23) - cleanup style (de33e4bfa8) - Bug 860857, support custom datatransfer types using a special type, r=smaug,jmathies,mstange (cf7e19deb6) - Bug 1248459 - Don't query out-of-bounds selection; r=masayuki (86c127f143) - Bug 1261671 - ContentEventHandler::ConvertToRootRelativeOffset() should return the root-relative result in the frame's own appUnits, not the root's appUnits in the case when they're different. r=masayuki (2cd72b5ebc) - Bug 1266702 - Clean up formatting in dom/events/DataTransfer.* and mark some methods const, r=jwatt (bd439cdad5)
571 lines
16 KiB
C++
571 lines
16 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
*
|
|
* 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/. */
|
|
|
|
#include "ImageLogging.h"
|
|
#include "ProgressTracker.h"
|
|
|
|
#include "imgIContainer.h"
|
|
#include "imgINotificationObserver.h"
|
|
#include "imgIRequest.h"
|
|
#include "Image.h"
|
|
#include "nsNetUtil.h"
|
|
#include "nsIObserverService.h"
|
|
|
|
#include "mozilla/Assertions.h"
|
|
#include "mozilla/Services.h"
|
|
|
|
using mozilla::WeakPtr;
|
|
|
|
namespace mozilla {
|
|
namespace image {
|
|
|
|
static void
|
|
CheckProgressConsistency(Progress aOldProgress, Progress aNewProgress)
|
|
{
|
|
// Check preconditions for every progress bit.
|
|
|
|
if (aNewProgress & FLAG_SIZE_AVAILABLE) {
|
|
// No preconditions.
|
|
}
|
|
if (aNewProgress & FLAG_DECODE_COMPLETE) {
|
|
MOZ_ASSERT(aNewProgress & FLAG_SIZE_AVAILABLE);
|
|
MOZ_ASSERT(aNewProgress & (FLAG_FRAME_COMPLETE | FLAG_HAS_ERROR));
|
|
}
|
|
if (aNewProgress & FLAG_FRAME_COMPLETE) {
|
|
MOZ_ASSERT(aNewProgress & FLAG_SIZE_AVAILABLE);
|
|
}
|
|
if (aNewProgress & FLAG_LOAD_COMPLETE) {
|
|
MOZ_ASSERT(aNewProgress & (FLAG_SIZE_AVAILABLE | FLAG_HAS_ERROR));
|
|
}
|
|
if (aNewProgress & FLAG_ONLOAD_BLOCKED) {
|
|
// No preconditions.
|
|
}
|
|
if (aNewProgress & FLAG_ONLOAD_UNBLOCKED) {
|
|
MOZ_ASSERT(aNewProgress & FLAG_ONLOAD_BLOCKED);
|
|
MOZ_ASSERT(aNewProgress & (FLAG_SIZE_AVAILABLE | FLAG_HAS_ERROR));
|
|
}
|
|
if (aNewProgress & FLAG_IS_ANIMATED) {
|
|
// No preconditions; like FLAG_HAS_TRANSPARENCY, we should normally never
|
|
// discover this *after* FLAG_SIZE_AVAILABLE, but unfortunately some corrupt
|
|
// GIFs may fool us.
|
|
}
|
|
if (aNewProgress & FLAG_HAS_TRANSPARENCY) {
|
|
// XXX We'd like to assert that transparency is only set during metadata
|
|
// decode but we don't have any way to assert that until bug 1254892 is fixed.
|
|
}
|
|
if (aNewProgress & FLAG_LAST_PART_COMPLETE) {
|
|
MOZ_ASSERT(aNewProgress & FLAG_LOAD_COMPLETE);
|
|
}
|
|
if (aNewProgress & FLAG_HAS_ERROR) {
|
|
// No preconditions.
|
|
}
|
|
}
|
|
|
|
void
|
|
ProgressTracker::SetImage(Image* aImage)
|
|
{
|
|
MutexAutoLock lock(mImageMutex);
|
|
MOZ_ASSERT(aImage, "Setting null image");
|
|
MOZ_ASSERT(!mImage, "Setting image when we already have one");
|
|
mImage = aImage;
|
|
}
|
|
|
|
void
|
|
ProgressTracker::ResetImage()
|
|
{
|
|
MutexAutoLock lock(mImageMutex);
|
|
MOZ_ASSERT(mImage, "Resetting image when it's already null!");
|
|
mImage = nullptr;
|
|
}
|
|
|
|
uint32_t
|
|
ProgressTracker::GetImageStatus() const
|
|
{
|
|
uint32_t status = imgIRequest::STATUS_NONE;
|
|
|
|
// Translate our current state to a set of imgIRequest::STATE_* flags.
|
|
if (mProgress & FLAG_SIZE_AVAILABLE) {
|
|
status |= imgIRequest::STATUS_SIZE_AVAILABLE;
|
|
}
|
|
if (mProgress & FLAG_DECODE_COMPLETE) {
|
|
status |= imgIRequest::STATUS_DECODE_COMPLETE;
|
|
}
|
|
if (mProgress & FLAG_FRAME_COMPLETE) {
|
|
status |= imgIRequest::STATUS_FRAME_COMPLETE;
|
|
}
|
|
if (mProgress & FLAG_LOAD_COMPLETE) {
|
|
status |= imgIRequest::STATUS_LOAD_COMPLETE;
|
|
}
|
|
if (mProgress & FLAG_IS_ANIMATED) {
|
|
status |= imgIRequest::STATUS_IS_ANIMATED;
|
|
}
|
|
if (mProgress & FLAG_HAS_TRANSPARENCY) {
|
|
status |= imgIRequest::STATUS_HAS_TRANSPARENCY;
|
|
}
|
|
if (mProgress & FLAG_HAS_ERROR) {
|
|
status |= imgIRequest::STATUS_ERROR;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
// A helper class to allow us to call SyncNotify asynchronously.
|
|
class AsyncNotifyRunnable : public nsRunnable
|
|
{
|
|
public:
|
|
AsyncNotifyRunnable(ProgressTracker* aTracker,
|
|
IProgressObserver* aObserver)
|
|
: mTracker(aTracker)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread(), "Should be created on the main thread");
|
|
MOZ_ASSERT(aTracker, "aTracker should not be null");
|
|
MOZ_ASSERT(aObserver, "aObserver should not be null");
|
|
mObservers.AppendElement(aObserver);
|
|
}
|
|
|
|
NS_IMETHOD Run()
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread(), "Should be running on the main thread");
|
|
MOZ_ASSERT(mTracker, "mTracker should not be null");
|
|
for (uint32_t i = 0; i < mObservers.Length(); ++i) {
|
|
mObservers[i]->SetNotificationsDeferred(false);
|
|
mTracker->SyncNotify(mObservers[i]);
|
|
}
|
|
|
|
mTracker->mRunnable = nullptr;
|
|
return NS_OK;
|
|
}
|
|
|
|
void AddObserver(IProgressObserver* aObserver)
|
|
{
|
|
mObservers.AppendElement(aObserver);
|
|
}
|
|
|
|
void RemoveObserver(IProgressObserver* aObserver)
|
|
{
|
|
mObservers.RemoveElement(aObserver);
|
|
}
|
|
|
|
private:
|
|
friend class ProgressTracker;
|
|
|
|
RefPtr<ProgressTracker> mTracker;
|
|
nsTArray<RefPtr<IProgressObserver>> mObservers;
|
|
};
|
|
|
|
void
|
|
ProgressTracker::Notify(IProgressObserver* aObserver)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (MOZ_LOG_TEST(gImgLog, LogLevel::Debug)) {
|
|
RefPtr<Image> image = GetImage();
|
|
if (image && image->GetURI()) {
|
|
RefPtr<ImageURL> uri(image->GetURI());
|
|
nsAutoCString spec;
|
|
uri->GetSpec(spec);
|
|
LOG_FUNC_WITH_PARAM(gImgLog,
|
|
"ProgressTracker::Notify async", "uri", spec.get());
|
|
} else {
|
|
LOG_FUNC_WITH_PARAM(gImgLog,
|
|
"ProgressTracker::Notify async", "uri", "<unknown>");
|
|
}
|
|
}
|
|
|
|
aObserver->SetNotificationsDeferred(true);
|
|
|
|
// If we have an existing runnable that we can use, we just append this
|
|
// observer to its list of observers to be notified. This ensures we don't
|
|
// unnecessarily delay onload.
|
|
AsyncNotifyRunnable* runnable =
|
|
static_cast<AsyncNotifyRunnable*>(mRunnable.get());
|
|
|
|
if (runnable) {
|
|
runnable->AddObserver(aObserver);
|
|
} else {
|
|
mRunnable = new AsyncNotifyRunnable(this, aObserver);
|
|
NS_DispatchToCurrentThread(mRunnable);
|
|
}
|
|
}
|
|
|
|
// A helper class to allow us to call SyncNotify asynchronously for a given,
|
|
// fixed, state.
|
|
class AsyncNotifyCurrentStateRunnable : public nsRunnable
|
|
{
|
|
public:
|
|
AsyncNotifyCurrentStateRunnable(ProgressTracker* aProgressTracker,
|
|
IProgressObserver* aObserver)
|
|
: mProgressTracker(aProgressTracker)
|
|
, mObserver(aObserver)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread(), "Should be created on the main thread");
|
|
MOZ_ASSERT(mProgressTracker, "mProgressTracker should not be null");
|
|
MOZ_ASSERT(mObserver, "mObserver should not be null");
|
|
mImage = mProgressTracker->GetImage();
|
|
}
|
|
|
|
NS_IMETHOD Run()
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread(), "Should be running on the main thread");
|
|
mObserver->SetNotificationsDeferred(false);
|
|
|
|
mProgressTracker->SyncNotify(mObserver);
|
|
return NS_OK;
|
|
}
|
|
|
|
private:
|
|
RefPtr<ProgressTracker> mProgressTracker;
|
|
RefPtr<IProgressObserver> mObserver;
|
|
|
|
// We have to hold on to a reference to the tracker's image, just in case
|
|
// it goes away while we're in the event queue.
|
|
RefPtr<Image> mImage;
|
|
};
|
|
|
|
void
|
|
ProgressTracker::NotifyCurrentState(IProgressObserver* aObserver)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (MOZ_LOG_TEST(gImgLog, LogLevel::Debug)) {
|
|
RefPtr<Image> image = GetImage();
|
|
nsAutoCString spec;
|
|
if (image && image->GetURI()) {
|
|
image->GetURI()->GetSpec(spec);
|
|
}
|
|
LOG_FUNC_WITH_PARAM(gImgLog,
|
|
"ProgressTracker::NotifyCurrentState", "uri", spec.get());
|
|
}
|
|
|
|
aObserver->SetNotificationsDeferred(true);
|
|
|
|
nsCOMPtr<nsIRunnable> ev = new AsyncNotifyCurrentStateRunnable(this,
|
|
aObserver);
|
|
NS_DispatchToCurrentThread(ev);
|
|
}
|
|
|
|
/**
|
|
* ImageObserverNotifier is a helper type that abstracts over the difference
|
|
* between sending notifications to all of the observers in an ObserverTable,
|
|
* and sending them to a single observer. This allows the same notification code
|
|
* to be used for both cases.
|
|
*/
|
|
template <typename T> struct ImageObserverNotifier;
|
|
|
|
template <>
|
|
struct MOZ_STACK_CLASS ImageObserverNotifier<const ObserverTable*>
|
|
{
|
|
explicit ImageObserverNotifier(const ObserverTable* aObservers,
|
|
bool aIgnoreDeferral = false)
|
|
: mObservers(aObservers)
|
|
, mIgnoreDeferral(aIgnoreDeferral)
|
|
{ }
|
|
|
|
template <typename Lambda>
|
|
void operator()(Lambda aFunc)
|
|
{
|
|
for (auto iter = mObservers->ConstIter(); !iter.Done(); iter.Next()) {
|
|
RefPtr<IProgressObserver> observer = iter.Data().get();
|
|
if (observer &&
|
|
(mIgnoreDeferral || !observer->NotificationsDeferred())) {
|
|
aFunc(observer);
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
const ObserverTable* mObservers;
|
|
const bool mIgnoreDeferral;
|
|
};
|
|
|
|
template <>
|
|
struct MOZ_STACK_CLASS ImageObserverNotifier<IProgressObserver*>
|
|
{
|
|
explicit ImageObserverNotifier(IProgressObserver* aObserver)
|
|
: mObserver(aObserver)
|
|
{ }
|
|
|
|
template <typename Lambda>
|
|
void operator()(Lambda aFunc)
|
|
{
|
|
if (mObserver && !mObserver->NotificationsDeferred()) {
|
|
aFunc(mObserver);
|
|
}
|
|
}
|
|
|
|
private:
|
|
IProgressObserver* mObserver;
|
|
};
|
|
|
|
template <typename T> void
|
|
SyncNotifyInternal(const T& aObservers,
|
|
bool aHasImage,
|
|
Progress aProgress,
|
|
const nsIntRect& aDirtyRect)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
typedef imgINotificationObserver I;
|
|
ImageObserverNotifier<T> notify(aObservers);
|
|
|
|
if (aProgress & FLAG_SIZE_AVAILABLE) {
|
|
notify([](IProgressObserver* aObs) { aObs->Notify(I::SIZE_AVAILABLE); });
|
|
}
|
|
|
|
if (aProgress & FLAG_ONLOAD_BLOCKED) {
|
|
notify([](IProgressObserver* aObs) { aObs->BlockOnload(); });
|
|
}
|
|
|
|
if (aHasImage) {
|
|
// OnFrameUpdate
|
|
// If there's any content in this frame at all (always true for
|
|
// vector images, true for raster images that have decoded at
|
|
// least one frame) then send OnFrameUpdate.
|
|
if (!aDirtyRect.IsEmpty()) {
|
|
notify([&](IProgressObserver* aObs) {
|
|
aObs->Notify(I::FRAME_UPDATE, &aDirtyRect);
|
|
});
|
|
}
|
|
|
|
if (aProgress & FLAG_FRAME_COMPLETE) {
|
|
notify([](IProgressObserver* aObs) { aObs->Notify(I::FRAME_COMPLETE); });
|
|
}
|
|
|
|
if (aProgress & FLAG_HAS_TRANSPARENCY) {
|
|
notify([](IProgressObserver* aObs) { aObs->Notify(I::HAS_TRANSPARENCY); });
|
|
}
|
|
|
|
if (aProgress & FLAG_IS_ANIMATED) {
|
|
notify([](IProgressObserver* aObs) { aObs->Notify(I::IS_ANIMATED); });
|
|
}
|
|
}
|
|
|
|
// Send UnblockOnload before OnStopDecode and OnStopRequest. This allows
|
|
// observers that can fire events when they receive those notifications to do
|
|
// so then, instead of being forced to wait for UnblockOnload.
|
|
if (aProgress & FLAG_ONLOAD_UNBLOCKED) {
|
|
notify([](IProgressObserver* aObs) { aObs->UnblockOnload(); });
|
|
}
|
|
|
|
if (aProgress & FLAG_DECODE_COMPLETE) {
|
|
MOZ_ASSERT(aHasImage, "Stopped decoding without ever having an image?");
|
|
notify([](IProgressObserver* aObs) { aObs->Notify(I::DECODE_COMPLETE); });
|
|
}
|
|
|
|
if (aProgress & FLAG_LOAD_COMPLETE) {
|
|
notify([=](IProgressObserver* aObs) {
|
|
aObs->OnLoadComplete(aProgress & FLAG_LAST_PART_COMPLETE);
|
|
});
|
|
}
|
|
}
|
|
|
|
void
|
|
ProgressTracker::SyncNotifyProgress(Progress aProgress,
|
|
const nsIntRect& aInvalidRect
|
|
/* = nsIntRect() */)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread(), "Use mObservers on main thread only");
|
|
|
|
// Don't unblock onload if we're not blocked.
|
|
Progress progress = Difference(aProgress);
|
|
if (!((mProgress | progress) & FLAG_ONLOAD_BLOCKED)) {
|
|
progress &= ~FLAG_ONLOAD_UNBLOCKED;
|
|
}
|
|
|
|
CheckProgressConsistency(mProgress, mProgress | progress);
|
|
|
|
// XXX(seth): Hack to work around the fact that some observers have bugs and
|
|
// need to get onload blocking notifications multiple times. We should fix
|
|
// those observers and remove this.
|
|
if ((aProgress & FLAG_DECODE_COMPLETE) &&
|
|
(mProgress & FLAG_ONLOAD_BLOCKED) &&
|
|
(mProgress & FLAG_ONLOAD_UNBLOCKED)) {
|
|
progress |= FLAG_ONLOAD_BLOCKED | FLAG_ONLOAD_UNBLOCKED;
|
|
}
|
|
|
|
// Apply the changes.
|
|
mProgress |= progress;
|
|
|
|
// Send notifications.
|
|
mObservers.Read([&](const ObserverTable* aTable) {
|
|
SyncNotifyInternal(aTable, HasImage(), progress, aInvalidRect);
|
|
});
|
|
|
|
if (progress & FLAG_HAS_ERROR) {
|
|
FireFailureNotification();
|
|
}
|
|
}
|
|
|
|
void
|
|
ProgressTracker::SyncNotify(IProgressObserver* aObserver)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
RefPtr<Image> image = GetImage();
|
|
|
|
nsAutoCString spec;
|
|
if (image && image->GetURI()) {
|
|
image->GetURI()->GetSpec(spec);
|
|
}
|
|
LOG_SCOPE_WITH_PARAM(gImgLog,
|
|
"ProgressTracker::SyncNotify", "uri", spec.get());
|
|
|
|
nsIntRect rect;
|
|
if (image) {
|
|
if (NS_FAILED(image->GetWidth(&rect.width)) ||
|
|
NS_FAILED(image->GetHeight(&rect.height))) {
|
|
// Either the image has no intrinsic size, or it has an error.
|
|
rect = GetMaxSizedIntRect();
|
|
}
|
|
}
|
|
|
|
SyncNotifyInternal(aObserver, !!image, mProgress, rect);
|
|
}
|
|
|
|
void
|
|
ProgressTracker::EmulateRequestFinished(IProgressObserver* aObserver)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread(),
|
|
"SyncNotifyState and mObservers are not threadsafe");
|
|
RefPtr<IProgressObserver> kungFuDeathGrip(aObserver);
|
|
|
|
if (mProgress & FLAG_ONLOAD_BLOCKED && !(mProgress & FLAG_ONLOAD_UNBLOCKED)) {
|
|
aObserver->UnblockOnload();
|
|
}
|
|
|
|
if (!(mProgress & FLAG_LOAD_COMPLETE)) {
|
|
aObserver->OnLoadComplete(true);
|
|
}
|
|
}
|
|
|
|
void
|
|
ProgressTracker::AddObserver(IProgressObserver* aObserver)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
RefPtr<IProgressObserver> observer = aObserver;
|
|
|
|
mObservers.Write([=](ObserverTable* aTable) {
|
|
MOZ_ASSERT(!aTable->Get(observer, nullptr),
|
|
"Adding duplicate entry for image observer");
|
|
|
|
WeakPtr<IProgressObserver> weakPtr = observer.get();
|
|
aTable->Put(observer, weakPtr);
|
|
});
|
|
}
|
|
|
|
bool
|
|
ProgressTracker::RemoveObserver(IProgressObserver* aObserver)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
RefPtr<IProgressObserver> observer = aObserver;
|
|
|
|
// Remove the observer from the list.
|
|
bool removed = mObservers.Write([=](ObserverTable* aTable) {
|
|
bool removed = aTable->Get(observer, nullptr);
|
|
aTable->Remove(observer);
|
|
return removed;
|
|
});
|
|
|
|
// Observers can get confused if they don't get all the proper teardown
|
|
// notifications. Part ways on good terms.
|
|
if (removed && !aObserver->NotificationsDeferred()) {
|
|
EmulateRequestFinished(aObserver);
|
|
}
|
|
|
|
// Make sure we don't give callbacks to an observer that isn't interested in
|
|
// them any more.
|
|
AsyncNotifyRunnable* runnable =
|
|
static_cast<AsyncNotifyRunnable*>(mRunnable.get());
|
|
|
|
if (aObserver->NotificationsDeferred() && runnable) {
|
|
runnable->RemoveObserver(aObserver);
|
|
aObserver->SetNotificationsDeferred(false);
|
|
}
|
|
|
|
return removed;
|
|
}
|
|
|
|
uint32_t
|
|
ProgressTracker::ObserverCount() const
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
return mObservers.Read([](const ObserverTable* aTable) {
|
|
return aTable->Count();
|
|
});
|
|
}
|
|
|
|
void
|
|
ProgressTracker::OnUnlockedDraw()
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
mObservers.Read([](const ObserverTable* aTable) {
|
|
ImageObserverNotifier<const ObserverTable*> notify(aTable);
|
|
notify([](IProgressObserver* aObs) {
|
|
aObs->Notify(imgINotificationObserver::UNLOCKED_DRAW);
|
|
});
|
|
});
|
|
}
|
|
|
|
void
|
|
ProgressTracker::ResetForNewRequest()
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
mProgress = NoProgress;
|
|
}
|
|
|
|
void
|
|
ProgressTracker::OnDiscard()
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
mObservers.Read([](const ObserverTable* aTable) {
|
|
ImageObserverNotifier<const ObserverTable*> notify(aTable);
|
|
notify([](IProgressObserver* aObs) {
|
|
aObs->Notify(imgINotificationObserver::DISCARD);
|
|
});
|
|
});
|
|
}
|
|
|
|
void
|
|
ProgressTracker::OnImageAvailable()
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
// Notify any imgRequestProxys that are observing us that we have an Image.
|
|
mObservers.Read([](const ObserverTable* aTable) {
|
|
ImageObserverNotifier<const ObserverTable*>
|
|
notify(aTable, /* aIgnoreDeferral = */ true);
|
|
notify([](IProgressObserver* aObs) {
|
|
aObs->SetHasImage();
|
|
});
|
|
});
|
|
}
|
|
|
|
void
|
|
ProgressTracker::FireFailureNotification()
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
// Some kind of problem has happened with image decoding.
|
|
// Report the URI to net:failed-to-process-uri-conent observers.
|
|
RefPtr<Image> image = GetImage();
|
|
if (image) {
|
|
// Should be on main thread, so ok to create a new nsIURI.
|
|
nsCOMPtr<nsIURI> uri;
|
|
{
|
|
RefPtr<ImageURL> threadsafeUriData = image->GetURI();
|
|
uri = threadsafeUriData ? threadsafeUriData->ToIURI() : nullptr;
|
|
}
|
|
if (uri) {
|
|
nsCOMPtr<nsIObserverService> os = mozilla::services::GetObserverService();
|
|
if (os) {
|
|
os->NotifyObservers(uri, "net:failed-to-process-uri-content", nullptr);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace image
|
|
} // namespace mozilla
|