1 // Copyright (c) 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "ash/desktop_background/wallpaper_resizer.h"
7 #include "ash/desktop_background/wallpaper_resizer_observer.h"
9 #include "base/logging.h"
10 #include "base/threading/sequenced_worker_pool.h"
11 #include "base/threading/worker_pool.h"
12 #include "content/public/browser/browser_thread.h"
13 #include "third_party/skia/include/core/SkImage.h"
14 #include "ui/gfx/image/image_skia_rep.h"
15 #include "ui/gfx/skia_util.h"
17 using content::BrowserThread
;
22 // For our scaling ratios we need to round positive numbers.
23 int RoundPositive(double x
) {
24 return static_cast<int>(floor(x
+ 0.5));
27 // Resizes |orig_bitmap| to |target_size| using |layout| and stores the
28 // resulting bitmap at |resized_bitmap_out|.
29 void Resize(SkBitmap orig_bitmap
,
30 const gfx::Size
& target_size
,
31 WallpaperLayout layout
,
32 SkBitmap
* resized_bitmap_out
) {
33 DCHECK(BrowserThread::GetBlockingPool()->RunsTasksOnCurrentThread());
34 SkBitmap new_bitmap
= orig_bitmap
;
36 const int orig_width
= orig_bitmap
.width();
37 const int orig_height
= orig_bitmap
.height();
38 const int new_width
= target_size
.width();
39 const int new_height
= target_size
.height();
41 if (orig_width
> new_width
|| orig_height
> new_height
) {
42 gfx::Rect
wallpaper_rect(0, 0, orig_width
, orig_height
);
43 gfx::Size cropped_size
= gfx::Size(std::min(new_width
, orig_width
),
44 std::min(new_height
, orig_height
));
46 case WALLPAPER_LAYOUT_CENTER
:
47 wallpaper_rect
.ClampToCenteredSize(cropped_size
);
48 orig_bitmap
.extractSubset(&new_bitmap
,
49 gfx::RectToSkIRect(wallpaper_rect
));
51 case WALLPAPER_LAYOUT_TILE
:
52 wallpaper_rect
.set_size(cropped_size
);
53 orig_bitmap
.extractSubset(&new_bitmap
,
54 gfx::RectToSkIRect(wallpaper_rect
));
56 case WALLPAPER_LAYOUT_STRETCH
:
57 new_bitmap
= skia::ImageOperations::Resize(
58 orig_bitmap
, skia::ImageOperations::RESIZE_LANCZOS3
,
59 new_width
, new_height
);
61 case WALLPAPER_LAYOUT_CENTER_CROPPED
:
62 if (orig_width
> new_width
&& orig_height
> new_height
) {
63 // The dimension with the smallest ratio must be cropped, the other
64 // one is preserved. Both are set in gfx::Size cropped_size.
65 double horizontal_ratio
= static_cast<double>(new_width
) /
66 static_cast<double>(orig_width
);
67 double vertical_ratio
= static_cast<double>(new_height
) /
68 static_cast<double>(orig_height
);
70 if (vertical_ratio
> horizontal_ratio
) {
71 cropped_size
= gfx::Size(
72 RoundPositive(static_cast<double>(new_width
) / vertical_ratio
),
75 cropped_size
= gfx::Size(orig_width
, RoundPositive(
76 static_cast<double>(new_height
) / horizontal_ratio
));
78 wallpaper_rect
.ClampToCenteredSize(cropped_size
);
80 orig_bitmap
.extractSubset(&sub_image
,
81 gfx::RectToSkIRect(wallpaper_rect
));
82 new_bitmap
= skia::ImageOperations::Resize(
83 sub_image
, skia::ImageOperations::RESIZE_LANCZOS3
,
84 new_width
, new_height
);
89 *resized_bitmap_out
= new_bitmap
;
90 resized_bitmap_out
->setImmutable();
96 uint32_t WallpaperResizer::GetImageId(const gfx::ImageSkia
& image
) {
97 const gfx::ImageSkiaRep
& image_rep
= image
.GetRepresentation(1.0f
);
98 return image_rep
.is_null() ? 0 : image_rep
.sk_bitmap().getGenerationID();
101 WallpaperResizer::WallpaperResizer(const gfx::ImageSkia
& image
,
102 const gfx::Size
& target_size
,
103 WallpaperLayout layout
)
105 original_image_id_(GetImageId(image_
)),
106 target_size_(target_size
),
108 weak_ptr_factory_(this) {
109 image_
.MakeThreadSafe();
112 WallpaperResizer::~WallpaperResizer() {
115 void WallpaperResizer::StartResize() {
116 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI
));
117 SkBitmap
* resized_bitmap
= new SkBitmap
;
118 if (!content::BrowserThread::PostBlockingPoolTaskAndReply(
120 base::Bind(&Resize
, *image_
.bitmap(), target_size_
,
121 layout_
, resized_bitmap
),
122 base::Bind(&WallpaperResizer::OnResizeFinished
,
123 weak_ptr_factory_
.GetWeakPtr(),
124 base::Owned(resized_bitmap
)))) {
125 LOG(WARNING
) << "PostSequencedWorkerTask failed. "
126 << "Wallpaper may not be resized.";
130 void WallpaperResizer::AddObserver(WallpaperResizerObserver
* observer
) {
131 observers_
.AddObserver(observer
);
134 void WallpaperResizer::RemoveObserver(WallpaperResizerObserver
* observer
) {
135 observers_
.RemoveObserver(observer
);
138 void WallpaperResizer::OnResizeFinished(SkBitmap
* resized_bitmap
) {
139 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI
));
140 image_
= gfx::ImageSkia::CreateFrom1xBitmap(*resized_bitmap
);
141 FOR_EACH_OBSERVER(WallpaperResizerObserver
, observers_
,
142 OnWallpaperResized());