blob: 1bc3e648cbbf7d98716d644130a88a6d24f15110 [file] [log] [blame]
// Copyright 2015 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/browser/android/contextualsearch/contextual_search_manager.h"
#include <memory>
#include <set>
#include "base/android/jni_string.h"
#include "base/callback.h"
#include "base/memory/weak_ptr.h"
#include "base/supports_user_data.h"
#include "base/time/time.h"
#include "chrome/browser/android/contextualsearch/contextual_search_delegate.h"
#include "chrome/browser/android/contextualsearch/resolved_search_term.h"
#include "chrome/browser/profiles/profile_manager.h"
#include "chrome/browser/search_engines/template_url_service_factory.h"
#include "components/contextual_search/content/browser/contextual_search_js_api_service_impl.h"
#include "components/navigation_interception/intercept_navigation_delegate.h"
#include "components/variations/variations_associated_data.h"
#include "content/public/browser/render_frame_host.h"
#include "content/public/browser/render_view_host.h"
#include "content/public/browser/web_contents.h"
#include "content/public/browser/web_contents_observer.h"
#include "jni/ContextualSearchManager_jni.h"
#include "net/url_request/url_fetcher_impl.h"
#include "services/network/public/cpp/shared_url_loader_factory.h"
#include "services/service_manager/public/cpp/binder_registry.h"
#include "services/service_manager/public/cpp/interface_provider.h"
using base::android::JavaParamRef;
using base::android::JavaRef;
using content::WebContents;
namespace {
const char kContextualSearchObserverKey[] = "contextual_search_observer";
class ContextualSearchObserver : public content::WebContentsObserver,
public base::SupportsUserData::Data {
public:
ContextualSearchObserver(
content::WebContents* contents,
contextual_search::ContextualSearchJsApiHandler* api_handler)
: content::WebContentsObserver(contents) {
registry_.AddInterface(base::Bind(
&contextual_search::CreateContextualSearchJsApiService, api_handler));
}
~ContextualSearchObserver() override = default;
static void EnsureForWebContents(
content::WebContents* contents,
contextual_search::ContextualSearchJsApiHandler* api_handler) {
// Clobber any prior registered observer.
contents->SetUserData(
kContextualSearchObserverKey,
std::make_unique<ContextualSearchObserver>(contents, api_handler));
}
private:
// content::WebContentsObserver:
void OnInterfaceRequestFromFrame(
content::RenderFrameHost* render_frame_host,
const std::string& interface_name,
mojo::ScopedMessagePipeHandle* interface_pipe) override {
registry_.TryBindInterface(interface_name, interface_pipe);
}
service_manager::BinderRegistry registry_;
DISALLOW_COPY_AND_ASSIGN(ContextualSearchObserver);
};
} // namespace
// This class manages the native behavior of the Contextual Search feature.
// Instances of this class are owned by the Java ContextualSearchManager.
// Most of the work is actually done in an associated delegate to this class:
// the ContextualSearchDelegate.
ContextualSearchManager::ContextualSearchManager(JNIEnv* env,
const JavaRef<jobject>& obj) {
java_manager_.Reset(obj);
Java_ContextualSearchManager_setNativeManager(
env, obj, reinterpret_cast<intptr_t>(this));
Profile* profile = ProfileManager::GetActiveUserProfile();
delegate_.reset(new ContextualSearchDelegate(
profile->GetURLLoaderFactory(),
TemplateURLServiceFactory::GetForProfile(profile),
base::Bind(&ContextualSearchManager::OnSearchTermResolutionResponse,
base::Unretained(this)),
base::Bind(&ContextualSearchManager::OnTextSurroundingSelectionAvailable,
base::Unretained(this))));
}
ContextualSearchManager::~ContextualSearchManager() {
JNIEnv* env = base::android::AttachCurrentThread();
Java_ContextualSearchManager_clearNativeManager(env, java_manager_);
}
void ContextualSearchManager::Destroy(JNIEnv* env,
const JavaParamRef<jobject>& obj) {
delete this;
}
void ContextualSearchManager::StartSearchTermResolutionRequest(
JNIEnv* env,
const JavaParamRef<jobject>& obj,
const base::android::JavaParamRef<jobject>& j_contextual_search_context,
const JavaParamRef<jobject>& j_base_web_contents) {
WebContents* base_web_contents =
WebContents::FromJavaWebContents(j_base_web_contents);
DCHECK(base_web_contents);
base::WeakPtr<ContextualSearchContext> contextual_search_context =
ContextualSearchContext::FromJavaContextualSearchContext(
j_contextual_search_context);
// Calls back to OnSearchTermResolutionResponse.
delegate_->StartSearchTermResolutionRequest(contextual_search_context,
base_web_contents);
}
void ContextualSearchManager::GatherSurroundingText(
JNIEnv* env,
const JavaParamRef<jobject>& obj,
const base::android::JavaParamRef<jobject>& j_contextual_search_context,
const JavaParamRef<jobject>& j_base_web_contents) {
WebContents* base_web_contents =
WebContents::FromJavaWebContents(j_base_web_contents);
DCHECK(base_web_contents);
base::WeakPtr<ContextualSearchContext> contextual_search_context =
ContextualSearchContext::FromJavaContextualSearchContext(
j_contextual_search_context);
delegate_->GatherAndSaveSurroundingText(contextual_search_context,
base_web_contents);
}
base::android::ScopedJavaLocalRef<jstring>
ContextualSearchManager::GetTargetLanguage(JNIEnv* env,
const JavaParamRef<jobject>& obj) {
DCHECK(delegate_);
std::string target_language(delegate_->GetTargetLanguage());
base::android::ScopedJavaLocalRef<jstring> j_target_language =
base::android::ConvertUTF8ToJavaString(env, target_language);
return j_target_language;
}
base::android::ScopedJavaLocalRef<jstring>
ContextualSearchManager::GetAcceptLanguages(JNIEnv* env,
const JavaParamRef<jobject>& obj) {
std::string accept_languages = delegate_->GetAcceptLanguages();
base::android::ScopedJavaLocalRef<jstring> j_accept_languages =
base::android::ConvertUTF8ToJavaString(env, accept_languages);
return j_accept_languages;
}
void ContextualSearchManager::OnSearchTermResolutionResponse(
const ResolvedSearchTerm& resolved_search_term) {
// Notify the Java UX of the result.
JNIEnv* env = base::android::AttachCurrentThread();
base::android::ScopedJavaLocalRef<jstring> j_search_term =
base::android::ConvertUTF8ToJavaString(env,
resolved_search_term.search_term);
base::android::ScopedJavaLocalRef<jstring> j_display_text =
base::android::ConvertUTF8ToJavaString(env,
resolved_search_term.display_text);
base::android::ScopedJavaLocalRef<jstring> j_alternate_term =
base::android::ConvertUTF8ToJavaString(
env, resolved_search_term.alternate_term);
base::android::ScopedJavaLocalRef<jstring> j_mid =
base::android::ConvertUTF8ToJavaString(env, resolved_search_term.mid);
base::android::ScopedJavaLocalRef<jstring> j_context_language =
base::android::ConvertUTF8ToJavaString(
env, resolved_search_term.context_language);
base::android::ScopedJavaLocalRef<jstring> j_thumbnail_url =
base::android::ConvertUTF8ToJavaString(
env, resolved_search_term.thumbnail_url);
base::android::ScopedJavaLocalRef<jstring> j_caption =
base::android::ConvertUTF8ToJavaString(env, resolved_search_term.caption);
base::android::ScopedJavaLocalRef<jstring> j_quick_action_uri =
base::android::ConvertUTF8ToJavaString(
env, resolved_search_term.quick_action_uri);
Java_ContextualSearchManager_onSearchTermResolutionResponse(
env, java_manager_, resolved_search_term.is_invalid,
resolved_search_term.response_code, j_search_term, j_display_text,
j_alternate_term, j_mid, resolved_search_term.prevent_preload,
resolved_search_term.selection_start_adjust,
resolved_search_term.selection_end_adjust, j_context_language,
j_thumbnail_url, j_caption, j_quick_action_uri,
resolved_search_term.quick_action_category);
}
void ContextualSearchManager::OnTextSurroundingSelectionAvailable(
const std::string& encoding,
const base::string16& surrounding_text,
size_t start_offset,
size_t end_offset) {
JNIEnv* env = base::android::AttachCurrentThread();
base::android::ScopedJavaLocalRef<jstring> j_encoding =
base::android::ConvertUTF8ToJavaString(env, encoding.c_str());
base::android::ScopedJavaLocalRef<jstring> j_surrounding_text =
base::android::ConvertUTF16ToJavaString(env, surrounding_text.c_str());
Java_ContextualSearchManager_onTextSurroundingSelectionAvailable(
env, java_manager_, j_encoding, j_surrounding_text, start_offset,
end_offset);
}
void ContextualSearchManager::EnableContextualSearchJsApiForWebContents(
JNIEnv* env,
jobject obj,
const JavaParamRef<jobject>& j_overlay_web_contents) {
DCHECK(j_overlay_web_contents);
WebContents* overlay_web_contents =
WebContents::FromJavaWebContents(j_overlay_web_contents);
DCHECK(overlay_web_contents);
ContextualSearchObserver::EnsureForWebContents(overlay_web_contents, this);
}
void ContextualSearchManager::WhitelistContextualSearchJsApiUrl(
JNIEnv* env,
jobject obj,
const base::android::JavaParamRef<jstring>& j_url) {
DCHECK(j_url);
overlay_gurl_ = GURL(base::android::ConvertJavaStringToUTF8(env, j_url));
}
void ContextualSearchManager::ShouldEnableJsApi(
const GURL& gurl,
contextual_search::mojom::ContextualSearchJsApiService::
ShouldEnableJsApiCallback callback) {
bool should_enable = (gurl == overlay_gurl_);
std::move(callback).Run(should_enable);
}
void ContextualSearchManager::SetCaption(const std::string& caption,
bool does_answer) {
JNIEnv* env = base::android::AttachCurrentThread();
base::android::ScopedJavaLocalRef<jstring> j_caption =
base::android::ConvertUTF8ToJavaString(env, caption.c_str());
Java_ContextualSearchManager_onSetCaption(env, java_manager_, j_caption,
does_answer);
}
void ContextualSearchManager::ChangeOverlayPosition(
contextual_search::mojom::OverlayPosition desired_position) {
JNIEnv* env = base::android::AttachCurrentThread();
Java_ContextualSearchManager_onChangeOverlayPosition(
env, java_manager_, static_cast<int>(desired_position));
}
jlong JNI_ContextualSearchManager_Init(JNIEnv* env,
const JavaParamRef<jobject>& obj) {
ContextualSearchManager* manager = new ContextualSearchManager(env, obj);
return reinterpret_cast<intptr_t>(manager);
}