Index: chrome/browser/android/contextualsearch/contextual_search_manager.cc |
diff --git a/chrome/browser/android/contextualsearch/contextual_search_manager.cc b/chrome/browser/android/contextualsearch/contextual_search_manager.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..64bd88c3c70553a5e67bbe215a57133d62fa9ea0 |
--- /dev/null |
+++ b/chrome/browser/android/contextualsearch/contextual_search_manager.cc |
@@ -0,0 +1,281 @@ |
+// 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 <set> |
+ |
+#include "base/android/jni_string.h" |
+#include "base/callback.h" |
+#include "base/memory/weak_ptr.h" |
+#include "base/time/time.h" |
+#include "chrome/browser/android/contextualsearch/contextual_search_delegate.h" |
+#include "chrome/browser/android/tab_android.h" |
+#include "chrome/browser/history/history_service_factory.h" |
+#include "chrome/browser/profiles/profile_manager.h" |
+#include "chrome/browser/search/contextual_search_promo_source_android.h" |
+#include "chrome/browser/search_engines/template_url_service_factory.h" |
+#include "chrome/browser/ui/android/window_android_helper.h" |
+#include "components/history/core/browser/history_service.h" |
+#include "components/navigation_interception/intercept_navigation_delegate.h" |
+#include "components/variations/variations_associated_data.h" |
+#include "components/web_contents_delegate_android/web_contents_delegate_android.h" |
+#include "content/public/browser/android/content_view_core.h" |
+#include "content/public/browser/web_contents.h" |
+#include "jni/ContextualSearchManager_jni.h" |
+#include "net/url_request/url_fetcher_impl.h" |
+ |
+using content::ContentViewCore; |
+ |
+namespace { |
+ |
+// Field trial related constants. |
+const char kContextualSearchFieldTrialName[] = "ContextualSearch"; |
+const char kContextualSearchHidePromoHeaderParam[] = "hide_promo_header"; |
+const char kContextualSearchEnabledValue[] = "enabled"; |
+ |
+const int kHistoryDeletionWindowSeconds = 2; |
+ |
+// Because we need a callback, this needs to exist. |
+void OnHistoryDeletionDone() { |
+} |
+ |
+} // 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, jobject obj) { |
+ java_manager_.Reset(env, obj); |
+ Java_ContextualSearchManager_setNativeManager( |
+ env, obj, reinterpret_cast<intptr_t>(this)); |
+ Profile* profile = ProfileManager::GetActiveUserProfile(); |
+ delegate_.reset(new ContextualSearchDelegate( |
+ profile->GetRequestContext(), |
+ TemplateURLServiceFactory::GetForProfile(profile), |
+ base::Bind(&ContextualSearchManager::OnSearchTermResolutionResponse, |
+ base::Unretained(this)), |
+ base::Bind(&ContextualSearchManager::OnSurroundingTextAvailable, |
+ base::Unretained(this)), |
+ base::Bind(&ContextualSearchManager::OnIcingSelectionAvailable, |
+ base::Unretained(this)))); |
+ content::URLDataSource::Add(ProfileManager::GetActiveUserProfile(), |
+ new ContextualSearchPromoSourceAndroid()); |
+} |
+ |
+ContextualSearchManager::~ContextualSearchManager() { |
+ JNIEnv* env = base::android::AttachCurrentThread(); |
+ Java_ContextualSearchManager_clearNativeManager(env, java_manager_.obj()); |
+} |
+ |
+void ContextualSearchManager::Destroy(JNIEnv* env, jobject obj) { delete this; } |
+ |
+void ContextualSearchManager::StartSearchTermResolutionRequest( |
+ JNIEnv* env, |
+ jobject obj, |
+ jstring j_selection, |
+ jboolean j_use_resolved_search_term, |
+ jobject j_base_content_view_core) { |
+ ContentViewCore* base_content_view_core = |
+ ContentViewCore::GetNativeContentViewCore(env, j_base_content_view_core); |
+ DCHECK(base_content_view_core); |
+ std::string selection( |
+ base::android::ConvertJavaStringToUTF8(env, j_selection)); |
+ bool use_resolved_search_term = j_use_resolved_search_term; |
+ // Calls back to OnSearchTermResolutionResponse. |
+ delegate_->StartSearchTermResolutionRequest( |
+ selection, use_resolved_search_term, base_content_view_core); |
+} |
+ |
+void ContextualSearchManager::GatherSurroundingText( |
+ JNIEnv* env, |
+ jobject obj, |
+ jstring j_selection, |
+ jboolean j_use_resolved_search_term, |
+ jobject j_base_content_view_core) { |
+ ContentViewCore* base_content_view_core = |
+ ContentViewCore::GetNativeContentViewCore(env, j_base_content_view_core); |
+ DCHECK(base_content_view_core); |
+ std::string selection( |
+ base::android::ConvertJavaStringToUTF8(env, j_selection)); |
+ bool use_resolved_search_term = j_use_resolved_search_term; |
+ delegate_->GatherAndSaveSurroundingText( |
+ selection, use_resolved_search_term, base_content_view_core); |
+} |
+ |
+void ContextualSearchManager::ContinueSearchTermResolutionRequest(JNIEnv* env, |
+ jobject obj) { |
+ delegate_->ContinueSearchTermResolutionRequest(); |
+} |
+ |
+void ContextualSearchManager::OnSearchTermResolutionResponse( |
+ bool is_invalid, |
+ int response_code, |
+ const std::string& search_term, |
+ const std::string& display_text, |
+ const std::string& alternate_term, |
+ bool prevent_preload) { |
+ // Notify the Java UX of the result. |
+ JNIEnv* env = base::android::AttachCurrentThread(); |
+ base::android::ScopedJavaLocalRef<jstring> j_search_term = |
+ base::android::ConvertUTF8ToJavaString(env, search_term.c_str()); |
+ base::android::ScopedJavaLocalRef<jstring> j_display_text = |
+ base::android::ConvertUTF8ToJavaString(env, display_text.c_str()); |
+ base::android::ScopedJavaLocalRef<jstring> j_alternate_term = |
+ base::android::ConvertUTF8ToJavaString(env, alternate_term.c_str()); |
+ Java_ContextualSearchManager_onSearchTermResolutionResponse( |
+ env, |
+ java_manager_.obj(), |
+ is_invalid, |
+ response_code, |
+ j_search_term.obj(), |
+ j_display_text.obj(), |
+ j_alternate_term.obj(), |
+ prevent_preload); |
+} |
+ |
+void ContextualSearchManager::OnSurroundingTextAvailable( |
+ const std::string& before_text, |
+ const std::string& after_text) { |
+ JNIEnv* env = base::android::AttachCurrentThread(); |
+ base::android::ScopedJavaLocalRef<jstring> j_before_text = |
+ base::android::ConvertUTF8ToJavaString(env, before_text.c_str()); |
+ base::android::ScopedJavaLocalRef<jstring> j_after_text = |
+ base::android::ConvertUTF8ToJavaString(env, after_text.c_str()); |
+ Java_ContextualSearchManager_onSurroundingTextAvailable( |
+ env, |
+ java_manager_.obj(), |
+ j_before_text.obj(), |
+ j_after_text.obj()); |
+} |
+ |
+void ContextualSearchManager::OnIcingSelectionAvailable( |
+ 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_onIcingSelectionAvailable( |
+ env, java_manager_.obj(), j_encoding.obj(), j_surrounding_text.obj(), |
+ start_offset, end_offset); |
+} |
+ |
+void ContextualSearchManager::RemoveLastSearchVisit( |
+ JNIEnv* env, |
+ jobject obj, |
+ jstring search_url, |
+ jlong search_start_time_ms) { |
+ // The deletion window is from the time a search URL was put in history, up |
+ // to a short amount of time later. |
+ base::Time begin_time = base::Time::FromJsTime(search_start_time_ms); |
+ base::Time end_time = begin_time + |
+ base::TimeDelta::FromSeconds(kHistoryDeletionWindowSeconds); |
+ |
+ history::HistoryService* service = HistoryServiceFactory::GetForProfile( |
+ ProfileManager::GetActiveUserProfile(), |
+ ServiceAccessType::EXPLICIT_ACCESS); |
+ if (service) { |
+ // NOTE(mathp): We are only removing |search_url| from the local history |
+ // because search results that are not promoted to a Tab do not make it to |
+ // the web history, only local. |
+ std::set<GURL> restrict_set; |
+ restrict_set.insert( |
+ GURL(base::android::ConvertJavaStringToUTF8(env, search_url))); |
+ service->ExpireHistoryBetween( |
+ restrict_set, |
+ begin_time, |
+ end_time, |
+ base::Bind(&OnHistoryDeletionDone), |
+ &history_task_tracker_); |
+ } |
+} |
+ |
+void ContextualSearchManager::SetWebContents(JNIEnv* env, |
+ jobject obj, |
+ jobject jcontent_view_core, |
+ jobject jweb_contents_delegate) { |
+ content::ContentViewCore* content_view_core = |
+ content::ContentViewCore::GetNativeContentViewCore(env, |
+ jcontent_view_core); |
+ DCHECK(content_view_core); |
+ DCHECK(content_view_core->GetWebContents()); |
+ |
+ // NOTE(pedrosimonetti): Takes ownership of the WebContents associated |
+ // with the ContentViewCore. This is to make sure that the WebContens |
+ // and the Compositor are in the same process. |
+ // TODO(pedrosimonetti): Confirm with dtrainor@ if the comment above |
+ // is accurate. |
+ web_contents_.reset(content_view_core->GetWebContents()); |
+ // TODO(pedrosimonetti): confirm if we need this after promoting it |
+ // to a real tab. |
+ TabAndroid::AttachTabHelpers(web_contents_.get()); |
+ WindowAndroidHelper::FromWebContents(web_contents_.get()) |
+ ->SetWindowAndroid(content_view_core->GetWindowAndroid()); |
+ web_contents_delegate_.reset( |
+ new web_contents_delegate_android::WebContentsDelegateAndroid( |
+ env, jweb_contents_delegate)); |
+ web_contents_->SetDelegate(web_contents_delegate_.get()); |
+} |
+ |
+void ContextualSearchManager::DestroyWebContents(JNIEnv* env, jobject jobj) { |
+ DCHECK(web_contents_.get()); |
+ web_contents_.reset(); |
+ // |web_contents_delegate_| may already be NULL at this point. |
+ web_contents_delegate_.reset(); |
+} |
+ |
+void ContextualSearchManager::ReleaseWebContents(JNIEnv* env, jobject jboj) { |
+ DCHECK(web_contents_.get()); |
+ web_contents_delegate_.reset(); |
+ ignore_result(web_contents_.release()); |
+} |
+ |
+void ContextualSearchManager::DestroyWebContentsFromContentViewCore( |
+ JNIEnv* env, |
+ jobject jobj, |
+ jobject jcontent_view_core) { |
+ content::ContentViewCore* content_view_core = |
+ content::ContentViewCore::GetNativeContentViewCore(env, |
+ jcontent_view_core); |
+ DCHECK(content_view_core); |
+ DCHECK(content_view_core->GetWebContents()); |
+ |
+ delete content_view_core->GetWebContents(); |
+} |
+ |
+bool ContextualSearchManager::ShouldHidePromoHeader(JNIEnv* env, |
+ jobject jobj) { |
+ return variations::GetVariationParamValue( |
+ kContextualSearchFieldTrialName, |
+ kContextualSearchHidePromoHeaderParam) == |
+ kContextualSearchEnabledValue; |
+} |
+ |
+void ContextualSearchManager::SetInterceptNavigationDelegate( |
+ JNIEnv* env, |
+ jobject obj, |
+ jobject delegate, |
+ jobject jweb_contents) { |
+ content::WebContents* web_contents = |
+ content::WebContents::FromJavaWebContents(jweb_contents); |
+ DCHECK(web_contents); |
+ navigation_interception::InterceptNavigationDelegate::Associate( |
+ web_contents, |
+ make_scoped_ptr(new navigation_interception::InterceptNavigationDelegate( |
+ env, delegate))); |
+} |
+ |
+bool RegisterContextualSearchManager(JNIEnv* env) { |
+ return RegisterNativesImpl(env); |
+} |
+ |
+jlong Init(JNIEnv* env, jobject obj) { |
+ ContextualSearchManager* manager = new ContextualSearchManager(env, obj); |
+ return reinterpret_cast<intptr_t>(manager); |
+} |