mirror of
https://github.com/flutter/flutter.git
synced 2026-02-20 02:29:02 +08:00
233 lines
12 KiB
C++
233 lines
12 KiB
C++
/*
|
|
* Copyright (C) 2006, 2008 Apple Inc. All rights reserved.
|
|
* Copyright (C) 2010 Google 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 THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "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 THE COPYRIGHT
|
|
* OWNER 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.
|
|
*/
|
|
|
|
#include "sky/engine/core/editing/ApplyBlockElementCommand.h"
|
|
|
|
#include "gen/sky/core/HTMLNames.h"
|
|
#include "sky/engine/bindings/exception_state.h"
|
|
#include "sky/engine/core/dom/NodeRenderStyle.h"
|
|
#include "sky/engine/core/dom/Text.h"
|
|
#include "sky/engine/core/editing/VisiblePosition.h"
|
|
#include "sky/engine/core/editing/VisibleUnits.h"
|
|
#include "sky/engine/core/editing/htmlediting.h"
|
|
#include "sky/engine/core/html/HTMLElement.h"
|
|
#include "sky/engine/core/rendering/RenderObject.h"
|
|
#include "sky/engine/core/rendering/style/RenderStyle.h"
|
|
|
|
namespace blink {
|
|
|
|
ApplyBlockElementCommand::ApplyBlockElementCommand(Document& document, const QualifiedName& tagName, const AtomicString& inlineStyle)
|
|
: CompositeEditCommand(document)
|
|
, m_tagName(tagName)
|
|
, m_inlineStyle(inlineStyle)
|
|
{
|
|
}
|
|
|
|
ApplyBlockElementCommand::ApplyBlockElementCommand(Document& document, const QualifiedName& tagName)
|
|
: CompositeEditCommand(document)
|
|
, m_tagName(tagName)
|
|
{
|
|
}
|
|
|
|
void ApplyBlockElementCommand::doApply()
|
|
{
|
|
if (!endingSelection().rootEditableElement())
|
|
return;
|
|
|
|
VisiblePosition visibleEnd = endingSelection().visibleEnd();
|
|
VisiblePosition visibleStart = endingSelection().visibleStart();
|
|
if (visibleStart.isNull() || visibleStart.isOrphan() || visibleEnd.isNull() || visibleEnd.isOrphan())
|
|
return;
|
|
|
|
// When a selection ends at the start of a paragraph, we rarely paint
|
|
// the selection gap before that paragraph, because there often is no gap.
|
|
// In a case like this, it's not obvious to the user that the selection
|
|
// ends "inside" that paragraph, so it would be confusing if Indent/Outdent
|
|
// operated on that paragraph.
|
|
// FIXME: We paint the gap before some paragraphs that are indented with left
|
|
// margin/padding, but not others. We should make the gap painting more consistent and
|
|
// then use a left margin/padding rule here.
|
|
if (visibleEnd != visibleStart && isStartOfParagraph(visibleEnd)) {
|
|
VisibleSelection newSelection(visibleStart, visibleEnd.previous(CannotCrossEditingBoundary), endingSelection().isDirectional());
|
|
if (newSelection.isNone())
|
|
return;
|
|
setEndingSelection(newSelection);
|
|
}
|
|
|
|
VisibleSelection selection = selectionForParagraphIteration(endingSelection());
|
|
VisiblePosition startOfSelection = selection.visibleStart();
|
|
VisiblePosition endOfSelection = selection.visibleEnd();
|
|
ASSERT(!startOfSelection.isNull());
|
|
ASSERT(!endOfSelection.isNull());
|
|
RefPtr<ContainerNode> startScope = nullptr;
|
|
int startIndex = indexForVisiblePosition(startOfSelection, startScope);
|
|
RefPtr<ContainerNode> endScope = nullptr;
|
|
int endIndex = indexForVisiblePosition(endOfSelection, endScope);
|
|
|
|
formatSelection(startOfSelection, endOfSelection);
|
|
|
|
document().updateLayout();
|
|
|
|
ASSERT(startScope == endScope);
|
|
ASSERT(startIndex >= 0);
|
|
ASSERT(startIndex <= endIndex);
|
|
if (startScope == endScope && startIndex >= 0 && startIndex <= endIndex) {
|
|
VisiblePosition start(visiblePositionForIndex(startIndex, startScope.get()));
|
|
VisiblePosition end(visiblePositionForIndex(endIndex, endScope.get()));
|
|
if (start.isNotNull() && end.isNotNull())
|
|
setEndingSelection(VisibleSelection(start, end, endingSelection().isDirectional()));
|
|
}
|
|
}
|
|
|
|
void ApplyBlockElementCommand::formatSelection(const VisiblePosition& startOfSelection, const VisiblePosition& endOfSelection)
|
|
{
|
|
}
|
|
|
|
static bool isNewLineAtPosition(const Position& position)
|
|
{
|
|
Node* textNode = position.containerNode();
|
|
int offset = position.offsetInContainerNode();
|
|
if (!textNode || !textNode->isTextNode() || offset < 0 || offset >= textNode->maxCharacterOffset())
|
|
return false;
|
|
|
|
TrackExceptionState exceptionState;
|
|
String textAtPosition = toText(textNode)->substringData(offset, 1, exceptionState);
|
|
if (exceptionState.had_exception())
|
|
return false;
|
|
|
|
return textAtPosition[0] == '\n';
|
|
}
|
|
|
|
static RenderStyle* renderStyleOfEnclosingTextNode(const Position& position)
|
|
{
|
|
if (position.anchorType() != Position::PositionIsOffsetInAnchor || !position.containerNode() || !position.containerNode()->isTextNode())
|
|
return 0;
|
|
return position.containerNode()->renderStyle();
|
|
}
|
|
|
|
void ApplyBlockElementCommand::rangeForParagraphSplittingTextNodesIfNeeded(const VisiblePosition& endOfCurrentParagraph, Position& start, Position& end)
|
|
{
|
|
start = startOfParagraph(endOfCurrentParagraph).deepEquivalent();
|
|
end = endOfCurrentParagraph.deepEquivalent();
|
|
|
|
document().updateRenderTreeIfNeeded();
|
|
|
|
bool isStartAndEndOnSameNode = false;
|
|
if (RenderStyle* startStyle = renderStyleOfEnclosingTextNode(start)) {
|
|
isStartAndEndOnSameNode = renderStyleOfEnclosingTextNode(end) && start.containerNode() == end.containerNode();
|
|
bool isStartAndEndOfLastParagraphOnSameNode = renderStyleOfEnclosingTextNode(m_endOfLastParagraph) && start.containerNode() == m_endOfLastParagraph.containerNode();
|
|
|
|
// Avoid obtanining the start of next paragraph for start
|
|
if (startStyle->preserveNewline() && isNewLineAtPosition(start) && !isNewLineAtPosition(start.previous()) && start.offsetInContainerNode() > 0)
|
|
start = startOfParagraph(VisiblePosition(end.previous())).deepEquivalent();
|
|
|
|
// If start is in the middle of a text node, split.
|
|
if (!startStyle->collapseWhiteSpace() && start.offsetInContainerNode() > 0) {
|
|
int startOffset = start.offsetInContainerNode();
|
|
Text* startText = start.containerText();
|
|
splitTextNode(startText, startOffset);
|
|
start = firstPositionInNode(startText);
|
|
if (isStartAndEndOnSameNode) {
|
|
ASSERT(end.offsetInContainerNode() >= startOffset);
|
|
end = Position(startText, end.offsetInContainerNode() - startOffset);
|
|
}
|
|
if (isStartAndEndOfLastParagraphOnSameNode) {
|
|
ASSERT(m_endOfLastParagraph.offsetInContainerNode() >= startOffset);
|
|
m_endOfLastParagraph = Position(startText, m_endOfLastParagraph.offsetInContainerNode() - startOffset);
|
|
}
|
|
}
|
|
}
|
|
|
|
document().updateRenderTreeIfNeeded();
|
|
|
|
if (RenderStyle* endStyle = renderStyleOfEnclosingTextNode(end)) {
|
|
bool isEndAndEndOfLastParagraphOnSameNode = renderStyleOfEnclosingTextNode(m_endOfLastParagraph) && end.deprecatedNode() == m_endOfLastParagraph.deprecatedNode();
|
|
// Include \n at the end of line if we're at an empty paragraph
|
|
if (endStyle->preserveNewline() && start == end && end.offsetInContainerNode() < end.containerNode()->maxCharacterOffset()) {
|
|
int endOffset = end.offsetInContainerNode();
|
|
if (!isNewLineAtPosition(end.previous()) && isNewLineAtPosition(end))
|
|
end = Position(end.containerText(), endOffset + 1);
|
|
if (isEndAndEndOfLastParagraphOnSameNode && end.offsetInContainerNode() >= m_endOfLastParagraph.offsetInContainerNode())
|
|
m_endOfLastParagraph = end;
|
|
}
|
|
|
|
// If end is in the middle of a text node, split.
|
|
if (!endStyle->collapseWhiteSpace() && end.offsetInContainerNode() && end.offsetInContainerNode() < end.containerNode()->maxCharacterOffset()) {
|
|
RefPtr<Text> endContainer = end.containerText();
|
|
splitTextNode(endContainer, end.offsetInContainerNode());
|
|
if (isStartAndEndOnSameNode)
|
|
start = firstPositionInOrBeforeNode(endContainer->previousSibling());
|
|
if (isEndAndEndOfLastParagraphOnSameNode) {
|
|
if (m_endOfLastParagraph.offsetInContainerNode() == end.offsetInContainerNode())
|
|
m_endOfLastParagraph = lastPositionInOrAfterNode(endContainer->previousSibling());
|
|
else
|
|
m_endOfLastParagraph = Position(endContainer, m_endOfLastParagraph.offsetInContainerNode() - end.offsetInContainerNode());
|
|
}
|
|
end = lastPositionInNode(endContainer->previousSibling());
|
|
}
|
|
}
|
|
}
|
|
|
|
VisiblePosition ApplyBlockElementCommand::endOfNextParagrahSplittingTextNodesIfNeeded(VisiblePosition& endOfCurrentParagraph, Position& start, Position& end)
|
|
{
|
|
VisiblePosition endOfNextParagraph = endOfParagraph(endOfCurrentParagraph.next());
|
|
Position position = endOfNextParagraph.deepEquivalent();
|
|
RenderStyle* style = renderStyleOfEnclosingTextNode(position);
|
|
if (!style)
|
|
return endOfNextParagraph;
|
|
|
|
RefPtr<Text> text = position.containerText();
|
|
if (!style->preserveNewline() || !position.offsetInContainerNode() || !isNewLineAtPosition(firstPositionInNode(text.get())))
|
|
return endOfNextParagraph;
|
|
|
|
// \n at the beginning of the text node immediately following the current paragraph is trimmed by moveParagraphWithClones.
|
|
// If endOfNextParagraph was pointing at this same text node, endOfNextParagraph will be shifted by one paragraph.
|
|
// Avoid this by splitting "\n"
|
|
splitTextNode(text, 1);
|
|
|
|
if (text == start.containerNode() && text->previousSibling() && text->previousSibling()->isTextNode()) {
|
|
ASSERT(start.offsetInContainerNode() < position.offsetInContainerNode());
|
|
start = Position(toText(text->previousSibling()), start.offsetInContainerNode());
|
|
}
|
|
if (text == end.containerNode() && text->previousSibling() && text->previousSibling()->isTextNode()) {
|
|
ASSERT(end.offsetInContainerNode() < position.offsetInContainerNode());
|
|
end = Position(toText(text->previousSibling()), end.offsetInContainerNode());
|
|
}
|
|
if (text == m_endOfLastParagraph.containerNode()) {
|
|
if (m_endOfLastParagraph.offsetInContainerNode() < position.offsetInContainerNode()) {
|
|
// We can only fix endOfLastParagraph if the previous node was still text and hasn't been modified by script.
|
|
if (text->previousSibling()->isTextNode()
|
|
&& static_cast<unsigned>(m_endOfLastParagraph.offsetInContainerNode()) <= toText(text->previousSibling())->length())
|
|
m_endOfLastParagraph = Position(toText(text->previousSibling()), m_endOfLastParagraph.offsetInContainerNode());
|
|
} else
|
|
m_endOfLastParagraph = Position(text.get(), m_endOfLastParagraph.offsetInContainerNode() - 1);
|
|
}
|
|
|
|
return VisiblePosition(Position(text.get(), position.offsetInContainerNode() - 1));
|
|
}
|
|
|
|
}
|