2014-07-23 06:08:01 +04:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* vim: set ts=2 et sw=2 tw=80: */
|
|
|
|
/* This Source Code is subject to the terms of the Mozilla Public License
|
|
|
|
* version 2.0 (the "License"). You can obtain a copy of the License at
|
|
|
|
* http://mozilla.org/MPL/2.0/. */
|
|
|
|
|
|
|
|
/* rendering object for CSS "display: ruby" */
|
|
|
|
#include "nsRubyFrame.h"
|
2014-08-15 21:34:20 +04:00
|
|
|
#include "nsLineLayout.h"
|
2014-07-23 06:08:01 +04:00
|
|
|
#include "nsPresContext.h"
|
|
|
|
#include "nsStyleContext.h"
|
2014-08-15 21:34:20 +04:00
|
|
|
#include "WritingModes.h"
|
|
|
|
#include "nsRubyBaseContainerFrame.h"
|
|
|
|
#include "nsRubyTextContainerFrame.h"
|
|
|
|
|
|
|
|
using namespace mozilla;
|
2014-07-23 06:08:01 +04:00
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
|
|
|
|
// Frame class boilerplate
|
|
|
|
// =======================
|
|
|
|
|
|
|
|
NS_QUERYFRAME_HEAD(nsRubyFrame)
|
|
|
|
NS_QUERYFRAME_ENTRY(nsRubyFrame)
|
|
|
|
NS_QUERYFRAME_TAIL_INHERITING(nsContainerFrame)
|
|
|
|
|
|
|
|
NS_IMPL_FRAMEARENA_HELPERS(nsRubyFrame)
|
|
|
|
|
|
|
|
nsContainerFrame*
|
|
|
|
NS_NewRubyFrame(nsIPresShell* aPresShell,
|
|
|
|
nsStyleContext* aContext)
|
|
|
|
{
|
|
|
|
return new (aPresShell) nsRubyFrame(aContext);
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
|
|
|
|
// nsRubyFrame Method Implementations
|
|
|
|
// ==================================
|
|
|
|
|
|
|
|
nsIAtom*
|
|
|
|
nsRubyFrame::GetType() const
|
|
|
|
{
|
|
|
|
return nsGkAtoms::rubyFrame;
|
|
|
|
}
|
|
|
|
|
2014-08-15 21:34:20 +04:00
|
|
|
/* virtual */ bool
|
2014-07-23 06:08:01 +04:00
|
|
|
nsRubyFrame::IsFrameOfType(uint32_t aFlags) const
|
|
|
|
{
|
|
|
|
return nsContainerFrame::IsFrameOfType(aFlags &
|
2014-08-15 21:34:20 +04:00
|
|
|
~(nsIFrame::eLineParticipant));
|
2014-07-23 06:08:01 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEBUG_FRAME_DUMP
|
|
|
|
nsresult
|
|
|
|
nsRubyFrame::GetFrameName(nsAString& aResult) const
|
|
|
|
{
|
|
|
|
return MakeFrameName(NS_LITERAL_STRING("Ruby"), aResult);
|
|
|
|
}
|
|
|
|
#endif
|
2014-08-15 21:34:20 +04:00
|
|
|
|
|
|
|
void
|
|
|
|
nsRubyFrame::CalculateColSizes(nsRenderingContext* aRenderingContext,
|
|
|
|
nsTArray<nscoord>& aColSizes)
|
|
|
|
{
|
|
|
|
nsFrameList::Enumerator e(this->PrincipalChildList());
|
|
|
|
uint32_t annotationNum = 0;
|
|
|
|
int segmentNum = -1;
|
|
|
|
|
|
|
|
nsTArray<int> segmentBaseCounts;
|
|
|
|
|
|
|
|
for(; !e.AtEnd(); e.Next()) {
|
|
|
|
nsIFrame* childFrame = e.get();
|
|
|
|
if (childFrame->GetType() == nsGkAtoms::rubyBaseContainerFrame) {
|
|
|
|
segmentNum++;
|
|
|
|
segmentBaseCounts.AppendElement(0);
|
|
|
|
nsFrameList::Enumerator bases(childFrame->PrincipalChildList());
|
|
|
|
for(; !bases.AtEnd(); bases.Next()) {
|
|
|
|
aColSizes.AppendElement(bases.get()->GetPrefISize(aRenderingContext));
|
|
|
|
segmentBaseCounts.ElementAt(segmentNum)++;
|
|
|
|
}
|
|
|
|
} else if (childFrame->GetType() == nsGkAtoms::rubyTextContainerFrame) {
|
|
|
|
if (segmentNum == -1) {
|
|
|
|
// No rbc exists for first segment, so act as if there is one
|
|
|
|
segmentNum++;
|
|
|
|
segmentBaseCounts.AppendElement(1);
|
|
|
|
aColSizes.AppendElement(0);
|
|
|
|
}
|
|
|
|
nsFrameList::Enumerator annotations(childFrame->PrincipalChildList());
|
|
|
|
uint32_t baseCount = segmentBaseCounts.ElementAt(segmentNum);
|
|
|
|
for(; !annotations.AtEnd(); annotations.Next()) {
|
|
|
|
nsIFrame* annotationFrame = annotations.get();
|
|
|
|
if (annotationNum > baseCount) {
|
|
|
|
aColSizes.AppendElement(annotationFrame->
|
|
|
|
GetPrefISize(aRenderingContext));
|
|
|
|
baseCount++;
|
|
|
|
segmentBaseCounts.ElementAt(segmentNum) = baseCount;
|
|
|
|
annotationNum++;
|
|
|
|
} else if (annotationNum < baseCount - 1) {
|
|
|
|
//there are fewer annotations than bases, so the last annotation is
|
|
|
|
//associated with (spans) the remaining bases. This means these
|
|
|
|
//columns can't be broken up, so gather their entire ISize in one
|
|
|
|
//entry of aColSizes and clear the other entries.
|
|
|
|
int baseSum = 0;
|
|
|
|
for (uint32_t i = annotationNum; i < annotationNum + baseCount; i++) {
|
|
|
|
baseSum += aColSizes.ElementAt(i);
|
|
|
|
if (i > annotationNum) {
|
|
|
|
aColSizes.ElementAt(i) = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
aColSizes.ElementAt(annotationNum) =
|
|
|
|
std::max(baseSum, annotationFrame->GetPrefISize(aRenderingContext));
|
|
|
|
annotationNum = baseCount;
|
|
|
|
} else {
|
|
|
|
aColSizes.ElementAt(annotationNum) =
|
|
|
|
std::max(aColSizes.ElementAt(annotationNum),
|
|
|
|
annotationFrame->GetPrefISize(aRenderingContext));
|
|
|
|
annotationNum++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
NS_ASSERTION(false, "Unrecognized child type for ruby frame.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* virtual */ void
|
|
|
|
nsRubyFrame::AddInlineMinISize(nsRenderingContext *aRenderingContext,
|
|
|
|
nsIFrame::InlineMinISizeData *aData)
|
|
|
|
{
|
|
|
|
//FIXME: This needs to handle the cases where it's possible for a ruby base to
|
|
|
|
//break, as well as forced breaks.
|
|
|
|
nsTArray<int> colSizes;
|
|
|
|
CalculateColSizes(aRenderingContext, colSizes);
|
|
|
|
|
|
|
|
nscoord max = 0;
|
|
|
|
for (uint32_t i = 0; i < colSizes.Length(); i++) {
|
|
|
|
if (colSizes.ElementAt(i) > max) {
|
|
|
|
max = colSizes.ElementAt(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
aData->currentLine += max;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* virtual */ void
|
|
|
|
nsRubyFrame::AddInlinePrefISize(nsRenderingContext *aRenderingContext,
|
|
|
|
nsIFrame::InlinePrefISizeData *aData)
|
|
|
|
{
|
|
|
|
nsTArray<int> colSizes;
|
|
|
|
CalculateColSizes(aRenderingContext, colSizes);
|
|
|
|
|
|
|
|
nscoord sum = 0;
|
|
|
|
for (uint32_t i = 0; i < colSizes.Length(); i++) {
|
|
|
|
sum += colSizes.ElementAt(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
aData->currentLine += sum;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* virtual */ nscoord
|
|
|
|
nsRubyFrame::GetLogicalBaseline(WritingMode aWritingMode) const
|
|
|
|
{
|
|
|
|
return mBaseline;
|
|
|
|
}
|
|
|
|
|
2014-08-15 21:34:20 +04:00
|
|
|
/* virtual */ bool
|
|
|
|
nsRubyFrame::CanContinueTextRun() const
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-08-15 21:34:20 +04:00
|
|
|
/* virtual */ void
|
|
|
|
nsRubyFrame::Reflow(nsPresContext* aPresContext,
|
|
|
|
nsHTMLReflowMetrics& aDesiredSize,
|
|
|
|
const nsHTMLReflowState& aReflowState,
|
|
|
|
nsReflowStatus& aStatus)
|
|
|
|
{
|
|
|
|
DO_GLOBAL_REFLOW_COUNT("nsRubyFrame");
|
|
|
|
DISPLAY_REFLOW(aPresContext, this, aReflowState, aDesiredSize, aStatus);
|
|
|
|
|
|
|
|
if (!aReflowState.mLineLayout) {
|
|
|
|
NS_ASSERTION(aReflowState.mLineLayout,
|
|
|
|
"No line layout provided to RubyFrame reflow method.");
|
|
|
|
aStatus = NS_FRAME_COMPLETE;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Begin the span for the ruby frame
|
|
|
|
WritingMode frameWM = aReflowState.GetWritingMode();
|
|
|
|
WritingMode lineWM = aReflowState.mLineLayout->GetWritingMode();
|
|
|
|
LogicalMargin borderPadding = aReflowState.ComputedLogicalBorderPadding();
|
|
|
|
nscoord availableISize = aReflowState.AvailableISize();
|
|
|
|
NS_ASSERTION(availableISize != NS_UNCONSTRAINEDSIZE,
|
|
|
|
"should no longer use available widths");
|
|
|
|
// Subtract off inline axis border+padding from availableISize
|
|
|
|
availableISize -= borderPadding.IStartEnd(frameWM);
|
|
|
|
aReflowState.mLineLayout->BeginSpan(this, &aReflowState,
|
|
|
|
borderPadding.IStart(frameWM),
|
|
|
|
availableISize, &mBaseline);
|
|
|
|
|
|
|
|
// FIXME: line breaking / continuations not yet implemented
|
|
|
|
aStatus = NS_FRAME_COMPLETE;
|
|
|
|
LogicalSize availSize(lineWM, aReflowState.AvailableISize(),
|
|
|
|
aReflowState.AvailableBSize());
|
|
|
|
// The ruby base container for the current segment
|
|
|
|
nsRubyBaseContainerFrame* segmentRBC = nullptr;
|
|
|
|
nscoord annotationBSize = 0;
|
|
|
|
for (nsFrameList::Enumerator e(mFrames); !e.AtEnd(); e.Next()) {
|
|
|
|
nsIFrame* childFrame = e.get();
|
|
|
|
if (e.get()->GetType() == nsGkAtoms::rubyBaseContainerFrame) {
|
|
|
|
if (segmentRBC) {
|
|
|
|
annotationBSize = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Figure out what all the text containers are for this segment (necessary
|
|
|
|
// for reflow calculations)
|
|
|
|
segmentRBC = do_QueryFrame(childFrame);
|
|
|
|
nsFrameList::Enumerator segment(e);
|
|
|
|
segment.Next();
|
|
|
|
while (!segment.AtEnd() && (segment.get()->GetType() !=
|
|
|
|
nsGkAtoms::rubyBaseContainerFrame)) {
|
|
|
|
if (segment.get()->GetType() == nsGkAtoms::rubyTextContainerFrame) {
|
|
|
|
segmentRBC->AppendTextContainer(segment.get());
|
|
|
|
}
|
|
|
|
segment.Next();
|
|
|
|
}
|
|
|
|
|
|
|
|
nsReflowStatus frameReflowStatus;
|
|
|
|
nsHTMLReflowMetrics metrics(aReflowState, aDesiredSize.mFlags);
|
|
|
|
nsHTMLReflowState childReflowState(aPresContext, aReflowState,
|
|
|
|
childFrame, availSize);
|
|
|
|
childReflowState.mLineLayout = aReflowState.mLineLayout;
|
|
|
|
childFrame->Reflow(aPresContext, metrics, childReflowState,
|
|
|
|
frameReflowStatus);
|
|
|
|
NS_ASSERTION(frameReflowStatus == NS_FRAME_COMPLETE,
|
|
|
|
"Ruby line breaking is not yet implemented");
|
|
|
|
childFrame->SetSize(LogicalSize(lineWM, metrics.ISize(lineWM),
|
|
|
|
metrics.BSize(lineWM)));
|
|
|
|
FinishReflowChild(childFrame, aPresContext, metrics, &childReflowState, 0,
|
|
|
|
0, NS_FRAME_NO_MOVE_FRAME | NS_FRAME_NO_MOVE_VIEW);
|
|
|
|
|
|
|
|
} else if (childFrame->GetType() == nsGkAtoms::rubyTextContainerFrame) {
|
|
|
|
nsReflowStatus frameReflowStatus;
|
|
|
|
nsHTMLReflowMetrics metrics(aReflowState, aDesiredSize.mFlags);
|
|
|
|
nsHTMLReflowState childReflowState(aPresContext, aReflowState, childFrame,
|
|
|
|
availSize);
|
|
|
|
childReflowState.mLineLayout = aReflowState.mLineLayout;
|
|
|
|
childFrame->Reflow(aPresContext, metrics, childReflowState,
|
|
|
|
frameReflowStatus);
|
|
|
|
NS_ASSERTION(frameReflowStatus == NS_FRAME_COMPLETE,
|
|
|
|
"Ruby line breaking is not yet implemented");
|
|
|
|
annotationBSize += metrics.BSize(lineWM);
|
|
|
|
childFrame->SetSize(LogicalSize(lineWM, metrics.ISize(lineWM),
|
|
|
|
metrics.BSize(lineWM)));
|
|
|
|
// FIXME: This is a temporary calculation for finding the block coordinate
|
|
|
|
// of the ruby text container. A better one replace it once it's been
|
|
|
|
// spec'ed.
|
|
|
|
nscoord baseContainerBCoord;
|
|
|
|
if (segmentRBC) {
|
|
|
|
// Find the starting block coordinate of the ruby base container for
|
|
|
|
// this segment. For now, the ruby text container will be placed so that
|
|
|
|
// its bottom edge touches this coordinate.
|
|
|
|
baseContainerBCoord = segmentRBC->
|
|
|
|
GetLogicalPosition(this->GetParent()->GetLogicalSize().ISize(lineWM)).
|
|
|
|
B(lineWM);
|
|
|
|
} else {
|
|
|
|
baseContainerBCoord = 0;
|
|
|
|
}
|
|
|
|
FinishReflowChild(childFrame, aPresContext, metrics, &childReflowState, 0,
|
|
|
|
baseContainerBCoord - metrics.BSize(lineWM), 0);
|
|
|
|
} else {
|
|
|
|
NS_NOTREACHED(
|
|
|
|
"Unrecognized child type for ruby frame will not be reflowed.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Null the pointers between child frames.
|
|
|
|
for (nsFrameList::Enumerator children(mFrames); !children.AtEnd();
|
|
|
|
children.Next()) {
|
|
|
|
nsRubyBaseContainerFrame* rbcFrame = do_QueryFrame(children.get());
|
|
|
|
if (rbcFrame) {
|
|
|
|
rbcFrame->ClearTextContainers();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
aDesiredSize.ISize(lineWM) = aReflowState.mLineLayout->EndSpan(this);
|
|
|
|
nsLayoutUtils::SetBSizeFromFontMetrics(this, aDesiredSize, aReflowState,
|
|
|
|
borderPadding, lineWM, frameWM);
|
|
|
|
}
|