 Chromium Code Reviews
 Chromium Code Reviews Issue 64853004:
  Use high resolution icons where possible for streamlined hosted app icons.  (Closed) 
  Base URL: https://chromium.googlesource.com/chromium/src.git@browser_experiment_create_app_from_page
    
  
    Issue 64853004:
  Use high resolution icons where possible for streamlined hosted app icons.  (Closed) 
  Base URL: https://chromium.googlesource.com/chromium/src.git@browser_experiment_create_app_from_page| Index: chrome/browser/ui/web_applications/hosted_app_tab_helper_unittest.cc | 
| diff --git a/chrome/browser/ui/web_applications/hosted_app_tab_helper_unittest.cc b/chrome/browser/ui/web_applications/hosted_app_tab_helper_unittest.cc | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..02dd0bfc79068c2d72e9891d9df93e3a5e2af893 | 
| --- /dev/null | 
| +++ b/chrome/browser/ui/web_applications/hosted_app_tab_helper_unittest.cc | 
| @@ -0,0 +1,221 @@ | 
| +// 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/web_applications/hosted_app_tab_helper.h" | 
| + | 
| +#include "base/command_line.h" | 
| +#include "base/files/scoped_temp_dir.h" | 
| +#include "base/strings/utf_string_conversions.h" | 
| +#include "chrome/browser/chrome_notification_types.h" | 
| +#include "chrome/browser/extensions/crx_installer.h" | 
| +#include "chrome/browser/extensions/extension_system.h" | 
| +#include "chrome/browser/extensions/test_extension_service.h" | 
| +#include "chrome/browser/extensions/test_extension_system.h" | 
| +#include "chrome/common/extensions/extension.h" | 
| +#include "chrome/common/extensions/manifest_handlers/icons_handler.h" | 
| +#include "chrome/common/web_application_info.h" | 
| +#include "chrome/test/base/chrome_render_view_host_test_harness.h" | 
| +#include "chrome/test/base/testing_profile.h" | 
| +#include "content/public/browser/notification_details.h" | 
| +#include "content/public/browser/notification_observer.h" | 
| +#include "content/public/browser/notification_registrar.h" | 
| +#include "content/public/browser/notification_service.h" | 
| +#include "content/public/browser/notification_source.h" | 
| +#include "content/public/common/favicon_url.h" | 
| +#include "content/public/test/test_renderer_host.h" | 
| +#include "content/public/test/test_utils.h" | 
| +#include "testing/gtest/include/gtest/gtest.h" | 
| +#include "third_party/skia/include/core/SkBitmap.h" | 
| + | 
| +using content::RenderViewHostTester; | 
| + | 
| +namespace { | 
| + | 
| +// Creates valid SkBitmaps of the dimensions found in |sizes| and pushes them | 
| +// into |bitmaps|. | 
| +std::vector<SkBitmap> CreateTestBitmaps(const std::vector<gfx::Size>& sizes) { | 
| + std::vector<SkBitmap> bitmaps(sizes.size()); | 
| + for (size_t i = 0; i < sizes.size(); ++i) { | 
| + SkBitmap& bitmap = bitmaps[i]; | 
| + bitmap.setConfig(SkBitmap::kARGB_8888_Config, | 
| + sizes[i].width(), | 
| + sizes[i].height()); | 
| + bitmap.allocPixels(); | 
| + bitmap.eraseColor(SK_ColorRED); | 
| + } | 
| + return bitmaps; | 
| +} | 
| + | 
| +class TestHostedAppTabHelper : public HostedAppTabHelper { | 
| + public: | 
| + explicit TestHostedAppTabHelper(content::WebContents* web_contents) | 
| + : HostedAppTabHelper(web_contents), | 
| + id_counter_(0) { | 
| + } | 
| + virtual ~TestHostedAppTabHelper() {} | 
| + | 
| + virtual int DownloadImage(const GURL& url) OVERRIDE { | 
| + return id_counter_++; | 
| + } | 
| + | 
| + private: | 
| + int id_counter_; | 
| + DISALLOW_COPY_AND_ASSIGN(TestHostedAppTabHelper); | 
| +}; | 
| + | 
| +class HostedAppTabHelperTest : public ChromeRenderViewHostTestHarness, | 
| + public content::NotificationObserver { | 
| + protected: | 
| + HostedAppTabHelperTest() : extension_(NULL) { | 
| + } | 
| + | 
| + virtual ~HostedAppTabHelperTest() { | 
| + } | 
| + | 
| + virtual void SetUp() OVERRIDE { | 
| + ChromeRenderViewHostTestHarness::SetUp(); | 
| + | 
| + web_app_info_.app_url = GURL("http://www.google.com"); | 
| + web_app_info_.title = ASCIIToUTF16("Hosted App"); | 
| + extension_ = NULL; | 
| + | 
| + // A real extension service is needed for the CrxInstaller. | 
| + ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); | 
| + static_cast<extensions::TestExtensionSystem*>( | 
| + extensions::ExtensionSystem::Get(profile()))-> | 
| + CreateExtensionService( | 
| + CommandLine::ForCurrentProcess(), | 
| + temp_dir_.path(), | 
| + false); | 
| + | 
| + HostedAppTabHelper::CreateForWebContents(web_contents()); | 
| 
benwells
2013/11/25 00:40:16
Why is this created here, when there are Test ones
 
calamity
2013/11/29 05:45:36
Removed.
 | 
| + | 
| + // Listen for notifications which indicate the hosted app installation has | 
| + // finished. | 
| + registrar_.Add(this, | 
| + chrome::NOTIFICATION_CRX_INSTALLER_DONE, | 
| + content::Source<extensions::CrxInstaller>(NULL)); | 
| + | 
| + registrar_.Add(this, | 
| + chrome::NOTIFICATION_EXTENSION_INSTALL_ERROR, | 
| + content::NotificationService::AllSources()); | 
| + } | 
| + | 
| + // Overriden from content::NotificationObserver: | 
| + void Observe(int type, | 
| + const content::NotificationSource& source, | 
| + const content::NotificationDetails& details) OVERRIDE { | 
| + if (type == chrome::NOTIFICATION_CRX_INSTALLER_DONE) | 
| + extension_ = content::Details<const extensions::Extension>(details).ptr(); | 
| + | 
| + base::MessageLoop::current()->Quit(); | 
| + } | 
| + | 
| + const extensions::Extension* extension() { | 
| + return extension_; | 
| + } | 
| + | 
| + WebApplicationInfo* web_app_info() { | 
| + return &web_app_info_; | 
| + } | 
| + | 
| + private: | 
| + const extensions::Extension* extension_; | 
| + WebApplicationInfo web_app_info_; | 
| + content::NotificationRegistrar registrar_; | 
| + base::ScopedTempDir temp_dir_; | 
| + | 
| + DISALLOW_COPY_AND_ASSIGN(HostedAppTabHelperTest); | 
| +}; | 
| + | 
| +} // namespace | 
| + | 
| +TEST_F(HostedAppTabHelperTest, CreateHostedApp) { | 
| + TestHostedAppTabHelper helper(web_contents()); | 
| + | 
| + std::vector<content::FaviconURL> favicon_urls; | 
| + favicon_urls.push_back( | 
| + content::FaviconURL(GURL("http://www.google.com/favicon.ico"), | 
| + content::FaviconURL::FAVICON)); | 
| + | 
| + // Download should not be initiated because |fetch_icons_immediately| is not | 
| 
benwells
2013/11/25 00:40:16
This comment is out of date.
 
calamity
2013/11/29 05:45:36
Done.
 | 
| + // set. | 
| + helper.DidUpdateFaviconURL(0, favicon_urls); | 
| + EXPECT_EQ(0, helper.pending_requests()); | 
| + | 
| + // Create hosted app which will initiate download. | 
| + helper.CreateHostedApp(*web_app_info()); | 
| + EXPECT_EQ(1, helper.pending_requests()); | 
| + | 
| + std::vector<gfx::Size> sizes(1, gfx::Size(32, 32)); | 
| + helper.DidDownloadFavicon(0, 200, favicon_urls[0].icon_url, | 
| + CreateTestBitmaps(sizes), sizes); | 
| + | 
| + // Create and install the hosted app. | 
| + content::RunMessageLoop(); | 
| + | 
| + EXPECT_TRUE(extension()); | 
| + EXPECT_TRUE(extension()->from_bookmark()); | 
| + | 
| + const ExtensionIconSet& icons = extensions::IconsInfo::GetIcons(extension()); | 
| + EXPECT_EQ(1, icons.map().size()); | 
| + EXPECT_FALSE(icons.Get(32, ExtensionIconSet::MATCH_EXACTLY).empty()); | 
| +} | 
| + | 
| +TEST_F(HostedAppTabHelperTest, CreateHostedAppBeforeFaviconURLUpdate) { | 
| + TestHostedAppTabHelper helper(web_contents()); | 
| + const GURL empty_favicon = GURL("http://www.google.com/empty_favicon.ico"); | 
| + | 
| + // This should get downloaded. | 
| + web_app_info()->icons.push_back(WebApplicationInfo::IconInfo()); | 
| + web_app_info()->icons[0].url = GURL("http://www.google.com/favicon2.ico"); | 
| + | 
| + // This is duplicated in the favicon urls and should only be downloaded once. | 
| + web_app_info()->icons.push_back(WebApplicationInfo::IconInfo()); | 
| + web_app_info()->icons[1].url = empty_favicon; | 
| + | 
| + // Initiate creation of hosted app before favicon URLs are loaded. | 
| + helper.CreateHostedApp(*web_app_info()); | 
| + | 
| + std::vector<content::FaviconURL> favicon_urls; | 
| + favicon_urls.push_back( | 
| + content::FaviconURL(GURL("http://www.google.com/favicon.ico"), | 
| + content::FaviconURL::FAVICON)); | 
| + // This is duplicated in the favicon urls and should only be downloaded once. | 
| + favicon_urls.push_back( | 
| + content::FaviconURL(empty_favicon, | 
| + content::FaviconURL::FAVICON)); | 
| + // Invalid icons shouldn't get put into the download queue. | 
| + favicon_urls.push_back( | 
| + content::FaviconURL(GURL("http://www.google.com/invalid.ico"), | 
| + content::FaviconURL::INVALID_ICON)); | 
| + helper.DidUpdateFaviconURL(0, favicon_urls); | 
| 
benwells
2013/11/25 00:40:16
I think it would be clearer to check the number of
 
calamity
2013/11/29 05:45:36
Done.
 | 
| + | 
| + std::vector<gfx::Size> sizes_1(1, gfx::Size(16, 16)); | 
| + helper.DidDownloadFavicon(0, 200, favicon_urls[0].icon_url, | 
| + CreateTestBitmaps(sizes_1), sizes_1); | 
| + | 
| + std::vector<gfx::Size> sizes_2; | 
| + sizes_2.push_back(gfx::Size(32, 32)); | 
| + sizes_2.push_back(gfx::Size(64, 64)); | 
| + helper.DidDownloadFavicon(1, 200, web_app_info()->icons[0].url, | 
| + CreateTestBitmaps(sizes_2), sizes_2); | 
| + | 
| + // Only 1 download should have been initiated for |empty_favicon| even though | 
| + // the URL was in both the web app info and the favicon urls. | 
| + helper.DidDownloadFavicon(2, 200, empty_favicon, | 
| + std::vector<SkBitmap>(), std::vector<gfx::Size>()); | 
| + | 
| + // Create and install the hosted app. | 
| + content::RunMessageLoop(); | 
| + | 
| + EXPECT_TRUE(extension()); | 
| + EXPECT_TRUE(extension()->from_bookmark()); | 
| + | 
| + const ExtensionIconSet& icons = extensions::IconsInfo::GetIcons(extension()); | 
| + EXPECT_EQ(3, icons.map().size()); | 
| + EXPECT_FALSE(icons.Get(16, ExtensionIconSet::MATCH_EXACTLY).empty()); | 
| + EXPECT_FALSE(icons.Get(32, ExtensionIconSet::MATCH_EXACTLY).empty()); | 
| + EXPECT_FALSE(icons.Get(64, ExtensionIconSet::MATCH_EXACTLY).empty()); | 
| +} |