| OLD | NEW |
| (Empty) |
| 1 /* | |
| 2 * Copyright (C) 2006, 2007, 2008, 2009, 2010, 2011, 2012 Apple Inc. All rights
reserved. | |
| 3 * | |
| 4 * Portions are Copyright (C) 1998 Netscape Communications Corporation. | |
| 5 * | |
| 6 * Other contributors: | |
| 7 * Robert O'Callahan <roc+@cs.cmu.edu> | |
| 8 * David Baron <dbaron@fas.harvard.edu> | |
| 9 * Christian Biesinger <cbiesinger@web.de> | |
| 10 * Randall Jesup <rjesup@wgate.com> | |
| 11 * Roland Mainz <roland.mainz@informatik.med.uni-giessen.de> | |
| 12 * Josh Soref <timeless@mac.com> | |
| 13 * Boris Zbarsky <bzbarsky@mit.edu> | |
| 14 * | |
| 15 * This library is free software; you can redistribute it and/or | |
| 16 * modify it under the terms of the GNU Lesser General Public | |
| 17 * License as published by the Free Software Foundation; either | |
| 18 * version 2.1 of the License, or (at your option) any later version. | |
| 19 * | |
| 20 * This library is distributed in the hope that it will be useful, | |
| 21 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
| 22 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | |
| 23 * Lesser General Public License for more details. | |
| 24 * | |
| 25 * You should have received a copy of the GNU Lesser General Public | |
| 26 * License along with this library; if not, write to the Free Software | |
| 27 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 US
A | |
| 28 * | |
| 29 * Alternatively, the contents of this file may be used under the terms | |
| 30 * of either the Mozilla Public License Version 1.1, found at | |
| 31 * http://www.mozilla.org/MPL/ (the "MPL") or the GNU General Public | |
| 32 * License Version 2.0, found at http://www.fsf.org/copyleft/gpl.html | |
| 33 * (the "GPL"), in which case the provisions of the MPL or the GPL are | |
| 34 * applicable instead of those above. If you wish to allow use of your | |
| 35 * version of this file only under the terms of one of those two | |
| 36 * licenses (the MPL or the GPL) and not to allow others to use your | |
| 37 * version of this file under the LGPL, indicate your decision by | |
| 38 * deletingthe provisions above and replace them with the notice and | |
| 39 * other provisions required by the MPL or the GPL, as the case may be. | |
| 40 * If you do not delete the provisions above, a recipient may use your | |
| 41 * version of this file under any of the LGPL, the MPL or the GPL. | |
| 42 */ | |
| 43 | |
| 44 #include "config.h" | |
| 45 #include "core/rendering/RenderLayerStackingNode.h" | |
| 46 | |
| 47 #include "core/layout/compositing/RenderLayerCompositor.h" | |
| 48 #include "core/rendering/RenderLayer.h" | |
| 49 #include "core/rendering/RenderView.h" | |
| 50 #include "public/platform/Platform.h" | |
| 51 | |
| 52 namespace blink { | |
| 53 | |
| 54 // FIXME: This should not require RenderLayer. There is currently a cycle where | |
| 55 // in order to determine if we shoulBeNormalFlowOnly() we have to ask the render | |
| 56 // layer about some of its state. | |
| 57 RenderLayerStackingNode::RenderLayerStackingNode(RenderLayer* layer) | |
| 58 : m_layer(layer) | |
| 59 , m_normalFlowListDirty(true) | |
| 60 #if ENABLE(ASSERT) | |
| 61 , m_layerListMutationAllowed(true) | |
| 62 , m_stackingParent(0) | |
| 63 #endif | |
| 64 { | |
| 65 m_isNormalFlowOnly = shouldBeNormalFlowOnly(); | |
| 66 | |
| 67 // Non-stacking contexts should have empty z-order lists. As this is already
the case, | |
| 68 // there is no need to dirty / recompute these lists. | |
| 69 m_zOrderListsDirty = isStackingContext(); | |
| 70 } | |
| 71 | |
| 72 RenderLayerStackingNode::~RenderLayerStackingNode() | |
| 73 { | |
| 74 #if ENABLE(ASSERT) | |
| 75 if (!renderer()->documentBeingDestroyed()) { | |
| 76 ASSERT(!isInStackingParentZOrderLists()); | |
| 77 ASSERT(!isInStackingParentNormalFlowList()); | |
| 78 | |
| 79 updateStackingParentForZOrderLists(0); | |
| 80 updateStackingParentForNormalFlowList(0); | |
| 81 } | |
| 82 #endif | |
| 83 } | |
| 84 | |
| 85 // Helper for the sorting of layers by z-index. | |
| 86 static inline bool compareZIndex(RenderLayerStackingNode* first, RenderLayerStac
kingNode* second) | |
| 87 { | |
| 88 return first->zIndex() < second->zIndex(); | |
| 89 } | |
| 90 | |
| 91 RenderLayerCompositor* RenderLayerStackingNode::compositor() const | |
| 92 { | |
| 93 ASSERT(renderer()->view()); | |
| 94 return renderer()->view()->compositor(); | |
| 95 } | |
| 96 | |
| 97 void RenderLayerStackingNode::dirtyZOrderLists() | |
| 98 { | |
| 99 ASSERT(m_layerListMutationAllowed); | |
| 100 ASSERT(isStackingContext()); | |
| 101 | |
| 102 #if ENABLE(ASSERT) | |
| 103 updateStackingParentForZOrderLists(0); | |
| 104 #endif | |
| 105 | |
| 106 if (m_posZOrderList) | |
| 107 m_posZOrderList->clear(); | |
| 108 if (m_negZOrderList) | |
| 109 m_negZOrderList->clear(); | |
| 110 m_zOrderListsDirty = true; | |
| 111 | |
| 112 if (!renderer()->documentBeingDestroyed()) | |
| 113 compositor()->setNeedsCompositingUpdate(CompositingUpdateRebuildTree); | |
| 114 } | |
| 115 | |
| 116 void RenderLayerStackingNode::dirtyStackingContextZOrderLists() | |
| 117 { | |
| 118 if (RenderLayerStackingNode* stackingNode = ancestorStackingContextNode()) | |
| 119 stackingNode->dirtyZOrderLists(); | |
| 120 } | |
| 121 | |
| 122 void RenderLayerStackingNode::dirtyNormalFlowList() | |
| 123 { | |
| 124 ASSERT(m_layerListMutationAllowed); | |
| 125 | |
| 126 #if ENABLE(ASSERT) | |
| 127 updateStackingParentForNormalFlowList(0); | |
| 128 #endif | |
| 129 | |
| 130 if (m_normalFlowList) | |
| 131 m_normalFlowList->clear(); | |
| 132 m_normalFlowListDirty = true; | |
| 133 | |
| 134 if (!renderer()->documentBeingDestroyed()) | |
| 135 compositor()->setNeedsCompositingUpdate(CompositingUpdateRebuildTree); | |
| 136 } | |
| 137 | |
| 138 void RenderLayerStackingNode::rebuildZOrderLists() | |
| 139 { | |
| 140 ASSERT(m_layerListMutationAllowed); | |
| 141 ASSERT(isDirtyStackingContext()); | |
| 142 | |
| 143 for (RenderLayer* child = layer()->firstChild(); child; child = child->nextS
ibling()) { | |
| 144 if (!layer()->reflectionInfo() || layer()->reflectionInfo()->reflectionL
ayer() != child) | |
| 145 child->stackingNode()->collectLayers(m_posZOrderList, m_negZOrderLis
t); | |
| 146 } | |
| 147 | |
| 148 // Sort the two lists. | |
| 149 if (m_posZOrderList) | |
| 150 std::stable_sort(m_posZOrderList->begin(), m_posZOrderList->end(), compa
reZIndex); | |
| 151 | |
| 152 if (m_negZOrderList) | |
| 153 std::stable_sort(m_negZOrderList->begin(), m_negZOrderList->end(), compa
reZIndex); | |
| 154 | |
| 155 // Append layers for top layer elements after normal layer collection, to en
sure they are on top regardless of z-indexes. | |
| 156 // The renderers of top layer elements are children of the view, sorted in t
op layer stacking order. | |
| 157 if (layer()->isRootLayer()) { | |
| 158 RenderView* view = renderer()->view(); | |
| 159 for (RenderObject* child = view->firstChild(); child; child = child->nex
tSibling()) { | |
| 160 Element* childElement = (child->node() && child->node()->isElementNo
de()) ? toElement(child->node()) : 0; | |
| 161 if (childElement && childElement->isInTopLayer()) { | |
| 162 RenderLayer* layer = toRenderLayerModelObject(child)->layer(); | |
| 163 // Create the buffer if it doesn't exist yet. | |
| 164 if (!m_posZOrderList) | |
| 165 m_posZOrderList = adoptPtr(new Vector<RenderLayerStackingNod
e*>); | |
| 166 m_posZOrderList->append(layer->stackingNode()); | |
| 167 } | |
| 168 } | |
| 169 } | |
| 170 | |
| 171 #if ENABLE(ASSERT) | |
| 172 updateStackingParentForZOrderLists(this); | |
| 173 #endif | |
| 174 | |
| 175 m_zOrderListsDirty = false; | |
| 176 } | |
| 177 | |
| 178 void RenderLayerStackingNode::updateNormalFlowList() | |
| 179 { | |
| 180 if (!m_normalFlowListDirty) | |
| 181 return; | |
| 182 | |
| 183 ASSERT(m_layerListMutationAllowed); | |
| 184 | |
| 185 for (RenderLayer* child = layer()->firstChild(); child; child = child->nextS
ibling()) { | |
| 186 if (child->stackingNode()->isNormalFlowOnly() && (!layer()->reflectionIn
fo() || layer()->reflectionInfo()->reflectionLayer() != child)) { | |
| 187 if (!m_normalFlowList) | |
| 188 m_normalFlowList = adoptPtr(new Vector<RenderLayerStackingNode*>
); | |
| 189 m_normalFlowList->append(child->stackingNode()); | |
| 190 } | |
| 191 } | |
| 192 | |
| 193 #if ENABLE(ASSERT) | |
| 194 updateStackingParentForNormalFlowList(this); | |
| 195 #endif | |
| 196 | |
| 197 m_normalFlowListDirty = false; | |
| 198 } | |
| 199 | |
| 200 void RenderLayerStackingNode::collectLayers(OwnPtr<Vector<RenderLayerStackingNod
e*> >& posBuffer, OwnPtr<Vector<RenderLayerStackingNode*> >& negBuffer) | |
| 201 { | |
| 202 if (layer()->isInTopLayer()) | |
| 203 return; | |
| 204 | |
| 205 if (!isNormalFlowOnly()) { | |
| 206 OwnPtr<Vector<RenderLayerStackingNode*> >& buffer = (zIndex() >= 0) ? po
sBuffer : negBuffer; | |
| 207 if (!buffer) | |
| 208 buffer = adoptPtr(new Vector<RenderLayerStackingNode*>); | |
| 209 buffer->append(this); | |
| 210 } | |
| 211 | |
| 212 if (!isStackingContext()) { | |
| 213 for (RenderLayer* child = layer()->firstChild(); child; child = child->n
extSibling()) { | |
| 214 if (!layer()->reflectionInfo() || layer()->reflectionInfo()->reflect
ionLayer() != child) | |
| 215 child->stackingNode()->collectLayers(posBuffer, negBuffer); | |
| 216 } | |
| 217 } | |
| 218 } | |
| 219 | |
| 220 #if ENABLE(ASSERT) | |
| 221 bool RenderLayerStackingNode::isInStackingParentZOrderLists() const | |
| 222 { | |
| 223 if (!m_stackingParent || m_stackingParent->zOrderListsDirty()) | |
| 224 return false; | |
| 225 | |
| 226 if (m_stackingParent->posZOrderList() && m_stackingParent->posZOrderList()->
find(this) != kNotFound) | |
| 227 return true; | |
| 228 | |
| 229 if (m_stackingParent->negZOrderList() && m_stackingParent->negZOrderList()->
find(this) != kNotFound) | |
| 230 return true; | |
| 231 | |
| 232 return false; | |
| 233 } | |
| 234 | |
| 235 bool RenderLayerStackingNode::isInStackingParentNormalFlowList() const | |
| 236 { | |
| 237 if (!m_stackingParent || m_stackingParent->normalFlowListDirty()) | |
| 238 return false; | |
| 239 | |
| 240 return (m_stackingParent->normalFlowList() && m_stackingParent->normalFlowLi
st()->find(this) != kNotFound); | |
| 241 } | |
| 242 | |
| 243 void RenderLayerStackingNode::updateStackingParentForZOrderLists(RenderLayerStac
kingNode* stackingParent) | |
| 244 { | |
| 245 if (m_posZOrderList) { | |
| 246 for (size_t i = 0; i < m_posZOrderList->size(); ++i) | |
| 247 m_posZOrderList->at(i)->setStackingParent(stackingParent); | |
| 248 } | |
| 249 | |
| 250 if (m_negZOrderList) { | |
| 251 for (size_t i = 0; i < m_negZOrderList->size(); ++i) | |
| 252 m_negZOrderList->at(i)->setStackingParent(stackingParent); | |
| 253 } | |
| 254 } | |
| 255 | |
| 256 void RenderLayerStackingNode::updateStackingParentForNormalFlowList(RenderLayerS
tackingNode* stackingParent) | |
| 257 { | |
| 258 if (m_normalFlowList) { | |
| 259 for (size_t i = 0; i < m_normalFlowList->size(); ++i) | |
| 260 m_normalFlowList->at(i)->setStackingParent(stackingParent); | |
| 261 } | |
| 262 } | |
| 263 #endif | |
| 264 | |
| 265 void RenderLayerStackingNode::updateLayerListsIfNeeded() | |
| 266 { | |
| 267 updateZOrderLists(); | |
| 268 updateNormalFlowList(); | |
| 269 | |
| 270 if (!layer()->reflectionInfo()) | |
| 271 return; | |
| 272 | |
| 273 RenderLayer* reflectionLayer = layer()->reflectionInfo()->reflectionLayer(); | |
| 274 reflectionLayer->stackingNode()->updateZOrderLists(); | |
| 275 reflectionLayer->stackingNode()->updateNormalFlowList(); | |
| 276 } | |
| 277 | |
| 278 void RenderLayerStackingNode::updateStackingNodesAfterStyleChange(const RenderSt
yle* oldStyle) | |
| 279 { | |
| 280 bool wasStackingContext = oldStyle ? !oldStyle->hasAutoZIndex() : false; | |
| 281 int oldZIndex = oldStyle ? oldStyle->zIndex() : 0; | |
| 282 | |
| 283 bool isStackingContext = this->isStackingContext(); | |
| 284 if (isStackingContext == wasStackingContext && oldZIndex == zIndex()) | |
| 285 return; | |
| 286 | |
| 287 dirtyStackingContextZOrderLists(); | |
| 288 | |
| 289 if (isStackingContext) | |
| 290 dirtyZOrderLists(); | |
| 291 else | |
| 292 clearZOrderLists(); | |
| 293 } | |
| 294 | |
| 295 // FIXME: Rename shouldBeNormalFlowOnly to something more accurate now that CSS | |
| 296 // 2.1 defines the term "normal flow". | |
| 297 bool RenderLayerStackingNode::shouldBeNormalFlowOnly() const | |
| 298 { | |
| 299 return !isStackingContext() && !renderer()->isPositioned(); | |
| 300 } | |
| 301 | |
| 302 void RenderLayerStackingNode::updateIsNormalFlowOnly() | |
| 303 { | |
| 304 bool isNormalFlowOnly = shouldBeNormalFlowOnly(); | |
| 305 if (isNormalFlowOnly == this->isNormalFlowOnly()) | |
| 306 return; | |
| 307 | |
| 308 m_isNormalFlowOnly = isNormalFlowOnly; | |
| 309 if (RenderLayer* p = layer()->parent()) | |
| 310 p->stackingNode()->dirtyNormalFlowList(); | |
| 311 dirtyStackingContextZOrderLists(); | |
| 312 } | |
| 313 | |
| 314 RenderLayerStackingNode* RenderLayerStackingNode::ancestorStackingContextNode()
const | |
| 315 { | |
| 316 for (RenderLayer* ancestor = layer()->parent(); ancestor; ancestor = ancesto
r->parent()) { | |
| 317 RenderLayerStackingNode* stackingNode = ancestor->stackingNode(); | |
| 318 if (stackingNode->isStackingContext()) | |
| 319 return stackingNode; | |
| 320 } | |
| 321 return 0; | |
| 322 } | |
| 323 | |
| 324 RenderLayerModelObject* RenderLayerStackingNode::renderer() const | |
| 325 { | |
| 326 return m_layer->renderer(); | |
| 327 } | |
| 328 | |
| 329 } // namespace blink | |
| OLD | NEW |