• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 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.
4 
5 #include "ui/gfx/image/image_skia_operations.h"
6 
7 #include "base/command_line.h"
8 #include "base/logging.h"
9 #include "skia/ext/image_operations.h"
10 #include "ui/gfx/canvas.h"
11 #include "ui/gfx/image/canvas_image_source.h"
12 #include "ui/gfx/image/image_skia.h"
13 #include "ui/gfx/image/image_skia_rep.h"
14 #include "ui/gfx/image/image_skia_source.h"
15 #include "ui/gfx/insets.h"
16 #include "ui/gfx/point.h"
17 #include "ui/gfx/point_conversions.h"
18 #include "ui/gfx/rect.h"
19 #include "ui/gfx/rect_conversions.h"
20 #include "ui/gfx/size.h"
21 #include "ui/gfx/size_conversions.h"
22 #include "ui/gfx/skbitmap_operations.h"
23 #include "ui/gfx/skia_util.h"
24 
25 namespace gfx {
26 namespace {
27 
DIPToPixelSize(gfx::Size dip_size,float scale)28 gfx::Size DIPToPixelSize(gfx::Size dip_size, float scale) {
29   return ToCeiledSize(ScaleSize(dip_size, scale));
30 }
31 
DIPToPixelBounds(gfx::Rect dip_bounds,float scale)32 gfx::Rect DIPToPixelBounds(gfx::Rect dip_bounds, float scale) {
33   return gfx::Rect(ToFlooredPoint(ScalePoint(dip_bounds.origin(), scale)),
34                    DIPToPixelSize(dip_bounds.size(), scale));
35 }
36 
37 // Returns an image rep for the ImageSkiaSource to return to visually indicate
38 // an error.
GetErrorImageRep(float scale,const gfx::Size & pixel_size)39 ImageSkiaRep GetErrorImageRep(float scale, const gfx::Size& pixel_size) {
40   SkBitmap bitmap;
41   bitmap.allocN32Pixels(pixel_size.width(), pixel_size.height());
42   bitmap.eraseColor(SK_ColorRED);
43   return gfx::ImageSkiaRep(bitmap, scale);
44 }
45 
46 // A base image source class that creates an image from two source images.
47 // This class guarantees that two ImageSkiaReps have have the same pixel size.
48 class BinaryImageSource : public gfx::ImageSkiaSource {
49  protected:
BinaryImageSource(const ImageSkia & first,const ImageSkia & second,const char * source_name)50   BinaryImageSource(const ImageSkia& first,
51                     const ImageSkia& second,
52                     const char* source_name)
53       : first_(first),
54         second_(second),
55         source_name_(source_name) {
56   }
~BinaryImageSource()57   virtual ~BinaryImageSource() {
58   }
59 
60   // gfx::ImageSkiaSource overrides:
GetImageForScale(float scale)61   virtual ImageSkiaRep GetImageForScale(float scale) OVERRIDE {
62     ImageSkiaRep first_rep = first_.GetRepresentation(scale);
63     ImageSkiaRep second_rep = second_.GetRepresentation(scale);
64     if (first_rep.pixel_size() != second_rep.pixel_size()) {
65       DCHECK_NE(first_rep.scale(), second_rep.scale());
66       if (first_rep.scale() == second_rep.scale()) {
67         LOG(ERROR) << "ImageSkiaRep size mismatch in " << source_name_;
68         return GetErrorImageRep(first_rep.scale(),first_rep.pixel_size());
69       }
70       first_rep = first_.GetRepresentation(1.0f);
71       second_rep = second_.GetRepresentation(1.0f);
72       DCHECK_EQ(first_rep.pixel_width(), second_rep.pixel_width());
73       DCHECK_EQ(first_rep.pixel_height(), second_rep.pixel_height());
74       if (first_rep.pixel_size() != second_rep.pixel_size()) {
75         LOG(ERROR) << "ImageSkiaRep size mismatch in " << source_name_;
76         return GetErrorImageRep(first_rep.scale(), first_rep.pixel_size());
77       }
78     } else {
79       DCHECK_EQ(first_rep.scale(), second_rep.scale());
80     }
81     return CreateImageSkiaRep(first_rep, second_rep);
82   }
83 
84   // Creates a final image from two ImageSkiaReps. The pixel size of
85   // the two images are guaranteed to be the same.
86   virtual ImageSkiaRep CreateImageSkiaRep(
87       const ImageSkiaRep& first_rep,
88       const ImageSkiaRep& second_rep) const = 0;
89 
90  private:
91   const ImageSkia first_;
92   const ImageSkia second_;
93   // The name of a class that implements the BinaryImageSource.
94   // The subclass is responsible for managing the memory.
95   const char* source_name_;
96 
97   DISALLOW_COPY_AND_ASSIGN(BinaryImageSource);
98 };
99 
100 class BlendingImageSource : public BinaryImageSource {
101  public:
BlendingImageSource(const ImageSkia & first,const ImageSkia & second,double alpha)102   BlendingImageSource(const ImageSkia& first,
103                       const ImageSkia& second,
104                       double alpha)
105       : BinaryImageSource(first, second, "BlendingImageSource"),
106         alpha_(alpha) {
107   }
108 
~BlendingImageSource()109   virtual ~BlendingImageSource() {
110   }
111 
112   // BinaryImageSource overrides:
CreateImageSkiaRep(const ImageSkiaRep & first_rep,const ImageSkiaRep & second_rep) const113   virtual ImageSkiaRep CreateImageSkiaRep(
114       const ImageSkiaRep& first_rep,
115       const ImageSkiaRep& second_rep) const OVERRIDE {
116     SkBitmap blended = SkBitmapOperations::CreateBlendedBitmap(
117         first_rep.sk_bitmap(), second_rep.sk_bitmap(), alpha_);
118     return ImageSkiaRep(blended, first_rep.scale());
119   }
120 
121  private:
122   double alpha_;
123 
124   DISALLOW_COPY_AND_ASSIGN(BlendingImageSource);
125 };
126 
127 class SuperimposedImageSource : public gfx::CanvasImageSource {
128  public:
SuperimposedImageSource(const ImageSkia & first,const ImageSkia & second)129   SuperimposedImageSource(const ImageSkia& first,
130                           const ImageSkia& second)
131       : gfx::CanvasImageSource(first.size(), false /* is opaque */),
132         first_(first),
133         second_(second) {
134   }
135 
~SuperimposedImageSource()136   virtual ~SuperimposedImageSource() {}
137 
138   // gfx::CanvasImageSource override.
Draw(Canvas * canvas)139   virtual void Draw(Canvas* canvas) OVERRIDE {
140     canvas->DrawImageInt(first_, 0, 0);
141     canvas->DrawImageInt(second_,
142                          (first_.width() - second_.width()) / 2,
143                          (first_.height() - second_.height()) / 2);
144   }
145 
146  private:
147   const ImageSkia first_;
148   const ImageSkia second_;
149 
150   DISALLOW_COPY_AND_ASSIGN(SuperimposedImageSource);
151 };
152 
153 class TransparentImageSource : public gfx::ImageSkiaSource {
154  public:
TransparentImageSource(const ImageSkia & image,double alpha)155   TransparentImageSource(const ImageSkia& image, double alpha)
156       : image_(image),
157         alpha_(alpha) {
158   }
159 
~TransparentImageSource()160   virtual ~TransparentImageSource() {}
161 
162  private:
163   // gfx::ImageSkiaSource overrides:
GetImageForScale(float scale)164   virtual ImageSkiaRep GetImageForScale(float scale) OVERRIDE {
165     ImageSkiaRep image_rep = image_.GetRepresentation(scale);
166     SkBitmap alpha;
167     alpha.allocN32Pixels(image_rep.pixel_width(),
168                          image_rep.pixel_height());
169     alpha.eraseColor(SkColorSetARGB(alpha_ * 255, 0, 0, 0));
170     return ImageSkiaRep(
171         SkBitmapOperations::CreateMaskedBitmap(image_rep.sk_bitmap(), alpha),
172         image_rep.scale());
173   }
174 
175   ImageSkia image_;
176   double alpha_;
177 
178   DISALLOW_COPY_AND_ASSIGN(TransparentImageSource);
179 };
180 
181 class MaskedImageSource : public BinaryImageSource {
182  public:
MaskedImageSource(const ImageSkia & rgb,const ImageSkia & alpha)183   MaskedImageSource(const ImageSkia& rgb, const ImageSkia& alpha)
184       : BinaryImageSource(rgb, alpha, "MaskedImageSource") {
185   }
186 
~MaskedImageSource()187   virtual ~MaskedImageSource() {
188   }
189 
190   // BinaryImageSource overrides:
CreateImageSkiaRep(const ImageSkiaRep & first_rep,const ImageSkiaRep & second_rep) const191   virtual ImageSkiaRep CreateImageSkiaRep(
192       const ImageSkiaRep& first_rep,
193       const ImageSkiaRep& second_rep) const OVERRIDE {
194     return ImageSkiaRep(SkBitmapOperations::CreateMaskedBitmap(
195         first_rep.sk_bitmap(), second_rep.sk_bitmap()),
196                         first_rep.scale());
197   }
198 
199  private:
200   DISALLOW_COPY_AND_ASSIGN(MaskedImageSource);
201 };
202 
203 class TiledImageSource : public gfx::ImageSkiaSource {
204  public:
TiledImageSource(const ImageSkia & source,int src_x,int src_y,int dst_w,int dst_h)205   TiledImageSource(const ImageSkia& source,
206                    int src_x, int src_y,
207                    int dst_w, int dst_h)
208       : source_(source),
209         src_x_(src_x),
210         src_y_(src_y),
211         dst_w_(dst_w),
212         dst_h_(dst_h) {
213   }
214 
~TiledImageSource()215   virtual ~TiledImageSource() {
216   }
217 
218   // gfx::ImageSkiaSource overrides:
GetImageForScale(float scale)219   virtual ImageSkiaRep GetImageForScale(float scale) OVERRIDE {
220     ImageSkiaRep source_rep = source_.GetRepresentation(scale);
221     gfx::Rect bounds = DIPToPixelBounds(gfx::Rect(src_x_, src_y_, dst_w_,
222                                                   dst_h_), source_rep.scale());
223     return ImageSkiaRep(
224         SkBitmapOperations::CreateTiledBitmap(
225             source_rep.sk_bitmap(),
226             bounds.x(), bounds.y(), bounds.width(), bounds.height()),
227         source_rep.scale());
228   }
229 
230  private:
231   const ImageSkia source_;
232   const int src_x_;
233   const int src_y_;
234   const int dst_w_;
235   const int dst_h_;
236 
237   DISALLOW_COPY_AND_ASSIGN(TiledImageSource);
238 };
239 
240 class HSLImageSource : public gfx::ImageSkiaSource {
241  public:
HSLImageSource(const ImageSkia & image,const color_utils::HSL & hsl_shift)242   HSLImageSource(const ImageSkia& image,
243                  const color_utils::HSL& hsl_shift)
244       : image_(image),
245         hsl_shift_(hsl_shift) {
246   }
247 
~HSLImageSource()248   virtual ~HSLImageSource() {
249   }
250 
251   // gfx::ImageSkiaSource overrides:
GetImageForScale(float scale)252   virtual ImageSkiaRep GetImageForScale(float scale) OVERRIDE {
253     ImageSkiaRep image_rep = image_.GetRepresentation(scale);
254     return gfx::ImageSkiaRep(
255         SkBitmapOperations::CreateHSLShiftedBitmap(image_rep.sk_bitmap(),
256             hsl_shift_), image_rep.scale());
257   }
258 
259  private:
260   const gfx::ImageSkia image_;
261   const color_utils::HSL hsl_shift_;
262   DISALLOW_COPY_AND_ASSIGN(HSLImageSource);
263 };
264 
265 // ImageSkiaSource which uses SkBitmapOperations::CreateButtonBackground
266 // to generate image reps for the target image.  The image and mask can be
267 // diferent sizes (crbug.com/171725).
268 class ButtonImageSource: public gfx::ImageSkiaSource {
269  public:
ButtonImageSource(SkColor color,const ImageSkia & image,const ImageSkia & mask)270   ButtonImageSource(SkColor color,
271                     const ImageSkia& image,
272                     const ImageSkia& mask)
273       : color_(color),
274         image_(image),
275         mask_(mask) {
276   }
277 
~ButtonImageSource()278   virtual ~ButtonImageSource() {
279   }
280 
281   // gfx::ImageSkiaSource overrides:
GetImageForScale(float scale)282   virtual ImageSkiaRep GetImageForScale(float scale) OVERRIDE {
283     ImageSkiaRep image_rep = image_.GetRepresentation(scale);
284     ImageSkiaRep mask_rep = mask_.GetRepresentation(scale);
285     if (image_rep.scale() != mask_rep.scale()) {
286       image_rep = image_.GetRepresentation(1.0f);
287       mask_rep = mask_.GetRepresentation(1.0f);
288     }
289     return gfx::ImageSkiaRep(
290         SkBitmapOperations::CreateButtonBackground(color_,
291               image_rep.sk_bitmap(), mask_rep.sk_bitmap()),
292           image_rep.scale());
293   }
294 
295  private:
296   const SkColor color_;
297   const ImageSkia image_;
298   const ImageSkia mask_;
299 
300   DISALLOW_COPY_AND_ASSIGN(ButtonImageSource);
301 };
302 
303 // ImageSkiaSource which uses SkBitmap::extractSubset to generate image reps
304 // for the target image.
305 class ExtractSubsetImageSource: public gfx::ImageSkiaSource {
306  public:
ExtractSubsetImageSource(const gfx::ImageSkia & image,const gfx::Rect & subset_bounds)307   ExtractSubsetImageSource(const gfx::ImageSkia& image,
308                            const gfx::Rect& subset_bounds)
309       : image_(image),
310         subset_bounds_(subset_bounds) {
311   }
312 
~ExtractSubsetImageSource()313   virtual ~ExtractSubsetImageSource() {
314   }
315 
316   // gfx::ImageSkiaSource overrides:
GetImageForScale(float scale)317   virtual ImageSkiaRep GetImageForScale(float scale) OVERRIDE {
318     ImageSkiaRep image_rep = image_.GetRepresentation(scale);
319     SkIRect subset_bounds_in_pixel = RectToSkIRect(
320         DIPToPixelBounds(subset_bounds_, image_rep.scale()));
321     SkBitmap dst;
322     bool success = image_rep.sk_bitmap().extractSubset(&dst,
323                                                        subset_bounds_in_pixel);
324     DCHECK(success);
325     return gfx::ImageSkiaRep(dst, image_rep.scale());
326   }
327 
328  private:
329   const gfx::ImageSkia image_;
330   const gfx::Rect subset_bounds_;
331 
332   DISALLOW_COPY_AND_ASSIGN(ExtractSubsetImageSource);
333 };
334 
335 // ResizeSource resizes relevant image reps in |source| to |target_dip_size|
336 // for requested scale factors.
337 class ResizeSource : public ImageSkiaSource {
338  public:
ResizeSource(const ImageSkia & source,skia::ImageOperations::ResizeMethod method,const Size & target_dip_size)339   ResizeSource(const ImageSkia& source,
340                skia::ImageOperations::ResizeMethod method,
341                const Size& target_dip_size)
342       : source_(source),
343         resize_method_(method),
344         target_dip_size_(target_dip_size) {
345   }
~ResizeSource()346   virtual ~ResizeSource() {}
347 
348   // gfx::ImageSkiaSource overrides:
GetImageForScale(float scale)349   virtual ImageSkiaRep GetImageForScale(float scale) OVERRIDE {
350     const ImageSkiaRep& image_rep = source_.GetRepresentation(scale);
351     if (image_rep.GetWidth() == target_dip_size_.width() &&
352         image_rep.GetHeight() == target_dip_size_.height())
353       return image_rep;
354 
355     const Size target_pixel_size = DIPToPixelSize(target_dip_size_, scale);
356     const SkBitmap resized = skia::ImageOperations::Resize(
357         image_rep.sk_bitmap(),
358         resize_method_,
359         target_pixel_size.width(),
360         target_pixel_size.height());
361     return ImageSkiaRep(resized, scale);
362   }
363 
364  private:
365   const ImageSkia source_;
366   skia::ImageOperations::ResizeMethod resize_method_;
367   const Size target_dip_size_;
368 
369   DISALLOW_COPY_AND_ASSIGN(ResizeSource);
370 };
371 
372 // DropShadowSource generates image reps with drop shadow for image reps in
373 // |source| that represent requested scale factors.
374 class DropShadowSource : public ImageSkiaSource {
375  public:
DropShadowSource(const ImageSkia & source,const ShadowValues & shadows_in_dip)376   DropShadowSource(const ImageSkia& source,
377                    const ShadowValues& shadows_in_dip)
378       : source_(source),
379         shaodws_in_dip_(shadows_in_dip) {
380   }
~DropShadowSource()381   virtual ~DropShadowSource() {}
382 
383   // gfx::ImageSkiaSource overrides:
GetImageForScale(float scale)384   virtual ImageSkiaRep GetImageForScale(float scale) OVERRIDE {
385     const ImageSkiaRep& image_rep = source_.GetRepresentation(scale);
386 
387     ShadowValues shadows_in_pixel;
388     for (size_t i = 0; i < shaodws_in_dip_.size(); ++i)
389       shadows_in_pixel.push_back(shaodws_in_dip_[i].Scale(scale));
390 
391     const SkBitmap shadow_bitmap = SkBitmapOperations::CreateDropShadow(
392         image_rep.sk_bitmap(),
393         shadows_in_pixel);
394     return ImageSkiaRep(shadow_bitmap, image_rep.scale());
395   }
396 
397  private:
398   const ImageSkia source_;
399   const ShadowValues shaodws_in_dip_;
400 
401   DISALLOW_COPY_AND_ASSIGN(DropShadowSource);
402 };
403 
404 // RotatedSource generates image reps that are rotations of those in
405 // |source| that represent requested scale factors.
406 class RotatedSource : public ImageSkiaSource {
407  public:
RotatedSource(const ImageSkia & source,SkBitmapOperations::RotationAmount rotation)408   RotatedSource(const ImageSkia& source,
409                 SkBitmapOperations::RotationAmount rotation)
410     : source_(source),
411       rotation_(rotation) {
412   }
~RotatedSource()413   virtual ~RotatedSource() {}
414 
415   // gfx::ImageSkiaSource overrides:
GetImageForScale(float scale)416   virtual ImageSkiaRep GetImageForScale(float scale) OVERRIDE {
417     const ImageSkiaRep& image_rep = source_.GetRepresentation(scale);
418     const SkBitmap rotated_bitmap =
419         SkBitmapOperations::Rotate(image_rep.sk_bitmap(), rotation_);
420     return ImageSkiaRep(rotated_bitmap, image_rep.scale());
421   }
422 
423  private:
424   const ImageSkia source_;
425   const SkBitmapOperations::RotationAmount rotation_;
426 
427   DISALLOW_COPY_AND_ASSIGN(RotatedSource);
428 };
429 
430 
431 }  // namespace
432 
433 // static
CreateBlendedImage(const ImageSkia & first,const ImageSkia & second,double alpha)434 ImageSkia ImageSkiaOperations::CreateBlendedImage(const ImageSkia& first,
435                                                   const ImageSkia& second,
436                                                   double alpha) {
437   if (first.isNull() || second.isNull())
438     return ImageSkia();
439 
440   return ImageSkia(new BlendingImageSource(first, second, alpha), first.size());
441 }
442 
443 // static
CreateSuperimposedImage(const ImageSkia & first,const ImageSkia & second)444 ImageSkia ImageSkiaOperations::CreateSuperimposedImage(
445     const ImageSkia& first,
446     const ImageSkia& second) {
447   if (first.isNull() || second.isNull())
448     return ImageSkia();
449 
450   return ImageSkia(new SuperimposedImageSource(first, second), first.size());
451 }
452 
453 // static
CreateTransparentImage(const ImageSkia & image,double alpha)454 ImageSkia ImageSkiaOperations::CreateTransparentImage(const ImageSkia& image,
455                                                       double alpha) {
456   if (image.isNull())
457     return ImageSkia();
458 
459   return ImageSkia(new TransparentImageSource(image, alpha), image.size());
460 }
461 
462 // static
CreateMaskedImage(const ImageSkia & rgb,const ImageSkia & alpha)463 ImageSkia ImageSkiaOperations::CreateMaskedImage(const ImageSkia& rgb,
464                                                  const ImageSkia& alpha) {
465   if (rgb.isNull() || alpha.isNull())
466     return ImageSkia();
467 
468   return ImageSkia(new MaskedImageSource(rgb, alpha), rgb.size());
469 }
470 
471 // static
CreateTiledImage(const ImageSkia & source,int src_x,int src_y,int dst_w,int dst_h)472 ImageSkia ImageSkiaOperations::CreateTiledImage(const ImageSkia& source,
473                                                 int src_x, int src_y,
474                                                 int dst_w, int dst_h) {
475   if (source.isNull())
476     return ImageSkia();
477 
478   return ImageSkia(new TiledImageSource(source, src_x, src_y, dst_w, dst_h),
479                    gfx::Size(dst_w, dst_h));
480 }
481 
482 // static
CreateHSLShiftedImage(const ImageSkia & image,const color_utils::HSL & hsl_shift)483 ImageSkia ImageSkiaOperations::CreateHSLShiftedImage(
484     const ImageSkia& image,
485     const color_utils::HSL& hsl_shift) {
486   if (image.isNull())
487     return ImageSkia();
488 
489   return ImageSkia(new HSLImageSource(image, hsl_shift), image.size());
490 }
491 
492 // static
CreateButtonBackground(SkColor color,const ImageSkia & image,const ImageSkia & mask)493 ImageSkia ImageSkiaOperations::CreateButtonBackground(SkColor color,
494                                                       const ImageSkia& image,
495                                                       const ImageSkia& mask) {
496   if (image.isNull() || mask.isNull())
497     return ImageSkia();
498 
499   return ImageSkia(new ButtonImageSource(color, image, mask), mask.size());
500 }
501 
502 // static
ExtractSubset(const ImageSkia & image,const Rect & subset_bounds)503 ImageSkia ImageSkiaOperations::ExtractSubset(const ImageSkia& image,
504                                              const Rect& subset_bounds) {
505   gfx::Rect clipped_bounds =
506       gfx::IntersectRects(subset_bounds, gfx::Rect(image.size()));
507   if (image.isNull() || clipped_bounds.IsEmpty()) {
508     return ImageSkia();
509   }
510 
511   return ImageSkia(new ExtractSubsetImageSource(image, clipped_bounds),
512                    clipped_bounds.size());
513 }
514 
515 // static
CreateResizedImage(const ImageSkia & source,skia::ImageOperations::ResizeMethod method,const Size & target_dip_size)516 ImageSkia ImageSkiaOperations::CreateResizedImage(
517     const ImageSkia& source,
518     skia::ImageOperations::ResizeMethod method,
519     const Size& target_dip_size) {
520   if (source.isNull())
521     return ImageSkia();
522 
523   return ImageSkia(new ResizeSource(source, method, target_dip_size),
524                    target_dip_size);
525 }
526 
527 // static
CreateImageWithDropShadow(const ImageSkia & source,const ShadowValues & shadows)528 ImageSkia ImageSkiaOperations::CreateImageWithDropShadow(
529     const ImageSkia& source,
530     const ShadowValues& shadows) {
531   if (source.isNull())
532     return ImageSkia();
533 
534   const gfx::Insets shadow_padding = -gfx::ShadowValue::GetMargin(shadows);
535   gfx::Size shadow_image_size = source.size();
536   shadow_image_size.Enlarge(shadow_padding.width(),
537                             shadow_padding.height());
538   return ImageSkia(new DropShadowSource(source, shadows), shadow_image_size);
539 }
540 
541 // static
CreateRotatedImage(const ImageSkia & source,SkBitmapOperations::RotationAmount rotation)542 ImageSkia ImageSkiaOperations::CreateRotatedImage(
543       const ImageSkia& source,
544       SkBitmapOperations::RotationAmount rotation) {
545   if (source.isNull())
546     return ImageSkia();
547 
548   return ImageSkia(new RotatedSource(source, rotation),
549       SkBitmapOperations::ROTATION_180_CW == rotation ?
550           source.size() :
551           gfx::Size(source.height(), source.width()));
552 
553 }
554 
555 }  // namespace gfx
556