blob: 2af7f8bb3c90a96adcb90d7e2294d20f6bd3cacf [file] [log] [blame]
// 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.";
const char* kObjectPathPrefix = "/org/freedesktop/IBus/Engine/";
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) {
}
InputMethodEngineIBus::~InputMethodEngineIBus() {
GetCurrentService()->UnsetEngine();
input_method::InputMethodManager::GetInstance()->
RemoveInputMethodExtension(ibus_id_);
}
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();
}
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() {
DVLOG(1) << "Failed to register input method components.";
// TODO(nona): Implement error handling.
}
void InputMethodEngineIBus::CreateEngineHandler(
const IBusEngineFactoryService::CreateEngineResponseSender& sender) {
DBusThreadManager::Get()->RemoveIBusEngineService(object_path_);
current_object_path_++;
object_path_ = dbus::ObjectPath(kObjectPathPrefix +
base::IntToString(current_object_path_));
GetCurrentService()->SetEngine(this);
sender.Run(object_path_);
}
} // namespace chromeos