blob: c0a8deebf95f7b505ee274383c3c0fb1c556b924 [file] [log] [blame]
fmalitacd56f812015-09-14 13:31:18 -07001/*
2 * Copyright 2015 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
8#include "SkImageSource.h"
9
10#include "SkCanvas.h"
fmalitacd56f812015-09-14 13:31:18 -070011#include "SkImage.h"
12#include "SkReadBuffer.h"
robertphillips6ac97b72016-03-09 05:17:10 -080013#include "SkSpecialImage.h"
14#include "SkSpecialSurface.h"
fmalitacd56f812015-09-14 13:31:18 -070015#include "SkWriteBuffer.h"
16#include "SkString.h"
17
robertphillips37bd7c32016-03-17 14:31:39 -070018SkImageFilter* SkImageSource::Create(SkImage* image) {
mdempsky106b1242015-09-22 06:10:35 -070019 return image ? new SkImageSource(image) : nullptr;
fmalitacd56f812015-09-14 13:31:18 -070020}
21
robertphillips37bd7c32016-03-17 14:31:39 -070022SkImageFilter* SkImageSource::Create(SkImage* image,
fmalitacd56f812015-09-14 13:31:18 -070023 const SkRect& srcRect,
24 const SkRect& dstRect,
25 SkFilterQuality filterQuality) {
mdempsky106b1242015-09-22 06:10:35 -070026 return image ? new SkImageSource(image, srcRect, dstRect, filterQuality) : nullptr;
fmalitacd56f812015-09-14 13:31:18 -070027}
28
robertphillips37bd7c32016-03-17 14:31:39 -070029SkImageSource::SkImageSource(SkImage* image)
fmalitacd56f812015-09-14 13:31:18 -070030 : INHERITED(0, nullptr)
31 , fImage(SkRef(image))
32 , fSrcRect(SkRect::MakeIWH(image->width(), image->height()))
33 , fDstRect(fSrcRect)
34 , fFilterQuality(kHigh_SkFilterQuality) { }
35
robertphillips37bd7c32016-03-17 14:31:39 -070036SkImageSource::SkImageSource(SkImage* image,
fmalitacd56f812015-09-14 13:31:18 -070037 const SkRect& srcRect,
38 const SkRect& dstRect,
39 SkFilterQuality filterQuality)
40 : INHERITED(0, nullptr)
41 , fImage(SkRef(image))
42 , fSrcRect(srcRect)
43 , fDstRect(dstRect)
44 , fFilterQuality(filterQuality) { }
45
46SkFlattenable* SkImageSource::CreateProc(SkReadBuffer& buffer) {
47 SkFilterQuality filterQuality = (SkFilterQuality)buffer.readInt();
48
49 SkRect src, dst;
50 buffer.readRect(&src);
51 buffer.readRect(&dst);
52
53 SkAutoTUnref<SkImage> image(buffer.readImage());
54 if (!image) {
55 return nullptr;
56 }
57
58 return SkImageSource::Create(image, src, dst, filterQuality);
59}
60
61void SkImageSource::flatten(SkWriteBuffer& buffer) const {
62 buffer.writeInt(fFilterQuality);
63 buffer.writeRect(fSrcRect);
64 buffer.writeRect(fDstRect);
robertphillips37bd7c32016-03-17 14:31:39 -070065 buffer.writeImage(fImage.get());
fmalitacd56f812015-09-14 13:31:18 -070066}
67
robertphillips6ac97b72016-03-09 05:17:10 -080068SkSpecialImage* SkImageSource::onFilterImage(SkSpecialImage* source, const Context& ctx,
69 SkIPoint* offset) const {
fmalitacd56f812015-09-14 13:31:18 -070070 SkRect dstRect;
71 ctx.ctm().mapRect(&dstRect, fDstRect);
robertphillips6ac97b72016-03-09 05:17:10 -080072
fmalitacd56f812015-09-14 13:31:18 -070073 SkRect bounds = SkRect::MakeIWH(fImage->width(), fImage->height());
74 if (fSrcRect == bounds && dstRect == bounds) {
75 // No regions cropped out or resized; return entire image.
76 offset->fX = offset->fY = 0;
robertphillips37bd7c32016-03-17 14:31:39 -070077 return SkSpecialImage::MakeFromImage(source->internal_getProxy(),
78 SkIRect::MakeWH(fImage->width(), fImage->height()),
79 fImage).release();
fmalitacd56f812015-09-14 13:31:18 -070080 }
81
82 const SkIRect dstIRect = dstRect.roundOut();
robertphillips6ac97b72016-03-09 05:17:10 -080083
84 const SkImageInfo info = SkImageInfo::MakeN32(dstIRect.width(), dstIRect.height(),
85 kPremul_SkAlphaType);
86
robertphillips37bd7c32016-03-17 14:31:39 -070087 sk_sp<SkSpecialSurface> surf(source->makeSurface(info));
robertphillips6ac97b72016-03-09 05:17:10 -080088 if (!surf) {
89 return nullptr;
fmalitacd56f812015-09-14 13:31:18 -070090 }
91
robertphillips6ac97b72016-03-09 05:17:10 -080092 SkCanvas* canvas = surf->getCanvas();
93 SkASSERT(canvas);
94
95 // TODO: it seems like this clear shouldn't be necessary (see skbug.com/5075)
96 canvas->clear(0x0);
97
fmalitacd56f812015-09-14 13:31:18 -070098 SkPaint paint;
99
robertphillips6ac97b72016-03-09 05:17:10 -0800100 // Subtract off the integer component of the translation (will be applied in offset, below).
fmalitacd56f812015-09-14 13:31:18 -0700101 dstRect.offset(-SkIntToScalar(dstIRect.fLeft), -SkIntToScalar(dstIRect.fTop));
102 paint.setXfermodeMode(SkXfermode::kSrc_Mode);
103 // FIXME: this probably shouldn't be necessary, but drawImageRect asserts
104 // None filtering when it's translate-only
105 paint.setFilterQuality(
106 fSrcRect.width() == dstRect.width() && fSrcRect.height() == dstRect.height() ?
107 kNone_SkFilterQuality : fFilterQuality);
robertphillips37bd7c32016-03-17 14:31:39 -0700108 canvas->drawImageRect(fImage.get(), fSrcRect, dstRect, &paint,
109 SkCanvas::kStrict_SrcRectConstraint);
fmalitacd56f812015-09-14 13:31:18 -0700110
fmalitacd56f812015-09-14 13:31:18 -0700111 offset->fX = dstIRect.fLeft;
112 offset->fY = dstIRect.fTop;
robertphillips37bd7c32016-03-17 14:31:39 -0700113 return surf->makeImageSnapshot().release();
fmalitacd56f812015-09-14 13:31:18 -0700114}
115
116void SkImageSource::computeFastBounds(const SkRect& src, SkRect* dst) const {
117 *dst = fDstRect;
118}
119
120#ifndef SK_IGNORE_TO_STRING
121void SkImageSource::toString(SkString* str) const {
122 str->appendf("SkImageSource: (");
123 str->appendf("src: (%f,%f,%f,%f) dst: (%f,%f,%f,%f) ",
124 fSrcRect.fLeft, fSrcRect.fTop, fSrcRect.fRight, fSrcRect.fBottom,
125 fDstRect.fLeft, fDstRect.fTop, fDstRect.fRight, fDstRect.fBottom);
126 str->appendf("image: (%d,%d)",
127 fImage->width(), fImage->height());
128 str->append(")");
129}
130#endif