// Copyright (c) 2014 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include "shell/renderer/api/electron_api_spell_check_client.h" #include #include #include #include #include #include #include #include "base/containers/contains.h" #include "base/logging.h" #include "base/numerics/safe_conversions.h" #include "base/strings/utf_string_conversion_utils.h" #include "base/task/single_thread_task_runner.h" #include "components/spellcheck/renderer/spellcheck_worditerator.h" #include "shell/common/gin_helper/dictionary.h" #include "shell/common/gin_helper/function_template.h" #include "shell/common/gin_helper/microtasks_scope.h" #include "third_party/blink/public/web/web_text_checking_completion.h" #include "third_party/blink/public/web/web_text_checking_result.h" #include "third_party/icu/source/common/unicode/uscript.h" namespace electron::api { namespace { bool HasWordCharacters(const std::u16string& text, size_t index) { base_icu::UChar32 code; while (base::ReadUnicodeCharacter(text.c_str(), text.size(), &index, &code)) { UErrorCode error = U_ZERO_ERROR; if (uscript_getScript(code, &error) != USCRIPT_COMMON) return true; } return false; } struct Word { blink::WebTextCheckingResult result; std::u16string text; std::vector contraction_words; }; } // namespace class SpellCheckClient::SpellcheckRequest { public: SpellcheckRequest( const std::u16string& text, std::unique_ptr completion) : text_(text), completion_(std::move(completion)) {} SpellcheckRequest(const SpellcheckRequest&) = delete; SpellcheckRequest& operator=(const SpellcheckRequest&) = delete; ~SpellcheckRequest() = default; const std::u16string& text() const { return text_; } blink::WebTextCheckingCompletion* completion() { return completion_.get(); } std::vector& wordlist() { return word_list_; } private: std::u16string text_; // Text to be checked in this task. std::vector word_list_; // List of Words found in text // The interface to send the misspelled ranges to Blink. std::unique_ptr completion_; }; SpellCheckClient::SpellCheckClient(const std::string& language, v8::Isolate* isolate, v8::Local provider) : isolate_(isolate), context_(isolate, isolate->GetCurrentContext()), provider_(isolate, provider) { DCHECK(!context_.IsEmpty()); character_attributes_.SetDefaultLanguage(language); // Persistent the method. v8::Local spell_check; gin_helper::Dictionary(isolate, provider).Get("spellCheck", &spell_check); spell_check_.Reset(isolate, spell_check); } SpellCheckClient::~SpellCheckClient() { context_.Reset(); } void SpellCheckClient::RequestCheckingOfText( const blink::WebString& textToCheck, std::unique_ptr completionCallback) { std::u16string text(textToCheck.Utf16()); // Ignore invalid requests. if (text.empty() || !HasWordCharacters(text, 0)) { completionCallback->DidCancelCheckingText(); return; } // Clean up the previous request before starting a new request. if (pending_request_param_) { pending_request_param_->completion()->DidCancelCheckingText(); } pending_request_param_ = std::make_unique(text, std::move(completionCallback)); base::SingleThreadTaskRunner::GetCurrentDefault()->PostTask( FROM_HERE, base::BindOnce(&SpellCheckClient::SpellCheckText, weak_factory_.GetWeakPtr())); } bool SpellCheckClient::IsSpellCheckingEnabled() const { return true; } void SpellCheckClient::ShowSpellingUI(bool show) {} bool SpellCheckClient::IsShowingSpellingUI() { return false; } void SpellCheckClient::UpdateSpellingUIWithMisspelledWord( const blink::WebString& word) {} void SpellCheckClient::SpellCheckText() { const auto& text = pending_request_param_->text(); if (text.empty() || spell_check_.IsEmpty()) { pending_request_param_->completion()->DidCancelCheckingText(); pending_request_param_ = nullptr; return; } if (!text_iterator_.IsInitialized() && !text_iterator_.Initialize(&character_attributes_, true)) { // We failed to initialize text_iterator_, return as spelled correctly. VLOG(1) << "Failed to initialize SpellcheckWordIterator"; return; } if (!contraction_iterator_.IsInitialized() && !contraction_iterator_.Initialize(&character_attributes_, false)) { // We failed to initialize the word iterator, return as spelled correctly. VLOG(1) << "Failed to initialize contraction_iterator_"; return; } text_iterator_.SetText(text); SpellCheckScope scope(*this); std::u16string word; size_t word_start; size_t word_length; std::set words; auto& word_list = pending_request_param_->wordlist(); Word word_entry; for (;;) { // Run until end of text const auto status = text_iterator_.GetNextWord(&word, &word_start, &word_length); if (status == SpellcheckWordIterator::IS_END_OF_TEXT) break; if (status == SpellcheckWordIterator::IS_SKIPPABLE) continue; word_entry.result.location = base::checked_cast(word_start); word_entry.result.length = base::checked_cast(word_length); word_entry.text = word; word_entry.contraction_words.clear(); word_list.push_back(word_entry); words.insert(word); // If the given word is a concatenated word of two or more valid words // (e.g. "hello:hello"), we should treat it as a valid word. if (IsContraction(scope, word, &word_entry.contraction_words)) { for (const auto& w : word_entry.contraction_words) { words.insert(w); } } } // Send out all the words data to the spellchecker to check SpellCheckWords(scope, words); } void SpellCheckClient::OnSpellCheckDone( const std::vector& misspelled_words) { std::vector results; std::unordered_set misspelled(misspelled_words.begin(), misspelled_words.end()); auto& word_list = pending_request_param_->wordlist(); for (const auto& word : word_list) { if (base::Contains(misspelled, word.text)) { // If this is a contraction, iterate through parts and accept the word // if none of them are misspelled if (!word.contraction_words.empty()) { auto all_correct = true; for (const auto& contraction_word : word.contraction_words) { if (base::Contains(misspelled, contraction_word)) { all_correct = false; break; } } if (all_correct) continue; } results.push_back(word.result); } } pending_request_param_->completion()->DidFinishCheckingText(results); pending_request_param_ = nullptr; } void SpellCheckClient::SpellCheckWords(const SpellCheckScope& scope, const std::set& words) { DCHECK(!scope.spell_check_.IsEmpty()); auto context = isolate_->GetCurrentContext(); gin_helper::MicrotasksScope microtasks_scope{ isolate_, context->GetMicrotaskQueue(), false, v8::MicrotasksScope::kDoNotRunMicrotasks}; v8::Local templ = gin_helper::CreateFunctionTemplate( isolate_, base::BindRepeating(&SpellCheckClient::OnSpellCheckDone, weak_factory_.GetWeakPtr())); v8::Local args[] = {gin::ConvertToV8(isolate_, words), templ->GetFunction(context).ToLocalChecked()}; // Call javascript with the words and the callback function scope.spell_check_->Call(context, scope.provider_, std::size(args), args) .IsEmpty(); } // Returns whether or not the given string is a contraction. // This function is a fall-back when the SpellcheckWordIterator class // returns a concatenated word which is not in the selected dictionary // (e.g. "in'n'out") but each word is valid. // Output variable contraction_words will contain individual // words in the contraction. bool SpellCheckClient::IsContraction( const SpellCheckScope& scope, const std::u16string& contraction, std::vector* contraction_words) { DCHECK(contraction_iterator_.IsInitialized()); contraction_iterator_.SetText(contraction); std::u16string word; size_t word_start; size_t word_length; for (auto status = contraction_iterator_.GetNextWord(&word, &word_start, &word_length); status != SpellcheckWordIterator::IS_END_OF_TEXT; status = contraction_iterator_.GetNextWord(&word, &word_start, &word_length)) { if (status == SpellcheckWordIterator::IS_SKIPPABLE) continue; contraction_words->push_back(word); } return contraction_words->size() > 1; } SpellCheckClient::SpellCheckScope::SpellCheckScope( const SpellCheckClient& client) : handle_scope_(client.isolate_), context_scope_( v8::Local::New(client.isolate_, client.context_)), provider_(v8::Local::New(client.isolate_, client.provider_)), spell_check_( v8::Local::New(client.isolate_, client.spell_check_)) {} SpellCheckClient::SpellCheckScope::~SpellCheckScope() = default; } // namespace electron::api