Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 2013 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/input_method/component_extension_ime_manager_i mpl.h" | 5 #include "chrome/browser/chromeos/input_method/component_extension_ime_manager_i mpl.h" |
| 6 | 6 |
| 7 #include <algorithm> | |
| 8 | |
| 7 #include "base/file_util.h" | 9 #include "base/file_util.h" |
| 10 #include "base/json/json_string_value_serializer.h" | |
| 8 #include "base/logging.h" | 11 #include "base/logging.h" |
| 9 #include "base/path_service.h" | 12 #include "base/path_service.h" |
| 10 #include "chrome/browser/extensions/component_loader.h" | 13 #include "chrome/browser/extensions/component_loader.h" |
| 11 #include "chrome/browser/extensions/extension_service.h" | 14 #include "chrome/browser/extensions/extension_service.h" |
| 12 #include "chrome/browser/profiles/profile.h" | 15 #include "chrome/browser/profiles/profile.h" |
| 13 #include "chrome/browser/profiles/profile_manager.h" | 16 #include "chrome/browser/profiles/profile_manager.h" |
| 14 #include "chrome/common/chrome_paths.h" | 17 #include "chrome/common/chrome_paths.h" |
| 15 #include "chrome/common/extensions/extension_constants.h" | 18 #include "chrome/common/extensions/extension_constants.h" |
| 16 #include "chrome/common/extensions/extension_file_util.h" | 19 #include "chrome/common/extensions/extension_file_util.h" |
| 17 #include "chromeos/ime/extension_ime_util.h" | 20 #include "chromeos/ime/extension_ime_util.h" |
| 18 #include "content/public/browser/browser_thread.h" | |
| 19 #include "extensions/browser/extension_system.h" | 21 #include "extensions/browser/extension_system.h" |
| 20 #include "extensions/common/extension.h" | 22 #include "extensions/common/extension.h" |
| 21 #include "extensions/common/extension_l10n_util.h" | 23 #include "extensions/common/extension_l10n_util.h" |
| 22 #include "extensions/common/file_util.h" | |
| 23 #include "extensions/common/manifest_constants.h" | 24 #include "extensions/common/manifest_constants.h" |
| 25 #include "grit/browser_resources.h" | |
| 26 #include "grit/generated_resources.h" | |
| 24 #include "ui/base/l10n/l10n_util.h" | 27 #include "ui/base/l10n/l10n_util.h" |
| 28 #include "ui/base/resource/resource_bundle.h" | |
| 25 | 29 |
| 26 namespace chromeos { | 30 namespace chromeos { |
| 27 | 31 |
| 28 namespace { | 32 namespace { |
| 29 | 33 |
| 30 struct WhitelistedComponentExtensionIME { | 34 struct WhitelistedComponentExtensionIME { |
|
nona
2014/07/14 06:03:48
Do we still needs this whitelist?
We can choose th
Shu Chen
2014/07/14 08:53:44
Removed #ifdef.
| |
| 31 const char* id; | 35 const char* id; |
| 32 const char* path; | 36 const char* path; |
|
nona
2014/07/14 06:46:07
Can we move this entry into corresponding manifest
Shu Chen
2014/07/14 08:53:44
Done.
| |
| 37 int manifest_resource_id; | |
| 33 } whitelisted_component_extension[] = { | 38 } whitelisted_component_extension[] = { |
| 34 { | 39 {// ChromeOS Hangul Input. |
| 35 // ChromeOS Hangul Input. | 40 extension_ime_util::kHangulExtensionId, |
| 36 extension_ime_util::kHangulExtensionId, | 41 "/usr/share/chromeos-assets/input_methods/hangul", IDR_HANGUL_MANIFEST, |
| 37 "/usr/share/chromeos-assets/input_methods/hangul", | 42 }, |
| 38 }, | |
| 39 #if defined(OFFICIAL_BUILD) | 43 #if defined(OFFICIAL_BUILD) |
| 40 { | 44 {// Official Google XKB Input. |
| 41 // Official Google XKB Input. | 45 extension_ime_util::kXkbExtensionId, |
| 42 extension_ime_util::kXkbExtensionId, | 46 "/usr/share/chromeos-assets/input_methods/google_xkb", |
| 43 "/usr/share/chromeos-assets/input_methods/google_xkb", | 47 IDR_GOOGLE_XKB_MANIFEST, |
| 44 }, | 48 }, |
| 45 { | 49 {// Google input tools. |
| 46 // Google input tools. | 50 extension_ime_util::kT13nExtensionId, |
| 47 extension_ime_util::kT13nExtensionId, | 51 "/usr/share/chromeos-assets/input_methods/input_tools", |
| 48 "/usr/share/chromeos-assets/input_methods/input_tools", | 52 IDR_GOOGLE_INPUT_TOOLS_MANIFEST, |
| 49 }, | 53 }, |
| 50 #else | 54 #else |
| 51 { | 55 {// Open-sourced ChromeOS xkb extension. |
| 52 // Open-sourced ChromeOS xkb extension. | 56 extension_ime_util::kXkbExtensionId, |
| 53 extension_ime_util::kXkbExtensionId, | 57 "/usr/share/chromeos-assets/input_methods/xkb", IDR_XKB_MANIFEST, |
| 54 "/usr/share/chromeos-assets/input_methods/xkb", | 58 }, |
| 55 }, | 59 {// Open-sourced ChromeOS Keyboards extension. |
| 56 { | 60 extension_ime_util::kM17nExtensionId, |
| 57 // Open-sourced ChromeOS Keyboards extension. | 61 "/usr/share/chromeos-assets/input_methods/keyboard_layouts", |
| 58 extension_ime_util::kM17nExtensionId, | 62 IDR_M17N_MANIFEST, |
| 59 "/usr/share/chromeos-assets/input_methods/keyboard_layouts", | 63 }, |
| 60 }, | 64 {// Open-sourced Pinyin Chinese Input Method. |
| 61 { | 65 extension_ime_util::kChinesePinyinExtensionId, |
| 62 // Open-sourced Pinyin Chinese Input Method. | 66 "/usr/share/chromeos-assets/input_methods/pinyin", IDR_PINYIN_MANIFEST, |
| 63 extension_ime_util::kChinesePinyinExtensionId, | 67 }, |
| 64 "/usr/share/chromeos-assets/input_methods/pinyin", | 68 {// Open-sourced Zhuyin Chinese Input Method. |
| 65 }, | 69 extension_ime_util::kChineseZhuyinExtensionId, |
| 66 { | 70 "/usr/share/chromeos-assets/input_methods/zhuyin", IDR_ZHUYIN_MANIFEST, |
| 67 // Open-sourced Zhuyin Chinese Input Method. | 71 }, |
| 68 extension_ime_util::kChineseZhuyinExtensionId, | 72 {// Open-sourced Cangjie Chinese Input Method. |
| 69 "/usr/share/chromeos-assets/input_methods/zhuyin", | 73 extension_ime_util::kChineseCangjieExtensionId, |
| 70 }, | 74 "/usr/share/chromeos-assets/input_methods/cangjie", IDR_CANGJIE_MANIFEST, |
| 71 { | 75 }, |
| 72 // Open-sourced Cangjie Chinese Input Method. | 76 {// Japanese Mozc Input. |
| 73 extension_ime_util::kChineseCangjieExtensionId, | 77 extension_ime_util::kMozcExtensionId, |
| 74 "/usr/share/chromeos-assets/input_methods/cangjie", | 78 "/usr/share/chromeos-assets/input_methods/nacl_mozc", IDR_MOZC_MANIFEST, |
| 75 }, | 79 }, |
| 76 { | |
| 77 // Japanese Mozc Input. | |
| 78 extension_ime_util::kMozcExtensionId, | |
| 79 "/usr/share/chromeos-assets/input_methods/nacl_mozc", | |
| 80 }, | |
| 81 #endif | 80 #endif |
| 82 { | 81 {// Braille hardware keyboard IME that works together with ChromeVox. |
| 83 // Braille hardware keyboard IME that works together with ChromeVox. | 82 extension_misc::kBrailleImeExtensionId, |
| 84 extension_misc::kBrailleImeExtensionId, | 83 extension_misc::kBrailleImeExtensionPath, IDR_BRAILLE_MANIFEST, |
| 85 extension_misc::kBrailleImeExtensionPath, | 84 }, |
| 86 }, | 85 }; |
| 86 | |
| 87 const struct InputMethodNameMap { | |
| 88 const char* message_name; | |
| 89 int resource_id; | |
| 90 bool operator()(const InputMethodNameMap& m1, | |
| 91 const InputMethodNameMap& m2) const { | |
| 92 return strcmp(m1.message_name, m2.message_name) < 0; | |
| 93 } | |
| 94 } kInputMethodNameMap[] = { | |
| 95 {"__MSG_ARMENIAN_PHONETIC_KEYBOARD__", | |
| 96 IDS_IME_NAME_ARMENIAN_PHONETIC_KEYBOARD}, | |
| 97 {"__MSG_ARRAY_INPUTMETHOD__", IDS_IME_NAME_ARRAY_INPUTMETHOD}, | |
| 98 {"__MSG_BELARUSIAN_KEYBOARD__", IDS_IME_NAME_BELARUSIAN_KEYBOARD}, | |
| 99 {"__MSG_BELGIAN_KEYBOARD__", IDS_IME_NAME_BELGIAN_KEYBOARD}, | |
| 100 {"__MSG_BRAZILIAN_KEYBOARD__", IDS_IME_NAME_BRAZILIAN_KEYBOARD}, | |
| 101 {"__MSG_BULGARIAN_KEYBOARD__", IDS_IME_NAME_BULGARIAN_KEYBOARD}, | |
| 102 {"__MSG_BULGARIAN_PHONETIC_KEYBOARD__", | |
| 103 IDS_IME_NAME_BULGARIAN_PHONETIC_KEYBOARD}, | |
| 104 {"__MSG_CANADIAN_ENGLISH_KEYBOARD__", | |
| 105 IDS_IME_NAME_CANADIAN_ENGLISH_KEYBOARD}, | |
| 106 {"__MSG_CANADIAN_FRENCH_KEYBOARD__", | |
| 107 IDS_IME_NAME_CANADIAN_FRENCH_KEYBOARD}, | |
| 108 {"__MSG_CANADIAN_MULTILINGUAL_KEYBOARD__", | |
| 109 IDS_IME_NAME_CANADIAN_MULTILINGUAL_KEYBOARD}, | |
| 110 {"__MSG_CANGJIE_INPUTMETHOD__", IDS_IME_NAME_CANGJIE_INPUTMETHOD}, | |
| 111 {"__MSG_CATALAN_KEYBOARD__", IDS_IME_NAME_CATALAN_KEYBOARD}, | |
| 112 {"__MSG_CROATIAN_KEYBOARD__", IDS_IME_NAME_CROATIAN_KEYBOARD}, | |
| 113 {"__MSG_CZECH_KEYBOARD__", IDS_IME_NAME_CZECH_KEYBOARD}, | |
| 114 {"__MSG_CZECH_QWERTY_KEYBOARD__", IDS_IME_NAME_CZECH_QWERTY_KEYBOARD}, | |
| 115 {"__MSG_DANISH_KEYBOARD__", IDS_IME_NAME_DANISH_KEYBOARD}, | |
| 116 {"__MSG_DAYI_INPUTMETHOD__", IDS_IME_NAME_DAYI_INPUTMETHOD}, | |
| 117 {"__MSG_ESTONIAN_KEYBOARD__", IDS_IME_NAME_ESTONIAN_KEYBOARD}, | |
| 118 {"__MSG_FINNISH_KEYBOARD__", IDS_IME_NAME_FINNISH_KEYBOARD}, | |
| 119 {"__MSG_FRENCH_KEYBOARD__", IDS_IME_NAME_FRENCH_KEYBOARD}, | |
| 120 {"__MSG_GEORGIAN_KEYBOARD__", IDS_IME_NAME_GEORGIAN_KEYBOARD}, | |
| 121 {"__MSG_GERMAN_KEYBOARD__", IDS_IME_NAME_GERMAN_KEYBOARD}, | |
| 122 {"__MSG_GERMAN_NEO_2_KEYBOARD__", IDS_IME_NAME_GERMAN_NEO_2_KEYBOARD}, | |
| 123 {"__MSG_GREEK_KEYBOARD__", IDS_IME_NAME_GREEK_KEYBOARD}, | |
| 124 {"__MSG_HANGUL_2_SET__", IDS_IME_NAME_HANGUL_2_SET}, | |
| 125 {"__MSG_HANGUL_3_SET_390__", IDS_IME_NAME_HANGUL_3_SET_390}, | |
| 126 {"__MSG_HANGUL_3_SET_FINAL__", IDS_IME_NAME_HANGUL_3_SET_FINAL}, | |
| 127 {"__MSG_HANGUL_3_SET_NO_SHIFT__", IDS_IME_NAME_HANGUL_3_SET_NO_SHIFT}, | |
| 128 {"__MSG_HANGUL_AHNMATAE__", IDS_IME_NAME_HANGUL_AHNMATAE}, | |
| 129 {"__MSG_HANGUL_ROMAJA__", IDS_IME_NAME_HANGUL_ROMAJA}, | |
| 130 {"__MSG_HEBREW_KEYBOARD__", IDS_IME_NAME_HEBREW_KEYBOARD}, | |
| 131 {"__MSG_HUNGARIAN_KEYBOARD__", IDS_IME_NAME_HUNGARIAN_KEYBOARD}, | |
| 132 {"__MSG_ICELANDIC_KEYBOARD__", IDS_IME_NAME_ICELANDIC_KEYBOARD}, | |
| 133 {"__MSG_IRISH_KEYBOARD__", IDS_IME_NAME_IRISH_KEYBOARD}, | |
| 134 {"__MSG_ITALIAN_KEYBOARD__", IDS_IME_NAME_ITALIAN_KEYBOARD}, | |
| 135 {"__MSG_JAPANESE_KEYBOARD__", IDS_IME_NAME_JAPANESE_KEYBOARD}, | |
| 136 {"__MSG_KEYBOARD_ARABIC__", IDS_IME_NAME_KEYBOARD_ARABIC}, | |
| 137 {"__MSG_KEYBOARD_BENGALI_PHONETIC__", | |
| 138 IDS_IME_NAME_KEYBOARD_BENGALI_PHONETIC}, | |
| 139 {"__MSG_KEYBOARD_DEVANAGARI_PHONETIC__", | |
| 140 IDS_IME_NAME_KEYBOARD_DEVANAGARI_PHONETIC}, | |
| 141 {"__MSG_KEYBOARD_ETHIOPIC__", IDS_IME_NAME_KEYBOARD_ETHIOPIC}, | |
| 142 {"__MSG_KEYBOARD_GUJARATI_PHONETIC__", | |
| 143 IDS_IME_NAME_KEYBOARD_GUJARATI_PHONETIC}, | |
| 144 {"__MSG_KEYBOARD_KANNADA_PHONETIC__", | |
| 145 IDS_IME_NAME_KEYBOARD_KANNADA_PHONETIC}, | |
| 146 {"__MSG_KEYBOARD_KHMER__", IDS_IME_NAME_KEYBOARD_KHMER}, | |
| 147 {"__MSG_KEYBOARD_LAO__", IDS_IME_NAME_KEYBOARD_LAO}, | |
| 148 {"__MSG_KEYBOARD_MALAYALAM_PHONETIC__", | |
| 149 IDS_IME_NAME_KEYBOARD_MALAYALAM_PHONETIC}, | |
| 150 {"__MSG_KEYBOARD_MYANMAR_MYANSAN__", | |
| 151 IDS_IME_NAME_KEYBOARD_MYANMAR_MYANSAN}, | |
| 152 {"__MSG_KEYBOARD_MYANMAR__", IDS_IME_NAME_KEYBOARD_MYANMAR}, | |
| 153 {"__MSG_KEYBOARD_NEPALI_INSCRIPT__", | |
| 154 IDS_IME_NAME_KEYBOARD_NEPALI_INSCRIPT}, | |
| 155 {"__MSG_KEYBOARD_NEPALI_PHONETIC__", | |
| 156 IDS_IME_NAME_KEYBOARD_NEPALI_PHONETIC}, | |
| 157 {"__MSG_KEYBOARD_PERSIAN__", IDS_IME_NAME_KEYBOARD_PERSIAN}, | |
| 158 {"__MSG_KEYBOARD_SINHALA__", IDS_IME_NAME_KEYBOARD_SINHALA}, | |
| 159 {"__MSG_KEYBOARD_SORANIKURDISH_AR__", | |
| 160 IDS_IME_NAME_KEYBOARD_SORANIKURDISH_AR}, | |
| 161 {"__MSG_KEYBOARD_SORANIKURDISH_EN__", | |
| 162 IDS_IME_NAME_KEYBOARD_SORANIKURDISH_EN}, | |
| 163 {"__MSG_KEYBOARD_TAMIL_INSCRIPT__", IDS_IME_NAME_KEYBOARD_TAMIL_INSCRIPT}, | |
| 164 {"__MSG_KEYBOARD_TAMIL_ITRANS__", IDS_IME_NAME_KEYBOARD_TAMIL_ITRANS}, | |
| 165 {"__MSG_KEYBOARD_TAMIL_PHONETIC__", IDS_IME_NAME_KEYBOARD_TAMIL_PHONETIC}, | |
| 166 {"__MSG_KEYBOARD_TAMIL_TAMIL99__", IDS_IME_NAME_KEYBOARD_TAMIL_TAMIL99}, | |
| 167 {"__MSG_KEYBOARD_TAMIL_TYPEWRITER__", | |
| 168 IDS_IME_NAME_KEYBOARD_TAMIL_TYPEWRITER}, | |
| 169 {"__MSG_KEYBOARD_TELUGU_PHONETIC__", | |
| 170 IDS_IME_NAME_KEYBOARD_TELUGU_PHONETIC}, | |
| 171 {"__MSG_KEYBOARD_THAI_KEDMANEE__", IDS_IME_NAME_KEYBOARD_THAI_KEDMANEE}, | |
| 172 {"__MSG_KEYBOARD_THAI_PATTACHOTE__", | |
| 173 IDS_IME_NAME_KEYBOARD_THAI_PATTACHOTE}, | |
| 174 {"__MSG_KEYBOARD_THAI_TIS__", IDS_IME_NAME_KEYBOARD_THAI_TIS}, | |
| 175 {"__MSG_KEYBOARD_VIETNAMESE_TCVN__", | |
| 176 IDS_IME_NAME_KEYBOARD_VIETNAMESE_TCVN}, | |
| 177 {"__MSG_KEYBOARD_VIETNAMESE_TELEX__", | |
| 178 IDS_IME_NAME_KEYBOARD_VIETNAMESE_TELEX}, | |
| 179 {"__MSG_KEYBOARD_VIETNAMESE_VIQR__", | |
| 180 IDS_IME_NAME_KEYBOARD_VIETNAMESE_VIQR}, | |
| 181 {"__MSG_KEYBOARD_VIETNAMESE_VNI__", IDS_IME_NAME_KEYBOARD_VIETNAMESE_VNI}, | |
| 182 {"__MSG_LATIN_AMERICAN_KEYBOARD__", IDS_IME_NAME_LATIN_AMERICAN_KEYBOARD}, | |
| 183 {"__MSG_LATVIAN_KEYBOARD__", IDS_IME_NAME_LATVIAN_KEYBOARD}, | |
| 184 {"__MSG_LITHUANIAN_KEYBOARD__", IDS_IME_NAME_LITHUANIAN_KEYBOARD}, | |
| 185 {"__MSG_MONGOLIAN_KEYBOARD__", IDS_IME_NAME_MONGOLIAN_KEYBOARD}, | |
| 186 {"__MSG_MOZC_JP_INPUTMETHOD__", IDS_IME_NAME_MOZC_JP_INPUTMETHOD}, | |
| 187 {"__MSG_MOZC_US_INPUTMETHOD__", IDS_IME_NAME_MOZC_US_INPUTMETHOD}, | |
| 188 {"__MSG_NORWEGIAN_KEYBOARD__", IDS_IME_NAME_NORWEGIAN_KEYBOARD}, | |
| 189 {"__MSG_PINYIN_INPUTMETHOD__", IDS_IME_NAME_PINYIN_INPUTMETHOD}, | |
| 190 {"__MSG_POLISH_KEYBOARD__", IDS_IME_NAME_POLISH_KEYBOARD}, | |
| 191 {"__MSG_PORTUGUESE_KEYBOARD__", IDS_IME_NAME_PORTUGUESE_KEYBOARD}, | |
| 192 {"__MSG_QUICK_INPUTMETHOD__", IDS_IME_NAME_QUICK_INPUTMETHOD}, | |
| 193 {"__MSG_ROMANIAN_KEYBOARD__", IDS_IME_NAME_ROMANIAN_KEYBOARD}, | |
| 194 {"__MSG_RUSSIAN_KEYBOARD__", IDS_IME_NAME_RUSSIAN_KEYBOARD}, | |
| 195 {"__MSG_RUSSIAN_PHONETIC_KEYBOARD__", | |
| 196 IDS_IME_NAME_RUSSIAN_PHONETIC_KEYBOARD}, | |
| 197 {"__MSG_SERBIAN_KEYBOARD__", IDS_IME_NAME_SERBIAN_KEYBOARD}, | |
| 198 {"__MSG_SLOVAKIAN_KEYBOARD__", IDS_IME_NAME_SLOVAKIAN_KEYBOARD}, | |
| 199 {"__MSG_SLOVENIAN_KEYBOARD__", IDS_IME_NAME_SLOVENIAN_KEYBOARD}, | |
| 200 {"__MSG_SPANISH_KEYBOARD__", IDS_IME_NAME_SPANISH_KEYBOARD}, | |
| 201 {"__MSG_SWEDISH_KEYBOARD__", IDS_IME_NAME_SWEDISH_KEYBOARD}, | |
| 202 {"__MSG_SWISS_FRENCH_KEYBOARD__", IDS_IME_NAME_SWISS_FRENCH_KEYBOARD}, | |
| 203 {"__MSG_SWISS_KEYBOARD__", IDS_IME_NAME_SWISS_KEYBOARD}, | |
| 204 {"__MSG_TRADITIONAL_PINYIN_INPUTMETHOD__", | |
| 205 IDS_IME_NAME_TRADITIONAL_PINYIN_INPUTMETHOD}, | |
| 206 {"__MSG_TRANSLITERATION_AM__", IDS_IME_NAME_TRANSLITERATION_AM}, | |
| 207 {"__MSG_TRANSLITERATION_AR__", IDS_IME_NAME_TRANSLITERATION_AR}, | |
| 208 {"__MSG_TRANSLITERATION_BN__", IDS_IME_NAME_TRANSLITERATION_BN}, | |
| 209 {"__MSG_TRANSLITERATION_EL__", IDS_IME_NAME_TRANSLITERATION_EL}, | |
| 210 {"__MSG_TRANSLITERATION_FA__", IDS_IME_NAME_TRANSLITERATION_FA}, | |
| 211 {"__MSG_TRANSLITERATION_GU__", IDS_IME_NAME_TRANSLITERATION_GU}, | |
| 212 {"__MSG_TRANSLITERATION_HE__", IDS_IME_NAME_TRANSLITERATION_HE}, | |
| 213 {"__MSG_TRANSLITERATION_HI__", IDS_IME_NAME_TRANSLITERATION_HI}, | |
| 214 {"__MSG_TRANSLITERATION_KN__", IDS_IME_NAME_TRANSLITERATION_KN}, | |
| 215 {"__MSG_TRANSLITERATION_ML__", IDS_IME_NAME_TRANSLITERATION_ML}, | |
| 216 {"__MSG_TRANSLITERATION_MR__", IDS_IME_NAME_TRANSLITERATION_MR}, | |
| 217 {"__MSG_TRANSLITERATION_NE__", IDS_IME_NAME_TRANSLITERATION_NE}, | |
| 218 {"__MSG_TRANSLITERATION_OR__", IDS_IME_NAME_TRANSLITERATION_OR}, | |
| 219 {"__MSG_TRANSLITERATION_PA__", IDS_IME_NAME_TRANSLITERATION_PA}, | |
| 220 {"__MSG_TRANSLITERATION_SA__", IDS_IME_NAME_TRANSLITERATION_SA}, | |
| 221 {"__MSG_TRANSLITERATION_SR__", IDS_IME_NAME_TRANSLITERATION_SR}, | |
| 222 {"__MSG_TRANSLITERATION_TA__", IDS_IME_NAME_TRANSLITERATION_TA}, | |
| 223 {"__MSG_TRANSLITERATION_TE__", IDS_IME_NAME_TRANSLITERATION_TE}, | |
| 224 {"__MSG_TRANSLITERATION_TI__", IDS_IME_NAME_TRANSLITERATION_TI}, | |
| 225 {"__MSG_TRANSLITERATION_UR__", IDS_IME_NAME_TRANSLITERATION_UR}, | |
| 226 {"__MSG_TURKISH_KEYBOARD__", IDS_IME_NAME_TURKISH_KEYBOARD}, | |
| 227 {"__MSG_UKRAINIAN_KEYBOARD__", IDS_IME_NAME_UKRAINIAN_KEYBOARD}, | |
| 228 {"__MSG_UK_DVORAK_KEYBOARD__", IDS_IME_NAME_UK_DVORAK_KEYBOARD}, | |
| 229 {"__MSG_UK_KEYBOARD__", IDS_IME_NAME_UK_KEYBOARD}, | |
| 230 {"__MSG_US_COLEMAK_KEYBOARD__", IDS_IME_NAME_US_COLEMAK_KEYBOARD}, | |
| 231 {"__MSG_US_DVORAK_KEYBOARD__", IDS_IME_NAME_US_DVORAK_KEYBOARD}, | |
| 232 {"__MSG_US_EXTENDED_KEYBOARD__", IDS_IME_NAME_US_EXTENDED_KEYBOARD}, | |
| 233 {"__MSG_US_INTERNATIONAL_KEYBOARD__", | |
| 234 IDS_IME_NAME_US_INTERNATIONAL_KEYBOARD}, | |
| 235 {"__MSG_US_KEYBOARD__", IDS_IME_NAME_US_KEYBOARD}, | |
| 236 {"__MSG_WUBI_INPUTMETHOD__", IDS_IME_NAME_WUBI_INPUTMETHOD}, | |
| 237 {"__MSG_ZHUYIN_INPUTMETHOD__", IDS_IME_NAME_ZHUYIN_INPUTMETHOD}, | |
| 87 }; | 238 }; |
| 88 | 239 |
| 89 extensions::ComponentLoader* GetComponentLoader() { | 240 extensions::ComponentLoader* GetComponentLoader() { |
| 90 // TODO(skuhne, nkostylev): At this time the only thing which makes sense here | 241 // TODO(skuhne, nkostylev): At this time the only thing which makes sense here |
| 91 // is to use the active profile. Nkostylev is working on getting IME settings | 242 // is to use the active profile. Nkostylev is working on getting IME settings |
| 92 // to work for multi user by collecting all settings from all users. Once that | 243 // to work for multi user by collecting all settings from all users. Once that |
| 93 // is done we might have to re-visit this decision. | 244 // is done we might have to re-visit this decision. |
| 94 Profile* profile = ProfileManager::GetActiveUserProfile(); | 245 Profile* profile = ProfileManager::GetActiveUserProfile(); |
| 95 extensions::ExtensionSystem* extension_system = | 246 extensions::ExtensionSystem* extension_system = |
| 96 extensions::ExtensionSystem::Get(profile); | 247 extensions::ExtensionSystem::Get(profile); |
| 97 ExtensionService* extension_service = extension_system->extension_service(); | 248 ExtensionService* extension_service = extension_system->extension_service(); |
| 98 return extension_service->component_loader(); | 249 return extension_service->component_loader(); |
| 99 } | 250 } |
| 100 } // namespace | 251 } // namespace |
| 101 | 252 |
| 102 ComponentExtensionIMEManagerImpl::ComponentExtensionIMEManagerImpl() | 253 ComponentExtensionIMEManagerImpl::ComponentExtensionIMEManagerImpl() |
| 103 : is_initialized_(false), | 254 : weak_ptr_factory_(this) { |
| 104 weak_ptr_factory_(this) { | 255 ReadComponentExtensionsInfo(&component_extension_list_); |
| 105 } | 256 } |
| 106 | 257 |
| 107 ComponentExtensionIMEManagerImpl::~ComponentExtensionIMEManagerImpl() { | 258 ComponentExtensionIMEManagerImpl::~ComponentExtensionIMEManagerImpl() { |
| 108 } | 259 } |
| 109 | 260 |
| 110 std::vector<ComponentExtensionIME> ComponentExtensionIMEManagerImpl::ListIME() { | 261 std::vector<ComponentExtensionIME> ComponentExtensionIMEManagerImpl::ListIME() { |
| 111 DCHECK(thread_checker_.CalledOnValidThread()); | |
| 112 return component_extension_list_; | 262 return component_extension_list_; |
| 113 } | 263 } |
| 114 | 264 |
| 115 bool ComponentExtensionIMEManagerImpl::Load(const std::string& extension_id, | 265 bool ComponentExtensionIMEManagerImpl::Load(const std::string& extension_id, |
| 116 const std::string& manifest, | 266 const std::string& manifest, |
| 117 const base::FilePath& file_path) { | 267 const base::FilePath& file_path) { |
| 118 DCHECK(thread_checker_.CalledOnValidThread()); | 268 if (!base::PathExists(file_path)) |
| 269 return false; | |
| 270 | |
| 119 Profile* profile = ProfileManager::GetActiveUserProfile(); | 271 Profile* profile = ProfileManager::GetActiveUserProfile(); |
| 120 extensions::ExtensionSystem* extension_system = | 272 extensions::ExtensionSystem* extension_system = |
| 121 extensions::ExtensionSystem::Get(profile); | 273 extensions::ExtensionSystem::Get(profile); |
| 122 ExtensionService* extension_service = extension_system->extension_service(); | 274 ExtensionService* extension_service = extension_system->extension_service(); |
| 123 if (extension_service->GetExtensionById(extension_id, false)) | 275 if (extension_service->GetExtensionById(extension_id, false)) |
| 124 return false; | 276 return false; |
| 125 const std::string loaded_extension_id = | 277 const std::string loaded_extension_id = |
| 126 GetComponentLoader()->Add(manifest, file_path); | 278 GetComponentLoader()->Add(manifest, file_path); |
| 127 DCHECK_EQ(loaded_extension_id, extension_id); | 279 DCHECK_EQ(loaded_extension_id, extension_id); |
| 128 return true; | 280 return true; |
| 129 } | 281 } |
| 130 | 282 |
| 131 void ComponentExtensionIMEManagerImpl::Unload(const std::string& extension_id, | 283 void ComponentExtensionIMEManagerImpl::Unload(const std::string& extension_id, |
| 132 const base::FilePath& file_path) { | 284 const base::FilePath& file_path) { |
| 133 DCHECK(thread_checker_.CalledOnValidThread()); | |
| 134 // Remove(extension_id) does nothing when the extension has already been | 285 // Remove(extension_id) does nothing when the extension has already been |
| 135 // removed or not been registered. | 286 // removed or not been registered. |
| 136 GetComponentLoader()->Remove(extension_id); | 287 GetComponentLoader()->Remove(extension_id); |
| 137 } | 288 } |
| 138 | 289 |
| 139 scoped_ptr<base::DictionaryValue> ComponentExtensionIMEManagerImpl::GetManifest( | 290 scoped_ptr<base::DictionaryValue> ComponentExtensionIMEManagerImpl::GetManifest( |
| 140 const base::FilePath& file_path) { | 291 const std::string& manifest_string) { |
| 141 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::FILE)); | |
| 142 std::string error; | 292 std::string error; |
| 143 scoped_ptr<base::DictionaryValue> manifest( | 293 JSONStringValueSerializer serializer(manifest_string); |
| 144 extensions::file_util::LoadManifest(file_path, &error)); | 294 scoped_ptr<base::Value> manifest(serializer.Deserialize(NULL, &error)); |
| 145 if (!manifest.get()) | 295 if (!manifest.get()) |
| 146 LOG(ERROR) << "Failed at getting manifest"; | 296 LOG(ERROR) << "Failed at getting manifest"; |
| 147 if (!extension_l10n_util::LocalizeExtension(file_path, | 297 if (!manifest->IsType(base::Value::TYPE_DICTIONARY)) |
| 148 manifest.get(), | 298 LOG(ERROR) << "Invalid manifest format"; |
| 149 &error)) | |
| 150 LOG(ERROR) << "Localization failed"; | |
| 151 | 299 |
| 152 return manifest.Pass(); | 300 return scoped_ptr<base::DictionaryValue>( |
| 153 } | 301 static_cast<base::DictionaryValue*>(manifest.release())).Pass(); |
| 154 | |
| 155 void ComponentExtensionIMEManagerImpl::InitializeAsync( | |
| 156 const base::Closure& callback) { | |
| 157 DCHECK(!is_initialized_); | |
| 158 DCHECK(thread_checker_.CalledOnValidThread()); | |
| 159 | |
| 160 std::vector<ComponentExtensionIME>* component_extension_ime_list | |
| 161 = new std::vector<ComponentExtensionIME>; | |
| 162 content::BrowserThread::PostTaskAndReply( | |
| 163 content::BrowserThread::FILE, | |
| 164 FROM_HERE, | |
| 165 base::Bind(&ComponentExtensionIMEManagerImpl::ReadComponentExtensionsInfo, | |
| 166 base::Unretained(component_extension_ime_list)), | |
| 167 base::Bind( | |
| 168 &ComponentExtensionIMEManagerImpl::OnReadComponentExtensionsInfo, | |
| 169 weak_ptr_factory_.GetWeakPtr(), | |
| 170 base::Owned(component_extension_ime_list), | |
| 171 callback)); | |
| 172 } | |
| 173 | |
| 174 bool ComponentExtensionIMEManagerImpl::IsInitialized() { | |
| 175 return is_initialized_; | |
| 176 } | 302 } |
| 177 | 303 |
| 178 // static | 304 // static |
| 179 bool ComponentExtensionIMEManagerImpl::ReadEngineComponent( | 305 bool ComponentExtensionIMEManagerImpl::ReadEngineComponent( |
| 180 const ComponentExtensionIME& component_extension, | 306 const ComponentExtensionIME& component_extension, |
| 181 const base::DictionaryValue& dict, | 307 const base::DictionaryValue& dict, |
| 182 ComponentExtensionEngine* out) { | 308 ComponentExtensionEngine* out) { |
| 183 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::FILE)); | |
| 184 DCHECK(out); | 309 DCHECK(out); |
| 185 std::string type; | 310 std::string type; |
| 186 if (!dict.GetString(extensions::manifest_keys::kType, &type)) | 311 if (!dict.GetString(extensions::manifest_keys::kType, &type)) |
| 187 return false; | 312 return false; |
| 188 if (type != "ime") | 313 if (type != "ime") |
| 189 return false; | 314 return false; |
| 190 if (!dict.GetString(extensions::manifest_keys::kId, &out->engine_id)) | 315 if (!dict.GetString(extensions::manifest_keys::kId, &out->engine_id)) |
| 191 return false; | 316 return false; |
| 192 if (!dict.GetString(extensions::manifest_keys::kName, &out->display_name)) | 317 if (!dict.GetString(extensions::manifest_keys::kName, &out->display_name)) |
| 193 return false; | 318 return false; |
| 194 | 319 |
| 320 // Localizes the input method name. | |
| 321 if (out->display_name.find("__MSG_") == 0) { | |
| 322 const InputMethodNameMap* map = kInputMethodNameMap; | |
| 323 size_t map_size = arraysize(kInputMethodNameMap); | |
| 324 std::string name = StringToUpperASCII(out->display_name); | |
| 325 const InputMethodNameMap map_key = {name.c_str(), 0}; | |
| 326 const InputMethodNameMap* p = | |
| 327 std::lower_bound(map, map + map_size, map_key, map_key); | |
| 328 if (p != map + map_size && !map_key(*p, map_key) && !map_key(map_key, *p)) | |
| 329 out->display_name = l10n_util::GetStringUTF8(p->resource_id); | |
| 330 } | |
| 331 | |
| 195 std::set<std::string> languages; | 332 std::set<std::string> languages; |
| 196 const base::Value* language_value = NULL; | 333 const base::Value* language_value = NULL; |
| 197 if (dict.Get(extensions::manifest_keys::kLanguage, &language_value)) { | 334 if (dict.Get(extensions::manifest_keys::kLanguage, &language_value)) { |
| 198 if (language_value->GetType() == base::Value::TYPE_STRING) { | 335 if (language_value->GetType() == base::Value::TYPE_STRING) { |
| 199 std::string language_str; | 336 std::string language_str; |
| 200 language_value->GetAsString(&language_str); | 337 language_value->GetAsString(&language_str); |
| 201 languages.insert(language_str); | 338 languages.insert(language_str); |
| 202 } else if (language_value->GetType() == base::Value::TYPE_LIST) { | 339 } else if (language_value->GetType() == base::Value::TYPE_LIST) { |
| 203 const base::ListValue* language_list = NULL; | 340 const base::ListValue* language_list = NULL; |
| 204 language_value->GetAsList(&language_list); | 341 language_value->GetAsList(&language_list); |
| (...skipping 44 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 249 } | 386 } |
| 250 | 387 |
| 251 return true; | 388 return true; |
| 252 } | 389 } |
| 253 | 390 |
| 254 // static | 391 // static |
| 255 bool ComponentExtensionIMEManagerImpl::ReadExtensionInfo( | 392 bool ComponentExtensionIMEManagerImpl::ReadExtensionInfo( |
| 256 const base::DictionaryValue& manifest, | 393 const base::DictionaryValue& manifest, |
| 257 const std::string& extension_id, | 394 const std::string& extension_id, |
| 258 ComponentExtensionIME* out) { | 395 ComponentExtensionIME* out) { |
| 259 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::FILE)); | |
| 260 if (!manifest.GetString(extensions::manifest_keys::kDescription, | 396 if (!manifest.GetString(extensions::manifest_keys::kDescription, |
| 261 &out->description)) | 397 &out->description)) |
| 262 return false; | 398 return false; |
| 263 std::string url_string; | 399 std::string url_string; |
| 264 if (manifest.GetString(extensions::manifest_keys::kOptionsPage, | 400 if (manifest.GetString(extensions::manifest_keys::kOptionsPage, |
| 265 &url_string)) { | 401 &url_string)) { |
| 266 GURL url = extensions::Extension::GetResourceURL( | 402 GURL url = extensions::Extension::GetResourceURL( |
| 267 extensions::Extension::GetBaseURLFromExtensionId(extension_id), | 403 extensions::Extension::GetBaseURLFromExtensionId(extension_id), |
| 268 url_string); | 404 url_string); |
| 269 if (!url.is_valid()) | 405 if (!url.is_valid()) |
| 270 return false; | 406 return false; |
| 271 out->options_page_url = url; | 407 out->options_page_url = url; |
| 272 } | 408 } |
| 273 // It's okay to return true on no option page and/or input view page case. | 409 // It's okay to return true on no option page and/or input view page case. |
| 274 return true; | 410 return true; |
| 275 } | 411 } |
| 276 | 412 |
| 277 // static | 413 // static |
| 278 void ComponentExtensionIMEManagerImpl::ReadComponentExtensionsInfo( | 414 void ComponentExtensionIMEManagerImpl::ReadComponentExtensionsInfo( |
| 279 std::vector<ComponentExtensionIME>* out_imes) { | 415 std::vector<ComponentExtensionIME>* out_imes) { |
| 280 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::FILE)); | |
| 281 DCHECK(out_imes); | 416 DCHECK(out_imes); |
| 282 for (size_t i = 0; i < arraysize(whitelisted_component_extension); ++i) { | 417 for (size_t i = 0; i < arraysize(whitelisted_component_extension); ++i) { |
| 283 ComponentExtensionIME component_ime; | 418 ComponentExtensionIME component_ime; |
| 284 component_ime.path = base::FilePath( | 419 component_ime.path = base::FilePath( |
| 285 whitelisted_component_extension[i].path); | 420 whitelisted_component_extension[i].path); |
| 286 | 421 |
| 287 if (!component_ime.path.IsAbsolute()) { | 422 ResourceBundle& rb = ResourceBundle::GetSharedInstance(); |
| 288 base::FilePath resources_path; | 423 component_ime.manifest = |
| 289 if (!PathService::Get(chrome::DIR_RESOURCES, &resources_path)) | 424 rb.GetRawDataResource( |
| 290 NOTREACHED(); | 425 whitelisted_component_extension[i].manifest_resource_id) |
| 291 component_ime.path = resources_path.Append(component_ime.path); | 426 .as_string(); |
| 292 } | |
| 293 const base::FilePath manifest_path = | |
| 294 component_ime.path.Append("manifest.json"); | |
| 295 | |
| 296 if (!base::PathExists(component_ime.path) || | |
| 297 !base::PathExists(manifest_path)) | |
| 298 continue; | |
| 299 | |
| 300 if (!base::ReadFileToString(manifest_path, &component_ime.manifest)) | |
| 301 continue; | |
| 302 | 427 |
| 303 scoped_ptr<base::DictionaryValue> manifest = | 428 scoped_ptr<base::DictionaryValue> manifest = |
| 304 GetManifest(component_ime.path); | 429 GetManifest(component_ime.manifest); |
| 305 if (!manifest.get()) | 430 if (!manifest.get()) |
| 306 continue; | 431 continue; |
| 307 | 432 |
| 308 if (!ReadExtensionInfo(*manifest.get(), | 433 if (!ReadExtensionInfo(*manifest.get(), |
| 309 whitelisted_component_extension[i].id, | 434 whitelisted_component_extension[i].id, |
| 310 &component_ime)) | 435 &component_ime)) |
| 311 continue; | 436 continue; |
| 312 component_ime.id = whitelisted_component_extension[i].id; | 437 component_ime.id = whitelisted_component_extension[i].id; |
| 313 | 438 |
| 314 const base::ListValue* component_list; | 439 const base::ListValue* component_list; |
| 315 if (!manifest->GetList(extensions::manifest_keys::kInputComponents, | 440 if (!manifest->GetList(extensions::manifest_keys::kInputComponents, |
| 316 &component_list)) | 441 &component_list)) |
| 317 continue; | 442 continue; |
| 318 | 443 |
| 319 for (size_t i = 0; i < component_list->GetSize(); ++i) { | 444 for (size_t i = 0; i < component_list->GetSize(); ++i) { |
| 320 const base::DictionaryValue* dictionary; | 445 const base::DictionaryValue* dictionary; |
| 321 if (!component_list->GetDictionary(i, &dictionary)) | 446 if (!component_list->GetDictionary(i, &dictionary)) |
| 322 continue; | 447 continue; |
| 323 | 448 |
| 324 ComponentExtensionEngine engine; | 449 ComponentExtensionEngine engine; |
| 325 ReadEngineComponent(component_ime, *dictionary, &engine); | 450 ReadEngineComponent(component_ime, *dictionary, &engine); |
| 326 component_ime.engines.push_back(engine); | 451 component_ime.engines.push_back(engine); |
| 327 } | 452 } |
| 328 out_imes->push_back(component_ime); | 453 out_imes->push_back(component_ime); |
| 329 } | 454 } |
| 330 } | 455 } |
| 331 | 456 |
| 332 void ComponentExtensionIMEManagerImpl::OnReadComponentExtensionsInfo( | |
| 333 std::vector<ComponentExtensionIME>* result, | |
| 334 const base::Closure& callback) { | |
| 335 DCHECK(thread_checker_.CalledOnValidThread()); | |
| 336 DCHECK(result); | |
| 337 component_extension_list_ = *result; | |
| 338 is_initialized_ = true; | |
| 339 callback.Run(); | |
| 340 } | |
| 341 | |
| 342 } // namespace chromeos | 457 } // namespace chromeos |
| OLD | NEW |