| // Copyright 2013 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/ui/android/context_menu_helper.h" |
| |
| #include <stdint.h> |
| |
| #include <vector> |
| |
| #include "base/android/callback_android.h" |
| #include "base/android/jni_string.h" |
| #include "base/bind_helpers.h" |
| #include "base/callback.h" |
| #include "chrome/browser/android/download/download_controller_base.h" |
| #include "chrome/browser/ui/tab_contents/core_tab_helper.h" |
| #include "chrome/common/thumbnail_capturer.mojom.h" |
| #include "components/data_reduction_proxy/core/common/data_reduction_proxy_headers.h" |
| #include "content/public/browser/android/content_view_core.h" |
| #include "content/public/browser/render_frame_host.h" |
| #include "content/public/browser/render_process_host.h" |
| #include "content/public/common/context_menu_params.h" |
| #include "jni/ContextMenuHelper_jni.h" |
| #include "jni/ContextMenuParams_jni.h" |
| #include "services/service_manager/public/cpp/interface_provider.h" |
| #include "third_party/WebKit/public/web/WebContextMenuData.h" |
| #include "ui/gfx/geometry/point.h" |
| #include "ui/gfx/geometry/size.h" |
| |
| using base::android::ConvertJavaStringToUTF8; |
| using base::android::ConvertUTF8ToJavaString; |
| using base::android::ConvertUTF16ToJavaString; |
| using base::android::JavaParamRef; |
| |
| DEFINE_WEB_CONTENTS_USER_DATA_KEY(ContextMenuHelper); |
| |
| const char kDataReductionProxyPassthroughHeader[] = |
| "Chrome-Proxy-Accept-Transform: identity\r\n"; |
| |
| namespace { |
| |
| void OnRetrieveImage(chrome::mojom::ThumbnailCapturerPtr thumbnail_capturer, |
| const base::android::JavaRef<jobject>& jcallback, |
| const std::vector<uint8_t>& thumbnail_data, |
| const gfx::Size& original_size) { |
| base::android::RunCallbackAndroid(jcallback, thumbnail_data); |
| } |
| |
| } // namespace |
| |
| ContextMenuHelper::ContextMenuHelper(content::WebContents* web_contents) |
| : web_contents_(web_contents) { |
| JNIEnv* env = base::android::AttachCurrentThread(); |
| java_obj_.Reset( |
| env, |
| Java_ContextMenuHelper_create(env, reinterpret_cast<long>(this)).obj()); |
| DCHECK(!java_obj_.is_null()); |
| } |
| |
| ContextMenuHelper::~ContextMenuHelper() { |
| JNIEnv* env = base::android::AttachCurrentThread(); |
| Java_ContextMenuHelper_destroy(env, java_obj_); |
| } |
| |
| void ContextMenuHelper::ShowContextMenu( |
| content::RenderFrameHost* render_frame_host, |
| const content::ContextMenuParams& params) { |
| content::ContentViewCore* content_view_core = |
| content::ContentViewCore::FromWebContents(web_contents_); |
| |
| if (!content_view_core) |
| return; |
| |
| base::android::ScopedJavaLocalRef<jobject> jcontent_view_core( |
| content_view_core->GetJavaObject()); |
| |
| if (jcontent_view_core.is_null()) |
| return; |
| |
| JNIEnv* env = base::android::AttachCurrentThread(); |
| context_menu_params_ = params; |
| render_frame_id_ = render_frame_host->GetRoutingID(); |
| render_process_id_ = render_frame_host->GetProcess()->GetID(); |
| |
| Java_ContextMenuHelper_showContextMenu( |
| env, java_obj_, jcontent_view_core, |
| ContextMenuHelper::CreateJavaContextMenuParams(params)); |
| } |
| |
| void ContextMenuHelper::OnContextMenuClosed( |
| JNIEnv* env, |
| const base::android::JavaParamRef<jobject>& obj) { |
| web_contents_->NotifyContextMenuClosed(context_menu_params_.custom_context); |
| } |
| |
| void ContextMenuHelper::SetPopulator(jobject jpopulator) { |
| JNIEnv* env = base::android::AttachCurrentThread(); |
| Java_ContextMenuHelper_setPopulator(env, java_obj_, jpopulator); |
| } |
| |
| base::android::ScopedJavaLocalRef<jobject> |
| ContextMenuHelper::CreateJavaContextMenuParams( |
| const content::ContextMenuParams& params) { |
| GURL sanitizedReferrer = (params.frame_url.is_empty() ? |
| params.page_url : params.frame_url).GetAsReferrer(); |
| |
| std::map<std::string, std::string>::const_iterator it = |
| params.properties.find( |
| data_reduction_proxy::chrome_proxy_content_transform_header()); |
| bool image_was_fetched_lo_fi = |
| it != params.properties.end() && |
| it->second == data_reduction_proxy::empty_image_directive(); |
| bool can_save = params.media_flags & blink::WebContextMenuData::kMediaCanSave; |
| JNIEnv* env = base::android::AttachCurrentThread(); |
| base::android::ScopedJavaLocalRef<jobject> jmenu_info = |
| ContextMenuParamsAndroid::Java_ContextMenuParams_create( |
| env, params.media_type, |
| ConvertUTF8ToJavaString(env, params.page_url.spec()), |
| ConvertUTF8ToJavaString(env, params.link_url.spec()), |
| ConvertUTF16ToJavaString(env, params.link_text), |
| ConvertUTF8ToJavaString(env, params.unfiltered_link_url.spec()), |
| ConvertUTF8ToJavaString(env, params.src_url.spec()), |
| ConvertUTF16ToJavaString(env, params.title_text), |
| image_was_fetched_lo_fi, |
| ConvertUTF8ToJavaString(env, sanitizedReferrer.spec()), |
| params.referrer_policy, can_save, params.x, params.y); |
| |
| return jmenu_info; |
| } |
| |
| base::android::ScopedJavaLocalRef<jobject> |
| ContextMenuHelper::GetJavaWebContents(JNIEnv* env, |
| const JavaParamRef<jobject>& obj) { |
| return web_contents_->GetJavaWebContents(); |
| } |
| |
| void ContextMenuHelper::OnStartDownload( |
| JNIEnv* env, |
| const JavaParamRef<jobject>& obj, |
| jboolean jis_link, |
| jboolean jis_data_reduction_proxy_enabled) { |
| std::string headers; |
| if (jis_data_reduction_proxy_enabled) |
| headers = kDataReductionProxyPassthroughHeader; |
| |
| DownloadControllerBase::Get()->StartContextMenuDownload( |
| context_menu_params_, |
| web_contents_, |
| jis_link, |
| headers); |
| } |
| |
| void ContextMenuHelper::SearchForImage(JNIEnv* env, |
| const JavaParamRef<jobject>& obj) { |
| content::RenderFrameHost* render_frame_host = |
| content::RenderFrameHost::FromID(render_process_id_, render_frame_id_); |
| if (!render_frame_host) |
| return; |
| |
| CoreTabHelper::FromWebContents(web_contents_)->SearchByImageInNewTab( |
| render_frame_host, context_menu_params_.src_url); |
| } |
| |
| void ContextMenuHelper::RetrieveImage(JNIEnv* env, |
| const JavaParamRef<jobject>& obj, |
| const JavaParamRef<jobject>& jcallback, |
| jint max_dimen_px) { |
| content::RenderFrameHost* render_frame_host = |
| content::RenderFrameHost::FromID(render_process_id_, render_frame_id_); |
| if (!render_frame_host) |
| return; |
| |
| chrome::mojom::ThumbnailCapturerPtr thumbnail_capturer; |
| render_frame_host->GetRemoteInterfaces()->GetInterface(&thumbnail_capturer); |
| // Bind the InterfacePtr into the callback so that it's kept alive until |
| // there's either a connection error or a response. |
| auto* thumbnail_capturer_proxy = thumbnail_capturer.get(); |
| thumbnail_capturer_proxy->RequestThumbnailForContextNode( |
| 0, gfx::Size(max_dimen_px, max_dimen_px), |
| base::Bind(&OnRetrieveImage, base::Passed(&thumbnail_capturer), |
| base::android::ScopedJavaGlobalRef<jobject>(env, jcallback))); |
| } |
| |
| bool RegisterContextMenuHelper(JNIEnv* env) { |
| return RegisterNativesImpl(env); |
| } |