OLD | NEW |
1 // Copyright (c) 2006-2009 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2006-2009 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 "app/l10n_util.h" | 5 #include "app/l10n_util.h" |
6 #include "chrome/browser/spellchecker.h" | 6 #include "chrome/browser/spellchecker.h" |
7 #include "chrome/browser/spellchecker_common.h" | 7 #include "chrome/browser/spellchecker_common.h" |
8 #include "chrome/browser/spellchecker_platform_engine.h" | 8 #include "chrome/browser/spellchecker_platform_engine.h" |
9 #include "base/basictypes.h" | 9 #include "base/basictypes.h" |
10 #include "base/compiler_specific.h" | 10 #include "base/compiler_specific.h" |
(...skipping 59 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
70 {"sk", "sk-SK"}, | 70 {"sk", "sk-SK"}, |
71 {"sl", "sl-SI"}, | 71 {"sl", "sl-SI"}, |
72 {"ca", "ca-ES"}, | 72 {"ca", "ca-ES"}, |
73 {"lv", "lv-LV"}, | 73 {"lv", "lv-LV"}, |
74 // {"uk", "uk-UA"}, // Not to be included in Spellchecker as per B=1277824 | 74 // {"uk", "uk-UA"}, // Not to be included in Spellchecker as per B=1277824 |
75 {"hi", "hi-IN"}, | 75 {"hi", "hi-IN"}, |
76 {"et", "et-EE"}, | 76 {"et", "et-EE"}, |
77 {"tr", "tr-TR"}, | 77 {"tr", "tr-TR"}, |
78 }; | 78 }; |
79 | 79 |
| 80 // Get the fallback folder (currently chrome::DIR_USER_DATA) where the |
| 81 // dictionary is downloaded in case of system-wide installations. |
| 82 FilePath GetFallbackDictionaryDownloadDirectory() { |
| 83 FilePath dict_dir_userdata; |
| 84 PathService::Get(chrome::DIR_USER_DATA, &dict_dir_userdata); |
| 85 dict_dir_userdata = dict_dir_userdata.AppendASCII("Dictionaries"); |
| 86 return dict_dir_userdata; |
| 87 } |
| 88 |
80 } | 89 } |
81 | 90 |
82 void SpellChecker::SpellCheckLanguages(std::vector<std::string>* languages) { | 91 void SpellChecker::SpellCheckLanguages(std::vector<std::string>* languages) { |
83 for (size_t i = 0; i < ARRAYSIZE_UNSAFE(g_supported_spellchecker_languages); | 92 for (size_t i = 0; i < ARRAYSIZE_UNSAFE(g_supported_spellchecker_languages); |
84 ++i) | 93 ++i) |
85 languages->push_back(g_supported_spellchecker_languages[i].language); | 94 languages->push_back(g_supported_spellchecker_languages[i].language); |
86 } | 95 } |
87 | 96 |
88 // This function returns the language-region version of language name. | 97 // This function returns the language-region version of language name. |
89 // e.g. returns hi-IN for hi. | 98 // e.g. returns hi-IN for hi. |
(...skipping 146 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
236 // which obtains the IO thread independently and invokes the task in the IO | 245 // which obtains the IO thread independently and invokes the task in the IO |
237 // thread if it's not NULL. After the flags are cleared, a (final) attempt is | 246 // thread if it's not NULL. After the flags are cleared, a (final) attempt is |
238 // made to initialize hunspell_. If it fails even then (dictionary could not | 247 // made to initialize hunspell_. If it fails even then (dictionary could not |
239 // download), no more attempts are made to initialize it. | 248 // download), no more attempts are made to initialize it. |
240 | 249 |
241 // ############################################################################ | 250 // ############################################################################ |
242 | 251 |
243 // This object downloads the dictionary files asynchronously by first | 252 // This object downloads the dictionary files asynchronously by first |
244 // fetching it to memory using URL fetcher and then writing it to | 253 // fetching it to memory using URL fetcher and then writing it to |
245 // disk using file_util::WriteFile. | 254 // disk using file_util::WriteFile. |
| 255 |
246 class SpellChecker::DictionaryDownloadController | 256 class SpellChecker::DictionaryDownloadController |
247 : public URLFetcher::Delegate, | 257 : public URLFetcher::Delegate, |
248 public base::RefCountedThreadSafe<DictionaryDownloadController> { | 258 public base::RefCountedThreadSafe<DictionaryDownloadController> { |
249 public: | 259 public: |
250 DictionaryDownloadController( | 260 DictionaryDownloadController( |
251 Task* spellchecker_flag_set_task, | 261 Task* spellchecker_flag_set_task, |
252 const FilePath& dic_file_path, | 262 const FilePath& dic_file_path, |
253 URLRequestContext* url_request_context, | 263 URLRequestContext* url_request_context, |
254 MessageLoop* ui_loop) | 264 MessageLoop* ui_loop) |
255 : spellchecker_flag_set_task_(spellchecker_flag_set_task), | 265 : spellchecker_flag_set_task_(spellchecker_flag_set_task), |
(...skipping 13 matching lines...) Expand all Loading... |
269 | 279 |
270 GURL url = GURL(std::string(kDownloadServerUrl) + WideToUTF8( | 280 GURL url = GURL(std::string(kDownloadServerUrl) + WideToUTF8( |
271 l10n_util::ToLower(file_name_.ToWStringHack()))); | 281 l10n_util::ToLower(file_name_.ToWStringHack()))); |
272 fetcher_.reset(new URLFetcher(url, URLFetcher::GET, this)); | 282 fetcher_.reset(new URLFetcher(url, URLFetcher::GET, this)); |
273 fetcher_->set_request_context(url_request_context_); | 283 fetcher_->set_request_context(url_request_context_); |
274 fetcher_->Start(); | 284 fetcher_->Start(); |
275 } | 285 } |
276 | 286 |
277 private: | 287 private: |
278 // The file has been downloaded in memory - need to write it down to file. | 288 // The file has been downloaded in memory - need to write it down to file. |
279 bool SaveBufferToFile(const std::string& data) { | 289 bool SaveBufferToFile(const std::string& data, |
280 FilePath file_to_write = dic_zip_file_path_.Append(file_name_); | 290 FilePath file_to_write) { |
281 int num_bytes = data.length(); | 291 int num_bytes = data.length(); |
282 return file_util::WriteFile(file_to_write, data.data(), num_bytes) == | 292 return file_util::WriteFile(file_to_write, data.data(), num_bytes) == |
283 num_bytes; | 293 num_bytes; |
284 } | 294 } |
285 | 295 |
286 // URLFetcher::Delegate interface. | 296 // URLFetcher::Delegate interface. |
287 virtual void OnURLFetchComplete(const URLFetcher* source, | 297 virtual void OnURLFetchComplete(const URLFetcher* source, |
288 const GURL& url, | 298 const GURL& url, |
289 const URLRequestStatus& status, | 299 const URLRequestStatus& status, |
290 int response_code, | 300 int response_code, |
291 const ResponseCookies& cookies, | 301 const ResponseCookies& cookies, |
292 const std::string& data) { | 302 const std::string& data) { |
293 DCHECK(source); | 303 DCHECK(source); |
294 bool save_success = false; | |
295 if ((response_code / 100) == 2 || | 304 if ((response_code / 100) == 2 || |
296 response_code == 401 || | 305 response_code == 401 || |
297 response_code == 407) { | 306 response_code == 407) { |
298 save_success = SaveBufferToFile(data); | 307 FilePath file_to_write = dic_zip_file_path_.Append(file_name_); |
| 308 if (!SaveBufferToFile(data, file_to_write)) { |
| 309 // Try saving it to user data/Dictionaries, which almost surely has |
| 310 // write permission. If even this fails, there is nothing to be done. |
| 311 FilePath user_data_dir = GetFallbackDictionaryDownloadDirectory(); |
| 312 |
| 313 // Create the directory if it does not exist. |
| 314 if (!file_util::PathExists(user_data_dir)) |
| 315 file_util::CreateDirectory(user_data_dir); |
| 316 |
| 317 file_to_write = user_data_dir.Append(file_name_); |
| 318 SaveBufferToFile(data.data(), file_to_write); |
| 319 } |
299 } // Unsuccessful save is taken care of in SpellChecker::Initialize(). | 320 } // Unsuccessful save is taken care of in SpellChecker::Initialize(). |
300 | 321 |
301 // Set Flag that dictionary is not downloading anymore. | 322 // Set Flag that dictionary is not downloading anymore. |
302 ui_loop_->PostTask(FROM_HERE, | 323 ui_loop_->PostTask(FROM_HERE, |
303 new UIProxyForIOTask(spellchecker_flag_set_task_)); | 324 new UIProxyForIOTask(spellchecker_flag_set_task_)); |
304 fetcher_.reset(NULL); | 325 fetcher_.reset(NULL); |
305 } | 326 } |
306 | 327 |
307 // factory object to invokelater back to spellchecker in io thread on | 328 // factory object to invokelater back to spellchecker in io thread on |
308 // download completion to change appropriate flags. | 329 // download completion to change appropriate flags. |
(...skipping 70 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
379 } | 400 } |
380 } | 401 } |
381 | 402 |
382 return dict_dir.AppendASCII(versioned_bdict_file_name); | 403 return dict_dir.AppendASCII(versioned_bdict_file_name); |
383 } | 404 } |
384 | 405 |
385 SpellChecker::SpellChecker(const FilePath& dict_dir, | 406 SpellChecker::SpellChecker(const FilePath& dict_dir, |
386 const std::string& language, | 407 const std::string& language, |
387 URLRequestContext* request_context, | 408 URLRequestContext* request_context, |
388 const FilePath& custom_dictionary_file_name) | 409 const FilePath& custom_dictionary_file_name) |
389 : custom_dictionary_file_name_(custom_dictionary_file_name), | 410 : given_dictionary_directory_(dict_dir), |
| 411 custom_dictionary_file_name_(custom_dictionary_file_name), |
390 tried_to_init_(false), | 412 tried_to_init_(false), |
| 413 language_(language), |
391 #ifndef NDEBUG | 414 #ifndef NDEBUG |
392 worker_loop_(NULL), | 415 worker_loop_(NULL), |
393 #endif | 416 #endif |
394 tried_to_download_(false), | 417 tried_to_download_dictionary_file_(false), |
395 file_loop_(NULL), | 418 file_loop_(NULL), |
396 url_request_context_(request_context), | 419 url_request_context_(request_context), |
397 dic_is_downloading_(false), | 420 dic_is_downloading_(false), |
398 auto_spell_correct_turned_on_(false), | 421 auto_spell_correct_turned_on_(false), |
399 is_using_platform_spelling_engine_(false), | 422 is_using_platform_spelling_engine_(false), |
400 ALLOW_THIS_IN_INITIALIZER_LIST( | 423 ALLOW_THIS_IN_INITIALIZER_LIST( |
401 dic_download_state_changer_factory_(this)) { | 424 dic_download_state_changer_factory_(this)) { |
402 if (SpellCheckerPlatform::SpellCheckerAvailable()) { | 425 if (SpellCheckerPlatform::SpellCheckerAvailable()) { |
403 SpellCheckerPlatform::Init(); | 426 SpellCheckerPlatform::Init(); |
404 if (SpellCheckerPlatform::PlatformSupportsLanguage(language)) { | 427 if (SpellCheckerPlatform::PlatformSupportsLanguage(language)) { |
405 // If we have reached here, then we know that the current platform | 428 // If we have reached here, then we know that the current platform |
406 // supports the given language and we will use it instead of hunspell. | 429 // supports the given language and we will use it instead of hunspell. |
407 SpellCheckerPlatform::SetLanguage(language); | 430 SpellCheckerPlatform::SetLanguage(language); |
408 is_using_platform_spelling_engine_ = true; | 431 is_using_platform_spelling_engine_ = true; |
409 } | 432 } |
410 } | 433 } |
411 | 434 |
412 // Remember UI loop to later use this as a proxy to get IO loop. | 435 // Remember UI loop to later use this as a proxy to get IO loop. |
413 ui_loop_ = MessageLoop::current(); | 436 ui_loop_ = MessageLoop::current(); |
414 | 437 |
415 // Get File Loop - hunspell gets initialized here. | 438 // Get File Loop - hunspell gets initialized here. |
416 base::Thread* file_thread = g_browser_process->file_thread(); | 439 base::Thread* file_thread = g_browser_process->file_thread(); |
417 if (file_thread) | 440 if (file_thread) |
418 file_loop_ = file_thread->message_loop(); | 441 file_loop_ = file_thread->message_loop(); |
419 | 442 |
420 // Get the path to the spellcheck file. | |
421 bdict_file_name_ = GetVersionedFileName(language, dict_dir); | |
422 | |
423 // Get the path to the custom dictionary file. | 443 // Get the path to the custom dictionary file. |
424 if (custom_dictionary_file_name_.empty()) { | 444 if (custom_dictionary_file_name_.empty()) { |
425 FilePath personal_file_directory; | 445 FilePath personal_file_directory; |
426 PathService::Get(chrome::DIR_USER_DATA, &personal_file_directory); | 446 PathService::Get(chrome::DIR_USER_DATA, &personal_file_directory); |
427 custom_dictionary_file_name_ = | 447 custom_dictionary_file_name_ = |
428 personal_file_directory.Append(chrome::kCustomDictionaryFileName); | 448 personal_file_directory.Append(chrome::kCustomDictionaryFileName); |
429 } | 449 } |
430 | 450 |
431 // Use this dictionary language as the default one of the | 451 // Use this dictionary language as the default one of the |
432 // SpellcheckCharAttribute object. | 452 // SpellcheckCharAttribute object. |
433 character_attributes_.SetDefaultLanguage(language); | 453 character_attributes_.SetDefaultLanguage(language); |
434 } | 454 } |
435 | 455 |
436 SpellChecker::~SpellChecker() { | 456 SpellChecker::~SpellChecker() { |
437 #ifndef NDEBUG | 457 #ifndef NDEBUG |
438 // This must be deleted on the I/O thread (see the header). This is the same | 458 // This must be deleted on the I/O thread (see the header). This is the same |
439 // thread thatSpellCheckWord is called on, so we verify that they were all the | 459 // thread thatSpellCheckWord is called on, so we verify that they were all the |
440 // same thread. | 460 // same thread. |
441 if (worker_loop_) | 461 if (worker_loop_) |
442 DCHECK(MessageLoop::current() == worker_loop_); | 462 DCHECK(MessageLoop::current() == worker_loop_); |
443 #endif | 463 #endif |
444 } | 464 } |
445 | 465 |
| 466 void SpellChecker::StartDictionaryDownloadInFileThread( |
| 467 const FilePath& file_name) { |
| 468 Task* dic_task = dic_download_state_changer_factory_.NewRunnableMethod( |
| 469 &SpellChecker::set_file_is_downloading, false); |
| 470 ddc_dic_ = new DictionaryDownloadController(dic_task, file_name, |
| 471 url_request_context_, ui_loop_); |
| 472 set_file_is_downloading(true); |
| 473 file_loop_->PostTask(FROM_HERE, NewRunnableMethod(ddc_dic_.get(), |
| 474 &DictionaryDownloadController::StartDownload)); |
| 475 } |
| 476 |
446 // Initialize SpellChecker. In this method, if the dicitonary is not present | 477 // Initialize SpellChecker. In this method, if the dicitonary is not present |
447 // in the local disk, it is fetched asynchronously. | 478 // in the local disk, it is fetched asynchronously. |
448 // TODO(sidchat): After dictionary is downloaded, initialize hunspell in | 479 // TODO(sidchat): After dictionary is downloaded, initialize hunspell in |
449 // file loop - this is currently being done in the io loop. | 480 // file loop - this is currently being done in the io loop. |
450 // Bug: http://b/issue?id=1123096 | 481 // Bug: http://b/issue?id=1123096 |
451 bool SpellChecker::Initialize() { | 482 bool SpellChecker::Initialize() { |
452 // Return false if the dictionary files are downloading. | 483 // Return false if the dictionary files are downloading. |
453 if (dic_is_downloading_) | 484 if (dic_is_downloading_) |
454 return false; | 485 return false; |
455 | 486 |
456 // Return false if tried to init and failed - don't try multiple times in | 487 // Return false if tried to init and failed - don't try multiple times in |
457 // this session. | 488 // this session. |
458 if (tried_to_init_) | 489 if (tried_to_init_) |
459 return hunspell_.get() != NULL; | 490 return hunspell_.get() != NULL; |
460 | 491 |
461 StatsScope<StatsCounterTimer> timer(chrome::Counters::spellcheck_init()); | 492 StatsScope<StatsCounterTimer> timer(chrome::Counters::spellcheck_init()); |
462 | 493 |
463 bool dic_exists = file_util::PathExists(bdict_file_name_); | 494 // The default place whether the spellcheck dictionary can reside is |
464 if (!dic_exists) { | 495 // chrome::DIR_APP_DICTIONARIES. However, for systemwide installations, |
465 if (file_loop_ && !tried_to_download_ && url_request_context_) { | 496 // this directory may not have permissions for download. In that case, the |
466 Task* dic_task = dic_download_state_changer_factory_.NewRunnableMethod( | 497 // alternate directory for download is chrome::DIR_USER_DATA. We have to check |
467 &SpellChecker::set_file_is_downloading, false); | 498 // for the spellcheck dictionaries in both the directories. If not found in |
468 ddc_dic_ = new DictionaryDownloadController(dic_task, bdict_file_name_, | 499 // either one, it has to be downloaded in either of the two. |
469 url_request_context_, ui_loop_); | 500 // TODO(sidchat): Some sort of UI to warn users that spellchecker is not |
470 set_file_is_downloading(true); | 501 // working at all (due to failed dictionary download)? |
471 file_loop_->PostTask(FROM_HERE, NewRunnableMethod(ddc_dic_.get(), | 502 |
472 &DictionaryDownloadController::StartDownload)); | 503 // File name for downloading in DIR_APP_DICTIONARIES. |
| 504 FilePath dictionary_file_name_app = GetVersionedFileName(language_, |
| 505 given_dictionary_directory_); |
| 506 |
| 507 // Filename for downloading in the fallback dictionary download directory, |
| 508 // DIR_USER_DATA. |
| 509 FilePath dict_dir_userdata = GetFallbackDictionaryDownloadDirectory(); |
| 510 FilePath dictionary_file_name_usr = GetVersionedFileName(language_, |
| 511 dict_dir_userdata); |
| 512 |
| 513 // Check in both the directories to see whether the spellcheck dictionary |
| 514 // already resides in one of these. |
| 515 FilePath bdic_file_name; |
| 516 if (file_util::PathExists(dictionary_file_name_app)) { |
| 517 bdic_file_name = dictionary_file_name_app; |
| 518 } else if (file_util::PathExists(dictionary_file_name_usr)) { |
| 519 bdic_file_name = dictionary_file_name_usr; |
| 520 } else { |
| 521 // Download the dictionary file. |
| 522 if (file_loop_ && url_request_context_) { |
| 523 if (!tried_to_download_dictionary_file_) { |
| 524 StartDictionaryDownloadInFileThread(dictionary_file_name_app); |
| 525 tried_to_download_dictionary_file_ = true; |
| 526 return false; |
| 527 } else { // There is no dictionary even after trying to download it. |
| 528 // Stop trying to download the dictionary in this session. |
| 529 tried_to_init_ = true; |
| 530 return false; |
| 531 } |
473 } | 532 } |
474 } | 533 } |
475 | 534 |
476 if (!dic_exists && !tried_to_download_) { | 535 // Control has come so far - the BDIC dictionary file probably exists. Now try |
477 tried_to_download_ = true; | 536 // to initialize hunspell using the available bdic dictionary file. |
478 return false; | |
479 } | |
480 | |
481 // Control has come so far - both files probably exist. | |
482 TimeTicks begin_time = TimeTicks::Now(); | 537 TimeTicks begin_time = TimeTicks::Now(); |
483 bdict_file_.reset(new file_util::MemoryMappedFile()); | 538 bdict_file_.reset(new file_util::MemoryMappedFile()); |
484 if (bdict_file_->Initialize(bdict_file_name_)) { | 539 if (bdict_file_->Initialize(bdic_file_name)) { |
485 hunspell_.reset(new Hunspell(bdict_file_->data(), bdict_file_->length())); | 540 hunspell_.reset(new Hunspell(bdict_file_->data(), bdict_file_->length())); |
486 AddCustomWordsToHunspell(); | 541 AddCustomWordsToHunspell(); |
487 } | 542 } |
488 DHISTOGRAM_TIMES("Spellcheck.InitTime", TimeTicks::Now() - begin_time); | 543 DHISTOGRAM_TIMES("Spellcheck.InitTime", TimeTicks::Now() - begin_time); |
489 | 544 |
490 tried_to_init_ = true; | 545 tried_to_init_ = true; |
491 return false; | 546 return false; |
492 } | 547 } |
493 | 548 |
494 void SpellChecker::GetAutoCorrectionWord(const std::wstring& word, | 549 void SpellChecker::GetAutoCorrectionWord(const std::wstring& word, |
(...skipping 232 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
727 | 782 |
728 // Populate the vector of WideStrings. | 783 // Populate the vector of WideStrings. |
729 for (int i = 0; i < number_of_suggestions; i++) { | 784 for (int i = 0; i < number_of_suggestions; i++) { |
730 if (i < kMaxSuggestions) | 785 if (i < kMaxSuggestions) |
731 optional_suggestions->push_back(UTF8ToWide(suggestions[i])); | 786 optional_suggestions->push_back(UTF8ToWide(suggestions[i])); |
732 free(suggestions[i]); | 787 free(suggestions[i]); |
733 } | 788 } |
734 if (suggestions != NULL) | 789 if (suggestions != NULL) |
735 free(suggestions); | 790 free(suggestions); |
736 } | 791 } |
OLD | NEW |