blob: 9a15ff2de46302d238bda8248931feb8f2bebc53 [file] [log] [blame]
Derek Sollenberger8872b382014-06-23 14:13:53 -04001/*
2 * Copyright (C) 2014 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Derek Sollenbergerc1908132016-07-15 10:28:16 -040017#include "SkiaCanvas.h"
Derek Sollenberger8872b382014-06-23 14:13:53 -040018
Derek Sollenbergerc1908132016-07-15 10:28:16 -040019#include "CanvasProperty.h"
Matt Sarett7de73852016-10-25 18:36:39 -040020#include "NinePatchUtils.h"
Derek Sollenbergerc1908132016-07-15 10:28:16 -040021#include "VectorDrawable.h"
sergeyvaed7f582016-10-14 16:30:21 -070022#include "hwui/Bitmap.h"
Yuqian Liafc221492016-07-18 13:07:42 -040023#include "hwui/MinikinUtils.h"
Ben Wagner0ed10be2018-06-28 17:08:16 -040024#include "hwui/PaintFilter.h"
Stan Iliev021693b2016-10-17 16:26:15 -040025#include "pipeline/skia/AnimatedDrawables.h"
Derek Sollenbergerc1908132016-07-15 10:28:16 -040026
Leon Scroggins III671cce22018-01-14 16:52:17 -050027#include <SkAnimatedImage.h>
Matt Sarettd0814db2017-04-13 09:33:18 -040028#include <SkCanvasStateUtils.h>
Derek Sollenbergerfb0c8fc2017-07-26 13:53:27 -040029#include <SkColorFilter.h>
Matt Sarettea70d222017-03-29 16:25:10 -040030#include <SkColorSpaceXformCanvas.h>
John Reck849911a2015-01-20 07:51:14 -080031#include <SkDeque.h>
John Reck1bcacfd2017-11-03 10:12:19 -070032#include <SkDrawable.h>
Mike Reed1c79eab2018-11-21 11:01:57 -050033#include <SkFont.h>
John Reck849911a2015-01-20 07:51:14 -080034#include <SkGraphics.h>
Derek Sollenberger6f485562015-07-30 10:00:39 -040035#include <SkImage.h>
Matt Sarett62feb3a2016-09-20 10:34:11 -040036#include <SkImagePriv.h>
Leon Scroggins III671cce22018-01-14 16:52:17 -050037#include <SkPicture.h>
Yuqian Liafc221492016-07-18 13:07:42 -040038#include <SkRSXform.h>
John Reck849911a2015-01-20 07:51:14 -080039#include <SkShader.h>
John Reck849911a2015-01-20 07:51:14 -080040#include <SkTemplates.h>
Stan Ilievf50806a2016-10-24 10:40:39 -040041#include <SkTextBlob.h>
Derek Sollenberger8872b382014-06-23 14:13:53 -040042
Ben Wagner60126ef2015-08-07 12:13:48 -040043#include <memory>
Ben Wagner0ed10be2018-06-28 17:08:16 -040044#include <optional>
45#include <utility>
Ben Wagner60126ef2015-08-07 12:13:48 -040046
Derek Sollenberger8872b382014-06-23 14:13:53 -040047namespace android {
48
Stan Ilievf50806a2016-10-24 10:40:39 -040049using uirenderer::PaintUtils;
50
John Reckc1b33d62015-04-22 09:04:45 -070051Canvas* Canvas::create_canvas(const SkBitmap& bitmap) {
Derek Sollenberger8872b382014-06-23 14:13:53 -040052 return new SkiaCanvas(bitmap);
53}
54
Derek Sollenbergerfa3e3402017-08-04 08:35:10 -040055Canvas* Canvas::create_canvas(SkCanvas* skiaCanvas) {
56 return new SkiaCanvas(skiaCanvas);
Derek Sollenberger8872b382014-06-23 14:13:53 -040057}
58
Stan Ilievf50806a2016-10-24 10:40:39 -040059SkiaCanvas::SkiaCanvas() {}
60
Derek Sollenbergerfa3e3402017-08-04 08:35:10 -040061SkiaCanvas::SkiaCanvas(SkCanvas* canvas) : mCanvas(canvas) {}
Stan Ilievf50806a2016-10-24 10:40:39 -040062
John Reckc1b33d62015-04-22 09:04:45 -070063SkiaCanvas::SkiaCanvas(const SkBitmap& bitmap) {
Romain Guy82426562017-04-04 19:38:50 -070064 sk_sp<SkColorSpace> cs = bitmap.refColorSpace();
Matt Sarettca9b7032017-04-13 12:18:47 -040065 mCanvasOwned =
66 std::unique_ptr<SkCanvas>(new SkCanvas(bitmap, SkCanvas::ColorBehavior::kLegacy));
Derek Sollenbergerfa3e3402017-08-04 08:35:10 -040067 if (cs.get() == nullptr || cs->isSRGB()) {
John Reck1072fff2018-04-12 15:20:09 -070068 mCanvas = mCanvasOwned.get();
Derek Sollenbergerfa3e3402017-08-04 08:35:10 -040069 } else {
70 /** The wrapper is needed if we are drawing into a non-sRGB destination, since
71 * we need to transform all colors (not just bitmaps via filters) into the
72 * destination's colorspace.
73 */
74 mCanvasWrapper = SkCreateColorSpaceXformCanvas(mCanvasOwned.get(), std::move(cs));
75 mCanvas = mCanvasWrapper.get();
76 }
Derek Sollenberger8872b382014-06-23 14:13:53 -040077}
78
Stan Iliev021693b2016-10-17 16:26:15 -040079SkiaCanvas::~SkiaCanvas() {}
80
Derek Sollenbergerc1908132016-07-15 10:28:16 -040081void SkiaCanvas::reset(SkCanvas* skiaCanvas) {
Mike Reed6acfe162016-11-18 17:21:09 -050082 if (mCanvas != skiaCanvas) {
83 mCanvas = skiaCanvas;
84 mCanvasOwned.reset();
Derek Sollenbergerfa3e3402017-08-04 08:35:10 -040085 mCanvasWrapper.reset();
Mike Reed6acfe162016-11-18 17:21:09 -050086 }
Derek Sollenbergerc1908132016-07-15 10:28:16 -040087 mSaveStack.reset(nullptr);
Derek Sollenbergerc1908132016-07-15 10:28:16 -040088}
89
Derek Sollenberger8872b382014-06-23 14:13:53 -040090// ----------------------------------------------------------------------------
91// Canvas state operations: Replace Bitmap
92// ----------------------------------------------------------------------------
93
John Reckc1b33d62015-04-22 09:04:45 -070094void SkiaCanvas::setBitmap(const SkBitmap& bitmap) {
Romain Guy82426562017-04-04 19:38:50 -070095 sk_sp<SkColorSpace> cs = bitmap.refColorSpace();
Matt Sarettca9b7032017-04-13 12:18:47 -040096 std::unique_ptr<SkCanvas> newCanvas =
97 std::unique_ptr<SkCanvas>(new SkCanvas(bitmap, SkCanvas::ColorBehavior::kLegacy));
Derek Sollenbergerfa3e3402017-08-04 08:35:10 -040098 std::unique_ptr<SkCanvas> newCanvasWrapper;
99 if (cs.get() != nullptr && !cs->isSRGB()) {
100 newCanvasWrapper = SkCreateColorSpaceXformCanvas(newCanvas.get(), std::move(cs));
Derek Sollenbergerfa3e3402017-08-04 08:35:10 -0400101 }
Tony Mantler4f641d12017-03-14 22:36:14 +0000102
Tony Mantler4f641d12017-03-14 22:36:14 +0000103 // deletes the previously owned canvas (if any)
Matt Sarettea70d222017-03-29 16:25:10 -0400104 mCanvasOwned = std::move(newCanvas);
105 mCanvasWrapper = std::move(newCanvasWrapper);
Derek Sollenbergerfa3e3402017-08-04 08:35:10 -0400106 mCanvas = mCanvasWrapper ? mCanvasWrapper.get() : mCanvasOwned.get();
Tony Mantler4f641d12017-03-14 22:36:14 +0000107
Derek Sollenberger8872b382014-06-23 14:13:53 -0400108 // clean up the old save stack
Stan Ilievf50806a2016-10-24 10:40:39 -0400109 mSaveStack.reset(nullptr);
Derek Sollenberger8872b382014-06-23 14:13:53 -0400110}
111
112// ----------------------------------------------------------------------------
113// Canvas state operations
114// ----------------------------------------------------------------------------
115
116bool SkiaCanvas::isOpaque() {
Leon Scroggins IIIf35b9892015-07-31 10:38:40 -0400117 return mCanvas->imageInfo().isOpaque();
Derek Sollenberger8872b382014-06-23 14:13:53 -0400118}
119
120int SkiaCanvas::width() {
Leon Scroggins IIIf35b9892015-07-31 10:38:40 -0400121 return mCanvas->imageInfo().width();
Derek Sollenberger8872b382014-06-23 14:13:53 -0400122}
123
124int SkiaCanvas::height() {
Leon Scroggins IIIf35b9892015-07-31 10:38:40 -0400125 return mCanvas->imageInfo().height();
Derek Sollenberger8872b382014-06-23 14:13:53 -0400126}
127
128// ----------------------------------------------------------------------------
129// Canvas state operations: Save (layer)
130// ----------------------------------------------------------------------------
131
132int SkiaCanvas::getSaveCount() const {
133 return mCanvas->getSaveCount();
134}
135
Florin Malitaeecff562015-12-21 10:43:01 -0500136int SkiaCanvas::save(SaveFlags::Flags flags) {
Derek Sollenberger8872b382014-06-23 14:13:53 -0400137 int count = mCanvas->save();
138 recordPartialSave(flags);
139 return count;
140}
141
Florin Malita5e271402015-11-04 14:36:02 -0500142// The SkiaCanvas::restore operation layers on the capability to preserve
143// either (or both) the matrix and/or clip state after a SkCanvas::restore
144// operation. It does this by explicitly saving off the clip & matrix state
145// when requested and playing it back after the SkCanvas::restore.
Derek Sollenberger8872b382014-06-23 14:13:53 -0400146void SkiaCanvas::restore() {
Stan Ilievf50806a2016-10-24 10:40:39 -0400147 const auto* rec = this->currentSaveRec();
148 if (!rec) {
Derek Sollenberger8872b382014-06-23 14:13:53 -0400149 // Fast path - no record for this frame.
150 mCanvas->restore();
151 return;
152 }
153
Florin Malitaeecff562015-12-21 10:43:01 -0500154 bool preserveMatrix = !(rec->saveFlags & SaveFlags::Matrix);
John Reck1bcacfd2017-11-03 10:12:19 -0700155 bool preserveClip = !(rec->saveFlags & SaveFlags::Clip);
Derek Sollenberger8872b382014-06-23 14:13:53 -0400156
157 SkMatrix savedMatrix;
158 if (preserveMatrix) {
159 savedMatrix = mCanvas->getTotalMatrix();
160 }
161
Stan Ilievf50806a2016-10-24 10:40:39 -0400162 const size_t clipIndex = rec->clipIndex;
Derek Sollenberger8872b382014-06-23 14:13:53 -0400163
164 mCanvas->restore();
Stan Ilievf50806a2016-10-24 10:40:39 -0400165 mSaveStack->pop_back();
Derek Sollenberger8872b382014-06-23 14:13:53 -0400166
167 if (preserveMatrix) {
168 mCanvas->setMatrix(savedMatrix);
169 }
170
Stan Ilievf50806a2016-10-24 10:40:39 -0400171 if (preserveClip) {
172 this->applyPersistentClips(clipIndex);
Derek Sollenberger8872b382014-06-23 14:13:53 -0400173 }
Derek Sollenberger8872b382014-06-23 14:13:53 -0400174}
175
176void SkiaCanvas::restoreToCount(int restoreCount) {
177 while (mCanvas->getSaveCount() > restoreCount) {
178 this->restore();
179 }
180}
181
Florin Malitaeecff562015-12-21 10:43:01 -0500182static inline SkCanvas::SaveLayerFlags layerFlags(SaveFlags::Flags flags) {
183 SkCanvas::SaveLayerFlags layerFlags = 0;
184
Florin Malitaeecff562015-12-21 10:43:01 -0500185 if (!(flags & SaveFlags::ClipToLayer)) {
186 layerFlags |= SkCanvas::kDontClipToLayer_Legacy_SaveLayerFlag;
187 }
188
189 return layerFlags;
190}
191
John Reck1bcacfd2017-11-03 10:12:19 -0700192int SkiaCanvas::saveLayer(float left, float top, float right, float bottom, const SkPaint* paint,
193 SaveFlags::Flags flags) {
Florin Malitaeecff562015-12-21 10:43:01 -0500194 const SkRect bounds = SkRect::MakeLTRB(left, top, right, bottom);
Derek Sollenbergerb8201192017-01-09 16:11:59 -0500195 const SkCanvas::SaveLayerRec rec(&bounds, paint, layerFlags(flags));
Florin Malitaeecff562015-12-21 10:43:01 -0500196
Stan Iliev68885e32016-12-14 11:18:34 -0500197 return mCanvas->saveLayer(rec);
Derek Sollenberger8872b382014-06-23 14:13:53 -0400198}
199
John Reck1bcacfd2017-11-03 10:12:19 -0700200int SkiaCanvas::saveLayerAlpha(float left, float top, float right, float bottom, int alpha,
201 SaveFlags::Flags flags) {
Florin Malitaeecff562015-12-21 10:43:01 -0500202 if (static_cast<unsigned>(alpha) < 0xFF) {
Yuqian Lifd92ee42016-04-27 17:03:38 -0400203 SkPaint alphaPaint;
204 alphaPaint.setAlpha(alpha);
205 return this->saveLayer(left, top, right, bottom, &alphaPaint, flags);
Florin Malitaeecff562015-12-21 10:43:01 -0500206 }
Yuqian Lifd92ee42016-04-27 17:03:38 -0400207 return this->saveLayer(left, top, right, bottom, nullptr, flags);
Derek Sollenberger8872b382014-06-23 14:13:53 -0400208}
209
Stan Ilievf50806a2016-10-24 10:40:39 -0400210class SkiaCanvas::Clip {
211public:
Mike Reed6e49c9f2016-12-02 15:36:59 -0500212 Clip(const SkRect& rect, SkClipOp op, const SkMatrix& m)
John Reck1bcacfd2017-11-03 10:12:19 -0700213 : mType(Type::Rect), mOp(op), mMatrix(m), mRRect(SkRRect::MakeRect(rect)) {}
Mike Reed6e49c9f2016-12-02 15:36:59 -0500214 Clip(const SkRRect& rrect, SkClipOp op, const SkMatrix& m)
John Reck1bcacfd2017-11-03 10:12:19 -0700215 : mType(Type::RRect), mOp(op), mMatrix(m), mRRect(rrect) {}
Mike Reed6e49c9f2016-12-02 15:36:59 -0500216 Clip(const SkPath& path, SkClipOp op, const SkMatrix& m)
Ben Wagner0ed10be2018-06-28 17:08:16 -0400217 : mType(Type::Path), mOp(op), mMatrix(m), mPath(std::in_place, path) {}
Stan Ilievf50806a2016-10-24 10:40:39 -0400218
219 void apply(SkCanvas* canvas) const {
220 canvas->setMatrix(mMatrix);
221 switch (mType) {
John Reck1bcacfd2017-11-03 10:12:19 -0700222 case Type::Rect:
223 canvas->clipRect(mRRect.rect(), mOp);
224 break;
225 case Type::RRect:
226 canvas->clipRRect(mRRect, mOp);
227 break;
228 case Type::Path:
Ben Wagner0ed10be2018-06-28 17:08:16 -0400229 canvas->clipPath(mPath.value(), mOp);
John Reck1bcacfd2017-11-03 10:12:19 -0700230 break;
Stan Ilievf50806a2016-10-24 10:40:39 -0400231 }
232 }
233
234private:
235 enum class Type {
236 Rect,
237 RRect,
238 Path,
239 };
240
John Reck1bcacfd2017-11-03 10:12:19 -0700241 Type mType;
242 SkClipOp mOp;
243 SkMatrix mMatrix;
Stan Ilievf50806a2016-10-24 10:40:39 -0400244
245 // These are logically a union (tracked separately due to non-POD path).
Ben Wagner0ed10be2018-06-28 17:08:16 -0400246 std::optional<SkPath> mPath;
John Reck1bcacfd2017-11-03 10:12:19 -0700247 SkRRect mRRect;
Stan Ilievf50806a2016-10-24 10:40:39 -0400248};
249
250const SkiaCanvas::SaveRec* SkiaCanvas::currentSaveRec() const {
John Reck1bcacfd2017-11-03 10:12:19 -0700251 const SaveRec* rec = mSaveStack ? static_cast<const SaveRec*>(mSaveStack->back()) : nullptr;
Stan Ilievf50806a2016-10-24 10:40:39 -0400252 int currentSaveCount = mCanvas->getSaveCount();
253 SkASSERT(!rec || currentSaveCount >= rec->saveCount);
254
255 return (rec && rec->saveCount == currentSaveCount) ? rec : nullptr;
256}
257
Derek Sollenberger8872b382014-06-23 14:13:53 -0400258// ----------------------------------------------------------------------------
259// functions to emulate legacy SaveFlags (i.e. independent matrix/clip flags)
260// ----------------------------------------------------------------------------
261
Florin Malitaeecff562015-12-21 10:43:01 -0500262void SkiaCanvas::recordPartialSave(SaveFlags::Flags flags) {
Derek Sollenberger8872b382014-06-23 14:13:53 -0400263 // A partial save is a save operation which doesn't capture the full canvas state.
Florin Malitaeecff562015-12-21 10:43:01 -0500264 // (either SaveFlags::Matrix or SaveFlags::Clip is missing).
Derek Sollenberger8872b382014-06-23 14:13:53 -0400265
266 // Mask-out non canvas state bits.
Florin Malitaeecff562015-12-21 10:43:01 -0500267 flags &= SaveFlags::MatrixClip;
Derek Sollenberger8872b382014-06-23 14:13:53 -0400268
Florin Malitaeecff562015-12-21 10:43:01 -0500269 if (flags == SaveFlags::MatrixClip) {
Derek Sollenberger8872b382014-06-23 14:13:53 -0400270 // not a partial save.
271 return;
272 }
273
Stan Ilievf50806a2016-10-24 10:40:39 -0400274 if (!mSaveStack) {
Ben Wagnerd1cbc162015-08-19 12:45:09 -0400275 mSaveStack.reset(new SkDeque(sizeof(struct SaveRec), 8));
Derek Sollenberger8872b382014-06-23 14:13:53 -0400276 }
277
278 SaveRec* rec = static_cast<SaveRec*>(mSaveStack->push_back());
Florin Malita5e271402015-11-04 14:36:02 -0500279 rec->saveCount = mCanvas->getSaveCount();
Derek Sollenberger8872b382014-06-23 14:13:53 -0400280 rec->saveFlags = flags;
Stan Ilievf50806a2016-10-24 10:40:39 -0400281 rec->clipIndex = mClipStack.size();
Derek Sollenberger8872b382014-06-23 14:13:53 -0400282}
283
Stan Ilievf50806a2016-10-24 10:40:39 -0400284template <typename T>
Mike Reed6e49c9f2016-12-02 15:36:59 -0500285void SkiaCanvas::recordClip(const T& clip, SkClipOp op) {
Stan Ilievf50806a2016-10-24 10:40:39 -0400286 // Only need tracking when in a partial save frame which
287 // doesn't restore the clip.
288 const SaveRec* rec = this->currentSaveRec();
289 if (rec && !(rec->saveFlags & SaveFlags::Clip)) {
290 mClipStack.emplace_back(clip, op, mCanvas->getTotalMatrix());
Derek Sollenberger8872b382014-06-23 14:13:53 -0400291 }
292}
293
Stan Ilievf50806a2016-10-24 10:40:39 -0400294// Applies and optionally removes all clips >= index.
295void SkiaCanvas::applyPersistentClips(size_t clipStartIndex) {
296 SkASSERT(clipStartIndex <= mClipStack.size());
297 const auto begin = mClipStack.cbegin() + clipStartIndex;
298 const auto end = mClipStack.cend();
Derek Sollenberger8872b382014-06-23 14:13:53 -0400299
Stan Ilievf50806a2016-10-24 10:40:39 -0400300 // Clip application mutates the CTM.
301 const SkMatrix saveMatrix = mCanvas->getTotalMatrix();
Derek Sollenberger8872b382014-06-23 14:13:53 -0400302
Stan Ilievf50806a2016-10-24 10:40:39 -0400303 for (auto clip = begin; clip != end; ++clip) {
Mike Reed6acfe162016-11-18 17:21:09 -0500304 clip->apply(mCanvas);
Derek Sollenberger8872b382014-06-23 14:13:53 -0400305 }
306
Stan Ilievf50806a2016-10-24 10:40:39 -0400307 mCanvas->setMatrix(saveMatrix);
308
309 // If the current/post-restore save rec is also persisting clips, we
310 // leave them on the stack to be reapplied part of the next restore().
311 // Otherwise we're done and just pop them.
312 const auto* rec = this->currentSaveRec();
313 if (!rec || (rec->saveFlags & SaveFlags::Clip)) {
314 mClipStack.erase(begin, end);
315 }
Derek Sollenberger8872b382014-06-23 14:13:53 -0400316}
317
318// ----------------------------------------------------------------------------
319// Canvas state operations: Matrix
320// ----------------------------------------------------------------------------
321
322void SkiaCanvas::getMatrix(SkMatrix* outMatrix) const {
323 *outMatrix = mCanvas->getTotalMatrix();
324}
325
326void SkiaCanvas::setMatrix(const SkMatrix& matrix) {
327 mCanvas->setMatrix(matrix);
328}
329
330void SkiaCanvas::concat(const SkMatrix& matrix) {
331 mCanvas->concat(matrix);
332}
333
334void SkiaCanvas::rotate(float degrees) {
335 mCanvas->rotate(degrees);
336}
337
338void SkiaCanvas::scale(float sx, float sy) {
339 mCanvas->scale(sx, sy);
340}
341
342void SkiaCanvas::skew(float sx, float sy) {
343 mCanvas->skew(sx, sy);
344}
345
346void SkiaCanvas::translate(float dx, float dy) {
347 mCanvas->translate(dx, dy);
348}
349
350// ----------------------------------------------------------------------------
351// Canvas state operations: Clips
352// ----------------------------------------------------------------------------
353
354// This function is a mirror of SkCanvas::getClipBounds except that it does
355// not outset the edge of the clip to account for anti-aliasing. There is
356// a skia bug to investigate pushing this logic into back into skia.
357// (see https://code.google.com/p/skia/issues/detail?id=1303)
358bool SkiaCanvas::getClipBounds(SkRect* outRect) const {
359 SkIRect ibounds;
Mike Reed5e438982017-01-25 08:23:25 -0500360 if (!mCanvas->getDeviceClipBounds(&ibounds)) {
Derek Sollenberger8872b382014-06-23 14:13:53 -0400361 return false;
362 }
363
364 SkMatrix inverse;
365 // if we can't invert the CTM, we can't return local clip bounds
366 if (!mCanvas->getTotalMatrix().invert(&inverse)) {
367 if (outRect) {
368 outRect->setEmpty();
369 }
370 return false;
371 }
372
373 if (NULL != outRect) {
374 SkRect r = SkRect::Make(ibounds);
375 inverse.mapRect(outRect, r);
376 }
377 return true;
378}
379
380bool SkiaCanvas::quickRejectRect(float left, float top, float right, float bottom) const {
381 SkRect bounds = SkRect::MakeLTRB(left, top, right, bottom);
382 return mCanvas->quickReject(bounds);
383}
384
385bool SkiaCanvas::quickRejectPath(const SkPath& path) const {
386 return mCanvas->quickReject(path);
387}
388
Mike Reed6e49c9f2016-12-02 15:36:59 -0500389bool SkiaCanvas::clipRect(float left, float top, float right, float bottom, SkClipOp op) {
Derek Sollenberger8872b382014-06-23 14:13:53 -0400390 SkRect rect = SkRect::MakeLTRB(left, top, right, bottom);
Stan Ilievf50806a2016-10-24 10:40:39 -0400391 this->recordClip(rect, op);
Derek Sollenberger8872b382014-06-23 14:13:53 -0400392 mCanvas->clipRect(rect, op);
Chris Craik5ec6a282015-06-23 15:42:12 -0700393 return !mCanvas->isClipEmpty();
Derek Sollenberger8872b382014-06-23 14:13:53 -0400394}
395
Mike Reed6e49c9f2016-12-02 15:36:59 -0500396bool SkiaCanvas::clipPath(const SkPath* path, SkClipOp op) {
Derek Sollenbergerf7d98f42017-04-17 11:27:36 -0400397 this->recordClip(*path, op);
398 mCanvas->clipPath(*path, op);
Chris Craik5ec6a282015-06-23 15:42:12 -0700399 return !mCanvas->isClipEmpty();
Derek Sollenberger8872b382014-06-23 14:13:53 -0400400}
401
Derek Sollenberger8872b382014-06-23 14:13:53 -0400402// ----------------------------------------------------------------------------
403// Canvas state operations: Filters
404// ----------------------------------------------------------------------------
405
Ben Wagner0ed10be2018-06-28 17:08:16 -0400406PaintFilter* SkiaCanvas::getPaintFilter() {
407 return mPaintFilter.get();
Derek Sollenbergeracb40992014-07-21 15:22:10 -0400408}
409
Ben Wagner0ed10be2018-06-28 17:08:16 -0400410void SkiaCanvas::setPaintFilter(sk_sp<PaintFilter> paintFilter) {
411 mPaintFilter = std::move(paintFilter);
Derek Sollenberger8872b382014-06-23 14:13:53 -0400412}
413
414// ----------------------------------------------------------------------------
Matt Sarettd0814db2017-04-13 09:33:18 -0400415// Canvas state operations: Capture
416// ----------------------------------------------------------------------------
417
418SkCanvasState* SkiaCanvas::captureCanvasState() const {
419 SkCanvas* canvas = mCanvas;
420 if (mCanvasOwned) {
421 // Important to use the underlying SkCanvas, not the wrapper.
422 canvas = mCanvasOwned.get();
423 }
424
425 // Workarounds for http://crbug.com/271096: SW draw only supports
426 // translate & scale transforms, and a simple rectangular clip.
427 // (This also avoids significant wasted time in calling
428 // SkCanvasStateUtils::CaptureCanvasState when the clip is complex).
John Reck1bcacfd2017-11-03 10:12:19 -0700429 if (!canvas->isClipRect() || (canvas->getTotalMatrix().getType() &
430 ~(SkMatrix::kTranslate_Mask | SkMatrix::kScale_Mask))) {
431 return nullptr;
Matt Sarettd0814db2017-04-13 09:33:18 -0400432 }
433
434 return SkCanvasStateUtils::CaptureCanvasState(canvas);
435}
436
437// ----------------------------------------------------------------------------
Derek Sollenberger8872b382014-06-23 14:13:53 -0400438// Canvas draw operations
439// ----------------------------------------------------------------------------
440
Mike Reed260ab722016-10-07 15:59:20 -0400441void SkiaCanvas::drawColor(int color, SkBlendMode mode) {
Derek Sollenberger8872b382014-06-23 14:13:53 -0400442 mCanvas->drawColor(color, mode);
443}
444
Ben Wagner0ed10be2018-06-28 17:08:16 -0400445SkiaCanvas::PaintCoW&& SkiaCanvas::filterPaint(PaintCoW&& paint) const {
446 if (mPaintFilter) {
447 mPaintFilter->filter(&paint.writeable());
448 }
449 return std::move(paint);
450}
451
Derek Sollenbergeracb40992014-07-21 15:22:10 -0400452void SkiaCanvas::drawPaint(const SkPaint& paint) {
Ben Wagner0ed10be2018-06-28 17:08:16 -0400453 mCanvas->drawPaint(*filterPaint(paint));
Derek Sollenberger8872b382014-06-23 14:13:53 -0400454}
455
456// ----------------------------------------------------------------------------
457// Canvas draw operations: Geometry
458// ----------------------------------------------------------------------------
459
Derek Sollenbergeracb40992014-07-21 15:22:10 -0400460void SkiaCanvas::drawPoints(const float* points, int count, const SkPaint& paint,
Derek Sollenberger8872b382014-06-23 14:13:53 -0400461 SkCanvas::PointMode mode) {
Derek Sollenberger792fb322017-03-03 14:02:09 -0500462 if (CC_UNLIKELY(count < 2 || paint.nothingToDraw())) return;
Derek Sollenberger8872b382014-06-23 14:13:53 -0400463 // convert the floats into SkPoints
John Reck1bcacfd2017-11-03 10:12:19 -0700464 count >>= 1; // now it is the number of points
Ben Wagner6bbf68d2015-08-19 11:26:06 -0400465 std::unique_ptr<SkPoint[]> pts(new SkPoint[count]);
Derek Sollenberger8872b382014-06-23 14:13:53 -0400466 for (int i = 0; i < count; i++) {
467 pts[i].set(points[0], points[1]);
468 points += 2;
469 }
Ben Wagner0ed10be2018-06-28 17:08:16 -0400470 mCanvas->drawPoints(mode, count, pts.get(), *filterPaint(paint));
Derek Sollenberger8872b382014-06-23 14:13:53 -0400471}
472
Derek Sollenbergeracb40992014-07-21 15:22:10 -0400473void SkiaCanvas::drawPoint(float x, float y, const SkPaint& paint) {
Ben Wagner0ed10be2018-06-28 17:08:16 -0400474 mCanvas->drawPoint(x, y, *filterPaint(paint));
Derek Sollenberger8872b382014-06-23 14:13:53 -0400475}
476
Derek Sollenbergeracb40992014-07-21 15:22:10 -0400477void SkiaCanvas::drawPoints(const float* points, int count, const SkPaint& paint) {
Ben Wagner0ed10be2018-06-28 17:08:16 -0400478 this->drawPoints(points, count, *filterPaint(paint), SkCanvas::kPoints_PointMode);
Derek Sollenberger8872b382014-06-23 14:13:53 -0400479}
480
481void SkiaCanvas::drawLine(float startX, float startY, float stopX, float stopY,
Derek Sollenbergeracb40992014-07-21 15:22:10 -0400482 const SkPaint& paint) {
Ben Wagner0ed10be2018-06-28 17:08:16 -0400483 mCanvas->drawLine(startX, startY, stopX, stopY, *filterPaint(paint));
Derek Sollenberger8872b382014-06-23 14:13:53 -0400484}
485
Derek Sollenbergeracb40992014-07-21 15:22:10 -0400486void SkiaCanvas::drawLines(const float* points, int count, const SkPaint& paint) {
Derek Sollenberger792fb322017-03-03 14:02:09 -0500487 if (CC_UNLIKELY(count < 4 || paint.nothingToDraw())) return;
Ben Wagner0ed10be2018-06-28 17:08:16 -0400488 this->drawPoints(points, count, *filterPaint(paint), SkCanvas::kLines_PointMode);
Derek Sollenberger8872b382014-06-23 14:13:53 -0400489}
490
John Reck1bcacfd2017-11-03 10:12:19 -0700491void SkiaCanvas::drawRect(float left, float top, float right, float bottom, const SkPaint& paint) {
Derek Sollenberger792fb322017-03-03 14:02:09 -0500492 if (CC_UNLIKELY(paint.nothingToDraw())) return;
Ben Wagner0ed10be2018-06-28 17:08:16 -0400493 mCanvas->drawRect({left, top, right, bottom}, *filterPaint(paint));
Derek Sollenberger8872b382014-06-23 14:13:53 -0400494}
495
Derek Sollenberger94394b32015-07-10 09:58:41 -0400496void SkiaCanvas::drawRegion(const SkRegion& region, const SkPaint& paint) {
Derek Sollenberger792fb322017-03-03 14:02:09 -0500497 if (CC_UNLIKELY(paint.nothingToDraw())) return;
Ben Wagner0ed10be2018-06-28 17:08:16 -0400498 mCanvas->drawRegion(region, *filterPaint(paint));
Derek Sollenberger94394b32015-07-10 09:58:41 -0400499}
500
John Reck1bcacfd2017-11-03 10:12:19 -0700501void SkiaCanvas::drawRoundRect(float left, float top, float right, float bottom, float rx, float ry,
502 const SkPaint& paint) {
Derek Sollenberger792fb322017-03-03 14:02:09 -0500503 if (CC_UNLIKELY(paint.nothingToDraw())) return;
Derek Sollenberger8872b382014-06-23 14:13:53 -0400504 SkRect rect = SkRect::MakeLTRB(left, top, right, bottom);
Ben Wagner0ed10be2018-06-28 17:08:16 -0400505 mCanvas->drawRoundRect(rect, rx, ry, *filterPaint(paint));
Derek Sollenberger8872b382014-06-23 14:13:53 -0400506}
507
Nader Jawadadfe1d92018-09-27 12:27:36 -0700508void SkiaCanvas::drawDoubleRoundRect(const SkRRect& outer, const SkRRect& inner,
509 const SkPaint& paint) {
510 mCanvas->drawDRRect(outer, inner, *filterPaint(paint));
511}
512
Derek Sollenbergeracb40992014-07-21 15:22:10 -0400513void SkiaCanvas::drawCircle(float x, float y, float radius, const SkPaint& paint) {
Derek Sollenberger792fb322017-03-03 14:02:09 -0500514 if (CC_UNLIKELY(radius <= 0 || paint.nothingToDraw())) return;
Ben Wagner0ed10be2018-06-28 17:08:16 -0400515 mCanvas->drawCircle(x, y, radius, *filterPaint(paint));
Derek Sollenberger8872b382014-06-23 14:13:53 -0400516}
517
Derek Sollenbergeracb40992014-07-21 15:22:10 -0400518void SkiaCanvas::drawOval(float left, float top, float right, float bottom, const SkPaint& paint) {
Derek Sollenberger792fb322017-03-03 14:02:09 -0500519 if (CC_UNLIKELY(paint.nothingToDraw())) return;
Derek Sollenberger8872b382014-06-23 14:13:53 -0400520 SkRect oval = SkRect::MakeLTRB(left, top, right, bottom);
Ben Wagner0ed10be2018-06-28 17:08:16 -0400521 mCanvas->drawOval(oval, *filterPaint(paint));
Derek Sollenberger8872b382014-06-23 14:13:53 -0400522}
523
John Reck1bcacfd2017-11-03 10:12:19 -0700524void SkiaCanvas::drawArc(float left, float top, float right, float bottom, float startAngle,
525 float sweepAngle, bool useCenter, const SkPaint& paint) {
Derek Sollenberger792fb322017-03-03 14:02:09 -0500526 if (CC_UNLIKELY(paint.nothingToDraw())) return;
Derek Sollenberger8872b382014-06-23 14:13:53 -0400527 SkRect arc = SkRect::MakeLTRB(left, top, right, bottom);
Derek Sollenbergeref3b2182017-11-06 13:55:59 -0500528 if (fabs(sweepAngle) >= 360.0f) {
Ben Wagner0ed10be2018-06-28 17:08:16 -0400529 mCanvas->drawOval(arc, *filterPaint(paint));
Derek Sollenbergeref3b2182017-11-06 13:55:59 -0500530 } else {
Ben Wagner0ed10be2018-06-28 17:08:16 -0400531 mCanvas->drawArc(arc, startAngle, sweepAngle, useCenter, *filterPaint(paint));
Derek Sollenbergeref3b2182017-11-06 13:55:59 -0500532 }
Derek Sollenberger8872b382014-06-23 14:13:53 -0400533}
534
Derek Sollenbergeracb40992014-07-21 15:22:10 -0400535void SkiaCanvas::drawPath(const SkPath& path, const SkPaint& paint) {
Derek Sollenberger792fb322017-03-03 14:02:09 -0500536 if (CC_UNLIKELY(paint.nothingToDraw())) return;
Stan Iliev6dcfdec2017-08-15 16:42:05 -0400537 if (CC_UNLIKELY(path.isEmpty() && (!path.isInverseFillType()))) {
538 return;
539 }
Ben Wagner0ed10be2018-06-28 17:08:16 -0400540 mCanvas->drawPath(path, *filterPaint(paint));
Derek Sollenberger8872b382014-06-23 14:13:53 -0400541}
542
Mike Reed826deef2017-04-04 15:32:04 -0400543void SkiaCanvas::drawVertices(const SkVertices* vertices, SkBlendMode mode, const SkPaint& paint) {
Ben Wagner0ed10be2018-06-28 17:08:16 -0400544 mCanvas->drawVertices(vertices, mode, *filterPaint(paint));
Derek Sollenberger8872b382014-06-23 14:13:53 -0400545}
546
547// ----------------------------------------------------------------------------
548// Canvas draw operations: Bitmaps
549// ----------------------------------------------------------------------------
550
Ben Wagner0ed10be2018-06-28 17:08:16 -0400551SkiaCanvas::PaintCoW&& SkiaCanvas::filterBitmap(PaintCoW&& paint,
552 sk_sp<SkColorFilter> colorSpaceFilter) const {
Derek Sollenbergerfa3e3402017-08-04 08:35:10 -0400553 /* We don't apply the colorSpace filter if this canvas is already wrapped with
554 * a SkColorSpaceXformCanvas since it already takes care of converting the
555 * contents of the bitmap into the appropriate colorspace. The mCanvasWrapper
556 * should only be used if this canvas is backed by a surface/bitmap that is known
557 * to have a non-sRGB colorspace.
558 */
559 if (!mCanvasWrapper && colorSpaceFilter) {
Ben Wagner0ed10be2018-06-28 17:08:16 -0400560 SkPaint& tmpPaint = paint.writeable();
561 if (tmpPaint.getColorFilter()) {
562 tmpPaint.setColorFilter(SkColorFilter::MakeComposeFilter(tmpPaint.refColorFilter(),
563 std::move(colorSpaceFilter)));
564 LOG_ALWAYS_FATAL_IF(!tmpPaint.getColorFilter());
Derek Sollenbergerfa3e3402017-08-04 08:35:10 -0400565 } else {
Ben Wagner0ed10be2018-06-28 17:08:16 -0400566 tmpPaint.setColorFilter(std::move(colorSpaceFilter));
Derek Sollenbergerfa3e3402017-08-04 08:35:10 -0400567 }
Derek Sollenbergerfb0c8fc2017-07-26 13:53:27 -0400568 }
Ben Wagner0ed10be2018-06-28 17:08:16 -0400569 return filterPaint(std::move(paint));
Derek Sollenberger8872b382014-06-23 14:13:53 -0400570}
571
Derek Sollenbergerfb0c8fc2017-07-26 13:53:27 -0400572void SkiaCanvas::drawBitmap(Bitmap& bitmap, float left, float top, const SkPaint* paint) {
Derek Sollenbergerfb0c8fc2017-07-26 13:53:27 -0400573 sk_sp<SkColorFilter> colorFilter;
574 sk_sp<SkImage> image = bitmap.makeImage(&colorFilter);
Ben Wagner0ed10be2018-06-28 17:08:16 -0400575 mCanvas->drawImage(image, left, top, filterBitmap(paint, std::move(colorFilter)));
Derek Sollenbergerfb0c8fc2017-07-26 13:53:27 -0400576}
577
578void SkiaCanvas::drawBitmap(Bitmap& bitmap, const SkMatrix& matrix, const SkPaint* paint) {
Mike Reed6acfe162016-11-18 17:21:09 -0500579 SkAutoCanvasRestore acr(mCanvas, true);
Mike Reed70ffbf92014-12-08 17:03:30 -0500580 mCanvas->concat(matrix);
Derek Sollenbergerfb0c8fc2017-07-26 13:53:27 -0400581
Derek Sollenbergerfb0c8fc2017-07-26 13:53:27 -0400582 sk_sp<SkColorFilter> colorFilter;
583 sk_sp<SkImage> image = bitmap.makeImage(&colorFilter);
Ben Wagner0ed10be2018-06-28 17:08:16 -0400584 mCanvas->drawImage(image, 0, 0, filterBitmap(paint, std::move(colorFilter)));
Derek Sollenberger8872b382014-06-23 14:13:53 -0400585}
586
John Reck1bcacfd2017-11-03 10:12:19 -0700587void SkiaCanvas::drawBitmap(Bitmap& bitmap, float srcLeft, float srcTop, float srcRight,
588 float srcBottom, float dstLeft, float dstTop, float dstRight,
589 float dstBottom, const SkPaint* paint) {
Derek Sollenberger8872b382014-06-23 14:13:53 -0400590 SkRect srcRect = SkRect::MakeLTRB(srcLeft, srcTop, srcRight, srcBottom);
591 SkRect dstRect = SkRect::MakeLTRB(dstLeft, dstTop, dstRight, dstBottom);
Derek Sollenbergerfb0c8fc2017-07-26 13:53:27 -0400592
Derek Sollenbergerfb0c8fc2017-07-26 13:53:27 -0400593 sk_sp<SkColorFilter> colorFilter;
594 sk_sp<SkImage> image = bitmap.makeImage(&colorFilter);
Ben Wagner0ed10be2018-06-28 17:08:16 -0400595 mCanvas->drawImageRect(image, srcRect, dstRect, filterBitmap(paint, std::move(colorFilter)),
John Reck1bcacfd2017-11-03 10:12:19 -0700596 SkCanvas::kFast_SrcRectConstraint);
Derek Sollenberger8872b382014-06-23 14:13:53 -0400597}
598
Derek Sollenbergerfb0c8fc2017-07-26 13:53:27 -0400599void SkiaCanvas::drawBitmapMesh(Bitmap& bitmap, int meshWidth, int meshHeight,
John Reck1bcacfd2017-11-03 10:12:19 -0700600 const float* vertices, const int* colors, const SkPaint* paint) {
Derek Sollenberger8872b382014-06-23 14:13:53 -0400601 const int ptCount = (meshWidth + 1) * (meshHeight + 1);
602 const int indexCount = meshWidth * meshHeight * 6;
Mike Reed871cd2d2017-03-17 10:15:52 -0400603 uint32_t flags = SkVertices::kHasTexCoords_BuilderFlag;
604 if (colors) {
605 flags |= SkVertices::kHasColors_BuilderFlag;
606 }
Mike Reed826deef2017-04-04 15:32:04 -0400607 SkVertices::Builder builder(SkVertices::kTriangles_VertexMode, ptCount, indexCount, flags);
Mike Reed871cd2d2017-03-17 10:15:52 -0400608 memcpy(builder.positions(), vertices, ptCount * sizeof(SkPoint));
609 if (colors) {
610 memcpy(builder.colors(), colors, ptCount * sizeof(SkColor));
611 }
612 SkPoint* texs = builder.texCoords();
613 uint16_t* indices = builder.indices();
Derek Sollenberger8872b382014-06-23 14:13:53 -0400614
615 // cons up texture coordinates and indices
616 {
Derek Sollenbergerfb0c8fc2017-07-26 13:53:27 -0400617 const SkScalar w = SkIntToScalar(bitmap.width());
618 const SkScalar h = SkIntToScalar(bitmap.height());
Derek Sollenberger8872b382014-06-23 14:13:53 -0400619 const SkScalar dx = w / meshWidth;
620 const SkScalar dy = h / meshHeight;
621
622 SkPoint* texsPtr = texs;
623 SkScalar y = 0;
624 for (int i = 0; i <= meshHeight; i++) {
625 if (i == meshHeight) {
626 y = h; // to ensure numerically we hit h exactly
627 }
628 SkScalar x = 0;
629 for (int j = 0; j < meshWidth; j++) {
630 texsPtr->set(x, y);
631 texsPtr += 1;
632 x += dx;
633 }
634 texsPtr->set(w, y);
635 texsPtr += 1;
636 y += dy;
637 }
638 SkASSERT(texsPtr - texs == ptCount);
639 }
640
641 // cons up indices
642 {
643 uint16_t* indexPtr = indices;
644 int index = 0;
645 for (int i = 0; i < meshHeight; i++) {
646 for (int j = 0; j < meshWidth; j++) {
647 // lower-left triangle
648 *indexPtr++ = index;
649 *indexPtr++ = index + meshWidth + 1;
650 *indexPtr++ = index + meshWidth + 2;
651 // upper-right triangle
652 *indexPtr++ = index;
653 *indexPtr++ = index + meshWidth + 2;
654 *indexPtr++ = index + 1;
655 // bump to the next cell
656 index += 1;
657 }
658 // bump to the next row
659 index += 1;
660 }
661 SkASSERT(indexPtr - indices == indexCount);
Derek Sollenberger8872b382014-06-23 14:13:53 -0400662 }
663
John Reck1bcacfd2017-11-03 10:12:19 -0700664// double-check that we have legal indices
Derek Sollenberger8872b382014-06-23 14:13:53 -0400665#ifdef SK_DEBUG
666 {
667 for (int i = 0; i < indexCount; i++) {
668 SkASSERT((unsigned)indices[i] < (unsigned)ptCount);
669 }
670 }
671#endif
672
673 // cons-up a shader for the bitmap
Ben Wagner0ed10be2018-06-28 17:08:16 -0400674 PaintCoW paintCoW(paint);
675 SkPaint& tmpPaint = paintCoW.writeable();
Stan Ilievf50806a2016-10-24 10:40:39 -0400676
Derek Sollenbergerfb0c8fc2017-07-26 13:53:27 -0400677 sk_sp<SkColorFilter> colorFilter;
678 sk_sp<SkImage> image = bitmap.makeImage(&colorFilter);
John Reck1bcacfd2017-11-03 10:12:19 -0700679 sk_sp<SkShader> shader =
680 image->makeShader(SkShader::kClamp_TileMode, SkShader::kClamp_TileMode);
681 if (colorFilter) {
Ben Wagner0ed10be2018-06-28 17:08:16 -0400682 shader = shader->makeWithColorFilter(std::move(colorFilter));
Derek Sollenbergerfb0c8fc2017-07-26 13:53:27 -0400683 }
Ben Wagner0ed10be2018-06-28 17:08:16 -0400684 tmpPaint.setShader(std::move(shader));
Derek Sollenberger8872b382014-06-23 14:13:53 -0400685
Ben Wagner0ed10be2018-06-28 17:08:16 -0400686 mCanvas->drawVertices(builder.detach(), SkBlendMode::kModulate,
687 *filterPaint(std::move(paintCoW)));
Derek Sollenberger8872b382014-06-23 14:13:53 -0400688}
689
John Reck1bcacfd2017-11-03 10:12:19 -0700690void SkiaCanvas::drawNinePatch(Bitmap& bitmap, const Res_png_9patch& chunk, float dstLeft,
691 float dstTop, float dstRight, float dstBottom,
692 const SkPaint* paint) {
Stan Ilievf50806a2016-10-24 10:40:39 -0400693 SkCanvas::Lattice lattice;
Derek Sollenbergerfb0c8fc2017-07-26 13:53:27 -0400694 NinePatchUtils::SetLatticeDivs(&lattice, chunk, bitmap.width(), bitmap.height());
Stan Ilievf50806a2016-10-24 10:40:39 -0400695
Stan Ilieve12d7312017-12-04 14:48:27 -0500696 lattice.fRectTypes = nullptr;
697 lattice.fColors = nullptr;
Stan Ilievf50806a2016-10-24 10:40:39 -0400698 int numFlags = 0;
Stan Iliev021693b2016-10-17 16:26:15 -0400699 if (chunk.numColors > 0 && chunk.numColors == NinePatchUtils::NumDistinctRects(lattice)) {
Stan Ilievf50806a2016-10-24 10:40:39 -0400700 // We can expect the framework to give us a color for every distinct rect.
701 // Skia requires a flag for every rect.
702 numFlags = (lattice.fXCount + 1) * (lattice.fYCount + 1);
703 }
704
Stan Ilieve12d7312017-12-04 14:48:27 -0500705 SkAutoSTMalloc<25, SkCanvas::Lattice::RectType> flags(numFlags);
706 SkAutoSTMalloc<25, SkColor> colors(numFlags);
Stan Ilievf50806a2016-10-24 10:40:39 -0400707 if (numFlags > 0) {
Stan Ilieve12d7312017-12-04 14:48:27 -0500708 NinePatchUtils::SetLatticeFlags(&lattice, flags.get(), numFlags, chunk, colors.get());
Stan Ilievf50806a2016-10-24 10:40:39 -0400709 }
710
711 lattice.fBounds = nullptr;
712 SkRect dst = SkRect::MakeLTRB(dstLeft, dstTop, dstRight, dstBottom);
Derek Sollenbergerfb0c8fc2017-07-26 13:53:27 -0400713
Derek Sollenbergerfb0c8fc2017-07-26 13:53:27 -0400714 sk_sp<SkColorFilter> colorFilter;
715 sk_sp<SkImage> image = bitmap.makeImage(&colorFilter);
Ben Wagner0ed10be2018-06-28 17:08:16 -0400716 mCanvas->drawImageLattice(image.get(), lattice, dst,
717 filterBitmap(paint, std::move(colorFilter)));
Derek Sollenbergeredca3202015-07-10 13:56:39 -0400718}
719
Derek Sollenberger2d142132018-01-22 10:25:26 -0500720double SkiaCanvas::drawAnimatedImage(AnimatedImageDrawable* imgDrawable) {
721 return imgDrawable->drawStaging(mCanvas);
Leon Scroggins III671cce22018-01-14 16:52:17 -0500722}
723
Doris Liu766431a2016-02-04 22:17:11 +0000724void SkiaCanvas::drawVectorDrawable(VectorDrawableRoot* vectorDrawable) {
Doris Liu1d8e1942016-03-02 15:16:28 -0800725 vectorDrawable->drawStaging(this);
Doris Liu766431a2016-02-04 22:17:11 +0000726}
727
Derek Sollenberger8872b382014-06-23 14:13:53 -0400728// ----------------------------------------------------------------------------
729// Canvas draw operations: Text
730// ----------------------------------------------------------------------------
731
Stan Iliev0b58d992017-03-30 18:22:27 -0400732void SkiaCanvas::drawGlyphs(ReadGlyphFunc glyphFunc, int count, const SkPaint& paint, float x,
John Reck1bcacfd2017-11-03 10:12:19 -0700733 float y, float boundsLeft, float boundsTop, float boundsRight,
734 float boundsBottom, float totalAdvance) {
Stan Iliev0b58d992017-03-30 18:22:27 -0400735 if (count <= 0 || paint.nothingToDraw()) return;
Mike Reed1c79eab2018-11-21 11:01:57 -0500736 SkFont font = SkFont::LEGACY_ExtractFromPaint(paint);
Stan Ilievf50806a2016-10-24 10:40:39 -0400737 SkPaint paintCopy(paint);
Ben Wagner0ed10be2018-06-28 17:08:16 -0400738 if (mPaintFilter) {
739 mPaintFilter->filter(&paintCopy);
740 }
Stan Ilieva39b7742017-11-29 13:15:45 -0500741 SkASSERT(paintCopy.getTextEncoding() == SkPaint::kGlyphID_TextEncoding);
Stan Iliev7717e222018-02-05 18:04:11 -0500742 // Stroke with a hairline is drawn on HW with a fill style for compatibility with Android O and
743 // older.
John Recke170fb62018-05-07 08:12:07 -0700744 if (!mCanvasOwned && sApiLevel <= 27 && paintCopy.getStrokeWidth() <= 0 &&
745 paintCopy.getStyle() == SkPaint::kStroke_Style) {
Stan Iliev7717e222018-02-05 18:04:11 -0500746 paintCopy.setStyle(SkPaint::kFill_Style);
747 }
Stan Ilievf50806a2016-10-24 10:40:39 -0400748
John Reck1bcacfd2017-11-03 10:12:19 -0700749 SkRect bounds =
750 SkRect::MakeLTRB(boundsLeft + x, boundsTop + y, boundsRight + x, boundsBottom + y);
Stan Ilievf50806a2016-10-24 10:40:39 -0400751
752 SkTextBlobBuilder builder;
Mike Reed1c79eab2018-11-21 11:01:57 -0500753 const SkTextBlobBuilder::RunBuffer& buffer = builder.allocRunPos(font, count, &bounds);
Stan Iliev0b58d992017-03-30 18:22:27 -0400754 glyphFunc(buffer.glyphs, buffer.pos);
Stan Ilievf50806a2016-10-24 10:40:39 -0400755
756 sk_sp<SkTextBlob> textBlob(builder.make());
757 mCanvas->drawTextBlob(textBlob, 0, 0, paintCopy);
758 drawTextDecorations(x, y, totalAdvance, paintCopy);
Derek Sollenberger8872b382014-06-23 14:13:53 -0400759}
760
Yuqian Liafc221492016-07-18 13:07:42 -0400761void SkiaCanvas::drawLayoutOnPath(const minikin::Layout& layout, float hOffset, float vOffset,
John Reck1bcacfd2017-11-03 10:12:19 -0700762 const SkPaint& paint, const SkPath& path, size_t start,
763 size_t end) {
Derek Sollenberger415a74b2018-03-14 15:03:13 -0400764 SkPaint paintCopy(paint);
Ben Wagner0ed10be2018-06-28 17:08:16 -0400765 if (mPaintFilter) {
766 mPaintFilter->filter(&paintCopy);
767 }
Derek Sollenberger415a74b2018-03-14 15:03:13 -0400768 SkASSERT(paintCopy.getTextEncoding() == SkPaint::kGlyphID_TextEncoding);
769
Yuqian Liafc221492016-07-18 13:07:42 -0400770 const int N = end - start;
Derek Sollenbergere547dd02016-11-09 11:55:59 -0500771 SkAutoSTMalloc<1024, uint8_t> storage(N * (sizeof(uint16_t) + sizeof(SkRSXform)));
Yuqian Liafc221492016-07-18 13:07:42 -0400772 SkRSXform* xform = (SkRSXform*)storage.get();
773 uint16_t* glyphs = (uint16_t*)(xform + N);
774 SkPathMeasure meas(path, false);
775
776 for (size_t i = start; i < end; i++) {
777 glyphs[i - start] = layout.getGlyphId(i);
Stan Ilievc6c96dd2018-01-10 16:06:04 -0500778 float halfWidth = layout.getCharAdvance(i) * 0.5f;
779 float x = hOffset + layout.getX(i) + halfWidth;
Yuqian Liafc221492016-07-18 13:07:42 -0400780 float y = vOffset + layout.getY(i);
781
782 SkPoint pos;
783 SkVector tan;
784 if (!meas.getPosTan(x, &pos, &tan)) {
785 pos.set(x, y);
786 tan.set(1, 0);
787 }
788 xform[i - start].fSCos = tan.x();
789 xform[i - start].fSSin = tan.y();
Stan Ilievc6c96dd2018-01-10 16:06:04 -0500790 xform[i - start].fTx = pos.x() - tan.y() * y - halfWidth * tan.x();
791 xform[i - start].fTy = pos.y() + tan.x() * y - halfWidth * tan.y();
Yuqian Liafc221492016-07-18 13:07:42 -0400792 }
793
Derek Sollenberger415a74b2018-03-14 15:03:13 -0400794 this->asSkCanvas()->drawTextRSXform(glyphs, sizeof(uint16_t) * N, xform, nullptr, paintCopy);
Derek Sollenberger8872b382014-06-23 14:13:53 -0400795}
796
Derek Sollenberger6f485562015-07-30 10:00:39 -0400797// ----------------------------------------------------------------------------
798// Canvas draw operations: Animations
799// ----------------------------------------------------------------------------
800
Derek Sollenberger6f485562015-07-30 10:00:39 -0400801void SkiaCanvas::drawRoundRect(uirenderer::CanvasPropertyPrimitive* left,
John Reck1bcacfd2017-11-03 10:12:19 -0700802 uirenderer::CanvasPropertyPrimitive* top,
803 uirenderer::CanvasPropertyPrimitive* right,
804 uirenderer::CanvasPropertyPrimitive* bottom,
805 uirenderer::CanvasPropertyPrimitive* rx,
806 uirenderer::CanvasPropertyPrimitive* ry,
807 uirenderer::CanvasPropertyPaint* paint) {
Stan Iliev021693b2016-10-17 16:26:15 -0400808 sk_sp<uirenderer::skiapipeline::AnimatedRoundRect> drawable(
John Reck1bcacfd2017-11-03 10:12:19 -0700809 new uirenderer::skiapipeline::AnimatedRoundRect(left, top, right, bottom, rx, ry,
810 paint));
Derek Sollenberger6f485562015-07-30 10:00:39 -0400811 mCanvas->drawDrawable(drawable.get());
812}
813
John Reck1bcacfd2017-11-03 10:12:19 -0700814void SkiaCanvas::drawCircle(uirenderer::CanvasPropertyPrimitive* x,
815 uirenderer::CanvasPropertyPrimitive* y,
816 uirenderer::CanvasPropertyPrimitive* radius,
817 uirenderer::CanvasPropertyPaint* paint) {
818 sk_sp<uirenderer::skiapipeline::AnimatedCircle> drawable(
819 new uirenderer::skiapipeline::AnimatedCircle(x, y, radius, paint));
Derek Sollenberger6f485562015-07-30 10:00:39 -0400820 mCanvas->drawDrawable(drawable.get());
821}
822
823// ----------------------------------------------------------------------------
824// Canvas draw operations: View System
825// ----------------------------------------------------------------------------
826
Stan Ilievf50806a2016-10-24 10:40:39 -0400827void SkiaCanvas::drawLayer(uirenderer::DeferredLayerUpdater* layerUpdater) {
Derek Sollenbergerc1908132016-07-15 10:28:16 -0400828 LOG_ALWAYS_FATAL("SkiaCanvas can't directly draw Layers");
829}
Derek Sollenberger6f485562015-07-30 10:00:39 -0400830
Derek Sollenbergerc1908132016-07-15 10:28:16 -0400831void SkiaCanvas::drawRenderNode(uirenderer::RenderNode* renderNode) {
832 LOG_ALWAYS_FATAL("SkiaCanvas can't directly draw RenderNodes");
833}
Derek Sollenberger6f485562015-07-30 10:00:39 -0400834
John Reckcd1c3eb2016-04-14 10:38:54 -0700835void SkiaCanvas::callDrawGLFunction(Functor* functor,
John Reck1bcacfd2017-11-03 10:12:19 -0700836 uirenderer::GlFunctorLifecycleListener* listener) {
Derek Sollenbergerc1908132016-07-15 10:28:16 -0400837 LOG_ALWAYS_FATAL("SkiaCanvas can't directly draw GL Content");
838}
Derek Sollenberger6f485562015-07-30 10:00:39 -0400839
John Reck1bcacfd2017-11-03 10:12:19 -0700840} // namespace android