Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(634)

Side by Side Diff: chrome/browser/manifest/manifest_icon_selector.cc

Issue 1285063003: manifest: rework icon selector to include small icon cut-off (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Fix test failure Created 5 years, 4 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 // Copyright 2015 The Chromium Authors. All rights reserved. 1 // Copyright 2015 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/manifest/manifest_icon_selector.h" 5 #include "chrome/browser/manifest/manifest_icon_selector.h"
6 6
7 #include <algorithm>
8 #include <cmath>
7 #include <limits> 9 #include <limits>
8 10
9 #include "base/strings/utf_string_conversions.h" 11 #include "base/strings/utf_string_conversions.h"
10 #include "components/mime_util/mime_util.h" 12 #include "components/mime_util/mime_util.h"
11 #include "content/public/browser/web_contents.h" 13 #include "content/public/browser/web_contents.h"
12 #include "ui/gfx/screen.h" 14 #include "ui/gfx/screen.h"
13 15
14 using content::Manifest; 16 using content::Manifest;
15 17
16 ManifestIconSelector::ManifestIconSelector(float preferred_icon_size_in_pixels) 18 ManifestIconSelector::ManifestIconSelector(float preferred_icon_size_in_pixels,
17 : preferred_icon_size_in_pixels_(preferred_icon_size_in_pixels) { 19 float minimum_icon_size_in_pixels)
20 : preferred_icon_size_in_pixels_(preferred_icon_size_in_pixels),
21 minimum_icon_size_in_pixels_(minimum_icon_size_in_pixels) {
18 } 22 }
19 23
20 bool ManifestIconSelector::IconSizesContainsPreferredSize( 24 bool ManifestIconSelector::IconSizesContainsPreferredSize(
21 const std::vector<gfx::Size>& sizes) { 25 const std::vector<gfx::Size>& sizes) {
22 for (size_t i = 0; i < sizes.size(); ++i) { 26 for (size_t i = 0; i < sizes.size(); ++i) {
23 if (sizes[i].height() != sizes[i].width()) 27 if (sizes[i].height() != sizes[i].width())
24 continue; 28 continue;
25 if (sizes[i].width() == preferred_icon_size_in_pixels_) 29 if (sizes[i].width() == preferred_icon_size_in_pixels_)
26 return true; 30 return true;
27 } 31 }
28 32
29 return false; 33 return false;
30 } 34 }
31 35
32 GURL ManifestIconSelector::FindBestMatchingIconForDensity( 36 bool ManifestIconSelector::IconSizesContainsBiggerThanMinimum(
37 const std::vector<gfx::Size>& sizes) {
38 for (size_t i = 0; i < sizes.size(); ++i) {
39 if (sizes[i].height() != sizes[i].width())
40 continue;
41 if (sizes[i].width() >= minimum_icon_size_in_pixels_)
42 return true;
43 }
44 return false;
45 }
46
47 int ManifestIconSelector::FindBestMatchingIconForDensity(
33 const std::vector<content::Manifest::Icon>& icons, 48 const std::vector<content::Manifest::Icon>& icons,
34 float density) { 49 float density) {
35 GURL url; 50 int best_index = -1;
36 int best_delta = std::numeric_limits<int>::min(); 51 int best_delta = std::numeric_limits<int>::min();
37 52
38 for (size_t i = 0; i < icons.size(); ++i) { 53 for (size_t i = 0; i < icons.size(); ++i) {
39 if (icons[i].density != density) 54 if (icons[i].density != density)
40 continue; 55 continue;
41 56
42 const std::vector<gfx::Size>& sizes = icons[i].sizes; 57 const std::vector<gfx::Size>& sizes = icons[i].sizes;
43 for (size_t j = 0; j < sizes.size(); ++j) { 58 for (size_t j = 0; j < sizes.size(); ++j) {
44 if (sizes[j].height() != sizes[j].width()) 59 if (sizes[j].height() != sizes[j].width())
45 continue; 60 continue;
46 int delta = sizes[j].width() - preferred_icon_size_in_pixels_; 61 int delta = sizes[j].width() - preferred_icon_size_in_pixels_;
47 if (delta == 0) 62 if (delta == 0)
48 return icons[i].src; 63 return i;
49 if (best_delta > 0 && delta < 0) 64 if (best_delta > 0 && delta < 0)
50 continue; 65 continue;
51 if ((best_delta > 0 && delta < best_delta) || 66 if ((best_delta > 0 && delta < best_delta) ||
52 (best_delta < 0 && delta > best_delta)) { 67 (best_delta < 0 && delta > best_delta)) {
53 url = icons[i].src; 68 best_index = i;
54 best_delta = delta; 69 best_delta = delta;
55 } 70 }
56 } 71 }
57 } 72 }
58 73
59 return url; 74 return best_index;
60 } 75 }
61 76
62 GURL ManifestIconSelector::FindBestMatchingIcon( 77 int ManifestIconSelector::FindBestMatchingIcon(
63 const std::vector<content::Manifest::Icon>& unfiltered_icons, 78 const std::vector<content::Manifest::Icon>& icons,
64 float density) { 79 float density) {
65 GURL url; 80 int best_index = -1;
66 std::vector<Manifest::Icon> icons = FilterIconsByType(unfiltered_icons);
67 81
68 // The first pass is to find the ideal icon. That icon is of the right size 82 // The first pass is to find the ideal icon. That icon is of the right size
69 // with the default density or the device's density. 83 // with the default density or the device's density.
70 for (size_t i = 0; i < icons.size(); ++i) { 84 for (size_t i = 0; i < icons.size(); ++i) {
71 if (icons[i].density == density && 85 if (icons[i].density == density &&
72 IconSizesContainsPreferredSize(icons[i].sizes)) { 86 IconSizesContainsPreferredSize(icons[i].sizes)) {
73 return icons[i].src; 87 return i;
74 } 88 }
75 89
76 // If there is an icon with the right size but not the right density, keep 90 // If there is an icon with the right size but not the right density, keep
77 // it on the side and only use it if nothing better is found. 91 // it on the side and only use it if nothing better is found.
78 if (icons[i].density == Manifest::Icon::kDefaultDensity && 92 if (icons[i].density == Manifest::Icon::kDefaultDensity &&
79 IconSizesContainsPreferredSize(icons[i].sizes)) { 93 IconSizesContainsPreferredSize(icons[i].sizes)) {
80 url = icons[i].src; 94 best_index = i;
81 } 95 }
82 } 96 }
83 97
98 // Do an early return here if we have a suitable icon.
mlamouri (slow - plz ping) 2015/08/20 22:18:18 nit: remove comment.
Lalit Maganti 2015/08/21 10:46:37 Done.
99 if (best_index != -1)
100 return best_index;
101
84 // The second pass is to find an icon with 'any'. The current device scale 102 // The second pass is to find an icon with 'any'. The current device scale
85 // factor is preferred. Otherwise, the default scale factor is used. 103 // factor is preferred. Otherwise, the default scale factor is used.
86 for (size_t i = 0; i < icons.size(); ++i) { 104 for (size_t i = 0; i < icons.size(); ++i) {
87 if (icons[i].density == density && 105 if (icons[i].density == density &&
88 IconSizesContainsAny(icons[i].sizes)) { 106 IconSizesContainsAny(icons[i].sizes)) {
89 return icons[i].src; 107 return i;
90 } 108 }
91 109
92 // If there is an icon with 'any' but not the right density, keep it on the 110 // If there is an icon with 'any' but not the right density, keep it on the
93 // side and only use it if nothing better is found. 111 // side and only use it if nothing better is found.
94 if (icons[i].density == Manifest::Icon::kDefaultDensity && 112 if (icons[i].density == Manifest::Icon::kDefaultDensity &&
95 IconSizesContainsAny(icons[i].sizes)) { 113 IconSizesContainsAny(icons[i].sizes)) {
96 url = icons[i].src; 114 best_index = i;
97 } 115 }
98 } 116 }
99 117
118 // Do an early return here if we have a suitable icon.
mlamouri (slow - plz ping) 2015/08/20 22:18:18 nit: remove comment.
Lalit Maganti 2015/08/21 10:46:37 Done.
119 if (best_index != -1)
120 return best_index;
121
100 // The last pass will try to find the best suitable icon for the device's 122 // The last pass will try to find the best suitable icon for the device's
101 // scale factor. If none, another pass will be run using kDefaultDensity. 123 // scale factor. If none, another pass will be run using kDefaultDensity.
102 if (!url.is_valid()) 124 best_index = FindBestMatchingIconForDensity(icons, density);
103 url = FindBestMatchingIconForDensity(icons, density); 125 if (best_index != -1 &&
104 if (!url.is_valid()) 126 IconSizesContainsBiggerThanMinimum(icons[best_index].sizes))
105 url = FindBestMatchingIconForDensity(icons, 127 return best_index;
106 Manifest::Icon::kDefaultDensity);
107 128
108 return url; 129 best_index = FindBestMatchingIconForDensity(icons,
130 Manifest::Icon::kDefaultDensity);
131 if (best_index != -1 &&
132 IconSizesContainsBiggerThanMinimum(icons[best_index].sizes))
133 return best_index;
134
135 return -1;
109 } 136 }
110 137
111 138
112 // static 139 // static
113 bool ManifestIconSelector::IconSizesContainsAny( 140 bool ManifestIconSelector::IconSizesContainsAny(
114 const std::vector<gfx::Size>& sizes) { 141 const std::vector<gfx::Size>& sizes) {
115 for (size_t i = 0; i < sizes.size(); ++i) { 142 for (size_t i = 0; i < sizes.size(); ++i) {
116 if (sizes[i].IsEmpty()) 143 if (sizes[i].IsEmpty())
117 return true; 144 return true;
118 } 145 }
119
120 return false; 146 return false;
121 } 147 }
122 148
123 // static 149 // static
124 std::vector<Manifest::Icon> ManifestIconSelector::FilterIconsByType( 150 std::vector<Manifest::Icon> ManifestIconSelector::FilterIconsByType(
125 const std::vector<content::Manifest::Icon>& icons) { 151 const std::vector<content::Manifest::Icon>& icons) {
126 std::vector<Manifest::Icon> result; 152 std::vector<Manifest::Icon> result;
127 153
128 for (size_t i = 0; i < icons.size(); ++i) { 154 for (size_t i = 0; i < icons.size(); ++i) {
129 if (icons[i].type.is_null() || 155 if (icons[i].type.is_null() ||
130 mime_util::IsSupportedImageMimeType( 156 mime_util::IsSupportedImageMimeType(
131 base::UTF16ToUTF8(icons[i].type.string()))) { 157 base::UTF16ToUTF8(icons[i].type.string()))) {
132 result.push_back(icons[i]); 158 result.push_back(icons[i]);
133 } 159 }
134 } 160 }
135 161
136 return result; 162 return result;
137 } 163 }
138 164
139 // static 165 // static
140 GURL ManifestIconSelector::FindBestMatchingIcon( 166 GURL ManifestIconSelector::FindBestMatchingIcon(
141 const std::vector<Manifest::Icon>& unfiltered_icons, 167 const std::vector<Manifest::Icon>& unfiltered_icons,
142 const float preferred_icon_size_in_dp, 168 const float preferred_icon_size_in_dp,
143 const gfx::Screen* screen) { 169 const gfx::Screen* screen) {
144 const float device_scale_factor = 170 const float device_scale_factor =
145 screen->GetPrimaryDisplay().device_scale_factor(); 171 screen->GetPrimaryDisplay().device_scale_factor();
146 const float preferred_icon_size_in_pixels = 172 const float preferred_icon_size_in_pixels =
147 preferred_icon_size_in_dp * device_scale_factor; 173 preferred_icon_size_in_dp * device_scale_factor;
148 174
149 ManifestIconSelector selector(preferred_icon_size_in_pixels); 175 const int minimum_scale_factor = std::max(
150 return selector.FindBestMatchingIcon(unfiltered_icons, device_scale_factor); 176 static_cast<int>(floor(device_scale_factor - 1)), 1);
177 const float minimum_icon_size_in_pixels =
178 preferred_icon_size_in_dp * minimum_scale_factor;
179
180 std::vector<Manifest::Icon> icons =
181 ManifestIconSelector::FilterIconsByType(unfiltered_icons);
182
183 ManifestIconSelector selector(preferred_icon_size_in_pixels,
184 minimum_icon_size_in_pixels);
185 int index = selector.FindBestMatchingIcon(icons, device_scale_factor);
186 if (index == -1)
187 return GURL();
188 return unfiltered_icons[index].src;
mlamouri (slow - plz ping) 2015/08/20 22:18:18 s/unfiltered_icons/icons/
Lalit Maganti 2015/08/21 10:46:37 Done.
151 } 189 }
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698