diff options
Diffstat (limited to 'layout/generic/nsRubyFrame.cpp')
-rw-r--r-- | layout/generic/nsRubyFrame.cpp | 419 |
1 files changed, 419 insertions, 0 deletions
diff --git a/layout/generic/nsRubyFrame.cpp b/layout/generic/nsRubyFrame.cpp new file mode 100644 index 0000000000..d7da5bc321 --- /dev/null +++ b/layout/generic/nsRubyFrame.cpp @@ -0,0 +1,419 @@ +/* -*- 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/. */ + +/* rendering object for CSS "display: ruby" */ + +#include "nsRubyFrame.h" + +#include "RubyUtils.h" +#include "mozilla/ComputedStyle.h" +#include "mozilla/Maybe.h" +#include "mozilla/PresShell.h" +#include "mozilla/StaticPrefs_layout.h" +#include "mozilla/WritingModes.h" +#include "nsLayoutUtils.h" +#include "nsLineLayout.h" +#include "nsPresContext.h" +#include "nsContainerFrameInlines.h" +#include "nsRubyBaseContainerFrame.h" +#include "nsRubyTextContainerFrame.h" + +using namespace mozilla; + +//---------------------------------------------------------------------- + +// Frame class boilerplate +// ======================= + +NS_QUERYFRAME_HEAD(nsRubyFrame) + NS_QUERYFRAME_ENTRY(nsRubyFrame) +NS_QUERYFRAME_TAIL_INHERITING(nsInlineFrame) + +NS_IMPL_FRAMEARENA_HELPERS(nsRubyFrame) + +nsContainerFrame* NS_NewRubyFrame(PresShell* aPresShell, + ComputedStyle* aStyle) { + return new (aPresShell) nsRubyFrame(aStyle, aPresShell->GetPresContext()); +} + +//---------------------------------------------------------------------- + +// nsRubyFrame Method Implementations +// ================================== + +#ifdef DEBUG_FRAME_DUMP +nsresult nsRubyFrame::GetFrameName(nsAString& aResult) const { + return MakeFrameName(u"Ruby"_ns, aResult); +} +#endif + +/* virtual */ +void nsRubyFrame::AddInlineMinISize(gfxContext* aRenderingContext, + nsIFrame::InlineMinISizeData* aData) { + auto handleChildren = [aRenderingContext](auto frame, auto data) { + for (RubySegmentEnumerator e(static_cast<nsRubyFrame*>(frame)); !e.AtEnd(); + e.Next()) { + e.GetBaseContainer()->AddInlineMinISize(aRenderingContext, data); + } + }; + DoInlineIntrinsicISize(aData, handleChildren); +} + +/* virtual */ +void nsRubyFrame::AddInlinePrefISize(gfxContext* aRenderingContext, + nsIFrame::InlinePrefISizeData* aData) { + auto handleChildren = [aRenderingContext](auto frame, auto data) { + for (RubySegmentEnumerator e(static_cast<nsRubyFrame*>(frame)); !e.AtEnd(); + e.Next()) { + e.GetBaseContainer()->AddInlinePrefISize(aRenderingContext, data); + } + }; + DoInlineIntrinsicISize(aData, handleChildren); + aData->mLineIsEmpty = false; +} + +static nsRubyBaseContainerFrame* FindRubyBaseContainerAncestor( + nsIFrame* aFrame) { + for (nsIFrame* ancestor = aFrame->GetParent(); + ancestor && ancestor->IsLineParticipant(); + ancestor = ancestor->GetParent()) { + if (ancestor->IsRubyBaseContainerFrame()) { + return static_cast<nsRubyBaseContainerFrame*>(ancestor); + } + } + return nullptr; +} + +/* virtual */ +void nsRubyFrame::Reflow(nsPresContext* aPresContext, + ReflowOutput& aDesiredSize, + const ReflowInput& aReflowInput, + nsReflowStatus& aStatus) { + MarkInReflow(); + DO_GLOBAL_REFLOW_COUNT("nsRubyFrame"); + DISPLAY_REFLOW(aPresContext, this, aReflowInput, aDesiredSize, aStatus); + MOZ_ASSERT(aStatus.IsEmpty(), "Caller should pass a fresh reflow status!"); + + if (!aReflowInput.mLineLayout) { + NS_ASSERTION(aReflowInput.mLineLayout, + "No line layout provided to RubyFrame reflow method."); + return; + } + + // Grab overflow frames from prev-in-flow and its own. + MoveInlineOverflowToChildList(aReflowInput.mLineLayout->LineContainerFrame()); + + // Clear leadings + mLeadings.Reset(); + + // Begin the span for the ruby frame + WritingMode frameWM = aReflowInput.GetWritingMode(); + WritingMode lineWM = aReflowInput.mLineLayout->GetWritingMode(); + LogicalMargin borderPadding = + aReflowInput.ComputedLogicalBorderPadding(frameWM); + nsLayoutUtils::SetBSizeFromFontMetrics(this, aDesiredSize, borderPadding, + lineWM, frameWM); + + nscoord startEdge = 0; + const bool boxDecorationBreakClone = + StyleBorder()->mBoxDecorationBreak == StyleBoxDecorationBreak::Clone; + if (boxDecorationBreakClone || !GetPrevContinuation()) { + startEdge = borderPadding.IStart(frameWM); + } + NS_ASSERTION(aReflowInput.AvailableISize() != NS_UNCONSTRAINEDSIZE, + "should no longer use available widths"); + nscoord endEdge = aReflowInput.AvailableISize() - borderPadding.IEnd(frameWM); + aReflowInput.mLineLayout->BeginSpan(this, &aReflowInput, startEdge, endEdge, + &mBaseline); + + for (RubySegmentEnumerator e(this); !e.AtEnd(); e.Next()) { + ReflowSegment(aPresContext, aReflowInput, aDesiredSize.BlockStartAscent(), + aDesiredSize.BSize(lineWM), e.GetBaseContainer(), aStatus); + + if (aStatus.IsInlineBreak()) { + // A break occurs when reflowing the segment. + // Don't continue reflowing more segments. + break; + } + } + + ContinuationTraversingState pullState(this); + while (aStatus.IsEmpty()) { + nsRubyBaseContainerFrame* baseContainer = + PullOneSegment(aReflowInput.mLineLayout, pullState); + if (!baseContainer) { + // No more continuations after, finish now. + break; + } + ReflowSegment(aPresContext, aReflowInput, aDesiredSize.BlockStartAscent(), + aDesiredSize.BSize(lineWM), baseContainer, aStatus); + } + // We never handle overflow in ruby. + MOZ_ASSERT(!aStatus.IsOverflowIncomplete()); + + aDesiredSize.ISize(lineWM) = aReflowInput.mLineLayout->EndSpan(this); + if (boxDecorationBreakClone || !GetPrevContinuation()) { + aDesiredSize.ISize(lineWM) += borderPadding.IStart(frameWM); + } + if (boxDecorationBreakClone || aStatus.IsComplete()) { + aDesiredSize.ISize(lineWM) += borderPadding.IEnd(frameWM); + } + + // Update descendant leadings of ancestor ruby base container. + if (nsRubyBaseContainerFrame* rbc = FindRubyBaseContainerAncestor(this)) { + rbc->UpdateDescendantLeadings(mLeadings); + } + + ReflowAbsoluteFrames(aPresContext, aDesiredSize, aReflowInput, aStatus); +} + +void nsRubyFrame::ReflowSegment(nsPresContext* aPresContext, + const ReflowInput& aReflowInput, + nscoord aBlockStartAscent, nscoord aBlockSize, + nsRubyBaseContainerFrame* aBaseContainer, + nsReflowStatus& aStatus) { + WritingMode lineWM = aReflowInput.mLineLayout->GetWritingMode(); + LogicalSize availSize(lineWM, aReflowInput.AvailableISize(), + aReflowInput.AvailableBSize()); + NS_ASSERTION(!GetWritingMode().IsOrthogonalTo(lineWM), + "Ruby frame writing-mode shouldn't be orthogonal to its line"); + + AutoRubyTextContainerArray textContainers(aBaseContainer); + const uint32_t rtcCount = textContainers.Length(); + + ReflowOutput baseMetrics(aReflowInput); + bool pushedFrame; + aReflowInput.mLineLayout->ReflowFrame(aBaseContainer, aStatus, &baseMetrics, + pushedFrame); + + if (aStatus.IsInlineBreakBefore()) { + if (aBaseContainer != mFrames.FirstChild()) { + // Some segments may have been reflowed before, hence it is not + // a break-before for the ruby container. + aStatus.Reset(); + aStatus.SetInlineLineBreakAfter(); + aStatus.SetIncomplete(); + PushChildrenToOverflow(aBaseContainer, aBaseContainer->GetPrevSibling()); + aReflowInput.mLineLayout->SetDirtyNextLine(); + } + // This base container is not placed at all, we can skip all + // text containers paired with it. + return; + } + if (aStatus.IsIncomplete()) { + // It always promise that if the status is incomplete, there is a + // break occurs. Break before has been processed above. However, + // it is possible that break after happens with the frame reflow + // completed. It happens if there is a force break at the end. + MOZ_ASSERT(aStatus.IsInlineBreakAfter()); + // Find the previous sibling which we will + // insert new continuations after. + nsIFrame* lastChild; + if (rtcCount > 0) { + lastChild = textContainers.LastElement(); + } else { + lastChild = aBaseContainer; + } + + // Create continuations for the base container + nsIFrame* newBaseContainer = CreateNextInFlow(aBaseContainer); + // newBaseContainer is null if there are existing next-in-flows. + // We only need to move and push if there were not. + if (newBaseContainer) { + // Move the new frame after all the text containers + mFrames.RemoveFrame(newBaseContainer); + mFrames.InsertFrame(nullptr, lastChild, newBaseContainer); + + // Create continuations for text containers + nsIFrame* newLastChild = newBaseContainer; + for (uint32_t i = 0; i < rtcCount; i++) { + nsIFrame* newTextContainer = CreateNextInFlow(textContainers[i]); + MOZ_ASSERT(newTextContainer, + "Next-in-flow of rtc should not exist " + "if the corresponding rbc does not"); + mFrames.RemoveFrame(newTextContainer); + mFrames.InsertFrame(nullptr, newLastChild, newTextContainer); + newLastChild = newTextContainer; + } + } + if (lastChild != mFrames.LastChild()) { + // Always push the next frame after the last child in this segment. + // It is possible that we pulled it back before our next-in-flow + // drain our overflow. + PushChildrenToOverflow(lastChild->GetNextSibling(), lastChild); + aReflowInput.mLineLayout->SetDirtyNextLine(); + } + } else if (rtcCount) { + DestroyContext context(PresShell()); + // If the ruby base container is reflowed completely, the line + // layout will remove the next-in-flows of that frame. But the + // line layout is not aware of the ruby text containers, hence + // it is necessary to remove them here. + for (uint32_t i = 0; i < rtcCount; i++) { + if (nsIFrame* nextRTC = textContainers[i]->GetNextInFlow()) { + nextRTC->GetParent()->DeleteNextInFlowChild(context, nextRTC, true); + } + } + } + + nscoord segmentISize = baseMetrics.ISize(lineWM); + const nsSize dummyContainerSize; + LogicalRect baseRect = + aBaseContainer->GetLogicalRect(lineWM, dummyContainerSize); + // We need to position our rtc frames on one side or the other of the + // base container's rect, using a coordinate space that's relative to + // the ruby frame. Right now, the base container's rect's block-axis + // position is relative to the block container frame containing the + // lines, so here we reset it to the different between the ascents of + // the ruby container and the ruby base container, assuming they are + // aligned with the baseline. + // XXX We may need to add border/padding here. See bug 1055667. + baseRect.BStart(lineWM) = aBlockStartAscent - baseMetrics.BlockStartAscent(); + // The rect for offsets of text containers. + LogicalRect offsetRect = baseRect; + RubyBlockLeadings descLeadings = aBaseContainer->GetDescendantLeadings(); + offsetRect.BStart(lineWM) -= descLeadings.mStart; + offsetRect.BSize(lineWM) += descLeadings.mStart + descLeadings.mEnd; + Maybe<LineRelativeDir> lastLineSide; + for (uint32_t i = 0; i < rtcCount; i++) { + nsRubyTextContainerFrame* textContainer = textContainers[i]; + WritingMode rtcWM = textContainer->GetWritingMode(); + nsReflowStatus textReflowStatus; + ReflowOutput textMetrics(aReflowInput); + ReflowInput textReflowInput(aPresContext, aReflowInput, textContainer, + availSize.ConvertTo(rtcWM, lineWM)); + textContainer->Reflow(aPresContext, textMetrics, textReflowInput, + textReflowStatus); + // Ruby text containers always return complete reflow status even when + // they have continuations, because the breaking has already been + // handled when reflowing the base containers. + NS_ASSERTION(textReflowStatus.IsEmpty(), + "Ruby text container must not break itself inside"); + const LogicalSize size = textMetrics.Size(lineWM); + textContainer->SetSize(lineWM, size); + + nscoord reservedISize = RubyUtils::GetReservedISize(textContainer); + segmentISize = std::max(segmentISize, size.ISize(lineWM) + reservedISize); + + Maybe<LineRelativeDir> lineSide; + switch (textContainer->StyleText()->mRubyPosition) { + case StyleRubyPosition::Over: + lineSide.emplace(eLineRelativeDirOver); + break; + case StyleRubyPosition::Under: + lineSide.emplace(eLineRelativeDirUnder); + break; + case StyleRubyPosition::AlternateOver: + if (lastLineSide.isSome() && + lastLineSide.value() == eLineRelativeDirOver) { + lineSide.emplace(eLineRelativeDirUnder); + } else { + lineSide.emplace(eLineRelativeDirOver); + } + break; + case StyleRubyPosition::AlternateUnder: + if (lastLineSide.isSome() && + lastLineSide.value() == eLineRelativeDirUnder) { + lineSide.emplace(eLineRelativeDirOver); + } else { + lineSide.emplace(eLineRelativeDirUnder); + } + break; + default: + // XXX inter-character support in bug 1055672 + MOZ_ASSERT_UNREACHABLE("Unsupported ruby-position"); + } + lastLineSide = lineSide; + + LogicalPoint position(lineWM); + if (lineSide.isSome()) { + LogicalSide logicalSide = + lineWM.LogicalSideForLineRelativeDir(lineSide.value()); + if (StaticPrefs::layout_css_ruby_intercharacter_enabled() && + rtcWM.IsVerticalRL() && + lineWM.GetInlineDir() == WritingMode::eInlineLTR) { + // Inter-character ruby annotations are only supported for vertical-rl + // in ltr horizontal writing. Fall back to non-inter-character behavior + // otherwise. + LogicalPoint offset( + lineWM, offsetRect.ISize(lineWM), + offsetRect.BSize(lineWM) > size.BSize(lineWM) + ? (offsetRect.BSize(lineWM) - size.BSize(lineWM)) / 2 + : 0); + position = offsetRect.Origin(lineWM) + offset; + aReflowInput.mLineLayout->AdvanceICoord(size.ISize(lineWM)); + } else if (logicalSide == eLogicalSideBStart) { + offsetRect.BStart(lineWM) -= size.BSize(lineWM); + offsetRect.BSize(lineWM) += size.BSize(lineWM); + position = offsetRect.Origin(lineWM); + } else if (logicalSide == eLogicalSideBEnd) { + position = offsetRect.Origin(lineWM) + + LogicalPoint(lineWM, 0, offsetRect.BSize(lineWM)); + offsetRect.BSize(lineWM) += size.BSize(lineWM); + } else { + MOZ_ASSERT_UNREACHABLE("???"); + } + } + // Using a dummy container-size here, so child positioning may not be + // correct. We will fix it in nsLineLayout after the whole line is + // reflowed. + FinishReflowChild(textContainer, aPresContext, textMetrics, + &textReflowInput, lineWM, position, dummyContainerSize, + ReflowChildFlags::Default); + } + MOZ_ASSERT(baseRect.ISize(lineWM) == offsetRect.ISize(lineWM), + "Annotations should only be placed on the block directions"); + + nscoord deltaISize = segmentISize - baseMetrics.ISize(lineWM); + if (deltaISize <= 0) { + RubyUtils::ClearReservedISize(aBaseContainer); + } else { + RubyUtils::SetReservedISize(aBaseContainer, deltaISize); + aReflowInput.mLineLayout->AdvanceICoord(deltaISize); + } + + // Set block leadings of the base container. + // The leadings are the difference between the offsetRect and the rect + // of this ruby container, which has block start zero and block size + // aBlockSize. + nscoord startLeading = -offsetRect.BStart(lineWM); + nscoord endLeading = offsetRect.BEnd(lineWM) - aBlockSize; + // XXX When bug 765861 gets fixed, this warning should be upgraded. + NS_WARNING_ASSERTION(startLeading >= 0 && endLeading >= 0, + "Leadings should be non-negative (because adding " + "ruby annotation can only increase the size)"); + mLeadings.Update(startLeading, endLeading); +} + +nsRubyBaseContainerFrame* nsRubyFrame::PullOneSegment( + const nsLineLayout* aLineLayout, ContinuationTraversingState& aState) { + // Pull a ruby base container + nsIFrame* baseFrame = GetNextInFlowChild(aState); + if (!baseFrame) { + return nullptr; + } + MOZ_ASSERT(baseFrame->IsRubyBaseContainerFrame()); + + // Get the float containing block of the base frame before we pull it. + nsBlockFrame* oldFloatCB = nsLayoutUtils::GetFloatContainingBlock(baseFrame); + PullNextInFlowChild(aState); + + // Pull all ruby text containers following the base container + nsIFrame* nextFrame; + while ((nextFrame = GetNextInFlowChild(aState)) != nullptr && + nextFrame->IsRubyTextContainerFrame()) { + PullNextInFlowChild(aState); + } + + if (nsBlockFrame* newFloatCB = + do_QueryFrame(aLineLayout->LineContainerFrame())) { + if (oldFloatCB && oldFloatCB != newFloatCB) { + newFloatCB->ReparentFloats(baseFrame, oldFloatCB, true); + } + } + + return static_cast<nsRubyBaseContainerFrame*>(baseFrame); +} |