| OLD | NEW |
| 1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 2013 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "chrome/browser/android/shortcut_helper.h" | 5 #include "chrome/browser/android/shortcut_helper.h" |
| 6 | 6 |
| 7 #include <jni.h> | 7 #include <jni.h> |
| 8 | 8 |
| 9 #include "base/android/jni_android.h" | 9 #include "base/android/jni_android.h" |
| 10 #include "base/android/jni_string.h" | 10 #include "base/android/jni_string.h" |
| (...skipping 10 matching lines...) Expand all Loading... |
| 21 #include "content/public/browser/web_contents.h" | 21 #include "content/public/browser/web_contents.h" |
| 22 #include "content/public/browser/web_contents_observer.h" | 22 #include "content/public/browser/web_contents_observer.h" |
| 23 #include "content/public/common/frame_navigate_params.h" | 23 #include "content/public/common/frame_navigate_params.h" |
| 24 #include "jni/ShortcutHelper_jni.h" | 24 #include "jni/ShortcutHelper_jni.h" |
| 25 #include "ui/gfx/android/java_bitmap.h" | 25 #include "ui/gfx/android/java_bitmap.h" |
| 26 #include "ui/gfx/codec/png_codec.h" | 26 #include "ui/gfx/codec/png_codec.h" |
| 27 #include "ui/gfx/color_analysis.h" | 27 #include "ui/gfx/color_analysis.h" |
| 28 #include "ui/gfx/favicon_size.h" | 28 #include "ui/gfx/favicon_size.h" |
| 29 #include "url/gurl.h" | 29 #include "url/gurl.h" |
| 30 | 30 |
| 31 ShortcutBuilder::ShortcutBuilder(content::WebContents* web_contents, | 31 jlong Initialize(JNIEnv* env, jobject obj, jlong tab_android_ptr) { |
| 32 const base::string16& title, | 32 TabAndroid* tab = reinterpret_cast<TabAndroid*>(tab_android_ptr); |
| 33 int launcher_large_icon_size) | 33 |
| 34 : launcher_large_icon_size_(launcher_large_icon_size), | 34 ShortcutHelper* shortcut_helper = |
| 35 new ShortcutHelper(env, obj, tab->web_contents()); |
| 36 shortcut_helper->Initialize(); |
| 37 |
| 38 return reinterpret_cast<intptr_t>(shortcut_helper); |
| 39 } |
| 40 |
| 41 ShortcutHelper::ShortcutHelper(JNIEnv* env, |
| 42 jobject obj, |
| 43 content::WebContents* web_contents) |
| 44 : WebContentsObserver(web_contents), |
| 45 java_ref_(env, obj), |
| 46 url_(web_contents->GetURL()), |
| 35 shortcut_type_(BOOKMARK) { | 47 shortcut_type_(BOOKMARK) { |
| 36 Observe(web_contents); | 48 } |
| 37 url_ = web_contents->GetURL(); | |
| 38 if (title.length() > 0) | |
| 39 title_ = title; | |
| 40 else | |
| 41 title_ = web_contents->GetTitle(); | |
| 42 | 49 |
| 50 void ShortcutHelper::Initialize() { |
| 43 // Send a message to the renderer to retrieve information about the page. | 51 // Send a message to the renderer to retrieve information about the page. |
| 44 Send(new ChromeViewMsg_RetrieveWebappInformation(routing_id(), url_)); | 52 Send(new ChromeViewMsg_RetrieveWebappInformation(routing_id(), url_)); |
| 45 } | 53 } |
| 46 | 54 |
| 47 void ShortcutBuilder::OnDidRetrieveWebappInformation( | 55 ShortcutHelper::~ShortcutHelper() { |
| 56 } |
| 57 |
| 58 void ShortcutHelper::OnDidRetrieveWebappInformation( |
| 48 bool success, | 59 bool success, |
| 49 bool is_mobile_webapp_capable, | 60 bool is_mobile_webapp_capable, |
| 50 bool is_apple_mobile_webapp_capable, | 61 bool is_apple_mobile_webapp_capable, |
| 51 const GURL& expected_url) { | 62 const GURL& expected_url) { |
| 52 Profile* profile = | 63 // This should silently fail. |
| 53 Profile::FromBrowserContext(web_contents()->GetBrowserContext()); | |
| 54 Observe(NULL); | |
| 55 | |
| 56 if (!success) { | 64 if (!success) { |
| 57 LOG(ERROR) << "Failed to parse webpage."; | 65 LOG(ERROR) << "Failed to parse webpage."; |
| 58 Destroy(); | |
| 59 return; | |
| 60 } else if (expected_url != url_) { | 66 } else if (expected_url != url_) { |
| 61 LOG(ERROR) << "Unexpected URL returned."; | 67 LOG(ERROR) << "Unexpected URL returned."; |
| 62 Destroy(); | |
| 63 return; | |
| 64 } | 68 } |
| 65 | 69 |
| 66 if (is_apple_mobile_webapp_capable && !is_mobile_webapp_capable) { | 70 if (is_apple_mobile_webapp_capable && !is_mobile_webapp_capable) { |
| 67 shortcut_type_ = APP_SHORTCUT_APPLE; | 71 shortcut_type_ = APP_SHORTCUT_APPLE; |
| 68 } else if (is_apple_mobile_webapp_capable || is_mobile_webapp_capable) { | 72 } else if (is_apple_mobile_webapp_capable || is_mobile_webapp_capable) { |
| 69 shortcut_type_ = APP_SHORTCUT; | 73 shortcut_type_ = APP_SHORTCUT; |
| 70 } else { | 74 } else { |
| 71 shortcut_type_ = BOOKMARK; | 75 shortcut_type_ = BOOKMARK; |
| 72 } | 76 } |
| 73 | 77 |
| 78 title_ = web_contents()->GetTitle(); |
| 79 |
| 80 JNIEnv* env = base::android::AttachCurrentThread(); |
| 81 ScopedJavaLocalRef<jobject> j_obj = java_ref_.get(env); |
| 82 ScopedJavaLocalRef<jstring> j_title = |
| 83 base::android::ConvertUTF16ToJavaString(env, title_); |
| 84 |
| 85 Java_ShortcutHelper_onInitialized(env, j_obj.obj(), j_title.obj()); |
| 86 } |
| 87 |
| 88 void ShortcutHelper::TearDown(JNIEnv*, jobject) { |
| 89 Destroy(); |
| 90 } |
| 91 |
| 92 void ShortcutHelper::Destroy() { |
| 93 delete this; |
| 94 } |
| 95 |
| 96 void ShortcutHelper::AddShortcut( |
| 97 JNIEnv* env, |
| 98 jobject obj, |
| 99 jstring jtitle, |
| 100 jint launcher_large_icon_size) { |
| 101 base::string16 title = base::android::ConvertJavaStringToUTF16(env, jtitle); |
| 102 if (!title.empty()) |
| 103 title_ = title; |
| 104 |
| 105 Profile* profile = |
| 106 Profile::FromBrowserContext(web_contents()->GetBrowserContext()); |
| 107 |
| 74 // Grab the best, largest icon we can find to represent this bookmark. | 108 // Grab the best, largest icon we can find to represent this bookmark. |
| 75 // TODO(dfalcantara): Try combining with the new BookmarksHandler once its | 109 // TODO(dfalcantara): Try combining with the new BookmarksHandler once its |
| 76 // rewrite is further along. | 110 // rewrite is further along. |
| 77 std::vector<int> icon_types; | 111 std::vector<int> icon_types; |
| 78 icon_types.push_back(favicon_base::FAVICON); | 112 icon_types.push_back(favicon_base::FAVICON); |
| 79 icon_types.push_back(favicon_base::TOUCH_PRECOMPOSED_ICON | | 113 icon_types.push_back(favicon_base::TOUCH_PRECOMPOSED_ICON | |
| 80 favicon_base::TOUCH_ICON); | 114 favicon_base::TOUCH_ICON); |
| 81 FaviconService* favicon_service = FaviconServiceFactory::GetForProfile( | 115 FaviconService* favicon_service = FaviconServiceFactory::GetForProfile( |
| 82 profile, Profile::EXPLICIT_ACCESS); | 116 profile, Profile::EXPLICIT_ACCESS); |
| 83 | 117 |
| 84 // Using favicon if its size is not smaller than platform required size, | 118 // Using favicon if its size is not smaller than platform required size, |
| 85 // otherwise using the largest icon among all avaliable icons. | 119 // otherwise using the largest icon among all avaliable icons. |
| 86 int threshold_to_get_any_largest_icon = launcher_large_icon_size_ - 1; | 120 int threshold_to_get_any_largest_icon = launcher_large_icon_size_ - 1; |
| 87 favicon_service->GetLargestRawFaviconForPageURL(url_, icon_types, | 121 favicon_service->GetLargestRawFaviconForPageURL(url_, icon_types, |
| 88 threshold_to_get_any_largest_icon, | 122 threshold_to_get_any_largest_icon, |
| 89 base::Bind(&ShortcutBuilder::FinishAddingShortcut, | 123 base::Bind(&ShortcutHelper::FinishAddingShortcut, |
| 90 base::Unretained(this)), | 124 base::Unretained(this)), |
| 91 &cancelable_task_tracker_); | 125 &cancelable_task_tracker_); |
| 92 } | 126 } |
| 93 | 127 |
| 94 void ShortcutBuilder::FinishAddingShortcut( | 128 void ShortcutHelper::FinishAddingShortcut( |
| 95 const favicon_base::FaviconRawBitmapResult& bitmap_result) { | 129 const favicon_base::FaviconRawBitmapResult& bitmap_result) { |
| 130 icon_ = bitmap_result; |
| 131 |
| 132 // Stop observing so we don't get destroyed while doing the last steps. |
| 133 Observe(NULL); |
| 134 |
| 96 base::WorkerPool::PostTask( | 135 base::WorkerPool::PostTask( |
| 97 FROM_HERE, | 136 FROM_HERE, |
| 98 base::Bind(&ShortcutHelper::AddShortcutInBackground, | 137 base::Bind(&ShortcutHelper::AddShortcutInBackground, |
| 99 url_, | 138 url_, |
| 100 title_, | 139 title_, |
| 101 shortcut_type_, | 140 shortcut_type_, |
| 102 bitmap_result), | 141 icon_), |
| 103 true); | 142 true); |
| 143 |
| 104 Destroy(); | 144 Destroy(); |
| 105 } | 145 } |
| 106 | 146 |
| 107 bool ShortcutBuilder::OnMessageReceived(const IPC::Message& message) { | 147 bool ShortcutHelper::OnMessageReceived(const IPC::Message& message) { |
| 108 bool handled = true; | 148 bool handled = true; |
| 109 IPC_BEGIN_MESSAGE_MAP(ShortcutBuilder, message) | 149 |
| 150 IPC_BEGIN_MESSAGE_MAP(ShortcutHelper, message) |
| 110 IPC_MESSAGE_HANDLER(ChromeViewHostMsg_DidRetrieveWebappInformation, | 151 IPC_MESSAGE_HANDLER(ChromeViewHostMsg_DidRetrieveWebappInformation, |
| 111 OnDidRetrieveWebappInformation) | 152 OnDidRetrieveWebappInformation) |
| 112 IPC_MESSAGE_UNHANDLED(handled = false) | 153 IPC_MESSAGE_UNHANDLED(handled = false) |
| 113 IPC_END_MESSAGE_MAP() | 154 IPC_END_MESSAGE_MAP() |
| 155 |
| 114 return handled; | 156 return handled; |
| 115 } | 157 } |
| 116 | 158 |
| 117 void ShortcutBuilder::WebContentsDestroyed() { | 159 void ShortcutHelper::WebContentsDestroyed() { |
| 118 Destroy(); | 160 Destroy(); |
| 119 } | 161 } |
| 120 | 162 |
| 121 void ShortcutBuilder::Destroy() { | |
| 122 if (cancelable_task_tracker_.HasTrackedTasks()) { | |
| 123 cancelable_task_tracker_.TryCancelAll(); | |
| 124 } | |
| 125 delete this; | |
| 126 } | |
| 127 | |
| 128 void ShortcutHelper::AddShortcut(content::WebContents* web_contents, | |
| 129 const base::string16& title, | |
| 130 int launcher_large_icon_size) { | |
| 131 // The ShortcutBuilder deletes itself when it's done. | |
| 132 new ShortcutBuilder(web_contents, title, launcher_large_icon_size); | |
| 133 } | |
| 134 | |
| 135 bool ShortcutHelper::RegisterShortcutHelper(JNIEnv* env) { | 163 bool ShortcutHelper::RegisterShortcutHelper(JNIEnv* env) { |
| 136 return RegisterNativesImpl(env); | 164 return RegisterNativesImpl(env); |
| 137 } | 165 } |
| 138 | 166 |
| 139 void ShortcutHelper::AddShortcutInBackground( | 167 void ShortcutHelper::AddShortcutInBackground( |
| 140 const GURL& url, | 168 const GURL& url, |
| 141 const base::string16& title, | 169 const base::string16& title, |
| 142 ShortcutBuilder::ShortcutType shortcut_type, | 170 ShortcutType shortcut_type, |
| 143 const favicon_base::FaviconRawBitmapResult& bitmap_result) { | 171 const favicon_base::FaviconRawBitmapResult& bitmap_result) { |
| 144 DCHECK(base::WorkerPool::RunsTasksOnCurrentThread()); | 172 DCHECK(base::WorkerPool::RunsTasksOnCurrentThread()); |
| 145 | 173 |
| 146 // Grab the average color from the bitmap. | 174 // Grab the average color from the bitmap. |
| 147 SkColor color = SK_ColorWHITE; | 175 SkColor color = SK_ColorWHITE; |
| 148 SkBitmap favicon_bitmap; | 176 SkBitmap favicon_bitmap; |
| 149 if (bitmap_result.is_valid()) { | 177 if (bitmap_result.is_valid()) { |
| 150 if (gfx::PNGCodec::Decode(bitmap_result.bitmap_data->front(), | 178 if (gfx::PNGCodec::Decode(bitmap_result.bitmap_data->front(), |
| 151 bitmap_result.bitmap_data->size(), | 179 bitmap_result.bitmap_data->size(), |
| 152 &favicon_bitmap)) | 180 &favicon_bitmap)) |
| (...skipping 15 matching lines...) Expand all Loading... |
| 168 java_bitmap = gfx::ConvertToJavaBitmap(&favicon_bitmap); | 196 java_bitmap = gfx::ConvertToJavaBitmap(&favicon_bitmap); |
| 169 | 197 |
| 170 Java_ShortcutHelper_addShortcut(env, | 198 Java_ShortcutHelper_addShortcut(env, |
| 171 base::android::GetApplicationContext(), | 199 base::android::GetApplicationContext(), |
| 172 java_url.obj(), | 200 java_url.obj(), |
| 173 java_title.obj(), | 201 java_title.obj(), |
| 174 java_bitmap.obj(), | 202 java_bitmap.obj(), |
| 175 r_value, | 203 r_value, |
| 176 g_value, | 204 g_value, |
| 177 b_value, | 205 b_value, |
| 178 shortcut_type != ShortcutBuilder::BOOKMARK); | 206 shortcut_type != BOOKMARK); |
| 179 | 207 |
| 180 // Record what type of shortcut was added by the user. | 208 // Record what type of shortcut was added by the user. |
| 181 switch (shortcut_type) { | 209 switch (shortcut_type) { |
| 182 case ShortcutBuilder::APP_SHORTCUT: | 210 case APP_SHORTCUT: |
| 183 content::RecordAction( | 211 content::RecordAction( |
| 184 base::UserMetricsAction("webapps.AddShortcut.AppShortcut")); | 212 base::UserMetricsAction("webapps.AddShortcut.AppShortcut")); |
| 185 break; | 213 break; |
| 186 case ShortcutBuilder::APP_SHORTCUT_APPLE: | 214 case APP_SHORTCUT_APPLE: |
| 187 content::RecordAction( | 215 content::RecordAction( |
| 188 base::UserMetricsAction("webapps.AddShortcut.AppShortcutApple")); | 216 base::UserMetricsAction("webapps.AddShortcut.AppShortcutApple")); |
| 189 break; | 217 break; |
| 190 case ShortcutBuilder::BOOKMARK: | 218 case BOOKMARK: |
| 191 content::RecordAction( | 219 content::RecordAction( |
| 192 base::UserMetricsAction("webapps.AddShortcut.Bookmark")); | 220 base::UserMetricsAction("webapps.AddShortcut.Bookmark")); |
| 193 break; | 221 break; |
| 194 default: | 222 default: |
| 195 NOTREACHED(); | 223 NOTREACHED(); |
| 196 } | 224 } |
| 197 } | 225 } |
| 198 | |
| 199 // Adds a shortcut to the current URL to the Android home screen, firing | |
| 200 // background tasks to pull all the data required. | |
| 201 // Note that we don't actually care about the tab here -- we just want | |
| 202 // its otherwise inaccessible WebContents. | |
| 203 static void AddShortcut(JNIEnv* env, | |
| 204 jclass clazz, | |
| 205 jlong tab_android_ptr, | |
| 206 jstring title, | |
| 207 jint launcher_large_icon_size) { | |
| 208 TabAndroid* tab = reinterpret_cast<TabAndroid*>(tab_android_ptr); | |
| 209 ShortcutHelper::AddShortcut( | |
| 210 tab->web_contents(), | |
| 211 base::android::ConvertJavaStringToUTF16(env, title), | |
| 212 launcher_large_icon_size); | |
| 213 } | |
| OLD | NEW |