| Index: ui/gfx/image/image_family.cc
|
| diff --git a/ui/gfx/image/image_family.cc b/ui/gfx/image/image_family.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..e7e9234f695b2d5e48220655c51b2cc0f344fff7
|
| --- /dev/null
|
| +++ b/ui/gfx/image/image_family.cc
|
| @@ -0,0 +1,124 @@
|
| +// Copyright 2013 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 "ui/gfx/image/image_family.h"
|
| +
|
| +#include <cmath>
|
| +
|
| +#include "ui/gfx/image/image.h"
|
| +#include "ui/gfx/image/image_skia.h"
|
| +#include "ui/gfx/size.h"
|
| +
|
| +namespace gfx {
|
| +
|
| +ImageFamily::const_iterator::const_iterator() {}
|
| +
|
| +ImageFamily::const_iterator::const_iterator(const const_iterator& other)
|
| + : map_iterator_(other.map_iterator_) {}
|
| +
|
| +ImageFamily::const_iterator::const_iterator(
|
| + const std::map<MapKey, gfx::Image>::const_iterator& other)
|
| + : map_iterator_(other) {}
|
| +
|
| +ImageFamily::ImageFamily() {}
|
| +ImageFamily::~ImageFamily() {}
|
| +
|
| +void ImageFamily::Add(const gfx::Image& image) {
|
| + gfx::Size size = image.Size();
|
| + if (size.IsEmpty()) {
|
| + map_[MapKey(1.0f, 0)] = image;
|
| + } else {
|
| + float aspect = static_cast<float>(size.width()) / size.height();
|
| + DCHECK_GT(aspect, 0.0f);
|
| + map_[MapKey(aspect, size.width())] = image;
|
| + }
|
| +}
|
| +
|
| +void ImageFamily::Add(const gfx::ImageSkia& image_skia) {
|
| + Add(gfx::Image(image_skia));
|
| +}
|
| +
|
| +const gfx::Image* ImageFamily::Get(int width, int height) const {
|
| + if (map_.empty())
|
| + return NULL;
|
| +
|
| + // If either |width| or |height| is 0, both are.
|
| + float desired_aspect;
|
| + if (height == 0 || width == 0) {
|
| + desired_aspect = 1.0f;
|
| + height = 0;
|
| + width = 0;
|
| + } else {
|
| + desired_aspect = static_cast<float>(width) / height;
|
| + }
|
| + DCHECK_GT(desired_aspect, 0.0f);
|
| +
|
| + float closest_aspect = GetClosestAspect(desired_aspect);
|
| +
|
| + // If thinner than desired, search for images with width such that the
|
| + // corresponding height is greater than or equal to the desired |height|.
|
| + int desired_width = closest_aspect <= desired_aspect ?
|
| + width : static_cast<int>(ceilf(height * closest_aspect));
|
| +
|
| + // Get the best-sized image with the aspect ratio.
|
| + return GetWithExactAspect(closest_aspect, desired_width);
|
| +}
|
| +
|
| +float ImageFamily::GetClosestAspect(float desired_aspect) const {
|
| + // Find the two aspect ratios on either side of |desired_aspect|.
|
| + std::map<MapKey, gfx::Image>::const_iterator greater_or_equal =
|
| + map_.lower_bound(MapKey(desired_aspect, 0));
|
| + // Early exit optimization if there is an exact match.
|
| + if (greater_or_equal != map_.end() &&
|
| + greater_or_equal->first.aspect() == desired_aspect) {
|
| + return desired_aspect;
|
| + }
|
| +
|
| + // No exact match; |greater_or_equal| will point to the first image with
|
| + // aspect ratio >= |desired_aspect|, and |less_than| will point to the last
|
| + // image with aspect ratio < |desired_aspect|.
|
| + if (greater_or_equal != map_.begin()) {
|
| + std::map<MapKey, gfx::Image>::const_iterator less_than =
|
| + greater_or_equal;
|
| + --less_than;
|
| + float thinner_aspect = less_than->first.aspect();
|
| + DCHECK_GT(thinner_aspect, 0.0f);
|
| + DCHECK_LT(thinner_aspect, desired_aspect);
|
| + if (greater_or_equal != map_.end()) {
|
| + float wider_aspect = greater_or_equal->first.aspect();
|
| + DCHECK_GT(wider_aspect, desired_aspect);
|
| + if ((wider_aspect / desired_aspect) < (desired_aspect / thinner_aspect))
|
| + return wider_aspect;
|
| + }
|
| + return thinner_aspect;
|
| + } else {
|
| + // No aspect ratio is less than or equal to |desired_aspect|.
|
| + DCHECK(greater_or_equal != map_.end());
|
| + float wider_aspect = greater_or_equal->first.aspect();
|
| + DCHECK_GT(wider_aspect, desired_aspect);
|
| + return wider_aspect;
|
| + }
|
| +}
|
| +
|
| +const gfx::Image* ImageFamily::GetWithExactAspect(float aspect,
|
| + int width) const {
|
| + // Find the two images of given aspect ratio on either side of |width|.
|
| + std::map<MapKey, gfx::Image>::const_iterator greater_or_equal =
|
| + map_.lower_bound(MapKey(aspect, width));
|
| + if (greater_or_equal != map_.end() &&
|
| + greater_or_equal->first.aspect() == aspect) {
|
| + // We have found the smallest image of the same size or greater.
|
| + return &greater_or_equal->second;
|
| + }
|
| +
|
| + DCHECK(greater_or_equal != map_.begin());
|
| + std::map<MapKey, gfx::Image>::const_iterator less_than = greater_or_equal;
|
| + --less_than;
|
| + // This must be true because there must be at least one image with |aspect|.
|
| + DCHECK_EQ(less_than->first.aspect(), aspect);
|
| + // We have found the largest image smaller than desired.
|
| + return &less_than->second;
|
| +}
|
| +
|
| +} // namespace gfx
|
|
|