mirror of
https://github.com/flutter/flutter.git
synced 2026-02-20 02:29:02 +08:00
The primary goal of this change was to remove EventTarget from the
sky_engine C++ code. Since EventTarget is so core to the entire event
system that sky_engine was based on, this is a rather invasive change.
As such, it had some knock-on effects. I deleted some of the files
that were affected, and cauterised the remainder.
In many cases, a file would depend on another file that it didn't
include directly, but instead included indirectly via another file
that I deleted. When this happened, if the features that this broke
were obsolete, I sometimes just removed the features instead.
Specifically:
- removed EventTarget
- removed EventQueue, since without a target, what's a queue going to
do?
- same with EventDispatch*
- removed ExecutionContext, since it had an EventQueue and nothing
else it did was relevant to Sky anymore
- removed ActiveDOMObject, which was all about ExecutionContexts
- removed ContextLifecycleNotifier since it dependend on
ExecutionContext and ActiveDOMObject
- removed the other Lifecycle classes for consistency, and replaced
them with four booleans in the Document class
- removed some of the attributes that are no longer relevant from
IDLExtendedAttributes (ConstructorCallWith and
CallWith=ExecutionContext)
- removed the Document member on DOMDartState since we never set it to
anything but null.
- removed BuiltinSky::InstallWindow since it relied on the Document
member of DOMDartState
- removed EventHandler, EventListener, and mentions of those in
various binding scripts
- removed NewEventHandler, since we're not using that either
- removed the following interfaces from the Sky Dart API:
- EventTarget
- EventListener (since without a target, there's no way to listen)
- FocusEvent (since it's only member was an EventTarget)
- HashChangeEvent (mostly by accident, but it's defunct anyway)
- FontFace (it used ConstructorCallWith=ExecutionContext)
- changed the following interfaces of the Sky DART API:
- MediaQueryList is no longer an EventTarget
- Node is no longer an EventTarget
- Document no longer has defaultView (depended on
DOMDartState's document)
- DocumentFragment, Element, Range, and Text no longer have a
constructor (they all depended on DOMDartState's document, which
is now gone)
- Event lost its EventTarget members and path.
- Window lost its WindowTimers partial interface (it used
EventTarget and ExecutionContext a lot)
- removed numerous hacks in the bindings around features that are now
gone, like addEventListener
- removed a bunch of console logging code, since that relied on
ExecutionContext
- cauterised the wound in FontFace.cpp by removing constructors and
methods that called now-removed features
- same with MediaQuery and friends
- same with some editor features and focus-related features
- same with Document
- removed DOMTimer classes since they use ExecutionContexts
216 lines
6.8 KiB
C++
216 lines
6.8 KiB
C++
/*
|
|
* Copyright (C) 1999 Lars Knoll (knoll@kde.org)
|
|
* (C) 1999 Antti Koivisto (koivisto@kde.org)
|
|
* Copyright (C) 2003, 2004, 2005, 2006, 2007, 2008, 2009 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 "sky/engine/core/dom/Text.h"
|
|
|
|
#include "sky/engine/bindings/exception_state.h"
|
|
#include "sky/engine/bindings/exception_state_placeholder.h"
|
|
#include "sky/engine/core/css/resolver/StyleResolver.h"
|
|
#include "sky/engine/core/dom/ExceptionCode.h"
|
|
#include "sky/engine/core/dom/NodeRenderStyle.h"
|
|
#include "sky/engine/core/dom/NodeRenderingTraversal.h"
|
|
#include "sky/engine/core/dom/NodeTraversal.h"
|
|
#include "sky/engine/core/dom/RenderTreeBuilder.h"
|
|
#include "sky/engine/core/dom/shadow/ShadowRoot.h"
|
|
#include "sky/engine/core/rendering/RenderText.h"
|
|
#include "sky/engine/wtf/text/CString.h"
|
|
#include "sky/engine/wtf/text/StringBuilder.h"
|
|
|
|
namespace blink {
|
|
|
|
PassRefPtr<Text> Text::create(Document& document, const String& data)
|
|
{
|
|
return adoptRef(new Text(document, data, CreateText));
|
|
}
|
|
|
|
PassRefPtr<Text> Text::createEditingText(Document& document, const String& data)
|
|
{
|
|
return adoptRef(new Text(document, data, CreateEditingText));
|
|
}
|
|
|
|
PassRefPtr<Text> Text::splitText(unsigned offset, ExceptionState& exceptionState)
|
|
{
|
|
// IndexSizeError: Raised if the specified offset is negative or greater than
|
|
// the number of 16-bit units in data.
|
|
if (offset > length()) {
|
|
exceptionState.ThrowDOMException(IndexSizeError, "The offset " + String::number(offset) + " is larger than the Text node's length.");
|
|
return nullptr;
|
|
}
|
|
|
|
String oldStr = data();
|
|
RefPtr<Text> newText = cloneWithData(oldStr.substring(offset));
|
|
setDataWithoutUpdate(oldStr.substring(0, offset));
|
|
|
|
didModifyData(oldStr);
|
|
|
|
if (parentNode())
|
|
parentNode()->insertBefore(newText.get(), nextSibling(), exceptionState);
|
|
if (exceptionState.had_exception())
|
|
return nullptr;
|
|
|
|
if (renderer())
|
|
renderer()->setTextWithOffset(dataImpl(), 0, oldStr.length());
|
|
|
|
if (parentNode())
|
|
document().didSplitTextNode(*this);
|
|
|
|
return newText.release();
|
|
}
|
|
|
|
String Text::nodeName() const
|
|
{
|
|
return "#text";
|
|
}
|
|
|
|
Node::NodeType Text::nodeType() const
|
|
{
|
|
return TEXT_NODE;
|
|
}
|
|
|
|
PassRefPtr<Node> Text::cloneNode(bool /*deep*/)
|
|
{
|
|
return cloneWithData(data());
|
|
}
|
|
|
|
bool Text::textRendererIsNeeded(const RenderStyle& style, const RenderObject& parent)
|
|
{
|
|
if (isEditingText())
|
|
return true;
|
|
|
|
if (!length())
|
|
return false;
|
|
|
|
if (style.display() == NONE)
|
|
return false;
|
|
|
|
if (style.requiresOnlyBlockChildren())
|
|
return false;
|
|
|
|
if (!containsOnlyWhitespace())
|
|
return true;
|
|
|
|
if (!parent.canHaveWhitespaceChildren())
|
|
return false;
|
|
|
|
if (style.preserveNewline()) // pre/pre-wrap/pre-line always make renderers.
|
|
return true;
|
|
|
|
const RenderObject* prev = NodeRenderingTraversal::previousSiblingRenderer(this);
|
|
|
|
if (parent.isRenderInline()) {
|
|
// <span><div/> <div/></span>
|
|
if (prev && !prev->isInline())
|
|
return false;
|
|
} else {
|
|
if (parent.isRenderBlock() && !parent.isRenderParagraph() && (!prev || !prev->isInline()))
|
|
return false;
|
|
|
|
// Avoiding creation of a Renderer for the text node is a non-essential memory optimization.
|
|
// So to avoid blowing up on very wide DOMs, we limit the number of siblings to visit.
|
|
unsigned maxSiblingsToVisit = 50;
|
|
|
|
RenderObject* first = parent.slowFirstChild();
|
|
while (first && first->isFloatingOrOutOfFlowPositioned() && maxSiblingsToVisit--)
|
|
first = first->nextSibling();
|
|
if (!first || NodeRenderingTraversal::nextSiblingRenderer(this) == first)
|
|
// Whitespace at the start of a block just goes away. Don't even
|
|
// make a render object for this text.
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
RenderText* Text::createTextRenderer(RenderStyle* style)
|
|
{
|
|
return new RenderText(this, dataImpl());
|
|
}
|
|
|
|
void Text::attach(const AttachContext& context)
|
|
{
|
|
RenderTreeBuilder(this, context.resolvedStyle).createRendererForTextIfNeeded();
|
|
CharacterData::attach(context);
|
|
}
|
|
|
|
void Text::recalcTextStyle(StyleRecalcChange change)
|
|
{
|
|
if (RenderText* renderer = this->renderer()) {
|
|
if (change != NoChange || needsStyleRecalc())
|
|
renderer->setStyle(document().styleResolver().styleForText(this));
|
|
if (needsStyleRecalc())
|
|
renderer->setText(dataImpl());
|
|
clearNeedsStyleRecalc();
|
|
} else if (needsStyleRecalc() || needsWhitespaceRenderer()) {
|
|
reattach();
|
|
}
|
|
}
|
|
|
|
// If a whitespace node had no renderer and goes through a recalcStyle it may
|
|
// need to create one if the parent style now has white-space: pre.
|
|
bool Text::needsWhitespaceRenderer()
|
|
{
|
|
ASSERT(!renderer());
|
|
if (RenderStyle* style = parentRenderStyle())
|
|
return style->preserveNewline();
|
|
return false;
|
|
}
|
|
|
|
void Text::updateTextRenderer(unsigned offsetOfReplacedData, unsigned lengthOfReplacedData, RecalcStyleBehavior recalcStyleBehavior)
|
|
{
|
|
if (!inActiveDocument())
|
|
return;
|
|
RenderText* textRenderer = renderer();
|
|
if (!textRenderer || !textRendererIsNeeded(*textRenderer->style(), *textRenderer->parent())) {
|
|
lazyReattachIfAttached();
|
|
// FIXME: Editing should be updated so this is not neccesary.
|
|
if (recalcStyleBehavior == DeprecatedRecalcStyleImmediatlelyForEditing)
|
|
document().updateRenderTreeIfNeeded();
|
|
return;
|
|
}
|
|
textRenderer->setTextWithOffset(dataImpl(), offsetOfReplacedData, lengthOfReplacedData);
|
|
}
|
|
|
|
PassRefPtr<Text> Text::cloneWithData(const String& data)
|
|
{
|
|
return create(document(), data);
|
|
}
|
|
|
|
#ifndef NDEBUG
|
|
void Text::formatForDebugger(char *buffer, unsigned length) const
|
|
{
|
|
StringBuilder result;
|
|
String s;
|
|
|
|
result.append(nodeName());
|
|
|
|
s = data();
|
|
if (s.length() > 0) {
|
|
if (result.length())
|
|
result.appendLiteral("; ");
|
|
result.appendLiteral("value=");
|
|
result.append(s);
|
|
}
|
|
|
|
strncpy(buffer, result.toString().utf8().data(), length - 1);
|
|
}
|
|
#endif
|
|
|
|
} // namespace blink
|