// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // The Mac interface forwards most of these commands to the application layer, // and I'm not really sure what to do about most of them. #include "config.h" #include "webkit/glue/editor_client_impl.h" #include "Document.h" #include "EditCommand.h" #include "Editor.h" #include "EventHandler.h" #include "EventNames.h" #include "Frame.h" #include "KeyboardCodes.h" #include "HTMLInputElement.h" #include "HTMLNames.h" #include "Frame.h" #include "KeyboardEvent.h" #include "PlatformKeyboardEvent.h" #include "PlatformString.h" #include "RenderObject.h" #undef LOG #include "base/message_loop.h" #include "base/string_util.h" #include "webkit/api/public/WebKit.h" #include "webkit/glue/autofill_form.h" #include "webkit/glue/dom_operations.h" #include "webkit/glue/editor_client_impl.h" #include "webkit/glue/glue_util.h" #include "webkit/glue/webkit_glue.h" #include "webkit/glue/webview.h" #include "webkit/glue/webview_impl.h" using webkit_glue::AutofillForm; // Arbitrary depth limit for the undo stack, to keep it from using // unbounded memory. This is the maximum number of distinct undoable // actions -- unbroken stretches of typed characters are coalesced // into a single action. static const size_t kMaximumUndoStackDepth = 1000; // The size above which we stop triggering autofill for an input text field // (so to avoid sending long strings through IPC). static const size_t kMaximumTextSizeForAutofill = 1000; namespace { // Record an editor command from the keyDownEntries[] below. We ignore the // Move* and Insert* commands because they're not that interesting. void MaybeRecordCommand(WebViewDelegate* d, const char* command_name) { if (!d) return; const char* move_prefix = "Move"; const char* insert_prefix = "Insert"; const char* delete_prefix = "Delete"; // Ignore all the Move*, Insert*, and Delete* commands. if (0 == strncmp(command_name, move_prefix, sizeof(move_prefix)) || 0 == strncmp(command_name, insert_prefix, sizeof(insert_prefix)) || 0 == strncmp(command_name, delete_prefix, sizeof(delete_prefix))) { return; } d->UserMetricsRecordComputedAction(UTF8ToWide(command_name)); } } EditorClientImpl::EditorClientImpl(WebView* web_view) : web_view_(static_cast(web_view)), use_editor_delegate_(false), in_redo_(false), backspace_or_delete_pressed_(false), spell_check_this_field_status_(SPELLCHECK_AUTOMATIC), // Don't complain about using "this" in initializer list. MSVC_PUSH_DISABLE_WARNING(4355) autofill_factory_(this) { MSVC_POP_WARNING() } EditorClientImpl::~EditorClientImpl() { } void EditorClientImpl::pageDestroyed() { // Called by the Page (which owns the editor client) when the page is going // away. This should cause us to delete ourselves, which is stupid. The page // should just delete us when it's going away. Oh well. delete this; } bool EditorClientImpl::shouldShowDeleteInterface(WebCore::HTMLElement* elem) { // Normally, we don't care to show WebCore's deletion UI, so we only enable // it if in testing mode and the test specifically requests it by using this // magic class name. return WebKit::layoutTestMode() && elem->getAttribute(WebCore::HTMLNames::classAttr) == "needsDeletionUI"; } bool EditorClientImpl::smartInsertDeleteEnabled() { if (use_editor_delegate_) { WebViewDelegate* d = web_view_->delegate(); if (d) return d->SmartInsertDeleteEnabled(); } return true; } bool EditorClientImpl::isSelectTrailingWhitespaceEnabled() { if (use_editor_delegate_) { WebViewDelegate* d = web_view_->delegate(); if (d) return d->IsSelectTrailingWhitespaceEnabled(); } #if defined(OS_WIN) return true; #elif defined(OS_MACOSX) || defined(OS_LINUX) return false; #endif } bool EditorClientImpl::ShouldSpellcheckByDefault() { // Spellcheck should be enabled for all editable areas (such as textareas, // contentEditable regions, and designMode docs), except text inputs. const WebCore::Frame* frame = web_view_->GetFocusedWebCoreFrame(); if (!frame) return false; const WebCore::Editor* editor = frame->editor(); if (!editor) return false; if (editor->spellCheckingEnabledInFocusedNode()) return true; const WebCore::Document* document = frame->document(); if (!document) return false; const WebCore::Node* node = document->focusedNode(); // If |node| is NULL, we default to allowing spellchecking. This is done in // order to mitigate the issue when the user clicks outside the textbox, as a // result of which |node| becomes NULL, resulting in all the spell check // markers being deleted. Also, the Frame will decide not to do spellchecking // if the user can't edit - so returning true here will not cause any problems // to the Frame's behavior. if (!node) return true; const WebCore::RenderObject* renderer = node->renderer(); if (!renderer) return false; return !renderer->isTextField(); } bool EditorClientImpl::isContinuousSpellCheckingEnabled() { if (spell_check_this_field_status_ == SPELLCHECK_FORCED_OFF) return false; else if (spell_check_this_field_status_ == SPELLCHECK_FORCED_ON) return true; else return ShouldSpellcheckByDefault(); } void EditorClientImpl::toggleContinuousSpellChecking() { if (isContinuousSpellCheckingEnabled()) spell_check_this_field_status_ = SPELLCHECK_FORCED_OFF; else spell_check_this_field_status_ = SPELLCHECK_FORCED_ON; } bool EditorClientImpl::isGrammarCheckingEnabled() { return false; } void EditorClientImpl::toggleGrammarChecking() { NOTIMPLEMENTED(); } int EditorClientImpl::spellCheckerDocumentTag() { NOTIMPLEMENTED(); return 0; } bool EditorClientImpl::isEditable() { return false; } bool EditorClientImpl::shouldBeginEditing(WebCore::Range* range) { if (use_editor_delegate_) { WebViewDelegate* d = web_view_->delegate(); if (d) return d->ShouldBeginEditing(web_view_, Describe(range)); } return true; } bool EditorClientImpl::shouldEndEditing(WebCore::Range* range) { if (use_editor_delegate_) { WebViewDelegate* d = web_view_->delegate(); if (d) return d->ShouldEndEditing(web_view_, Describe(range)); } return true; } bool EditorClientImpl::shouldInsertNode(WebCore::Node* node, WebCore::Range* range, WebCore::EditorInsertAction action) { if (use_editor_delegate_) { WebViewDelegate* d = web_view_->delegate(); if (d) { return d->ShouldInsertNode(web_view_, Describe(node), Describe(range), Describe(action)); } } return true; } bool EditorClientImpl::shouldInsertText(const WebCore::String& text, WebCore::Range* range, WebCore::EditorInsertAction action) { if (use_editor_delegate_) { WebViewDelegate* d = web_view_->delegate(); if (d) { std::wstring wstr = webkit_glue::StringToStdWString(text); return d->ShouldInsertText(web_view_, wstr, Describe(range), Describe(action)); } } return true; } bool EditorClientImpl::shouldDeleteRange(WebCore::Range* range) { if (use_editor_delegate_) { WebViewDelegate* d = web_view_->delegate(); if (d) return d->ShouldDeleteRange(web_view_, Describe(range)); } return true; } bool EditorClientImpl::shouldChangeSelectedRange(WebCore::Range* from_range, WebCore::Range* to_range, WebCore::EAffinity affinity, bool still_selecting) { if (use_editor_delegate_) { WebViewDelegate* d = web_view_->delegate(); if (d) { return d->ShouldChangeSelectedRange(web_view_, Describe(from_range), Describe(to_range), Describe(affinity), still_selecting); } } return true; } bool EditorClientImpl::shouldApplyStyle(WebCore::CSSStyleDeclaration* style, WebCore::Range* range) { if (use_editor_delegate_) { WebViewDelegate* d = web_view_->delegate(); if (d) return d->ShouldApplyStyle(web_view_, Describe(style), Describe(range)); } return true; } bool EditorClientImpl::shouldMoveRangeAfterDelete( WebCore::Range* /*range*/, WebCore::Range* /*rangeToBeReplaced*/) { return true; } void EditorClientImpl::didBeginEditing() { if (use_editor_delegate_) { WebViewDelegate* d = web_view_->delegate(); if (d) d->DidBeginEditing(); } } void EditorClientImpl::respondToChangedSelection() { if (use_editor_delegate_) { WebViewDelegate* d = web_view_->delegate(); if (d) { WebCore::Frame* frame = web_view_->GetFocusedWebCoreFrame(); if (frame) d->DidChangeSelection(!frame->selection()->isRange()); } } } void EditorClientImpl::respondToChangedContents() { if (use_editor_delegate_) { WebViewDelegate* d = web_view_->delegate(); if (d) d->DidChangeContents(); } } void EditorClientImpl::didEndEditing() { if (use_editor_delegate_) { WebViewDelegate* d = web_view_->delegate(); if (d) d->DidEndEditing(); } } void EditorClientImpl::didWriteSelectionToPasteboard() { } void EditorClientImpl::didSetSelectionTypesForPasteboard() { } void EditorClientImpl::registerCommandForUndo( PassRefPtr command) { if (undo_stack_.size() == kMaximumUndoStackDepth) undo_stack_.pop_front(); // drop oldest item off the far end if (!in_redo_) redo_stack_.clear(); undo_stack_.push_back(command); } void EditorClientImpl::registerCommandForRedo( PassRefPtr command) { redo_stack_.push_back(command); } void EditorClientImpl::clearUndoRedoOperations() { undo_stack_.clear(); redo_stack_.clear(); } bool EditorClientImpl::canUndo() const { return !undo_stack_.empty(); } bool EditorClientImpl::canRedo() const { return !redo_stack_.empty(); } void EditorClientImpl::undo() { if (canUndo()) { RefPtr command(undo_stack_.back()); undo_stack_.pop_back(); command->unapply(); // unapply will call us back to push this command onto the redo stack. } } void EditorClientImpl::redo() { if (canRedo()) { RefPtr command(redo_stack_.back()); redo_stack_.pop_back(); ASSERT(!in_redo_); in_redo_ = true; command->reapply(); // reapply will call us back to push this command onto the undo stack. in_redo_ = false; } } // The below code was adapted from the WebKit file webview.cpp // provided by Apple, Inc, and is subject to the following copyright // notice and disclaimer. /* * Copyright (C) 2006, 2007 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. */ static const unsigned CtrlKey = 1 << 0; static const unsigned AltKey = 1 << 1; static const unsigned ShiftKey = 1 << 2; static const unsigned MetaKey = 1 << 3; #if defined(OS_MACOSX) // Aliases for the generic key defintions to make kbd shortcuts definitions more // readable on OS X. static const unsigned OptionKey = AltKey; static const unsigned CommandKey = MetaKey; #endif // Keys with special meaning. These will be delegated to the editor using // the execCommand() method struct KeyDownEntry { unsigned virtualKey; unsigned modifiers; const char* name; }; struct KeyPressEntry { unsigned charCode; unsigned modifiers; const char* name; }; static const KeyDownEntry keyDownEntries[] = { { WebCore::VKEY_LEFT, 0, "MoveLeft" }, { WebCore::VKEY_LEFT, ShiftKey, "MoveLeftAndModifySelection" }, #if defined(OS_MACOSX) { WebCore::VKEY_LEFT, OptionKey, "MoveWordLeft" }, { WebCore::VKEY_LEFT, OptionKey | ShiftKey, "MoveWordLeftAndModifySelection" }, #else { WebCore::VKEY_LEFT, CtrlKey, "MoveWordLeft" }, { WebCore::VKEY_LEFT, CtrlKey | ShiftKey, "MoveWordLeftAndModifySelection"}, #endif { WebCore::VKEY_RIGHT, 0, "MoveRight" }, { WebCore::VKEY_RIGHT, ShiftKey, "MoveRightAndModifySelection" }, #if defined(OS_MACOSX) { WebCore::VKEY_RIGHT, OptionKey, "MoveWordRight" }, { WebCore::VKEY_RIGHT, OptionKey | ShiftKey, "MoveWordRightAndModifySelection" }, #else { WebCore::VKEY_RIGHT, CtrlKey, "MoveWordRight" }, { WebCore::VKEY_RIGHT, CtrlKey | ShiftKey, "MoveWordRightAndModifySelection" }, #endif { WebCore::VKEY_UP, 0, "MoveUp" }, { WebCore::VKEY_UP, ShiftKey, "MoveUpAndModifySelection" }, { WebCore::VKEY_PRIOR, ShiftKey, "MovePageUpAndModifySelection" }, { WebCore::VKEY_DOWN, 0, "MoveDown" }, { WebCore::VKEY_DOWN, ShiftKey, "MoveDownAndModifySelection" }, { WebCore::VKEY_NEXT, ShiftKey, "MovePageDownAndModifySelection"}, { WebCore::VKEY_PRIOR, 0, "MovePageUp" }, { WebCore::VKEY_NEXT, 0, "MovePageDown" }, { WebCore::VKEY_HOME, 0, "MoveToBeginningOfLine" }, { WebCore::VKEY_HOME, ShiftKey, "MoveToBeginningOfLineAndModifySelection" }, #if defined(OS_MACOSX) { WebCore::VKEY_LEFT, CommandKey, "MoveToBeginningOfLine" }, { WebCore::VKEY_LEFT, CommandKey | ShiftKey, "MoveToBeginningOfLineAndModifySelection" }, #endif #if defined(OS_MACOSX) { WebCore::VKEY_UP, CommandKey, "MoveToBeginningOfDocument" }, { WebCore::VKEY_UP, CommandKey | ShiftKey, "MoveToBeginningOfDocumentAndModifySelection" }, #else { WebCore::VKEY_HOME, CtrlKey, "MoveToBeginningOfDocument" }, { WebCore::VKEY_HOME, CtrlKey | ShiftKey, "MoveToBeginningOfDocumentAndModifySelection" }, #endif { WebCore::VKEY_END, 0, "MoveToEndOfLine" }, { WebCore::VKEY_END, ShiftKey, "MoveToEndOfLineAndModifySelection" }, #if defined(OS_MACOSX) { WebCore::VKEY_DOWN, CommandKey, "MoveToEndOfDocument" }, { WebCore::VKEY_DOWN, CommandKey | ShiftKey, "MoveToEndOfDocumentAndModifySelection" }, #else { WebCore::VKEY_END, CtrlKey, "MoveToEndOfDocument" }, { WebCore::VKEY_END, CtrlKey | ShiftKey, "MoveToEndOfDocumentAndModifySelection" }, #endif #if defined(OS_MACOSX) { WebCore::VKEY_RIGHT, CommandKey, "MoveToEndOfLine" }, { WebCore::VKEY_RIGHT, CommandKey | ShiftKey, "MoveToEndOfLineAndModifySelection" }, #endif { WebCore::VKEY_BACK, 0, "DeleteBackward" }, { WebCore::VKEY_BACK, ShiftKey, "DeleteBackward" }, { WebCore::VKEY_DELETE, 0, "DeleteForward" }, #if defined(OS_MACOSX) { WebCore::VKEY_BACK, OptionKey, "DeleteWordBackward" }, { WebCore::VKEY_DELETE, OptionKey, "DeleteWordForward" }, #else { WebCore::VKEY_BACK, CtrlKey, "DeleteWordBackward" }, { WebCore::VKEY_DELETE, CtrlKey, "DeleteWordForward" }, #endif { 'B', CtrlKey, "ToggleBold" }, { 'I', CtrlKey, "ToggleItalic" }, { 'U', CtrlKey, "ToggleUnderline" }, { WebCore::VKEY_ESCAPE, 0, "Cancel" }, { WebCore::VKEY_OEM_PERIOD, CtrlKey, "Cancel" }, { WebCore::VKEY_TAB, 0, "InsertTab" }, { WebCore::VKEY_TAB, ShiftKey, "InsertBacktab" }, { WebCore::VKEY_RETURN, 0, "InsertNewline" }, { WebCore::VKEY_RETURN, CtrlKey, "InsertNewline" }, { WebCore::VKEY_RETURN, AltKey, "InsertNewline" }, { WebCore::VKEY_RETURN, AltKey | ShiftKey, "InsertNewline" }, { WebCore::VKEY_RETURN, ShiftKey, "InsertLineBreak" }, { WebCore::VKEY_INSERT, CtrlKey, "Copy" }, { WebCore::VKEY_INSERT, ShiftKey, "Paste" }, { WebCore::VKEY_DELETE, ShiftKey, "Cut" }, #if defined(OS_MACOSX) { 'C', CommandKey, "Copy" }, { 'V', CommandKey, "Paste" }, { 'V', CommandKey | ShiftKey, "PasteAndMatchStyle" }, { 'X', CommandKey, "Cut" }, { 'A', CommandKey, "SelectAll" }, { 'Z', CommandKey, "Undo" }, { 'Z', CommandKey | ShiftKey, "Redo" }, { 'Y', CommandKey, "Redo" }, #else { 'C', CtrlKey, "Copy" }, { 'V', CtrlKey, "Paste" }, { 'V', CtrlKey | ShiftKey, "PasteAndMatchStyle" }, { 'X', CtrlKey, "Cut" }, { 'A', CtrlKey, "SelectAll" }, { 'Z', CtrlKey, "Undo" }, { 'Z', CtrlKey | ShiftKey, "Redo" }, { 'Y', CtrlKey, "Redo" }, #endif }; static const KeyPressEntry keyPressEntries[] = { { '\t', 0, "InsertTab" }, { '\t', ShiftKey, "InsertBacktab" }, { '\r', 0, "InsertNewline" }, { '\r', CtrlKey, "InsertNewline" }, { '\r', ShiftKey, "InsertLineBreak" }, { '\r', AltKey, "InsertNewline" }, { '\r', AltKey | ShiftKey, "InsertNewline" }, }; const char* EditorClientImpl::interpretKeyEvent( const WebCore::KeyboardEvent* evt) { const WebCore::PlatformKeyboardEvent* keyEvent = evt->keyEvent(); if (!keyEvent) return ""; static HashMap* keyDownCommandsMap = 0; static HashMap* keyPressCommandsMap = 0; if (!keyDownCommandsMap) { keyDownCommandsMap = new HashMap; keyPressCommandsMap = new HashMap; for (unsigned i = 0; i < arraysize(keyDownEntries); i++) { keyDownCommandsMap->set( keyDownEntries[i].modifiers << 16 | keyDownEntries[i].virtualKey, keyDownEntries[i].name); } for (unsigned i = 0; i < arraysize(keyPressEntries); i++) { keyPressCommandsMap->set( keyPressEntries[i].modifiers << 16 | keyPressEntries[i].charCode, keyPressEntries[i].name); } } unsigned modifiers = 0; if (keyEvent->shiftKey()) modifiers |= ShiftKey; if (keyEvent->altKey()) modifiers |= AltKey; if (keyEvent->ctrlKey()) modifiers |= CtrlKey; if (keyEvent->metaKey()) modifiers |= MetaKey; if (keyEvent->type() == WebCore::PlatformKeyboardEvent::RawKeyDown) { int mapKey = modifiers << 16 | evt->keyCode(); return mapKey ? keyDownCommandsMap->get(mapKey) : 0; } int mapKey = modifiers << 16 | evt->charCode(); return mapKey ? keyPressCommandsMap->get(mapKey) : 0; } bool EditorClientImpl::handleEditingKeyboardEvent( WebCore::KeyboardEvent* evt) { const WebCore::PlatformKeyboardEvent* keyEvent = evt->keyEvent(); // do not treat this as text input if it's a system key event if (!keyEvent || keyEvent->isSystemKey()) return false; WebCore::Frame* frame = evt->target()->toNode()->document()->frame(); if (!frame) return false; const char* command_name = interpretKeyEvent(evt); WebCore::Editor::Command command = frame->editor()->command(command_name); if (keyEvent->type() == WebCore::PlatformKeyboardEvent::RawKeyDown) { // WebKit doesn't have enough information about mode to decide how // commands that just insert text if executed via Editor should be treated, // so we leave it upon WebCore to either handle them immediately // (e.g. Tab that changes focus) or let a keypress event be generated // (e.g. Tab that inserts a Tab character, or Enter). if (command.isTextInsertion() || !command_name) return false; if (command.execute(evt)) { WebViewDelegate* d = web_view_->delegate(); MaybeRecordCommand(d, command_name); return true; } return false; } if (command.execute(evt)) { WebViewDelegate* d = web_view_->delegate(); MaybeRecordCommand(d, command_name); return true; } // Here we need to filter key events. // On Gtk/Linux, it emits key events with ASCII text and ctrl on for ctrl-. // In Webkit, EditorClient::handleKeyboardEvent in // WebKit/gtk/WebCoreSupport/EditorClientGtk.cpp drop such events. // On Mac, it emits key events with ASCII text and meta on for Command-. // These key events should not emit text insert event. // Alt key would be used to insert alternative character, so we should let // through. Also note that Ctrl-Alt combination equals to AltGr key which is // also used to insert alternative character. // http://code.google.com/p/chromium/issues/detail?id=10846 // Windows sets both alt and meta are on when "Alt" key pressed. // http://code.google.com/p/chromium/issues/detail?id=2215 // Also, we should not rely on an assumption that keyboards don't // send ASCII characters when pressing a control key on Windows, // which may be configured to do it so by user. // See also http://en.wikipedia.org/wiki/Keyboard_Layout // TODO(ukai): investigate more detail for various keyboard layout. if (evt->keyEvent()->text().length() == 1) { UChar ch = evt->keyEvent()->text()[0U]; // Don't insert null or control characters as they can result in // unexpected behaviour if (ch < ' ') return false; #if !defined(OS_WIN) // Don't insert ASCII character if ctrl w/o alt or meta is on. // On Mac, we should ignore events when meta is on (Command-). if (ch < 0x80) { if (evt->keyEvent()->ctrlKey() && !evt->keyEvent()->altKey()) return false; #if defined(OS_MACOSX) if (evt->keyEvent()->metaKey()) return false; #endif } #endif } if (!frame->editor()->canEdit()) return false; return frame->editor()->insertText(evt->keyEvent()->text(), evt); } // // End of code block subject to Apple, Inc. copyright // void EditorClientImpl::handleKeyboardEvent(WebCore::KeyboardEvent* evt) { if (evt->keyCode() == WebCore::VKEY_DOWN || evt->keyCode() == WebCore::VKEY_UP) { DCHECK(evt->target()->toNode()); ShowFormAutofillForNode(evt->target()->toNode()); } if (handleEditingKeyboardEvent(evt)) evt->setDefaultHandled(); } void EditorClientImpl::handleInputMethodKeydown(WebCore::KeyboardEvent* keyEvent) { // We handle IME within chrome. } void EditorClientImpl::textFieldDidBeginEditing(WebCore::Element*) { } void EditorClientImpl::textFieldDidEndEditing(WebCore::Element* element) { // Notification that focus was lost. Be careful with this, it's also sent // when the page is being closed. // Cancel any pending DoAutofill calls. autofill_factory_.RevokeAll(); // Hide any showing popup. web_view_->HideAutoCompletePopup(); // Notify any password-listener of the focus change. WebCore::HTMLInputElement* input_element = webkit_glue::ElementToHTMLInputElement(element); if (!input_element) return; WebFrameImpl* webframe = WebFrameImpl::FromFrame(input_element->document()->frame()); if (webframe->GetWebViewImpl() && !webframe->GetWebViewImpl()->GetDelegate()) return; // The page is getting closed, don't fill the password. webkit_glue::PasswordAutocompleteListener* listener = webframe->GetPasswordListener(input_element); if (!listener) return; std::wstring value = webkit_glue::StringToStdWString(input_element->value()); listener->OnBlur(input_element, value); } void EditorClientImpl::textDidChangeInTextField(WebCore::Element* element) { DCHECK(element->hasLocalName(WebCore::HTMLNames::inputTag)); // Note that we only show the autofill popup in this case if the caret is at // the end. This matches FireFox and Safari but not IE. Autofill(static_cast(element), false, false, true); } bool EditorClientImpl::ShowFormAutofillForNode(WebCore::Node* node) { WebCore::HTMLInputElement* input_element = webkit_glue::NodeToHTMLInputElement(node); if (input_element) return Autofill(input_element, true, true, false); return false; } bool EditorClientImpl::Autofill(WebCore::HTMLInputElement* input_element, bool form_autofill_only, bool autofill_on_empty_value, bool requires_caret_at_end) { // Cancel any pending DoAutofill calls. autofill_factory_.RevokeAll(); // Let's try to trigger autofill for that field, if applicable. if (!input_element->isEnabledFormControl() || !input_element->isTextField() || input_element->isPasswordField() || !input_element->autoComplete()) { return false; } std::wstring name = AutofillForm::GetNameForInputElement(input_element); if (name.empty()) // If the field has no name, then we won't have values. return false; // Don't attempt to autofill with values that are too large. if (input_element->value().length() > kMaximumTextSizeForAutofill) return false; if (!requires_caret_at_end) { DoAutofill(input_element, form_autofill_only, autofill_on_empty_value, false, backspace_or_delete_pressed_); } else { // We post a task for doing the autofill as the caret position is not set // properly at this point (http://bugs.webkit.org/show_bug.cgi?id=16976) and // we need it to determine whether or not to trigger autofill. std::wstring value = webkit_glue::StringToStdWString(input_element->value()); MessageLoop::current()->PostTask( FROM_HERE, autofill_factory_.NewRunnableMethod(&EditorClientImpl::DoAutofill, input_element, form_autofill_only, autofill_on_empty_value, true, backspace_or_delete_pressed_)); } return true; } void EditorClientImpl::DoAutofill(WebCore::HTMLInputElement* input_element, bool form_autofill_only, bool autofill_on_empty_value, bool requires_caret_at_end, bool backspace) { std::wstring value = webkit_glue::StringToStdWString(input_element->value()); // Enforce autofill_on_empty_value and caret_at_end. bool is_caret_at_end = requires_caret_at_end ? input_element->selectionStart() == input_element->selectionEnd() && input_element->selectionEnd() == static_cast(value.length()) : true; // When |requires_caret_at_end| is false, just pretend we are at // the end. if ((!autofill_on_empty_value && value.empty()) || !is_caret_at_end) { web_view_->HideAutoCompletePopup(); return; } // First let's see if there is a password listener for that element. // We won't trigger form autofill in that case, as having both behavior on // a node would be confusing. WebFrameImpl* webframe = WebFrameImpl::FromFrame(input_element->document()->frame()); webkit_glue::PasswordAutocompleteListener* listener = webframe->GetPasswordListener(input_element); if (listener) { if (form_autofill_only) return; listener->OnInlineAutocompleteNeeded(input_element, value, backspace, true); return; } // Then trigger form autofill. std::wstring name = AutofillForm::GetNameForInputElement(input_element); DCHECK_GT(static_cast(name.length()), 0); web_view_->delegate()->QueryFormFieldAutofill(name, value, reinterpret_cast(input_element)); } void EditorClientImpl::OnAutofillSuggestionAccepted( WebCore::HTMLInputElement* text_field) { WebFrameImpl* webframe = WebFrameImpl::FromFrame(text_field->document()->frame()); webkit_glue::PasswordAutocompleteListener* listener = webframe->GetPasswordListener(text_field); std::wstring value = webkit_glue::StringToStdWString(text_field->value()); // Password listeners need to autocomplete other fields that depend on the // input element with autofill suggestions. if (listener) listener->OnInlineAutocompleteNeeded(text_field, value, false, false); } bool EditorClientImpl::doTextFieldCommandFromEvent( WebCore::Element* element, WebCore::KeyboardEvent* event) { // Remember if backspace was pressed for the autofill. It is not clear how to // find if backspace was pressed from textFieldDidBeginEditing and // textDidChangeInTextField as when these methods are called the value of the // input element already contains the type character. backspace_or_delete_pressed_ = (event->keyCode() == WebCore::VKEY_BACK) || (event->keyCode() == WebCore::VKEY_DELETE); // The Mac code appears to use this method as a hook to implement special // keyboard commands specific to Safari's auto-fill implementation. We // just return false to allow the default action. return false; } void EditorClientImpl::textWillBeDeletedInTextField(WebCore::Element*) { } void EditorClientImpl::textDidChangeInTextArea(WebCore::Element*) { } void EditorClientImpl::ignoreWordInSpellDocument(const WebCore::String&) { NOTIMPLEMENTED(); } void EditorClientImpl::learnWord(const WebCore::String&) { NOTIMPLEMENTED(); } void EditorClientImpl::checkSpellingOfString(const UChar* str, int length, int* misspellingLocation, int* misspellingLength) { // SpellCheckWord will write (0, 0) into the output vars, which is what our // caller expects if the word is spelled correctly. int spell_location = -1; int spell_length = 0; WebViewDelegate* d = web_view_->delegate(); // Check to see if the provided str is spelled correctly. if (isContinuousSpellCheckingEnabled() && d) { std::wstring word = webkit_glue::StringToStdWString(WebCore::String(str, length)); d->SpellCheck(word, &spell_location, &spell_length); } else { spell_location = 0; spell_length = 0; } // Note: the Mac code checks if the pointers are NULL before writing to them, // so we do too. if (misspellingLocation) *misspellingLocation = spell_location; if (misspellingLength) *misspellingLength = spell_length; } WebCore::String EditorClientImpl::getAutoCorrectSuggestionForMisspelledWord( const WebCore::String& misspelledWord) { WebViewDelegate* d = web_view_->delegate(); if (!(isContinuousSpellCheckingEnabled() && d)) return WebCore::String(); std::wstring word = webkit_glue::StringToStdWString(misspelledWord); // Do not autocorrect words with capital letters in it except the // first letter. This will remove cases changing "IMB" to "IBM". for (size_t i = 1; i < word.length(); i++) { if (u_isupper(static_cast(word[i]))) return WebCore::String(); } std::wstring autocorrect_word = d->GetAutoCorrectWord(word); return webkit_glue::StdWStringToString(autocorrect_word); } void EditorClientImpl::checkGrammarOfString(const UChar*, int length, WTF::Vector&, int* badGrammarLocation, int* badGrammarLength) { NOTIMPLEMENTED(); if (badGrammarLocation) *badGrammarLocation = 0; if (badGrammarLength) *badGrammarLength = 0; } void EditorClientImpl::updateSpellingUIWithGrammarString(const WebCore::String&, const WebCore::GrammarDetail& detail) { NOTIMPLEMENTED(); } void EditorClientImpl::updateSpellingUIWithMisspelledWord(const WebCore::String&) { NOTIMPLEMENTED(); } void EditorClientImpl::showSpellingUI(bool show) { NOTIMPLEMENTED(); } bool EditorClientImpl::spellingUIIsShowing() { return false; } void EditorClientImpl::getGuessesForWord(const WebCore::String&, WTF::Vector& guesses) { NOTIMPLEMENTED(); } void EditorClientImpl::setInputMethodState(bool enabled) { WebViewDelegate* d = web_view_->delegate(); if (d) { d->SetInputMethodState(enabled); } } std::wstring EditorClientImpl::DescribeOrError(int number, WebCore::ExceptionCode ec) { if (ec) return L"ERROR"; return IntToWString(number); } std::wstring EditorClientImpl::DescribeOrError(WebCore::Node* node, WebCore::ExceptionCode ec) { if (ec) return L"ERROR"; return Describe(node); } // These Describe() functions match the output expected by the layout tests. std::wstring EditorClientImpl::Describe(WebCore::Range* range) { if (range) { WebCore::ExceptionCode exception = 0; std::wstring str = L"range from "; int offset = range->startOffset(exception); str.append(DescribeOrError(offset, exception)); str.append(L" of "); WebCore::Node* container = range->startContainer(exception); str.append(DescribeOrError(container, exception)); str.append(L" to "); offset = range->endOffset(exception); str.append(DescribeOrError(offset, exception)); str.append(L" of "); container = range->endContainer(exception); str.append(DescribeOrError(container, exception)); return str; } return L"(null)"; } // See comment for Describe(), above. std::wstring EditorClientImpl::Describe(WebCore::Node* node) { if (node) { std::wstring str = webkit_glue::StringToStdWString(node->nodeName()); WebCore::Node* parent = node->parentNode(); if (parent) { str.append(L" > "); str.append(Describe(parent)); } return str; } return L"(null)"; } // See comment for Describe(), above. std::wstring EditorClientImpl::Describe(WebCore::EditorInsertAction action) { switch (action) { case WebCore::EditorInsertActionTyped: return L"WebViewInsertActionTyped"; case WebCore::EditorInsertActionPasted: return L"WebViewInsertActionPasted"; case WebCore::EditorInsertActionDropped: return L"WebViewInsertActionDropped"; } return L"(UNKNOWN ACTION)"; } // See comment for Describe(), above. std::wstring EditorClientImpl::Describe(WebCore::EAffinity affinity) { switch (affinity) { case WebCore::UPSTREAM: return L"NSSelectionAffinityUpstream"; case WebCore::DOWNSTREAM: return L"NSSelectionAffinityDownstream"; } return L"(UNKNOWN AFFINITY)"; } std::wstring EditorClientImpl::Describe(WebCore::CSSStyleDeclaration* style) { // TODO(pamg): Implement me. It's not clear what WebKit produces for this // (their [style description] method), and none of the layout tests provide // an example. But because none of them use it, it's not yet important. return std::wstring(); }