blob: 17bbd9c65b5dbd1f4981e1e12051173e8f59ddc4 [file] [log] [blame]
// 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 "components/spellcheck/renderer/spellcheck_provider.h"
#include "base/bind.h"
#include "base/metrics/histogram_macros.h"
#include "build/build_config.h"
#include "components/spellcheck/common/spellcheck.mojom.h"
#include "components/spellcheck/common/spellcheck_result.h"
#include "components/spellcheck/renderer/spellcheck.h"
#include "components/spellcheck/renderer/spellcheck_language.h"
#include "components/spellcheck/spellcheck_buildflags.h"
#include "content/public/common/service_names.mojom.h"
#include "content/public/renderer/render_frame.h"
#include "content/public/renderer/render_thread.h"
#include "services/service_manager/public/cpp/local_interface_provider.h"
#include "third_party/blink/public/platform/web_vector.h"
#include "third_party/blink/public/web/web_document.h"
#include "third_party/blink/public/web/web_element.h"
#include "third_party/blink/public/web/web_local_frame.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/blink/public/web/web_text_decoration_type.h"
using blink::WebElement;
using blink::WebLocalFrame;
using blink::WebString;
using blink::WebTextCheckingCompletion;
using blink::WebTextCheckingResult;
using blink::WebTextDecorationType;
using blink::WebVector;
static_assert(int(blink::kWebTextDecorationTypeSpelling) ==
int(SpellCheckResult::SPELLING),
"mismatching enums");
static_assert(int(blink::kWebTextDecorationTypeGrammar) ==
int(SpellCheckResult::GRAMMAR),
"mismatching enums");
class SpellCheckProvider::DictionaryUpdateObserverImpl
: public DictionaryUpdateObserver {
public:
explicit DictionaryUpdateObserverImpl(SpellCheckProvider* owner);
~DictionaryUpdateObserverImpl() override;
// DictionaryUpdateObserver:
void OnDictionaryUpdated(const WebVector<WebString>& words_added) override;
private:
SpellCheckProvider* owner_;
};
SpellCheckProvider::DictionaryUpdateObserverImpl::DictionaryUpdateObserverImpl(
SpellCheckProvider* owner)
: owner_(owner) {
owner_->spellcheck_->AddDictionaryUpdateObserver(this);
}
SpellCheckProvider::DictionaryUpdateObserverImpl::
~DictionaryUpdateObserverImpl() {
owner_->spellcheck_->RemoveDictionaryUpdateObserver(this);
}
void SpellCheckProvider::DictionaryUpdateObserverImpl::OnDictionaryUpdated(
const WebVector<WebString>& words_added) {
// Clear only cache. Current pending requests should continue as they are.
owner_->last_request_.clear();
owner_->last_results_.Assign(
blink::WebVector<blink::WebTextCheckingResult>());
// owner_->render_frame() is nullptr in unit tests.
if (auto* render_frame = owner_->render_frame()) {
DCHECK(render_frame->GetWebFrame());
render_frame->GetWebFrame()->RemoveSpellingMarkersUnderWords(words_added);
}
}
SpellCheckProvider::SpellCheckProvider(
content::RenderFrame* render_frame,
SpellCheck* spellcheck,
service_manager::LocalInterfaceProvider* embedder_provider)
: content::RenderFrameObserver(render_frame),
content::RenderFrameObserverTracker<SpellCheckProvider>(render_frame),
spellcheck_(spellcheck),
embedder_provider_(embedder_provider),
weak_factory_(this) {
DCHECK(spellcheck_);
DCHECK(embedder_provider);
if (render_frame) // NULL in unit tests.
render_frame->GetWebFrame()->SetTextCheckClient(this);
dictionary_update_observer_ =
std::make_unique<DictionaryUpdateObserverImpl>(this);
}
SpellCheckProvider::~SpellCheckProvider() {
}
void SpellCheckProvider::ResetDictionaryUpdateObserverForTesting() {
dictionary_update_observer_.reset();
}
spellcheck::mojom::SpellCheckHost& SpellCheckProvider::GetSpellCheckHost() {
if (spell_check_host_)
return *spell_check_host_;
embedder_provider_->GetInterface(&spell_check_host_);
return *spell_check_host_;
}
void SpellCheckProvider::RequestTextChecking(
const base::string16& text,
WebTextCheckingCompletion* completion) {
// Ignore invalid requests.
if (text.empty() || !HasWordCharacters(text, 0)) {
completion->DidCancelCheckingText();
return;
}
// Try to satisfy check from cache.
if (SatisfyRequestFromCache(text, completion))
return;
// Send this text to a browser. A browser checks the user profile and send
// this text to the Spelling service only if a user enables this feature.
last_request_.clear();
last_results_.Assign(blink::WebVector<blink::WebTextCheckingResult>());
last_identifier_ = text_check_completions_.Add(completion);
#if BUILDFLAG(USE_BROWSER_SPELLCHECKER)
// Text check (unified request for grammar and spell check) is only
// available for browser process, so we ask the system spellchecker
// over mojo or return an empty result if the checker is not available.
GetSpellCheckHost().RequestTextCheck(
text, routing_id(),
base::BindOnce(&SpellCheckProvider::OnRespondTextCheck,
weak_factory_.GetWeakPtr(), last_identifier_, text));
#else
GetSpellCheckHost().CallSpellingService(
text, base::BindOnce(&SpellCheckProvider::OnRespondSpellingService,
weak_factory_.GetWeakPtr(), last_identifier_, text));
#endif // !USE_BROWSER_SPELLCHECKER
}
void SpellCheckProvider::FocusedNodeChanged(const blink::WebNode& unused) {
#if defined(OS_ANDROID)
if (!spell_check_host_.is_bound())
return;
WebLocalFrame* frame = render_frame()->GetWebFrame();
WebElement element = frame->GetDocument().IsNull()
? WebElement()
: frame->GetDocument().FocusedElement();
bool enabled = !element.IsNull() && element.IsEditable();
if (!enabled)
GetSpellCheckHost().DisconnectSessionBridge();
#endif // defined(OS_ANDROID)
}
bool SpellCheckProvider::IsSpellCheckingEnabled() const {
return spellcheck_->IsSpellcheckEnabled();
}
void SpellCheckProvider::CheckSpelling(
const WebString& text,
size_t& offset,
size_t& length,
WebVector<WebString>* optional_suggestions) {
base::string16 word = text.Utf16();
std::vector<base::string16> suggestions;
const int kWordStart = 0;
spellcheck_->SpellCheckWord(word.c_str(), kWordStart, word.size(),
routing_id(), &offset, &length,
optional_suggestions ? &suggestions : nullptr);
if (optional_suggestions) {
WebVector<WebString> web_suggestions(suggestions.size());
std::transform(
suggestions.begin(), suggestions.end(), web_suggestions.begin(),
[](const base::string16& s) { return WebString::FromUTF16(s); });
*optional_suggestions = web_suggestions;
UMA_HISTOGRAM_COUNTS_1M("SpellCheck.api.check.suggestions",
base::saturated_cast<int>(word.size()));
} else {
UMA_HISTOGRAM_COUNTS_1M("SpellCheck.api.check",
base::saturated_cast<int>(word.size()));
// If optional_suggestions is not requested, the API is called
// for marking. So we use this for counting markable words.
GetSpellCheckHost().NotifyChecked(word, 0 < length);
}
}
void SpellCheckProvider::RequestCheckingOfText(
const WebString& text,
WebTextCheckingCompletion* completion) {
RequestTextChecking(text.Utf16(), completion);
UMA_HISTOGRAM_COUNTS_1M("SpellCheck.api.async",
base::saturated_cast<int>(text.length()));
}
#if !BUILDFLAG(USE_BROWSER_SPELLCHECKER)
void SpellCheckProvider::OnRespondSpellingService(
int identifier,
const base::string16& line,
bool success,
const std::vector<SpellCheckResult>& results) {
WebTextCheckingCompletion* completion =
text_check_completions_.Lookup(identifier);
if (!completion)
return;
text_check_completions_.Remove(identifier);
// If |success| is false, we use local spellcheck as a fallback.
if (!success) {
spellcheck_->RequestTextChecking(line, completion);
return;
}
// Double-check the returned spellchecking results with our spellchecker to
// visualize the differences between ours and the on-line spellchecker.
blink::WebVector<blink::WebTextCheckingResult> textcheck_results;
spellcheck_->CreateTextCheckingResults(SpellCheck::USE_NATIVE_CHECKER,
0,
line,
results,
&textcheck_results);
completion->DidFinishCheckingText(textcheck_results);
// Cache the request and the converted results.
last_request_ = line;
last_results_.Swap(textcheck_results);
}
#endif
bool SpellCheckProvider::HasWordCharacters(const base::string16& text,
size_t index) const {
const base::char16* data = text.data();
size_t length = text.length();
while (index < length) {
uint32_t code = 0;
U16_NEXT(data, index, length, code);
UErrorCode error = U_ZERO_ERROR;
if (uscript_getScript(code, &error) != USCRIPT_COMMON)
return true;
}
return false;
}
#if BUILDFLAG(USE_BROWSER_SPELLCHECKER)
void SpellCheckProvider::OnRespondTextCheck(
int identifier,
const base::string16& line,
const std::vector<SpellCheckResult>& results) {
// TODO(groby): Unify with SpellCheckProvider::OnRespondSpellingService
DCHECK(spellcheck_);
WebTextCheckingCompletion* completion =
text_check_completions_.Lookup(identifier);
if (!completion)
return;
text_check_completions_.Remove(identifier);
blink::WebVector<blink::WebTextCheckingResult> textcheck_results;
spellcheck_->CreateTextCheckingResults(SpellCheck::DO_NOT_MODIFY,
0,
line,
results,
&textcheck_results);
completion->DidFinishCheckingText(textcheck_results);
// Cache the request and the converted results.
last_request_ = line;
last_results_.Swap(textcheck_results);
}
#endif
bool SpellCheckProvider::SatisfyRequestFromCache(
const base::string16& text,
WebTextCheckingCompletion* completion) {
size_t last_length = last_request_.length();
if (!last_length)
return false;
// Send back the |last_results_| if the |last_request_| is a substring of
// |text| and |text| does not have more words to check. Provider cannot cancel
// the spellcheck request here, because WebKit might have discarded the
// previous spellcheck results and erased the spelling markers in response to
// the user editing the text.
base::string16 request(text);
size_t text_length = request.length();
if (text_length >= last_length &&
!request.compare(0, last_length, last_request_)) {
if (text_length == last_length || !HasWordCharacters(text, last_length)) {
completion->DidFinishCheckingText(last_results_);
return true;
}
}
// Create a subset of the cached results and return it if the given text is a
// substring of the cached text.
if (text_length < last_length &&
!last_request_.compare(0, text_length, request)) {
size_t result_size = 0;
for (size_t i = 0; i < last_results_.size(); ++i) {
size_t start = last_results_[i].location;
size_t end = start + last_results_[i].length;
if (start <= text_length && end <= text_length)
++result_size;
}
blink::WebVector<blink::WebTextCheckingResult> results(last_results_.Data(),
result_size);
completion->DidFinishCheckingText(results);
return true;
}
return false;
}
void SpellCheckProvider::OnDestruct() {
delete this;
}