1 | /* |
2 | * Copyright (C) 2006, 2007, 2008, 2013, 2014 Apple Inc. All rights reserved. |
3 | * |
4 | * Redistribution and use in source and binary forms, with or without |
5 | * modification, are permitted provided that the following conditions |
6 | * are met: |
7 | * 1. Redistributions of source code must retain the above copyright |
8 | * notice, this list of conditions and the following disclaimer. |
9 | * 2. Redistributions in binary form must reproduce the above copyright |
10 | * notice, this list of conditions and the following disclaimer in the |
11 | * documentation and/or other materials provided with the distribution. |
12 | * |
13 | * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY |
14 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
15 | * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
16 | * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR |
17 | * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
18 | * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
19 | * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
20 | * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY |
21 | * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
22 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
23 | * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
24 | */ |
25 | |
26 | #pragma once |
27 | |
28 | #include "CompositionUnderline.h" |
29 | #include "DictationAlternative.h" |
30 | #include "DocumentMarker.h" |
31 | #include "EditAction.h" |
32 | #include "EditingBehavior.h" |
33 | #include "EditingStyle.h" |
34 | #include "EditorInsertAction.h" |
35 | #include "FindOptions.h" |
36 | #include "FrameSelection.h" |
37 | #include "PasteboardWriterData.h" |
38 | #include "TextChecking.h" |
39 | #include "TextEventInputType.h" |
40 | #include "TextIteratorBehavior.h" |
41 | #include "VisibleSelection.h" |
42 | #include "WritingDirection.h" |
43 | #include <memory> |
44 | |
45 | #if PLATFORM(COCOA) |
46 | OBJC_CLASS NSAttributedString; |
47 | OBJC_CLASS NSDictionary; |
48 | OBJC_CLASS NSMutableDictionary; |
49 | #endif |
50 | |
51 | namespace PAL { |
52 | class KillRing; |
53 | } |
54 | |
55 | namespace WebCore { |
56 | |
57 | class AlternativeTextController; |
58 | class ArchiveResource; |
59 | class DataTransfer; |
60 | class CompositeEditCommand; |
61 | class CustomUndoStep; |
62 | class DeleteButtonController; |
63 | class EditCommand; |
64 | class EditCommandComposition; |
65 | class EditorClient; |
66 | class EditorInternalCommand; |
67 | class File; |
68 | class Frame; |
69 | class HTMLElement; |
70 | class HitTestResult; |
71 | class KeyboardEvent; |
72 | class KillRing; |
73 | class Pasteboard; |
74 | class PasteboardWriterData; |
75 | class SharedBuffer; |
76 | class Font; |
77 | class SpellCheckRequest; |
78 | class SpellChecker; |
79 | class StaticRange; |
80 | class StyleProperties; |
81 | class Text; |
82 | class TextCheckerClient; |
83 | class TextEvent; |
84 | |
85 | struct FontAttributes; |
86 | struct PasteboardPlainText; |
87 | struct PasteboardURL; |
88 | struct TextCheckingResult; |
89 | |
90 | #if ENABLE(ATTACHMENT_ELEMENT) |
91 | struct SerializedAttachmentData; |
92 | #endif |
93 | |
94 | enum EditorCommandSource { CommandFromMenuOrKeyBinding, CommandFromDOM, CommandFromDOMWithUserInterface }; |
95 | enum EditorParagraphSeparator { EditorParagraphSeparatorIsDiv, EditorParagraphSeparatorIsP }; |
96 | |
97 | enum class MailBlockquoteHandling { |
98 | RespectBlockquote, |
99 | IgnoreBlockquote, |
100 | }; |
101 | |
102 | #if ENABLE(ATTACHMENT_ELEMENT) |
103 | class HTMLAttachmentElement; |
104 | #endif |
105 | |
106 | enum class TemporarySelectionOption : uint8_t { |
107 | RevealSelection = 1 << 0, |
108 | DoNotSetFocus = 1 << 1, |
109 | |
110 | // Don't propagate selection changes to the client layer. |
111 | IgnoreSelectionChanges = 1 << 2, |
112 | |
113 | // Force the render tree to update selection state. Only respected on iOS. |
114 | EnableAppearanceUpdates = 1 << 3, |
115 | }; |
116 | |
117 | class TemporarySelectionChange { |
118 | public: |
119 | TemporarySelectionChange(Frame&, Optional<VisibleSelection> = WTF::nullopt, OptionSet<TemporarySelectionOption> = { }); |
120 | ~TemporarySelectionChange(); |
121 | |
122 | private: |
123 | void setSelection(const VisibleSelection&); |
124 | |
125 | Ref<Frame> m_frame; |
126 | OptionSet<TemporarySelectionOption> m_options; |
127 | bool m_wasIgnoringSelectionChanges; |
128 | #if PLATFORM(IOS_FAMILY) |
129 | bool m_appearanceUpdatesWereEnabled; |
130 | #endif |
131 | Optional<VisibleSelection> m_selectionToRestore; |
132 | }; |
133 | |
134 | class Editor { |
135 | WTF_MAKE_FAST_ALLOCATED; |
136 | public: |
137 | explicit Editor(Frame&); |
138 | ~Editor(); |
139 | |
140 | enum class PasteOption : uint8_t { |
141 | AllowPlainText = 1 << 0, |
142 | IgnoreMailBlockquote = 1 << 1, |
143 | AsQuotation = 1 << 2, |
144 | }; |
145 | |
146 | WEBCORE_EXPORT EditorClient* client() const; |
147 | WEBCORE_EXPORT TextCheckerClient* textChecker() const; |
148 | |
149 | CompositeEditCommand* lastEditCommand() { return m_lastEditCommand.get(); } |
150 | |
151 | void handleKeyboardEvent(KeyboardEvent&); |
152 | void handleInputMethodKeydown(KeyboardEvent&); |
153 | bool handleTextEvent(TextEvent&); |
154 | |
155 | WEBCORE_EXPORT bool canEdit() const; |
156 | WEBCORE_EXPORT bool canEditRichly() const; |
157 | |
158 | bool canDHTMLCut(); |
159 | bool canDHTMLCopy(); |
160 | WEBCORE_EXPORT bool canDHTMLPaste(); |
161 | bool tryDHTMLCopy(); |
162 | bool tryDHTMLCut(); |
163 | |
164 | WEBCORE_EXPORT bool canCut() const; |
165 | WEBCORE_EXPORT bool canCopy() const; |
166 | WEBCORE_EXPORT bool canPaste() const; |
167 | WEBCORE_EXPORT bool canDelete() const; |
168 | WEBCORE_EXPORT bool canSmartCopyOrDelete(); |
169 | bool shouldSmartDelete(); |
170 | |
171 | WEBCORE_EXPORT void cut(); |
172 | WEBCORE_EXPORT void copy(); |
173 | WEBCORE_EXPORT void paste(); |
174 | void paste(Pasteboard&); |
175 | WEBCORE_EXPORT void pasteAsPlainText(); |
176 | void pasteAsQuotation(); |
177 | WEBCORE_EXPORT void performDelete(); |
178 | |
179 | WEBCORE_EXPORT void copyURL(const URL&, const String& title); |
180 | void copyURL(const URL&, const String& title, Pasteboard&); |
181 | PasteboardWriterData::URLData pasteboardWriterURL(const URL&, const String& title); |
182 | #if !PLATFORM(IOS_FAMILY) |
183 | WEBCORE_EXPORT void copyImage(const HitTestResult&); |
184 | #endif |
185 | |
186 | String readPlainTextFromPasteboard(Pasteboard&); |
187 | |
188 | WEBCORE_EXPORT void indent(); |
189 | WEBCORE_EXPORT void outdent(); |
190 | void transpose(); |
191 | |
192 | bool shouldInsertFragment(DocumentFragment&, Range*, EditorInsertAction); |
193 | bool shouldInsertText(const String&, Range*, EditorInsertAction) const; |
194 | WEBCORE_EXPORT bool shouldDeleteRange(Range*) const; |
195 | bool shouldApplyStyle(StyleProperties*, Range*); |
196 | |
197 | void respondToChangedContents(const VisibleSelection& endingSelection); |
198 | |
199 | bool selectionStartHasStyle(CSSPropertyID, const String& value) const; |
200 | WEBCORE_EXPORT TriState selectionHasStyle(CSSPropertyID, const String& value) const; |
201 | String selectionStartCSSPropertyValue(CSSPropertyID); |
202 | |
203 | TriState selectionUnorderedListState() const; |
204 | TriState selectionOrderedListState() const; |
205 | WEBCORE_EXPORT RefPtr<Node> insertOrderedList(); |
206 | WEBCORE_EXPORT RefPtr<Node> insertUnorderedList(); |
207 | WEBCORE_EXPORT bool canIncreaseSelectionListLevel(); |
208 | WEBCORE_EXPORT bool canDecreaseSelectionListLevel(); |
209 | WEBCORE_EXPORT RefPtr<Node> increaseSelectionListLevel(); |
210 | WEBCORE_EXPORT RefPtr<Node> increaseSelectionListLevelOrdered(); |
211 | WEBCORE_EXPORT RefPtr<Node> increaseSelectionListLevelUnordered(); |
212 | WEBCORE_EXPORT void decreaseSelectionListLevel(); |
213 | WEBCORE_EXPORT void changeSelectionListType(); |
214 | |
215 | void removeFormattingAndStyle(); |
216 | |
217 | void clearLastEditCommand(); |
218 | #if PLATFORM(IOS_FAMILY) |
219 | WEBCORE_EXPORT void ensureLastEditCommandHasCurrentSelectionIfOpenForMoreTyping(); |
220 | #endif |
221 | |
222 | WEBCORE_EXPORT bool deleteWithDirection(SelectionDirection, TextGranularity, bool killRing, bool isTypingAction); |
223 | WEBCORE_EXPORT void deleteSelectionWithSmartDelete(bool smartDelete, EditAction = EditAction::Delete); |
224 | void clearText(); |
225 | #if PLATFORM(IOS_FAMILY) |
226 | WEBCORE_EXPORT void removeUnchangeableStyles(); |
227 | #endif |
228 | |
229 | WEBCORE_EXPORT void applyStyle(StyleProperties*, EditAction = EditAction::Unspecified); |
230 | enum class ColorFilterMode { InvertColor, UseOriginalColor }; |
231 | void applyStyle(RefPtr<EditingStyle>&&, EditAction, ColorFilterMode); |
232 | void applyParagraphStyle(StyleProperties*, EditAction = EditAction::Unspecified); |
233 | WEBCORE_EXPORT void applyStyleToSelection(StyleProperties*, EditAction); |
234 | WEBCORE_EXPORT void applyStyleToSelection(Ref<EditingStyle>&&, EditAction, ColorFilterMode); |
235 | void applyParagraphStyleToSelection(StyleProperties*, EditAction); |
236 | |
237 | // Returns whether or not we should proceed with editing. |
238 | bool willApplyEditing(CompositeEditCommand&, Vector<RefPtr<StaticRange>>&&) const; |
239 | bool willUnapplyEditing(const EditCommandComposition&) const; |
240 | bool willReapplyEditing(const EditCommandComposition&) const; |
241 | |
242 | void appliedEditing(CompositeEditCommand&); |
243 | void unappliedEditing(EditCommandComposition&); |
244 | void reappliedEditing(EditCommandComposition&); |
245 | void unappliedSpellCorrection(const VisibleSelection& selectionOfCorrected, const String& corrected, const String& correction); |
246 | |
247 | // This is off by default, since most editors want this behavior (originally matched IE but not Firefox). |
248 | void setShouldStyleWithCSS(bool flag) { m_shouldStyleWithCSS = flag; } |
249 | bool shouldStyleWithCSS() const { return m_shouldStyleWithCSS; } |
250 | |
251 | class Command { |
252 | public: |
253 | WEBCORE_EXPORT Command(); |
254 | Command(const EditorInternalCommand*, EditorCommandSource, Frame&); |
255 | |
256 | WEBCORE_EXPORT bool execute(const String& parameter = String(), Event* triggeringEvent = nullptr) const; |
257 | WEBCORE_EXPORT bool execute(Event* triggeringEvent) const; |
258 | |
259 | WEBCORE_EXPORT bool isSupported() const; |
260 | WEBCORE_EXPORT bool isEnabled(Event* triggeringEvent = nullptr) const; |
261 | |
262 | WEBCORE_EXPORT TriState state(Event* triggeringEvent = nullptr) const; |
263 | String value(Event* triggeringEvent = nullptr) const; |
264 | |
265 | WEBCORE_EXPORT bool isTextInsertion() const; |
266 | WEBCORE_EXPORT bool allowExecutionWhenDisabled() const; |
267 | |
268 | private: |
269 | const EditorInternalCommand* m_command { nullptr }; |
270 | EditorCommandSource m_source; |
271 | RefPtr<Frame> m_frame; |
272 | }; |
273 | WEBCORE_EXPORT Command command(const String& commandName); // Command source is CommandFromMenuOrKeyBinding. |
274 | Command command(const String& commandName, EditorCommandSource); |
275 | WEBCORE_EXPORT static bool commandIsSupportedFromMenuOrKeyBinding(const String& commandName); // Works without a frame. |
276 | |
277 | WEBCORE_EXPORT bool insertText(const String&, Event* triggeringEvent, TextEventInputType = TextEventInputKeyboard); |
278 | bool insertTextForConfirmedComposition(const String& text); |
279 | WEBCORE_EXPORT bool insertDictatedText(const String&, const Vector<DictationAlternative>& dictationAlternatives, Event* triggeringEvent); |
280 | bool insertTextWithoutSendingTextEvent(const String&, bool selectInsertedText, TextEvent* triggeringEvent); |
281 | bool insertLineBreak(); |
282 | bool insertParagraphSeparator(); |
283 | WEBCORE_EXPORT bool insertParagraphSeparatorInQuotedContent(); |
284 | |
285 | WEBCORE_EXPORT bool isContinuousSpellCheckingEnabled() const; |
286 | WEBCORE_EXPORT void toggleContinuousSpellChecking(); |
287 | bool isGrammarCheckingEnabled(); |
288 | void toggleGrammarChecking(); |
289 | void ignoreSpelling(); |
290 | void learnSpelling(); |
291 | int spellCheckerDocumentTag(); |
292 | WEBCORE_EXPORT bool isSelectionUngrammatical(); |
293 | String misspelledSelectionString() const; |
294 | String misspelledWordAtCaretOrRange(Node* clickedNode) const; |
295 | Vector<String> guessesForMisspelledWord(const String&) const; |
296 | Vector<String> guessesForMisspelledOrUngrammatical(bool& misspelled, bool& ungrammatical); |
297 | bool isSpellCheckingEnabledInFocusedNode() const; |
298 | bool isSpellCheckingEnabledFor(Node*) const; |
299 | WEBCORE_EXPORT void markMisspellingsAfterTypingToWord(const VisiblePosition &wordStart, const VisibleSelection& selectionAfterTyping, bool doReplacement); |
300 | void markMisspellings(const VisibleSelection&, RefPtr<Range>& firstMisspellingRange); |
301 | void markBadGrammar(const VisibleSelection&); |
302 | void markMisspellingsAndBadGrammar(const VisibleSelection& spellingSelection, bool markGrammar, const VisibleSelection& grammarSelection); |
303 | void markAndReplaceFor(const SpellCheckRequest&, const Vector<TextCheckingResult>&); |
304 | WEBCORE_EXPORT void replaceRangeForSpellChecking(Range&, const String&); |
305 | |
306 | bool isOverwriteModeEnabled() const { return m_overwriteModeEnabled; } |
307 | WEBCORE_EXPORT void toggleOverwriteModeEnabled(); |
308 | |
309 | void markAllMisspellingsAndBadGrammarInRanges(OptionSet<TextCheckingType>, RefPtr<Range>&& spellingRange, RefPtr<Range>&& automaticReplacementRange, RefPtr<Range>&& grammarRange); |
310 | #if PLATFORM(IOS_FAMILY) |
311 | NO_RETURN_DUE_TO_ASSERT |
312 | #endif |
313 | void changeBackToReplacedString(const String& replacedString); |
314 | |
315 | #if !PLATFORM(IOS_FAMILY) |
316 | WEBCORE_EXPORT void advanceToNextMisspelling(bool startBeforeSelection = false); |
317 | #endif |
318 | void showSpellingGuessPanel(); |
319 | bool spellingPanelIsShowing(); |
320 | |
321 | bool shouldBeginEditing(Range*); |
322 | bool shouldEndEditing(Range*); |
323 | |
324 | void clearUndoRedoOperations(); |
325 | bool canUndo() const; |
326 | void undo(); |
327 | bool canRedo() const; |
328 | void redo(); |
329 | |
330 | void registerCustomUndoStep(Ref<CustomUndoStep>&&); |
331 | |
332 | void didBeginEditing(); |
333 | void didEndEditing(); |
334 | void willWriteSelectionToPasteboard(Range*); |
335 | void didWriteSelectionToPasteboard(); |
336 | |
337 | void showFontPanel(); |
338 | void showStylesPanel(); |
339 | void showColorPanel(); |
340 | void toggleBold(); |
341 | void toggleUnderline(); |
342 | WEBCORE_EXPORT void setBaseWritingDirection(WritingDirection); |
343 | |
344 | // smartInsertDeleteEnabled and selectTrailingWhitespaceEnabled are |
345 | // mutually exclusive, meaning that enabling one will disable the other. |
346 | bool smartInsertDeleteEnabled(); |
347 | bool isSelectTrailingWhitespaceEnabled() const; |
348 | |
349 | WEBCORE_EXPORT bool hasBidiSelection() const; |
350 | |
351 | // international text input composition |
352 | bool hasComposition() const { return m_compositionNode; } |
353 | WEBCORE_EXPORT void setComposition(const String&, const Vector<CompositionUnderline>&, unsigned selectionStart, unsigned selectionEnd); |
354 | WEBCORE_EXPORT void confirmComposition(); |
355 | WEBCORE_EXPORT void confirmComposition(const String&); // if no existing composition, replaces selection |
356 | WEBCORE_EXPORT void cancelComposition(); |
357 | bool cancelCompositionIfSelectionIsInvalid(); |
358 | WEBCORE_EXPORT RefPtr<Range> compositionRange() const; |
359 | WEBCORE_EXPORT bool getCompositionSelection(unsigned& selectionStart, unsigned& selectionEnd) const; |
360 | |
361 | // getting international text input composition state (for use by InlineTextBox) |
362 | Text* compositionNode() const { return m_compositionNode.get(); } |
363 | unsigned compositionStart() const { return m_compositionStart; } |
364 | unsigned compositionEnd() const { return m_compositionEnd; } |
365 | bool compositionUsesCustomUnderlines() const { return !m_customCompositionUnderlines.isEmpty(); } |
366 | const Vector<CompositionUnderline>& customCompositionUnderlines() const { return m_customCompositionUnderlines; } |
367 | |
368 | enum class RevealSelection { No, Yes }; |
369 | WEBCORE_EXPORT void setIgnoreSelectionChanges(bool, RevealSelection shouldRevealExistingSelection = RevealSelection::Yes); |
370 | bool ignoreSelectionChanges() const { return m_ignoreSelectionChanges; } |
371 | |
372 | WEBCORE_EXPORT RefPtr<Range> rangeForPoint(const IntPoint& windowPoint); |
373 | |
374 | void clear(); |
375 | |
376 | VisibleSelection selectionForCommand(Event*); |
377 | |
378 | PAL::KillRing& killRing() const { return *m_killRing; } |
379 | SpellChecker& spellChecker() const { return *m_spellChecker; } |
380 | |
381 | EditingBehavior behavior() const; |
382 | |
383 | RefPtr<Range> selectedRange(); |
384 | |
385 | #if PLATFORM(IOS_FAMILY) |
386 | WEBCORE_EXPORT void confirmMarkedText(); |
387 | WEBCORE_EXPORT void setTextAsChildOfElement(const String&, Element&); |
388 | WEBCORE_EXPORT void setTextAlignmentForChangedBaseWritingDirection(WritingDirection); |
389 | WEBCORE_EXPORT void insertDictationPhrases(Vector<Vector<String>>&& dictationPhrases, RetainPtr<id> metadata); |
390 | WEBCORE_EXPORT void setDictationPhrasesAsChildOfElement(const Vector<Vector<String>>& dictationPhrases, RetainPtr<id> metadata, Element&); |
391 | #endif |
392 | |
393 | enum class KillRingInsertionMode { PrependText, AppendText }; |
394 | void addRangeToKillRing(const Range&, KillRingInsertionMode); |
395 | void addTextToKillRing(const String&, KillRingInsertionMode); |
396 | void setStartNewKillRingSequence(bool); |
397 | |
398 | void startAlternativeTextUITimer(); |
399 | // If user confirmed a correction in the correction panel, correction has non-zero length, otherwise it means that user has dismissed the panel. |
400 | WEBCORE_EXPORT void handleAlternativeTextUIResult(const String& correction); |
401 | void dismissCorrectionPanelAsIgnored(); |
402 | |
403 | WEBCORE_EXPORT void pasteAsFragment(Ref<DocumentFragment>&&, bool smartReplace, bool matchStyle, MailBlockquoteHandling = MailBlockquoteHandling::RespectBlockquote); |
404 | WEBCORE_EXPORT void pasteAsPlainText(const String&, bool smartReplace); |
405 | |
406 | // This is only called on the mac where paste is implemented primarily at the WebKit level. |
407 | WEBCORE_EXPORT void pasteAsPlainTextBypassingDHTML(); |
408 | |
409 | void clearMisspellingsAndBadGrammar(const VisibleSelection&); |
410 | void markMisspellingsAndBadGrammar(const VisibleSelection&); |
411 | |
412 | Element* findEventTargetFrom(const VisibleSelection& selection) const; |
413 | |
414 | WEBCORE_EXPORT String selectedText() const; |
415 | String selectedTextForDataTransfer() const; |
416 | WEBCORE_EXPORT bool findString(const String&, FindOptions); |
417 | |
418 | WEBCORE_EXPORT RefPtr<Range> rangeOfString(const String&, Range*, FindOptions); |
419 | |
420 | const VisibleSelection& mark() const; // Mark, to be used as emacs uses it. |
421 | void setMark(const VisibleSelection&); |
422 | |
423 | void computeAndSetTypingStyle(EditingStyle& , EditAction = EditAction::Unspecified); |
424 | WEBCORE_EXPORT void computeAndSetTypingStyle(StyleProperties& , EditAction = EditAction::Unspecified); |
425 | WEBCORE_EXPORT void applyEditingStyleToBodyElement() const; |
426 | void applyEditingStyleToElement(Element*) const; |
427 | |
428 | WEBCORE_EXPORT IntRect firstRectForRange(Range*) const; |
429 | |
430 | void selectionWillChange(); |
431 | void respondToChangedSelection(const VisibleSelection& oldSelection, OptionSet<FrameSelection::SetSelectionOption>); |
432 | WEBCORE_EXPORT void updateEditorUINowIfScheduled(); |
433 | bool shouldChangeSelection(const VisibleSelection& oldSelection, const VisibleSelection& newSelection, EAffinity, bool stillSelecting) const; |
434 | WEBCORE_EXPORT unsigned countMatchesForText(const String&, Range*, FindOptions, unsigned limit, bool markMatches, Vector<RefPtr<Range>>*); |
435 | bool markedTextMatchesAreHighlighted() const; |
436 | WEBCORE_EXPORT void setMarkedTextMatchesAreHighlighted(bool); |
437 | |
438 | void textFieldDidBeginEditing(Element*); |
439 | void textFieldDidEndEditing(Element*); |
440 | void textDidChangeInTextField(Element*); |
441 | bool doTextFieldCommandFromEvent(Element*, KeyboardEvent*); |
442 | void textWillBeDeletedInTextField(Element* input); |
443 | void textDidChangeInTextArea(Element*); |
444 | WEBCORE_EXPORT WritingDirection baseWritingDirectionForSelectionStart() const; |
445 | |
446 | enum class SelectReplacement : bool { No, Yes }; |
447 | enum class SmartReplace : bool { No, Yes }; |
448 | enum class MatchStyle : bool { No, Yes }; |
449 | WEBCORE_EXPORT void replaceSelectionWithFragment(DocumentFragment&, SelectReplacement, SmartReplace, MatchStyle, EditAction = EditAction::Insert, MailBlockquoteHandling = MailBlockquoteHandling::RespectBlockquote); |
450 | WEBCORE_EXPORT void replaceSelectionWithText(const String&, SelectReplacement, SmartReplace, EditAction = EditAction::Insert); |
451 | WEBCORE_EXPORT bool selectionStartHasMarkerFor(DocumentMarker::MarkerType, int from, int length) const; |
452 | void updateMarkersForWordsAffectedByEditing(bool onlyHandleWordsContainingSelection); |
453 | void deletedAutocorrectionAtPosition(const Position&, const String& originalString); |
454 | |
455 | WEBCORE_EXPORT void simplifyMarkup(Node* startNode, Node* endNode); |
456 | |
457 | EditorParagraphSeparator defaultParagraphSeparator() const { return m_defaultParagraphSeparator; } |
458 | void setDefaultParagraphSeparator(EditorParagraphSeparator separator) { m_defaultParagraphSeparator = separator; } |
459 | Vector<String> dictationAlternativesForMarker(const DocumentMarker&); |
460 | void applyDictationAlternativelternative(const String& alternativeString); |
461 | |
462 | #if USE(APPKIT) |
463 | WEBCORE_EXPORT void uppercaseWord(); |
464 | WEBCORE_EXPORT void lowercaseWord(); |
465 | WEBCORE_EXPORT void capitalizeWord(); |
466 | #endif |
467 | |
468 | #if USE(AUTOMATIC_TEXT_REPLACEMENT) |
469 | WEBCORE_EXPORT void showSubstitutionsPanel(); |
470 | WEBCORE_EXPORT bool substitutionsPanelIsShowing(); |
471 | WEBCORE_EXPORT void toggleSmartInsertDelete(); |
472 | WEBCORE_EXPORT bool isAutomaticQuoteSubstitutionEnabled(); |
473 | WEBCORE_EXPORT void toggleAutomaticQuoteSubstitution(); |
474 | WEBCORE_EXPORT bool isAutomaticLinkDetectionEnabled(); |
475 | WEBCORE_EXPORT void toggleAutomaticLinkDetection(); |
476 | WEBCORE_EXPORT bool isAutomaticDashSubstitutionEnabled(); |
477 | WEBCORE_EXPORT void toggleAutomaticDashSubstitution(); |
478 | WEBCORE_EXPORT bool isAutomaticTextReplacementEnabled(); |
479 | WEBCORE_EXPORT void toggleAutomaticTextReplacement(); |
480 | WEBCORE_EXPORT bool isAutomaticSpellingCorrectionEnabled(); |
481 | WEBCORE_EXPORT void toggleAutomaticSpellingCorrection(); |
482 | #endif |
483 | |
484 | RefPtr<DocumentFragment> webContentFromPasteboard(Pasteboard&, Range& context, bool allowPlainText, bool& chosePlainText); |
485 | |
486 | WEBCORE_EXPORT const Font* fontForSelection(bool& hasMultipleFonts) const; |
487 | WEBCORE_EXPORT static const RenderStyle* styleForSelectionStart(Frame* , Node *&nodeToRemove); |
488 | WEBCORE_EXPORT FontAttributes fontAttributesAtSelectionStart() const; |
489 | |
490 | #if PLATFORM(COCOA) |
491 | WEBCORE_EXPORT String stringSelectionForPasteboard(); |
492 | String stringSelectionForPasteboardWithImageAltText(); |
493 | void takeFindStringFromSelection(); |
494 | #if !PLATFORM(IOS_FAMILY) |
495 | WEBCORE_EXPORT void readSelectionFromPasteboard(const String& pasteboardName); |
496 | WEBCORE_EXPORT void replaceNodeFromPasteboard(Node*, const String& pasteboardName); |
497 | WEBCORE_EXPORT RefPtr<SharedBuffer> dataSelectionForPasteboard(const String& pasteboardName); |
498 | #endif // !PLATFORM(IOS_FAMILY) |
499 | WEBCORE_EXPORT void replaceSelectionWithAttributedString(NSAttributedString *, MailBlockquoteHandling = MailBlockquoteHandling::RespectBlockquote); |
500 | #endif |
501 | |
502 | bool canCopyExcludingStandaloneImages() const; |
503 | |
504 | #if !PLATFORM(WIN) |
505 | WEBCORE_EXPORT void writeSelectionToPasteboard(Pasteboard&); |
506 | WEBCORE_EXPORT void writeImageToPasteboard(Pasteboard&, Element& imageElement, const URL&, const String& title); |
507 | void writeSelection(PasteboardWriterData&); |
508 | #endif |
509 | |
510 | #if ENABLE(TELEPHONE_NUMBER_DETECTION) && !PLATFORM(IOS_FAMILY) |
511 | void scanSelectionForTelephoneNumbers(); |
512 | const Vector<RefPtr<Range>>& detectedTelephoneNumberRanges() const { return m_detectedTelephoneNumberRanges; } |
513 | #endif |
514 | |
515 | WEBCORE_EXPORT String stringForCandidateRequest() const; |
516 | WEBCORE_EXPORT void handleAcceptedCandidate(TextCheckingResult); |
517 | WEBCORE_EXPORT RefPtr<Range> contextRangeForCandidateRequest() const; |
518 | RefPtr<Range> rangeForTextCheckingResult(const TextCheckingResult&) const; |
519 | bool isHandlingAcceptedCandidate() const { return m_isHandlingAcceptedCandidate; } |
520 | |
521 | void (bool b) { m_isGettingDictionaryPopupInfo = b; } |
522 | bool () const { return m_isGettingDictionaryPopupInfo; } |
523 | |
524 | #if ENABLE(ATTACHMENT_ELEMENT) |
525 | WEBCORE_EXPORT void insertAttachment(const String& identifier, Optional<uint64_t>&& fileSize, const String& fileName, const String& contentType); |
526 | void registerAttachmentIdentifier(const String&, const String& contentType, const String& preferredFileName, Ref<SharedBuffer>&& fileData); |
527 | void registerAttachments(Vector<SerializedAttachmentData>&&); |
528 | void registerAttachmentIdentifier(const String&, const String& contentType, const String& filePath); |
529 | void registerAttachmentIdentifier(const String&); |
530 | void cloneAttachmentData(const String& fromIdentifier, const String& toIdentifier); |
531 | void didInsertAttachmentElement(HTMLAttachmentElement&); |
532 | void didRemoveAttachmentElement(HTMLAttachmentElement&); |
533 | |
534 | #if PLATFORM(COCOA) |
535 | void getPasteboardTypesAndDataForAttachment(Element&, Vector<String>& outTypes, Vector<RefPtr<SharedBuffer>>& outData); |
536 | #endif |
537 | #endif |
538 | |
539 | WEBCORE_EXPORT RefPtr<HTMLImageElement> insertEditableImage(); |
540 | |
541 | private: |
542 | Document& document() const; |
543 | |
544 | bool canDeleteRange(Range*) const; |
545 | bool canSmartReplaceWithPasteboard(Pasteboard&); |
546 | void pasteAsPlainTextWithPasteboard(Pasteboard&); |
547 | void pasteWithPasteboard(Pasteboard*, OptionSet<PasteOption>); |
548 | String plainTextFromPasteboard(const PasteboardPlainText&); |
549 | |
550 | void quoteFragmentForPasting(DocumentFragment&); |
551 | |
552 | void revealSelectionAfterEditingOperation(const ScrollAlignment& = ScrollAlignment::alignCenterIfNeeded, RevealExtentOption = DoNotRevealExtent); |
553 | void markMisspellingsOrBadGrammar(const VisibleSelection&, bool checkSpelling, RefPtr<Range>& firstMisspellingRange); |
554 | OptionSet<TextCheckingType> resolveTextCheckingTypeMask(const Node& rootEditableElement, OptionSet<TextCheckingType>); |
555 | |
556 | WEBCORE_EXPORT String selectedText(TextIteratorBehavior) const; |
557 | |
558 | void selectComposition(); |
559 | enum SetCompositionMode { ConfirmComposition, CancelComposition }; |
560 | void setComposition(const String&, SetCompositionMode); |
561 | |
562 | void changeSelectionAfterCommand(const VisibleSelection& newSelection, OptionSet<FrameSelection::SetSelectionOption>); |
563 | |
564 | enum EditorActionSpecifier { CutAction, CopyAction }; |
565 | void performCutOrCopy(EditorActionSpecifier); |
566 | |
567 | void editorUIUpdateTimerFired(); |
568 | |
569 | Element* findEventTargetFromSelection() const; |
570 | |
571 | bool unifiedTextCheckerEnabled() const; |
572 | |
573 | RefPtr<Range> adjustedSelectionRange(); |
574 | |
575 | #if PLATFORM(COCOA) |
576 | RefPtr<SharedBuffer> selectionInWebArchiveFormat(); |
577 | String selectionInHTMLFormat(); |
578 | RefPtr<SharedBuffer> imageInWebArchiveFormat(Element&); |
579 | static String userVisibleString(const URL&); |
580 | static RefPtr<SharedBuffer> dataInRTFDFormat(NSAttributedString *); |
581 | static RefPtr<SharedBuffer> dataInRTFFormat(NSAttributedString *); |
582 | #endif |
583 | void platformFontAttributesAtSelectionStart(FontAttributes&, const RenderStyle&) const; |
584 | |
585 | void scheduleEditorUIUpdate(); |
586 | |
587 | #if ENABLE(ATTACHMENT_ELEMENT) |
588 | void notifyClientOfAttachmentUpdates(); |
589 | #endif |
590 | |
591 | void postTextStateChangeNotificationForCut(const String&, const VisibleSelection&); |
592 | |
593 | Frame& m_frame; |
594 | RefPtr<CompositeEditCommand> m_lastEditCommand; |
595 | RefPtr<Text> m_compositionNode; |
596 | unsigned m_compositionStart; |
597 | unsigned m_compositionEnd; |
598 | Vector<CompositionUnderline> m_customCompositionUnderlines; |
599 | bool m_ignoreSelectionChanges { false }; |
600 | bool m_shouldStartNewKillRingSequence { false }; |
601 | bool m_shouldStyleWithCSS { false }; |
602 | const std::unique_ptr<PAL::KillRing> m_killRing; |
603 | const std::unique_ptr<SpellChecker> m_spellChecker; |
604 | const std::unique_ptr<AlternativeTextController> m_alternativeTextController; |
605 | EditorParagraphSeparator m_defaultParagraphSeparator { EditorParagraphSeparatorIsDiv }; |
606 | bool m_overwriteModeEnabled { false }; |
607 | |
608 | #if ENABLE(ATTACHMENT_ELEMENT) |
609 | HashSet<String> m_insertedAttachmentIdentifiers; |
610 | HashSet<String> m_removedAttachmentIdentifiers; |
611 | #endif |
612 | |
613 | VisibleSelection m_mark; |
614 | bool { false }; |
615 | |
616 | VisibleSelection m_oldSelectionForEditorUIUpdate; |
617 | Timer m_editorUIUpdateTimer; |
618 | bool m_editorUIUpdateTimerShouldCheckSpellingAndGrammar { false }; |
619 | bool m_editorUIUpdateTimerWasTriggeredByDictation { false }; |
620 | bool m_isHandlingAcceptedCandidate { false }; |
621 | |
622 | #if ENABLE(TELEPHONE_NUMBER_DETECTION) && !PLATFORM(IOS_FAMILY) |
623 | bool shouldDetectTelephoneNumbers(); |
624 | void scanRangeForTelephoneNumbers(Range&, const StringView&, Vector<RefPtr<Range>>& markedRanges); |
625 | |
626 | Timer m_telephoneNumberDetectionUpdateTimer; |
627 | Vector<RefPtr<Range>> m_detectedTelephoneNumberRanges; |
628 | #endif |
629 | |
630 | bool { false }; |
631 | }; |
632 | |
633 | inline void Editor::setStartNewKillRingSequence(bool flag) |
634 | { |
635 | m_shouldStartNewKillRingSequence = flag; |
636 | } |
637 | |
638 | inline const VisibleSelection& Editor::mark() const |
639 | { |
640 | return m_mark; |
641 | } |
642 | |
643 | inline void Editor::setMark(const VisibleSelection& selection) |
644 | { |
645 | m_mark = selection; |
646 | } |
647 | |
648 | inline bool Editor::markedTextMatchesAreHighlighted() const |
649 | { |
650 | return m_areMarkedTextMatchesHighlighted; |
651 | } |
652 | |
653 | } // namespace WebCore |
654 | |