| // 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/manifest/manifest_icon_selector.h" |
| |
| #include <string> |
| #include <vector> |
| |
| #include "base/macros.h" |
| #include "base/strings/utf_string_conversions.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "ui/display/screen.h" |
| #include "ui/display/test/test_screen.h" |
| |
| namespace { |
| const int DEFAULT_PREFERRED_ICON_SIZE = 48; |
| } |
| |
| class ManifestIconSelectorTest : public testing::Test { |
| protected: |
| ManifestIconSelectorTest() { |
| const display::Display test_display = test_screen_.GetPrimaryDisplay(); |
| display::Display display(test_display); |
| display.set_id(0x1337); |
| display.set_bounds(gfx::Rect(0, 0, 2560, 1440)); |
| test_screen_.display_list().RemoveDisplay(test_display.id()); |
| test_screen_.display_list().AddDisplay(display, |
| display::DisplayList::Type::PRIMARY); |
| display::Screen::SetScreenInstance(&test_screen_); |
| } |
| |
| ~ManifestIconSelectorTest() override { |
| display::Screen::SetScreenInstance(nullptr); |
| } |
| |
| void SetUp() override { |
| SetPreferredIconSizeInDp(DEFAULT_PREFERRED_ICON_SIZE); |
| } |
| |
| GURL FindBestMatchingIconWithMinimum( |
| const std::vector<content::Manifest::Icon>& icons, |
| int minimum_icon_size_in_dp) { |
| return ManifestIconSelector::FindBestMatchingIcon( |
| icons, GetPreferredIconSizeInDp(), minimum_icon_size_in_dp); |
| } |
| |
| GURL FindBestMatchingIcon(const std::vector<content::Manifest::Icon>& icons) { |
| return FindBestMatchingIconWithMinimum(icons, 0); |
| } |
| |
| void SetDisplayDeviceScaleFactor(float device_scale_factor) { |
| display::Display display(test_screen_.GetPrimaryDisplay()); |
| display.set_device_scale_factor(device_scale_factor); |
| test_screen_.display_list().UpdateDisplay(display); |
| } |
| |
| int GetPreferredIconSizeInDp() { |
| return preferred_icon_size_; |
| } |
| |
| void SetPreferredIconSizeInDp(int new_size) { |
| preferred_icon_size_ = new_size; |
| } |
| |
| static content::Manifest::Icon CreateIcon( |
| const std::string& url, |
| const std::string& type, |
| const std::vector<gfx::Size> sizes) { |
| content::Manifest::Icon icon; |
| icon.src = GURL(url); |
| icon.type = base::UTF8ToUTF16(type); |
| icon.sizes = sizes; |
| |
| return icon; |
| } |
| |
| private: |
| display::test::TestScreen test_screen_; |
| int preferred_icon_size_ = DEFAULT_PREFERRED_ICON_SIZE; |
| |
| DISALLOW_COPY_AND_ASSIGN(ManifestIconSelectorTest); |
| }; |
| |
| TEST_F(ManifestIconSelectorTest, NoIcons) { |
| // No icons should return the empty URL. |
| std::vector<content::Manifest::Icon> icons; |
| GURL url = FindBestMatchingIcon(icons); |
| EXPECT_TRUE(url.is_empty()); |
| } |
| |
| TEST_F(ManifestIconSelectorTest, NoSizes) { |
| // Icon with no sizes are ignored. |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back( |
| CreateIcon("http://foo.com/icon.png", "", std::vector<gfx::Size>())); |
| |
| GURL url = FindBestMatchingIcon(icons); |
| EXPECT_TRUE(url.is_empty()); |
| } |
| |
| TEST_F(ManifestIconSelectorTest, MIMETypeFiltering) { |
| // Icons with type specified to a MIME type that isn't a valid image MIME type |
| // are ignored. |
| std::vector<gfx::Size> sizes; |
| sizes.push_back(gfx::Size(1024, 1024)); |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back( |
| CreateIcon("http://foo.com/icon.png", "image/foo_bar", sizes)); |
| icons.push_back(CreateIcon("http://foo.com/icon.png", "image/", sizes)); |
| icons.push_back(CreateIcon("http://foo.com/icon.png", "image/", sizes)); |
| icons.push_back(CreateIcon("http://foo.com/icon.png", "video/mp4", sizes)); |
| |
| GURL url = FindBestMatchingIcon(icons); |
| EXPECT_TRUE(url.is_empty()); |
| |
| icons.clear(); |
| icons.push_back(CreateIcon("http://foo.com/icon.png", "image/png", sizes)); |
| url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon.png", url.spec()); |
| |
| icons.clear(); |
| icons.push_back(CreateIcon("http://foo.com/icon.png", "image/gif", sizes)); |
| url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon.png", url.spec()); |
| |
| icons.clear(); |
| icons.push_back(CreateIcon("http://foo.com/icon.png", "image/jpeg", sizes)); |
| url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon.png", url.spec()); |
| } |
| |
| TEST_F(ManifestIconSelectorTest, PreferredSizeIsUsedFirst) { |
| // Each icon is marked with sizes that match the preferred icon size. |
| std::vector<gfx::Size> sizes_1; |
| sizes_1.push_back(gfx::Size(GetPreferredIconSizeInDp(), |
| GetPreferredIconSizeInDp())); |
| |
| std::vector<gfx::Size> sizes_2; |
| sizes_2.push_back(gfx::Size(GetPreferredIconSizeInDp() * 2, |
| GetPreferredIconSizeInDp() * 2)); |
| |
| std::vector<gfx::Size> sizes_3; |
| sizes_3.push_back(gfx::Size(GetPreferredIconSizeInDp() * 3, |
| GetPreferredIconSizeInDp() * 3)); |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back(CreateIcon("http://foo.com/icon_x1.png", "", sizes_1)); |
| icons.push_back(CreateIcon("http://foo.com/icon_x2.png", "", sizes_2)); |
| icons.push_back(CreateIcon("http://foo.com/icon_x3.png", "", sizes_3)); |
| |
| SetDisplayDeviceScaleFactor(1.0f); |
| GURL url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon_x1.png", url.spec()); |
| |
| SetDisplayDeviceScaleFactor(2.0f); |
| url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon_x2.png", url.spec()); |
| |
| SetDisplayDeviceScaleFactor(3.0f); |
| url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon_x3.png", url.spec()); |
| } |
| |
| TEST_F(ManifestIconSelectorTest, FirstIconWithPreferredSizeIsUsedFirst) { |
| // This test has three icons. The first icon is going to be used because it |
| // has sizes which matches the preferred size for each device scale factor. |
| std::vector<gfx::Size> sizes_1; |
| sizes_1.push_back(gfx::Size(GetPreferredIconSizeInDp(), |
| GetPreferredIconSizeInDp())); |
| sizes_1.push_back(gfx::Size(GetPreferredIconSizeInDp() * 2, |
| GetPreferredIconSizeInDp() * 2)); |
| sizes_1.push_back(gfx::Size(GetPreferredIconSizeInDp() * 3, |
| GetPreferredIconSizeInDp() * 3)); |
| |
| std::vector<gfx::Size> sizes_2; |
| sizes_2.push_back(gfx::Size(1024, 1024)); |
| |
| std::vector<gfx::Size> sizes_3; |
| sizes_3.push_back(gfx::Size(1024, 1024)); |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back(CreateIcon("http://foo.com/icon_x1.png", "", sizes_1)); |
| icons.push_back(CreateIcon("http://foo.com/icon_x2.png", "", sizes_2)); |
| icons.push_back(CreateIcon("http://foo.com/icon_x3.png", "", sizes_3)); |
| |
| SetDisplayDeviceScaleFactor(1.0f); |
| GURL url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon_x1.png", url.spec()); |
| |
| SetDisplayDeviceScaleFactor(2.0f); |
| url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon_x1.png", url.spec()); |
| |
| SetDisplayDeviceScaleFactor(3.0f); |
| url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon_x1.png", url.spec()); |
| } |
| |
| TEST_F(ManifestIconSelectorTest, UseDeviceDensity) { |
| // If there is no perfect icon, the smallest larger icon will be chosen. |
| std::vector<gfx::Size> sizes_1; |
| sizes_1.push_back(gfx::Size(90, 90)); |
| |
| std::vector<gfx::Size> sizes_2; |
| sizes_2.push_back(gfx::Size(128, 128)); |
| |
| std::vector<gfx::Size> sizes_3; |
| sizes_3.push_back(gfx::Size(192, 192)); |
| |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back(CreateIcon("http://foo.com/icon_x1.png", "", sizes_1)); |
| icons.push_back(CreateIcon("http://foo.com/icon_x2.png", "", sizes_2)); |
| icons.push_back(CreateIcon("http://foo.com/icon_x3.png", "", sizes_3)); |
| |
| SetDisplayDeviceScaleFactor(1.0f); |
| GURL url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon_x1.png", url.spec()); |
| |
| SetDisplayDeviceScaleFactor(2.0f); |
| url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon_x2.png", url.spec()); |
| |
| SetDisplayDeviceScaleFactor(3.0f); |
| url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon_x3.png", url.spec()); |
| } |
| |
| TEST_F(ManifestIconSelectorTest, CheckDifferentDeviceScaleFactors) { |
| // When an icon of the correct size has not been found, we fall back to the |
| // closest non-matching sizes. Make sure that the minimum passed is enforced. |
| std::vector<gfx::Size> sizes_1_2; |
| std::vector<gfx::Size> sizes_3; |
| |
| sizes_1_2.push_back(gfx::Size(47, 47)); |
| sizes_3.push_back(gfx::Size(95, 95)); |
| |
| // Set to a value which will not affect the calculations. |
| SetPreferredIconSizeInDp(1024); |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back(CreateIcon("http://foo.com/icon_x1.png", "", sizes_1_2)); |
| icons.push_back(CreateIcon("http://foo.com/icon_x2.png", "", sizes_1_2)); |
| icons.push_back(CreateIcon("http://foo.com/icon_x3.png", "", sizes_3)); |
| |
| // Icon 3 should match. |
| SetDisplayDeviceScaleFactor(1.0f); |
| GURL url = FindBestMatchingIconWithMinimum(icons, 48); |
| EXPECT_EQ("http://foo.com/icon_x3.png", url.spec()); |
| |
| // Nothing matches here because the minimum is 48 and all icon sizes are now |
| // too small. |
| SetDisplayDeviceScaleFactor(2.0f); |
| url = FindBestMatchingIconWithMinimum(icons, 48); |
| EXPECT_TRUE(url.is_empty()); |
| |
| // Nothing matches here as the minimum is 96. |
| SetDisplayDeviceScaleFactor(3.0f); |
| url = FindBestMatchingIconWithMinimum(icons, 96); |
| EXPECT_TRUE(url.is_empty()); |
| } |
| |
| TEST_F(ManifestIconSelectorTest, IdealVeryCloseToMinimumMatches) { |
| std::vector<gfx::Size> sizes; |
| sizes.push_back(gfx::Size(2, 2)); |
| SetPreferredIconSizeInDp(2); |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back(CreateIcon("http://foo.com/icon_x1.png", "", sizes)); |
| |
| SetDisplayDeviceScaleFactor(1.0f); |
| GURL url = FindBestMatchingIconWithMinimum(icons, 1); |
| EXPECT_EQ("http://foo.com/icon_x1.png", url.spec()); |
| } |
| |
| TEST_F(ManifestIconSelectorTest, SizeVeryCloseToMinimumMatches) { |
| std::vector<gfx::Size> sizes; |
| sizes.push_back(gfx::Size(2, 2)); |
| |
| SetPreferredIconSizeInDp(200); |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back(CreateIcon("http://foo.com/icon_x1.png", "", sizes)); |
| |
| SetDisplayDeviceScaleFactor(1.0f); |
| GURL url = FindBestMatchingIconWithMinimum(icons, 1); |
| EXPECT_EQ("http://foo.com/icon_x1.png", url.spec()); |
| } |
| |
| TEST_F(ManifestIconSelectorTest, NotSquareIconsAreIgnored) { |
| std::vector<gfx::Size> sizes; |
| sizes.push_back(gfx::Size(1024, 1023)); |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back(CreateIcon("http://foo.com/icon.png", "", sizes)); |
| |
| GURL url = FindBestMatchingIcon(icons); |
| EXPECT_TRUE(url.is_empty()); |
| } |
| |
| TEST_F(ManifestIconSelectorTest, ClosestIconToPreferred) { |
| // Ensure ManifestIconSelector::FindBestMatchingIcon selects the closest icon |
| // to the preferred size when presented with a number of options. |
| int very_small = GetPreferredIconSizeInDp() / 4; |
| int small_size = GetPreferredIconSizeInDp() / 2; |
| int bit_small = GetPreferredIconSizeInDp() - 1; |
| int bit_big = GetPreferredIconSizeInDp() + 1; |
| int big = GetPreferredIconSizeInDp() * 2; |
| int very_big = GetPreferredIconSizeInDp() * 4; |
| |
| SetDisplayDeviceScaleFactor(1.0f); |
| // (very_small, bit_small) => bit_small |
| { |
| std::vector<gfx::Size> sizes_1; |
| sizes_1.push_back(gfx::Size(very_small, very_small)); |
| |
| std::vector<gfx::Size> sizes_2; |
| sizes_2.push_back(gfx::Size(bit_small, bit_small)); |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back(CreateIcon("http://foo.com/icon_no.png", "", sizes_1)); |
| icons.push_back(CreateIcon("http://foo.com/icon.png", "", sizes_2)); |
| |
| GURL url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon.png", url.spec()); |
| } |
| |
| // (very_small, bit_small, small_size) => bit_small |
| { |
| std::vector<gfx::Size> sizes_1; |
| sizes_1.push_back(gfx::Size(very_small, very_small)); |
| |
| std::vector<gfx::Size> sizes_2; |
| sizes_2.push_back(gfx::Size(bit_small, bit_small)); |
| |
| std::vector<gfx::Size> sizes_3; |
| sizes_3.push_back(gfx::Size(small_size, small_size)); |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back(CreateIcon("http://foo.com/icon_no_1.png", "", sizes_1)); |
| icons.push_back(CreateIcon("http://foo.com/icon.png", "", sizes_2)); |
| icons.push_back(CreateIcon("http://foo.com/icon_no_2.png", "", sizes_3)); |
| |
| GURL url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon.png", url.spec()); |
| } |
| |
| // (very_big, big) => big |
| { |
| std::vector<gfx::Size> sizes_1; |
| sizes_1.push_back(gfx::Size(very_big, very_big)); |
| |
| std::vector<gfx::Size> sizes_2; |
| sizes_2.push_back(gfx::Size(big, big)); |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back(CreateIcon("http://foo.com/icon_no.png", "", sizes_1)); |
| icons.push_back(CreateIcon("http://foo.com/icon.png", "", sizes_2)); |
| |
| GURL url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon.png", url.spec()); |
| } |
| |
| // (very_big, big, bit_big) => bit_big |
| { |
| std::vector<gfx::Size> sizes_1; |
| sizes_1.push_back(gfx::Size(very_big, very_big)); |
| |
| std::vector<gfx::Size> sizes_2; |
| sizes_2.push_back(gfx::Size(big, big)); |
| |
| std::vector<gfx::Size> sizes_3; |
| sizes_3.push_back(gfx::Size(bit_big, bit_big)); |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back(CreateIcon("http://foo.com/icon_no.png", "", sizes_1)); |
| icons.push_back(CreateIcon("http://foo.com/icon_no.png", "", sizes_2)); |
| icons.push_back(CreateIcon("http://foo.com/icon.png", "", sizes_3)); |
| |
| GURL url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon.png", url.spec()); |
| } |
| |
| // (bit_small, very_big) => very_big |
| { |
| std::vector<gfx::Size> sizes_1; |
| sizes_1.push_back(gfx::Size(bit_small, bit_small)); |
| |
| std::vector<gfx::Size> sizes_2; |
| sizes_2.push_back(gfx::Size(very_big, very_big)); |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back(CreateIcon("http://foo.com/icon_no.png", "", sizes_1)); |
| icons.push_back(CreateIcon("http://foo.com/icon.png", "", sizes_2)); |
| |
| GURL url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon.png", url.spec()); |
| } |
| |
| // (bit_small, bit_big) => bit_big |
| { |
| std::vector<gfx::Size> sizes_1; |
| sizes_1.push_back(gfx::Size(bit_small, bit_small)); |
| |
| std::vector<gfx::Size> sizes_2; |
| sizes_2.push_back(gfx::Size(bit_big, bit_big)); |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back(CreateIcon("http://foo.com/icon_no.png", "", sizes_1)); |
| icons.push_back(CreateIcon("http://foo.com/icon.png", "", sizes_2)); |
| |
| GURL url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon.png", url.spec()); |
| } |
| } |
| |
| TEST_F(ManifestIconSelectorTest, UseAnyIfNoPreferredSize) { |
| // 'any' (ie. gfx::Size(0,0)) should be used if there is no icon of a |
| // preferred size. An icon with the current device scale factor is preferred |
| // over one with the default density. |
| |
| // Icon with 'any' and icon with preferred size => preferred size is chosen. |
| { |
| std::vector<gfx::Size> sizes_1; |
| sizes_1.push_back(gfx::Size(GetPreferredIconSizeInDp(), |
| GetPreferredIconSizeInDp())); |
| std::vector<gfx::Size> sizes_2; |
| sizes_2.push_back(gfx::Size(0, 0)); |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back(CreateIcon("http://foo.com/icon.png", "", sizes_1)); |
| icons.push_back(CreateIcon("http://foo.com/icon_no.png", "", sizes_2)); |
| |
| GURL url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon.png", url.spec()); |
| } |
| |
| // Icon with 'any' and icon larger than preferred size => any is chosen. |
| { |
| std::vector<gfx::Size> sizes_1; |
| sizes_1.push_back(gfx::Size(GetPreferredIconSizeInDp() + 1, |
| GetPreferredIconSizeInDp() + 1)); |
| std::vector<gfx::Size> sizes_2; |
| sizes_2.push_back(gfx::Size(0, 0)); |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back(CreateIcon("http://foo.com/icon_no.png", "", sizes_1)); |
| icons.push_back(CreateIcon("http://foo.com/icon.png", "", sizes_2)); |
| |
| GURL url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon.png", url.spec()); |
| } |
| |
| // Multiple icons with 'any' => the last one is chosen. |
| { |
| std::vector<gfx::Size> sizes; |
| sizes.push_back(gfx::Size(0, 0)); |
| |
| std::vector<content::Manifest::Icon> icons; |
| icons.push_back(CreateIcon("http://foo.com/icon_no1.png", "", sizes)); |
| icons.push_back(CreateIcon("http://foo.com/icon_no2.png", "", sizes)); |
| icons.push_back(CreateIcon("http://foo.com/icon.png", "", sizes)); |
| |
| SetDisplayDeviceScaleFactor(3.0f); |
| GURL url = FindBestMatchingIcon(icons); |
| EXPECT_EQ("http://foo.com/icon.png", url.spec()); |
| } |
| } |