blob: 5769e677a1fd294b2e6cb1561b9f3cf738badfae [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
robertphillips225db442016-04-17 14:27:05 -070018sk_sp<SkImageFilter> SkImageSource::Make(sk_sp<SkImage> image) {
19 if (!image) {
20 return nullptr;
21 }
22
23 return sk_sp<SkImageFilter>(new SkImageSource(std::move(image)));
24}
25
26sk_sp<SkImageFilter> SkImageSource::Make(sk_sp<SkImage> image,
27 const SkRect& srcRect,
28 const SkRect& dstRect,
29 SkFilterQuality filterQuality) {
30 if (!image) {
31 return nullptr;
32 }
33
34 return sk_sp<SkImageFilter>(new SkImageSource(std::move(image),
35 srcRect, dstRect,
36 filterQuality));
37}
fmalitacd56f812015-09-14 13:31:18 -070038
robertphillips549c8992016-04-01 09:28:51 -070039SkImageSource::SkImageSource(sk_sp<SkImage> image)
robertphillips6b134732016-04-15 09:58:37 -070040 : INHERITED(nullptr, 0, nullptr)
robertphillips549c8992016-04-01 09:28:51 -070041 , fImage(std::move(image))
42 , fSrcRect(SkRect::MakeIWH(fImage->width(), fImage->height()))
fmalitacd56f812015-09-14 13:31:18 -070043 , fDstRect(fSrcRect)
robertphillips549c8992016-04-01 09:28:51 -070044 , fFilterQuality(kHigh_SkFilterQuality) {
45}
fmalitacd56f812015-09-14 13:31:18 -070046
robertphillips549c8992016-04-01 09:28:51 -070047SkImageSource::SkImageSource(sk_sp<SkImage> image,
fmalitacd56f812015-09-14 13:31:18 -070048 const SkRect& srcRect,
49 const SkRect& dstRect,
50 SkFilterQuality filterQuality)
robertphillips6b134732016-04-15 09:58:37 -070051 : INHERITED(nullptr, 0, nullptr)
robertphillips549c8992016-04-01 09:28:51 -070052 , fImage(std::move(image))
fmalitacd56f812015-09-14 13:31:18 -070053 , fSrcRect(srcRect)
54 , fDstRect(dstRect)
robertphillips549c8992016-04-01 09:28:51 -070055 , fFilterQuality(filterQuality) {
56}
fmalitacd56f812015-09-14 13:31:18 -070057
reed60c9b582016-04-03 09:11:13 -070058sk_sp<SkFlattenable> SkImageSource::CreateProc(SkReadBuffer& buffer) {
fmalitacd56f812015-09-14 13:31:18 -070059 SkFilterQuality filterQuality = (SkFilterQuality)buffer.readInt();
60
61 SkRect src, dst;
62 buffer.readRect(&src);
63 buffer.readRect(&dst);
64
robertphillips549c8992016-04-01 09:28:51 -070065 sk_sp<SkImage> image(buffer.readImage());
fmalitacd56f812015-09-14 13:31:18 -070066 if (!image) {
67 return nullptr;
68 }
69
reed60c9b582016-04-03 09:11:13 -070070 return SkImageSource::Make(std::move(image), src, dst, filterQuality);
fmalitacd56f812015-09-14 13:31:18 -070071}
72
73void SkImageSource::flatten(SkWriteBuffer& buffer) const {
74 buffer.writeInt(fFilterQuality);
75 buffer.writeRect(fSrcRect);
76 buffer.writeRect(fDstRect);
robertphillips37bd7c32016-03-17 14:31:39 -070077 buffer.writeImage(fImage.get());
fmalitacd56f812015-09-14 13:31:18 -070078}
79
robertphillips2302de92016-03-24 07:26:32 -070080sk_sp<SkSpecialImage> SkImageSource::onFilterImage(SkSpecialImage* source, const Context& ctx,
81 SkIPoint* offset) const {
fmalitacd56f812015-09-14 13:31:18 -070082 SkRect dstRect;
83 ctx.ctm().mapRect(&dstRect, fDstRect);
robertphillips6ac97b72016-03-09 05:17:10 -080084
fmalitacd56f812015-09-14 13:31:18 -070085 SkRect bounds = SkRect::MakeIWH(fImage->width(), fImage->height());
Robert Phillips8bb3b212017-02-06 12:32:55 -050086 if (fSrcRect == bounds) {
87 int iLeft = dstRect.fLeft;
88 int iTop = dstRect.fTop;
89 // TODO: this seems to be a very noise-prone way to determine this (esp. the floating-point
90 // widths & heights).
91 if (dstRect.width() == bounds.width() && dstRect.height() == bounds.height() &&
92 iLeft == dstRect.fLeft && iTop == dstRect.fTop) {
93 // The dest is just an un-scaled integer translation of the entire image; return it
94 offset->fX = iLeft;
95 offset->fY = iTop;
96 return SkSpecialImage::MakeFromImage(SkIRect::MakeWH(fImage->width(), fImage->height()),
97 fImage, ctx.outputProperties().colorSpace(),
98 &source->props());
99 }
fmalitacd56f812015-09-14 13:31:18 -0700100 }
101
102 const SkIRect dstIRect = dstRect.roundOut();
robertphillips6ac97b72016-03-09 05:17:10 -0800103
brianosmaneed6b0e2016-09-23 13:04:05 -0700104 sk_sp<SkSpecialSurface> surf(source->makeSurface(ctx.outputProperties(), dstIRect.size()));
robertphillips6ac97b72016-03-09 05:17:10 -0800105 if (!surf) {
106 return nullptr;
fmalitacd56f812015-09-14 13:31:18 -0700107 }
108
robertphillips6ac97b72016-03-09 05:17:10 -0800109 SkCanvas* canvas = surf->getCanvas();
110 SkASSERT(canvas);
111
112 // TODO: it seems like this clear shouldn't be necessary (see skbug.com/5075)
113 canvas->clear(0x0);
114
fmalitacd56f812015-09-14 13:31:18 -0700115 SkPaint paint;
116
robertphillips6ac97b72016-03-09 05:17:10 -0800117 // Subtract off the integer component of the translation (will be applied in offset, below).
fmalitacd56f812015-09-14 13:31:18 -0700118 dstRect.offset(-SkIntToScalar(dstIRect.fLeft), -SkIntToScalar(dstIRect.fTop));
reed374772b2016-10-05 17:33:02 -0700119 paint.setBlendMode(SkBlendMode::kSrc);
fmalitacd56f812015-09-14 13:31:18 -0700120 // FIXME: this probably shouldn't be necessary, but drawImageRect asserts
121 // None filtering when it's translate-only
122 paint.setFilterQuality(
123 fSrcRect.width() == dstRect.width() && fSrcRect.height() == dstRect.height() ?
124 kNone_SkFilterQuality : fFilterQuality);
robertphillips37bd7c32016-03-17 14:31:39 -0700125 canvas->drawImageRect(fImage.get(), fSrcRect, dstRect, &paint,
126 SkCanvas::kStrict_SrcRectConstraint);
fmalitacd56f812015-09-14 13:31:18 -0700127
fmalitacd56f812015-09-14 13:31:18 -0700128 offset->fX = dstIRect.fLeft;
129 offset->fY = dstIRect.fTop;
robertphillips2302de92016-03-24 07:26:32 -0700130 return surf->makeImageSnapshot();
fmalitacd56f812015-09-14 13:31:18 -0700131}
132
senorblancoe5e79842016-03-21 14:51:59 -0700133SkRect SkImageSource::computeFastBounds(const SkRect& src) const {
134 return fDstRect;
fmalitacd56f812015-09-14 13:31:18 -0700135}
136
137#ifndef SK_IGNORE_TO_STRING
138void SkImageSource::toString(SkString* str) const {
139 str->appendf("SkImageSource: (");
140 str->appendf("src: (%f,%f,%f,%f) dst: (%f,%f,%f,%f) ",
141 fSrcRect.fLeft, fSrcRect.fTop, fSrcRect.fRight, fSrcRect.fBottom,
142 fDstRect.fLeft, fDstRect.fTop, fDstRect.fRight, fDstRect.fBottom);
143 str->appendf("image: (%d,%d)",
144 fImage->width(), fImage->height());
145 str->append(")");
146}
147#endif