summaryrefslogtreecommitdiffstats
path: root/layout/generic/nsSplittableFrame.cpp
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-19 00:47:55 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-19 00:47:55 +0000
commit26a029d407be480d791972afb5975cf62c9360a6 (patch)
treef435a8308119effd964b339f76abb83a57c29483 /layout/generic/nsSplittableFrame.cpp
parentInitial commit. (diff)
downloadfirefox-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/nsSplittableFrame.cpp')
-rw-r--r--layout/generic/nsSplittableFrame.cpp346
1 files changed, 346 insertions, 0 deletions
diff --git a/layout/generic/nsSplittableFrame.cpp b/layout/generic/nsSplittableFrame.cpp
new file mode 100644
index 0000000000..4054e45f8c
--- /dev/null
+++ b/layout/generic/nsSplittableFrame.cpp
@@ -0,0 +1,346 @@
+/* -*- 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/. */
+
+/*
+ * base class for rendering objects that can be split across lines,
+ * columns, or pages
+ */
+
+#include "nsSplittableFrame.h"
+#include "nsContainerFrame.h"
+#include "nsFieldSetFrame.h"
+#include "nsIFrameInlines.h"
+
+using namespace mozilla;
+
+NS_QUERYFRAME_HEAD(nsSplittableFrame)
+ NS_QUERYFRAME_ENTRY(nsSplittableFrame)
+NS_QUERYFRAME_TAIL_INHERITING(nsIFrame)
+
+// These frame properties cache the first-continuation and first-in-flow frame
+// pointers. All nsSplittableFrames other than the first one in the continuation
+// chain will have these properties set.
+NS_DECLARE_FRAME_PROPERTY_WITHOUT_DTOR(FirstContinuationProperty, nsIFrame);
+NS_DECLARE_FRAME_PROPERTY_WITHOUT_DTOR(FirstInFlowProperty, nsIFrame);
+
+void nsSplittableFrame::Init(nsIContent* aContent, nsContainerFrame* aParent,
+ nsIFrame* aPrevInFlow) {
+ if (aPrevInFlow) {
+ // Hook the frame into the flow
+ SetPrevInFlow(aPrevInFlow);
+ aPrevInFlow->SetNextInFlow(this);
+ }
+ nsIFrame::Init(aContent, aParent, aPrevInFlow);
+}
+
+void nsSplittableFrame::Destroy(DestroyContext& aContext) {
+ // Disconnect from the flow list
+ if (mPrevContinuation || mNextContinuation) {
+ RemoveFromFlow(this);
+ }
+
+ // Let the base class destroy the frame
+ nsIFrame::Destroy(aContext);
+}
+
+nsIFrame* nsSplittableFrame::GetPrevContinuation() const {
+ return mPrevContinuation;
+}
+
+void nsSplittableFrame::SetPrevContinuation(nsIFrame* aFrame) {
+ NS_ASSERTION(!aFrame || Type() == aFrame->Type(),
+ "setting a prev continuation with incorrect type!");
+ NS_ASSERTION(!IsInPrevContinuationChain(aFrame, this),
+ "creating a loop in continuation chain!");
+ mPrevContinuation = aFrame;
+ RemoveStateBits(NS_FRAME_IS_FLUID_CONTINUATION);
+ UpdateFirstContinuationAndFirstInFlowCache();
+}
+
+nsIFrame* nsSplittableFrame::GetNextContinuation() const {
+ return mNextContinuation;
+}
+
+void nsSplittableFrame::SetNextContinuation(nsIFrame* aFrame) {
+ NS_ASSERTION(!aFrame || Type() == aFrame->Type(),
+ "setting a next continuation with incorrect type!");
+ NS_ASSERTION(!IsInNextContinuationChain(aFrame, this),
+ "creating a loop in continuation chain!");
+ mNextContinuation = aFrame;
+ if (mNextContinuation) {
+ mNextContinuation->RemoveStateBits(NS_FRAME_IS_FLUID_CONTINUATION);
+ }
+}
+
+nsIFrame* nsSplittableFrame::FirstContinuation() const {
+ if (!GetPrevContinuation()) {
+ MOZ_ASSERT(
+ !HasProperty(FirstContinuationProperty()),
+ "The property shouldn't be present on first-continuation itself!");
+ return const_cast<nsSplittableFrame*>(this);
+ }
+
+ nsIFrame* firstContinuation = GetProperty(FirstContinuationProperty());
+ MOZ_ASSERT(firstContinuation,
+ "The property should be set and non-null on all continuations "
+ "after the first!");
+ MOZ_ASSERT(!firstContinuation->GetPrevContinuation(),
+ "First continuation shouldn't have a prev continuation!");
+ return firstContinuation;
+}
+
+nsIFrame* nsSplittableFrame::LastContinuation() const {
+ nsSplittableFrame* lastContinuation = const_cast<nsSplittableFrame*>(this);
+ while (lastContinuation->mNextContinuation) {
+ lastContinuation =
+ static_cast<nsSplittableFrame*>(lastContinuation->mNextContinuation);
+ }
+ MOZ_ASSERT(lastContinuation, "post-condition failed");
+ return lastContinuation;
+}
+
+#ifdef DEBUG
+bool nsSplittableFrame::IsInPrevContinuationChain(nsIFrame* aFrame1,
+ nsIFrame* aFrame2) {
+ int32_t iterations = 0;
+ while (aFrame1 && iterations < 10) {
+ // Bail out after 10 iterations so we don't bog down debug builds too much
+ if (aFrame1 == aFrame2) return true;
+ aFrame1 = aFrame1->GetPrevContinuation();
+ ++iterations;
+ }
+ return false;
+}
+
+bool nsSplittableFrame::IsInNextContinuationChain(nsIFrame* aFrame1,
+ nsIFrame* aFrame2) {
+ int32_t iterations = 0;
+ while (aFrame1 && iterations < 10) {
+ // Bail out after 10 iterations so we don't bog down debug builds too much
+ if (aFrame1 == aFrame2) return true;
+ aFrame1 = aFrame1->GetNextContinuation();
+ ++iterations;
+ }
+ return false;
+}
+#endif
+
+nsIFrame* nsSplittableFrame::GetPrevInFlow() const {
+ return HasAnyStateBits(NS_FRAME_IS_FLUID_CONTINUATION) ? mPrevContinuation
+ : nullptr;
+}
+
+void nsSplittableFrame::SetPrevInFlow(nsIFrame* aFrame) {
+ NS_ASSERTION(!aFrame || Type() == aFrame->Type(),
+ "setting a prev in flow with incorrect type!");
+ NS_ASSERTION(!IsInPrevContinuationChain(aFrame, this),
+ "creating a loop in continuation chain!");
+ mPrevContinuation = aFrame;
+ AddStateBits(NS_FRAME_IS_FLUID_CONTINUATION);
+ UpdateFirstContinuationAndFirstInFlowCache();
+}
+
+nsIFrame* nsSplittableFrame::GetNextInFlow() const {
+ return mNextContinuation && mNextContinuation->HasAnyStateBits(
+ NS_FRAME_IS_FLUID_CONTINUATION)
+ ? mNextContinuation
+ : nullptr;
+}
+
+void nsSplittableFrame::SetNextInFlow(nsIFrame* aFrame) {
+ NS_ASSERTION(!aFrame || Type() == aFrame->Type(),
+ "setting a next in flow with incorrect type!");
+ NS_ASSERTION(!IsInNextContinuationChain(aFrame, this),
+ "creating a loop in continuation chain!");
+ mNextContinuation = aFrame;
+ if (mNextContinuation) {
+ mNextContinuation->AddStateBits(NS_FRAME_IS_FLUID_CONTINUATION);
+ }
+}
+
+nsIFrame* nsSplittableFrame::FirstInFlow() const {
+ if (!GetPrevInFlow()) {
+ MOZ_ASSERT(!HasProperty(FirstInFlowProperty()),
+ "The property shouldn't be present on first-in-flow itself!");
+ return const_cast<nsSplittableFrame*>(this);
+ }
+
+ nsIFrame* firstInFlow = GetProperty(FirstInFlowProperty());
+ MOZ_ASSERT(firstInFlow,
+ "The property should be set and non-null on all in-flows after "
+ "the first!");
+ MOZ_ASSERT(!firstInFlow->GetPrevInFlow(),
+ "First-in-flow shouldn't have a prev-in-flow!");
+ return firstInFlow;
+}
+
+nsIFrame* nsSplittableFrame::LastInFlow() const {
+ nsSplittableFrame* lastInFlow = const_cast<nsSplittableFrame*>(this);
+ while (nsIFrame* next = lastInFlow->GetNextInFlow()) {
+ lastInFlow = static_cast<nsSplittableFrame*>(next);
+ }
+ MOZ_ASSERT(lastInFlow, "post-condition failed");
+ return lastInFlow;
+}
+
+// Remove this frame from the flow. Connects prev in flow and next in flow
+void nsSplittableFrame::RemoveFromFlow(nsIFrame* aFrame) {
+ nsIFrame* prevContinuation = aFrame->GetPrevContinuation();
+ nsIFrame* nextContinuation = aFrame->GetNextContinuation();
+
+ // The new continuation is fluid only if the continuation on both sides
+ // of the removed frame was fluid
+ if (aFrame->GetPrevInFlow() && aFrame->GetNextInFlow()) {
+ if (prevContinuation) {
+ prevContinuation->SetNextInFlow(nextContinuation);
+ }
+ if (nextContinuation) {
+ nextContinuation->SetPrevInFlow(prevContinuation);
+ }
+ } else {
+ if (prevContinuation) {
+ prevContinuation->SetNextContinuation(nextContinuation);
+ }
+ if (nextContinuation) {
+ nextContinuation->SetPrevContinuation(prevContinuation);
+ }
+ }
+
+ // **Note: it is important here that we clear the Next link from aFrame
+ // BEFORE clearing its Prev link, because in nsContinuingTextFrame,
+ // SetPrevInFlow() would follow the Next pointers, wiping out the cached
+ // mFirstContinuation field from each following frame in the list.
+ aFrame->SetNextInFlow(nullptr);
+ aFrame->SetPrevInFlow(nullptr);
+}
+
+void nsSplittableFrame::UpdateFirstContinuationAndFirstInFlowCache() {
+ nsIFrame* oldCachedFirstContinuation =
+ GetProperty(FirstContinuationProperty());
+ nsIFrame* newFirstContinuation;
+ if (nsIFrame* prevContinuation = GetPrevContinuation()) {
+ newFirstContinuation = prevContinuation->FirstContinuation();
+ SetProperty(FirstContinuationProperty(), newFirstContinuation);
+ } else {
+ newFirstContinuation = this;
+ RemoveProperty(FirstContinuationProperty());
+ }
+
+ if (oldCachedFirstContinuation != newFirstContinuation) {
+ // Update the first-continuation cache for our next-continuations in the
+ // chain.
+ for (nsIFrame* next = GetNextContinuation(); next;
+ next = next->GetNextContinuation()) {
+ next->SetProperty(FirstContinuationProperty(), newFirstContinuation);
+ }
+ }
+
+ nsIFrame* oldCachedFirstInFlow = GetProperty(FirstInFlowProperty());
+ nsIFrame* newFirstInFlow;
+ if (nsIFrame* prevInFlow = GetPrevInFlow()) {
+ newFirstInFlow = prevInFlow->FirstInFlow();
+ SetProperty(FirstInFlowProperty(), newFirstInFlow);
+ } else {
+ newFirstInFlow = this;
+ RemoveProperty(FirstInFlowProperty());
+ }
+
+ if (oldCachedFirstInFlow != newFirstInFlow) {
+ // Update the first-in-flow cache for our next-in-flows in the chain.
+ for (nsIFrame* next = GetNextInFlow(); next; next = next->GetNextInFlow()) {
+ next->SetProperty(FirstInFlowProperty(), newFirstInFlow);
+ }
+ }
+}
+
+NS_DECLARE_FRAME_PROPERTY_SMALL_VALUE(ConsumedBSizeProperty, nscoord);
+
+nscoord nsSplittableFrame::CalcAndCacheConsumedBSize() {
+ nsIFrame* prev = GetPrevContinuation();
+ if (!prev) {
+ return 0;
+ }
+ const auto wm = GetWritingMode();
+ nscoord bSize = 0;
+ for (; prev; prev = prev->GetPrevContinuation()) {
+ if (prev->IsTrueOverflowContainer()) {
+ // Overflow containers might not get reflowed, and they have no bSize
+ // anyways.
+ continue;
+ }
+
+ bSize += prev->ContentBSize(wm);
+ bool found = false;
+ nscoord consumed = prev->GetProperty(ConsumedBSizeProperty(), &found);
+ if (found) {
+ bSize += consumed;
+ break;
+ }
+ MOZ_ASSERT(!prev->GetPrevContinuation(),
+ "Property should always be set on prev continuation if not "
+ "the first continuation");
+ }
+ SetProperty(ConsumedBSizeProperty(), bSize);
+ return bSize;
+}
+
+nscoord nsSplittableFrame::GetEffectiveComputedBSize(
+ const ReflowInput& aReflowInput, nscoord aConsumedBSize) const {
+ nscoord bSize = aReflowInput.ComputedBSize();
+ if (bSize == NS_UNCONSTRAINEDSIZE) {
+ return NS_UNCONSTRAINEDSIZE;
+ }
+
+ bSize -= aConsumedBSize;
+
+ // nsFieldSetFrame's inner frames are special since some of their content-box
+ // BSize may be consumed by positioning it below the legend. So we always
+ // report zero for true overflow containers here.
+ // XXXmats: hmm, can we fix this so that the sizes actually adds up instead?
+ if (IsTrueOverflowContainer() &&
+ Style()->GetPseudoType() == PseudoStyleType::fieldsetContent) {
+ for (nsFieldSetFrame* fieldset = do_QueryFrame(GetParent()); fieldset;
+ fieldset = static_cast<nsFieldSetFrame*>(fieldset->GetPrevInFlow())) {
+ bSize -= fieldset->LegendSpace();
+ }
+ }
+
+ // We may have stretched the frame beyond its computed height. Oh well.
+ return std::max(0, bSize);
+}
+
+LogicalSides nsSplittableFrame::GetBlockLevelLogicalSkipSides(
+ bool aAfterReflow) const {
+ LogicalSides skip(mWritingMode);
+ if (MOZ_UNLIKELY(IsTrueOverflowContainer())) {
+ skip |= eLogicalSideBitsBBoth;
+ return skip;
+ }
+
+ if (MOZ_UNLIKELY(StyleBorder()->mBoxDecorationBreak ==
+ StyleBoxDecorationBreak::Clone)) {
+ return skip;
+ }
+
+ if (GetPrevContinuation()) {
+ skip |= eLogicalSideBitsBStart;
+ }
+
+ // Always skip block-end side if we have a *later* sibling across column-span
+ // split.
+ if (HasColumnSpanSiblings()) {
+ skip |= eLogicalSideBitsBEnd;
+ }
+
+ if (aAfterReflow) {
+ nsIFrame* nif = GetNextContinuation();
+ if (nif && !nif->IsTrueOverflowContainer()) {
+ skip |= eLogicalSideBitsBEnd;
+ }
+ }
+
+ return skip;
+}