// Copyright (c) 2012 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. #include "ui/views/ime/input_method_bridge.h" #include "ui/base/ime/input_method.h" #include "ui/base/ime/input_method_observer.h" #include "ui/events/event.h" #include "ui/gfx/rect.h" #include "ui/views/view.h" #include "ui/views/widget/widget.h" namespace views { // InputMethodBridge::HostObserver class --------------------------------------- // An observer class for observing the host input method. When the host input // method is destroyed, it will null out the |host_| field on the // InputMethodBridge object. class InputMethodBridge::HostObserver : public ui::InputMethodObserver { public: explicit HostObserver(InputMethodBridge* bridge); virtual ~HostObserver(); virtual void OnTextInputTypeChanged( const ui::TextInputClient* client) OVERRIDE {} virtual void OnFocus() OVERRIDE {} virtual void OnBlur() OVERRIDE {} virtual void OnCaretBoundsChanged( const ui::TextInputClient* client) OVERRIDE {} virtual void OnTextInputStateChanged( const ui::TextInputClient* client) OVERRIDE {} virtual void OnInputMethodDestroyed( const ui::InputMethod* input_method) OVERRIDE; virtual void OnShowImeIfNeeded() OVERRIDE {} private: InputMethodBridge* bridge_; DISALLOW_COPY_AND_ASSIGN(HostObserver); }; InputMethodBridge::HostObserver::HostObserver(InputMethodBridge* bridge) : bridge_(bridge) { bridge_->host_->AddObserver(this); } InputMethodBridge::HostObserver::~HostObserver() { if (bridge_->host_) bridge_->host_->RemoveObserver(this); } void InputMethodBridge::HostObserver::OnInputMethodDestroyed( const ui::InputMethod* input_method) { DCHECK_EQ(bridge_->host_, input_method); bridge_->host_->RemoveObserver(this); bridge_->host_ = NULL; } // InputMethodBridge class ----------------------------------------------------- InputMethodBridge::InputMethodBridge(internal::InputMethodDelegate* delegate, ui::InputMethod* host, bool shared_input_method) : host_(host), shared_input_method_(shared_input_method) { DCHECK(host_); SetDelegate(delegate); host_observer_.reset(new HostObserver(this)); } InputMethodBridge::~InputMethodBridge() { // By the time we get here it's very likely |widget_|'s NativeWidget has been // destroyed. This means any calls to |widget_| that go to the NativeWidget, // such as IsActive(), will crash. SetFocusedTextInputClient() may callback to // this and go into |widget_|. NULL out |widget_| so we don't attempt to use // it. DetachFromWidget(); // Host input method might have been destroyed at this point. if (host_) host_->DetachTextInputClient(this); } void InputMethodBridge::OnFocus() { DCHECK(host_); // Direct the shared IME to send TextInputClient messages to |this| object. if (shared_input_method_ || !host_->GetTextInputClient()) host_->SetFocusedTextInputClient(this); // TODO(yusukes): We don't need to call OnTextInputTypeChanged() once we move // text input type tracker code to ui::InputMethodBase. if (GetFocusedView()) { OnTextInputTypeChanged(GetFocusedView()); OnCaretBoundsChanged(GetFocusedView()); } } void InputMethodBridge::OnBlur() { DCHECK(host_); if (HasCompositionText()) { ConfirmCompositionText(); host_->CancelComposition(this); } if (host_->GetTextInputClient() == this) host_->SetFocusedTextInputClient(NULL); } bool InputMethodBridge::OnUntranslatedIMEMessage(const base::NativeEvent& event, NativeEventResult* result) { DCHECK(host_); return host_->OnUntranslatedIMEMessage(event, result); } void InputMethodBridge::DispatchKeyEvent(const ui::KeyEvent& key) { DCHECK(key.type() == ui::ET_KEY_PRESSED || key.type() == ui::ET_KEY_RELEASED); // We can just dispatch the event here since the |key| is already processed by // the system-wide IME. DispatchKeyEventPostIME(key); } void InputMethodBridge::OnTextInputTypeChanged(View* view) { DCHECK(host_); if (IsViewFocused(view)) host_->OnTextInputTypeChanged(this); InputMethodBase::OnTextInputTypeChanged(view); } void InputMethodBridge::OnCaretBoundsChanged(View* view) { DCHECK(host_); if (IsViewFocused(view) && !IsTextInputTypeNone()) host_->OnCaretBoundsChanged(this); } void InputMethodBridge::CancelComposition(View* view) { DCHECK(host_); if (IsViewFocused(view)) host_->CancelComposition(this); } void InputMethodBridge::OnInputLocaleChanged() { DCHECK(host_); host_->OnInputLocaleChanged(); } std::string InputMethodBridge::GetInputLocale() { DCHECK(host_); return host_->GetInputLocale(); } bool InputMethodBridge::IsActive() { DCHECK(host_); return host_->IsActive(); } bool InputMethodBridge::IsCandidatePopupOpen() const { DCHECK(host_); return host_->IsCandidatePopupOpen(); } void InputMethodBridge::ShowImeIfNeeded() { DCHECK(host_); host_->ShowImeIfNeeded(); } // Overridden from TextInputClient. Forward an event from the system-wide IME // to the text input |client|, which is e.g. views::Textfield. void InputMethodBridge::SetCompositionText( const ui::CompositionText& composition) { TextInputClient* client = GetTextInputClient(); if (client) client->SetCompositionText(composition); } void InputMethodBridge::ConfirmCompositionText() { TextInputClient* client = GetTextInputClient(); if (client) client->ConfirmCompositionText(); } void InputMethodBridge::ClearCompositionText() { TextInputClient* client = GetTextInputClient(); if (client) client->ClearCompositionText(); } void InputMethodBridge::InsertText(const base::string16& text) { TextInputClient* client = GetTextInputClient(); if (client) client->InsertText(text); } void InputMethodBridge::InsertChar(base::char16 ch, int flags) { TextInputClient* client = GetTextInputClient(); if (client) client->InsertChar(ch, flags); } gfx::NativeWindow InputMethodBridge::GetAttachedWindow() const { TextInputClient* client = GetTextInputClient(); return client ? client->GetAttachedWindow() : static_cast<gfx::NativeWindow>(NULL); } ui::TextInputType InputMethodBridge::GetTextInputType() const { TextInputClient* client = GetTextInputClient(); return client ? client->GetTextInputType() : ui::TEXT_INPUT_TYPE_NONE; } ui::TextInputMode InputMethodBridge::GetTextInputMode() const { TextInputClient* client = GetTextInputClient(); return client ? client->GetTextInputMode() : ui::TEXT_INPUT_MODE_DEFAULT; } bool InputMethodBridge::CanComposeInline() const { TextInputClient* client = GetTextInputClient(); return client ? client->CanComposeInline() : true; } gfx::Rect InputMethodBridge::GetCaretBounds() const { TextInputClient* client = GetTextInputClient(); if (!client) return gfx::Rect(); return client->GetCaretBounds(); } bool InputMethodBridge::GetCompositionCharacterBounds(uint32 index, gfx::Rect* rect) const { DCHECK(rect); TextInputClient* client = GetTextInputClient(); if (!client) return false; return client->GetCompositionCharacterBounds(index, rect); } bool InputMethodBridge::HasCompositionText() const { TextInputClient* client = GetTextInputClient(); return client ? client->HasCompositionText() : false; } bool InputMethodBridge::GetTextRange(gfx::Range* range) const { TextInputClient* client = GetTextInputClient(); return client ? client->GetTextRange(range) : false; } bool InputMethodBridge::GetCompositionTextRange(gfx::Range* range) const { TextInputClient* client = GetTextInputClient(); return client ? client->GetCompositionTextRange(range) : false; } bool InputMethodBridge::GetSelectionRange(gfx::Range* range) const { TextInputClient* client = GetTextInputClient(); return client ? client->GetSelectionRange(range) : false; } bool InputMethodBridge::SetSelectionRange(const gfx::Range& range) { TextInputClient* client = GetTextInputClient(); return client ? client->SetSelectionRange(range) : false; } bool InputMethodBridge::DeleteRange(const gfx::Range& range) { TextInputClient* client = GetTextInputClient(); return client ? client->DeleteRange(range) : false; } bool InputMethodBridge::GetTextFromRange(const gfx::Range& range, base::string16* text) const { TextInputClient* client = GetTextInputClient(); return client ? client->GetTextFromRange(range, text) : false; } void InputMethodBridge::OnInputMethodChanged() { TextInputClient* client = GetTextInputClient(); if (client) client->OnInputMethodChanged(); } bool InputMethodBridge::ChangeTextDirectionAndLayoutAlignment( base::i18n::TextDirection direction) { TextInputClient* client = GetTextInputClient(); return client ? client->ChangeTextDirectionAndLayoutAlignment(direction) : false; } void InputMethodBridge::ExtendSelectionAndDelete(size_t before, size_t after) { TextInputClient* client = GetTextInputClient(); if (client) client->ExtendSelectionAndDelete(before, after); } void InputMethodBridge::EnsureCaretInRect(const gfx::Rect& rect) { TextInputClient* client = GetTextInputClient(); if (client) client->EnsureCaretInRect(rect); } void InputMethodBridge::OnCandidateWindowShown() { } void InputMethodBridge::OnCandidateWindowUpdated() { } void InputMethodBridge::OnCandidateWindowHidden() { } bool InputMethodBridge::IsEditingCommandEnabled(int command_id) { return false; } void InputMethodBridge::ExecuteEditingCommand(int command_id) { } // Overridden from FocusChangeListener. void InputMethodBridge::OnWillChangeFocus(View* focused_before, View* focused) { if (HasCompositionText()) { ConfirmCompositionText(); CancelComposition(focused_before); } } void InputMethodBridge::OnDidChangeFocus(View* focused_before, View* focused) { DCHECK_EQ(GetFocusedView(), focused); OnTextInputTypeChanged(focused); OnCaretBoundsChanged(focused); } ui::InputMethod* InputMethodBridge::GetHostInputMethod() const { return host_; } } // namespace views