Index: chrome/browser/chromeos/input_method/input_method_engine_ibus.cc |
diff --git a/chrome/browser/chromeos/input_method/input_method_engine_ibus.cc b/chrome/browser/chromeos/input_method/input_method_engine_ibus.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..c52c4cd517342b947bc6559fb54047aed472d7a6 |
--- /dev/null |
+++ b/chrome/browser/chromeos/input_method/input_method_engine_ibus.cc |
@@ -0,0 +1,577 @@ |
+// 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 "chrome/browser/chromeos/input_method/input_method_engine_ibus.h" |
+ |
+#include <map> |
+ |
+#include "base/logging.h" |
+#include "base/memory/scoped_ptr.h" |
+#include "base/string_number_conversions.h" |
+#include "base/string_util.h" |
+#include "chrome/browser/chromeos/input_method/ibus_keymap.h" |
+#include "chrome/browser/chromeos/input_method/input_method_manager.h" |
+#include "chrome/browser/chromeos/input_method/input_method_util.h" |
+#include "chromeos/dbus/dbus_thread_manager.h" |
+#include "chromeos/dbus/ibus/ibus_client.h" |
+#include "chromeos/dbus/ibus/ibus_component.h" |
+#include "chromeos/dbus/ibus/ibus_engine_factory_service.h" |
+#include "chromeos/dbus/ibus/ibus_engine_service.h" |
+#include "chromeos/dbus/ibus/ibus_lookup_table.h" |
+#include "chromeos/dbus/ibus/ibus_property.h" |
+#include "chromeos/dbus/ibus/ibus_text.h" |
+#include "dbus/object_path.h" |
+ |
+namespace chromeos { |
+const char* kExtensionImePrefix = "_ext_ime_"; |
+const char* kErrorNotActive = "IME is not active"; |
+const char* kErrorWrongContext = "Context is not active"; |
+const char* kCandidateNotFound = "Candidate not found"; |
+const char* kEngineBusPrefix = "org.freedesktop.IBus."; |
+ |
+namespace { |
+const uint32 kIBusAltKeyMask = 1 << 3; |
+const uint32 kIBusCtrlKeyMask = 1 << 2; |
+const uint32 kIBusShiftKeyMask = 1 << 0; |
+const uint32 kIBusKeyReleaseMask = 1 << 30; |
+} |
+ |
+InputMethodEngineIBus::InputMethodEngineIBus() |
+ : focused_(false), |
+ active_(false), |
+ context_id_(0), |
+ next_context_id_(1), |
+ current_object_path_(0), |
+ aux_text_(new ibus::IBusText()), |
+ aux_text_visible_(false), |
+ observer_(NULL), |
+ preedit_text_(new ibus::IBusText()), |
+ preedit_cursor_(0), |
+ component_(new ibus::IBusComponent()), |
+ table_(new ibus::IBusLookupTable()), |
+ table_visible_(false), |
+ weak_ptr_factory_(this) { |
+} |
+ |
+void InputMethodEngineIBus::Initialize( |
+ InputMethodEngine::Observer* observer, |
+ const char* engine_name, |
+ const char* extension_id, |
+ const char* engine_id, |
+ const char* description, |
+ const char* language, |
+ const std::vector<std::string>& layouts, |
+ std::string* error) { |
+ DCHECK(observer) << "Observer must not be null."; |
+ |
+ observer_ = observer; |
+ engine_id_ = engine_id; |
+ ibus_id_ = kExtensionImePrefix; |
+ ibus_id_ += extension_id; |
+ ibus_id_ += engine_id; |
+ |
+ input_method::InputMethodManager* manager = |
+ input_method::InputMethodManager::GetInstance(); |
+ std::string layout; |
+ if (!layouts.empty()) { |
+ layout = JoinString(layouts, ','); |
+ } else { |
+ input_method::InputMethodManager* manager = |
+ input_method::InputMethodManager::GetInstance(); |
+ const std::string fallback_id = |
+ manager->GetInputMethodUtil()->GetHardwareInputMethodId(); |
+ const input_method::InputMethodDescriptor* fallback_desc = |
+ manager->GetInputMethodUtil()->GetInputMethodDescriptorFromId( |
+ fallback_id); |
+ layout = fallback_desc->keyboard_layout(); |
+ } |
+ |
+ component_.reset(new ibus::IBusComponent()); |
+ component_->set_name(std::string(kEngineBusPrefix) + std::string(engine_id)); |
+ component_->set_description(description); |
+ component_->set_author(engine_name); |
+ |
+ chromeos::ibus::IBusComponent::EngineDescription engine_desc; |
+ engine_desc.engine_id = ibus_id_; |
+ engine_desc.display_name = description; |
+ engine_desc.description = description; |
+ engine_desc.language_code = language; |
+ engine_desc.author = ibus_id_; |
+ engine_desc.layout = layout.c_str(); |
+ |
+ component_->mutable_engine_description()->push_back(engine_desc); |
+ manager->AddInputMethodExtension(ibus_id_, engine_name, layouts, language, |
+ this); |
+ // If connection is avaiable, register component. If there are no connection |
+ // to ibus-daemon, OnConnected callback will register component instead. |
+ if (IsConnected()) |
+ RegisterComponent(); |
+} |
+ |
+InputMethodEngineIBus::~InputMethodEngineIBus() { |
+} |
+ |
+bool InputMethodEngineIBus::SetComposition( |
+ int context_id, |
+ const char* text, |
+ int selection_start, |
+ int selection_end, |
+ int cursor, |
+ const std::vector<SegmentInfo>& segments, |
+ std::string* error) { |
+ if (!active_) { |
+ *error = kErrorNotActive; |
+ return false; |
+ } |
+ if (context_id != context_id_ || context_id_ == -1) { |
+ *error = kErrorWrongContext; |
+ return false; |
+ } |
+ |
+ preedit_cursor_ = cursor; |
+ preedit_text_.reset(new ibus::IBusText()); |
+ preedit_text_->set_text(text); |
+ |
+ // TODO: Add support for displaying selected text in the composition string. |
+ for (std::vector<SegmentInfo>::const_iterator segment = segments.begin(); |
+ segment != segments.end(); ++segment) { |
+ ibus::IBusText::UnderlineAttribute underline; |
+ |
+ switch (segment->style) { |
+ case SEGMENT_STYLE_UNDERLINE: |
+ underline.type = ibus::IBusText::IBUS_TEXT_UNDERLINE_SINGLE; |
+ break; |
+ case SEGMENT_STYLE_DOUBLE_UNDERLINE: |
+ underline.type = ibus::IBusText::IBUS_TEXT_UNDERLINE_DOUBLE; |
+ break; |
+ default: |
+ continue; |
+ } |
+ |
+ underline.start_index = segment->start; |
+ underline.end_index = segment->end; |
+ preedit_text_->mutable_underline_attributes()->push_back(underline); |
+ } |
+ |
+ // TODO(nona): Makes focus out mode configuable, if necessary. |
+ GetCurrentService()->UpdatePreedit( |
+ *preedit_text_.get(), |
+ preedit_cursor_, |
+ true, |
+ chromeos::IBusEngineService::IBUS_ENGINE_PREEEDIT_FOCUS_OUT_MODE_COMMIT); |
+ return true; |
+} |
+ |
+bool InputMethodEngineIBus::ClearComposition(int context_id, |
+ std::string* error) { |
+ if (!active_) { |
+ *error = kErrorNotActive; |
+ return false; |
+ } |
+ if (context_id != context_id_ || context_id_ == -1) { |
+ *error = kErrorWrongContext; |
+ return false; |
+ } |
+ |
+ preedit_cursor_ = 0; |
+ preedit_text_.reset(new ibus::IBusText()); |
+ GetCurrentService()->UpdatePreedit( |
+ *preedit_text_.get(), |
+ 0, |
+ true, |
+ chromeos::IBusEngineService::IBUS_ENGINE_PREEEDIT_FOCUS_OUT_MODE_COMMIT); |
+ return true; |
+} |
+ |
+bool InputMethodEngineIBus::CommitText(int context_id, const char* text, |
+ std::string* error) { |
+ if (!active_) { |
+ // TODO: Commit the text anyways. |
+ *error = kErrorNotActive; |
+ return false; |
+ } |
+ if (context_id != context_id_ || context_id_ == -1) { |
+ *error = kErrorWrongContext; |
+ return false; |
+ } |
+ |
+ GetCurrentService()->CommitText(text); |
+ return true; |
+} |
+ |
+bool InputMethodEngineIBus::SetCandidateWindowVisible(bool visible, |
+ std::string* error) { |
+ if (!active_) { |
+ *error = kErrorNotActive; |
+ return false; |
+ } |
+ |
+ table_visible_ = visible; |
+ GetCurrentService()->UpdateLookupTable(*table_.get(), table_visible_); |
+ return true; |
+} |
+ |
+void InputMethodEngineIBus::SetCandidateWindowCursorVisible(bool visible) { |
+ if (!active_) |
+ return; |
+ table_->set_is_cursor_visible(visible); |
+ GetCurrentService()->UpdateLookupTable(*table_.get(), table_visible_); |
+} |
+ |
+void InputMethodEngineIBus::SetCandidateWindowVertical(bool vertical) { |
+ if (!active_) |
+ return; |
+ table_->set_orientation( |
+ vertical ? ibus::IBusLookupTable::IBUS_LOOKUP_TABLE_ORIENTATION_VERTICAL : |
+ ibus::IBusLookupTable::IBUS_LOOKUP_TABLE_ORIENTATION_HORIZONTAL); |
+ GetCurrentService()->UpdateLookupTable(*table_.get(), table_visible_); |
+} |
+ |
+void InputMethodEngineIBus::SetCandidateWindowPageSize(int size) { |
+ if (!active_) |
+ return; |
+ table_->set_page_size(size); |
+ GetCurrentService()->UpdateLookupTable(*table_.get(), table_visible_); |
+} |
+ |
+void InputMethodEngineIBus::SetCandidateWindowAuxText(const char* text) { |
+ if (!active_) |
+ return; |
+ aux_text_->set_text(text); |
+ GetCurrentService()->UpdateAuxiliaryText(*aux_text_.get(), aux_text_visible_); |
+} |
+ |
+void InputMethodEngineIBus::SetCandidateWindowAuxTextVisible(bool visible) { |
+ if (!active_) |
+ return; |
+ aux_text_visible_ = visible; |
+ GetCurrentService()->UpdateAuxiliaryText(*aux_text_.get(), aux_text_visible_); |
+} |
+ |
+bool InputMethodEngineIBus::SetCandidates( |
+ int context_id, |
+ const std::vector<Candidate>& candidates, |
+ std::string* error) { |
+ if (!active_) { |
+ *error = kErrorNotActive; |
+ return false; |
+ } |
+ if (context_id != context_id_ || context_id_ == -1) { |
+ *error = kErrorWrongContext; |
+ return false; |
+ } |
+ |
+ // TODO: Nested candidates |
+ candidate_ids_.clear(); |
+ candidate_indexes_.clear(); |
+ table_->mutable_candidates()->clear(); |
+ for (std::vector<Candidate>::const_iterator ix = candidates.begin(); |
+ ix != candidates.end(); ++ix) { |
+ ibus::IBusLookupTable::Entry entry; |
+ // TODO(nona): support annotation(crbug.com/140186). |
+ entry.value = ix->value + " " + ix->annotation; |
+ entry.label = ix->label; |
+ |
+ // Store a mapping from the user defined ID to the candidate index. |
+ candidate_indexes_[ix->id] = candidate_ids_.size(); |
+ candidate_ids_.push_back(ix->id); |
+ |
+ table_->mutable_candidates()->push_back(entry); |
+ } |
+ GetCurrentService()->UpdateLookupTable(*table_.get(), table_visible_); |
+ return true; |
+} |
+ |
+bool InputMethodEngineIBus::SetCursorPosition(int context_id, int candidate_id, |
+ std::string* error) { |
+ if (!active_) { |
+ *error = kErrorNotActive; |
+ return false; |
+ } |
+ if (context_id != context_id_ || context_id_ == -1) { |
+ *error = kErrorWrongContext; |
+ return false; |
+ } |
+ |
+ std::map<int, int>::const_iterator position = |
+ candidate_indexes_.find(candidate_id); |
+ if (position == candidate_indexes_.end()) { |
+ *error = kCandidateNotFound; |
+ return false; |
+ } |
+ |
+ table_->set_cursor_position(position->second); |
+ GetCurrentService()->UpdateLookupTable(*table_.get(), table_visible_); |
+ return true; |
+} |
+ |
+bool InputMethodEngineIBus::SetMenuItems(const std::vector<MenuItem>& items) { |
+ if (!active_) |
+ return false; |
+ |
+ ibus::IBusPropertyList properties; |
+ for (std::vector<MenuItem>::const_iterator item = items.begin(); |
+ item != items.end(); ++item) { |
+ ibus::IBusProperty* property = new ibus::IBusProperty(); |
+ if (!MenuItemToProperty(*item, property)) { |
+ delete property; |
+ DVLOG(1) << "Bad menu item"; |
+ return false; |
+ } |
+ properties.push_back(property); |
+ } |
+ GetCurrentService()->RegisterProperties(properties); |
+ return true; |
+} |
+ |
+bool InputMethodEngineIBus::UpdateMenuItems( |
+ const std::vector<MenuItem>& items) { |
+ if (!active_) |
+ return false; |
+ |
+ ibus::IBusPropertyList properties; |
+ for (std::vector<MenuItem>::const_iterator item = items.begin(); |
+ item != items.end(); ++item) { |
+ ibus::IBusProperty* property = new ibus::IBusProperty(); |
+ if (!MenuItemToProperty(*item, property)) { |
+ delete property; |
+ DVLOG(1) << "Bad menu item"; |
+ return false; |
+ } |
+ properties.push_back(property); |
+ } |
+ GetCurrentService()->RegisterProperties(properties); |
+ return true; |
+} |
+ |
+bool InputMethodEngineIBus::IsActive() const { |
+ return active_; |
+} |
+ |
+void InputMethodEngineIBus::KeyEventDone(input_method::KeyEventHandle* key_data, |
+ bool handled) { |
+ KeyEventDoneCallback* callback = |
+ reinterpret_cast<KeyEventDoneCallback*>(key_data); |
+ callback->Run(handled); |
+ delete callback; |
+} |
+ |
+void InputMethodEngineIBus::FocusIn() { |
+ focused_ = true; |
+ if (!active_) |
+ return; |
+ context_id_ = next_context_id_; |
+ ++next_context_id_; |
+ |
+ InputContext context; |
+ context.id = context_id_; |
+ // TODO: Other types |
+ context.type = "text"; |
+ |
+ observer_->OnFocus(context); |
+} |
+ |
+void InputMethodEngineIBus::FocusOut() { |
+ focused_ = false; |
+ if (!active_) |
+ return; |
+ int context_id = context_id_; |
+ context_id_ = -1; |
+ observer_->OnBlur(context_id); |
+} |
+ |
+void InputMethodEngineIBus::Enable() { |
+ active_ = true; |
+ observer_->OnActivate(engine_id_); |
+ FocusIn(); |
+} |
+ |
+void InputMethodEngineIBus::Disable() { |
+ active_ = false; |
+ observer_->OnDeactivated(engine_id_); |
+} |
+ |
+void InputMethodEngineIBus::PropertyActivate( |
+ const std::string& property_name, |
+ IBusPropertyState property_state) { |
+ observer_->OnMenuItemActivated(engine_id_, property_name); |
+} |
+ |
+void InputMethodEngineIBus::PropertyShow( |
+ const std::string& property_name) { |
+} |
+ |
+void InputMethodEngineIBus::PropertyHide( |
+ const std::string& property_name) { |
+} |
+ |
+void InputMethodEngineIBus::SetCapability( |
+ IBusCapability capability) { |
+} |
+ |
+void InputMethodEngineIBus::Reset() { |
+} |
+ |
+void InputMethodEngineIBus::ProcessKeyEvent( |
+ uint32 keysym, |
+ uint32 keycode, |
+ uint32 state, |
+ const KeyEventDoneCallback& callback) { |
+ |
+ KeyEventDoneCallback *handler = new KeyEventDoneCallback(); |
+ *handler = callback; |
+ |
+ KeyboardEvent event; |
+ event.type = !(state & kIBusKeyReleaseMask) ? "keydown" : "keyup"; |
+ event.key = input_method::GetIBusKey(keysym); |
+ event.alt_key = state & kIBusAltKeyMask; |
+ event.ctrl_key = state & kIBusCtrlKeyMask; |
+ event.shift_key = state & kIBusShiftKeyMask; |
+ observer_->OnKeyEvent( |
+ engine_id_, |
+ event, |
+ reinterpret_cast<input_method::KeyEventHandle*>(handler)); |
+} |
+ |
+void InputMethodEngineIBus::CandidateClicked( |
+ uint32 index, |
+ IBusMouseButton button, |
+ uint32 state) { |
+ if (index > candidate_ids_.size()) { |
+ return; |
+ } |
+ |
+ MouseButtonEvent pressed_button; |
+ switch (button) { |
+ case IBusEngineHandlerInterface::IBUS_MOUSE_BUTTON_LEFT: |
+ pressed_button = MOUSE_BUTTON_LEFT; |
+ break; |
+ case IBusEngineHandlerInterface::IBUS_MOUSE_BUTTON_MIDDLE: |
+ pressed_button = MOUSE_BUTTON_MIDDLE; |
+ break; |
+ case IBusEngineHandlerInterface::IBUS_MOUSE_BUTTON_RIGHT: |
+ pressed_button = MOUSE_BUTTON_RIGHT; |
+ break; |
+ default: |
+ DVLOG(1) << "Unknown button: " << button; |
+ pressed_button = MOUSE_BUTTON_LEFT; |
+ break; |
+ } |
+ |
+ observer_->OnCandidateClicked( |
+ engine_id_, candidate_ids_.at(index), pressed_button); |
+} |
+ |
+void InputMethodEngineIBus::SetSurroundingText( |
+ const std::string& text, |
+ uint32 cursor_pos, |
+ uint32 anchor_pos) { |
+} |
+ |
+IBusEngineService* InputMethodEngineIBus::GetCurrentService() { |
+ return DBusThreadManager::Get()->GetIBusEngineService(object_path_); |
+} |
+ |
+bool InputMethodEngineIBus::MenuItemToProperty( |
+ const MenuItem& item, |
+ ibus::IBusProperty* property) { |
+ property->set_key(item.id); |
+ |
+ if (item.modified & MENU_ITEM_MODIFIED_LABEL) { |
+ property->set_label(item.label); |
+ } |
+ if (item.modified & MENU_ITEM_MODIFIED_VISIBLE) { |
+ property->set_visible(item.visible); |
+ } |
+ if (item.modified & MENU_ITEM_MODIFIED_CHECKED) { |
+ property->set_checked(item.checked); |
+ } |
+ if (item.modified & MENU_ITEM_MODIFIED_ENABLED) { |
+ // TODO(nona): implement sensitive entry(crbug.com/140192). |
+ } |
+ if (item.modified & MENU_ITEM_MODIFIED_STYLE) { |
+ ibus::IBusProperty::IBusPropertyType type = |
+ ibus::IBusProperty::IBUS_PROPERTY_TYPE_NORMAL; |
+ if (!item.children.empty()) { |
+ type = ibus::IBusProperty::IBUS_PROPERTY_TYPE_MENU; |
+ } else { |
+ switch (item.style) { |
+ case MENU_ITEM_STYLE_NONE: |
+ type = ibus::IBusProperty::IBUS_PROPERTY_TYPE_NORMAL; |
+ break; |
+ case MENU_ITEM_STYLE_CHECK: |
+ type = ibus::IBusProperty::IBUS_PROPERTY_TYPE_TOGGLE; |
+ break; |
+ case MENU_ITEM_STYLE_RADIO: |
+ type = ibus::IBusProperty::IBUS_PROPERTY_TYPE_RADIO; |
+ break; |
+ case MENU_ITEM_STYLE_SEPARATOR: |
+ type = ibus::IBusProperty::IBUS_PROPERTY_TYPE_SEPARATOR; |
+ break; |
+ } |
+ } |
+ property->set_type(type); |
+ } |
+ |
+ for (std::vector<MenuItem>::const_iterator child = item.children.begin(); |
+ child != item.children.end(); ++child) { |
+ ibus::IBusProperty* new_property = new ibus::IBusProperty(); |
+ if (!MenuItemToProperty(*child, new_property)) { |
+ delete new_property; |
+ DVLOG(1) << "Bad menu item child"; |
+ return false; |
+ } |
+ property->mutable_sub_properties()->push_back(new_property); |
+ } |
+ |
+ return true; |
+} |
+ |
+void InputMethodEngineIBus::OnConnected() { |
+ RegisterComponent(); |
+} |
+ |
+void InputMethodEngineIBus::OnDisconnected() { |
+} |
+ |
+bool InputMethodEngineIBus::IsConnected() { |
+ return DBusThreadManager::Get()->GetIBusClient() != NULL; |
+} |
+ |
+void InputMethodEngineIBus::RegisterComponent() { |
+ chromeos::IBusClient* client = |
+ chromeos::DBusThreadManager::Get()->GetIBusClient(); |
+ client->RegisterComponent( |
+ *component_.get(), |
+ base::Bind(&InputMethodEngineIBus::OnComponentRegistered, |
+ weak_ptr_factory_.GetWeakPtr()), |
+ base::Bind(&InputMethodEngineIBus::OnComponentRegistrationFailed, |
+ weak_ptr_factory_.GetWeakPtr())); |
+} |
+ |
+void InputMethodEngineIBus::OnComponentRegistered() { |
+ DBusThreadManager::Get()->GetIBusEngineFactoryService()-> |
+ SetCreateEngineHandler(ibus_id_, |
+ base::Bind( |
+ &InputMethodEngineIBus::CreateEngineHandler, |
+ weak_ptr_factory_.GetWeakPtr())); |
+} |
+ |
+void InputMethodEngineIBus::OnComponentRegistrationFailed() { |
+ // TODO(nona): Implement error handling. |
Zachary Kuznia
2012/08/07 06:41:04
Log an error here.
Seigo Nonaka
2012/08/07 17:39:21
Done.
|
+} |
+ |
+void InputMethodEngineIBus::CreateEngineHandler( |
+ const IBusEngineFactoryService::CreateEngineResponseSender& sender) { |
+ DBusThreadManager::Get()->RemoveIBusEngineService(object_path_); |
+ |
+ const std::string kObjectPathPrefix = "/org/freedesktop/IBus/Engine/"; |
Zachary Kuznia
2012/08/07 06:41:04
Put this constant at the top of the file.
Seigo Nonaka
2012/08/07 17:39:21
Done.
|
+ current_object_path_++; |
+ object_path_ = dbus::ObjectPath(kObjectPathPrefix + |
+ base::IntToString(current_object_path_)); |
+ GetCurrentService()->Initialize(this); |
+ sender.Run(object_path_); |
+} |
+ |
+} // namespace chromeos |