|  | // Copyright 2015 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 "content/renderer/image_downloader/image_downloader_impl.h" | 
|  |  | 
|  | #include <utility> | 
|  |  | 
|  | #include "base/bind.h" | 
|  | #include "base/logging.h" | 
|  | #include "content/public/renderer/render_frame.h" | 
|  | #include "skia/ext/image_operations.h" | 
|  | #include "ui/gfx/geometry/size.h" | 
|  | #include "ui/gfx/skbitmap_operations.h" | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | //  Proportionally resizes the |image| to fit in a box of size | 
|  | // |max_image_size|. | 
|  | SkBitmap ResizeImage(const SkBitmap& image, uint32_t max_image_size) { | 
|  | if (max_image_size == 0) | 
|  | return image; | 
|  | uint32_t max_dimension = std::max(image.width(), image.height()); | 
|  | if (max_dimension <= max_image_size) | 
|  | return image; | 
|  | // Proportionally resize the minimal image to fit in a box of size | 
|  | // max_image_size. | 
|  | return skia::ImageOperations::Resize( | 
|  | image, skia::ImageOperations::RESIZE_BEST, | 
|  | static_cast<uint64_t>(image.width()) * max_image_size / max_dimension, | 
|  | static_cast<uint64_t>(image.height()) * max_image_size / max_dimension); | 
|  | } | 
|  |  | 
|  | // Filters the array of bitmaps, removing all images that do not fit in a box of | 
|  | // size |max_image_size|. Returns the result if it is not empty. Otherwise, | 
|  | // find the smallest image in the array and resize it proportionally to fit | 
|  | // in a box of size |max_image_size|. | 
|  | // Sets |original_image_sizes| to the sizes of |images| before resizing. | 
|  | void FilterAndResizeImagesForMaximalSize( | 
|  | const std::vector<SkBitmap>& unfiltered, | 
|  | uint32_t max_image_size, | 
|  | std::vector<SkBitmap>* images, | 
|  | std::vector<gfx::Size>* original_image_sizes) { | 
|  | images->clear(); | 
|  | original_image_sizes->clear(); | 
|  |  | 
|  | if (unfiltered.empty()) | 
|  | return; | 
|  |  | 
|  | if (max_image_size == 0) | 
|  | max_image_size = std::numeric_limits<uint32_t>::max(); | 
|  |  | 
|  | const SkBitmap* min_image = nullptr; | 
|  | uint32_t min_image_size = std::numeric_limits<uint32_t>::max(); | 
|  | // Filter the images by |max_image_size|, and also identify the smallest image | 
|  | // in case all the images are bigger than |max_image_size|. | 
|  | for (std::vector<SkBitmap>::const_iterator it = unfiltered.begin(); | 
|  | it != unfiltered.end(); ++it) { | 
|  | const SkBitmap& image = *it; | 
|  | uint32_t current_size = std::max(it->width(), it->height()); | 
|  | if (current_size < min_image_size) { | 
|  | min_image = ℑ | 
|  | min_image_size = current_size; | 
|  | } | 
|  | if (static_cast<uint32_t>(image.width()) <= max_image_size && | 
|  | static_cast<uint32_t>(image.height()) <= max_image_size) { | 
|  | images->push_back(image); | 
|  | original_image_sizes->push_back(gfx::Size(image.width(), image.height())); | 
|  | } | 
|  | } | 
|  | DCHECK(min_image); | 
|  | if (images->size()) | 
|  | return; | 
|  | // Proportionally resize the minimal image to fit in a box of size | 
|  | // |max_image_size|. | 
|  | SkBitmap resized = ResizeImage(*min_image, max_image_size); | 
|  | // Drop null or empty SkBitmap. | 
|  | if (resized.drawsNothing()) | 
|  | return; | 
|  | images->push_back(resized); | 
|  | original_image_sizes->push_back( | 
|  | gfx::Size(min_image->width(), min_image->height())); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | namespace content { | 
|  |  | 
|  | ImageDownloaderImpl::ImageDownloaderImpl(RenderFrame* render_frame, | 
|  | mojom::ImageDownloaderRequest request) | 
|  | : ImageDownloaderBase(render_frame), binding_(this, std::move(request)) { | 
|  | DCHECK(render_frame); | 
|  | binding_.set_connection_error_handler( | 
|  | base::BindOnce(&ImageDownloaderImpl::OnDestruct, base::Unretained(this))); | 
|  | } | 
|  |  | 
|  | ImageDownloaderImpl::~ImageDownloaderImpl() {} | 
|  |  | 
|  | // static | 
|  | void ImageDownloaderImpl::CreateMojoService( | 
|  | RenderFrame* render_frame, | 
|  | mojom::ImageDownloaderRequest request) { | 
|  | DVLOG(1) << "ImageDownloaderImpl::CreateMojoService"; | 
|  | DCHECK(render_frame); | 
|  |  | 
|  | // Owns itself. Will be deleted when message pipe is destroyed or RenderFrame | 
|  | // is destructed. | 
|  | new ImageDownloaderImpl(render_frame, std::move(request)); | 
|  | } | 
|  |  | 
|  | // ImageDownloader methods: | 
|  | void ImageDownloaderImpl::DownloadImage(const GURL& image_url, | 
|  | bool is_favicon, | 
|  | uint32_t max_bitmap_size, | 
|  | bool bypass_cache, | 
|  | DownloadImageCallback callback) { | 
|  | std::vector<SkBitmap> result_images; | 
|  | std::vector<gfx::Size> result_original_image_sizes; | 
|  |  | 
|  | ImageDownloaderBase::DownloadImage( | 
|  | image_url, is_favicon, bypass_cache, | 
|  | base::BindOnce(&ImageDownloaderImpl::DidDownloadImage, | 
|  | base::Unretained(this), max_bitmap_size, | 
|  | std::move(callback))); | 
|  | } | 
|  |  | 
|  | void ImageDownloaderImpl::DidDownloadImage( | 
|  | uint32_t max_image_size, | 
|  | DownloadImageCallback callback, | 
|  | int32_t http_status_code, | 
|  | const std::vector<SkBitmap>& images) { | 
|  | std::vector<SkBitmap> result_images; | 
|  | std::vector<gfx::Size> result_original_image_sizes; | 
|  | FilterAndResizeImagesForMaximalSize(images, max_image_size, &result_images, | 
|  | &result_original_image_sizes); | 
|  |  | 
|  | std::move(callback).Run(http_status_code, result_images, | 
|  | result_original_image_sizes); | 
|  | } | 
|  |  | 
|  | void ImageDownloaderImpl::OnDestruct() { | 
|  | delete this; | 
|  | } | 
|  |  | 
|  | }  // namespace content |