Chromium Code Reviews| OLD | NEW |
|---|---|
| 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::IconSizesBiggerThanMinimum( |
| 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 GURL ManifestIconSelector::FindBestMatchingIcon( |
| 63 const std::vector<content::Manifest::Icon>& unfiltered_icons, | 78 const std::vector<content::Manifest::Icon>& unfiltered_icons, |
| 64 float density) { | 79 float density) { |
| 65 GURL url; | 80 GURL url; |
| 66 std::vector<Manifest::Icon> icons = FilterIconsByType(unfiltered_icons); | 81 std::vector<Manifest::Icon> icons = FilterIconsByType(unfiltered_icons); |
| 67 | 82 |
| 68 // The first pass is to find the ideal icon. That icon is of the right size | 83 // 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. | 84 // with the default density or the device's density. |
| (...skipping 20 matching lines...) Expand all Loading... | |
| 90 } | 105 } |
| 91 | 106 |
| 92 // If there is an icon with 'any' but not the right density, keep it on the | 107 // 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. | 108 // side and only use it if nothing better is found. |
| 94 if (icons[i].density == Manifest::Icon::kDefaultDensity && | 109 if (icons[i].density == Manifest::Icon::kDefaultDensity && |
| 95 IconSizesContainsAny(icons[i].sizes)) { | 110 IconSizesContainsAny(icons[i].sizes)) { |
| 96 url = icons[i].src; | 111 url = icons[i].src; |
| 97 } | 112 } |
| 98 } | 113 } |
| 99 | 114 |
| 115 // Do an early return here if we have a suitable URL. | |
|
gone
2015/08/19 22:31:56
Is the first pass's fallback URL (where you don't
Lalit Maganti
2015/08/20 14:53:49
Very good point. Mounir: is there a reason for thi
mlamouri (slow - plz ping)
2015/08/20 22:18:18
It's on purpose. We don't want to return in the fi
| |
| 116 if (url.is_valid()) | |
| 117 return url; | |
| 118 | |
| 100 // The last pass will try to find the best suitable icon for the device's | 119 // 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. | 120 // scale factor. If none, another pass will be run using kDefaultDensity. |
| 102 if (!url.is_valid()) | 121 int best_index = FindBestMatchingIconForDensity(icons, density); |
|
mlamouri (slow - plz ping)
2015/08/19 15:02:53
For consistency, you could have a best_index only
Lalit Maganti
2015/08/20 14:53:49
Done.
| |
| 103 url = FindBestMatchingIconForDensity(icons, density); | 122 if (best_index == -1 || |
|
mlamouri (slow - plz ping)
2015/08/19 15:02:53
if (best_index != -1 && IconSizesBiggerThanMinimum
Lalit Maganti
2015/08/20 14:53:49
Done.
| |
| 104 if (!url.is_valid()) | 123 !IconSizesBiggerThanMinimum(icons[best_index].sizes)) { |
| 105 url = FindBestMatchingIconForDensity(icons, | 124 best_index = FindBestMatchingIconForDensity( |
| 106 Manifest::Icon::kDefaultDensity); | 125 icons, Manifest::Icon::kDefaultDensity); |
| 107 | 126 } |
| 108 return url; | 127 return best_index == -1 || |
| 128 !IconSizesBiggerThanMinimum(icons[best_index].sizes) | |
| 129 ? GURL() : icons[best_index].src; | |
| 109 } | 130 } |
| 110 | 131 |
| 111 | 132 |
| 112 // static | 133 // static |
| 113 bool ManifestIconSelector::IconSizesContainsAny( | 134 bool ManifestIconSelector::IconSizesContainsAny( |
| 114 const std::vector<gfx::Size>& sizes) { | 135 const std::vector<gfx::Size>& sizes) { |
| 115 for (size_t i = 0; i < sizes.size(); ++i) { | 136 for (size_t i = 0; i < sizes.size(); ++i) { |
| 116 if (sizes[i].IsEmpty()) | 137 if (sizes[i].IsEmpty()) |
| 117 return true; | 138 return true; |
| 118 } | 139 } |
| (...skipping 20 matching lines...) Expand all Loading... | |
| 139 // static | 160 // static |
| 140 GURL ManifestIconSelector::FindBestMatchingIcon( | 161 GURL ManifestIconSelector::FindBestMatchingIcon( |
| 141 const std::vector<Manifest::Icon>& unfiltered_icons, | 162 const std::vector<Manifest::Icon>& unfiltered_icons, |
| 142 const float preferred_icon_size_in_dp, | 163 const float preferred_icon_size_in_dp, |
| 143 const gfx::Screen* screen) { | 164 const gfx::Screen* screen) { |
| 144 const float device_scale_factor = | 165 const float device_scale_factor = |
| 145 screen->GetPrimaryDisplay().device_scale_factor(); | 166 screen->GetPrimaryDisplay().device_scale_factor(); |
| 146 const float preferred_icon_size_in_pixels = | 167 const float preferred_icon_size_in_pixels = |
| 147 preferred_icon_size_in_dp * device_scale_factor; | 168 preferred_icon_size_in_dp * device_scale_factor; |
| 148 | 169 |
| 149 ManifestIconSelector selector(preferred_icon_size_in_pixels); | 170 const int minimum_scale_factor = std::max( |
| 171 static_cast<int>(floor(device_scale_factor - 1)), 1); | |
| 172 const float minimum_icon_size_in_pixels = | |
| 173 preferred_icon_size_in_dp * minimum_scale_factor; | |
| 174 | |
| 175 ManifestIconSelector selector(preferred_icon_size_in_pixels, | |
| 176 minimum_icon_size_in_pixels); | |
| 150 return selector.FindBestMatchingIcon(unfiltered_icons, device_scale_factor); | 177 return selector.FindBestMatchingIcon(unfiltered_icons, device_scale_factor); |
|
mlamouri (slow - plz ping)
2015/08/19 15:02:53
int index = selector.FindBestMatchingIcon(unfilter
Lalit Maganti
2015/08/20 14:53:49
Done.
| |
| 151 } | 178 } |
| OLD | NEW |