// 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/renderer/spellchecker/spellcheck.h" #include "base/bind.h" #include "base/message_loop_proxy.h" #include "base/utf_string_conversions.h" #include "chrome/common/render_messages.h" #include "chrome/common/spellcheck_common.h" #include "chrome/common/spellcheck_messages.h" #include "chrome/common/spellcheck_result.h" #include "chrome/renderer/spellchecker/spellcheck_language.h" #include "chrome/renderer/spellchecker/spellcheck_provider.h" #include "content/public/renderer/render_view.h" #include "content/public/renderer/render_view_visitor.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebTextCheckingCompletion.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebTextCheckingResult.h" using WebKit::WebVector; using WebKit::WebTextCheckingResult; using WebKit::WebTextCheckingType; namespace { class UpdateSpellcheckEnabled : public content::RenderViewVisitor { public: explicit UpdateSpellcheckEnabled(bool enabled) : enabled_(enabled) {} virtual bool Visit(content::RenderView* render_view) OVERRIDE; private: bool enabled_; // New spellcheck-enabled state. DISALLOW_COPY_AND_ASSIGN(UpdateSpellcheckEnabled); }; bool UpdateSpellcheckEnabled::Visit(content::RenderView* render_view) { SpellCheckProvider* provider = SpellCheckProvider::Get(render_view); DCHECK(provider); provider->EnableSpellcheck(enabled_); return true; } } // namespace class SpellCheck::SpellcheckRequest { public: SpellcheckRequest(const string16& text, int offset, WebKit::WebTextCheckingCompletion* completion) : text_(text), offset_(offset), completion_(completion) { DCHECK(completion); } ~SpellcheckRequest() {} string16 text() { return text_; } int offset() { return offset_; } WebKit::WebTextCheckingCompletion* completion() { return completion_; } private: string16 text_; // Text to be checked in this task. int offset_; // The text offset from the beginning. // The interface to send the misspelled ranges to WebKit. WebKit::WebTextCheckingCompletion* completion_; DISALLOW_COPY_AND_ASSIGN(SpellcheckRequest); }; // Initializes SpellCheck object. // spellcheck_enabled_ currently MUST be set to true, due to peculiarities of // the initialization sequence. // Since it defaults to true, newly created SpellCheckProviders will enable // spellchecking. After the first word is typed, the provider requests a check, // which in turn triggers the delayed initialization sequence in SpellCheck. // This does send a message to the browser side, which triggers the creation // of the SpellcheckService. That does create the observer for the preference // responsible for enabling/disabling checking, which allows subsequent changes // to that preference to be sent to all SpellCheckProviders. // Setting |spellcheck_enabled_| to false by default prevents that mechanism, // and as such the SpellCheckProviders will never be notified of different // values. // TODO(groby): Simplify this. SpellCheck::SpellCheck() : auto_spell_correct_turned_on_(false), spellcheck_enabled_(true) { } SpellCheck::~SpellCheck() { } bool SpellCheck::OnControlMessageReceived(const IPC::Message& message) { bool handled = true; IPC_BEGIN_MESSAGE_MAP(SpellCheck, message) IPC_MESSAGE_HANDLER(SpellCheckMsg_Init, OnInit) IPC_MESSAGE_HANDLER(SpellCheckMsg_CustomDictionaryChanged, OnCustomDictionaryChanged) IPC_MESSAGE_HANDLER(SpellCheckMsg_EnableAutoSpellCorrect, OnEnableAutoSpellCorrect) IPC_MESSAGE_HANDLER(SpellCheckMsg_EnableSpellCheck, OnEnableSpellCheck) IPC_MESSAGE_UNHANDLED(handled = false) IPC_END_MESSAGE_MAP() return handled; } void SpellCheck::OnInit(IPC::PlatformFileForTransit bdict_file, const std::vector& custom_words, const std::string& language, bool auto_spell_correct) { Init(IPC::PlatformFileForTransitToPlatformFile(bdict_file), custom_words, language); auto_spell_correct_turned_on_ = auto_spell_correct; #if !defined(OS_MACOSX) PostDelayedSpellCheckTask(pending_request_param_.release()); #endif } void SpellCheck::OnCustomDictionaryChanged( const std::vector& words_added, const std::vector& words_removed) { custom_dictionary_.OnCustomDictionaryChanged(words_added, words_removed); } void SpellCheck::OnEnableAutoSpellCorrect(bool enable) { auto_spell_correct_turned_on_ = enable; } void SpellCheck::OnEnableSpellCheck(bool enable) { spellcheck_enabled_ = enable; UpdateSpellcheckEnabled updater(enable); content::RenderView::ForEach(&updater); } // TODO(groby): Make sure we always have a spelling engine, even before Init() // is called. void SpellCheck::Init(base::PlatformFile file, const std::vector& custom_words, const std::string& language) { spellcheck_.Init(file, language); custom_dictionary_.Init(custom_words); } bool SpellCheck::SpellCheckWord( const char16* in_word, int in_word_len, int tag, int* misspelling_start, int* misspelling_len, std::vector* optional_suggestions) { DCHECK(in_word_len >= 0); DCHECK(misspelling_start && misspelling_len) << "Out vars must be given."; // Do nothing if we need to delay initialization. (Rather than blocking, // report the word as correctly spelled.) if (InitializeIfNeeded()) return true; return spellcheck_.SpellCheckWord(in_word, in_word_len, tag, misspelling_start, misspelling_len, optional_suggestions); } bool SpellCheck::SpellCheckParagraph( const string16& text, WebVector* results) { #if !defined(OS_MACOSX) // Mac has its own spell checker, so this method will not be used. DCHECK(results); std::vector textcheck_results; size_t length = text.length(); size_t offset = 0; // Spellcheck::SpellCheckWord() automatically breaks text into words and // checks the spellings of the extracted words. This function sets the // position and length of the first misspelled word and returns false when // the text includes misspelled words. Therefore, we just repeat calling the // function until it returns true to check the whole text. int misspelling_start = 0; int misspelling_length = 0; while (offset <= length) { if (SpellCheckWord(&text[offset], length - offset, 0, &misspelling_start, &misspelling_length, NULL)) { results->assign(textcheck_results); return true; } if (!custom_dictionary_.SpellCheckWord( &text[offset], misspelling_start, misspelling_length)) { string16 replacement; textcheck_results.push_back(WebTextCheckingResult( WebKit::WebTextCheckingTypeSpelling, misspelling_start + offset, misspelling_length, replacement)); } offset += misspelling_start + misspelling_length; } results->assign(textcheck_results); return false; #else return true; #endif } string16 SpellCheck::GetAutoCorrectionWord(const string16& word, int tag) { string16 autocorrect_word; if (!auto_spell_correct_turned_on_) return autocorrect_word; // Return the empty string. int word_length = static_cast(word.size()); if (word_length < 2 || word_length > chrome::spellcheck_common::kMaxAutoCorrectWordSize) return autocorrect_word; if (InitializeIfNeeded()) return autocorrect_word; char16 misspelled_word[ chrome::spellcheck_common::kMaxAutoCorrectWordSize + 1]; const char16* word_char = word.c_str(); for (int i = 0; i <= chrome::spellcheck_common::kMaxAutoCorrectWordSize; ++i) { if (i >= word_length) misspelled_word[i] = 0; else misspelled_word[i] = word_char[i]; } // Swap adjacent characters and spellcheck. int misspelling_start, misspelling_len; for (int i = 0; i < word_length - 1; i++) { // Swap. std::swap(misspelled_word[i], misspelled_word[i + 1]); // Check spelling. misspelling_start = misspelling_len = 0; SpellCheckWord(misspelled_word, word_length, tag, &misspelling_start, &misspelling_len, NULL); // Make decision: if only one swap produced a valid word, then we want to // return it. If we found two or more, we don't do autocorrection. if (misspelling_len == 0) { if (autocorrect_word.empty()) { autocorrect_word.assign(misspelled_word); } else { autocorrect_word.clear(); break; } } // Restore the swapped characters. std::swap(misspelled_word[i], misspelled_word[i + 1]); } return autocorrect_word; } #if !defined(OS_MACOSX) // OSX uses its own spell checker void SpellCheck::RequestTextChecking( const string16& text, int offset, WebKit::WebTextCheckingCompletion* completion) { // Clean up the previous request before starting a new request. if (pending_request_param_.get()) pending_request_param_->completion()->didCancelCheckingText(); pending_request_param_.reset(new SpellcheckRequest( text, offset, completion)); // We will check this text after we finish loading the hunspell dictionary. if (InitializeIfNeeded()) return; PostDelayedSpellCheckTask(pending_request_param_.release()); } #endif bool SpellCheck::InitializeIfNeeded() { return spellcheck_.InitializeIfNeeded(); } #if !defined(OS_MACOSX) // OSX doesn't have |pending_request_param_| void SpellCheck::PostDelayedSpellCheckTask(SpellcheckRequest* request) { if (!request) return; base::MessageLoopProxy::current()->PostTask(FROM_HERE, base::Bind(&SpellCheck::PerformSpellCheck, AsWeakPtr(), base::Owned(request))); } #endif #if !defined(OS_MACOSX) // Mac uses its native engine instead. void SpellCheck::PerformSpellCheck(SpellcheckRequest* param) { DCHECK(param); if (!spellcheck_.IsEnabled()) { param->completion()->didCancelCheckingText(); } else { WebVector results; SpellCheckParagraph(param->text(), &results); param->completion()->didFinishCheckingText(results); } } #endif void SpellCheck::CreateTextCheckingResults( ResultFilter filter, int line_offset, const string16& line_text, const std::vector& spellcheck_results, WebVector* textcheck_results) { // Double-check misspelled words with our spellchecker and attach grammar // markers to them if our spellchecker tells they are correct words, i.e. they // are probably contextually-misspelled words. const char16* text = line_text.c_str(); std::vector list; for (size_t i = 0; i < spellcheck_results.size(); ++i) { WebTextCheckingType type = static_cast(spellcheck_results[i].type); int word_location = spellcheck_results[i].location; int word_length = spellcheck_results[i].length; int misspelling_start = 0; int misspelling_length = 0; if (type == WebKit::WebTextCheckingTypeSpelling && filter == USE_NATIVE_CHECKER) { if (SpellCheckWord(text + word_location, word_length, 0, &misspelling_start, &misspelling_length, NULL)) { type = WebKit::WebTextCheckingTypeGrammar; } } if (!custom_dictionary_.SpellCheckWord(text, word_location, word_length)) { list.push_back(WebTextCheckingResult( type, word_location + line_offset, word_length, spellcheck_results[i].replacement)); } } textcheck_results->assign(list); }