| // 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/android/shortcut_helper.h" |
| |
| #include <jni.h> |
| #include <utility> |
| |
| #include "base/android/jni_android.h" |
| #include "base/android/jni_array.h" |
| #include "base/android/jni_string.h" |
| #include "base/bind.h" |
| #include "base/callback.h" |
| #include "base/command_line.h" |
| #include "base/strings/string16.h" |
| #include "base/strings/utf_string_conversions.h" |
| #include "chrome/browser/android/webapk/chrome_webapk_host.h" |
| #include "chrome/browser/android/webapk/webapk_install_service.h" |
| #include "chrome/browser/manifest/manifest_icon_downloader.h" |
| #include "chrome/common/chrome_switches.h" |
| #include "content/public/browser/browser_thread.h" |
| #include "content/public/browser/web_contents.h" |
| #include "jni/ShortcutHelper_jni.h" |
| #include "ui/gfx/android/java_bitmap.h" |
| #include "ui/gfx/color_analysis.h" |
| #include "url/gurl.h" |
| |
| using base::android::JavaParamRef; |
| using base::android::ScopedJavaLocalRef; |
| using content::Manifest; |
| |
| namespace { |
| |
| int g_ideal_homescreen_icon_size = -1; |
| int g_minimum_homescreen_icon_size = -1; |
| int g_ideal_splash_image_size = -1; |
| int g_minimum_splash_image_size = -1; |
| int g_ideal_badge_icon_size = -1; |
| |
| int g_default_rgb_icon_value = 145; |
| |
| // Retrieves and caches the ideal and minimum sizes of the Home screen icon |
| // and the splash screen image. |
| void GetHomescreenIconAndSplashImageSizes() { |
| JNIEnv* env = base::android::AttachCurrentThread(); |
| ScopedJavaLocalRef<jintArray> java_size_array = |
| Java_ShortcutHelper_getHomeScreenIconAndSplashImageSizes(env); |
| std::vector<int> sizes; |
| base::android::JavaIntArrayToIntVector(env, java_size_array.obj(), &sizes); |
| |
| // Check that the size returned is what is expected. |
| DCHECK(sizes.size() == 5); |
| |
| // This ordering must be kept up to date with the Java ShortcutHelper. |
| g_ideal_homescreen_icon_size = sizes[0]; |
| g_minimum_homescreen_icon_size = sizes[1]; |
| g_ideal_splash_image_size = sizes[2]; |
| g_minimum_splash_image_size = sizes[3]; |
| g_ideal_badge_icon_size = sizes[4]; |
| |
| // Try to ensure that the data returned is sane. |
| DCHECK(g_minimum_homescreen_icon_size <= g_ideal_homescreen_icon_size); |
| DCHECK(g_minimum_splash_image_size <= g_ideal_splash_image_size); |
| } |
| |
| } // anonymous namespace |
| |
| // static |
| void ShortcutHelper::AddToLauncherWithSkBitmap( |
| content::BrowserContext* browser_context, |
| const ShortcutInfo& info, |
| const std::string& webapp_id, |
| const SkBitmap& icon_bitmap, |
| const base::Closure& splash_image_callback) { |
| if (info.display == blink::WebDisplayModeStandalone || |
| info.display == blink::WebDisplayModeFullscreen) { |
| AddWebappWithSkBitmap(info, webapp_id, icon_bitmap, splash_image_callback); |
| return; |
| } |
| AddShortcutWithSkBitmap(info, icon_bitmap); |
| } |
| |
| // static |
| void ShortcutHelper::InstallWebApkWithSkBitmap( |
| content::BrowserContext* browser_context, |
| const ShortcutInfo& info, |
| const SkBitmap& icon_bitmap, |
| const WebApkInstaller::FinishCallback& callback) { |
| WebApkInstallService::Get(browser_context) |
| ->InstallAsync(info, icon_bitmap, callback); |
| } |
| |
| // static |
| void ShortcutHelper::AddWebappWithSkBitmap( |
| const ShortcutInfo& info, |
| const std::string& webapp_id, |
| const SkBitmap& icon_bitmap, |
| const base::Closure& splash_image_callback) { |
| // Send the data to the Java side to create the shortcut. |
| JNIEnv* env = base::android::AttachCurrentThread(); |
| ScopedJavaLocalRef<jstring> java_webapp_id = |
| base::android::ConvertUTF8ToJavaString(env, webapp_id); |
| ScopedJavaLocalRef<jstring> java_url = |
| base::android::ConvertUTF8ToJavaString(env, info.url.spec()); |
| ScopedJavaLocalRef<jstring> java_scope_url = |
| base::android::ConvertUTF8ToJavaString(env, info.scope.spec()); |
| ScopedJavaLocalRef<jstring> java_user_title = |
| base::android::ConvertUTF16ToJavaString(env, info.user_title); |
| ScopedJavaLocalRef<jstring> java_name = |
| base::android::ConvertUTF16ToJavaString(env, info.name); |
| ScopedJavaLocalRef<jstring> java_short_name = |
| base::android::ConvertUTF16ToJavaString(env, info.short_name); |
| ScopedJavaLocalRef<jstring> java_best_primary_icon_url = |
| base::android::ConvertUTF8ToJavaString(env, |
| info.best_primary_icon_url.spec()); |
| ScopedJavaLocalRef<jobject> java_bitmap; |
| if (icon_bitmap.getSize()) |
| java_bitmap = gfx::ConvertToJavaBitmap(&icon_bitmap); |
| |
| // The callback will need to be run after shortcut creation completes in order |
| // to download the splash image and save it to the WebappDataStorage. Create a |
| // copy of the callback here and send the pointer to Java, which will send it |
| // back once the asynchronous shortcut creation process finishes. |
| uintptr_t callback_pointer = |
| reinterpret_cast<uintptr_t>(new base::Closure(splash_image_callback)); |
| |
| Java_ShortcutHelper_addWebapp( |
| env, java_webapp_id, java_url, java_scope_url, java_user_title, java_name, |
| java_short_name, java_best_primary_icon_url, java_bitmap, info.display, |
| info.orientation, info.source, info.theme_color, info.background_color, |
| callback_pointer); |
| } |
| |
| void ShortcutHelper::AddShortcutWithSkBitmap(const ShortcutInfo& info, |
| const SkBitmap& icon_bitmap) { |
| JNIEnv* env = base::android::AttachCurrentThread(); |
| ScopedJavaLocalRef<jstring> java_url = |
| base::android::ConvertUTF8ToJavaString(env, info.url.spec()); |
| ScopedJavaLocalRef<jstring> java_user_title = |
| base::android::ConvertUTF16ToJavaString(env, info.user_title); |
| ScopedJavaLocalRef<jobject> java_bitmap; |
| if (icon_bitmap.getSize()) |
| java_bitmap = gfx::ConvertToJavaBitmap(&icon_bitmap); |
| |
| Java_ShortcutHelper_addShortcut(env, java_url, java_user_title, java_bitmap, |
| info.source); |
| } |
| |
| void ShortcutHelper::ShowWebApkInstallInProgressToast() { |
| Java_ShortcutHelper_showWebApkInstallInProgressToast( |
| base::android::AttachCurrentThread()); |
| } |
| |
| int ShortcutHelper::GetIdealHomescreenIconSizeInPx() { |
| if (g_ideal_homescreen_icon_size == -1) |
| GetHomescreenIconAndSplashImageSizes(); |
| return g_ideal_homescreen_icon_size; |
| } |
| |
| int ShortcutHelper::GetMinimumHomescreenIconSizeInPx() { |
| if (g_minimum_homescreen_icon_size == -1) |
| GetHomescreenIconAndSplashImageSizes(); |
| return g_minimum_homescreen_icon_size; |
| } |
| |
| int ShortcutHelper::GetIdealSplashImageSizeInPx() { |
| if (g_ideal_splash_image_size == -1) |
| GetHomescreenIconAndSplashImageSizes(); |
| return g_ideal_splash_image_size; |
| } |
| |
| int ShortcutHelper::GetMinimumSplashImageSizeInPx() { |
| if (g_minimum_splash_image_size == -1) |
| GetHomescreenIconAndSplashImageSizes(); |
| return g_minimum_splash_image_size; |
| } |
| |
| int ShortcutHelper::GetIdealBadgeIconSizeInPx() { |
| if (g_ideal_badge_icon_size == -1) |
| GetHomescreenIconAndSplashImageSizes(); |
| return g_ideal_badge_icon_size; |
| } |
| |
| // static |
| void ShortcutHelper::FetchSplashScreenImage( |
| content::WebContents* web_contents, |
| const GURL& image_url, |
| const int ideal_splash_image_size_in_px, |
| const int minimum_splash_image_size_in_px, |
| const std::string& webapp_id) { |
| // This is a fire and forget task. It is not vital for the splash screen image |
| // to be downloaded so if the downloader returns false there is no fallback. |
| ManifestIconDownloader::Download( |
| web_contents, image_url, ideal_splash_image_size_in_px, |
| minimum_splash_image_size_in_px, |
| base::Bind(&ShortcutHelper::StoreWebappSplashImage, webapp_id)); |
| } |
| |
| // static |
| void ShortcutHelper::StoreWebappSplashImage(const std::string& webapp_id, |
| const SkBitmap& splash_image) { |
| if (splash_image.drawsNothing()) |
| return; |
| |
| JNIEnv* env = base::android::AttachCurrentThread(); |
| ScopedJavaLocalRef<jstring> java_webapp_id = |
| base::android::ConvertUTF8ToJavaString(env, webapp_id); |
| ScopedJavaLocalRef<jobject> java_splash_image = |
| gfx::ConvertToJavaBitmap(&splash_image); |
| |
| Java_ShortcutHelper_storeWebappSplashImage(env, java_webapp_id, |
| java_splash_image); |
| } |
| |
| // static |
| SkBitmap ShortcutHelper::FinalizeLauncherIconInBackground( |
| const SkBitmap& bitmap, |
| const GURL& url, |
| bool* is_generated) { |
| DCHECK(content::BrowserThread::GetBlockingPool()->RunsTasksOnCurrentThread()); |
| |
| JNIEnv* env = base::android::AttachCurrentThread(); |
| ScopedJavaLocalRef<jobject> result; |
| *is_generated = false; |
| |
| if (!bitmap.isNull()) { |
| if (Java_ShortcutHelper_isIconLargeEnoughForLauncher(env, bitmap.width(), |
| bitmap.height())) { |
| ScopedJavaLocalRef<jobject> java_bitmap = |
| gfx::ConvertToJavaBitmap(&bitmap); |
| result = |
| Java_ShortcutHelper_createHomeScreenIconFromWebIcon(env, java_bitmap); |
| } |
| } |
| |
| if (result.is_null()) { |
| ScopedJavaLocalRef<jstring> java_url = |
| base::android::ConvertUTF8ToJavaString(env, url.spec()); |
| SkColor mean_color = |
| SkColorSetRGB(g_default_rgb_icon_value, g_default_rgb_icon_value, |
| g_default_rgb_icon_value); |
| |
| if (!bitmap.isNull()) |
| mean_color = color_utils::CalculateKMeanColorOfBitmap(bitmap); |
| |
| *is_generated = true; |
| result = Java_ShortcutHelper_generateHomeScreenIcon( |
| env, java_url, SkColorGetR(mean_color), SkColorGetG(mean_color), |
| SkColorGetB(mean_color)); |
| } |
| |
| return result.obj() |
| ? gfx::CreateSkBitmapFromJavaBitmap(gfx::JavaBitmap(result)) |
| : SkBitmap(); |
| } |
| |
| // static |
| std::string ShortcutHelper::QueryWebApkPackage(const GURL& url) { |
| JNIEnv* env = base::android::AttachCurrentThread(); |
| ScopedJavaLocalRef<jstring> java_url = |
| base::android::ConvertUTF8ToJavaString(env, url.spec()); |
| ScopedJavaLocalRef<jstring> java_webapk_package_name = |
| Java_ShortcutHelper_queryWebApkPackage(env, java_url); |
| |
| std::string webapk_package_name = ""; |
| if (java_webapk_package_name.obj()) { |
| webapk_package_name = |
| base::android::ConvertJavaStringToUTF8(env, java_webapk_package_name); |
| } |
| return webapk_package_name; |
| } |
| |
| // static |
| bool ShortcutHelper::IsWebApkInstalled(content::BrowserContext* browser_context, |
| const GURL& start_url, |
| const GURL& manifest_url) { |
| return !QueryWebApkPackage(start_url).empty() || |
| WebApkInstallService::Get(browser_context) |
| ->IsInstallInProgress(manifest_url); |
| } |
| |
| GURL ShortcutHelper::GetScopeFromURL(const GURL& url) { |
| JNIEnv* env = base::android::AttachCurrentThread(); |
| ScopedJavaLocalRef<jstring> java_url = |
| base::android::ConvertUTF8ToJavaString(env, url.spec()); |
| ScopedJavaLocalRef<jstring> java_scope_url = |
| Java_ShortcutHelper_getScopeFromUrl(env, java_url); |
| return GURL(base::android::ConvertJavaStringToUTF16(env, java_scope_url)); |
| } |
| |
| void ShortcutHelper::RetrieveWebApks(const WebApkInfoCallback& callback) { |
| uintptr_t callback_pointer = |
| reinterpret_cast<uintptr_t>(new WebApkInfoCallback(callback)); |
| Java_ShortcutHelper_retrieveWebApks(base::android::AttachCurrentThread(), |
| callback_pointer); |
| } |
| |
| // Callback used by Java when the shortcut has been created. |
| // |splash_image_callback| is a pointer to a base::Closure allocated in |
| // AddShortcutWithSkBitmap, so reinterpret_cast it back and run it. |
| // |
| // This callback should only ever be called when the shortcut was for a |
| // webapp-capable site; otherwise, |splash_image_callback| will have never been |
| // allocated and doesn't need to be run or deleted. |
| void OnWebappDataStored(JNIEnv* env, |
| const JavaParamRef<jclass>& clazz, |
| jlong jsplash_image_callback) { |
| DCHECK(jsplash_image_callback); |
| base::Closure* splash_image_callback = |
| reinterpret_cast<base::Closure*>(jsplash_image_callback); |
| splash_image_callback->Run(); |
| delete splash_image_callback; |
| } |
| |
| void OnWebApksRetrieved(JNIEnv* env, |
| const JavaParamRef<jclass>& clazz, |
| const jlong jcallback_pointer, |
| const JavaParamRef<jobjectArray>& jshort_names, |
| const JavaParamRef<jobjectArray>& jpackage_names, |
| const JavaParamRef<jintArray>& jshell_apk_versions, |
| const JavaParamRef<jintArray>& jversion_codes) { |
| DCHECK(jcallback_pointer); |
| std::vector<std::string> short_names; |
| base::android::AppendJavaStringArrayToStringVector(env, jshort_names, |
| &short_names); |
| std::vector<std::string> package_names; |
| base::android::AppendJavaStringArrayToStringVector(env, jpackage_names, |
| &package_names); |
| std::vector<int> shell_apk_versions; |
| base::android::JavaIntArrayToIntVector(env, jshell_apk_versions, |
| &shell_apk_versions); |
| std::vector<int> version_codes; |
| base::android::JavaIntArrayToIntVector(env, jversion_codes, &version_codes); |
| |
| DCHECK(short_names.size() == package_names.size()); |
| DCHECK(short_names.size() == shell_apk_versions.size()); |
| DCHECK(short_names.size() == version_codes.size()); |
| |
| std::vector<WebApkInfo> webapk_list; |
| webapk_list.reserve(short_names.size()); |
| for (size_t i = 0; i < short_names.size(); ++i) { |
| webapk_list.push_back(WebApkInfo(std::move(short_names[i]), |
| std::move(package_names[i]), |
| shell_apk_versions[i], version_codes[i])); |
| } |
| |
| ShortcutHelper::WebApkInfoCallback* webapk_list_callback = |
| reinterpret_cast<ShortcutHelper::WebApkInfoCallback*>(jcallback_pointer); |
| webapk_list_callback->Run(webapk_list); |
| delete webapk_list_callback; |
| } |
| |
| bool ShortcutHelper::RegisterShortcutHelper(JNIEnv* env) { |
| return RegisterNativesImpl(env); |
| } |