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 <stddef.h> | |
| 8 | |
| 9 #include <algorithm> | |
| 10 #include <cmath> | |
| 11 #include <limits> | 7 #include <limits> |
| 12 | 8 |
| 9 #include "base/stl_util.h" | |
| 13 #include "base/strings/utf_string_conversions.h" | 10 #include "base/strings/utf_string_conversions.h" |
| 14 #include "components/mime_util/mime_util.h" | 11 #include "components/mime_util/mime_util.h" |
| 15 #include "content/public/browser/web_contents.h" | |
| 16 | 12 |
| 17 using content::Manifest; | 13 // static |
| 18 | 14 GURL ManifestIconSelector::FindBestMatchingIcon( |
| 19 ManifestIconSelector::ManifestIconSelector(int ideal_icon_size_in_px, | 15 const std::vector<content::Manifest::Icon>& icons, |
| 20 int minimum_icon_size_in_px) | 16 int ideal_icon_size_in_px, |
| 21 : ideal_icon_size_in_px_(ideal_icon_size_in_px), | 17 int minimum_icon_size_in_px, |
| 22 minimum_icon_size_in_px_(minimum_icon_size_in_px) { | 18 content::Manifest::Icon::IconPurpose purpose) { |
| 23 } | 19 // Icon with exact matching size has priority over icon with size "any", which |
| 24 | 20 // has priority over icon with closest matching size. |
| 25 bool ManifestIconSelector::IconSizesContainsPreferredSize( | 21 int latest_size_any_index = -1; |
| 26 const std::vector<gfx::Size>& sizes) const { | 22 int closest_size_match_index = -1; |
| 27 for (size_t i = 0; i < sizes.size(); ++i) { | 23 int best_delta_in_size = std::numeric_limits<int>::min(); |
| 28 if (sizes[i].height() != sizes[i].width()) | |
| 29 continue; | |
| 30 if (sizes[i].width() == ideal_icon_size_in_px_) | |
| 31 return true; | |
| 32 } | |
| 33 | |
| 34 return false; | |
| 35 } | |
| 36 | |
| 37 bool ManifestIconSelector::IconSizesContainsBiggerThanMinimumSize( | |
| 38 const std::vector<gfx::Size>& sizes) const { | |
| 39 for (size_t i = 0; i < sizes.size(); ++i) { | |
| 40 if (sizes[i].height() != sizes[i].width()) | |
| 41 continue; | |
| 42 if (sizes[i].width() >= minimum_icon_size_in_px_) | |
| 43 return true; | |
| 44 } | |
| 45 return false; | |
| 46 } | |
| 47 | |
| 48 int ManifestIconSelector::FindClosestIconToIdealSize( | |
| 49 const std::vector<content::Manifest::Icon>& icons) const { | |
| 50 int best_index = -1; | |
| 51 int best_delta = std::numeric_limits<int>::min(); | |
| 52 | 24 |
| 53 for (size_t i = 0; i < icons.size(); ++i) { | 25 for (size_t i = 0; i < icons.size(); ++i) { |
| 54 const std::vector<gfx::Size>& sizes = icons[i].sizes; | 26 const auto& icon = icons[i]; |
| 55 for (size_t j = 0; j < sizes.size(); ++j) { | 27 |
| 56 if (sizes[j].height() != sizes[j].width()) | 28 // Check for supported image MIME types. |
| 29 if (!(icon.type.empty() || | |
| 30 mime_util::IsSupportedImageMimeType(base::UTF16ToUTF8(icon.type)))) { | |
|
pkotwicz
2017/01/31 18:24:53
Nit: Expand this out
icon.type.empty() && !mime_u
F
2017/01/31 18:42:50
Done.
| |
| 31 continue; | |
| 32 } | |
| 33 | |
| 34 // Check for icon purpose. | |
| 35 if (!base::ContainsValue(icon.purpose, purpose)) | |
|
pkotwicz
2017/01/31 18:24:53
Cool! I didn't know about base::ContainsValue()
F
2017/01/31 18:42:50
:)
| |
| 36 continue; | |
| 37 | |
| 38 // Check for size constraints. | |
| 39 for (const auto& size : icon.sizes) { | |
|
pkotwicz
2017/01/31 18:24:53
Nit: "const auto& size" -> "const gfx::Size& size"
F
2017/01/31 18:42:50
Done.
| |
| 40 // Check for size "any". Return this icon if no better one is found. | |
|
pkotwicz
2017/01/31 18:24:53
Remove "Return this icon if no better one is found
F
2017/01/31 18:42:50
Resolved offline.
| |
| 41 if (size.IsEmpty()) { | |
| 42 latest_size_any_index = i; | |
| 57 continue; | 43 continue; |
| 58 int delta = sizes[j].width() - ideal_icon_size_in_px_; | 44 } |
| 59 if (delta == 0) | 45 |
| 60 return i; | 46 // Check for squareness. |
| 61 if (best_delta > 0 && delta < 0) | 47 if (size.width() != size.height()) |
| 62 continue; | 48 continue; |
| 63 if ((best_delta > 0 && delta < best_delta) || | 49 |
| 64 (best_delta < 0 && delta > best_delta)) { | 50 // Check for minimum size. |
| 65 best_index = i; | 51 if (size.width() < minimum_icon_size_in_px) |
| 66 best_delta = delta; | 52 continue; |
| 53 | |
| 54 // Check for ideal size. Return this icon immediately. | |
|
pkotwicz
2017/01/31 18:24:53
Remove "Return this icon immediately." It is obvio
F
2017/01/31 18:42:50
Resolved offline.
| |
| 55 if (size.width() == ideal_icon_size_in_px) | |
| 56 return icon.src; | |
| 57 | |
| 58 // Check for closest match. | |
| 59 int delta = size.width() - ideal_icon_size_in_px; | |
| 60 | |
| 61 // Smallest icon larger than ideal size has priority over largest icon | |
| 62 // smaller than ideal size. | |
| 63 if (best_delta_in_size > 0 && delta < 0) | |
| 64 continue; | |
| 65 | |
| 66 if ((best_delta_in_size > 0 && delta < best_delta_in_size) || | |
| 67 (best_delta_in_size < 0 && delta > best_delta_in_size)) { | |
| 68 closest_size_match_index = i; | |
| 69 best_delta_in_size = delta; | |
| 67 } | 70 } |
| 68 } | 71 } |
| 69 } | 72 } |
| 70 | 73 |
| 71 return best_index; | 74 if (latest_size_any_index != -1) |
| 75 return icons[latest_size_any_index].src; | |
| 76 else if (closest_size_match_index != -1) | |
| 77 return icons[closest_size_match_index].src; | |
| 78 else | |
| 79 return GURL(); | |
| 72 } | 80 } |
| 73 | |
| 74 int ManifestIconSelector::FindBestMatchingIcon( | |
| 75 const std::vector<content::Manifest::Icon>& icons) const { | |
| 76 int best_index = -1; | |
| 77 | |
| 78 // The first pass is to find the ideal icon - one with the exact right size. | |
| 79 for (size_t i = 0; i < icons.size(); ++i) { | |
| 80 if (IconSizesContainsPreferredSize(icons[i].sizes)) | |
| 81 return i; | |
| 82 | |
| 83 // If there is an icon size 'any', keep it on the side and only use it if | |
| 84 // nothing better is found. | |
| 85 if (IconSizesContainsAny(icons[i].sizes)) | |
| 86 best_index = i; | |
| 87 } | |
| 88 if (best_index != -1) | |
| 89 return best_index; | |
| 90 | |
| 91 // The last pass will try to find the smallest icon larger than the ideal | |
| 92 // size, or the largest icon smaller than the ideal size. | |
| 93 best_index = FindClosestIconToIdealSize(icons); | |
| 94 | |
| 95 if (best_index != -1 && | |
| 96 IconSizesContainsBiggerThanMinimumSize(icons[best_index].sizes)) | |
| 97 return best_index; | |
| 98 | |
| 99 return -1; | |
| 100 } | |
| 101 | |
| 102 | |
| 103 // static | |
| 104 bool ManifestIconSelector::IconSizesContainsAny( | |
| 105 const std::vector<gfx::Size>& sizes) { | |
| 106 for (size_t i = 0; i < sizes.size(); ++i) { | |
| 107 if (sizes[i].IsEmpty()) | |
| 108 return true; | |
| 109 } | |
| 110 return false; | |
| 111 } | |
| 112 | |
| 113 // static | |
| 114 std::vector<Manifest::Icon> ManifestIconSelector::FilterIconsByType( | |
| 115 const std::vector<content::Manifest::Icon>& icons) { | |
| 116 std::vector<Manifest::Icon> result; | |
| 117 | |
| 118 for (size_t i = 0; i < icons.size(); ++i) { | |
| 119 if (icons[i].type.empty() || | |
| 120 mime_util::IsSupportedImageMimeType(base::UTF16ToUTF8(icons[i].type))) { | |
| 121 result.push_back(icons[i]); | |
| 122 } | |
| 123 } | |
| 124 | |
| 125 return result; | |
| 126 } | |
| 127 | |
| 128 // static | |
| 129 GURL ManifestIconSelector::FindBestMatchingIcon( | |
| 130 const std::vector<Manifest::Icon>& unfiltered_icons, | |
| 131 const int ideal_icon_size_in_px, | |
| 132 const int minimum_icon_size_in_px) { | |
| 133 DCHECK(minimum_icon_size_in_px <= ideal_icon_size_in_px); | |
|
pkotwicz
2017/01/31 18:24:53
Shouldn't we keep this DCHECK?
F
2017/01/31 18:42:50
Done. Thanks!
| |
| 134 | |
| 135 std::vector<Manifest::Icon> icons = | |
| 136 ManifestIconSelector::FilterIconsByType(unfiltered_icons); | |
| 137 | |
| 138 ManifestIconSelector selector(ideal_icon_size_in_px, | |
| 139 minimum_icon_size_in_px); | |
| 140 int index = selector.FindBestMatchingIcon(icons); | |
| 141 if (index == -1) | |
| 142 return GURL(); | |
| 143 return icons[index].src; | |
| 144 } | |
| OLD | NEW |