1 /*
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"
11 #include "SkColorSpaceXformer.h"
12 #include "SkImage.h"
13 #include "SkReadBuffer.h"
14 #include "SkSpecialImage.h"
15 #include "SkSpecialSurface.h"
16 #include "SkWriteBuffer.h"
17 #include "SkString.h"
18
Make(sk_sp<SkImage> image)19 sk_sp<SkImageFilter> SkImageSource::Make(sk_sp<SkImage> image) {
20 if (!image) {
21 return nullptr;
22 }
23
24 return sk_sp<SkImageFilter>(new SkImageSource(std::move(image)));
25 }
26
Make(sk_sp<SkImage> image,const SkRect & srcRect,const SkRect & dstRect,SkFilterQuality filterQuality)27 sk_sp<SkImageFilter> SkImageSource::Make(sk_sp<SkImage> image,
28 const SkRect& srcRect,
29 const SkRect& dstRect,
30 SkFilterQuality filterQuality) {
31 if (!image || srcRect.width() <= 0.0f || srcRect.height() <= 0.0f) {
32 return nullptr;
33 }
34
35 return sk_sp<SkImageFilter>(new SkImageSource(std::move(image),
36 srcRect, dstRect,
37 filterQuality));
38 }
39
SkImageSource(sk_sp<SkImage> image)40 SkImageSource::SkImageSource(sk_sp<SkImage> image)
41 : INHERITED(nullptr, 0, nullptr)
42 , fImage(std::move(image))
43 , fSrcRect(SkRect::MakeIWH(fImage->width(), fImage->height()))
44 , fDstRect(fSrcRect)
45 , fFilterQuality(kHigh_SkFilterQuality) {
46 }
47
SkImageSource(sk_sp<SkImage> image,const SkRect & srcRect,const SkRect & dstRect,SkFilterQuality filterQuality)48 SkImageSource::SkImageSource(sk_sp<SkImage> image,
49 const SkRect& srcRect,
50 const SkRect& dstRect,
51 SkFilterQuality filterQuality)
52 : INHERITED(nullptr, 0, nullptr)
53 , fImage(std::move(image))
54 , fSrcRect(srcRect)
55 , fDstRect(dstRect)
56 , fFilterQuality(filterQuality) {
57 }
58
CreateProc(SkReadBuffer & buffer)59 sk_sp<SkFlattenable> SkImageSource::CreateProc(SkReadBuffer& buffer) {
60 SkFilterQuality filterQuality = (SkFilterQuality)buffer.readInt();
61
62 SkRect src, dst;
63 buffer.readRect(&src);
64 buffer.readRect(&dst);
65
66 sk_sp<SkImage> image(buffer.readImage());
67 if (!image) {
68 return nullptr;
69 }
70
71 return SkImageSource::Make(std::move(image), src, dst, filterQuality);
72 }
73
flatten(SkWriteBuffer & buffer) const74 void SkImageSource::flatten(SkWriteBuffer& buffer) const {
75 buffer.writeInt(fFilterQuality);
76 buffer.writeRect(fSrcRect);
77 buffer.writeRect(fDstRect);
78 buffer.writeImage(fImage.get());
79 }
80
onFilterImage(SkSpecialImage * source,const Context & ctx,SkIPoint * offset) const81 sk_sp<SkSpecialImage> SkImageSource::onFilterImage(SkSpecialImage* source, const Context& ctx,
82 SkIPoint* offset) const {
83 SkRect dstRect;
84 ctx.ctm().mapRect(&dstRect, fDstRect);
85
86 SkRect bounds = SkRect::MakeIWH(fImage->width(), fImage->height());
87 if (fSrcRect == bounds) {
88 int iLeft = dstRect.fLeft;
89 int iTop = dstRect.fTop;
90 // TODO: this seems to be a very noise-prone way to determine this (esp. the floating-point
91 // widths & heights).
92 if (dstRect.width() == bounds.width() && dstRect.height() == bounds.height() &&
93 iLeft == dstRect.fLeft && iTop == dstRect.fTop) {
94 // The dest is just an un-scaled integer translation of the entire image; return it
95 offset->fX = iLeft;
96 offset->fY = iTop;
97
98 return SkSpecialImage::MakeFromImage(source->getContext(),
99 SkIRect::MakeWH(fImage->width(), fImage->height()),
100 fImage, &source->props());
101 }
102 }
103
104 const SkIRect dstIRect = dstRect.roundOut();
105
106 sk_sp<SkSpecialSurface> surf(source->makeSurface(ctx.outputProperties(), dstIRect.size()));
107 if (!surf) {
108 return nullptr;
109 }
110
111 SkCanvas* canvas = surf->getCanvas();
112 SkASSERT(canvas);
113
114 // TODO: it seems like this clear shouldn't be necessary (see skbug.com/5075)
115 canvas->clear(0x0);
116
117 SkPaint paint;
118
119 // Subtract off the integer component of the translation (will be applied in offset, below).
120 dstRect.offset(-SkIntToScalar(dstIRect.fLeft), -SkIntToScalar(dstIRect.fTop));
121 paint.setBlendMode(SkBlendMode::kSrc);
122 // FIXME: this probably shouldn't be necessary, but drawImageRect asserts
123 // None filtering when it's translate-only
124 paint.setFilterQuality(
125 fSrcRect.width() == dstRect.width() && fSrcRect.height() == dstRect.height() ?
126 kNone_SkFilterQuality : fFilterQuality);
127 canvas->drawImageRect(fImage.get(), fSrcRect, dstRect, &paint,
128 SkCanvas::kStrict_SrcRectConstraint);
129
130 offset->fX = dstIRect.fLeft;
131 offset->fY = dstIRect.fTop;
132 return surf->makeImageSnapshot();
133 }
134
onMakeColorSpace(SkColorSpaceXformer * xformer) const135 sk_sp<SkImageFilter> SkImageSource::onMakeColorSpace(SkColorSpaceXformer* xformer) const {
136 SkASSERT(0 == this->countInputs());
137
138 auto image = xformer->apply(fImage.get());
139 if (image != fImage) {
140 return SkImageSource::Make(image, fSrcRect, fDstRect, fFilterQuality);
141 }
142 return this->refMe();
143 }
144
computeFastBounds(const SkRect & src) const145 SkRect SkImageSource::computeFastBounds(const SkRect& src) const {
146 return fDstRect;
147 }
148
onFilterNodeBounds(const SkIRect & src,const SkMatrix & ctm,MapDirection direction,const SkIRect * inputRect) const149 SkIRect SkImageSource::onFilterNodeBounds(const SkIRect& src, const SkMatrix& ctm,
150 MapDirection direction, const SkIRect* inputRect) const {
151 if (kReverse_MapDirection == direction) {
152 return SkImageFilter::onFilterNodeBounds(src, ctm, direction, inputRect);
153 }
154
155 SkRect dstRect = fDstRect;
156 ctm.mapRect(&dstRect);
157 return dstRect.roundOut();
158 }
159
160