summaryrefslogtreecommitdiffstats
path: root/chrome/browser/history/select_favicon_frames.cc
diff options
context:
space:
mode:
authorpkotwicz@chromium.org <pkotwicz@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2012-09-06 07:27:15 +0000
committerpkotwicz@chromium.org <pkotwicz@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2012-09-06 07:27:15 +0000
commit60b10c7b8b3a39adde92adbd86f50a827ea41523 (patch)
tree7e3fe92fd40742356bcb657aaaf88b0b82ac1b62 /chrome/browser/history/select_favicon_frames.cc
parent5b9cf6fdc0d54e811a3461355cca186cbdb1a20c (diff)
downloadchromium_src-60b10c7b8b3a39adde92adbd86f50a827ea41523.zip
chromium_src-60b10c7b8b3a39adde92adbd86f50a827ea41523.tar.gz
chromium_src-60b10c7b8b3a39adde92adbd86f50a827ea41523.tar.bz2
Move SelectFaviconFrames from favicon/ to history/ so that HistoryBackend can use it.
BUG=10802066 Test=chrome and unit_tests compile R=thakis TBR=sky Review URL: https://chromiumcodereview.appspot.com/10908114 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@155139 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'chrome/browser/history/select_favicon_frames.cc')
-rw-r--r--chrome/browser/history/select_favicon_frames.cc248
1 files changed, 248 insertions, 0 deletions
diff --git a/chrome/browser/history/select_favicon_frames.cc b/chrome/browser/history/select_favicon_frames.cc
new file mode 100644
index 0000000..6619f0f
--- /dev/null
+++ b/chrome/browser/history/select_favicon_frames.cc
@@ -0,0 +1,248 @@
+// Copyright (c) 2012 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/history/select_favicon_frames.h"
+
+#include "skia/ext/image_operations.h"
+#include "third_party/skia/include/core/SkCanvas.h"
+#include "ui/gfx/image/image.h"
+#include "ui/gfx/image/image_skia.h"
+#include "ui/gfx/size.h"
+
+namespace {
+
+void SizesFromBitmaps(const std::vector<SkBitmap>& bitmaps,
+ std::vector<gfx::Size>* sizes) {
+ for (size_t i = 0; i < bitmaps.size(); ++i)
+ sizes->push_back(gfx::Size(bitmaps[i].width(), bitmaps[i].height()));
+}
+
+size_t BiggestCandidate(const std::vector<gfx::Size>& candidate_sizes) {
+ size_t max_index = 0;
+ int max_area = candidate_sizes[0].GetArea();
+ for (size_t i = 1; i < candidate_sizes.size(); ++i) {
+ int area = candidate_sizes[i].GetArea();
+ if (area > max_area) {
+ max_area = area;
+ max_index = i;
+ }
+ }
+ return max_index;
+}
+
+SkBitmap PadWithBorder(const SkBitmap& contents,
+ int desired_size,
+ int source_size) {
+ SkBitmap bitmap;
+ bitmap.setConfig(
+ SkBitmap::kARGB_8888_Config, desired_size, desired_size);
+ bitmap.allocPixels();
+ bitmap.eraseARGB(0, 0, 0, 0);
+
+ {
+ SkCanvas canvas(bitmap);
+ int shift = (desired_size - source_size) / 2;
+ SkRect dest(SkRect::MakeXYWH(shift, shift, source_size, source_size));
+ canvas.drawBitmapRect(contents, NULL, dest);
+ }
+
+ return bitmap;
+}
+
+SkBitmap SampleNearestNeighbor(const SkBitmap& contents, int desired_size) {
+ SkBitmap bitmap;
+ bitmap.setConfig(
+ SkBitmap::kARGB_8888_Config, desired_size, desired_size);
+ bitmap.allocPixels();
+ if (!contents.isOpaque())
+ bitmap.eraseARGB(0, 0, 0, 0);
+
+ {
+ SkCanvas canvas(bitmap);
+ SkRect dest(SkRect::MakeWH(desired_size, desired_size));
+ canvas.drawBitmapRect(contents, NULL, dest);
+ }
+
+ return bitmap;
+}
+
+enum ResizeMethod {
+NONE,
+PAD_WITH_BORDER,
+SAMPLE_NEAREST_NEIGHBOUR,
+LANCZOS
+};
+
+size_t GetCandidateIndexWithBestScore(
+ const std::vector<gfx::Size>& candidate_sizes,
+ ui::ScaleFactor scale_factor,
+ int desired_size,
+ float* score,
+ ResizeMethod* resize_method) {
+ float scale = ui::GetScaleFactorScale(scale_factor);
+ desired_size = static_cast<int>(desired_size * scale + 0.5f);
+
+ // Try to find an exact match.
+ for (size_t i = 0; i < candidate_sizes.size(); ++i) {
+ if (candidate_sizes[i].width() == desired_size &&
+ candidate_sizes[i].height() == desired_size) {
+ *score = 1;
+ *resize_method = NONE;
+ return i;
+ }
+ }
+
+ // If that failed, the following special rules apply:
+ // 1. 17px-24px images are built from 16px images by adding
+ // a transparent border.
+ if (desired_size > 16 * scale && desired_size <= 24 * scale) {
+ int source_size = static_cast<int>(16 * scale + 0.5f);
+ for (size_t i = 0; i < candidate_sizes.size(); ++i) {
+ if (candidate_sizes[i].width() == source_size &&
+ candidate_sizes[i].height() == source_size) {
+ *score = 0.2f;
+ *resize_method = PAD_WITH_BORDER;
+ return i;
+ }
+ }
+ // Try again, with upsizing the base variant.
+ for (size_t i = 0; i < candidate_sizes.size(); ++i) {
+ if (candidate_sizes[i].width() * scale == source_size &&
+ candidate_sizes[i].height() * scale == source_size) {
+ *score = 0.15f;
+ *resize_method = PAD_WITH_BORDER;
+ return i;
+ }
+ }
+ }
+
+ // 2. Integer multiples are built using nearest neighbor sampling.
+ // 3. Else, use Lancosz scaling:
+ // b) If available, from the next bigger variant.
+ int candidate_index = -1;
+ int min_area = INT_MAX;
+ for (size_t i = 0; i < candidate_sizes.size(); ++i) {
+ int area = candidate_sizes[i].GetArea();
+ if (candidate_sizes[i].width() > desired_size &&
+ candidate_sizes[i].height() > desired_size &&
+ (candidate_index == -1 || area < min_area)) {
+ candidate_index = i;
+ min_area = area;
+ }
+ }
+ *score = 0.1f;
+ // c) Else, from the biggest smaller variant.
+ if (candidate_index == -1) {
+ *score = 0;
+ candidate_index = BiggestCandidate(candidate_sizes);
+ }
+
+ const gfx::Size& candidate_size = candidate_sizes[candidate_index];
+ bool is_integer_multiple = desired_size % candidate_size.width() == 0 &&
+ desired_size % candidate_size.height() == 0;
+ *resize_method = is_integer_multiple ? SAMPLE_NEAREST_NEIGHBOUR : LANCZOS;
+ return candidate_index;
+}
+
+// Represents the index of the best candidate for a |scale_factor| from the
+// |candidate_sizes| passed into GetCandidateIndicesWithBestScores().
+struct SelectionResult {
+ // index in |candidate_sizes| of the best candidate.
+ size_t index;
+
+ // The ScaleFactor for which |index| is the best candidate.
+ ui::ScaleFactor scale_factor;
+
+ // How the bitmap data that the bitmap with |candidate_sizes[index]| should
+ // be resized for displaying in the UI.
+ ResizeMethod resize_method;
+};
+
+void GetCandidateIndicesWithBestScores(
+ const std::vector<gfx::Size>& candidate_sizes,
+ const std::vector<ui::ScaleFactor>& scale_factors,
+ int desired_size,
+ float* match_score,
+ std::vector<SelectionResult>* results) {
+ if (candidate_sizes.empty())
+ return;
+
+ if (desired_size == 0) {
+ // Just return the biggest image available.
+ SelectionResult result;
+ result.index = BiggestCandidate(candidate_sizes);
+ result.scale_factor = ui::SCALE_FACTOR_100P;
+ result.resize_method = NONE;
+ results->push_back(result);
+ if (match_score)
+ *match_score = 0.8f;
+ return;
+ }
+
+ float total_score = 0;
+ for (size_t i = 0; i < scale_factors.size(); ++i) {
+ float score;
+ SelectionResult result;
+ result.scale_factor = scale_factors[i];
+ result.index = GetCandidateIndexWithBestScore(candidate_sizes,
+ result.scale_factor, desired_size, &score, &result.resize_method);
+ results->push_back(result);
+ total_score += score;
+ }
+
+ if (match_score)
+ *match_score = total_score / scale_factors.size();
+}
+
+// Resize |source_bitmap| using |resize_method|.
+SkBitmap GetResizedBitmap(const SkBitmap& source_bitmap,
+ int desired_size_in_dip,
+ ui::ScaleFactor scale_factor,
+ ResizeMethod resize_method) {
+ float scale = ui::GetScaleFactorScale(scale_factor);
+ int desired_size_in_pixel = static_cast<int>(
+ desired_size_in_dip * scale + 0.5f);
+
+ switch(resize_method) {
+ case NONE:
+ return source_bitmap;
+ case PAD_WITH_BORDER: {
+ int inner_border_in_pixel = static_cast<int>(16 * scale + 0.5f);
+ return PadWithBorder(source_bitmap, desired_size_in_pixel,
+ inner_border_in_pixel);
+ }
+ case SAMPLE_NEAREST_NEIGHBOUR:
+ return SampleNearestNeighbor(source_bitmap, desired_size_in_pixel);
+ case LANCZOS:
+ return skia::ImageOperations::Resize(
+ source_bitmap, skia::ImageOperations::RESIZE_LANCZOS3,
+ desired_size_in_pixel, desired_size_in_pixel);
+ }
+ return source_bitmap;
+}
+
+} // namespace
+
+gfx::ImageSkia SelectFaviconFrames(
+ const std::vector<SkBitmap>& bitmaps,
+ const std::vector<ui::ScaleFactor>& scale_factors,
+ int desired_size,
+ float* match_score) {
+ std::vector<gfx::Size> candidate_sizes;
+ SizesFromBitmaps(bitmaps, &candidate_sizes);
+
+ std::vector<SelectionResult> results;
+ GetCandidateIndicesWithBestScores(candidate_sizes, scale_factors,
+ desired_size, match_score, &results);
+
+ gfx::ImageSkia multi_image;
+ for (size_t i = 0; i < results.size(); ++i) {
+ const SelectionResult& result = results[i];
+ SkBitmap resized_bitmap = GetResizedBitmap(bitmaps[result.index],
+ desired_size, result.scale_factor, result.resize_method);
+ multi_image.AddRepresentation(
+ gfx::ImageSkiaRep(resized_bitmap, result.scale_factor));
+ }
+ return multi_image;
+}