blob: 32d2681fea6b9042c17366bdf35f5b4a1d17d74b [file] [log] [blame]
/*
* Copyright (C) 2008 Apple Inc. 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 APPLE INC. ``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 APPLE INC. OR
* CONTRIBUTORS 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.
*/
#ifndef RangeBoundaryPoint_h
#define RangeBoundaryPoint_h
#include "core/dom/Node.h"
#include "core/dom/NodeTraversal.h"
#include "core/editing/Position.h"
namespace blink {
class RangeBoundaryPoint {
DISALLOW_NEW();
public:
explicit RangeBoundaryPoint(Node* container);
explicit RangeBoundaryPoint(const RangeBoundaryPoint&);
bool isConnected() const;
const Position toPosition() const;
Node* container() const;
unsigned offset() const;
Node* childBefore() const;
void clear();
void set(Node* container, unsigned offset, Node* childBefore);
void setOffset(unsigned);
void setToBeforeChild(Node&);
void setToStartOfNode(Node&);
void setToEndOfNode(Node&);
void childBeforeWillBeRemoved();
void invalidateOffset();
void markValid() const;
DEFINE_INLINE_TRACE() {
visitor->trace(m_containerNode);
visitor->trace(m_childBeforeBoundary);
}
private:
uint64_t domTreeVersion() const;
void ensureOffsetIsValid() const;
bool isOffsetValid() const;
static const unsigned invalidOffset = static_cast<unsigned>(-1);
Member<Node> m_containerNode;
Member<Node> m_childBeforeBoundary;
mutable uint64_t m_domTreeVersion;
mutable unsigned m_offsetInContainer;
};
inline RangeBoundaryPoint::RangeBoundaryPoint(Node* container)
: m_containerNode(container),
m_childBeforeBoundary(nullptr),
m_domTreeVersion(domTreeVersion()),
m_offsetInContainer(0) {}
inline RangeBoundaryPoint::RangeBoundaryPoint(const RangeBoundaryPoint& other)
: m_containerNode(other.container()),
m_childBeforeBoundary(other.childBefore()),
m_domTreeVersion(other.m_domTreeVersion),
m_offsetInContainer(other.offset()) {}
inline Node* RangeBoundaryPoint::container() const {
return m_containerNode.get();
}
inline Node* RangeBoundaryPoint::childBefore() const {
return m_childBeforeBoundary.get();
}
inline uint64_t RangeBoundaryPoint::domTreeVersion() const {
return m_containerNode->document().domTreeVersion();
}
inline void RangeBoundaryPoint::ensureOffsetIsValid() const {
if (isOffsetValid())
return;
DCHECK(!m_containerNode->isCharacterDataNode());
markValid();
if (!m_childBeforeBoundary) {
m_offsetInContainer = 0;
return;
}
m_offsetInContainer = m_childBeforeBoundary->nodeIndex() + 1;
}
inline bool RangeBoundaryPoint::isConnected() const {
return m_containerNode && m_containerNode->isConnected();
}
inline bool RangeBoundaryPoint::isOffsetValid() const {
if (m_offsetInContainer == invalidOffset) {
DCHECK(!m_containerNode->isTextNode());
return false;
}
return domTreeVersion() == m_domTreeVersion ||
m_containerNode->isCharacterDataNode();
}
inline const Position RangeBoundaryPoint::toPosition() const {
ensureOffsetIsValid();
// TODO(yosin): We should return |Position::beforeAnchor| when
// |m_containerNode| isn't |Text| node.
return Position::editingPositionOf(m_containerNode.get(),
m_offsetInContainer);
}
inline unsigned RangeBoundaryPoint::offset() const {
ensureOffsetIsValid();
return m_offsetInContainer;
}
inline void RangeBoundaryPoint::clear() {
m_containerNode.clear();
m_offsetInContainer = 0;
m_childBeforeBoundary = nullptr;
m_domTreeVersion = 0;
}
inline void RangeBoundaryPoint::set(Node* container,
unsigned offset,
Node* childBefore) {
DCHECK(container);
DCHECK_GE(offset, 0u);
DCHECK_EQ(childBefore,
offset ? NodeTraversal::childAt(*container, offset - 1) : 0);
m_containerNode = container;
m_offsetInContainer = offset;
m_childBeforeBoundary = childBefore;
markValid();
}
inline void RangeBoundaryPoint::setOffset(unsigned offset) {
DCHECK(m_containerNode);
DCHECK(m_containerNode->isCharacterDataNode());
DCHECK_GE(m_offsetInContainer, 0u);
DCHECK(!m_childBeforeBoundary);
m_offsetInContainer = offset;
markValid();
}
inline void RangeBoundaryPoint::setToBeforeChild(Node& child) {
DCHECK(child.parentNode());
m_childBeforeBoundary = child.previousSibling();
m_containerNode = child.parentNode();
m_offsetInContainer = m_childBeforeBoundary ? invalidOffset : 0;
markValid();
}
inline void RangeBoundaryPoint::setToStartOfNode(Node& container) {
m_containerNode = &container;
m_offsetInContainer = 0;
m_childBeforeBoundary = nullptr;
markValid();
}
inline void RangeBoundaryPoint::setToEndOfNode(Node& container) {
m_containerNode = &container;
if (m_containerNode->isCharacterDataNode()) {
m_offsetInContainer = m_containerNode->maxCharacterOffset();
m_childBeforeBoundary = nullptr;
} else {
m_childBeforeBoundary = m_containerNode->lastChild();
m_offsetInContainer = m_childBeforeBoundary ? invalidOffset : 0;
}
markValid();
}
inline void RangeBoundaryPoint::childBeforeWillBeRemoved() {
m_childBeforeBoundary = m_childBeforeBoundary->previousSibling();
if (!isOffsetValid())
return;
DCHECK_GT(m_offsetInContainer, 0u);
if (!m_childBeforeBoundary)
m_offsetInContainer = 0;
else if (m_offsetInContainer > 0)
--m_offsetInContainer;
markValid();
}
inline void RangeBoundaryPoint::invalidateOffset() {
m_offsetInContainer = invalidOffset;
}
inline void RangeBoundaryPoint::markValid() const {
m_domTreeVersion = domTreeVersion();
}
inline bool operator==(const RangeBoundaryPoint& a,
const RangeBoundaryPoint& b) {
if (a.container() != b.container())
return false;
if (a.childBefore() || b.childBefore()) {
if (a.childBefore() != b.childBefore())
return false;
} else {
if (a.offset() != b.offset())
return false;
}
return true;
}
} // namespace blink
#endif