senorblanco@chromium.org | 5333300 | 2013-12-12 23:28:52 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2013 The Android Open Source Project |
| 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 "SkPictureImageFilter.h" |
robertphillips | fef2860 | 2016-04-11 11:08:52 -0700 | [diff] [blame] | 9 | |
senorblanco@chromium.org | 5333300 | 2013-12-12 23:28:52 +0000 | [diff] [blame] | 10 | #include "SkCanvas.h" |
Matt Sarett | e22a6a2 | 2017-04-12 16:26:21 -0400 | [diff] [blame] | 11 | #include "SkColorSpaceXformCanvas.h" |
| 12 | #include "SkColorSpaceXformer.h" |
Cary Clark | 4dc5a45 | 2018-05-21 11:56:57 -0400 | [diff] [blame] | 13 | #include "SkFlattenablePriv.h" |
Mike Reed | 4872315 | 2017-11-10 14:35:54 -0500 | [diff] [blame] | 14 | #include "SkImageSource.h" |
Cary Clark | efd99cc | 2018-06-11 16:25:43 -0400 | [diff] [blame] | 15 | #include "SkPicturePriv.h" |
commit-bot@chromium.org | 8b0e8ac | 2014-01-30 18:58:24 +0000 | [diff] [blame] | 16 | #include "SkReadBuffer.h" |
robertphillips | fef2860 | 2016-04-11 11:08:52 -0700 | [diff] [blame] | 17 | #include "SkSpecialImage.h" |
| 18 | #include "SkSpecialSurface.h" |
commit-bot@chromium.org | 8b0e8ac | 2014-01-30 18:58:24 +0000 | [diff] [blame] | 19 | #include "SkWriteBuffer.h" |
senorblanco@chromium.org | 5333300 | 2013-12-12 23:28:52 +0000 | [diff] [blame] | 20 | #include "SkValidationUtils.h" |
| 21 | |
robertphillips | 225db44 | 2016-04-17 14:27:05 -0700 | [diff] [blame] | 22 | sk_sp<SkImageFilter> SkPictureImageFilter::Make(sk_sp<SkPicture> picture) { |
| 23 | return sk_sp<SkImageFilter>(new SkPictureImageFilter(std::move(picture))); |
| 24 | } |
| 25 | |
| 26 | sk_sp<SkImageFilter> SkPictureImageFilter::Make(sk_sp<SkPicture> picture, |
| 27 | const SkRect& cropRect) { |
Mike Reed | 4872315 | 2017-11-10 14:35:54 -0500 | [diff] [blame] | 28 | return sk_sp<SkImageFilter>(new SkPictureImageFilter(std::move(picture), cropRect, nullptr)); |
robertphillips | 225db44 | 2016-04-17 14:27:05 -0700 | [diff] [blame] | 29 | } |
| 30 | |
robertphillips | 5ff17b1 | 2016-03-28 13:13:42 -0700 | [diff] [blame] | 31 | SkPictureImageFilter::SkPictureImageFilter(sk_sp<SkPicture> picture) |
robertphillips | 372177e | 2016-03-30 07:32:28 -0700 | [diff] [blame] | 32 | : INHERITED(nullptr, 0, nullptr) |
robertphillips | 5ff17b1 | 2016-03-28 13:13:42 -0700 | [diff] [blame] | 33 | , fPicture(std::move(picture)) |
Mike Reed | 4872315 | 2017-11-10 14:35:54 -0500 | [diff] [blame] | 34 | , fCropRect(fPicture ? fPicture->cullRect() : SkRect::MakeEmpty()) { |
senorblanco@chromium.org | 5333300 | 2013-12-12 23:28:52 +0000 | [diff] [blame] | 35 | } |
| 36 | |
robertphillips | 5ff17b1 | 2016-03-28 13:13:42 -0700 | [diff] [blame] | 37 | SkPictureImageFilter::SkPictureImageFilter(sk_sp<SkPicture> picture, const SkRect& cropRect, |
Matt Sarett | e22a6a2 | 2017-04-12 16:26:21 -0400 | [diff] [blame] | 38 | sk_sp<SkColorSpace> colorSpace) |
robertphillips | 372177e | 2016-03-30 07:32:28 -0700 | [diff] [blame] | 39 | : INHERITED(nullptr, 0, nullptr) |
robertphillips | 5ff17b1 | 2016-03-28 13:13:42 -0700 | [diff] [blame] | 40 | , fPicture(std::move(picture)) |
Justin Novosad | 5234075 | 2014-12-02 14:50:56 -0500 | [diff] [blame] | 41 | , fCropRect(cropRect) |
Matt Sarett | e22a6a2 | 2017-04-12 16:26:21 -0400 | [diff] [blame] | 42 | , fColorSpace(std::move(colorSpace)) { |
senorblanco@chromium.org | 5333300 | 2013-12-12 23:28:52 +0000 | [diff] [blame] | 43 | } |
| 44 | |
Mike Reed | 4872315 | 2017-11-10 14:35:54 -0500 | [diff] [blame] | 45 | enum PictureResolution { |
| 46 | kDeviceSpace_PictureResolution, |
| 47 | kLocalSpace_PictureResolution |
| 48 | }; |
| 49 | static sk_sp<SkImageFilter> make_localspace_filter(sk_sp<SkPicture> pic, const SkRect& cropRect, |
| 50 | SkFilterQuality fq) { |
| 51 | SkISize dim = { SkScalarRoundToInt(cropRect.width()), SkScalarRoundToInt(cropRect.height()) }; |
| 52 | auto img = SkImage::MakeFromPicture(std::move(pic), dim, nullptr, nullptr, |
| 53 | SkImage::BitDepth::kU8, SkColorSpace::MakeSRGB()); |
| 54 | return SkImageSource::Make(img, cropRect, cropRect, fq); |
| 55 | } |
| 56 | |
reed | 60c9b58 | 2016-04-03 09:11:13 -0700 | [diff] [blame] | 57 | sk_sp<SkFlattenable> SkPictureImageFilter::CreateProc(SkReadBuffer& buffer) { |
reed | ca2622b | 2016-03-18 07:25:55 -0700 | [diff] [blame] | 58 | sk_sp<SkPicture> picture; |
reed | 9fa60da | 2014-08-21 07:59:51 -0700 | [diff] [blame] | 59 | SkRect cropRect; |
| 60 | |
Mike Reed | 5a0f345 | 2018-01-19 08:25:11 -0500 | [diff] [blame] | 61 | if (buffer.readBool()) { |
Cary Clark | efd99cc | 2018-06-11 16:25:43 -0400 | [diff] [blame] | 62 | picture = SkPicturePriv::MakeFromBuffer(buffer); |
reed | 9fa60da | 2014-08-21 07:59:51 -0700 | [diff] [blame] | 63 | } |
| 64 | buffer.readRect(&cropRect); |
| 65 | |
Mike Reed | 77e487d | 2017-11-09 21:50:20 +0000 | [diff] [blame] | 66 | if (buffer.isVersionLT(SkReadBuffer::kRemovePictureImageFilterLocalSpace)) { |
Mike Reed | 4872315 | 2017-11-10 14:35:54 -0500 | [diff] [blame] | 67 | PictureResolution pictureResolution = buffer.checkRange<PictureResolution>( |
| 68 | kDeviceSpace_PictureResolution, kLocalSpace_PictureResolution); |
Mike Reed | 77e487d | 2017-11-09 21:50:20 +0000 | [diff] [blame] | 69 | if (kLocalSpace_PictureResolution == pictureResolution) { |
Mike Reed | 4872315 | 2017-11-10 14:35:54 -0500 | [diff] [blame] | 70 | return make_localspace_filter(std::move(picture), cropRect, |
| 71 | buffer.checkFilterQuality()); |
Mike Reed | 77e487d | 2017-11-09 21:50:20 +0000 | [diff] [blame] | 72 | } |
Justin Novosad | 5234075 | 2014-12-02 14:50:56 -0500 | [diff] [blame] | 73 | } |
Mike Reed | 4872315 | 2017-11-10 14:35:54 -0500 | [diff] [blame] | 74 | return sk_sp<SkImageFilter>(new SkPictureImageFilter(picture, cropRect, nullptr)); |
reed | 9fa60da | 2014-08-21 07:59:51 -0700 | [diff] [blame] | 75 | } |
senorblanco@chromium.org | 5333300 | 2013-12-12 23:28:52 +0000 | [diff] [blame] | 76 | |
commit-bot@chromium.org | 8b0e8ac | 2014-01-30 18:58:24 +0000 | [diff] [blame] | 77 | void SkPictureImageFilter::flatten(SkWriteBuffer& buffer) const { |
Mike Reed | 5a0f345 | 2018-01-19 08:25:11 -0500 | [diff] [blame] | 78 | bool hasPicture = (fPicture != nullptr); |
| 79 | buffer.writeBool(hasPicture); |
| 80 | if (hasPicture) { |
Cary Clark | efd99cc | 2018-06-11 16:25:43 -0400 | [diff] [blame] | 81 | SkPicturePriv::Flatten(fPicture, buffer); |
commit-bot@chromium.org | 5e0995e | 2014-02-07 12:20:04 +0000 | [diff] [blame] | 82 | } |
commit-bot@chromium.org | cac5fd5 | 2014-03-10 10:51:58 +0000 | [diff] [blame] | 83 | buffer.writeRect(fCropRect); |
senorblanco@chromium.org | 5333300 | 2013-12-12 23:28:52 +0000 | [diff] [blame] | 84 | } |
| 85 | |
robertphillips | fef2860 | 2016-04-11 11:08:52 -0700 | [diff] [blame] | 86 | sk_sp<SkSpecialImage> SkPictureImageFilter::onFilterImage(SkSpecialImage* source, |
| 87 | const Context& ctx, |
| 88 | SkIPoint* offset) const { |
senorblanco@chromium.org | 5333300 | 2013-12-12 23:28:52 +0000 | [diff] [blame] | 89 | if (!fPicture) { |
robertphillips | fef2860 | 2016-04-11 11:08:52 -0700 | [diff] [blame] | 90 | return nullptr; |
senorblanco@chromium.org | 5333300 | 2013-12-12 23:28:52 +0000 | [diff] [blame] | 91 | } |
| 92 | |
| 93 | SkRect floatBounds; |
senorblanco@chromium.org | 4cb543d | 2014-03-14 15:44:01 +0000 | [diff] [blame] | 94 | ctx.ctm().mapRect(&floatBounds, fCropRect); |
reed | b07a94f | 2014-11-19 05:03:18 -0800 | [diff] [blame] | 95 | SkIRect bounds = floatBounds.roundOut(); |
senorblanco | 3d822c2 | 2014-07-30 14:49:31 -0700 | [diff] [blame] | 96 | if (!bounds.intersect(ctx.clipBounds())) { |
robertphillips | fef2860 | 2016-04-11 11:08:52 -0700 | [diff] [blame] | 97 | return nullptr; |
senorblanco | 3d822c2 | 2014-07-30 14:49:31 -0700 | [diff] [blame] | 98 | } |
senorblanco@chromium.org | 5333300 | 2013-12-12 23:28:52 +0000 | [diff] [blame] | 99 | |
robertphillips | fef2860 | 2016-04-11 11:08:52 -0700 | [diff] [blame] | 100 | SkASSERT(!bounds.isEmpty()); |
| 101 | |
brianosman | eed6b0e | 2016-09-23 13:04:05 -0700 | [diff] [blame] | 102 | sk_sp<SkSpecialSurface> surf(source->makeSurface(ctx.outputProperties(), bounds.size())); |
robertphillips | fef2860 | 2016-04-11 11:08:52 -0700 | [diff] [blame] | 103 | if (!surf) { |
| 104 | return nullptr; |
senorblanco@chromium.org | 5333300 | 2013-12-12 23:28:52 +0000 | [diff] [blame] | 105 | } |
| 106 | |
robertphillips | fef2860 | 2016-04-11 11:08:52 -0700 | [diff] [blame] | 107 | SkCanvas* canvas = surf->getCanvas(); |
| 108 | SkASSERT(canvas); |
robertphillips | fef2860 | 2016-04-11 11:08:52 -0700 | [diff] [blame] | 109 | canvas->clear(0x0); |
| 110 | |
Mike Reed | 4872315 | 2017-11-10 14:35:54 -0500 | [diff] [blame] | 111 | std::unique_ptr<SkCanvas> xformCanvas; |
| 112 | if (fColorSpace) { |
| 113 | // Only non-null in the case where onMakeColorSpace() was called. This instructs |
| 114 | // us to do the color space xform on playback. |
| 115 | xformCanvas = SkCreateColorSpaceXformCanvas(canvas, fColorSpace); |
| 116 | canvas = xformCanvas.get(); |
Justin Novosad | 5234075 | 2014-12-02 14:50:56 -0500 | [diff] [blame] | 117 | } |
Mike Reed | 4872315 | 2017-11-10 14:35:54 -0500 | [diff] [blame] | 118 | canvas->translate(-SkIntToScalar(bounds.fLeft), -SkIntToScalar(bounds.fTop)); |
| 119 | canvas->concat(ctx.ctm()); |
| 120 | canvas->drawPicture(fPicture); |
senorblanco@chromium.org | 5333300 | 2013-12-12 23:28:52 +0000 | [diff] [blame] | 121 | |
senorblanco@chromium.org | 6776b82 | 2014-01-03 21:48:22 +0000 | [diff] [blame] | 122 | offset->fX = bounds.fLeft; |
| 123 | offset->fY = bounds.fTop; |
robertphillips | fef2860 | 2016-04-11 11:08:52 -0700 | [diff] [blame] | 124 | return surf->makeImageSnapshot(); |
senorblanco@chromium.org | 5333300 | 2013-12-12 23:28:52 +0000 | [diff] [blame] | 125 | } |
Justin Novosad | 5234075 | 2014-12-02 14:50:56 -0500 | [diff] [blame] | 126 | |
Matt Sarett | e22a6a2 | 2017-04-12 16:26:21 -0400 | [diff] [blame] | 127 | sk_sp<SkImageFilter> SkPictureImageFilter::onMakeColorSpace(SkColorSpaceXformer* xformer) const { |
Florin Malita | a252c66 | 2017-07-10 12:27:52 -0400 | [diff] [blame] | 128 | sk_sp<SkColorSpace> dstCS = xformer->dst(); |
Florin Malita | 604f0d5 | 2017-07-13 14:29:12 -0400 | [diff] [blame] | 129 | if (SkColorSpace::Equals(dstCS.get(), fColorSpace.get())) { |
Florin Malita | a252c66 | 2017-07-10 12:27:52 -0400 | [diff] [blame] | 130 | return this->refMe(); |
| 131 | } |
| 132 | |
Mike Reed | 4872315 | 2017-11-10 14:35:54 -0500 | [diff] [blame] | 133 | return sk_sp<SkImageFilter>(new SkPictureImageFilter(fPicture, fCropRect, std::move(dstCS))); |
Justin Novosad | 5234075 | 2014-12-02 14:50:56 -0500 | [diff] [blame] | 134 | } |