blob: 0437f885d3c1484a22b8659ba796bbe67a2fa6ad [file] [log] [blame]
/*
* Copyright (C) 2005 Allan Sandfeld Jensen (kde@carewolf.com)
* Copyright (C) 2006, 2007 Apple Inc. All rights reserved.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 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
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public License
* along with this library; see the file COPYING.LIB. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*
*/
#include "core/layout/CounterNode.h"
#include "core/layout/LayoutCounter.h"
#ifndef NDEBUG
#include <stdio.h>
#endif
namespace blink {
CounterNode::CounterNode(LayoutObject& o, bool hasResetType, int value)
: m_hasResetType(hasResetType),
m_value(value),
m_countInParent(0),
m_owner(o),
m_rootLayoutObject(nullptr),
m_parent(nullptr),
m_previousSibling(nullptr),
m_nextSibling(nullptr),
m_firstChild(nullptr),
m_lastChild(nullptr) {}
CounterNode::~CounterNode() {
// Ideally this would be an assert and this would never be reached. In reality this happens a lot
// so we need to handle these cases. The node is still connected to the tree so we need to detach it.
if (m_parent || m_previousSibling || m_nextSibling || m_firstChild ||
m_lastChild) {
CounterNode* oldParent = nullptr;
CounterNode* oldPreviousSibling = nullptr;
// Instead of calling removeChild() we do this safely as the tree is likely broken if we get here.
if (m_parent) {
if (m_parent->m_firstChild == this)
m_parent->m_firstChild = m_nextSibling;
if (m_parent->m_lastChild == this)
m_parent->m_lastChild = m_previousSibling;
oldParent = m_parent;
m_parent = nullptr;
}
if (m_previousSibling) {
if (m_previousSibling->m_nextSibling == this)
m_previousSibling->m_nextSibling = m_nextSibling;
oldPreviousSibling = m_previousSibling;
m_previousSibling = nullptr;
}
if (m_nextSibling) {
if (m_nextSibling->m_previousSibling == this)
m_nextSibling->m_previousSibling = oldPreviousSibling;
m_nextSibling = nullptr;
}
if (m_firstChild) {
// The node's children are reparented to the old parent.
for (CounterNode* child = m_firstChild; child;) {
CounterNode* nextChild = child->m_nextSibling;
CounterNode* nextSibling = nullptr;
child->m_parent = oldParent;
if (oldPreviousSibling) {
nextSibling = oldPreviousSibling->m_nextSibling;
child->m_previousSibling = oldPreviousSibling;
oldPreviousSibling->m_nextSibling = child;
child->m_nextSibling = nextSibling;
nextSibling->m_previousSibling = child;
oldPreviousSibling = child;
}
child = nextChild;
}
}
}
resetLayoutObjects();
}
PassRefPtr<CounterNode> CounterNode::create(LayoutObject& owner,
bool hasResetType,
int value) {
return adoptRef(new CounterNode(owner, hasResetType, value));
}
CounterNode* CounterNode::nextInPreOrderAfterChildren(
const CounterNode* stayWithin) const {
if (this == stayWithin)
return nullptr;
const CounterNode* current = this;
CounterNode* next = current->m_nextSibling;
for (; !next; next = current->m_nextSibling) {
current = current->m_parent;
if (!current || current == stayWithin)
return nullptr;
}
return next;
}
CounterNode* CounterNode::nextInPreOrder(const CounterNode* stayWithin) const {
if (CounterNode* next = m_firstChild)
return next;
return nextInPreOrderAfterChildren(stayWithin);
}
CounterNode* CounterNode::lastDescendant() const {
CounterNode* last = m_lastChild;
if (!last)
return nullptr;
while (CounterNode* lastChild = last->m_lastChild)
last = lastChild;
return last;
}
CounterNode* CounterNode::previousInPreOrder() const {
CounterNode* previous = m_previousSibling;
if (!previous)
return m_parent;
while (CounterNode* lastChild = previous->m_lastChild)
previous = lastChild;
return previous;
}
int CounterNode::computeCountInParent() const {
int increment = actsAsReset() ? 0 : m_value;
if (m_previousSibling)
return m_previousSibling->m_countInParent + increment;
ASSERT(m_parent->m_firstChild == this);
return m_parent->m_value + increment;
}
void CounterNode::addLayoutObject(LayoutCounter* value) {
if (!value) {
ASSERT_NOT_REACHED();
return;
}
if (value->m_counterNode) {
ASSERT_NOT_REACHED();
value->m_counterNode->removeLayoutObject(value);
}
ASSERT(!value->m_nextForSameCounter);
for (LayoutCounter* iterator = m_rootLayoutObject; iterator;
iterator = iterator->m_nextForSameCounter) {
if (iterator == value) {
ASSERT_NOT_REACHED();
return;
}
}
value->m_nextForSameCounter = m_rootLayoutObject;
m_rootLayoutObject = value;
if (value->m_counterNode != this) {
if (value->m_counterNode) {
ASSERT_NOT_REACHED();
value->m_counterNode->removeLayoutObject(value);
}
value->m_counterNode = this;
}
}
void CounterNode::removeLayoutObject(LayoutCounter* value) {
if (!value) {
ASSERT_NOT_REACHED();
return;
}
if (value->m_counterNode && value->m_counterNode != this) {
ASSERT_NOT_REACHED();
value->m_counterNode->removeLayoutObject(value);
}
LayoutCounter* previous = nullptr;
for (LayoutCounter* iterator = m_rootLayoutObject; iterator;
iterator = iterator->m_nextForSameCounter) {
if (iterator == value) {
if (previous)
previous->m_nextForSameCounter = value->m_nextForSameCounter;
else
m_rootLayoutObject = value->m_nextForSameCounter;
value->m_nextForSameCounter = nullptr;
value->m_counterNode = nullptr;
return;
}
previous = iterator;
}
ASSERT_NOT_REACHED();
}
void CounterNode::resetLayoutObjects() {
while (m_rootLayoutObject)
m_rootLayoutObject
->invalidate(); // This makes m_rootLayoutObject point to the next layoutObject if any since it disconnects the m_rootLayoutObject from this.
}
void CounterNode::resetThisAndDescendantsLayoutObjects() {
CounterNode* node = this;
do {
node->resetLayoutObjects();
node = node->nextInPreOrder(this);
} while (node);
}
void CounterNode::recount() {
for (CounterNode* node = this; node; node = node->m_nextSibling) {
int oldCount = node->m_countInParent;
int newCount = node->computeCountInParent();
if (oldCount == newCount)
break;
node->m_countInParent = newCount;
node->resetThisAndDescendantsLayoutObjects();
}
}
void CounterNode::insertAfter(CounterNode* newChild,
CounterNode* refChild,
const AtomicString& identifier) {
ASSERT(newChild);
ASSERT(!newChild->m_parent);
ASSERT(!newChild->m_previousSibling);
ASSERT(!newChild->m_nextSibling);
// If the refChild is not our child we can not complete the request. This hardens against bugs in LayoutCounter.
// When layoutObjects are reparented it may request that we insert counter nodes improperly.
if (refChild && refChild->m_parent != this)
return;
if (newChild->m_hasResetType) {
while (m_lastChild != refChild)
LayoutCounter::destroyCounterNode(m_lastChild->owner(), identifier);
}
CounterNode* next;
if (refChild) {
next = refChild->m_nextSibling;
refChild->m_nextSibling = newChild;
} else {
next = m_firstChild;
m_firstChild = newChild;
}
newChild->m_parent = this;
newChild->m_previousSibling = refChild;
if (next) {
ASSERT(next->m_previousSibling == refChild);
next->m_previousSibling = newChild;
newChild->m_nextSibling = next;
} else {
ASSERT(m_lastChild == refChild);
m_lastChild = newChild;
}
if (!newChild->m_firstChild || newChild->m_hasResetType) {
newChild->m_countInParent = newChild->computeCountInParent();
newChild->resetThisAndDescendantsLayoutObjects();
if (next)
next->recount();
return;
}
// The code below handles the case when a formerly root increment counter is loosing its root position
// and therefore its children become next siblings.
CounterNode* last = newChild->m_lastChild;
CounterNode* first = newChild->m_firstChild;
if (first) {
ASSERT(last);
newChild->m_nextSibling = first;
if (m_lastChild == newChild)
m_lastChild = last;
first->m_previousSibling = newChild;
// The case when the original next sibling of the inserted node becomes a child of
// one of the former children of the inserted node is not handled as it is believed
// to be impossible since:
// 1. if the increment counter node lost it's root position as a result of another
// counter node being created, it will be inserted as the last child so next is null.
// 2. if the increment counter node lost it's root position as a result of a layoutObject being
// inserted into the document's layout tree, all its former children counters are attached
// to children of the inserted layoutObject and hence cannot be in scope for counter nodes
// attached to layoutObjects that were already in the document's layout tree.
last->m_nextSibling = next;
if (next) {
ASSERT(next->m_previousSibling == newChild);
next->m_previousSibling = last;
} else {
m_lastChild = last;
}
for (next = first;; next = next->m_nextSibling) {
next->m_parent = this;
if (last == next)
break;
}
}
newChild->m_firstChild = nullptr;
newChild->m_lastChild = nullptr;
newChild->m_countInParent = newChild->computeCountInParent();
newChild->resetLayoutObjects();
first->recount();
}
void CounterNode::removeChild(CounterNode* oldChild) {
ASSERT(oldChild);
ASSERT(!oldChild->m_firstChild);
ASSERT(!oldChild->m_lastChild);
CounterNode* next = oldChild->m_nextSibling;
CounterNode* previous = oldChild->m_previousSibling;
oldChild->m_nextSibling = nullptr;
oldChild->m_previousSibling = nullptr;
oldChild->m_parent = nullptr;
if (previous) {
previous->m_nextSibling = next;
} else {
ASSERT(m_firstChild == oldChild);
m_firstChild = next;
}
if (next) {
next->m_previousSibling = previous;
} else {
ASSERT(m_lastChild == oldChild);
m_lastChild = previous;
}
if (next)
next->recount();
}
#ifndef NDEBUG
static void showTreeAndMark(const CounterNode* node) {
const CounterNode* root = node;
while (root->parent())
root = root->parent();
for (const CounterNode* current = root; current;
current = current->nextInPreOrder()) {
fprintf(stderr, "%c", (current == node) ? '*' : ' ');
for (const CounterNode* parent = current; parent && parent != root;
parent = parent->parent())
fprintf(stderr, " ");
fprintf(stderr, "%p %s: %d %d P:%p PS:%p NS:%p R:%p\n", current,
current->actsAsReset() ? "reset____" : "increment",
current->value(), current->countInParent(), current->parent(),
current->previousSibling(), current->nextSibling(),
&current->owner());
}
fflush(stderr);
}
#endif
} // namespace blink
#ifndef NDEBUG
void showCounterTree(const blink::CounterNode* counter) {
if (counter)
showTreeAndMark(counter);
else
fprintf(stderr, "Cannot showCounterTree for (nil).\n");
}
#endif