diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-19 00:47:55 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-19 00:47:55 +0000 |
commit | 26a029d407be480d791972afb5975cf62c9360a6 (patch) | |
tree | f435a8308119effd964b339f76abb83a57c29483 /layout/generic/ScrollbarActivity.cpp | |
parent | Initial commit. (diff) | |
download | firefox-26a029d407be480d791972afb5975cf62c9360a6.tar.xz firefox-26a029d407be480d791972afb5975cf62c9360a6.zip |
Adding upstream version 124.0.1.upstream/124.0.1
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'layout/generic/ScrollbarActivity.cpp')
-rw-r--r-- | layout/generic/ScrollbarActivity.cpp | 280 |
1 files changed, 280 insertions, 0 deletions
diff --git a/layout/generic/ScrollbarActivity.cpp b/layout/generic/ScrollbarActivity.cpp new file mode 100644 index 0000000000..e9865c62f9 --- /dev/null +++ b/layout/generic/ScrollbarActivity.cpp @@ -0,0 +1,280 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* vim: set ts=8 sts=2 et sw=2 tw=80: */ +/* 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 "ScrollbarActivity.h" +#include "nsIScrollbarMediator.h" +#include "nsIContent.h" +#include "nsIFrame.h" +#include "nsContentUtils.h" +#include "nsITimer.h" +#include "nsQueryFrame.h" +#include "nsIScrollableFrame.h" +#include "PresShell.h" +#include "nsLayoutUtils.h" +#include "nsScrollbarFrame.h" +#include "nsRefreshDriver.h" +#include "mozilla/dom/Element.h" +#include "mozilla/dom/Event.h" +#include "mozilla/dom/Document.h" +#include "mozilla/LookAndFeel.h" + +namespace mozilla::layout { + +using mozilla::dom::Element; + +NS_IMPL_ISUPPORTS(ScrollbarActivity, nsIDOMEventListener) + +static bool DisplayOnMouseMove() { + return LookAndFeel::GetInt(LookAndFeel::IntID::ScrollbarDisplayOnMouseMove); +} + +void ScrollbarActivity::Destroy() { + StopListeningForScrollbarEvents(); + StopListeningForScrollAreaEvents(); + CancelFadeTimer(); +} + +void ScrollbarActivity::ActivityOccurred() { + ActivityStarted(); + ActivityStopped(); +} + +static void SetBooleanAttribute(Element* aElement, nsAtom* aAttribute, + bool aValue) { + if (aElement) { + if (aValue) { + aElement->SetAttr(kNameSpaceID_None, aAttribute, u"true"_ns, true); + } else { + aElement->UnsetAttr(kNameSpaceID_None, aAttribute, true); + } + } +} + +void ScrollbarActivity::ActivityStarted() { + const bool wasActive = IsActive(); + mNestedActivityCounter++; + if (wasActive) { + return; + } + CancelFadeTimer(); + StartListeningForScrollbarEvents(); + StartListeningForScrollAreaEvents(); + SetBooleanAttribute(GetHorizontalScrollbar(), nsGkAtoms::active, true); + SetBooleanAttribute(GetVerticalScrollbar(), nsGkAtoms::active, true); + mScrollbarEffectivelyVisible = true; +} + +void ScrollbarActivity::ActivityStopped() { + if (!IsActive()) { + // This can happen if there was a frame reconstruction while the activity + // was ongoing. In this case we just do nothing. We should probably handle + // this case better. + return; + } + mNestedActivityCounter--; + if (IsActive()) { + return; + } + // Clear sticky scrollbar hover status. + HoveredScrollbar(nullptr); + StartFadeTimer(); +} + +NS_IMETHODIMP +ScrollbarActivity::HandleEvent(dom::Event* aEvent) { + if (!mScrollbarEffectivelyVisible && !DisplayOnMouseMove()) { + return NS_OK; + } + + nsAutoString type; + aEvent->GetType(type); + + if (type.EqualsLiteral("mousemove")) { + // Mouse motions anywhere in the scrollable frame should keep the + // scrollbars visible, but we have to be careful as content descendants of + // our scrollable content aren't necessarily scrolled by our scroll frame + // (if they are out of flow and their containing block is not a descendant + // of our scroll frame) and we don't want those to activate us. + nsIFrame* scrollFrame = do_QueryFrame(mScrollableFrame); + MOZ_ASSERT(scrollFrame); + nsIScrollableFrame* scrollableFrame = do_QueryFrame(mScrollableFrame); + nsCOMPtr<nsIContent> targetContent = + do_QueryInterface(aEvent->GetOriginalTarget()); + nsIFrame* targetFrame = + targetContent ? targetContent->GetPrimaryFrame() : nullptr; + if ((scrollableFrame && scrollableFrame->IsRootScrollFrameOfDocument()) || + !targetFrame || + nsLayoutUtils::IsAncestorFrameCrossDocInProcess( + scrollFrame, targetFrame, + scrollFrame->PresShell()->GetRootFrame())) { + ActivityOccurred(); + } + return NS_OK; + } + + nsCOMPtr<nsIContent> targetContent = + do_QueryInterface(aEvent->GetOriginalTarget()); + + HandleEventForScrollbar(type, targetContent, GetHorizontalScrollbar(), + &mHScrollbarHovered); + HandleEventForScrollbar(type, targetContent, GetVerticalScrollbar(), + &mVScrollbarHovered); + + return NS_OK; +} + +void ScrollbarActivity::HandleEventForScrollbar(const nsAString& aType, + nsIContent* aTarget, + Element* aScrollbar, + bool* aStoredHoverState) { + if (!aTarget || !aScrollbar || + !aTarget->IsInclusiveDescendantOf(aScrollbar)) { + return; + } + + if (aType.EqualsLiteral("mousedown")) { + ActivityStarted(); + } else if (aType.EqualsLiteral("mouseup")) { + ActivityStopped(); + } else if (aType.EqualsLiteral("mouseover") || + aType.EqualsLiteral("mouseout")) { + bool newHoveredState = aType.EqualsLiteral("mouseover"); + if (newHoveredState && !*aStoredHoverState) { + ActivityStarted(); + HoveredScrollbar(aScrollbar); + } else if (*aStoredHoverState && !newHoveredState) { + ActivityStopped(); + // Don't call HoveredScrollbar(nullptr) here because we want the hover + // attribute to stick until the scrollbars are hidden. + } + *aStoredHoverState = newHoveredState; + } +} + +void ScrollbarActivity::StartListeningForScrollbarEvents() { + if (mListeningForScrollbarEvents) { + return; + } + + mHorizontalScrollbar = GetHorizontalScrollbar(); + mVerticalScrollbar = GetVerticalScrollbar(); + + AddScrollbarEventListeners(mHorizontalScrollbar); + AddScrollbarEventListeners(mVerticalScrollbar); + + mListeningForScrollbarEvents = true; +} + +void ScrollbarActivity::StopListeningForScrollbarEvents() { + if (!mListeningForScrollbarEvents) return; + + RemoveScrollbarEventListeners(mHorizontalScrollbar); + RemoveScrollbarEventListeners(mVerticalScrollbar); + + mHorizontalScrollbar = nullptr; + mVerticalScrollbar = nullptr; + mListeningForScrollbarEvents = false; +} + +void ScrollbarActivity::StartListeningForScrollAreaEvents() { + if (mListeningForScrollAreaEvents) { + return; + } + nsIFrame* scrollArea = do_QueryFrame(mScrollableFrame); + scrollArea->GetContent()->AddEventListener(u"mousemove"_ns, this, true); + mListeningForScrollAreaEvents = true; +} + +void ScrollbarActivity::StopListeningForScrollAreaEvents() { + if (!mListeningForScrollAreaEvents) { + return; + } + nsIFrame* scrollArea = do_QueryFrame(mScrollableFrame); + scrollArea->GetContent()->RemoveEventListener(u"mousemove"_ns, this, true); + mListeningForScrollAreaEvents = false; +} + +void ScrollbarActivity::AddScrollbarEventListeners( + dom::EventTarget* aScrollbar) { + if (aScrollbar) { + aScrollbar->AddEventListener(u"mousedown"_ns, this, true); + aScrollbar->AddEventListener(u"mouseup"_ns, this, true); + aScrollbar->AddEventListener(u"mouseover"_ns, this, true); + aScrollbar->AddEventListener(u"mouseout"_ns, this, true); + } +} + +void ScrollbarActivity::RemoveScrollbarEventListeners( + dom::EventTarget* aScrollbar) { + if (aScrollbar) { + aScrollbar->RemoveEventListener(u"mousedown"_ns, this, true); + aScrollbar->RemoveEventListener(u"mouseup"_ns, this, true); + aScrollbar->RemoveEventListener(u"mouseover"_ns, this, true); + aScrollbar->RemoveEventListener(u"mouseout"_ns, this, true); + } +} + +void ScrollbarActivity::CancelFadeTimer() { + if (mFadeTimer) { + mFadeTimer->Cancel(); + } +} + +void ScrollbarActivity::StartFadeTimer() { + CancelFadeTimer(); + if (StaticPrefs::layout_testing_overlay_scrollbars_always_visible()) { + return; + } + if (!mFadeTimer) { + mFadeTimer = NS_NewTimer(); + } + mFadeTimer->InitWithNamedFuncCallback( + [](nsITimer*, void* aClosure) { + RefPtr<ScrollbarActivity> activity = + static_cast<ScrollbarActivity*>(aClosure); + activity->BeginFade(); + }, + this, LookAndFeel::GetInt(LookAndFeel::IntID::ScrollbarFadeBeginDelay), + nsITimer::TYPE_ONE_SHOT, "ScrollbarActivity::FadeBeginTimerFired"); +} + +void ScrollbarActivity::BeginFade() { + MOZ_ASSERT(!IsActive()); + mScrollbarEffectivelyVisible = false; + SetBooleanAttribute(GetHorizontalScrollbar(), nsGkAtoms::active, false); + SetBooleanAttribute(GetVerticalScrollbar(), nsGkAtoms::active, false); +} + +static void MaybeInvalidateScrollbarForHover( + Element* aScrollbarToInvalidate, Element* aScrollbarAboutToGetHover) { + if (aScrollbarToInvalidate) { + bool hasHover = aScrollbarToInvalidate->HasAttr(nsGkAtoms::hover); + bool willHaveHover = aScrollbarAboutToGetHover == aScrollbarToInvalidate; + if (hasHover != willHaveHover) { + if (nsIFrame* f = aScrollbarToInvalidate->GetPrimaryFrame()) { + f->SchedulePaint(); + } + } + } +} + +void ScrollbarActivity::HoveredScrollbar(Element* aScrollbar) { + Element* vertScrollbar = GetVerticalScrollbar(); + Element* horzScrollbar = GetHorizontalScrollbar(); + MaybeInvalidateScrollbarForHover(vertScrollbar, aScrollbar); + MaybeInvalidateScrollbarForHover(horzScrollbar, aScrollbar); + + SetBooleanAttribute(horzScrollbar, nsGkAtoms::hover, false); + SetBooleanAttribute(vertScrollbar, nsGkAtoms::hover, false); + SetBooleanAttribute(aScrollbar, nsGkAtoms::hover, true); +} + +Element* ScrollbarActivity::GetScrollbarContent(bool aVertical) { + nsIFrame* box = mScrollableFrame->GetScrollbarBox(aVertical); + return box ? box->GetContent()->AsElement() : nullptr; +} + +} // namespace mozilla::layout |