Chromium Code Reviews

Side by Side Diff: chrome/browser/chromeos/cros/input_method_library.cc

Issue 6032005: Fix UI-thread blocking issue in SetImeConfig (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: review fixes Created 9 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments.
Jump to:
View unified diff | | Annotate | Revision Log
OLDNEW
1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #include "chrome/browser/chromeos/cros/input_method_library.h" 5 #include "chrome/browser/chromeos/cros/input_method_library.h"
6 6
7 #include <glib.h> 7 #include <glib.h>
8 #include <signal.h> 8 #include <signal.h>
9 9
10 #include "unicode/uloc.h" 10 #include "unicode/uloc.h"
(...skipping 45 matching lines...)
56 : input_method_status_connection_(NULL), 56 : input_method_status_connection_(NULL),
57 previous_input_method_("", "", "", ""), 57 previous_input_method_("", "", "", ""),
58 current_input_method_("", "", "", ""), 58 current_input_method_("", "", "", ""),
59 should_launch_ime_(false), 59 should_launch_ime_(false),
60 ime_connected_(false), 60 ime_connected_(false),
61 defer_ime_startup_(false), 61 defer_ime_startup_(false),
62 enable_auto_ime_shutdown_(true), 62 enable_auto_ime_shutdown_(true),
63 should_change_input_method_(false), 63 should_change_input_method_(false),
64 ibus_daemon_process_id_(0), 64 ibus_daemon_process_id_(0),
65 candidate_window_process_id_(0) { 65 candidate_window_process_id_(0) {
66 // TODO(yusukes): Using both CreateFallbackInputMethodDescriptors and
67 // chromeos::GetHardwareKeyboardLayoutName doesn't look clean. Probably
68 // we should unify these APIs.
66 scoped_ptr<InputMethodDescriptors> input_method_descriptors( 69 scoped_ptr<InputMethodDescriptors> input_method_descriptors(
67 CreateFallbackInputMethodDescriptors()); 70 CreateFallbackInputMethodDescriptors());
68 current_input_method_ = input_method_descriptors->at(0); 71 current_input_method_ = input_method_descriptors->at(0);
69 if (CrosLibrary::Get()->EnsureLoaded()) { 72 if (CrosLibrary::Get()->EnsureLoaded()) {
70 current_input_method_id_ = chromeos::GetHardwareKeyboardLayoutName(); 73 current_input_method_id_ = chromeos::GetHardwareKeyboardLayoutName();
71 } 74 }
72 // Observe APP_EXITING to stop input method processes gracefully. 75 // Observe APP_EXITING to stop input method processes gracefully.
73 // Note that even if we fail to stop input method processes from 76 // Note that even if we fail to stop input method processes from
74 // Chrome in case of a sudden crash, we have a way to do it from an 77 // Chrome in case of a sudden crash, we have a way to do it from an
75 // upstart script. See crosbug.com/6515 and crosbug.com/6995 for 78 // upstart script. See crosbug.com/6515 and crosbug.com/6995 for
(...skipping 65 matching lines...)
141 scoped_ptr<InputMethodDescriptors> active_input_method_descriptors( 144 scoped_ptr<InputMethodDescriptors> active_input_method_descriptors(
142 GetActiveInputMethods()); 145 GetActiveInputMethods());
143 for (size_t i = 0; i < active_input_method_descriptors->size(); ++i) { 146 for (size_t i = 0; i < active_input_method_descriptors->size(); ++i) {
144 if (active_input_method_descriptors->at(i).id == input_method_id) { 147 if (active_input_method_descriptors->at(i).id == input_method_id) {
145 return true; 148 return true;
146 } 149 }
147 } 150 }
148 return false; 151 return false;
149 } 152 }
150 153
151 bool GetImeConfig(const char* section, const char* config_name, 154 bool GetImeConfig(const std::string& section, const std::string& config_name,
152 ImeConfigValue* out_value) { 155 ImeConfigValue* out_value) {
153 bool success = false; 156 bool success = false;
154 if (EnsureLoadedAndStarted()) { 157 if (EnsureLoadedAndStarted()) {
155 success = chromeos::GetImeConfig(input_method_status_connection_, 158 success = chromeos::GetImeConfig(input_method_status_connection_,
156 section, config_name, out_value); 159 section.c_str(),
160 config_name.c_str(),
161 out_value);
157 } 162 }
158 return success; 163 return success;
159 } 164 }
160 165
161 bool SetImeConfig(const char* section, const char* config_name, 166 bool SetImeConfig(const std::string& section, const std::string& config_name,
162 const ImeConfigValue& value) { 167 const ImeConfigValue& value) {
163 MaybeStartOrStopInputMethodProcesses(section, config_name, value); 168 // Before calling FlushImeConfig(), start input method process if necessary.
169 MaybeStartInputMethodProcesses(section, config_name, value);
164 170
165 const ConfigKeyType key = std::make_pair(section, config_name); 171 const ConfigKeyType key = std::make_pair(section, config_name);
166 current_config_values_[key] = value; 172 current_config_values_[key] = value;
167 if (ime_connected_) { 173 if (ime_connected_) {
168 pending_config_requests_[key] = value; 174 pending_config_requests_[key] = value;
169 FlushImeConfig(); 175 FlushImeConfig();
170 } 176 }
177
178 // Stop input method process if necessary.
179 MaybeStopInputMethodProcesses(section, config_name, value);
171 return pending_config_requests_.empty(); 180 return pending_config_requests_.empty();
172 } 181 }
173 182
174 virtual const InputMethodDescriptor& previous_input_method() const { 183 virtual const InputMethodDescriptor& previous_input_method() const {
175 return previous_input_method_; 184 return previous_input_method_;
176 } 185 }
177 virtual const InputMethodDescriptor& current_input_method() const { 186 virtual const InputMethodDescriptor& current_input_method() const {
178 return current_input_method_; 187 return current_input_method_;
179 } 188 }
180 189
181 virtual const ImePropertyList& current_ime_properties() const { 190 virtual const ImePropertyList& current_ime_properties() const {
182 return current_ime_properties_; 191 return current_ime_properties_;
183 } 192 }
184 193
185 private: 194 private:
186 // Starts or stops the input method processes based on the current state. 195 // Starts input method processes based on the |defer_ime_startup_| flag and
187 void MaybeStartOrStopInputMethodProcesses( 196 // input method configuration being updated. |section| is a section name of
188 const char* section, 197 // the input method configuration (e.g. "general", "general/hotkey").
189 const char* config_name, 198 // |config_name| is a name of the configuration (e.g. "preload_engines",
190 const ImeConfigValue& value) { 199 // "previous_engine"). |value| is the configuration value to be set.
191 if (!strcmp(language_prefs::kGeneralSectionName, section) && 200 void MaybeStartInputMethodProcesses(const std::string& section,
192 !strcmp(language_prefs::kPreloadEnginesConfigName, config_name)) { 201 const std::string& config_name,
202 const ImeConfigValue& value) {
203 if (section == language_prefs::kGeneralSectionName &&
204 config_name == language_prefs::kPreloadEnginesConfigName) {
193 if (EnsureLoadedAndStarted()) { 205 if (EnsureLoadedAndStarted()) {
194 // If there are no input methods other than one for the hardware 206 const std::string hardware_layout_name =
195 // keyboard, we'll stop the input method processes. 207 chromeos::GetHardwareKeyboardLayoutName(); // e.g. "xkb:us::eng"
196 if (value.type == ImeConfigValue::kValueTypeStringList && 208 if (!(value.type == ImeConfigValue::kValueTypeStringList &&
197 value.string_list_value.size() == 1 && 209 value.string_list_value.size() == 1 &&
198 value.string_list_value[0] == 210 value.string_list_value[0] == hardware_layout_name) &&
199 chromeos::GetHardwareKeyboardLayoutName()) { 211 !defer_ime_startup_) {
200 if (enable_auto_ime_shutdown_) 212 // If there are no input methods other than one for the hardware
201 StopInputMethodProcesses(); 213 // keyboard, we don't start the input method processes.
202 } else if (!defer_ime_startup_) { 214 // When |defer_ime_startup_| is true, we don't start it either.
203 StartInputMethodProcesses(); 215 StartInputMethodProcesses();
204 } 216 }
205 chromeos::SetActiveInputMethods(input_method_status_connection_, value); 217 chromeos::SetActiveInputMethods(input_method_status_connection_, value);
206 } 218 }
219 }
220 }
221
222 // Stops input method processes based on the |enable_auto_ime_shutdown_| flag
223 // and input method configuration being updated.
224 // See also: MaybeStartInputMethodProcesses().
225 void MaybeStopInputMethodProcesses(const std::string& section,
226 const std::string& config_name,
227 const ImeConfigValue& value) {
228 if (section == language_prefs::kGeneralSectionName &&
229 config_name == language_prefs::kPreloadEnginesConfigName) {
230 if (EnsureLoadedAndStarted()) {
231 const std::string hardware_layout_name =
232 chromeos::GetHardwareKeyboardLayoutName(); // e.g. "xkb:us::eng"
233 if (value.type == ImeConfigValue::kValueTypeStringList &&
234 value.string_list_value.size() == 1 &&
235 value.string_list_value[0] == hardware_layout_name &&
236 enable_auto_ime_shutdown_) {
237 // If there are no input methods other than one for the hardware
238 // keyboard, and |enable_auto_ime_shutdown_| is true, we'll stop the
239 // input method processes.
240 StopInputMethodProcesses();
241 }
242 chromeos::SetActiveInputMethods(input_method_status_connection_, value);
243 }
207 } 244 }
208 } 245 }
209 246
210 // Changes the current input method to |input_method_id|. If the id is not in 247 // Changes the current input method to |input_method_id|. If the id is not in
211 // the preload_engine list, this function changes the current method to the 248 // the preload_engine list, this function changes the current method to the
212 // first preloaded engine. Returns true if the current engine is switched to 249 // first preloaded engine. Returns true if the current engine is switched to
213 // |input_method_id| or the first one. 250 // |input_method_id| or the first one.
214 bool ChangeInputMethodInternal(const std::string& input_method_id) { 251 bool ChangeInputMethodInternal(const std::string& input_method_id) {
215 DCHECK(EnsureLoadedAndStarted()); 252 DCHECK(EnsureLoadedAndStarted());
216 std::string input_method_id_to_switch = input_method_id; 253 std::string input_method_id_to_switch = input_method_id;
(...skipping 80 matching lines...)
297 current_input_method_, 334 current_input_method_,
298 num_active_input_methods)); 335 num_active_input_methods));
299 } 336 }
300 } 337 }
301 338
302 static void InputMethodChangedHandler( 339 static void InputMethodChangedHandler(
303 void* object, 340 void* object,
304 const chromeos::InputMethodDescriptor& current_input_method) { 341 const chromeos::InputMethodDescriptor& current_input_method) {
305 // The handler is called when the input method method change is 342 // The handler is called when the input method method change is
306 // notified via a DBus connection. Since the DBus notificatiosn are 343 // notified via a DBus connection. Since the DBus notificatiosn are
307 // handled in the UI thread, we can assume that this functionalways 344 // handled in the UI thread, we can assume that this function always
308 // runs on the UI thread, but just in case. 345 // runs on the UI thread, but just in case.
309 if (!BrowserThread::CurrentlyOn(BrowserThread::UI)) { 346 if (!BrowserThread::CurrentlyOn(BrowserThread::UI)) {
310 LOG(ERROR) << "Not on UI thread"; 347 LOG(ERROR) << "Not on UI thread";
311 return; 348 return;
312 } 349 }
313 350
314 InputMethodLibraryImpl* input_method_library = 351 InputMethodLibraryImpl* input_method_library =
315 static_cast<InputMethodLibraryImpl*>(object); 352 static_cast<InputMethodLibraryImpl*>(object);
316 input_method_library->ChangeCurrentInputMethod(current_input_method); 353 input_method_library->ChangeCurrentInputMethod(current_input_method);
317 } 354 }
(...skipping 261 matching lines...)
579 // This is used to register this object to APP_EXITING notification. 616 // This is used to register this object to APP_EXITING notification.
580 NotificationRegistrar notification_registrar_; 617 NotificationRegistrar notification_registrar_;
581 618
582 // True if we should launch the input method processes. 619 // True if we should launch the input method processes.
583 bool should_launch_ime_; 620 bool should_launch_ime_;
584 // True if the connection to the IBus daemon is alive. 621 // True if the connection to the IBus daemon is alive.
585 bool ime_connected_; 622 bool ime_connected_;
586 // If true, we'll defer the startup until a non-default method is 623 // If true, we'll defer the startup until a non-default method is
587 // activated. 624 // activated.
588 bool defer_ime_startup_; 625 bool defer_ime_startup_;
626 // True if we should stop input method processes when there are no input
627 // methods other than one for the hardware keyboard.
589 bool enable_auto_ime_shutdown_; 628 bool enable_auto_ime_shutdown_;
590 // The ID of the current input method (ex. "mozc"). 629 // The ID of the current input method (ex. "mozc").
591 std::string current_input_method_id_; 630 std::string current_input_method_id_;
592 // True if we should change the input method once the queue of the 631 // True if we should change the input method once the queue of the
593 // pending config requests becomes empty. 632 // pending config requests becomes empty.
594 bool should_change_input_method_; 633 bool should_change_input_method_;
595 634
596 // The process id of the IBus daemon. 0 if it's not running. The process 635 // The process id of the IBus daemon. 0 if it's not running. The process
597 // ID 0 is not used in Linux, hence it's safe to use 0 for this purpose. 636 // ID 0 is not used in Linux, hence it's safe to use 0 for this purpose.
598 int ibus_daemon_process_id_; 637 int ibus_daemon_process_id_;
(...skipping 31 matching lines...)
630 return CreateRealisticInputMethodDescriptors(); 669 return CreateRealisticInputMethodDescriptors();
631 } 670 }
632 671
633 void ChangeInputMethod(const std::string& input_method_id) {} 672 void ChangeInputMethod(const std::string& input_method_id) {}
634 void SetImePropertyActivated(const std::string& key, bool activated) {} 673 void SetImePropertyActivated(const std::string& key, bool activated) {}
635 674
636 bool InputMethodIsActivated(const std::string& input_method_id) { 675 bool InputMethodIsActivated(const std::string& input_method_id) {
637 return true; 676 return true;
638 } 677 }
639 678
640 bool GetImeConfig(const char* section, 679 bool GetImeConfig(const std::string& section,
641 const char* config_name, 680 const std::string& config_name,
642 ImeConfigValue* out_value) { 681 ImeConfigValue* out_value) {
643 return false; 682 return false;
644 } 683 }
645 684
646 bool SetImeConfig(const char* section, 685 bool SetImeConfig(const std::string& section,
647 const char* config_name, 686 const std::string& config_name,
648 const ImeConfigValue& value) { 687 const ImeConfigValue& value) {
649 return false; 688 return false;
650 } 689 }
651 690
652 virtual const InputMethodDescriptor& previous_input_method() const { 691 virtual const InputMethodDescriptor& previous_input_method() const {
653 return previous_input_method_; 692 return previous_input_method_;
654 } 693 }
655 694
656 virtual const InputMethodDescriptor& current_input_method() const { 695 virtual const InputMethodDescriptor& current_input_method() const {
657 return current_input_method_; 696 return current_input_method_;
(...skipping 150 matching lines...)
808 return new InputMethodLibraryStubImpl(); 847 return new InputMethodLibraryStubImpl();
809 else 848 else
810 return new InputMethodLibraryImpl(); 849 return new InputMethodLibraryImpl();
811 } 850 }
812 851
813 } // namespace chromeos 852 } // namespace chromeos
814 853
815 // Allows InvokeLater without adding refcounting. This class is a Singleton and 854 // Allows InvokeLater without adding refcounting. This class is a Singleton and
816 // won't be deleted until it's last InvokeLater is run. 855 // won't be deleted until it's last InvokeLater is run.
817 DISABLE_RUNNABLE_METHOD_REFCOUNT(chromeos::InputMethodLibraryImpl); 856 DISABLE_RUNNABLE_METHOD_REFCOUNT(chromeos::InputMethodLibraryImpl);
OLDNEW
« no previous file with comments | « chrome/browser/chromeos/cros/input_method_library.h ('k') | chrome/browser/chromeos/cros/mock_input_method_library.h » ('j') | no next file with comments »

Powered by Google App Engine