blob: 975305d2e20a2b51123359e6a183a5da330438a1 [file] [log] [blame]
/*
* Copyright (C) 2003, 2009, 2012 Apple Inc. All rights reserved.
* Copyright (C) 2013 Intel Corporation. All rights reserved.
*
* Portions are Copyright (C) 1998 Netscape Communications Corporation.
*
* Other contributors:
* Robert O'Callahan <roc+@cs.cmu.edu>
* David Baron <dbaron@fas.harvard.edu>
* Christian Biesinger <cbiesinger@web.de>
* Randall Jesup <rjesup@wgate.com>
* Roland Mainz <roland.mainz@informatik.med.uni-giessen.de>
* Josh Soref <timeless@mac.com>
* Boris Zbarsky <bzbarsky@mit.edu>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*
* Alternatively, the contents of this file may be used under the terms
* of either the Mozilla Public License Version 1.1, found at
* http://www.mozilla.org/MPL/ (the "MPL") or the GNU General Public
* License Version 2.0, found at http://www.fsf.org/copyleft/gpl.html
* (the "GPL"), in which case the provisions of the MPL or the GPL are
* applicable instead of those above. If you wish to allow use of your
* version of this file only under the terms of one of those two
* licenses (the MPL or the GPL) and not to allow others to use your
* version of this file under the LGPL, indicate your decision by
* deletingthe provisions above and replace them with the notice and
* other provisions required by the MPL or the GPL, as the case may be.
* If you do not delete the provisions above, a recipient may use your
* version of this file under any of the LGPL, the MPL or the GPL.
*/
#ifndef DeprecatedPaintLayerStackingNode_h
#define DeprecatedPaintLayerStackingNode_h
#include "core/CoreExport.h"
#include "core/layout/LayoutBoxModelObject.h"
#include "wtf/Noncopyable.h"
#include "wtf/OwnPtr.h"
#include "wtf/Vector.h"
namespace blink {
class DeprecatedPaintLayer;
class DeprecatedPaintLayerCompositor;
class ComputedStyle;
class LayoutBoxModelObject;
// DeprecatedPaintLayerStackingNode represents anything that is a stacking
// context or treated as a stacking context.
//
// Stacking contexts are the basis for the CSS painting algorithm. The paint
// order is determined by walking stacking contexts (or elements treated like a
// stacking context like positioned objects or floats) in an order defined by
// ‘z-index’. This walk is interleaved with content that is not a stacking.
// context. See CSS 2.1 appendix E for the actual algorithm
// http://www.w3.org/TR/CSS21/zindex.html
// See also DeprecatedPaintLayerPainter (in particular paintLayerContents) for
// our implementation of the walk.
//
// Stacking contexts form a subtree over the layout tree. Ideally we would want
// objects of this class to be a node in this tree but there are potential
// issues with stale pointers so we rely on DeprecatedPaintLayer's tree
// structure.
//
// This class's purpose is to represent a node in the stacking context tree
// (aka paint tree). It currently caches the z-order lists for painting and
// hit-testing.
//
// To implement any z-order list iterations, use
// DeprecatedPaintLayerStackingNodeIterator and
// DeprecatedPaintLayerStackingNodeReverseIterator.
//
// This class is NOT DEPRECATED, DeprecatedPaintLayer is and we match its
// naming.
class CORE_EXPORT DeprecatedPaintLayerStackingNode {
WTF_MAKE_FAST_ALLOCATED(DeprecatedPaintLayerStackingNode);
WTF_MAKE_NONCOPYABLE(DeprecatedPaintLayerStackingNode);
public:
explicit DeprecatedPaintLayerStackingNode(DeprecatedPaintLayer*);
~DeprecatedPaintLayerStackingNode();
int zIndex() const { return layoutObject()->style()->zIndex(); }
bool isStackingContext() const { return layoutObject()->style()->isStackingContext(); }
// Update our normal and z-index lists.
void updateLayerListsIfNeeded();
bool zOrderListsDirty() const { return m_zOrderListsDirty; }
void dirtyZOrderLists();
void updateZOrderLists();
void clearZOrderLists();
void dirtyStackingContextZOrderLists();
bool hasPositiveZOrderList() const { return posZOrderList() && posZOrderList()->size(); }
bool hasNegativeZOrderList() const { return negZOrderList() && negZOrderList()->size(); }
bool isTreatedAsOrStackingContext() const { return m_isTreatedAsOrStackingContext; }
void updateIsTreatedAsStackingContext();
void updateStackingNodesAfterStyleChange(const ComputedStyle* oldStyle);
DeprecatedPaintLayerStackingNode* ancestorStackingContextNode() const;
DeprecatedPaintLayer* layer() const { return m_layer; }
#if ENABLE(ASSERT)
bool layerListMutationAllowed() const { return m_layerListMutationAllowed; }
void setLayerListMutationAllowed(bool flag) { m_layerListMutationAllowed = flag; }
#endif
private:
friend class DeprecatedPaintLayerStackingNodeIterator;
friend class DeprecatedPaintLayerStackingNodeReverseIterator;
friend class LayoutTreeAsText;
Vector<DeprecatedPaintLayerStackingNode*>* posZOrderList() const
{
ASSERT(!m_zOrderListsDirty);
ASSERT(isStackingContext() || !m_posZOrderList);
return m_posZOrderList.get();
}
Vector<DeprecatedPaintLayerStackingNode*>* negZOrderList() const
{
ASSERT(!m_zOrderListsDirty);
ASSERT(isStackingContext() || !m_negZOrderList);
return m_negZOrderList.get();
}
void rebuildZOrderLists();
void collectLayers(OwnPtr<Vector<DeprecatedPaintLayerStackingNode*>>& posZOrderList, OwnPtr<Vector<DeprecatedPaintLayerStackingNode*>>& negZOrderList);
#if ENABLE(ASSERT)
bool isInStackingParentZOrderLists() const;
void updateStackingParentForZOrderLists(DeprecatedPaintLayerStackingNode* stackingParent);
void setStackingParent(DeprecatedPaintLayerStackingNode* stackingParent) { m_stackingParent = stackingParent; }
#endif
bool shouldBeTreatedAsOrStackingContext() const { return layoutObject()->style()->isTreatedAsOrStackingContext(); }
bool isDirtyStackingContext() const { return m_zOrderListsDirty && isStackingContext(); }
DeprecatedPaintLayerCompositor* compositor() const;
// We can't return a LayoutBox as LayoutInline can be a stacking context.
LayoutBoxModelObject* layoutObject() const;
DeprecatedPaintLayer* m_layer;
// m_posZOrderList holds a sorted list of all the descendant nodes within
// that have z-indices of 0 or greater (auto will count as 0).
// m_negZOrderList holds descendants within our stacking context with
// negative z-indices.
OwnPtr<Vector<DeprecatedPaintLayerStackingNode*>> m_posZOrderList;
OwnPtr<Vector<DeprecatedPaintLayerStackingNode*>> m_negZOrderList;
// This boolean caches whether the z-order lists above are dirty.
// It is only ever set for stacking contexts, as no other element can
// have z-order lists.
unsigned m_zOrderListsDirty : 1;
// This attribute caches whether the element was a stacking context or
// was treated like a stacking context, so that we can do comparison against
// it during style change (styleDidChange in particular), as we have lost
// the previous style information.
unsigned m_isTreatedAsOrStackingContext: 1;
#if ENABLE(ASSERT)
unsigned m_layerListMutationAllowed : 1;
DeprecatedPaintLayerStackingNode* m_stackingParent;
#endif
};
inline void DeprecatedPaintLayerStackingNode::clearZOrderLists()
{
ASSERT(!isStackingContext());
#if ENABLE(ASSERT)
updateStackingParentForZOrderLists(0);
#endif
m_posZOrderList.clear();
m_negZOrderList.clear();
}
inline void DeprecatedPaintLayerStackingNode::updateZOrderLists()
{
if (!m_zOrderListsDirty)
return;
if (!isStackingContext()) {
clearZOrderLists();
m_zOrderListsDirty = false;
return;
}
rebuildZOrderLists();
}
#if ENABLE(ASSERT)
class LayerListMutationDetector {
public:
explicit LayerListMutationDetector(DeprecatedPaintLayerStackingNode* stackingNode)
: m_stackingNode(stackingNode)
, m_previousMutationAllowedState(stackingNode->layerListMutationAllowed())
{
m_stackingNode->setLayerListMutationAllowed(false);
}
~LayerListMutationDetector()
{
m_stackingNode->setLayerListMutationAllowed(m_previousMutationAllowedState);
}
private:
DeprecatedPaintLayerStackingNode* m_stackingNode;
bool m_previousMutationAllowedState;
};
#endif
} // namespace blink
#endif // DeprecatedPaintLayerStackingNode_h