2 * Copyright 2015 Google Inc.
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
8 #include "SkImageSource.h"
12 #include "SkReadBuffer.h"
13 #include "SkSpecialImage.h"
14 #include "SkSpecialSurface.h"
15 #include "SkWriteBuffer.h"
18 sk_sp<SkImageFilter> SkImageSource::Make(sk_sp<SkImage> image) {
23 return sk_sp<SkImageFilter>(new SkImageSource(std::move(image)));
26 sk_sp<SkImageFilter> SkImageSource::Make(sk_sp<SkImage> image,
27 const SkRect& srcRect,
28 const SkRect& dstRect,
29 SkFilterQuality filterQuality) {
34 return sk_sp<SkImageFilter>(new SkImageSource(std::move(image),
39 SkImageSource::SkImageSource(sk_sp<SkImage> image)
40 : INHERITED(nullptr, 0, nullptr)
41 , fImage(std::move(image))
42 , fSrcRect(SkRect::MakeIWH(fImage->width(), fImage->height()))
44 , fFilterQuality(kHigh_SkFilterQuality) {
47 SkImageSource::SkImageSource(sk_sp<SkImage> image,
48 const SkRect& srcRect,
49 const SkRect& dstRect,
50 SkFilterQuality filterQuality)
51 : INHERITED(nullptr, 0, nullptr)
52 , fImage(std::move(image))
55 , fFilterQuality(filterQuality) {
58 sk_sp<SkFlattenable> SkImageSource::CreateProc(SkReadBuffer& buffer) {
59 SkFilterQuality filterQuality = (SkFilterQuality)buffer.readInt();
62 buffer.readRect(&src);
63 buffer.readRect(&dst);
65 sk_sp<SkImage> image(buffer.readImage());
70 return SkImageSource::Make(std::move(image), src, dst, filterQuality);
73 void SkImageSource::flatten(SkWriteBuffer& buffer) const {
74 buffer.writeInt(fFilterQuality);
75 buffer.writeRect(fSrcRect);
76 buffer.writeRect(fDstRect);
77 buffer.writeImage(fImage.get());
80 sk_sp<SkSpecialImage> SkImageSource::onFilterImage(SkSpecialImage* source, const Context& ctx,
81 SkIPoint* offset) const {
83 ctx.ctm().mapRect(&dstRect, fDstRect);
85 SkRect bounds = SkRect::MakeIWH(fImage->width(), fImage->height());
86 if (fSrcRect == bounds && dstRect == bounds) {
87 // No regions cropped out or resized; return entire image.
88 offset->fX = offset->fY = 0;
89 return SkSpecialImage::MakeFromImage(SkIRect::MakeWH(fImage->width(), fImage->height()),
90 fImage, ctx.outputProperties().colorSpace(),
94 const SkIRect dstIRect = dstRect.roundOut();
96 sk_sp<SkSpecialSurface> surf(source->makeSurface(ctx.outputProperties(), dstIRect.size()));
101 SkCanvas* canvas = surf->getCanvas();
104 // TODO: it seems like this clear shouldn't be necessary (see skbug.com/5075)
109 // Subtract off the integer component of the translation (will be applied in offset, below).
110 dstRect.offset(-SkIntToScalar(dstIRect.fLeft), -SkIntToScalar(dstIRect.fTop));
111 paint.setBlendMode(SkBlendMode::kSrc);
112 // FIXME: this probably shouldn't be necessary, but drawImageRect asserts
113 // None filtering when it's translate-only
114 paint.setFilterQuality(
115 fSrcRect.width() == dstRect.width() && fSrcRect.height() == dstRect.height() ?
116 kNone_SkFilterQuality : fFilterQuality);
117 canvas->drawImageRect(fImage.get(), fSrcRect, dstRect, &paint,
118 SkCanvas::kStrict_SrcRectConstraint);
120 offset->fX = dstIRect.fLeft;
121 offset->fY = dstIRect.fTop;
122 return surf->makeImageSnapshot();
125 SkRect SkImageSource::computeFastBounds(const SkRect& src) const {
129 #ifndef SK_IGNORE_TO_STRING
130 void SkImageSource::toString(SkString* str) const {
131 str->appendf("SkImageSource: (");
132 str->appendf("src: (%f,%f,%f,%f) dst: (%f,%f,%f,%f) ",
133 fSrcRect.fLeft, fSrcRect.fTop, fSrcRect.fRight, fSrcRect.fBottom,
134 fDstRect.fLeft, fDstRect.fTop, fDstRect.fRight, fDstRect.fBottom);
135 str->appendf("image: (%d,%d)",
136 fImage->width(), fImage->height());