blob: 7e39bb99a54ccbbbb6ea46f99d797d6d0840cfea [file] [log] [blame]
/*
* Copyright (C) 2006, 2007, 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 COMPUTER, 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 COMPUTER, 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 Editor_h
#define Editor_h
#include "core/CoreExport.h"
#include "core/clipboard/DataTransferAccessPolicy.h"
#include "core/editing/EditingBehavior.h"
#include "core/editing/EditingStyle.h"
#include "core/editing/EphemeralRange.h"
#include "core/editing/FindOptions.h"
#include "core/editing/FrameSelection.h"
#include "core/editing/VisibleSelection.h"
#include "core/editing/WritingDirection.h"
#include "core/editing/iterators/TextIterator.h"
#include "core/editing/markers/DocumentMarker.h"
#include "core/events/InputEvent.h"
#include "platform/PasteMode.h"
#include "platform/heap/Handle.h"
#include <memory>
namespace blink {
class CompositeEditCommand;
class DragData;
class EditorClient;
class EditorInternalCommand;
class LocalFrame;
class HitTestResult;
class KillRing;
class Pasteboard;
class SpellChecker;
class StylePropertySet;
class TextEvent;
class TypingCommand;
class UndoStack;
class UndoStep;
enum class DeleteDirection;
enum class DeleteMode { Simple, Smart };
enum class InsertMode { Simple, Smart };
enum class DragSourceType { HTMLSource, PlainTextSource };
enum EditorCommandSource { CommandFromMenuOrKeyBinding, CommandFromDOM };
enum EditorParagraphSeparator {
EditorParagraphSeparatorIsDiv,
EditorParagraphSeparatorIsP
};
class CORE_EXPORT Editor final : public GarbageCollectedFinalized<Editor> {
WTF_MAKE_NONCOPYABLE(Editor);
public:
static Editor* create(LocalFrame&);
~Editor();
EditorClient& client() const;
CompositeEditCommand* lastEditCommand() { return m_lastEditCommand.get(); }
TypingCommand* lastTypingCommandIfStillOpenForTyping() const;
void handleKeyboardEvent(KeyboardEvent*);
bool handleTextEvent(TextEvent*);
bool canEdit() const;
bool canEditRichly() const;
bool canDHTMLCut();
bool canDHTMLCopy();
bool canCut() const;
bool canCopy() const;
bool canPaste() const;
bool canDelete() const;
bool canSmartCopyOrDelete() const;
void cut(EditorCommandSource);
void copy();
void paste(EditorCommandSource);
void pasteAsPlainText(EditorCommandSource);
void performDelete();
static void countEvent(ExecutionContext*, const Event*);
void copyImage(const HitTestResult&);
void transpose();
void respondToChangedContents(const Position&);
bool selectionStartHasStyle(CSSPropertyID, const String& value) const;
TriState selectionHasStyle(CSSPropertyID, const String& value) const;
String selectionStartCSSPropertyValue(CSSPropertyID);
void removeFormattingAndStyle();
void registerCommandGroup(CompositeEditCommand* commandGroupWrapper);
bool deleteWithDirection(DeleteDirection,
TextGranularity,
bool killRing,
bool isTypingAction);
void deleteSelectionWithSmartDelete(
DeleteMode,
InputEvent::InputType,
const Position& referenceMovePosition = Position());
void applyStyle(StylePropertySet*, InputEvent::InputType);
void applyParagraphStyle(StylePropertySet*, InputEvent::InputType);
void applyStyleToSelection(StylePropertySet*, InputEvent::InputType);
void applyParagraphStyleToSelection(StylePropertySet*, InputEvent::InputType);
void appliedEditing(CompositeEditCommand*);
void unappliedEditing(UndoStep*);
void reappliedEditing(UndoStep*);
void setShouldStyleWithCSS(bool flag) { m_shouldStyleWithCSS = flag; }
bool shouldStyleWithCSS() const { return m_shouldStyleWithCSS; }
class CORE_EXPORT Command {
STACK_ALLOCATED();
public:
Command();
Command(const EditorInternalCommand*, EditorCommandSource, LocalFrame*);
bool execute(const String& parameter = String(),
Event* triggeringEvent = nullptr) const;
bool execute(Event* triggeringEvent) const;
bool isSupported() const;
bool isEnabled(Event* triggeringEvent = nullptr) const;
TriState state(Event* triggeringEvent = nullptr) const;
String value(Event* triggeringEvent = nullptr) const;
bool isTextInsertion() const;
// Returns 0 if this Command is not supported.
int idForHistogram() const;
private:
LocalFrame& frame() const {
DCHECK(m_frame);
return *m_frame;
}
// Returns target ranges for the command, currently only supports delete
// related commands. Used by InputEvent.
const StaticRangeVector* getTargetRanges() const;
const EditorInternalCommand* m_command;
EditorCommandSource m_source;
Member<LocalFrame> m_frame;
};
Command createCommand(
const String&
commandName); // Command source is CommandFromMenuOrKeyBinding.
Command createCommand(const String& commandName, EditorCommandSource);
// |Editor::executeCommand| is implementation of |WebFrame::executeCommand|
// rather than |Document::execCommand|.
bool executeCommand(const String&);
bool executeCommand(const String& commandName, const String& value);
bool insertText(const String&, KeyboardEvent* triggeringEvent);
bool insertTextWithoutSendingTextEvent(const String&,
bool selectInsertedText,
TextEvent* triggeringEvent);
bool insertLineBreak();
bool insertParagraphSeparator();
bool isOverwriteModeEnabled() const { return m_overwriteModeEnabled; }
void toggleOverwriteModeEnabled();
bool canUndo();
void undo();
bool canRedo();
void redo();
// Exposed for IdleSpellCheckCallback only.
// Supposed to be used as |const UndoStack&|.
UndoStack& undoStack() const { return *m_undoStack; }
void setBaseWritingDirection(WritingDirection);
// smartInsertDeleteEnabled and selectTrailingWhitespaceEnabled are
// mutually exclusive, meaning that enabling one will disable the other.
bool smartInsertDeleteEnabled() const;
bool isSelectTrailingWhitespaceEnabled() const;
bool preventRevealSelection() const { return m_preventRevealSelection; }
void setStartNewKillRingSequence(bool);
void clear();
VisibleSelection selectionForCommand(Event*);
KillRing& killRing() const { return *m_killRing; }
EditingBehavior behavior() const;
EphemeralRange selectedRange();
void addToKillRing(const EphemeralRange&);
void pasteAsFragment(DocumentFragment*, bool smartReplace, bool matchStyle);
void pasteAsPlainText(const String&, bool smartReplace);
Element* findEventTargetFrom(const VisibleSelection&) const;
Element* findEventTargetFromSelection() const;
bool findString(const String&, FindOptions);
Range* findStringAndScrollToVisible(const String&, Range*, FindOptions);
Range* findRangeOfString(const String& target,
const EphemeralRange& referenceRange,
FindOptions);
Range* findRangeOfString(const String& target,
const EphemeralRangeInFlatTree& referenceRange,
FindOptions);
const VisibleSelection& mark() const; // Mark, to be used as emacs uses it.
void setMark(const VisibleSelection&);
void computeAndSetTypingStyle(StylePropertySet*, InputEvent::InputType);
// |firstRectForRange| requires up-to-date layout.
IntRect firstRectForRange(const EphemeralRange&) const;
void respondToChangedSelection(const Position& oldSelectionStart,
FrameSelection::SetSelectionOptions);
bool markedTextMatchesAreHighlighted() const;
void setMarkedTextMatchesAreHighlighted(bool);
void replaceSelectionWithFragment(DocumentFragment*,
bool selectReplacement,
bool smartReplace,
bool matchStyle,
InputEvent::InputType);
void replaceSelectionWithText(const String&,
bool selectReplacement,
bool smartReplace,
InputEvent::InputType);
// Implementation of WebLocalFrameImpl::replaceSelection.
void replaceSelection(const String&);
void replaceSelectionAfterDragging(DocumentFragment*,
InsertMode,
DragSourceType);
// Return false if frame was destroyed by event handler, should stop executing
// remaining actions.
bool deleteSelectionAfterDraggingWithEvents(
Element* dragSource,
DeleteMode,
const Position& referenceMovePosition);
bool replaceSelectionAfterDraggingWithEvents(Element* dropTarget,
DragData*,
DocumentFragment*,
Range* dropCaretRange,
InsertMode,
DragSourceType);
EditorParagraphSeparator defaultParagraphSeparator() const {
return m_defaultParagraphSeparator;
}
void setDefaultParagraphSeparator(EditorParagraphSeparator separator) {
m_defaultParagraphSeparator = separator;
}
static void tidyUpHTMLStructure(Document&);
class RevealSelectionScope {
WTF_MAKE_NONCOPYABLE(RevealSelectionScope);
DISALLOW_NEW_EXCEPT_PLACEMENT_NEW();
public:
explicit RevealSelectionScope(Editor*);
~RevealSelectionScope();
DECLARE_TRACE();
private:
Member<Editor> m_editor;
};
friend class RevealSelectionScope;
EditingStyle* typingStyle() const;
void setTypingStyle(EditingStyle*);
void clearTypingStyle();
DECLARE_TRACE();
private:
Member<LocalFrame> m_frame;
Member<CompositeEditCommand> m_lastEditCommand;
const Member<UndoStack> m_undoStack;
int m_preventRevealSelection;
bool m_shouldStartNewKillRingSequence;
bool m_shouldStyleWithCSS;
const std::unique_ptr<KillRing> m_killRing;
VisibleSelection m_mark;
bool m_areMarkedTextMatchesHighlighted;
EditorParagraphSeparator m_defaultParagraphSeparator;
bool m_overwriteModeEnabled;
Member<EditingStyle> m_typingStyle;
explicit Editor(LocalFrame&);
LocalFrame& frame() const {
DCHECK(m_frame);
return *m_frame;
}
bool canDeleteRange(const EphemeralRange&) const;
bool tryDHTMLCopy();
bool tryDHTMLCut();
bool tryDHTMLPaste(PasteMode);
bool canSmartReplaceWithPasteboard(Pasteboard*);
void pasteAsPlainTextWithPasteboard(Pasteboard*);
void pasteWithPasteboard(Pasteboard*);
void writeSelectionToPasteboard();
bool dispatchCPPEvent(const AtomicString&,
DataTransferAccessPolicy,
PasteMode = AllMimeTypes);
void revealSelectionAfterEditingOperation(
const ScrollAlignment& = ScrollAlignment::alignCenterIfNeeded,
RevealExtentOption = DoNotRevealExtent);
void changeSelectionAfterCommand(const SelectionInDOMTree&,
FrameSelection::SetSelectionOptions);
SpellChecker& spellChecker() const;
bool handleEditingKeyboardEvent(KeyboardEvent*);
};
inline void Editor::setStartNewKillRingSequence(bool flag) {
m_shouldStartNewKillRingSequence = flag;
}
inline const VisibleSelection& Editor::mark() const {
return m_mark;
}
inline void Editor::setMark(const VisibleSelection& selection) {
m_mark = selection;
}
inline bool Editor::markedTextMatchesAreHighlighted() const {
return m_areMarkedTextMatchesHighlighted;
}
inline EditingStyle* Editor::typingStyle() const {
return m_typingStyle.get();
}
inline void Editor::clearTypingStyle() {
m_typingStyle.clear();
}
inline void Editor::setTypingStyle(EditingStyle* style) {
m_typingStyle = style;
}
} // namespace blink
#endif // Editor_h