/* -*- Mode: C++; tab-width: 8; 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 "nsBrowserStatusFilter.h" #include "nsITimer.h" #include "nsString.h" #include "nsThreadUtils.h" using namespace mozilla; //----------------------------------------------------------------------------- // nsBrowserStatusFilter //----------------------------------------------------------------------------- nsBrowserStatusFilter::nsBrowserStatusFilter() : mTarget(GetMainThreadEventTarget()), mCurProgress(0), mMaxProgress(0), mCurrentPercentage(0), mStatusIsDirty(true), mIsLoadingDocument(false), mDelayedStatus(false), mDelayedProgress(false) {} nsBrowserStatusFilter::~nsBrowserStatusFilter() { if (mTimer) { mTimer->Cancel(); } } //----------------------------------------------------------------------------- // nsBrowserStatusFilter::nsISupports //----------------------------------------------------------------------------- NS_IMPL_CYCLE_COLLECTION_WEAK(nsBrowserStatusFilter, mListener, mTarget) NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(nsBrowserStatusFilter) NS_INTERFACE_MAP_ENTRY(nsIWebProgress) NS_INTERFACE_MAP_ENTRY(nsIWebProgressListener) NS_INTERFACE_MAP_ENTRY(nsIWebProgressListener2) NS_INTERFACE_MAP_ENTRY(nsISupportsWeakReference) NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsIWebProgress) NS_INTERFACE_MAP_END NS_IMPL_CYCLE_COLLECTING_ADDREF(nsBrowserStatusFilter) NS_IMPL_CYCLE_COLLECTING_RELEASE(nsBrowserStatusFilter) //----------------------------------------------------------------------------- // nsBrowserStatusFilter::nsIWebProgress //----------------------------------------------------------------------------- NS_IMETHODIMP nsBrowserStatusFilter::AddProgressListener(nsIWebProgressListener* aListener, uint32_t aNotifyMask) { mListener = aListener; return NS_OK; } NS_IMETHODIMP nsBrowserStatusFilter::RemoveProgressListener( nsIWebProgressListener* aListener) { if (aListener == mListener) mListener = nullptr; return NS_OK; } NS_IMETHODIMP nsBrowserStatusFilter::GetBrowsingContextXPCOM( mozilla::dom::BrowsingContext** aResult) { MOZ_ASSERT_UNREACHABLE("nsBrowserStatusFilter::GetBrowsingContextXPCOM"); return NS_ERROR_NOT_IMPLEMENTED; } mozilla::dom::BrowsingContext* nsBrowserStatusFilter::GetBrowsingContext() { MOZ_ASSERT_UNREACHABLE("nsBrowserStatusFilter::GetBrowsingContext"); return nullptr; } NS_IMETHODIMP nsBrowserStatusFilter::GetDOMWindow(mozIDOMWindowProxy** aResult) { MOZ_ASSERT_UNREACHABLE("nsBrowserStatusFilter::GetDOMWindow"); return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsBrowserStatusFilter::GetIsTopLevel(bool* aIsTopLevel) { *aIsTopLevel = false; MOZ_ASSERT_UNREACHABLE("nsBrowserStatusFilter::GetIsTopLevel"); return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsBrowserStatusFilter::GetIsLoadingDocument(bool* aIsLoadingDocument) { MOZ_ASSERT_UNREACHABLE("nsBrowserStatusFilter::GetIsLoadingDocument"); return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsBrowserStatusFilter::GetLoadType(uint32_t* aLoadType) { *aLoadType = 0; MOZ_ASSERT_UNREACHABLE("nsBrowserStatusFilter::GetLoadType"); return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsBrowserStatusFilter::GetTarget(nsIEventTarget** aTarget) { nsCOMPtr target = mTarget; target.forget(aTarget); return NS_OK; } NS_IMETHODIMP nsBrowserStatusFilter::SetTarget(nsIEventTarget* aTarget) { mTarget = aTarget; return NS_OK; } //----------------------------------------------------------------------------- // nsBrowserStatusFilter::nsIWebProgressListener //----------------------------------------------------------------------------- NS_IMETHODIMP nsBrowserStatusFilter::OnStateChange(nsIWebProgress* aWebProgress, nsIRequest* aRequest, uint32_t aStateFlags, nsresult aStatus) { if (!mListener) return NS_OK; if (aStateFlags & STATE_START) { // Reset members on beginning of document loading, but we don't want // subframe document loading followed by the root document loading // resets members accidentally, so for non-toplevel load we check if // there hasn't been a document load started. if (aStateFlags & STATE_IS_DOCUMENT) { bool isTopLevel = false; aWebProgress->GetIsTopLevel(&isTopLevel); if (!mIsLoadingDocument || isTopLevel) { ResetMembers(); } mIsLoadingDocument = true; } } else if (aStateFlags & STATE_STOP) { // Flush pending status / progress update during document loading. if (mIsLoadingDocument) { bool isLoadingDocument = true; aWebProgress->GetIsLoadingDocument(&isLoadingDocument); mIsLoadingDocument &= isLoadingDocument; if (mTimer) { mTimer->Cancel(); CallDelayedProgressListeners(); // CallDelayedProgressListeners() may trigger OnStatusChange and/or // OnProgressChange handlers, which can run JS and may even call // RemoveProgressListener. if (!mListener) { return NS_OK; } } } } else { // No need to forward this state change. return NS_OK; } // Only notify listener for STATE_IS_NETWORK or STATE_IS_REDIRECTED_DOCUMENT if (aStateFlags & STATE_IS_NETWORK || aStateFlags & STATE_IS_REDIRECTED_DOCUMENT) { return mListener->OnStateChange(aWebProgress, aRequest, aStateFlags, aStatus); } return NS_OK; } NS_IMETHODIMP nsBrowserStatusFilter::OnProgressChange(nsIWebProgress* aWebProgress, nsIRequest* aRequest, int32_t aCurSelfProgress, int32_t aMaxSelfProgress, int32_t aCurTotalProgress, int32_t aMaxTotalProgress) { if (!mListener) return NS_OK; // // limit frequency of calls to OnProgressChange // mCurProgress = (int64_t)aCurTotalProgress; mMaxProgress = (int64_t)aMaxTotalProgress; if (mDelayedProgress) return NS_OK; if (!mDelayedStatus) { MaybeSendProgress(); StartDelayTimer(); } mDelayedProgress = true; return NS_OK; } NS_IMETHODIMP nsBrowserStatusFilter::OnLocationChange(nsIWebProgress* aWebProgress, nsIRequest* aRequest, nsIURI* aLocation, uint32_t aFlags) { if (!mListener) return NS_OK; return mListener->OnLocationChange(aWebProgress, aRequest, aLocation, aFlags); } NS_IMETHODIMP nsBrowserStatusFilter::OnStatusChange(nsIWebProgress* aWebProgress, nsIRequest* aRequest, nsresult aStatus, const char16_t* aMessage) { if (!mListener) return NS_OK; // // limit frequency of calls to OnStatusChange // if (mStatusIsDirty || !mCurrentStatusMsg.Equals(aMessage)) { mStatusIsDirty = true; mStatusMsg = aMessage; } if (mDelayedStatus) return NS_OK; if (!mDelayedProgress) { MaybeSendStatus(); StartDelayTimer(); } mDelayedStatus = true; return NS_OK; } NS_IMETHODIMP nsBrowserStatusFilter::OnSecurityChange(nsIWebProgress* aWebProgress, nsIRequest* aRequest, uint32_t aState) { if (!mListener) return NS_OK; return mListener->OnSecurityChange(aWebProgress, aRequest, aState); } NS_IMETHODIMP nsBrowserStatusFilter::OnContentBlockingEvent(nsIWebProgress* aWebProgress, nsIRequest* aRequest, uint32_t aEvent) { if (!mListener) return NS_OK; return mListener->OnContentBlockingEvent(aWebProgress, aRequest, aEvent); } //----------------------------------------------------------------------------- // nsBrowserStatusFilter::nsIWebProgressListener2 //----------------------------------------------------------------------------- NS_IMETHODIMP nsBrowserStatusFilter::OnProgressChange64(nsIWebProgress* aWebProgress, nsIRequest* aRequest, int64_t aCurSelfProgress, int64_t aMaxSelfProgress, int64_t aCurTotalProgress, int64_t aMaxTotalProgress) { // XXX truncates 64-bit to 32-bit return OnProgressChange(aWebProgress, aRequest, (int32_t)aCurSelfProgress, (int32_t)aMaxSelfProgress, (int32_t)aCurTotalProgress, (int32_t)aMaxTotalProgress); } NS_IMETHODIMP nsBrowserStatusFilter::OnRefreshAttempted(nsIWebProgress* aWebProgress, nsIURI* aUri, uint32_t aDelay, bool aSameUri, bool* allowRefresh) { nsCOMPtr listener = do_QueryInterface(mListener); if (!listener) { *allowRefresh = true; return NS_OK; } return listener->OnRefreshAttempted(aWebProgress, aUri, aDelay, aSameUri, allowRefresh); } //----------------------------------------------------------------------------- // nsBrowserStatusFilter //----------------------------------------------------------------------------- void nsBrowserStatusFilter::ResetMembers() { mMaxProgress = 0; mCurProgress = 0; mCurrentPercentage = 0; mStatusIsDirty = true; // We don't reset mIsLoadingDocument here. // It's controlled by OnStateChange based on webProgress states. } void nsBrowserStatusFilter::MaybeSendProgress() { if (mCurProgress > mMaxProgress || mCurProgress <= 0) return; // check our percentage int32_t percentage = (int32_t) double(mCurProgress) * 100 / mMaxProgress; // The progress meter only updates for increases greater than 3 percent if (percentage > (mCurrentPercentage + 3)) { mCurrentPercentage = percentage; // XXX truncates 64-bit to 32-bit mListener->OnProgressChange(nullptr, nullptr, 0, 0, (int32_t)mCurProgress, (int32_t)mMaxProgress); } } void nsBrowserStatusFilter::MaybeSendStatus() { if (mStatusIsDirty) { mListener->OnStatusChange(nullptr, nullptr, NS_OK, mStatusMsg.get()); mCurrentStatusMsg = mStatusMsg; mStatusIsDirty = false; } } nsresult nsBrowserStatusFilter::StartDelayTimer() { NS_ASSERTION(!DelayInEffect(), "delay should not be in effect"); return NS_NewTimerWithFuncCallback(getter_AddRefs(mTimer), TimeoutHandler, this, 160, nsITimer::TYPE_ONE_SHOT, "nsBrowserStatusFilter::TimeoutHandler", mTarget); } void nsBrowserStatusFilter::CallDelayedProgressListeners() { mTimer = nullptr; if (!mListener) return; if (mDelayedStatus) { mDelayedStatus = false; MaybeSendStatus(); } if (mDelayedProgress) { mDelayedProgress = false; MaybeSendProgress(); } } void nsBrowserStatusFilter::TimeoutHandler(nsITimer* aTimer, void* aClosure) { nsBrowserStatusFilter* self = reinterpret_cast(aClosure); if (!self) { NS_ERROR("no self"); return; } self->CallDelayedProgressListeners(); } NS_IMETHODIMP nsBrowserStatusFilter::GetDocumentRequest(nsIRequest** aRequest) { *aRequest = nullptr; return NS_ERROR_NOT_IMPLEMENTED; }