mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-30 16:22:00 +00:00
1e80f659a2
Differential Revision: https://phabricator.services.mozilla.com/D211498
596 lines
20 KiB
C++
596 lines
20 KiB
C++
/* -*- 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/. */
|
|
|
|
//
|
|
// Eric Vaughan
|
|
// Netscape Communications
|
|
//
|
|
// See documentation in associated header file
|
|
//
|
|
|
|
#include "nsScrollbarFrame.h"
|
|
#include "nsSliderFrame.h"
|
|
#include "nsScrollbarButtonFrame.h"
|
|
#include "nsContentCreatorFunctions.h"
|
|
#include "nsGkAtoms.h"
|
|
#include "nsIScrollbarMediator.h"
|
|
#include "nsStyleConsts.h"
|
|
#include "nsIContent.h"
|
|
#include "nsLayoutUtils.h"
|
|
#include "mozilla/LookAndFeel.h"
|
|
#include "mozilla/PresShell.h"
|
|
#include "mozilla/dom/Element.h"
|
|
#include "mozilla/dom/MutationEventBinding.h"
|
|
#include "mozilla/ScrollContainerFrame.h"
|
|
#include "mozilla/StaticPrefs_apz.h"
|
|
|
|
using namespace mozilla;
|
|
using mozilla::dom::Element;
|
|
|
|
//
|
|
// NS_NewScrollbarFrame
|
|
//
|
|
// Creates a new scrollbar frame and returns it
|
|
//
|
|
nsIFrame* NS_NewScrollbarFrame(PresShell* aPresShell, ComputedStyle* aStyle) {
|
|
return new (aPresShell)
|
|
nsScrollbarFrame(aStyle, aPresShell->GetPresContext());
|
|
}
|
|
|
|
NS_IMPL_FRAMEARENA_HELPERS(nsScrollbarFrame)
|
|
|
|
NS_QUERYFRAME_HEAD(nsScrollbarFrame)
|
|
NS_QUERYFRAME_ENTRY(nsScrollbarFrame)
|
|
NS_QUERYFRAME_ENTRY(nsIAnonymousContentCreator)
|
|
NS_QUERYFRAME_TAIL_INHERITING(nsContainerFrame)
|
|
|
|
void nsScrollbarFrame::Init(nsIContent* aContent, nsContainerFrame* aParent,
|
|
nsIFrame* aPrevInFlow) {
|
|
nsContainerFrame::Init(aContent, aParent, aPrevInFlow);
|
|
|
|
// We want to be a reflow root since we use reflows to move the
|
|
// slider. Any reflow inside the scrollbar frame will be a reflow to
|
|
// move the slider and will thus not change anything outside of the
|
|
// scrollbar or change the size of the scrollbar frame.
|
|
AddStateBits(NS_FRAME_REFLOW_ROOT);
|
|
}
|
|
|
|
void nsScrollbarFrame::Destroy(DestroyContext& aContext) {
|
|
aContext.AddAnonymousContent(mUpTopButton.forget());
|
|
aContext.AddAnonymousContent(mDownTopButton.forget());
|
|
aContext.AddAnonymousContent(mSlider.forget());
|
|
aContext.AddAnonymousContent(mUpBottomButton.forget());
|
|
aContext.AddAnonymousContent(mDownBottomButton.forget());
|
|
nsContainerFrame::Destroy(aContext);
|
|
}
|
|
|
|
void nsScrollbarFrame::Reflow(nsPresContext* aPresContext,
|
|
ReflowOutput& aDesiredSize,
|
|
const ReflowInput& aReflowInput,
|
|
nsReflowStatus& aStatus) {
|
|
MarkInReflow();
|
|
MOZ_ASSERT(aStatus.IsEmpty(), "Caller should pass a fresh reflow status!");
|
|
|
|
// We always take all the space we're given, and our track size in the other
|
|
// axis.
|
|
const bool horizontal = IsHorizontal();
|
|
const auto wm = GetWritingMode();
|
|
const auto minSize = aReflowInput.ComputedMinSize();
|
|
|
|
aDesiredSize.ISize(wm) = aReflowInput.ComputedISize();
|
|
aDesiredSize.BSize(wm) = [&] {
|
|
if (aReflowInput.ComputedBSize() != NS_UNCONSTRAINEDSIZE) {
|
|
return aReflowInput.ComputedBSize();
|
|
}
|
|
// We don't want to change our size during incremental reflow, see the
|
|
// reflow root comment in init.
|
|
if (!aReflowInput.mParentReflowInput) {
|
|
return GetLogicalSize(wm).BSize(wm);
|
|
}
|
|
return minSize.BSize(wm);
|
|
}();
|
|
|
|
const nsSize containerSize = aDesiredSize.PhysicalSize();
|
|
const LogicalSize totalAvailSize = aDesiredSize.Size(wm);
|
|
LogicalPoint nextKidPos(wm);
|
|
|
|
MOZ_ASSERT(!wm.IsVertical());
|
|
const bool movesInInlineDirection = horizontal;
|
|
|
|
// Layout our kids left to right / top to bottom.
|
|
for (nsIFrame* kid : mFrames) {
|
|
MOZ_ASSERT(!kid->GetWritingMode().IsOrthogonalTo(wm),
|
|
"We don't expect orthogonal scrollbar parts");
|
|
const bool isSlider = kid->GetContent() == mSlider;
|
|
LogicalSize availSize = totalAvailSize;
|
|
{
|
|
// Assume we'll consume the same size before and after the slider. This is
|
|
// not a technically correct assumption if we have weird scrollbar button
|
|
// setups, but those will be going away, see bug 1824254.
|
|
const int32_t factor = isSlider ? 2 : 1;
|
|
if (movesInInlineDirection) {
|
|
availSize.ISize(wm) =
|
|
std::max(0, totalAvailSize.ISize(wm) - nextKidPos.I(wm) * factor);
|
|
} else {
|
|
availSize.BSize(wm) =
|
|
std::max(0, totalAvailSize.BSize(wm) - nextKidPos.B(wm) * factor);
|
|
}
|
|
}
|
|
|
|
ReflowInput kidRI(aPresContext, aReflowInput, kid, availSize);
|
|
if (isSlider) {
|
|
// We want for the slider to take all the remaining available space.
|
|
kidRI.SetComputedISize(availSize.ISize(wm));
|
|
kidRI.SetComputedBSize(availSize.BSize(wm));
|
|
} else if (movesInInlineDirection) {
|
|
// Otherwise we want all the space in the axis we're not advancing in, and
|
|
// the default / minimum size on the other axis.
|
|
kidRI.SetComputedBSize(availSize.BSize(wm));
|
|
} else {
|
|
kidRI.SetComputedISize(availSize.ISize(wm));
|
|
}
|
|
|
|
ReflowOutput kidDesiredSize(wm);
|
|
nsReflowStatus status;
|
|
const auto flags = ReflowChildFlags::Default;
|
|
ReflowChild(kid, aPresContext, kidDesiredSize, kidRI, wm, nextKidPos,
|
|
containerSize, flags, status);
|
|
// We haven't seen the slider yet, we can advance
|
|
FinishReflowChild(kid, aPresContext, kidDesiredSize, &kidRI, wm, nextKidPos,
|
|
containerSize, flags);
|
|
if (movesInInlineDirection) {
|
|
nextKidPos.I(wm) += kidDesiredSize.ISize(wm);
|
|
} else {
|
|
nextKidPos.B(wm) += kidDesiredSize.BSize(wm);
|
|
}
|
|
}
|
|
|
|
aDesiredSize.SetOverflowAreasToDesiredBounds();
|
|
}
|
|
|
|
nsresult nsScrollbarFrame::AttributeChanged(int32_t aNameSpaceID,
|
|
nsAtom* aAttribute,
|
|
int32_t aModType) {
|
|
nsresult rv =
|
|
nsContainerFrame::AttributeChanged(aNameSpaceID, aAttribute, aModType);
|
|
|
|
// Update value in our children
|
|
UpdateChildrenAttributeValue(aAttribute, true);
|
|
|
|
// if the current position changes, notify any ScrollContainerFrame
|
|
// parent we may have
|
|
if (aAttribute != nsGkAtoms::curpos) {
|
|
return rv;
|
|
}
|
|
|
|
ScrollContainerFrame* scrollContainerFrame = do_QueryFrame(GetParent());
|
|
if (!scrollContainerFrame) {
|
|
return rv;
|
|
}
|
|
|
|
nsCOMPtr<nsIContent> content(mContent);
|
|
scrollContainerFrame->CurPosAttributeChanged(content);
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsScrollbarFrame::HandlePress(nsPresContext* aPresContext,
|
|
WidgetGUIEvent* aEvent,
|
|
nsEventStatus* aEventStatus) {
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsScrollbarFrame::HandleMultiplePress(nsPresContext* aPresContext,
|
|
WidgetGUIEvent* aEvent,
|
|
nsEventStatus* aEventStatus,
|
|
bool aControlHeld) {
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsScrollbarFrame::HandleDrag(nsPresContext* aPresContext,
|
|
WidgetGUIEvent* aEvent,
|
|
nsEventStatus* aEventStatus) {
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsScrollbarFrame::HandleRelease(nsPresContext* aPresContext,
|
|
WidgetGUIEvent* aEvent,
|
|
nsEventStatus* aEventStatus) {
|
|
return NS_OK;
|
|
}
|
|
|
|
void nsScrollbarFrame::SetScrollbarMediatorContent(nsIContent* aMediator) {
|
|
mScrollbarMediator = aMediator;
|
|
}
|
|
|
|
nsIScrollbarMediator* nsScrollbarFrame::GetScrollbarMediator() {
|
|
if (!mScrollbarMediator) {
|
|
return nullptr;
|
|
}
|
|
nsIFrame* f = mScrollbarMediator->GetPrimaryFrame();
|
|
ScrollContainerFrame* scrollContainerFrame = do_QueryFrame(f);
|
|
nsIScrollbarMediator* sbm;
|
|
|
|
if (scrollContainerFrame) {
|
|
nsIFrame* scrolledFrame = scrollContainerFrame->GetScrolledFrame();
|
|
sbm = do_QueryFrame(scrolledFrame);
|
|
if (sbm) {
|
|
return sbm;
|
|
}
|
|
}
|
|
sbm = do_QueryFrame(f);
|
|
if (f && !sbm) {
|
|
f = f->PresShell()->GetRootScrollContainerFrame();
|
|
if (f && f->GetContent() == mScrollbarMediator) {
|
|
return do_QueryFrame(f);
|
|
}
|
|
}
|
|
return sbm;
|
|
}
|
|
|
|
bool nsScrollbarFrame::IsHorizontal() const {
|
|
auto appearance = StyleDisplay()->EffectiveAppearance();
|
|
MOZ_ASSERT(appearance == StyleAppearance::ScrollbarHorizontal ||
|
|
appearance == StyleAppearance::ScrollbarVertical);
|
|
return appearance == StyleAppearance::ScrollbarHorizontal;
|
|
}
|
|
|
|
nsSize nsScrollbarFrame::ScrollbarMinSize() const {
|
|
nsPresContext* pc = PresContext();
|
|
const LayoutDeviceIntSize widget =
|
|
pc->Theme()->GetMinimumWidgetSize(pc, const_cast<nsScrollbarFrame*>(this),
|
|
StyleDisplay()->EffectiveAppearance());
|
|
return LayoutDeviceIntSize::ToAppUnits(widget, pc->AppUnitsPerDevPixel());
|
|
}
|
|
|
|
StyleScrollbarWidth nsScrollbarFrame::ScrollbarWidth() const {
|
|
return nsLayoutUtils::StyleForScrollbar(this)
|
|
->StyleUIReset()
|
|
->ScrollbarWidth();
|
|
}
|
|
|
|
nscoord nsScrollbarFrame::ScrollbarTrackSize() const {
|
|
nsPresContext* pc = PresContext();
|
|
auto overlay = pc->UseOverlayScrollbars() ? nsITheme::Overlay::Yes
|
|
: nsITheme::Overlay::No;
|
|
return LayoutDevicePixel::ToAppUnits(
|
|
pc->Theme()->GetScrollbarSize(pc, ScrollbarWidth(), overlay),
|
|
pc->AppUnitsPerDevPixel());
|
|
}
|
|
|
|
void nsScrollbarFrame::SetIncrementToLine(int32_t aDirection) {
|
|
mSmoothScroll = true;
|
|
mDirection = aDirection;
|
|
mScrollUnit = ScrollUnit::LINES;
|
|
|
|
// get the scrollbar's content node
|
|
nsIContent* content = GetContent();
|
|
mIncrement = aDirection * nsSliderFrame::GetIncrement(content);
|
|
}
|
|
|
|
void nsScrollbarFrame::SetIncrementToPage(int32_t aDirection) {
|
|
mSmoothScroll = true;
|
|
mDirection = aDirection;
|
|
mScrollUnit = ScrollUnit::PAGES;
|
|
|
|
// get the scrollbar's content node
|
|
nsIContent* content = GetContent();
|
|
mIncrement = aDirection * nsSliderFrame::GetPageIncrement(content);
|
|
}
|
|
|
|
void nsScrollbarFrame::SetIncrementToWhole(int32_t aDirection) {
|
|
// Don't repeat or use smooth scrolling if scrolling to beginning or end
|
|
// of a page.
|
|
mSmoothScroll = false;
|
|
mDirection = aDirection;
|
|
mScrollUnit = ScrollUnit::WHOLE;
|
|
|
|
// get the scrollbar's content node
|
|
nsIContent* content = GetContent();
|
|
if (aDirection == -1)
|
|
mIncrement = -nsSliderFrame::GetCurrentPosition(content);
|
|
else
|
|
mIncrement = nsSliderFrame::GetMaxPosition(content) -
|
|
nsSliderFrame::GetCurrentPosition(content);
|
|
}
|
|
|
|
int32_t nsScrollbarFrame::MoveToNewPosition(
|
|
ImplementsScrollByUnit aImplementsScrollByUnit) {
|
|
if (aImplementsScrollByUnit == ImplementsScrollByUnit::Yes &&
|
|
StaticPrefs::apz_scrollbarbuttonrepeat_enabled()) {
|
|
nsIScrollbarMediator* m = GetScrollbarMediator();
|
|
MOZ_ASSERT(m);
|
|
// aImplementsScrollByUnit being Yes indicates the caller doesn't care
|
|
// about the return value.
|
|
// Note that this `MoveToNewPosition` is used for scrolling triggered by
|
|
// repeating scrollbar button press, so we'd use an intended-direction
|
|
// scroll snap flag.
|
|
m->ScrollByUnit(
|
|
this, mSmoothScroll ? ScrollMode::Smooth : ScrollMode::Instant,
|
|
mDirection, mScrollUnit, ScrollSnapFlags::IntendedDirection);
|
|
return 0;
|
|
}
|
|
|
|
// get the scrollbar's content node
|
|
RefPtr<Element> content = GetContent()->AsElement();
|
|
|
|
// get the current pos
|
|
int32_t curpos = nsSliderFrame::GetCurrentPosition(content);
|
|
|
|
// get the max pos
|
|
int32_t maxpos = nsSliderFrame::GetMaxPosition(content);
|
|
|
|
// increment the given amount
|
|
if (mIncrement) {
|
|
curpos += mIncrement;
|
|
}
|
|
|
|
// make sure the current position is between the current and max positions
|
|
if (curpos < 0) {
|
|
curpos = 0;
|
|
} else if (curpos > maxpos) {
|
|
curpos = maxpos;
|
|
}
|
|
|
|
// set the current position of the slider.
|
|
nsAutoString curposStr;
|
|
curposStr.AppendInt(curpos);
|
|
|
|
AutoWeakFrame weakFrame(this);
|
|
if (mSmoothScroll) {
|
|
content->SetAttr(kNameSpaceID_None, nsGkAtoms::smooth, u"true"_ns, false);
|
|
}
|
|
content->SetAttr(kNameSpaceID_None, nsGkAtoms::curpos, curposStr, false);
|
|
// notify the nsScrollbarFrame of the change
|
|
AttributeChanged(kNameSpaceID_None, nsGkAtoms::curpos,
|
|
dom::MutationEvent_Binding::MODIFICATION);
|
|
if (!weakFrame.IsAlive()) {
|
|
return curpos;
|
|
}
|
|
// notify all nsSliderFrames of the change
|
|
for (const auto& childList : ChildLists()) {
|
|
for (nsIFrame* f : childList.mList) {
|
|
nsSliderFrame* sliderFrame = do_QueryFrame(f);
|
|
if (sliderFrame) {
|
|
sliderFrame->AttributeChanged(kNameSpaceID_None, nsGkAtoms::curpos,
|
|
dom::MutationEvent_Binding::MODIFICATION);
|
|
if (!weakFrame.IsAlive()) {
|
|
return curpos;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
content->UnsetAttr(kNameSpaceID_None, nsGkAtoms::smooth, false);
|
|
return curpos;
|
|
}
|
|
|
|
static already_AddRefed<Element> MakeScrollbarButton(
|
|
dom::NodeInfo* aNodeInfo, bool aVertical, bool aBottom, bool aDown,
|
|
AnonymousContentKey& aKey) {
|
|
MOZ_ASSERT(aNodeInfo);
|
|
MOZ_ASSERT(
|
|
aNodeInfo->Equals(nsGkAtoms::scrollbarbutton, nullptr, kNameSpaceID_XUL));
|
|
|
|
static constexpr nsLiteralString kSbattrValues[2][2] = {
|
|
{
|
|
u"scrollbar-up-top"_ns,
|
|
u"scrollbar-up-bottom"_ns,
|
|
},
|
|
{
|
|
u"scrollbar-down-top"_ns,
|
|
u"scrollbar-down-bottom"_ns,
|
|
},
|
|
};
|
|
|
|
static constexpr nsLiteralString kTypeValues[2] = {
|
|
u"decrement"_ns,
|
|
u"increment"_ns,
|
|
};
|
|
|
|
aKey = AnonymousContentKey::Type_ScrollbarButton;
|
|
if (aVertical) {
|
|
aKey |= AnonymousContentKey::Flag_Vertical;
|
|
}
|
|
if (aBottom) {
|
|
aKey |= AnonymousContentKey::Flag_ScrollbarButton_Bottom;
|
|
}
|
|
if (aDown) {
|
|
aKey |= AnonymousContentKey::Flag_ScrollbarButton_Down;
|
|
}
|
|
|
|
RefPtr<Element> e;
|
|
NS_TrustedNewXULElement(getter_AddRefs(e), do_AddRef(aNodeInfo));
|
|
e->SetAttr(kNameSpaceID_None, nsGkAtoms::sbattr,
|
|
kSbattrValues[aDown][aBottom], false);
|
|
e->SetAttr(kNameSpaceID_None, nsGkAtoms::type, kTypeValues[aDown], false);
|
|
return e.forget();
|
|
}
|
|
|
|
nsresult nsScrollbarFrame::CreateAnonymousContent(
|
|
nsTArray<ContentInfo>& aElements) {
|
|
nsNodeInfoManager* nodeInfoManager = mContent->NodeInfo()->NodeInfoManager();
|
|
Element* el = GetContent()->AsElement();
|
|
|
|
// If there are children already in the node, don't create any anonymous
|
|
// content (this only apply to crashtests/369038-1.xhtml)
|
|
if (el->HasChildren()) {
|
|
return NS_OK;
|
|
}
|
|
|
|
nsAutoString orient;
|
|
el->GetAttr(nsGkAtoms::orient, orient);
|
|
bool vertical = orient.EqualsLiteral("vertical");
|
|
|
|
RefPtr<dom::NodeInfo> sbbNodeInfo =
|
|
nodeInfoManager->GetNodeInfo(nsGkAtoms::scrollbarbutton, nullptr,
|
|
kNameSpaceID_XUL, nsINode::ELEMENT_NODE);
|
|
|
|
bool createButtons = PresContext()->Theme()->ThemeSupportsScrollbarButtons();
|
|
|
|
if (createButtons) {
|
|
AnonymousContentKey key;
|
|
mUpTopButton =
|
|
MakeScrollbarButton(sbbNodeInfo, vertical, /* aBottom */ false,
|
|
/* aDown */ false, key);
|
|
aElements.AppendElement(ContentInfo(mUpTopButton, key));
|
|
}
|
|
|
|
if (createButtons) {
|
|
AnonymousContentKey key;
|
|
mDownTopButton =
|
|
MakeScrollbarButton(sbbNodeInfo, vertical, /* aBottom */ false,
|
|
/* aDown */ true, key);
|
|
aElements.AppendElement(ContentInfo(mDownTopButton, key));
|
|
}
|
|
|
|
{
|
|
AnonymousContentKey key = AnonymousContentKey::Type_Slider;
|
|
if (vertical) {
|
|
key |= AnonymousContentKey::Flag_Vertical;
|
|
}
|
|
|
|
NS_TrustedNewXULElement(
|
|
getter_AddRefs(mSlider),
|
|
nodeInfoManager->GetNodeInfo(nsGkAtoms::slider, nullptr,
|
|
kNameSpaceID_XUL, nsINode::ELEMENT_NODE));
|
|
mSlider->SetAttr(kNameSpaceID_None, nsGkAtoms::orient, orient, false);
|
|
|
|
aElements.AppendElement(ContentInfo(mSlider, key));
|
|
|
|
NS_TrustedNewXULElement(
|
|
getter_AddRefs(mThumb),
|
|
nodeInfoManager->GetNodeInfo(nsGkAtoms::thumb, nullptr,
|
|
kNameSpaceID_XUL, nsINode::ELEMENT_NODE));
|
|
mThumb->SetAttr(kNameSpaceID_None, nsGkAtoms::orient, orient, false);
|
|
mSlider->AppendChildTo(mThumb, false, IgnoreErrors());
|
|
}
|
|
|
|
if (createButtons) {
|
|
AnonymousContentKey key;
|
|
mUpBottomButton =
|
|
MakeScrollbarButton(sbbNodeInfo, vertical, /* aBottom */ true,
|
|
/* aDown */ false, key);
|
|
aElements.AppendElement(ContentInfo(mUpBottomButton, key));
|
|
}
|
|
|
|
if (createButtons) {
|
|
AnonymousContentKey key;
|
|
mDownBottomButton =
|
|
MakeScrollbarButton(sbbNodeInfo, vertical, /* aBottom */ true,
|
|
/* aDown */ true, key);
|
|
aElements.AppendElement(ContentInfo(mDownBottomButton, key));
|
|
}
|
|
|
|
// Don't cache styles if we are inside a <select> element, since we have
|
|
// some UA style sheet rules that depend on the <select>'s attributes.
|
|
if (GetContent()->GetParent() &&
|
|
GetContent()->GetParent()->IsHTMLElement(nsGkAtoms::select)) {
|
|
for (auto& info : aElements) {
|
|
info.mKey = AnonymousContentKey::None;
|
|
}
|
|
}
|
|
|
|
UpdateChildrenAttributeValue(nsGkAtoms::curpos, false);
|
|
UpdateChildrenAttributeValue(nsGkAtoms::maxpos, false);
|
|
UpdateChildrenAttributeValue(nsGkAtoms::disabled, false);
|
|
UpdateChildrenAttributeValue(nsGkAtoms::pageincrement, false);
|
|
UpdateChildrenAttributeValue(nsGkAtoms::increment, false);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
void nsScrollbarFrame::UpdateChildrenAttributeValue(nsAtom* aAttribute,
|
|
bool aNotify) {
|
|
Element* el = GetContent()->AsElement();
|
|
|
|
nsAutoString value;
|
|
el->GetAttr(aAttribute, value);
|
|
|
|
if (!el->HasAttr(aAttribute)) {
|
|
if (mUpTopButton) {
|
|
mUpTopButton->UnsetAttr(kNameSpaceID_None, aAttribute, aNotify);
|
|
}
|
|
if (mDownTopButton) {
|
|
mDownTopButton->UnsetAttr(kNameSpaceID_None, aAttribute, aNotify);
|
|
}
|
|
if (mSlider) {
|
|
mSlider->UnsetAttr(kNameSpaceID_None, aAttribute, aNotify);
|
|
}
|
|
if (mUpBottomButton) {
|
|
mUpBottomButton->UnsetAttr(kNameSpaceID_None, aAttribute, aNotify);
|
|
}
|
|
if (mDownBottomButton) {
|
|
mDownBottomButton->UnsetAttr(kNameSpaceID_None, aAttribute, aNotify);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (aAttribute == nsGkAtoms::curpos || aAttribute == nsGkAtoms::maxpos) {
|
|
if (mUpTopButton) {
|
|
mUpTopButton->SetAttr(kNameSpaceID_None, aAttribute, value, aNotify);
|
|
}
|
|
if (mDownTopButton) {
|
|
mDownTopButton->SetAttr(kNameSpaceID_None, aAttribute, value, aNotify);
|
|
}
|
|
if (mSlider) {
|
|
mSlider->SetAttr(kNameSpaceID_None, aAttribute, value, aNotify);
|
|
}
|
|
if (mUpBottomButton) {
|
|
mUpBottomButton->SetAttr(kNameSpaceID_None, aAttribute, value, aNotify);
|
|
}
|
|
if (mDownBottomButton) {
|
|
mDownBottomButton->SetAttr(kNameSpaceID_None, aAttribute, value, aNotify);
|
|
}
|
|
} else if (aAttribute == nsGkAtoms::disabled) {
|
|
if (mUpTopButton) {
|
|
mUpTopButton->SetAttr(kNameSpaceID_None, aAttribute, value, aNotify);
|
|
}
|
|
if (mDownTopButton) {
|
|
mDownTopButton->SetAttr(kNameSpaceID_None, aAttribute, value, aNotify);
|
|
}
|
|
if (mSlider) {
|
|
mSlider->SetAttr(kNameSpaceID_None, aAttribute, value, aNotify);
|
|
}
|
|
if (mUpBottomButton) {
|
|
mUpBottomButton->SetAttr(kNameSpaceID_None, aAttribute, value, aNotify);
|
|
}
|
|
if (mDownBottomButton) {
|
|
mDownBottomButton->SetAttr(kNameSpaceID_None, aAttribute, value, aNotify);
|
|
}
|
|
} else if (aAttribute == nsGkAtoms::pageincrement ||
|
|
aAttribute == nsGkAtoms::increment) {
|
|
if (mSlider) {
|
|
mSlider->SetAttr(kNameSpaceID_None, aAttribute, value, aNotify);
|
|
}
|
|
}
|
|
}
|
|
|
|
void nsScrollbarFrame::AppendAnonymousContentTo(
|
|
nsTArray<nsIContent*>& aElements, uint32_t aFilter) {
|
|
if (mUpTopButton) {
|
|
aElements.AppendElement(mUpTopButton);
|
|
}
|
|
|
|
if (mDownTopButton) {
|
|
aElements.AppendElement(mDownTopButton);
|
|
}
|
|
|
|
if (mSlider) {
|
|
aElements.AppendElement(mSlider);
|
|
}
|
|
|
|
if (mUpBottomButton) {
|
|
aElements.AppendElement(mUpBottomButton);
|
|
}
|
|
|
|
if (mDownBottomButton) {
|
|
aElements.AppendElement(mDownBottomButton);
|
|
}
|
|
}
|