blob: 868f86bfd98439ac79f6c67e4a9726c7b1063cf0 [file] [log] [blame]
/*
* Copyright (C) 2013 Adobe Systems Incorporated. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above
* copyright notice, this list of conditions and the following
* disclaimer.
* 2. Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following
* disclaimer in the documentation and/or other materials
* provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER "AS IS" AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY,
* OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR
* TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
* THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*/
#include "config.h"
#include "RenderNamedFlowFragment.h"
#include "FlowThreadController.h"
#include "PaintInfo.h"
#include "RenderBoxRegionInfo.h"
#include "RenderFlowThread.h"
#include "RenderNamedFlowThread.h"
#include "RenderView.h"
#include "StyleResolver.h"
namespace WebCore {
RenderNamedFlowFragment::RenderNamedFlowFragment(Document& document, PassRef<RenderStyle> style)
: RenderRegion(document, std::move(style), nullptr)
, m_hasCustomRegionStyle(false)
{
}
RenderNamedFlowFragment::~RenderNamedFlowFragment()
{
}
PassRef<RenderStyle> RenderNamedFlowFragment::createStyle(const RenderStyle& parentStyle)
{
auto style = RenderStyle::createAnonymousStyleWithDisplay(&parentStyle, BLOCK);
style.get().setFlowThread(parentStyle.flowThread());
style.get().setRegionThread(parentStyle.regionThread());
style.get().setRegionFragment(parentStyle.regionFragment());
style.get().setOverflowX(parentStyle.overflowX());
style.get().setOverflowY(parentStyle.overflowY());
#if ENABLE(CSS_SHAPES)
style.get().setShapeInside(parentStyle.shapeInside());
#endif
return style;
}
void RenderNamedFlowFragment::styleDidChange(StyleDifference diff, const RenderStyle* oldStyle)
{
RenderRegion::styleDidChange(diff, oldStyle);
// If the region is not attached to any thread, there is no need to check
// whether the region has region styling since no content will be displayed
// into the region.
if (!m_flowThread) {
setHasCustomRegionStyle(false);
return;
}
checkRegionStyle();
if (parent() && parent()->needsLayout())
setNeedsLayout(MarkOnlyThis);
}
bool RenderNamedFlowFragment::shouldHaveAutoLogicalHeight() const
{
ASSERT(parent());
const RenderStyle& styleToUse = parent()->style();
bool hasSpecifiedEndpointsForHeight = styleToUse.logicalTop().isSpecified() && styleToUse.logicalBottom().isSpecified();
bool hasAnchoredEndpointsForHeight = parent()->isOutOfFlowPositioned() && hasSpecifiedEndpointsForHeight;
return styleToUse.logicalHeight().isAuto() && !hasAnchoredEndpointsForHeight;
}
LayoutUnit RenderNamedFlowFragment::maxPageLogicalHeight() const
{
ASSERT(m_flowThread);
ASSERT(hasAutoLogicalHeight() && m_flowThread->inMeasureContentLayoutPhase());
ASSERT(isAnonymous());
ASSERT(parent());
const RenderStyle& styleToUse = parent()->style();
return styleToUse.logicalMaxHeight().isUndefined() ? RenderFlowThread::maxLogicalHeight() : toRenderBlock(parent())->computeReplacedLogicalHeightUsing(styleToUse.logicalMaxHeight());
}
void RenderNamedFlowFragment::checkRegionStyle()
{
ASSERT(m_flowThread);
bool customRegionStyle = false;
// FIXME: Region styling doesn't work for pseudo elements.
if (!isPseudoElement())
customRegionStyle = view().document().ensureStyleResolver().checkRegionStyle(generatingElement());
setHasCustomRegionStyle(customRegionStyle);
toRenderNamedFlowThread(m_flowThread)->checkRegionsWithStyling();
}
PassRefPtr<RenderStyle> RenderNamedFlowFragment::computeStyleInRegion(const RenderObject* object)
{
ASSERT(object);
ASSERT(!object->isAnonymous());
ASSERT(object->node() && object->node()->isElementNode());
// FIXME: Region styling fails for pseudo-elements because the renderers don't have a node.
Element* element = toElement(object->node());
RefPtr<RenderStyle> renderObjectRegionStyle = object->view().document().ensureStyleResolver().styleForElement(element, 0, DisallowStyleSharing, MatchAllRules, this);
return renderObjectRegionStyle.release();
}
void RenderNamedFlowFragment::computeChildrenStyleInRegion(const RenderElement* object)
{
for (RenderObject* child = object->firstChild(); child; child = child->nextSibling()) {
auto it = m_renderObjectRegionStyle.find(child);
RefPtr<RenderStyle> childStyleInRegion;
bool objectRegionStyleCached = false;
if (it != m_renderObjectRegionStyle.end()) {
childStyleInRegion = it->value.style;
objectRegionStyleCached = true;
} else {
if (child->isAnonymous() || child->isInFlowRenderFlowThread())
childStyleInRegion = RenderStyle::createAnonymousStyleWithDisplay(&object->style(), child->style().display());
else if (child->isText())
childStyleInRegion = RenderStyle::clone(&object->style());
else
childStyleInRegion = computeStyleInRegion(child);
}
setObjectStyleInRegion(child, childStyleInRegion, objectRegionStyleCached);
if (child->isRenderElement())
computeChildrenStyleInRegion(toRenderElement(child));
}
}
void RenderNamedFlowFragment::setObjectStyleInRegion(RenderObject* object, PassRefPtr<RenderStyle> styleInRegion, bool objectRegionStyleCached)
{
ASSERT(object->flowThreadContainingBlock());
RefPtr<RenderStyle> objectOriginalStyle = &object->style();
if (object->isRenderElement())
toRenderElement(object)->setStyleInternal(*styleInRegion);
if (object->isBoxModelObject() && !object->hasBoxDecorations()) {
bool hasBoxDecorations = object->isTableCell()
|| object->style().hasBackground()
|| object->style().hasBorder()
|| object->style().hasAppearance()
|| object->style().boxShadow();
object->setHasBoxDecorations(hasBoxDecorations);
}
ObjectRegionStyleInfo styleInfo;
styleInfo.style = objectOriginalStyle;
styleInfo.cached = objectRegionStyleCached;
m_renderObjectRegionStyle.set(object, styleInfo);
}
void RenderNamedFlowFragment::clearObjectStyleInRegion(const RenderObject* object)
{
ASSERT(object);
m_renderObjectRegionStyle.remove(object);
// Clear the style for the children of this object.
for (RenderObject* child = object->firstChildSlow(); child; child = child->nextSibling())
clearObjectStyleInRegion(child);
}
void RenderNamedFlowFragment::setRegionObjectsRegionStyle()
{
if (!hasCustomRegionStyle())
return;
// Start from content nodes and recursively compute the style in region for the render objects below.
// If the style in region was already computed, used that style instead of computing a new one.
const RenderNamedFlowThread& namedFlow = view().flowThreadController().ensureRenderFlowThreadWithName(style().regionThread());
const NamedFlowContentElements& contentElements = namedFlow.contentElements();
for (auto iter = contentElements.begin(), end = contentElements.end(); iter != end; ++iter) {
const Element* element = *iter;
// The list of content nodes contains also the nodes with display:none.
if (!element->renderer())
continue;
RenderElement* object = element->renderer();
// If the content node does not flow any of its children in this region,
// we do not compute any style for them in this region.
if (!flowThread()->objectInFlowRegion(object, this))
continue;
// If the object has style in region, use that instead of computing a new one.
auto it = m_renderObjectRegionStyle.find(object);
RefPtr<RenderStyle> objectStyleInRegion;
bool objectRegionStyleCached = false;
if (it != m_renderObjectRegionStyle.end()) {
objectStyleInRegion = it->value.style;
ASSERT(it->value.cached);
objectRegionStyleCached = true;
} else
objectStyleInRegion = computeStyleInRegion(object);
setObjectStyleInRegion(object, objectStyleInRegion, objectRegionStyleCached);
computeChildrenStyleInRegion(object);
}
}
void RenderNamedFlowFragment::restoreRegionObjectsOriginalStyle()
{
if (!hasCustomRegionStyle())
return;
RenderObjectRegionStyleMap temp;
for (auto iter = m_renderObjectRegionStyle.begin(), end = m_renderObjectRegionStyle.end(); iter != end; ++iter) {
RenderObject* object = const_cast<RenderObject*>(iter->key);
RefPtr<RenderStyle> objectRegionStyle = &object->style();
RefPtr<RenderStyle> objectOriginalStyle = iter->value.style;
if (object->isRenderElement())
toRenderElement(object)->setStyleInternal(*objectOriginalStyle);
bool shouldCacheRegionStyle = iter->value.cached;
if (!shouldCacheRegionStyle) {
// Check whether we should cache the computed style in region.
unsigned changedContextSensitiveProperties = ContextSensitivePropertyNone;
StyleDifference styleDiff = objectOriginalStyle->diff(objectRegionStyle.get(), changedContextSensitiveProperties);
if (styleDiff < StyleDifferenceLayoutPositionedMovementOnly)
shouldCacheRegionStyle = true;
}
if (shouldCacheRegionStyle) {
ObjectRegionStyleInfo styleInfo;
styleInfo.style = objectRegionStyle;
styleInfo.cached = true;
temp.set(object, styleInfo);
}
}
m_renderObjectRegionStyle.swap(temp);
}
RenderNamedFlowThread* RenderNamedFlowFragment::namedFlowThread() const
{
return toRenderNamedFlowThread(flowThread());
}
} // namespace WebCore