blob: 93fa4994ecb91a2a58923f72ea4cde40d8a7930c [file] [log] [blame]
Cary Clark8032b982017-07-28 11:04:54 -04001#Topic Canvas
Cary Clark137b8742018-05-30 09:21:49 -04002#Alias Canvas_Reference ##
Cary Clark8032b982017-07-28 11:04:54 -04003
Cary Clarke4aa3712017-09-15 02:56:12 -04004#Class SkCanvas
5
Cary Clark61313f32018-10-08 14:57:48 -04006#Code
7#Populate
8##
9
Cary Clark8032b982017-07-28 11:04:54 -040010Canvas provides an interface for drawing, and how the drawing is clipped and transformed.
11Canvas contains a stack of Matrix and Clip values.
12
13Canvas and Paint together provide the state to draw into Surface or Device.
Cary Clarkbad5ad72017-08-03 17:14:08 -040014Each Canvas draw call transforms the geometry of the object by the concatenation of all
15Matrix values in the stack. The transformed geometry is clipped by the intersection
16of all of Clip values in the stack. The Canvas draw calls use Paint to supply drawing
17state such as Color, Typeface, text size, stroke width, Shader and so on.
Cary Clark8032b982017-07-28 11:04:54 -040018
Herb Derbyefe39bc2018-05-01 17:06:20 -040019To draw to a pixel-based destination, create Raster_Surface or GPU_Surface.
20Request Canvas from Surface to obtain the interface to draw.
21Canvas generated by Raster_Surface draws to memory visible to the CPU.
Cary Clark8032b982017-07-28 11:04:54 -040022Canvas generated by GPU_Surface uses Vulkan or OpenGL to draw to the GPU.
23
Cary Clarkbad5ad72017-08-03 17:14:08 -040024To draw to a document, obtain Canvas from SVG_Canvas, Document_PDF, or Picture_Recorder.
Cary Clark09d80c02018-10-31 12:14:03 -040025Document based Canvas and other Canvas subclasses reference Device describing the
Cary Clarkbad5ad72017-08-03 17:14:08 -040026destination.
27
Cary Clark8032b982017-07-28 11:04:54 -040028Canvas can be constructed to draw to Bitmap without first creating Raster_Surface.
Herb Derbyefe39bc2018-05-01 17:06:20 -040029This approach may be deprecated in the future.
Cary Clark8032b982017-07-28 11:04:54 -040030
Cary Clark09d80c02018-10-31 12:14:03 -040031Canvas may be created directly when no Surface is required; some Canvas methods
32implicitly create Raster_Surface.
Cary Clark8032b982017-07-28 11:04:54 -040033
34# ------------------------------------------------------------------------------
35
Cary Clarka560c472017-11-27 10:44:06 -050036#Method static std::unique_ptr<SkCanvas> MakeRasterDirect(const SkImageInfo& info, void* pixels,
37 size_t rowBytes,
38 const SkSurfaceProps* props = nullptr)
Cary Clark61313f32018-10-08 14:57:48 -040039#In Constructors
Cary Clarkab2621d2018-01-30 10:08:57 -050040#Line # creates from SkImageInfo and Pixel_Storage ##
Cary Clark09d80c02018-10-31 12:14:03 -040041#Populate
Cary Clark8032b982017-07-28 11:04:54 -040042
43#Example
44 #Description
45 Allocates a three by three bitmap, clears it to white, and draws a black pixel
46 in the center.
47 ##
Cary Clarkbad5ad72017-08-03 17:14:08 -040048void draw(SkCanvas* ) {
Cary Clarkce101242017-09-01 15:51:02 -040049 SkImageInfo info = SkImageInfo::MakeN32Premul(3, 3); // device aligned, 32 bpp, Premultiplied
Cary Clarkbad5ad72017-08-03 17:14:08 -040050 const size_t minRowBytes = info.minRowBytes(); // bytes used by one bitmap row
Cary Clarka560c472017-11-27 10:44:06 -050051 const size_t size = info.computeMinByteSize(); // bytes used by all rows
Cary Clarkbad5ad72017-08-03 17:14:08 -040052 SkAutoTMalloc<SkPMColor> storage(size); // allocate storage for pixels
53 SkPMColor* pixels = storage.get(); // get pointer to allocated storage
54 // create a SkCanvas backed by a raster device, and delete it when the
55 // function goes out of scope.
56 std::unique_ptr<SkCanvas> canvas = SkCanvas::MakeRasterDirect(info, pixels, minRowBytes);
Cary Clarkce101242017-09-01 15:51:02 -040057 canvas->clear(SK_ColorWHITE); // white is Unpremultiplied, in ARGB order
Cary Clarkbad5ad72017-08-03 17:14:08 -040058 canvas->flush(); // ensure that pixels are cleared
Cary Clarkce101242017-09-01 15:51:02 -040059 SkPMColor pmWhite = pixels[0]; // the Premultiplied format may vary
Cary Clarkbad5ad72017-08-03 17:14:08 -040060 SkPaint paint; // by default, draws black
61 canvas->drawPoint(1, 1, paint); // draw in the center
62 canvas->flush(); // ensure that point was drawn
63 for (int y = 0; y < info.height(); ++y) {
64 for (int x = 0; x < info.width(); ++x) {
65 SkDebugf("%c", *pixels++ == pmWhite ? '-' : 'x');
66 }
67 SkDebugf("\n");
68 }
Cary Clark8032b982017-07-28 11:04:54 -040069}
70 #StdOut
71 ---
72 -x-
73 ---
74 ##
75##
76
Cary Clark8032b982017-07-28 11:04:54 -040077#SeeAlso MakeRasterDirectN32 SkSurface::MakeRasterDirect
Cary Clark2ade9972017-11-02 17:49:34 -040078
Cary Clark8032b982017-07-28 11:04:54 -040079##
80
81# ------------------------------------------------------------------------------
82
83#Method static std::unique_ptr<SkCanvas> MakeRasterDirectN32(int width, int height, SkPMColor* pixels,
Herb Derbyefe39bc2018-05-01 17:06:20 -040084 size_t rowBytes)
Cary Clark61313f32018-10-08 14:57:48 -040085#In Constructors
Cary Clarkab2621d2018-01-30 10:08:57 -050086#Line # creates from image data and Pixel_Storage ##
Cary Clark09d80c02018-10-31 12:14:03 -040087#Populate
Cary Clark8032b982017-07-28 11:04:54 -040088
89#Example
90 #Description
91 Allocates a three by three bitmap, clears it to white, and draws a black pixel
92 in the center.
93 ##
94void draw(SkCanvas* ) {
95 const int width = 3;
96 const int height = 3;
Cary Clark77b3f3a2018-11-07 14:59:03 -050097 SkPMColor pixels[height][width]; // allocate a 3 by 3 Premultiplied bitmap on the stack
Cary Clark8032b982017-07-28 11:04:54 -040098 // create a SkCanvas backed by a raster device, and delete it when the
99 // function goes out of scope.
100 std::unique_ptr<SkCanvas> canvas = SkCanvas::MakeRasterDirectN32(
101 width,
102 height,
Cary Clarkbc5697d2017-10-04 14:31:33 -0400103 pixels[0], // top-left of the bitmap
Cary Clark8032b982017-07-28 11:04:54 -0400104 sizeof(pixels[0])); // byte width of the each row
Cary Clark682c58d2018-05-16 07:07:07 -0400105 // write a Premultiplied value for white into all pixels in the bitmap
Cary Clark8032b982017-07-28 11:04:54 -0400106 canvas->clear(SK_ColorWHITE);
Cary Clarkce101242017-09-01 15:51:02 -0400107 SkPMColor pmWhite = pixels[0][0]; // the Premultiplied format may vary
Cary Clark8032b982017-07-28 11:04:54 -0400108 SkPaint paint; // by default, draws black
109 canvas->drawPoint(1, 1, paint); // draw in the center
110 canvas->flush(); // ensure that pixels is ready to be read
111 for (int y = 0; y < height; ++y) {
112 for (int x = 0; x < width; ++x) {
113 SkDebugf("%c", pixels[y][x] == pmWhite ? '-' : 'x');
114 }
115 SkDebugf("\n");
116 }
117}
118 #StdOut
119 ---
120 -x-
121 ---
122 ##
123##
124
Cary Clark2ade9972017-11-02 17:49:34 -0400125#SeeAlso MakeRasterDirect SkSurface::MakeRasterDirect SkImageInfo::MakeN32Premul
Cary Clark8032b982017-07-28 11:04:54 -0400126
127##
128
129# ------------------------------------------------------------------------------
130
131#Method SkCanvas()
132
Cary Clarkab2621d2018-01-30 10:08:57 -0500133#Line # creates with no Surface, no dimensions ##
Cary Clark09d80c02018-10-31 12:14:03 -0400134#Populate
Cary Clark8032b982017-07-28 11:04:54 -0400135
136#Example
137
138#Description
139Passes a placeholder to a function that requires one.
140##
141
142#Function
Mike Reed07055a32018-10-23 13:57:46 -0400143static void check_for_rotated_ctm(const SkCanvas* canvas) {
Cary Clarkbad5ad72017-08-03 17:14:08 -0400144 const SkMatrix& matrix = canvas->getTotalMatrix();
Mike Reed07055a32018-10-23 13:57:46 -0400145 SkDebugf("rect stays rect is %s\n", matrix.rectStaysRect() ? "true" : "false");
Cary Clarkbad5ad72017-08-03 17:14:08 -0400146}
147
Cary Clark8032b982017-07-28 11:04:54 -0400148##
Cary Clarkbad5ad72017-08-03 17:14:08 -0400149void draw(SkCanvas* canvas) {
Mike Reed07055a32018-10-23 13:57:46 -0400150 check_for_rotated_ctm(canvas);
151 canvas->rotate(30);
152 check_for_rotated_ctm(canvas);
153
154 SkCanvas defaultCanvas;
155 check_for_rotated_ctm(&defaultCanvas);
Cary Clarkbad5ad72017-08-03 17:14:08 -0400156}
Cary Clark8032b982017-07-28 11:04:54 -0400157
158 #StdOut
Mike Reed07055a32018-10-23 13:57:46 -0400159 rect stays rect is true
160 rect stays rect is false
161 rect stays rect is true
Cary Clark8032b982017-07-28 11:04:54 -0400162 ##
163##
164
Cary Clark2ade9972017-11-02 17:49:34 -0400165#SeeAlso MakeRasterDirect SkRasterHandleAllocator::MakeCanvas SkSurface::getCanvas SkCreateColorSpaceXformCanvas
Cary Clark8032b982017-07-28 11:04:54 -0400166
167##
168
169# ------------------------------------------------------------------------------
170
Cary Clark73fa9722017-08-29 17:36:51 -0400171#Method SkCanvas(int width, int height, const SkSurfaceProps* props = nullptr)
Cary Clark8032b982017-07-28 11:04:54 -0400172
Cary Clark682c58d2018-05-16 07:07:07 -0400173#Line # creates with no Surface, set dimensions, Surface_Properties ##
Cary Clark09d80c02018-10-31 12:14:03 -0400174#Populate
Cary Clark8032b982017-07-28 11:04:54 -0400175
176#Example
177 SkCanvas canvas(10, 20); // 10 units wide, 20 units high
178 canvas.clipRect(SkRect::MakeXYWH(30, 40, 5, 10)); // clip is outside canvas' device
179 SkDebugf("canvas %s empty\n", canvas.getDeviceClipBounds().isEmpty() ? "is" : "is not");
180
181 #StdOut
182 canvas is empty
183 ##
184##
185
Cary Clark2ade9972017-11-02 17:49:34 -0400186#SeeAlso MakeRasterDirect SkSurfaceProps SkPixelGeometry SkCreateColorSpaceXformCanvas
Cary Clark8032b982017-07-28 11:04:54 -0400187
188##
189
190# ------------------------------------------------------------------------------
191
Cary Clark8032b982017-07-28 11:04:54 -0400192#Method explicit SkCanvas(const SkBitmap& bitmap)
193
Cary Clarkab2621d2018-01-30 10:08:57 -0500194#Line # uses existing Bitmap ##
Cary Clark09d80c02018-10-31 12:14:03 -0400195#Populate
Cary Clark8032b982017-07-28 11:04:54 -0400196
197#Example
198#Description
199The actual output depends on the installed fonts.
200##
201 SkBitmap bitmap;
202 // create a bitmap 5 wide and 11 high
203 bitmap.allocPixels(SkImageInfo::MakeN32Premul(5, 11));
204 SkCanvas canvas(bitmap);
Cary Clarkce101242017-09-01 15:51:02 -0400205 canvas.clear(SK_ColorWHITE); // white is Unpremultiplied, in ARGB order
Cary Clark8032b982017-07-28 11:04:54 -0400206 SkPixmap pixmap; // provides guaranteed access to the drawn pixels
207 if (!canvas.peekPixels(&pixmap)) {
208 SkDebugf("peekPixels should never fail.\n");
209 }
Cary Clarkbc5697d2017-10-04 14:31:33 -0400210 const SkPMColor* pixels = pixmap.addr32(); // points to top-left of bitmap
Cary Clarkce101242017-09-01 15:51:02 -0400211 SkPMColor pmWhite = pixels[0]; // the Premultiplied format may vary
Cary Clark8032b982017-07-28 11:04:54 -0400212 SkPaint paint; // by default, draws black, 12 point text
Mike Reed33535cb2019-01-09 11:25:18 -0500213 canvas.drawString("!", 1, 10, SkFont(), paint); // 1 char at baseline (1, 10)
Cary Clark8032b982017-07-28 11:04:54 -0400214 for (int y = 0; y < bitmap.height(); ++y) {
215 for (int x = 0; x < bitmap.width(); ++x) {
216 SkDebugf("%c", *pixels++ == pmWhite ? '-' : 'x');
217 }
218 SkDebugf("\n");
219 }
220
221 #StdOut
Cary Clarkd0530ba2017-09-14 11:25:39 -0400222 -----
223 ---x-
224 ---x-
225 ---x-
226 ---x-
227 ---x-
228 ---x-
229 -----
230 ---x-
231 ---x-
Cary Clark8032b982017-07-28 11:04:54 -0400232 -----
233 #StdOut ##
234##
235
Cary Clark2ade9972017-11-02 17:49:34 -0400236#SeeAlso MakeRasterDirect SkRasterHandleAllocator::MakeCanvas SkSurface::getCanvas SkCreateColorSpaceXformCanvas
Cary Clark8032b982017-07-28 11:04:54 -0400237
238##
239
Cary Clark8032b982017-07-28 11:04:54 -0400240# ------------------------------------------------------------------------------
241
242#Method SkCanvas(const SkBitmap& bitmap, const SkSurfaceProps& props)
243
Cary Clarkab2621d2018-01-30 10:08:57 -0500244#Line # uses existing Bitmap and Surface_Properties ##
Cary Clark09d80c02018-10-31 12:14:03 -0400245#Populate
Cary Clark8032b982017-07-28 11:04:54 -0400246
247#Example
248#Description
249The actual output depends on the installed fonts.
250##
251 SkBitmap bitmap;
252 // create a bitmap 5 wide and 11 high
253 bitmap.allocPixels(SkImageInfo::MakeN32Premul(5, 11));
254 SkCanvas canvas(bitmap, SkSurfaceProps(0, kUnknown_SkPixelGeometry));
Cary Clarkce101242017-09-01 15:51:02 -0400255 canvas.clear(SK_ColorWHITE); // white is Unpremultiplied, in ARGB order
Cary Clark8032b982017-07-28 11:04:54 -0400256 SkPixmap pixmap; // provides guaranteed access to the drawn pixels
257 if (!canvas.peekPixels(&pixmap)) {
258 SkDebugf("peekPixels should never fail.\n");
259 }
Cary Clarkbc5697d2017-10-04 14:31:33 -0400260 const SkPMColor* pixels = pixmap.addr32(); // points to top-left of bitmap
Cary Clarkce101242017-09-01 15:51:02 -0400261 SkPMColor pmWhite = pixels[0]; // the Premultiplied format may vary
Cary Clark8032b982017-07-28 11:04:54 -0400262 SkPaint paint; // by default, draws black, 12 point text
Mike Reed33535cb2019-01-09 11:25:18 -0500263 canvas.drawString("!", 1, 10, SkFont(), paint); // 1 char at baseline (1, 10)
Cary Clark8032b982017-07-28 11:04:54 -0400264 for (int y = 0; y < bitmap.height(); ++y) {
265 for (int x = 0; x < bitmap.width(); ++x) {
266 SkDebugf("%c", *pixels++ == pmWhite ? '-' : 'x');
267 }
268 SkDebugf("\n");
269 }
270
271 #StdOut
272 -----
273 ---x-
274 ---x-
275 ---x-
276 ---x-
277 ---x-
278 ---x-
279 -----
280 ---x-
281 ---x-
282 -----
283 #StdOut ##
284##
285
Cary Clark2ade9972017-11-02 17:49:34 -0400286#SeeAlso MakeRasterDirect SkRasterHandleAllocator::MakeCanvas SkSurface::getCanvas SkCreateColorSpaceXformCanvas
Cary Clark8032b982017-07-28 11:04:54 -0400287
288##
289
290# ------------------------------------------------------------------------------
291
292#Method virtual ~SkCanvas()
293
Cary Clarkab2621d2018-01-30 10:08:57 -0500294#Line # draws saved Layers, frees resources ##
Cary Clark09d80c02018-10-31 12:14:03 -0400295#Populate
Cary Clark8032b982017-07-28 11:04:54 -0400296
297#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -0400298#Description
Cary Clarkce101242017-09-01 15:51:02 -0400299Canvas Layer draws into bitmap. saveLayerAlpha sets up an additional
300drawing surface that blends with the bitmap. When Layer goes out of
Cary Clark09d80c02018-10-31 12:14:03 -0400301scope, Layer destructor is called. The saved Layer is restored, drawing
Cary Clarkbad5ad72017-08-03 17:14:08 -0400302transparent letters.
303##
Cary Clarkf05bdda2017-08-24 12:59:48 -0400304void draw(SkCanvas* canvas) {
305 SkBitmap bitmap;
306 bitmap.allocPixels(SkImageInfo::MakeN32Premul(200, 200));
307 {
308 SkCanvas offscreen(bitmap);
309 SkPaint paint;
Mike Reed33535cb2019-01-09 11:25:18 -0500310 SkFont font(nullptr, 100);
311 offscreen.drawString("ABC", 20, 160, font, paint);
Cary Clarkf05bdda2017-08-24 12:59:48 -0400312 SkRect layerBounds = SkRect::MakeXYWH(32, 32, 192, 192);
313 offscreen.saveLayerAlpha(&layerBounds, 128);
314 offscreen.clear(SK_ColorWHITE);
Mike Reed33535cb2019-01-09 11:25:18 -0500315 offscreen.drawString("DEF", 20, 160, font, paint);
Cary Clarkf05bdda2017-08-24 12:59:48 -0400316 }
317 canvas->drawBitmap(bitmap, 0, 0, nullptr);
Cary Clarkbad5ad72017-08-03 17:14:08 -0400318}
Cary Clark8032b982017-07-28 11:04:54 -0400319##
320
Cary Clarkbad5ad72017-08-03 17:14:08 -0400321#SeeAlso State_Stack
Cary Clark8032b982017-07-28 11:04:54 -0400322
323##
324
325# ------------------------------------------------------------------------------
Cary Clark78de7512018-02-07 07:27:09 -0500326#Subtopic Property
Cary Clark78de7512018-02-07 07:27:09 -0500327#Line # metrics and attributes ##
328##
Cary Clark8032b982017-07-28 11:04:54 -0400329
Cary Clark8032b982017-07-28 11:04:54 -0400330#Method SkImageInfo imageInfo() const
Cary Clark78de7512018-02-07 07:27:09 -0500331#In Property
Cary Clarkab2621d2018-01-30 10:08:57 -0500332#Line # returns Image_Info for Canvas ##
Cary Clark09d80c02018-10-31 12:14:03 -0400333#Populate
Cary Clark8032b982017-07-28 11:04:54 -0400334
335#Example
Cary Clarkf05bdda2017-08-24 12:59:48 -0400336 SkCanvas emptyCanvas;
337 SkImageInfo canvasInfo = emptyCanvas.imageInfo();
338 SkImageInfo emptyInfo;
339 SkDebugf("emptyInfo %c= canvasInfo\n", emptyInfo == canvasInfo ? '=' : '!');
340
341 #StdOut
342 emptyInfo == canvasInfo
343 ##
Cary Clark8032b982017-07-28 11:04:54 -0400344##
345
Cary Clark2ade9972017-11-02 17:49:34 -0400346#SeeAlso SkImageInfo MakeRasterDirect makeSurface
Cary Clark8032b982017-07-28 11:04:54 -0400347
348##
349
350# ------------------------------------------------------------------------------
351
352#Method bool getProps(SkSurfaceProps* props) const
Cary Clark78de7512018-02-07 07:27:09 -0500353#In Property
Cary Clarkab2621d2018-01-30 10:08:57 -0500354#Line # copies Surface_Properties if available ##
Cary Clark09d80c02018-10-31 12:14:03 -0400355#Populate
Cary Clark8032b982017-07-28 11:04:54 -0400356
357#Example
358 SkBitmap bitmap;
359 SkCanvas canvas(bitmap, SkSurfaceProps(0, kRGB_V_SkPixelGeometry));
360 SkSurfaceProps surfaceProps(0, kUnknown_SkPixelGeometry);
361 SkDebugf("isRGB:%d\n", SkPixelGeometryIsRGB(surfaceProps.pixelGeometry()));
362 if (!canvas.getProps(&surfaceProps)) {
363 SkDebugf("getProps failed unexpectedly.\n");
364 }
365 SkDebugf("isRGB:%d\n", SkPixelGeometryIsRGB(surfaceProps.pixelGeometry()));
366
367 #StdOut
368 isRGB:0
369 isRGB:1
370 #StdOut ##
371##
372
Cary Clark2ade9972017-11-02 17:49:34 -0400373#SeeAlso SkSurfaceProps makeSurface
Cary Clark8032b982017-07-28 11:04:54 -0400374
375##
376
377# ------------------------------------------------------------------------------
Cary Clark78de7512018-02-07 07:27:09 -0500378#Subtopic Utility
Cary Clark78de7512018-02-07 07:27:09 -0500379#Line # rarely called management functions ##
380##
Cary Clark8032b982017-07-28 11:04:54 -0400381
382#Method void flush()
Cary Clark78de7512018-02-07 07:27:09 -0500383#In Utility
Cary Clarkab2621d2018-01-30 10:08:57 -0500384#Line # triggers execution of all pending draw operations ##
Cary Clark09d80c02018-10-31 12:14:03 -0400385#Populate
Cary Clark2ade9972017-11-02 17:49:34 -0400386
Cary Clark08895c42018-02-01 09:37:32 -0500387#NoExample
Cary Clark8032b982017-07-28 11:04:54 -0400388##
389
Cary Clarkabaffd82018-11-15 08:25:12 -0500390#SeeAlso peekPixels SkSurface::flush GrContext::flush GrContext::abandonContext
Cary Clark8032b982017-07-28 11:04:54 -0400391
392##
393
394# ------------------------------------------------------------------------------
395
396#Method virtual SkISize getBaseLayerSize() const
Cary Clark78de7512018-02-07 07:27:09 -0500397#In Property
Cary Clarkab2621d2018-01-30 10:08:57 -0500398#Line # returns size of base Layer in global coordinates ##
Cary Clark09d80c02018-10-31 12:14:03 -0400399#Populate
Cary Clark8032b982017-07-28 11:04:54 -0400400
401#Example
402 SkBitmap bitmap;
403 bitmap.allocPixels(SkImageInfo::MakeN32Premul(20, 30));
404 SkCanvas canvas(bitmap, SkSurfaceProps(0, kUnknown_SkPixelGeometry));
405 canvas.clipRect(SkRect::MakeWH(10, 40));
406 SkIRect clipDeviceBounds = canvas.getDeviceClipBounds();
407 if (clipDeviceBounds.isEmpty()) {
408 SkDebugf("Empty clip bounds is unexpected!\n");
409 }
410 SkDebugf("clip=%d,%d\n", clipDeviceBounds.width(), clipDeviceBounds.height());
411 SkISize baseLayerSize = canvas.getBaseLayerSize();
412 SkDebugf("size=%d,%d\n", baseLayerSize.width(), baseLayerSize.height());
413
414 #StdOut
415 clip=10,30
416 size=20,30
417 ##
418##
419
420#ToDo is this the same as the width and height of surface? ##
421
Cary Clark2ade9972017-11-02 17:49:34 -0400422#SeeAlso getDeviceClipBounds
423
Cary Clark8032b982017-07-28 11:04:54 -0400424##
425
426# ------------------------------------------------------------------------------
427
428#Method sk_sp<SkSurface> makeSurface(const SkImageInfo& info, const SkSurfaceProps* props = nullptr)
Cary Clark61313f32018-10-08 14:57:48 -0400429#In Constructors
Cary Clarkab2621d2018-01-30 10:08:57 -0500430#Line # creates Surface matching SkImageInfo and SkSurfaceProps ##
Cary Clark09d80c02018-10-31 12:14:03 -0400431#Populate
Cary Clark8032b982017-07-28 11:04:54 -0400432
433#Example
434 sk_sp<SkSurface> surface = SkSurface::MakeRasterN32Premul(5, 6);
435 SkCanvas* smallCanvas = surface->getCanvas();
436 SkImageInfo imageInfo = SkImageInfo::MakeN32Premul(3, 4);
437 sk_sp<SkSurface> compatible = smallCanvas->makeSurface(imageInfo);
438 SkDebugf("compatible %c= nullptr\n", compatible == nullptr ? '=' : '!');
439 SkDebugf("size = %d, %d\n", compatible->width(), compatible->height());
440
441 #StdOut
Cary Clarkbad5ad72017-08-03 17:14:08 -0400442 compatible != nullptr
Cary Clark8032b982017-07-28 11:04:54 -0400443 size = 3, 4
444 ##
445##
446
Cary Clark2ade9972017-11-02 17:49:34 -0400447#SeeAlso SkSurface SkSurface::makeSurface SkImageInfo SkSurfaceProps
Cary Clark8032b982017-07-28 11:04:54 -0400448
449##
450
451# ------------------------------------------------------------------------------
452
453#Method virtual GrContext* getGrContext()
Cary Clark78de7512018-02-07 07:27:09 -0500454#In Property
Cary Clarkab2621d2018-01-30 10:08:57 -0500455#Line # returns GPU_Context of the GPU_Surface ##
Cary Clark09d80c02018-10-31 12:14:03 -0400456#Populate
Cary Clark8032b982017-07-28 11:04:54 -0400457
458#Example
459void draw(SkCanvas* canvas) {
460 if (canvas->getGrContext()) {
461 canvas->clear(SK_ColorRED);
462 } else {
463 canvas->clear(SK_ColorBLUE);
464 }
465}
466##
467
468#ToDo fiddle should show both CPU and GPU out ##
469
Herb Derbyefe39bc2018-05-01 17:06:20 -0400470#SeeAlso GrContext
Cary Clark2ade9972017-11-02 17:49:34 -0400471
Cary Clark8032b982017-07-28 11:04:54 -0400472##
473
474# ------------------------------------------------------------------------------
475
Cary Clark73fa9722017-08-29 17:36:51 -0400476#Method void* accessTopLayerPixels(SkImageInfo* info, size_t* rowBytes, SkIPoint* origin = nullptr)
Cary Clark78de7512018-02-07 07:27:09 -0500477#In Utility
478#In Property
Cary Clarkab2621d2018-01-30 10:08:57 -0500479#Line # returns writable pixel access if available ##
Cary Clark09d80c02018-10-31 12:14:03 -0400480#Populate
Cary Clark8032b982017-07-28 11:04:54 -0400481
482#Example
483void draw(SkCanvas* canvas) {
484 if (canvas->accessTopLayerPixels(nullptr, nullptr)) {
485 canvas->clear(SK_ColorRED);
486 } else {
487 canvas->clear(SK_ColorBLUE);
488 }
489}
490##
491
492#Example
493#Description
Cary Clarkce101242017-09-01 15:51:02 -0400494Draws "ABC" on the device. Then draws "DEF" in Layer, and reads
495Layer to add a large dotted "DEF". Finally blends Layer with the
Herb Derbyefe39bc2018-05-01 17:06:20 -0400496device.
Cary Clark8032b982017-07-28 11:04:54 -0400497
Cary Clarkce101242017-09-01 15:51:02 -0400498The Layer and blended result appear on the CPU and GPU but the large dotted
Cary Clark8032b982017-07-28 11:04:54 -0400499"DEF" appear only on the CPU.
500##
501void draw(SkCanvas* canvas) {
Cary Clarkbad5ad72017-08-03 17:14:08 -0400502 SkPaint paint;
Mike Reed33535cb2019-01-09 11:25:18 -0500503 SkFont font(nullptr, 100);
504 canvas->drawString("ABC", 20, 160, font, paint);
Cary Clarkbad5ad72017-08-03 17:14:08 -0400505 SkRect layerBounds = SkRect::MakeXYWH(32, 32, 192, 192);
506 canvas->saveLayerAlpha(&layerBounds, 128);
507 canvas->clear(SK_ColorWHITE);
Mike Reed33535cb2019-01-09 11:25:18 -0500508 canvas->drawString("DEF", 20, 160, font, paint);
Cary Clarkbad5ad72017-08-03 17:14:08 -0400509 SkImageInfo imageInfo;
510 size_t rowBytes;
511 SkIPoint origin;
512 uint32_t* access = (uint32_t*) canvas->accessTopLayerPixels(&imageInfo, &rowBytes, &origin);
513 if (access) {
514 int h = imageInfo.height();
515 int v = imageInfo.width();
516 int rowWords = rowBytes / sizeof(uint32_t);
517 for (int y = 0; y < h; ++y) {
518 int newY = (y - h / 2) * 2 + h / 2;
519 if (newY < 0 || newY >= h) {
520 continue;
521 }
522 for (int x = 0; x < v; ++x) {
523 int newX = (x - v / 2) * 2 + v / 2;
524 if (newX < 0 || newX >= v) {
525 continue;
526 }
527 if (access[y * rowWords + x] == SK_ColorBLACK) {
528 access[newY * rowWords + newX] = SK_ColorGRAY;
529 }
530 }
531 }
532
533 }
Cary Clark8032b982017-07-28 11:04:54 -0400534 canvas->restore();
535}
536##
537
538#ToDo there are no callers of this that I can find. Deprecate? ##
539#ToDo fiddle should show both CPU and GPU out ##
540
Cary Clark2ade9972017-11-02 17:49:34 -0400541#SeeAlso SkImageInfo SkPixmap
542
Cary Clark8032b982017-07-28 11:04:54 -0400543##
544
545# ------------------------------------------------------------------------------
546
547#Method SkRasterHandleAllocator::Handle accessTopRasterHandle() const
Cary Clark78de7512018-02-07 07:27:09 -0500548#In Utility
549#In Property
Cary Clarkab2621d2018-01-30 10:08:57 -0500550#Line # returns context that tracks Clip and Matrix ##
Cary Clark09d80c02018-10-31 12:14:03 -0400551#Populate
Cary Clark8032b982017-07-28 11:04:54 -0400552
553#Example
554#Description
555#ToDo ##
556##
557#Function
558 static void DeleteCallback(void*, void* context) {
559 delete (char*) context;
560 }
561
562 class CustomAllocator : public SkRasterHandleAllocator {
563 public:
564 bool allocHandle(const SkImageInfo& info, Rec* rec) override {
565 char* context = new char[4]{'s', 'k', 'i', 'a'};
566 rec->fReleaseProc = DeleteCallback;
567 rec->fReleaseCtx = context;
568 rec->fHandle = context;
569 rec->fPixels = context;
570 rec->fRowBytes = 4;
571 return true;
572 }
573
574 void updateHandle(Handle handle, const SkMatrix& ctm, const SkIRect& clip_bounds) override {
575 // apply canvas matrix and clip to custom environment
576 }
577 };
578
579##
580 void draw(SkCanvas* canvas) {
581 const SkImageInfo info = SkImageInfo::MakeN32Premul(1, 1);
582 std::unique_ptr<SkCanvas> c2 =
583 SkRasterHandleAllocator::MakeCanvas(std::unique_ptr<CustomAllocator>(
584 new CustomAllocator()), info);
585 char* context = (char*) c2->accessTopRasterHandle();
586 SkDebugf("context = %.4s\n", context);
587
588 }
589 #StdOut
590 context = skia
591 ##
592 #ToDo skstd::make_unique could not be used because def is private -- note to fix in c++14? ##
593##
594
595#SeeAlso SkRasterHandleAllocator
596
597##
598
599# ------------------------------------------------------------------------------
Cary Clark78de7512018-02-07 07:27:09 -0500600#Subtopic Pixels
Cary Clark78de7512018-02-07 07:27:09 -0500601#Line # read and write pixel values ##
602##
Cary Clark8032b982017-07-28 11:04:54 -0400603
604#Method bool peekPixels(SkPixmap* pixmap)
Cary Clark78de7512018-02-07 07:27:09 -0500605#In Pixels
Cary Clarkab2621d2018-01-30 10:08:57 -0500606#Line # returns if Canvas has direct access to its pixels ##
Cary Clark09d80c02018-10-31 12:14:03 -0400607#Populate
Cary Clark8032b982017-07-28 11:04:54 -0400608
609#Example
610 SkPixmap pixmap;
611 if (canvas->peekPixels(&pixmap)) {
612 SkDebugf("width=%d height=%d\n", pixmap.bounds().width(), pixmap.bounds().height());
613 }
614 #StdOut
615 width=256 height=256
616 ##
617##
618
Cary Clark2ade9972017-11-02 17:49:34 -0400619#SeeAlso readPixels SkBitmap::peekPixels SkImage::peekPixels SkSurface::peekPixels
620
Cary Clark8032b982017-07-28 11:04:54 -0400621##
622
623# ------------------------------------------------------------------------------
624
625#Method bool readPixels(const SkImageInfo& dstInfo, void* dstPixels, size_t dstRowBytes,
626 int srcX, int srcY)
Cary Clark78de7512018-02-07 07:27:09 -0500627#In Pixels
Cary Clarkab2621d2018-01-30 10:08:57 -0500628#Line # copies and converts rectangle of pixels from Canvas ##
Cary Clark8032b982017-07-28 11:04:54 -0400629
Cary Clark154beea2017-10-26 07:58:48 -0400630Copies Rect of pixels from Canvas into dstPixels. Matrix and Clip are
Herb Derbyefe39bc2018-05-01 17:06:20 -0400631ignored.
Cary Clark6fc50412017-09-21 12:31:06 -0400632
Cary Clarka560c472017-11-27 10:44:06 -0500633Source Rect corners are (srcX, srcY) and (imageInfo().width(), imageInfo().height()).
634Destination Rect corners are (0, 0) and (dstInfo.width(), dstInfo.height()).
Cary Clark6fc50412017-09-21 12:31:06 -0400635Copies each readable pixel intersecting both rectangles, without scaling,
Cary Clarkf05bdda2017-08-24 12:59:48 -0400636converting to dstInfo.colorType() and dstInfo.alphaType() if required.
Cary Clark8032b982017-07-28 11:04:54 -0400637
Cary Clarkf05bdda2017-08-24 12:59:48 -0400638Pixels are readable when Device is raster, or backed by a GPU.
639Pixels are not readable when SkCanvas is returned by SkDocument::beginPage,
640returned by SkPictureRecorder::beginRecording, or Canvas is the base of a utility
Brian Osman46fe9c72018-03-09 15:44:34 -0500641class like SkDebugCanvas.
Cary Clark8032b982017-07-28 11:04:54 -0400642
Cary Clarkf05bdda2017-08-24 12:59:48 -0400643The destination pixel storage must be allocated by the caller.
Cary Clark8032b982017-07-28 11:04:54 -0400644
Cary Clark2dc84ad2018-01-26 12:56:22 -0500645Pixel values are converted only if Color_Type and Alpha_Type
Cary Clarkf05bdda2017-08-24 12:59:48 -0400646do not match. Only pixels within both source and destination rectangles
Cary Clark154beea2017-10-26 07:58:48 -0400647are copied. dstPixels contents outside Rect intersection are unchanged.
Cary Clarkf05bdda2017-08-24 12:59:48 -0400648
649Pass negative values for srcX or srcY to offset pixels across or down destination.
Cary Clark8032b982017-07-28 11:04:54 -0400650
651Does not copy, and returns false if:
652
653#List
Cary Clarkf05bdda2017-08-24 12:59:48 -0400654# Source and destination rectangles do not intersect. ##
655# Canvas pixels could not be converted to dstInfo.colorType() or dstInfo.alphaType(). ##
656# Canvas pixels are not readable; for instance, Canvas is document-based. ##
Cary Clark8032b982017-07-28 11:04:54 -0400657# dstRowBytes is too small to contain one row of pixels. ##
658##
659
Cary Clark2dc84ad2018-01-26 12:56:22 -0500660#Param dstInfo width, height, Color_Type, and Alpha_Type of dstPixels ##
Cary Clarkf05bdda2017-08-24 12:59:48 -0400661#Param dstPixels storage for pixels; dstInfo.height() times dstRowBytes, or larger ##
662#Param dstRowBytes size of one destination row; dstInfo.width() times pixel size, or larger ##
Cary Clark5538c132018-06-14 12:28:14 -0400663#Param srcX offset into readable pixels on x-axis; may be negative ##
664#Param srcY offset into readable pixels on y-axis; may be negative ##
Cary Clark8032b982017-07-28 11:04:54 -0400665
Cary Clarkbad5ad72017-08-03 17:14:08 -0400666#Return true if pixels were copied ##
Cary Clark8032b982017-07-28 11:04:54 -0400667
668#Example
Cary Clarkf05bdda2017-08-24 12:59:48 -0400669#Width 64
670#Height 64
671#Description
672 A black circle drawn on a blue background provides an image to copy.
673 readPixels copies one quarter of the canvas into each of the four corners.
Cary Clarka560c472017-11-27 10:44:06 -0500674 The copied quarter circles overdraw the original circle.
Cary Clarkf05bdda2017-08-24 12:59:48 -0400675##
676 canvas->clear(SK_ColorBLUE);
677 SkPaint paint;
678 canvas->drawCircle(32, 32, 28, paint);
679 SkImageInfo info = SkImageInfo::Make(64, 64, kBGRA_8888_SkColorType, kPremul_SkAlphaType);
680 sk_sp<SkData> data(SkData::MakeUninitialized(info.minRowBytes() * info.height()));
681 sk_bzero(data->writable_data(), info.minRowBytes() * info.height());
682 for (int x : { 32, -32 } ) {
683 for (int y : { 32, -32 } ) {
684 canvas->readPixels(info, data->writable_data(), info.minRowBytes(), x, y);
Herb Derbyefe39bc2018-05-01 17:06:20 -0400685 }
Cary Clarkf05bdda2017-08-24 12:59:48 -0400686 }
687 sk_sp<SkImage> image = SkImage::MakeRasterData(info, data, info.minRowBytes());
688 canvas->drawImage(image, 0, 0);
689##
690
691#Example
Cary Clark8032b982017-07-28 11:04:54 -0400692#Description
Cary Clarkce101242017-09-01 15:51:02 -0400693 Canvas returned by Raster_Surface has Premultiplied pixel values.
694 clear() takes Unpremultiplied input with Color_Alpha equal 0x80
Cary Clarkffb3d682018-05-17 12:17:28 -0400695 and RGB equal 0x55, 0xAA, 0xFF. RGB is multiplied by Color_Alpha
Cary Clarkce101242017-09-01 15:51:02 -0400696 to generate Premultiplied value 0x802B5580. readPixels converts pixel back
697 to Unpremultiplied value 0x8056A9FF, introducing error.
Cary Clark8032b982017-07-28 11:04:54 -0400698##
699 canvas->clear(0x8055aaff);
700 for (SkAlphaType alphaType : { kPremul_SkAlphaType, kUnpremul_SkAlphaType } ) {
701 uint32_t pixel = 0;
702 SkImageInfo info = SkImageInfo::Make(1, 1, kBGRA_8888_SkColorType, alphaType);
703 if (canvas->readPixels(info, &pixel, 4, 0, 0)) {
704 SkDebugf("pixel = %08x\n", pixel);
705 }
706 }
707
708 #StdOut
709 pixel = 802b5580
710 pixel = 8056a9ff
711 ##
712##
713
Cary Clark2ade9972017-11-02 17:49:34 -0400714#SeeAlso peekPixels writePixels drawBitmap drawImage SkBitmap::readPixels SkPixmap::readPixels SkImage::readPixels SkSurface::readPixels
Cary Clark8032b982017-07-28 11:04:54 -0400715
716##
717
718# ------------------------------------------------------------------------------
719
720#Method bool readPixels(const SkPixmap& pixmap, int srcX, int srcY)
721
Cary Clark154beea2017-10-26 07:58:48 -0400722Copies Rect of pixels from Canvas into pixmap. Matrix and Clip are
Cary Clarka560c472017-11-27 10:44:06 -0500723ignored.
Cary Clark6fc50412017-09-21 12:31:06 -0400724
Cary Clarka560c472017-11-27 10:44:06 -0500725Source Rect corners are (srcX, srcY) and (imageInfo().width(), imageInfo().height()).
726Destination Rect corners are (0, 0) and (pixmap.width(), pixmap.height()).
Cary Clark6fc50412017-09-21 12:31:06 -0400727Copies each readable pixel intersecting both rectangles, without scaling,
Cary Clarkf05bdda2017-08-24 12:59:48 -0400728converting to pixmap.colorType() and pixmap.alphaType() if required.
Cary Clark8032b982017-07-28 11:04:54 -0400729
Cary Clarkf05bdda2017-08-24 12:59:48 -0400730Pixels are readable when Device is raster, or backed by a GPU.
731Pixels are not readable when SkCanvas is returned by SkDocument::beginPage,
732returned by SkPictureRecorder::beginRecording, or Canvas is the base of a utility
Brian Osman46fe9c72018-03-09 15:44:34 -0500733class like SkDebugCanvas.
Cary Clark8032b982017-07-28 11:04:54 -0400734
Cary Clark6fc50412017-09-21 12:31:06 -0400735Caller must allocate pixel storage in pixmap if needed.
Cary Clark8032b982017-07-28 11:04:54 -0400736
Cary Clark2dc84ad2018-01-26 12:56:22 -0500737Pixel values are converted only if Color_Type and Alpha_Type
Cary Clark154beea2017-10-26 07:58:48 -0400738do not match. Only pixels within both source and destination Rects
739are copied. pixmap pixels contents outside Rect intersection are unchanged.
Cary Clarkf05bdda2017-08-24 12:59:48 -0400740
741Pass negative values for srcX or srcY to offset pixels across or down pixmap.
Cary Clark8032b982017-07-28 11:04:54 -0400742
743Does not copy, and returns false if:
744
745#List
Cary Clarkf05bdda2017-08-24 12:59:48 -0400746# Source and destination rectangles do not intersect. ##
747# Canvas pixels could not be converted to pixmap.colorType() or pixmap.alphaType(). ##
748# Canvas pixels are not readable; for instance, Canvas is document-based. ##
749# Pixmap pixels could not be allocated. ##
750# pixmap.rowBytes() is too small to contain one row of pixels. ##
Cary Clark8032b982017-07-28 11:04:54 -0400751##
752
Cary Clarkbad5ad72017-08-03 17:14:08 -0400753#Param pixmap storage for pixels copied from Canvas ##
Cary Clark5538c132018-06-14 12:28:14 -0400754#Param srcX offset into readable pixels on x-axis; may be negative ##
755#Param srcY offset into readable pixels on y-axis; may be negative ##
Cary Clark8032b982017-07-28 11:04:54 -0400756
Cary Clarkbad5ad72017-08-03 17:14:08 -0400757#Return true if pixels were copied ##
Cary Clark8032b982017-07-28 11:04:54 -0400758
759#Example
Cary Clarkf05bdda2017-08-24 12:59:48 -0400760 #Description
Cary Clarkce101242017-09-01 15:51:02 -0400761 clear() takes Unpremultiplied input with Color_Alpha equal 0x80
Cary Clarkffb3d682018-05-17 12:17:28 -0400762 and RGB equal 0x55, 0xAA, 0xFF. RGB is multiplied by Color_Alpha
Cary Clarkce101242017-09-01 15:51:02 -0400763 to generate Premultiplied value 0x802B5580.
Cary Clarkf05bdda2017-08-24 12:59:48 -0400764 ##
765 void draw(SkCanvas* canvas) {
766 canvas->clear(0x8055aaff);
767 uint32_t pixels[1] = { 0 };
768 SkPixmap pixmap(SkImageInfo::MakeN32Premul(1, 1), pixels, 4);
769 canvas->readPixels(pixmap, 0, 0);
770 SkDebugf("pixel = %08x\n", pixels[0]);
771 }
Cary Clark8032b982017-07-28 11:04:54 -0400772 #StdOut
773 pixel = 802b5580
774 ##
775##
776
Cary Clark2ade9972017-11-02 17:49:34 -0400777#SeeAlso peekPixels writePixels drawBitmap drawImage SkBitmap::readPixels SkPixmap::readPixels SkImage::readPixels SkSurface::readPixels
Cary Clark8032b982017-07-28 11:04:54 -0400778
779##
780
781# ------------------------------------------------------------------------------
782
783#Method bool readPixels(const SkBitmap& bitmap, int srcX, int srcY)
784
Cary Clark154beea2017-10-26 07:58:48 -0400785Copies Rect of pixels from Canvas into bitmap. Matrix and Clip are
Cary Clarka560c472017-11-27 10:44:06 -0500786ignored.
Cary Clark6fc50412017-09-21 12:31:06 -0400787
Cary Clarka560c472017-11-27 10:44:06 -0500788Source Rect corners are (srcX, srcY) and (imageInfo().width(), imageInfo().height()).
Cary Clark154beea2017-10-26 07:58:48 -0400789Destination Rect corners are (0, 0) and (bitmap.width(), bitmap.height()).
Cary Clarkf05bdda2017-08-24 12:59:48 -0400790Copies each readable pixel intersecting both rectangles, without scaling,
791converting to bitmap.colorType() and bitmap.alphaType() if required.
Cary Clark8032b982017-07-28 11:04:54 -0400792
Cary Clarkf05bdda2017-08-24 12:59:48 -0400793Pixels are readable when Device is raster, or backed by a GPU.
794Pixels are not readable when SkCanvas is returned by SkDocument::beginPage,
795returned by SkPictureRecorder::beginRecording, or Canvas is the base of a utility
Brian Osman46fe9c72018-03-09 15:44:34 -0500796class like SkDebugCanvas.
Cary Clark8032b982017-07-28 11:04:54 -0400797
Cary Clark6fc50412017-09-21 12:31:06 -0400798Caller must allocate pixel storage in bitmap if needed.
Cary Clarkf05bdda2017-08-24 12:59:48 -0400799
Cary Clark2dc84ad2018-01-26 12:56:22 -0500800Bitmap values are converted only if Color_Type and Alpha_Type
Cary Clarkf05bdda2017-08-24 12:59:48 -0400801do not match. Only pixels within both source and destination rectangles
Cary Clark154beea2017-10-26 07:58:48 -0400802are copied. Bitmap pixels outside Rect intersection are unchanged.
Cary Clarkf05bdda2017-08-24 12:59:48 -0400803
804Pass negative values for srcX or srcY to offset pixels across or down bitmap.
Cary Clark8032b982017-07-28 11:04:54 -0400805
806Does not copy, and returns false if:
807
808#List
Cary Clarkf05bdda2017-08-24 12:59:48 -0400809# Source and destination rectangles do not intersect. ##
810# Canvas pixels could not be converted to bitmap.colorType() or bitmap.alphaType(). ##
811# Canvas pixels are not readable; for instance, Canvas is document-based. ##
Cary Clark8032b982017-07-28 11:04:54 -0400812# bitmap pixels could not be allocated. ##
Cary Clarkf05bdda2017-08-24 12:59:48 -0400813# bitmap.rowBytes() is too small to contain one row of pixels. ##
Cary Clark8032b982017-07-28 11:04:54 -0400814##
815
Cary Clarkbad5ad72017-08-03 17:14:08 -0400816#Param bitmap storage for pixels copied from Canvas ##
Cary Clark5538c132018-06-14 12:28:14 -0400817#Param srcX offset into readable pixels on x-axis; may be negative ##
818#Param srcY offset into readable pixels on y-axis; may be negative ##
Cary Clark8032b982017-07-28 11:04:54 -0400819
Cary Clarkbad5ad72017-08-03 17:14:08 -0400820#Return true if pixels were copied ##
Cary Clark8032b982017-07-28 11:04:54 -0400821
822#Example
Cary Clarkf05bdda2017-08-24 12:59:48 -0400823 #Description
Cary Clarkce101242017-09-01 15:51:02 -0400824 clear() takes Unpremultiplied input with Color_Alpha equal 0x80
Cary Clarkffb3d682018-05-17 12:17:28 -0400825 and RGB equal 0x55, 0xAA, 0xFF. RGB is multiplied by Color_Alpha
Cary Clarkce101242017-09-01 15:51:02 -0400826 to generate Premultiplied value 0x802B5580.
Cary Clarkf05bdda2017-08-24 12:59:48 -0400827 ##
Cary Clark8032b982017-07-28 11:04:54 -0400828void draw(SkCanvas* canvas) {
829 canvas->clear(0x8055aaff);
830 SkBitmap bitmap;
831 bitmap.allocPixels(SkImageInfo::MakeN32Premul(1, 1));
832 canvas->readPixels(bitmap, 0, 0);
833 SkDebugf("pixel = %08x\n", bitmap.getAddr32(0, 0)[0]);
834}
835 #StdOut
836 pixel = 802b5580
837 ##
838##
839
Cary Clark2ade9972017-11-02 17:49:34 -0400840#SeeAlso peekPixels writePixels drawBitmap drawImage SkBitmap::readPixels SkPixmap::readPixels SkImage::readPixels SkSurface::readPixels
Cary Clark8032b982017-07-28 11:04:54 -0400841
842##
843
844# ------------------------------------------------------------------------------
845
846#Method bool writePixels(const SkImageInfo& info, const void* pixels, size_t rowBytes, int x, int y)
Cary Clark78de7512018-02-07 07:27:09 -0500847#In Pixels
Cary Clarkab2621d2018-01-30 10:08:57 -0500848#Line # copies and converts rectangle of pixels to Canvas ##
Cary Clark154beea2017-10-26 07:58:48 -0400849Copies Rect from pixels to Canvas. Matrix and Clip are ignored.
850Source Rect corners are (0, 0) and (info.width(), info.height()).
851Destination Rect corners are (x, y) and
852(imageInfo().width(), imageInfo().height()).
Cary Clark6fc50412017-09-21 12:31:06 -0400853
854Copies each readable pixel intersecting both rectangles, without scaling,
Cary Clark154beea2017-10-26 07:58:48 -0400855converting to imageInfo().colorType() and imageInfo().alphaType() if required.
Cary Clark8032b982017-07-28 11:04:54 -0400856
Cary Clarkf05bdda2017-08-24 12:59:48 -0400857Pixels are writable when Device is raster, or backed by a GPU.
858Pixels are not writable when SkCanvas is returned by SkDocument::beginPage,
859returned by SkPictureRecorder::beginRecording, or Canvas is the base of a utility
Brian Osman46fe9c72018-03-09 15:44:34 -0500860class like SkDebugCanvas.
Cary Clark8032b982017-07-28 11:04:54 -0400861
Cary Clark2dc84ad2018-01-26 12:56:22 -0500862Pixel values are converted only if Color_Type and Alpha_Type
Cary Clarkf05bdda2017-08-24 12:59:48 -0400863do not match. Only pixels within both source and destination rectangles
Cary Clark154beea2017-10-26 07:58:48 -0400864are copied. Canvas pixels outside Rect intersection are unchanged.
Cary Clark8032b982017-07-28 11:04:54 -0400865
Cary Clarkf05bdda2017-08-24 12:59:48 -0400866Pass negative values for x or y to offset pixels to the left or
867above Canvas pixels.
Cary Clark8032b982017-07-28 11:04:54 -0400868
869Does not copy, and returns false if:
870
871#List
Cary Clarkf05bdda2017-08-24 12:59:48 -0400872# Source and destination rectangles do not intersect. ##
Cary Clarkac47b882018-01-11 10:35:44 -0500873# pixels could not be converted to Canvas imageInfo().colorType() or
874 imageInfo().alphaType(). ##
Cary Clark8032b982017-07-28 11:04:54 -0400875# Canvas pixels are not writable; for instance, Canvas is document-based. ##
876# rowBytes is too small to contain one row of pixels. ##
877##
878
Cary Clark2dc84ad2018-01-26 12:56:22 -0500879#Param info width, height, Color_Type, and Alpha_Type of pixels ##
Cary Clarkf05bdda2017-08-24 12:59:48 -0400880#Param pixels pixels to copy, of size info.height() times rowBytes, or larger ##
Cary Clarkd0530ba2017-09-14 11:25:39 -0400881#Param rowBytes size of one row of pixels; info.width() times pixel size, or larger ##
Cary Clark5538c132018-06-14 12:28:14 -0400882#Param x offset into Canvas writable pixels on x-axis; may be negative ##
883#Param y offset into Canvas writable pixels on y-axis; may be negative ##
Cary Clark8032b982017-07-28 11:04:54 -0400884
Cary Clarkbad5ad72017-08-03 17:14:08 -0400885#Return true if pixels were written to Canvas ##
Cary Clark8032b982017-07-28 11:04:54 -0400886
887#Example
888 SkImageInfo imageInfo = SkImageInfo::MakeN32(256, 1, kPremul_SkAlphaType);
889 for (int y = 0; y < 256; ++y) {
890 uint32_t pixels[256];
891 for (int x = 0; x < 256; ++x) {
892 pixels[x] = SkColorSetARGB(x, x + y, x, x - y);
893 }
894 canvas->writePixels(imageInfo, &pixels, sizeof(pixels), 0, y);
895 }
896##
897
Cary Clark2ade9972017-11-02 17:49:34 -0400898#SeeAlso readPixels drawBitmap drawImage SkBitmap::writePixels
Cary Clark8032b982017-07-28 11:04:54 -0400899
900##
901
902# ------------------------------------------------------------------------------
903
904#Method bool writePixels(const SkBitmap& bitmap, int x, int y)
905
Cary Clark154beea2017-10-26 07:58:48 -0400906Copies Rect from pixels to Canvas. Matrix and Clip are ignored.
907Source Rect corners are (0, 0) and (bitmap.width(), bitmap.height()).
Cary Clark6fc50412017-09-21 12:31:06 -0400908
Cary Clark154beea2017-10-26 07:58:48 -0400909Destination Rect corners are (x, y) and
910(imageInfo().width(), imageInfo().height()).
Cary Clark6fc50412017-09-21 12:31:06 -0400911
912Copies each readable pixel intersecting both rectangles, without scaling,
Cary Clark154beea2017-10-26 07:58:48 -0400913converting to imageInfo().colorType() and imageInfo().alphaType() if required.
Cary Clark8032b982017-07-28 11:04:54 -0400914
Cary Clarkf05bdda2017-08-24 12:59:48 -0400915Pixels are writable when Device is raster, or backed by a GPU.
916Pixels are not writable when SkCanvas is returned by SkDocument::beginPage,
917returned by SkPictureRecorder::beginRecording, or Canvas is the base of a utility
Brian Osman46fe9c72018-03-09 15:44:34 -0500918class like SkDebugCanvas.
Cary Clark8032b982017-07-28 11:04:54 -0400919
Cary Clark2dc84ad2018-01-26 12:56:22 -0500920Pixel values are converted only if Color_Type and Alpha_Type
Cary Clarkf05bdda2017-08-24 12:59:48 -0400921do not match. Only pixels within both source and destination rectangles
Cary Clark154beea2017-10-26 07:58:48 -0400922are copied. Canvas pixels outside Rect intersection are unchanged.
Cary Clark8032b982017-07-28 11:04:54 -0400923
Cary Clarkf05bdda2017-08-24 12:59:48 -0400924Pass negative values for x or y to offset pixels to the left or
925above Canvas pixels.
Cary Clark8032b982017-07-28 11:04:54 -0400926
927Does not copy, and returns false if:
928
929#List
Cary Clarkf05bdda2017-08-24 12:59:48 -0400930# Source and destination rectangles do not intersect. ##
Cary Clark8032b982017-07-28 11:04:54 -0400931# bitmap does not have allocated pixels. ##
Cary Clarkac47b882018-01-11 10:35:44 -0500932# bitmap pixels could not be converted to Canvas imageInfo().colorType() or
933 imageInfo().alphaType(). ##
Cary Clarkce101242017-09-01 15:51:02 -0400934# Canvas pixels are not writable; for instance, Canvas is document based. ##
Cary Clark8032b982017-07-28 11:04:54 -0400935# bitmap pixels are inaccessible; for instance, bitmap wraps a texture. ##
936##
937
Cary Clarkbad5ad72017-08-03 17:14:08 -0400938#Param bitmap contains pixels copied to Canvas ##
Cary Clarkf05bdda2017-08-24 12:59:48 -0400939#Param x offset into Canvas writable pixels in x; may be negative ##
940#Param y offset into Canvas writable pixels in y; may be negative ##
Cary Clark8032b982017-07-28 11:04:54 -0400941
Cary Clarkbad5ad72017-08-03 17:14:08 -0400942#Return true if pixels were written to Canvas ##
Cary Clark8032b982017-07-28 11:04:54 -0400943
944#Example
945void draw(SkCanvas* canvas) {
946 SkImageInfo imageInfo = SkImageInfo::MakeN32Premul(2, 2);
947 SkBitmap bitmap;
948 bitmap.setInfo(imageInfo);
949 uint32_t pixels[4];
950 bitmap.setPixels(pixels);
951 for (int y = 0; y < 256; y += 2) {
952 for (int x = 0; x < 256; x += 2) {
953 pixels[0] = SkColorSetRGB(x, y, x | y);
954 pixels[1] = SkColorSetRGB(x ^ y, y, x);
955 pixels[2] = SkColorSetRGB(x, x & y, y);
956 pixels[3] = SkColorSetRGB(~x, ~y, x);
957 canvas->writePixels(bitmap, x, y);
958 }
959 }
960}
961##
962
Cary Clark2ade9972017-11-02 17:49:34 -0400963#SeeAlso readPixels drawBitmap drawImage SkBitmap::writePixels
Cary Clark8032b982017-07-28 11:04:54 -0400964
965##
966
967# ------------------------------------------------------------------------------
Cary Clark08895c42018-02-01 09:37:32 -0500968#Subtopic State_Stack
969#Line # stack of state for hierarchical drawing ##
Cary Clark8032b982017-07-28 11:04:54 -0400970
971Canvas maintains a stack of state that allows hierarchical drawing, commonly used
Cary Clarkbad5ad72017-08-03 17:14:08 -0400972to implement windows and views. The initial state has an identity matrix and and
973an infinite clip. Even with a wide-open clip, drawing is constrained by the
974bounds of the Canvas Surface or Device.
Cary Clark8032b982017-07-28 11:04:54 -0400975
Cary Clark939fd6c2018-07-12 08:40:13 -0400976Canvas savable state consists of Clip and Matrix.
Cary Clark8032b982017-07-28 11:04:54 -0400977Clip describes the area that may be drawn to.
978Matrix transforms the geometry.
Cary Clark8032b982017-07-28 11:04:54 -0400979
Mike Reed9a229e92019-01-08 10:52:52 -0500980save(), saveLayer, and saveLayerAlpha
Cary Clark8032b982017-07-28 11:04:54 -0400981save state and return the depth of the stack.
982
Cary Clarkbad5ad72017-08-03 17:14:08 -0400983restore(), restoreToCount, and ~SkCanvas() revert state to its value when saved.
Cary Clark8032b982017-07-28 11:04:54 -0400984
985Each state on the stack intersects Clip with the previous Clip,
986and concatenates Matrix with the previous Matrix.
987The intersected Clip makes the drawing area the same or smaller;
988the concatenated Matrix may move the origin and potentially scale or rotate
989the coordinate space.
990
991Canvas does not require balancing the state stack but it is a good idea
992to do so. Calling save() without restore() will eventually cause Skia to fail;
993mismatched save() and restore() create hard to find bugs.
994
995It is not possible to use state to draw outside of the clip defined by the
996previous state.
997
998#Example
999#Description
1000Draw to ever smaller clips; then restore drawing to full canvas.
1001Note that the second clipRect is not permitted to enlarge Clip.
1002##
1003#Height 160
Cary Clarkbad5ad72017-08-03 17:14:08 -04001004void draw(SkCanvas* canvas) {
1005 SkPaint paint;
Herb Derbyefe39bc2018-05-01 17:06:20 -04001006 canvas->save(); // records stack depth to restore
Cary Clarkbad5ad72017-08-03 17:14:08 -04001007 canvas->clipRect(SkRect::MakeWH(100, 100)); // constrains drawing to clip
1008 canvas->clear(SK_ColorRED); // draws to limit of clip
Herb Derbyefe39bc2018-05-01 17:06:20 -04001009 canvas->save(); // records stack depth to restore
Cary Clarkbad5ad72017-08-03 17:14:08 -04001010 canvas->clipRect(SkRect::MakeWH(50, 150)); // Rect below 100 is ignored
1011 canvas->clear(SK_ColorBLUE); // draws to smaller clip
1012 canvas->restore(); // enlarges clip
1013 canvas->drawLine(20, 20, 150, 150, paint); // line below 100 is not drawn
1014 canvas->restore(); // enlarges clip
1015 canvas->drawLine(150, 20, 50, 120, paint); // line below 100 is drawn
Cary Clark8032b982017-07-28 11:04:54 -04001016}
Herb Derbyefe39bc2018-05-01 17:06:20 -04001017##
Cary Clark8032b982017-07-28 11:04:54 -04001018
1019Each Clip uses the current Matrix for its coordinates.
1020
1021#Example
1022#Description
1023While clipRect is given the same rectangle twice, Matrix makes the second
1024clipRect draw at half the size of the first.
1025##
1026#Height 128
Cary Clarkbad5ad72017-08-03 17:14:08 -04001027void draw(SkCanvas* canvas) {
1028 canvas->clipRect(SkRect::MakeWH(100, 100));
1029 canvas->clear(SK_ColorRED);
1030 canvas->scale(.5, .5);
1031 canvas->clipRect(SkRect::MakeWH(100, 100));
1032 canvas->clear(SK_ColorBLUE);
Cary Clark8032b982017-07-28 11:04:54 -04001033}
1034##
1035
Mike Reed9a229e92019-01-08 10:52:52 -05001036#SeeAlso save saveLayer saveLayerAlpha restore() restoreToCount
Cary Clark8032b982017-07-28 11:04:54 -04001037
1038#Method int save()
1039
Cary Clarkab2621d2018-01-30 10:08:57 -05001040#In State_Stack
1041#Line # saves Clip and Matrix on stack ##
Cary Clark09d80c02018-10-31 12:14:03 -04001042#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001043
1044#Example
Herb Derbyefe39bc2018-05-01 17:06:20 -04001045#Description
Cary Clark8032b982017-07-28 11:04:54 -04001046The black square is translated 50 pixels down and to the right.
1047Restoring Canvas state removes translate() from Canvas stack;
1048the red square is not translated, and is drawn at the origin.
1049##
1050#Height 100
1051void draw(SkCanvas* canvas) {
1052 SkPaint paint;
1053 SkRect rect = { 0, 0, 25, 25 };
1054 canvas->drawRect(rect, paint);
1055 canvas->save();
1056 canvas->translate(50, 50);
1057 canvas->drawRect(rect, paint);
1058 canvas->restore();
1059 paint.setColor(SK_ColorRED);
1060 canvas->drawRect(rect, paint);
1061}
1062##
1063
Mike Reed9a229e92019-01-08 10:52:52 -05001064#SeeAlso saveLayer saveLayerAlpha restore restoreToCount
Cary Clark8032b982017-07-28 11:04:54 -04001065
1066##
1067
1068# ------------------------------------------------------------------------------
Cary Clark8032b982017-07-28 11:04:54 -04001069
1070#Method void restore()
1071
Cary Clarkab2621d2018-01-30 10:08:57 -05001072#In State_Stack
1073#Line # restores changes to Clip and Matrix, pops save stack ##
Cary Clark09d80c02018-10-31 12:14:03 -04001074#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001075
1076#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04001077void draw(SkCanvas* canvas) {
1078 SkCanvas simple;
1079 SkDebugf("depth = %d\n", simple.getSaveCount());
1080 simple.restore();
1081 SkDebugf("depth = %d\n", simple.getSaveCount());
Cary Clark8032b982017-07-28 11:04:54 -04001082}
1083##
1084
Mike Reed9a229e92019-01-08 10:52:52 -05001085#SeeAlso save saveLayer saveLayerAlpha restoreToCount
Cary Clark2ade9972017-11-02 17:49:34 -04001086
Cary Clark8032b982017-07-28 11:04:54 -04001087##
1088
1089# ------------------------------------------------------------------------------
1090
1091#Method int getSaveCount() const
1092
Cary Clarkab2621d2018-01-30 10:08:57 -05001093#In State_Stack
1094#Line # returns depth of stack containing Clip and Matrix ##
Cary Clark09d80c02018-10-31 12:14:03 -04001095#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001096
1097#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04001098void draw(SkCanvas* canvas) {
1099 SkCanvas simple;
1100 SkDebugf("depth = %d\n", simple.getSaveCount());
1101 simple.save();
1102 SkDebugf("depth = %d\n", simple.getSaveCount());
1103 simple.restore();
1104 SkDebugf("depth = %d\n", simple.getSaveCount());
Cary Clark8032b982017-07-28 11:04:54 -04001105}
1106#StdOut
Cary Clarkbad5ad72017-08-03 17:14:08 -04001107depth = 1
1108depth = 2
Cary Clark8032b982017-07-28 11:04:54 -04001109depth = 1
1110##
1111##
1112
Cary Clarka90ea222018-10-16 10:30:28 -04001113#SeeAlso save restore restoreToCount
Cary Clark2ade9972017-11-02 17:49:34 -04001114
Cary Clark8032b982017-07-28 11:04:54 -04001115##
1116
1117# ------------------------------------------------------------------------------
1118
1119#Method void restoreToCount(int saveCount)
1120
Cary Clarkab2621d2018-01-30 10:08:57 -05001121#In State_Stack
1122#Line # restores changes to Clip and Matrix to given depth ##
Cary Clark09d80c02018-10-31 12:14:03 -04001123#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001124
1125#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04001126void draw(SkCanvas* canvas) {
1127 SkDebugf("depth = %d\n", canvas->getSaveCount());
1128 canvas->save();
1129 canvas->save();
1130 SkDebugf("depth = %d\n", canvas->getSaveCount());
1131 canvas->restoreToCount(0);
1132 SkDebugf("depth = %d\n", canvas->getSaveCount());
Cary Clark8032b982017-07-28 11:04:54 -04001133}
1134#StdOut
Cary Clarkbad5ad72017-08-03 17:14:08 -04001135depth = 1
1136depth = 3
Cary Clark8032b982017-07-28 11:04:54 -04001137depth = 1
1138##
1139##
1140
Cary Clarka90ea222018-10-16 10:30:28 -04001141#SeeAlso restore getSaveCount save
Cary Clark2ade9972017-11-02 17:49:34 -04001142
Cary Clark8032b982017-07-28 11:04:54 -04001143##
1144
Cary Clark08895c42018-02-01 09:37:32 -05001145#Subtopic State_Stack ##
Cary Clark8032b982017-07-28 11:04:54 -04001146
1147# ------------------------------------------------------------------------------
Cary Clarkce101242017-09-01 15:51:02 -04001148
Cary Clark08895c42018-02-01 09:37:32 -05001149#Subtopic Layer
Cary Clarkd0530ba2017-09-14 11:25:39 -04001150#Substitute layer
Cary Clarkce101242017-09-01 15:51:02 -04001151#Alias Layers
Cary Clark137b8742018-05-30 09:21:49 -04001152#Substitute layers
1153##
Cary Clark08895c42018-02-01 09:37:32 -05001154#Line # temporary Bitmap to draw into ##
Cary Clarkce101242017-09-01 15:51:02 -04001155
1156Layer allocates a temporary Bitmap to draw into. When the drawing is
Herb Derbyefe39bc2018-05-01 17:06:20 -04001157complete, the Bitmap is drawn into the Canvas.
Cary Clarkce101242017-09-01 15:51:02 -04001158
1159Layer is saved in a stack along with other saved state. When state with a Layer
1160is restored, the Bitmap is drawn into the previous Layer.
1161
1162Layer may be initialized with the contents of the previous Layer. When Layer is
1163restored, its Bitmap can be modified by Paint passed to Layer to apply
1164Color_Alpha, Color_Filter, Image_Filter, and Blend_Mode.
1165
1166#Method int saveLayer(const SkRect* bounds, const SkPaint* paint)
1167
Cary Clarkab2621d2018-01-30 10:08:57 -05001168#In Layer
1169#Line # saves Clip and Matrix on stack; creates Layer ##
Cary Clark09d80c02018-10-31 12:14:03 -04001170#Populate
Cary Clarkce101242017-09-01 15:51:02 -04001171
1172#Example
1173#Description
1174Rectangles are blurred by Image_Filter when restore() draws Layer to main
1175Canvas.
1176##
1177#Height 128
Cary Clark81abc432018-06-25 16:30:08 -04001178#Function
1179###$
1180#include "SkBlurImageFilter.h"
1181$$$#
1182##
1183
Cary Clarkce101242017-09-01 15:51:02 -04001184void draw(SkCanvas* canvas) {
1185 SkPaint paint, blur;
Cary Clarka560c472017-11-27 10:44:06 -05001186 blur.setImageFilter(SkBlurImageFilter::Make(3, 3, nullptr));
Cary Clarkce101242017-09-01 15:51:02 -04001187 canvas->saveLayer(nullptr, &blur);
1188 SkRect rect = { 25, 25, 50, 50};
1189 canvas->drawRect(rect, paint);
1190 canvas->translate(50, 50);
1191 paint.setColor(SK_ColorRED);
1192 canvas->drawRect(rect, paint);
1193 canvas->restore();
1194}
1195##
1196
Mike Reed9a229e92019-01-08 10:52:52 -05001197#SeeAlso save restore saveLayer saveLayerAlpha SaveLayerRec
Cary Clarkce101242017-09-01 15:51:02 -04001198
1199##
1200
Herb Derbyefe39bc2018-05-01 17:06:20 -04001201#Method int saveLayer(const SkRect& bounds, const SkPaint* paint)
Cary Clarkce101242017-09-01 15:51:02 -04001202
Cary Clarkab2621d2018-01-30 10:08:57 -05001203#In Layer
Cary Clark09d80c02018-10-31 12:14:03 -04001204#Populate
Cary Clarkce101242017-09-01 15:51:02 -04001205
1206#Example
1207#Description
1208Rectangles are blurred by Image_Filter when restore() draws Layer to main Canvas.
Herb Derbyefe39bc2018-05-01 17:06:20 -04001209The red rectangle is clipped; it does not fully fit on Layer.
Cary Clarkce101242017-09-01 15:51:02 -04001210Image_Filter blurs past edge of Layer so red rectangle is blurred on all sides.
1211##
1212#Height 128
Cary Clark81abc432018-06-25 16:30:08 -04001213#Function
1214###$
1215#include "SkBlurImageFilter.h"
1216$$$#
1217##
1218
Cary Clarkce101242017-09-01 15:51:02 -04001219void draw(SkCanvas* canvas) {
1220 SkPaint paint, blur;
Cary Clarka560c472017-11-27 10:44:06 -05001221 blur.setImageFilter(SkBlurImageFilter::Make(3, 3, nullptr));
Cary Clarkce101242017-09-01 15:51:02 -04001222 canvas->saveLayer(SkRect::MakeWH(90, 90), &blur);
1223 SkRect rect = { 25, 25, 50, 50};
1224 canvas->drawRect(rect, paint);
1225 canvas->translate(50, 50);
1226 paint.setColor(SK_ColorRED);
1227 canvas->drawRect(rect, paint);
1228 canvas->restore();
1229}
1230##
1231
Mike Reed9a229e92019-01-08 10:52:52 -05001232#SeeAlso save restore saveLayerAlpha SaveLayerRec
Cary Clarkce101242017-09-01 15:51:02 -04001233
1234##
1235
1236#Method int saveLayerAlpha(const SkRect* bounds, U8CPU alpha)
1237
Cary Clarkab2621d2018-01-30 10:08:57 -05001238#In Layer
1239#Line # saves Clip and Matrix on stack; creates Layer; sets opacity ##
Cary Clark09d80c02018-10-31 12:14:03 -04001240#Populate
Cary Clarkce101242017-09-01 15:51:02 -04001241
1242#Example
1243 SkPaint paint;
1244 paint.setColor(SK_ColorRED);
1245 canvas->drawCircle(50, 50, 50, paint);
1246 canvas->saveLayerAlpha(nullptr, 128);
1247 paint.setColor(SK_ColorBLUE);
1248 canvas->drawCircle(100, 50, 50, paint);
1249 paint.setColor(SK_ColorGREEN);
1250 paint.setAlpha(128);
1251 canvas->drawCircle(75, 90, 50, paint);
1252 canvas->restore();
1253##
1254
Mike Reed9a229e92019-01-08 10:52:52 -05001255#SeeAlso save restore saveLayer SaveLayerRec
Cary Clarkce101242017-09-01 15:51:02 -04001256
1257##
1258
Cary Clarkd98f78c2018-04-26 08:32:37 -04001259#Enum SaveLayerFlagsSet
Cary Clark08895c42018-02-01 09:37:32 -05001260#Line # sets SaveLayerRec options ##
Cary Clarkce101242017-09-01 15:51:02 -04001261#Code
Cary Clark61313f32018-10-08 14:57:48 -04001262#Populate
Cary Clarkce101242017-09-01 15:51:02 -04001263##
1264
Cary Clark682c58d2018-05-16 07:07:07 -04001265
1266#Typedef uint32_t SaveLayerFlags
1267#Line # options for SaveLayerRec ##
Cary Clark137b8742018-05-30 09:21:49 -04001268##
Cary Clark682c58d2018-05-16 07:07:07 -04001269
Cary Clarkce101242017-09-01 15:51:02 -04001270SaveLayerFlags provides options that may be used in any combination in SaveLayerRec,
Mike Reed9a229e92019-01-08 10:52:52 -05001271defining how Layer allocated by saveLayer operates. It may be set to zero or
1272kInitWithPrevious_SaveLayerFlag.
Cary Clarkce101242017-09-01 15:51:02 -04001273
1274#Const kInitWithPrevious_SaveLayerFlag 4
Cary Clark682c58d2018-05-16 07:07:07 -04001275#Line # initializes with previous contents ##
Cary Clarkce101242017-09-01 15:51:02 -04001276 Initializes Layer with the contents of the previous Layer.
1277##
1278
Cary Clarkce101242017-09-01 15:51:02 -04001279#Example
1280#Height 160
1281#Description
1282Canvas Layer captures red and blue circles scaled up by four.
Herb Derbyefe39bc2018-05-01 17:06:20 -04001283scalePaint blends Layer back with transparency.
Cary Clarkce101242017-09-01 15:51:02 -04001284##
1285void draw(SkCanvas* canvas) {
1286 SkPaint redPaint, bluePaint, scalePaint;
1287 redPaint.setColor(SK_ColorRED);
1288 canvas->drawCircle(21, 21, 8, redPaint);
1289 bluePaint.setColor(SK_ColorBLUE);
1290 canvas->drawCircle(31, 21, 8, bluePaint);
1291 SkMatrix matrix;
1292 matrix.setScale(4, 4);
1293 scalePaint.setAlpha(0x40);
1294 scalePaint.setImageFilter(
1295 SkImageFilter::MakeMatrixFilter(matrix, kNone_SkFilterQuality, nullptr));
1296 SkCanvas::SaveLayerRec saveLayerRec(nullptr, &scalePaint,
Herb Derbyefe39bc2018-05-01 17:06:20 -04001297 SkCanvas::kInitWithPrevious_SaveLayerFlag);
Cary Clarkce101242017-09-01 15:51:02 -04001298 canvas->saveLayer(saveLayerRec);
1299 canvas->restore();
1300}
1301##
1302
Mike Reed9a229e92019-01-08 10:52:52 -05001303#SeeAlso save restore saveLayer saveLayerAlpha SaveLayerRec
Cary Clarkce101242017-09-01 15:51:02 -04001304
1305#Enum ##
1306
Cary Clark682c58d2018-05-16 07:07:07 -04001307#Subtopic SaveLayerRec
1308#Line # contains the state used to create the Layer ##
Cary Clarka560c472017-11-27 10:44:06 -05001309
Cary Clarkce101242017-09-01 15:51:02 -04001310#Struct SaveLayerRec
Cary Clark08895c42018-02-01 09:37:32 -05001311#Line # contains the state used to create the Layer ##
Cary Clark682c58d2018-05-16 07:07:07 -04001312
Cary Clarkce101242017-09-01 15:51:02 -04001313#Code
Cary Clark61313f32018-10-08 14:57:48 -04001314#Populate
Cary Clarkce101242017-09-01 15:51:02 -04001315##
1316
Cary Clark137b8742018-05-30 09:21:49 -04001317SaveLayerRec contains the state used to create the Layer.
1318
Cary Clarkce101242017-09-01 15:51:02 -04001319#Member const SkRect* fBounds
Cary Clark682c58d2018-05-16 07:07:07 -04001320#Line # hints at Layer size limit ##
Cary Clarkce101242017-09-01 15:51:02 -04001321 fBounds is used as a hint to limit the size of Layer; may be nullptr.
1322 fBounds suggests but does not define Layer size. To clip drawing to
1323 a specific rectangle, use clipRect.
1324##
1325
1326#Member const SkPaint* fPaint
Cary Clark682c58d2018-05-16 07:07:07 -04001327#Line # modifies overlay ##
Cary Clarkce101242017-09-01 15:51:02 -04001328 fPaint modifies how Layer overlays the prior Layer; may be nullptr.
1329 Color_Alpha, Blend_Mode, Color_Filter, Draw_Looper, Image_Filter, and
1330 Mask_Filter affect Layer draw.
1331##
1332
1333#Member const SkImageFilter* fBackdrop
Cary Clark682c58d2018-05-16 07:07:07 -04001334#Line # applies Image_Filter to prior Layer ##
Cary Clarkce101242017-09-01 15:51:02 -04001335 fBackdrop applies Image_Filter to the prior Layer when copying to the Layer;
1336 may be nullptr. Use kInitWithPrevious_SaveLayerFlag to copy the
1337 prior Layer without an Image_Filter.
1338##
1339
1340#Member const SkImage* fClipMask
Cary Clark682c58d2018-05-16 07:07:07 -04001341#Line # clips Layer with Mask_Alpha ##
Cary Clarkce101242017-09-01 15:51:02 -04001342 restore() clips Layer by the Color_Alpha channel of fClipMask when
1343 Layer is copied to Device. fClipMask may be nullptr. .
1344##
1345
1346#Member const SkMatrix* fClipMatrix
Cary Clark682c58d2018-05-16 07:07:07 -04001347#Line # transforms Mask_Alpha used to clip ##
Herb Derbyefe39bc2018-05-01 17:06:20 -04001348 fClipMatrix transforms fClipMask before it clips Layer. If
Cary Clarkce101242017-09-01 15:51:02 -04001349 fClipMask describes a translucent gradient, it may be scaled and rotated
1350 without introducing artifacts. fClipMatrix may be nullptr.
1351##
1352
1353#Member SaveLayerFlags fSaveLayerFlags
Mike Reed9a229e92019-01-08 10:52:52 -05001354#Line # creates with prior Layer contents ##
Cary Clarkce101242017-09-01 15:51:02 -04001355 fSaveLayerFlags are used to create Layer without transparency,
Mike Reed9a229e92019-01-08 10:52:52 -05001356 and to create Layer with the ontents of the previous Layer.
Cary Clarkce101242017-09-01 15:51:02 -04001357##
1358
1359#Example
1360#Height 160
1361#Description
Cary Clarkffb3d682018-05-17 12:17:28 -04001362Canvas Layer captures a red Anti_Aliased circle and a blue Aliased circle scaled
Cary Clarkce101242017-09-01 15:51:02 -04001363up by four. After drawing another red circle without scaling on top, the Layer is
Herb Derbyefe39bc2018-05-01 17:06:20 -04001364transferred to the main canvas.
Cary Clarkce101242017-09-01 15:51:02 -04001365##
1366void draw(SkCanvas* canvas) {
1367 SkPaint redPaint, bluePaint;
1368 redPaint.setAntiAlias(true);
1369 redPaint.setColor(SK_ColorRED);
1370 canvas->drawCircle(21, 21, 8, redPaint);
1371 bluePaint.setColor(SK_ColorBLUE);
1372 canvas->drawCircle(31, 21, 8, bluePaint);
1373 SkMatrix matrix;
1374 matrix.setScale(4, 4);
1375 auto scaler = SkImageFilter::MakeMatrixFilter(matrix, kNone_SkFilterQuality, nullptr);
Herb Derbyefe39bc2018-05-01 17:06:20 -04001376 SkCanvas::SaveLayerRec saveLayerRec(nullptr, nullptr, scaler.get(), 0);
Cary Clarkce101242017-09-01 15:51:02 -04001377 canvas->saveLayer(saveLayerRec);
1378 canvas->drawCircle(125, 85, 8, redPaint);
1379 canvas->restore();
1380}
1381##
1382
Cary Clark61313f32018-10-08 14:57:48 -04001383#Subtopic Constructors
Cary Clark682c58d2018-05-16 07:07:07 -04001384##
Cary Clarkce101242017-09-01 15:51:02 -04001385
Cary Clark682c58d2018-05-16 07:07:07 -04001386#Method SaveLayerRec()
1387#Line # constructs SaveLayerRec ##
Cary Clark09d80c02018-10-31 12:14:03 -04001388#Populate
Cary Clarkce101242017-09-01 15:51:02 -04001389
1390#Example
1391 SkCanvas::SaveLayerRec rec1;
Mike Reed9a229e92019-01-08 10:52:52 -05001392 rec1.fSaveLayerFlags = SkCanvas::kInitWithPrevious_SaveLayerFlag;
1393 SkCanvas::SaveLayerRec rec2(nullptr, nullptr, SkCanvas::kInitWithPrevious_SaveLayerFlag);
Cary Clarkce101242017-09-01 15:51:02 -04001394 SkDebugf("rec1 %c= rec2\n", rec1.fBounds == rec2.fBounds
1395 && rec1.fPaint == rec2.fPaint
1396 && rec1.fBackdrop == rec2.fBackdrop
1397 && rec1.fSaveLayerFlags == rec2.fSaveLayerFlags ? '=' : '!');
1398 #StdOut
1399 rec1 == rec2
1400 ##
1401##
1402
Mike Reed9a229e92019-01-08 10:52:52 -05001403#SeeAlso save restore saveLayer saveLayerAlpha
Cary Clark2ade9972017-11-02 17:49:34 -04001404
Cary Clarkce101242017-09-01 15:51:02 -04001405##
1406
Cary Clark224c7002018-06-27 11:00:21 -04001407#Method SaveLayerRec(const SkRect* bounds, const SkPaint* paint, SaveLayerFlags saveLayerFlags = 0)
Cary Clark09d80c02018-10-31 12:14:03 -04001408#Populate
Cary Clarkce101242017-09-01 15:51:02 -04001409
1410#Example
1411 SkCanvas::SaveLayerRec rec1;
1412 SkCanvas::SaveLayerRec rec2(nullptr, nullptr);
1413 SkDebugf("rec1 %c= rec2\n", rec1.fBounds == rec2.fBounds
1414 && rec1.fPaint == rec2.fPaint
1415 && rec1.fBackdrop == rec2.fBackdrop
1416 && rec1.fSaveLayerFlags == rec2.fSaveLayerFlags ? '=' : '!');
1417 #StdOut
1418 rec1 == rec2
1419 ##
1420##
1421
Mike Reed9a229e92019-01-08 10:52:52 -05001422#SeeAlso save restore saveLayer saveLayerAlpha
Cary Clark2ade9972017-11-02 17:49:34 -04001423
Cary Clarkce101242017-09-01 15:51:02 -04001424##
1425
1426#Method SaveLayerRec(const SkRect* bounds, const SkPaint* paint, const SkImageFilter* backdrop,
1427 SaveLayerFlags saveLayerFlags)
Cary Clark09d80c02018-10-31 12:14:03 -04001428#Populate
Cary Clarkce101242017-09-01 15:51:02 -04001429
1430#Example
1431 SkCanvas::SaveLayerRec rec1;
1432 SkCanvas::SaveLayerRec rec2(nullptr, nullptr, nullptr, 0);
1433 SkDebugf("rec1 %c= rec2\n", rec1.fBounds == rec2.fBounds
1434 && rec1.fPaint == rec2.fPaint
1435 && rec1.fBackdrop == rec2.fBackdrop
1436 && rec1.fSaveLayerFlags == rec2.fSaveLayerFlags ? '=' : '!');
1437 #StdOut
1438 rec1 == rec2
1439 ##
1440##
1441
Mike Reed9a229e92019-01-08 10:52:52 -05001442#SeeAlso save restore saveLayer saveLayerAlpha
Cary Clark2ade9972017-11-02 17:49:34 -04001443
Cary Clarkce101242017-09-01 15:51:02 -04001444##
1445
Cary Clarkce101242017-09-01 15:51:02 -04001446#Struct ##
1447
Cary Clark682c58d2018-05-16 07:07:07 -04001448#Subtopic ##
1449
Cary Clarkce101242017-09-01 15:51:02 -04001450#Method int saveLayer(const SaveLayerRec& layerRec)
1451
Cary Clarkab2621d2018-01-30 10:08:57 -05001452#In Layer
Cary Clark09d80c02018-10-31 12:14:03 -04001453#Populate
Cary Clarkce101242017-09-01 15:51:02 -04001454
1455#Example
1456#Description
1457The example draws an image, and saves it into a Layer with kInitWithPrevious_SaveLayerFlag.
1458Next it punches a hole in Layer and restore with SkBlendMode::kPlus.
1459Where Layer was cleared, the original image will draw unchanged.
1460Outside of the circle the mandrill is brightened.
1461##
1462 #Image 3
Hal Canaryc465d132017-12-08 10:21:31 -05001463 // sk_sp<SkImage> image = GetResourceAsImage("images/mandrill_256.png");
Cary Clarkce101242017-09-01 15:51:02 -04001464 canvas->drawImage(image, 0, 0, nullptr);
1465 SkCanvas::SaveLayerRec rec;
1466 SkPaint paint;
1467 paint.setBlendMode(SkBlendMode::kPlus);
1468 rec.fSaveLayerFlags = SkCanvas::kInitWithPrevious_SaveLayerFlag;
1469 rec.fPaint = &paint;
1470 canvas->saveLayer(rec);
1471 paint.setBlendMode(SkBlendMode::kClear);
1472 canvas->drawCircle(128, 128, 96, paint);
1473 canvas->restore();
1474##
1475
1476#ToDo above example needs to replace GetResourceAsImage with way to select image in fiddle ##
1477
Mike Reed9a229e92019-01-08 10:52:52 -05001478#SeeAlso save restore saveLayer saveLayerAlpha
Cary Clark2ade9972017-11-02 17:49:34 -04001479
Cary Clarkce101242017-09-01 15:51:02 -04001480##
1481
Cary Clark08895c42018-02-01 09:37:32 -05001482#Subtopic Layer ##
Cary Clarkce101242017-09-01 15:51:02 -04001483
1484# ------------------------------------------------------------------------------
Cary Clark08895c42018-02-01 09:37:32 -05001485#Subtopic Matrix
1486#Line # coordinate transformation ##
Cary Clark8032b982017-07-28 11:04:54 -04001487
1488#Method void translate(SkScalar dx, SkScalar dy)
1489
Cary Clarkab2621d2018-01-30 10:08:57 -05001490#In Matrix
1491#Line # translates Matrix ##
Cary Clark09d80c02018-10-31 12:14:03 -04001492#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001493
1494#Example
1495#Height 128
1496#Description
1497scale() followed by translate() produces different results from translate() followed
Herb Derbyefe39bc2018-05-01 17:06:20 -04001498by scale().
Cary Clark8032b982017-07-28 11:04:54 -04001499
1500The blue stroke follows translate of (50, 50); a black
Herb Derbyefe39bc2018-05-01 17:06:20 -04001501fill follows scale of (2, 1/2.f). After restoring the clip, which resets
Cary Clark8032b982017-07-28 11:04:54 -04001502Matrix, a red frame follows the same scale of (2, 1/2.f); a gray fill
1503follows translate of (50, 50).
1504##
Cary Clarkbad5ad72017-08-03 17:14:08 -04001505void draw(SkCanvas* canvas) {
1506 SkPaint filledPaint;
1507 SkPaint outlinePaint;
1508 outlinePaint.setStyle(SkPaint::kStroke_Style);
1509 outlinePaint.setColor(SK_ColorBLUE);
1510 canvas->save();
1511 canvas->translate(50, 50);
1512 canvas->drawCircle(28, 28, 15, outlinePaint); // blue center: (50+28, 50+28)
1513 canvas->scale(2, 1/2.f);
1514 canvas->drawCircle(28, 28, 15, filledPaint); // black center: (50+(28*2), 50+(28/2))
1515 canvas->restore();
1516 filledPaint.setColor(SK_ColorGRAY);
1517 outlinePaint.setColor(SK_ColorRED);
1518 canvas->scale(2, 1/2.f);
1519 canvas->drawCircle(28, 28, 15, outlinePaint); // red center: (28*2, 28/2)
1520 canvas->translate(50, 50);
1521 canvas->drawCircle(28, 28, 15, filledPaint); // gray center: ((50+28)*2, (50+28)/2)
Cary Clark8032b982017-07-28 11:04:54 -04001522}
1523##
1524
Cary Clark2ade9972017-11-02 17:49:34 -04001525#SeeAlso concat() scale() skew() rotate() setMatrix
Cary Clark8032b982017-07-28 11:04:54 -04001526
1527##
1528
1529# ------------------------------------------------------------------------------
1530
1531#Method void scale(SkScalar sx, SkScalar sy)
1532
Cary Clarkab2621d2018-01-30 10:08:57 -05001533#In Matrix
1534#Line # scales Matrix ##
Cary Clark09d80c02018-10-31 12:14:03 -04001535#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001536
1537#Example
1538#Height 160
Cary Clarkbad5ad72017-08-03 17:14:08 -04001539void draw(SkCanvas* canvas) {
1540 SkPaint paint;
1541 SkRect rect = { 10, 20, 60, 120 };
1542 canvas->translate(20, 20);
1543 canvas->drawRect(rect, paint);
1544 canvas->scale(2, .5f);
1545 paint.setColor(SK_ColorGRAY);
1546 canvas->drawRect(rect, paint);
Cary Clark8032b982017-07-28 11:04:54 -04001547}
1548##
1549
Cary Clark2ade9972017-11-02 17:49:34 -04001550#SeeAlso concat() translate() skew() rotate() setMatrix
Cary Clark8032b982017-07-28 11:04:54 -04001551
1552##
1553
1554# ------------------------------------------------------------------------------
1555
1556#Method void rotate(SkScalar degrees)
1557
Cary Clarkab2621d2018-01-30 10:08:57 -05001558#In Matrix
1559#Line # rotates Matrix ##
Cary Clark09d80c02018-10-31 12:14:03 -04001560#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001561
1562#Example
1563#Description
1564Draw clock hands at time 5:10. The hour hand and minute hand point up and
1565are rotated clockwise.
1566##
Cary Clarkbad5ad72017-08-03 17:14:08 -04001567void draw(SkCanvas* canvas) {
1568 SkPaint paint;
1569 paint.setStyle(SkPaint::kStroke_Style);
1570 canvas->translate(128, 128);
1571 canvas->drawCircle(0, 0, 60, paint);
1572 canvas->save();
1573 canvas->rotate(10 * 360 / 60); // 10 minutes of 60 scaled to 360 degrees
Herb Derbyefe39bc2018-05-01 17:06:20 -04001574 canvas->drawLine(0, 0, 0, -50, paint);
Cary Clarkbad5ad72017-08-03 17:14:08 -04001575 canvas->restore();
1576 canvas->rotate((5 + 10.f/60) * 360 / 12); // 5 and 10/60 hours of 12 scaled to 360 degrees
1577 canvas->drawLine(0, 0, 0, -30, paint);
Cary Clark8032b982017-07-28 11:04:54 -04001578}
1579##
1580
Cary Clark2ade9972017-11-02 17:49:34 -04001581#SeeAlso concat() translate() skew() scale() setMatrix
Cary Clark8032b982017-07-28 11:04:54 -04001582
1583##
1584
1585# ------------------------------------------------------------------------------
1586
1587#Method void rotate(SkScalar degrees, SkScalar px, SkScalar py)
1588
Cary Clarkab2621d2018-01-30 10:08:57 -05001589#In Matrix
Cary Clark09d80c02018-10-31 12:14:03 -04001590#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001591
1592#Example
1593#Height 192
Cary Clarkbad5ad72017-08-03 17:14:08 -04001594void draw(SkCanvas* canvas) {
1595 SkPaint paint;
Mike Reed33535cb2019-01-09 11:25:18 -05001596 SkFont font(nullptr, 96);
1597 canvas->drawString("A1", 130, 100, font, paint);
Cary Clarkbad5ad72017-08-03 17:14:08 -04001598 canvas->rotate(180, 130, 100);
Mike Reed33535cb2019-01-09 11:25:18 -05001599 canvas->drawString("A1", 130, 100, font, paint);
Cary Clark8032b982017-07-28 11:04:54 -04001600}
1601##
1602
Cary Clark2ade9972017-11-02 17:49:34 -04001603#SeeAlso concat() translate() skew() scale() setMatrix
Cary Clark8032b982017-07-28 11:04:54 -04001604
1605##
1606
1607# ------------------------------------------------------------------------------
1608
1609#Method void skew(SkScalar sx, SkScalar sy)
1610
Cary Clarkab2621d2018-01-30 10:08:57 -05001611#In Matrix
1612#Line # skews Matrix ##
Cary Clark09d80c02018-10-31 12:14:03 -04001613#Populate
Cary Clarkbad5ad72017-08-03 17:14:08 -04001614
Cary Clark8032b982017-07-28 11:04:54 -04001615#Example
Herb Derbyefe39bc2018-05-01 17:06:20 -04001616 #Description
Cary Clark5538c132018-06-14 12:28:14 -04001617 Black text mimics an oblique text style by using a negative skew on x-axis
1618 that shifts the geometry to the right as the y-axis values decrease.
1619 Red text uses a positive skew on y-axis to shift the geometry down
1620 as the x-axis values increase.
1621 Blue text combines sx and sy skew to rotate and scale.
Cary Clark8032b982017-07-28 11:04:54 -04001622 ##
Cary Clarkbad5ad72017-08-03 17:14:08 -04001623 SkPaint paint;
Mike Reed33535cb2019-01-09 11:25:18 -05001624 SkFont font(nullptr, 128);
Cary Clarkbad5ad72017-08-03 17:14:08 -04001625 canvas->translate(30, 130);
1626 canvas->save();
1627 canvas->skew(-.5, 0);
Mike Reed33535cb2019-01-09 11:25:18 -05001628 canvas->drawString("A1", 0, 0, font, paint);
Cary Clarkbad5ad72017-08-03 17:14:08 -04001629 canvas->restore();
1630 canvas->save();
1631 canvas->skew(0, .5);
1632 paint.setColor(SK_ColorRED);
Mike Reed33535cb2019-01-09 11:25:18 -05001633 canvas->drawString("A1", 0, 0, font, paint);
Cary Clarkbad5ad72017-08-03 17:14:08 -04001634 canvas->restore();
1635 canvas->skew(-.5, .5);
1636 paint.setColor(SK_ColorBLUE);
Mike Reed33535cb2019-01-09 11:25:18 -05001637 canvas->drawString("A1", 0, 0, font, paint);
Cary Clark8032b982017-07-28 11:04:54 -04001638##
1639
Cary Clark2ade9972017-11-02 17:49:34 -04001640#SeeAlso concat() translate() rotate() scale() setMatrix
Cary Clark8032b982017-07-28 11:04:54 -04001641
1642##
1643
1644# ------------------------------------------------------------------------------
1645
1646#Method void concat(const SkMatrix& matrix)
1647
Cary Clarkab2621d2018-01-30 10:08:57 -05001648#In Matrix
1649#Line # multiplies Matrix by Matrix ##
Cary Clark09d80c02018-10-31 12:14:03 -04001650#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001651
1652#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04001653void draw(SkCanvas* canvas) {
1654 SkPaint paint;
Mike Reed33535cb2019-01-09 11:25:18 -05001655 SkFont font(nullptr, 80);
1656 font.setScaleX(.3);
Cary Clarkbad5ad72017-08-03 17:14:08 -04001657 SkMatrix matrix;
1658 SkRect rect[2] = {{ 10, 20, 90, 110 }, { 40, 130, 140, 180 }};
1659 matrix.setRectToRect(rect[0], rect[1], SkMatrix::kFill_ScaleToFit);
1660 canvas->drawRect(rect[0], paint);
1661 canvas->drawRect(rect[1], paint);
1662 paint.setColor(SK_ColorWHITE);
Mike Reed33535cb2019-01-09 11:25:18 -05001663 canvas->drawString("Here", rect[0].fLeft + 10, rect[0].fBottom - 10, font, paint);
Cary Clarkbad5ad72017-08-03 17:14:08 -04001664 canvas->concat(matrix);
Mike Reed33535cb2019-01-09 11:25:18 -05001665 canvas->drawString("There", rect[0].fLeft + 10, rect[0].fBottom - 10, font, paint);
Cary Clark8032b982017-07-28 11:04:54 -04001666}
1667##
1668
Cary Clark2ade9972017-11-02 17:49:34 -04001669#SeeAlso translate() rotate() scale() skew() setMatrix
Cary Clark8032b982017-07-28 11:04:54 -04001670
1671##
1672
1673# ------------------------------------------------------------------------------
1674
1675#Method void setMatrix(const SkMatrix& matrix)
1676
Cary Clarkab2621d2018-01-30 10:08:57 -05001677#In Matrix
1678#Line # sets Matrix ##
Cary Clark09d80c02018-10-31 12:14:03 -04001679#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001680
1681#Example
1682#Height 128
Cary Clarkbad5ad72017-08-03 17:14:08 -04001683void draw(SkCanvas* canvas) {
1684 SkPaint paint;
Mike Reed33535cb2019-01-09 11:25:18 -05001685 SkFont font;
Cary Clarkbad5ad72017-08-03 17:14:08 -04001686 canvas->scale(4, 6);
Mike Reed33535cb2019-01-09 11:25:18 -05001687 canvas->drawString("truth", 2, 10, font, paint);
Cary Clarkbad5ad72017-08-03 17:14:08 -04001688 SkMatrix matrix;
1689 matrix.setScale(2.8f, 6);
1690 canvas->setMatrix(matrix);
Mike Reed33535cb2019-01-09 11:25:18 -05001691 canvas->drawString("consequences", 2, 20, font, paint);
Cary Clark8032b982017-07-28 11:04:54 -04001692}
1693##
1694
Cary Clark2ade9972017-11-02 17:49:34 -04001695#SeeAlso resetMatrix concat() translate() rotate() scale() skew()
Cary Clark8032b982017-07-28 11:04:54 -04001696
1697##
1698
1699# ------------------------------------------------------------------------------
1700
1701#Method void resetMatrix()
1702
Cary Clarkab2621d2018-01-30 10:08:57 -05001703#In Matrix
1704#Line # resets Matrix to identity ##
Cary Clark09d80c02018-10-31 12:14:03 -04001705#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001706
1707#Example
1708#Height 128
Cary Clarkbad5ad72017-08-03 17:14:08 -04001709void draw(SkCanvas* canvas) {
1710 SkPaint paint;
Mike Reed33535cb2019-01-09 11:25:18 -05001711 SkFont font;
Cary Clarkbad5ad72017-08-03 17:14:08 -04001712 canvas->scale(4, 6);
Mike Reed33535cb2019-01-09 11:25:18 -05001713 canvas->drawString("truth", 2, 10, font, paint);
Cary Clarkbad5ad72017-08-03 17:14:08 -04001714 canvas->resetMatrix();
1715 canvas->scale(2.8f, 6);
Mike Reed33535cb2019-01-09 11:25:18 -05001716 canvas->drawString("consequences", 2, 20, font, paint);
Cary Clark8032b982017-07-28 11:04:54 -04001717}
1718##
1719
Cary Clark2ade9972017-11-02 17:49:34 -04001720#SeeAlso setMatrix concat() translate() rotate() scale() skew()
Cary Clark8032b982017-07-28 11:04:54 -04001721
1722##
1723
1724# ------------------------------------------------------------------------------
1725
1726#Method const SkMatrix& getTotalMatrix() const
1727
Cary Clarkab2621d2018-01-30 10:08:57 -05001728#In Matrix
1729#Line # returns Matrix ##
Cary Clark09d80c02018-10-31 12:14:03 -04001730#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001731
1732#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04001733 SkDebugf("isIdentity %s\n", canvas->getTotalMatrix().isIdentity() ? "true" : "false");
1734 #StdOut
1735 isIdentity true
1736 ##
Cary Clark8032b982017-07-28 11:04:54 -04001737##
1738
Cary Clark2ade9972017-11-02 17:49:34 -04001739#SeeAlso setMatrix resetMatrix concat()
Cary Clark8032b982017-07-28 11:04:54 -04001740
1741##
1742
Cary Clark08895c42018-02-01 09:37:32 -05001743#Subtopic Matrix ##
Cary Clark8032b982017-07-28 11:04:54 -04001744
1745# ------------------------------------------------------------------------------
Cary Clark08895c42018-02-01 09:37:32 -05001746#Subtopic Clip
1747#Line # stack of clipping Paths ##
Cary Clark8032b982017-07-28 11:04:54 -04001748
1749Clip is built from a stack of clipping paths. Each Path in the
Herb Derbyefe39bc2018-05-01 17:06:20 -04001750stack can be constructed from one or more Path_Contour elements. The
Cary Clark8032b982017-07-28 11:04:54 -04001751Path_Contour may be composed of any number of Path_Verb segments. Each
1752Path_Contour forms a closed area; Path_Fill_Type defines the area enclosed
1753by Path_Contour.
1754
1755Clip stack of Path elements successfully restrict the Path area. Each
Herb Derbyefe39bc2018-05-01 17:06:20 -04001756Path is transformed by Matrix, then intersected with or subtracted from the
Cary Clark8032b982017-07-28 11:04:54 -04001757prior Clip to form the replacement Clip. Use SkClipOp::kDifference
1758to subtract Path from Clip; use SkClipOp::kIntersect to intersect Path
1759with Clip.
1760
Cary Clarkffb3d682018-05-17 12:17:28 -04001761A clipping Path may be Anti_Aliased; if Path, after transformation, is
1762composed of horizontal and vertical lines, clearing Anti_Alias allows whole pixels
Cary Clarkce101242017-09-01 15:51:02 -04001763to either be inside or outside the clip. The fastest drawing has a Aliased,
1764rectangular clip.
Cary Clark8032b982017-07-28 11:04:54 -04001765
Cary Clarkffb3d682018-05-17 12:17:28 -04001766If clipping Path has Anti_Alias set, clip may partially clip a pixel, requiring
Herb Derbyefe39bc2018-05-01 17:06:20 -04001767that drawing blend partially with the destination along the edge. A rotated
Cary Clarkffb3d682018-05-17 12:17:28 -04001768rectangular Anti_Aliased clip looks smoother but draws slower.
Cary Clark8032b982017-07-28 11:04:54 -04001769
1770Clip can combine with Rect and Round_Rect primitives; like
1771Path, these are transformed by Matrix before they are combined with Clip.
1772
1773Clip can combine with Region. Region is assumed to be in Device coordinates
1774and is unaffected by Matrix.
1775
1776#Example
1777#Height 90
1778 #Description
Cary Clarkffb3d682018-05-17 12:17:28 -04001779 Draw a red circle with an Aliased clip and an Anti_Aliased clip.
Cary Clark8032b982017-07-28 11:04:54 -04001780 Use an image filter to zoom into the pixels drawn.
Cary Clarkce101242017-09-01 15:51:02 -04001781 The edge of the Aliased clip fully draws pixels in the red circle.
Cary Clarkffb3d682018-05-17 12:17:28 -04001782 The edge of the Anti_Aliased clip partially draws pixels in the red circle.
Cary Clark8032b982017-07-28 11:04:54 -04001783 ##
Cary Clarkbad5ad72017-08-03 17:14:08 -04001784 SkPaint redPaint, scalePaint;
1785 redPaint.setAntiAlias(true);
1786 redPaint.setColor(SK_ColorRED);
1787 canvas->save();
1788 for (bool antialias : { false, true } ) {
1789 canvas->save();
1790 canvas->clipRect(SkRect::MakeWH(19.5f, 11.5f), antialias);
1791 canvas->drawCircle(17, 11, 8, redPaint);
1792 canvas->restore();
1793 canvas->translate(16, 0);
1794 }
1795 canvas->restore();
1796 SkMatrix matrix;
1797 matrix.setScale(6, 6);
1798 scalePaint.setImageFilter(
1799 SkImageFilter::MakeMatrixFilter(matrix, kNone_SkFilterQuality, nullptr));
1800 SkCanvas::SaveLayerRec saveLayerRec(
Herb Derbyefe39bc2018-05-01 17:06:20 -04001801 nullptr, &scalePaint, SkCanvas::kInitWithPrevious_SaveLayerFlag);
Cary Clarkbad5ad72017-08-03 17:14:08 -04001802 canvas->saveLayer(saveLayerRec);
Cary Clark8032b982017-07-28 11:04:54 -04001803 canvas->restore();
1804##
1805
1806#Method void clipRect(const SkRect& rect, SkClipOp op, bool doAntiAlias)
1807
Cary Clarkab2621d2018-01-30 10:08:57 -05001808#In Clip
1809#Line # combines Clip with Rect ##
Cary Clark09d80c02018-10-31 12:14:03 -04001810#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001811
1812#Example
1813#Height 128
Cary Clarkbad5ad72017-08-03 17:14:08 -04001814void draw(SkCanvas* canvas) {
1815 canvas->rotate(10);
1816 SkPaint paint;
1817 paint.setAntiAlias(true);
1818 for (auto alias: { false, true } ) {
1819 canvas->save();
1820 canvas->clipRect(SkRect::MakeWH(90, 80), SkClipOp::kIntersect, alias);
1821 canvas->drawCircle(100, 60, 60, paint);
1822 canvas->restore();
1823 canvas->translate(80, 0);
1824 }
Cary Clark8032b982017-07-28 11:04:54 -04001825}
1826##
1827
Cary Clark2ade9972017-11-02 17:49:34 -04001828#SeeAlso clipRRect clipPath clipRegion
Cary Clark8032b982017-07-28 11:04:54 -04001829
1830##
1831
Herb Derbyefe39bc2018-05-01 17:06:20 -04001832#Method void clipRect(const SkRect& rect, SkClipOp op)
Cary Clark8032b982017-07-28 11:04:54 -04001833
Cary Clarkab2621d2018-01-30 10:08:57 -05001834#In Clip
Cary Clark09d80c02018-10-31 12:14:03 -04001835#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001836
1837#Example
1838#Height 192
1839#Width 280
Cary Clarkbad5ad72017-08-03 17:14:08 -04001840void draw(SkCanvas* canvas) {
1841 SkPaint paint;
1842 for (SkClipOp op: { SkClipOp::kIntersect, SkClipOp::kDifference } ) {
1843 canvas->save();
1844 canvas->clipRect(SkRect::MakeWH(90, 120), op, false);
1845 canvas->drawCircle(100, 100, 60, paint);
1846 canvas->restore();
1847 canvas->translate(80, 0);
1848 }
Cary Clark8032b982017-07-28 11:04:54 -04001849}
1850##
1851
Cary Clark2ade9972017-11-02 17:49:34 -04001852#SeeAlso clipRRect clipPath clipRegion
Cary Clark8032b982017-07-28 11:04:54 -04001853
1854##
1855
Herb Derbyefe39bc2018-05-01 17:06:20 -04001856#Method void clipRect(const SkRect& rect, bool doAntiAlias = false)
Cary Clark8032b982017-07-28 11:04:54 -04001857
Cary Clarkab2621d2018-01-30 10:08:57 -05001858#In Clip
Cary Clark09d80c02018-10-31 12:14:03 -04001859#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001860
1861#Example
1862#Height 133
1863 #Description
Cary Clarkce101242017-09-01 15:51:02 -04001864 A circle drawn in pieces looks uniform when drawn Aliased.
Cary Clarkffb3d682018-05-17 12:17:28 -04001865 The same circle pieces blend with pixels more than once when Anti_Aliased,
Cary Clark8032b982017-07-28 11:04:54 -04001866 visible as a thin pair of lines through the right circle.
1867 ##
Cary Clarkbad5ad72017-08-03 17:14:08 -04001868void draw(SkCanvas* canvas) {
1869 canvas->clear(SK_ColorWHITE);
1870 SkPaint paint;
1871 paint.setAntiAlias(true);
1872 paint.setColor(0x8055aaff);
1873 SkRect clipRect = { 0, 0, 87.4f, 87.4f };
1874 for (auto alias: { false, true } ) {
1875 canvas->save();
1876 canvas->clipRect(clipRect, SkClipOp::kIntersect, alias);
1877 canvas->drawCircle(67, 67, 60, paint);
1878 canvas->restore();
1879 canvas->save();
1880 canvas->clipRect(clipRect, SkClipOp::kDifference, alias);
1881 canvas->drawCircle(67, 67, 60, paint);
1882 canvas->restore();
1883 canvas->translate(120, 0);
1884 }
Cary Clark8032b982017-07-28 11:04:54 -04001885}
1886##
1887
Cary Clark2ade9972017-11-02 17:49:34 -04001888#SeeAlso clipRRect clipPath clipRegion
Cary Clark8032b982017-07-28 11:04:54 -04001889
1890##
1891
Cary Clark8032b982017-07-28 11:04:54 -04001892#Method void clipRRect(const SkRRect& rrect, SkClipOp op, bool doAntiAlias)
1893
Cary Clarkab2621d2018-01-30 10:08:57 -05001894#In Clip
1895#Line # combines Clip with Round_Rect ##
Cary Clark09d80c02018-10-31 12:14:03 -04001896#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001897
1898#Example
1899#Height 128
Cary Clarkbad5ad72017-08-03 17:14:08 -04001900void draw(SkCanvas* canvas) {
1901 canvas->clear(SK_ColorWHITE);
1902 SkPaint paint;
1903 paint.setAntiAlias(true);
1904 paint.setColor(0x8055aaff);
1905 SkRRect oval;
1906 oval.setOval({10, 20, 90, 100});
1907 canvas->clipRRect(oval, SkClipOp::kIntersect, true);
1908 canvas->drawCircle(70, 100, 60, paint);
Cary Clark8032b982017-07-28 11:04:54 -04001909}
1910##
1911
Cary Clark2ade9972017-11-02 17:49:34 -04001912#SeeAlso clipRect clipPath clipRegion
Cary Clark8032b982017-07-28 11:04:54 -04001913
1914##
1915
Herb Derbyefe39bc2018-05-01 17:06:20 -04001916#Method void clipRRect(const SkRRect& rrect, SkClipOp op)
Cary Clark8032b982017-07-28 11:04:54 -04001917
Cary Clarkab2621d2018-01-30 10:08:57 -05001918#In Clip
Cary Clark09d80c02018-10-31 12:14:03 -04001919#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001920
1921#Example
1922#Height 128
Cary Clarkbad5ad72017-08-03 17:14:08 -04001923void draw(SkCanvas* canvas) {
1924 SkPaint paint;
1925 paint.setColor(0x8055aaff);
1926 auto oval = SkRRect::MakeOval({10, 20, 90, 100});
1927 canvas->clipRRect(oval, SkClipOp::kIntersect);
1928 canvas->drawCircle(70, 100, 60, paint);
Cary Clark8032b982017-07-28 11:04:54 -04001929}
1930##
1931
Cary Clark2ade9972017-11-02 17:49:34 -04001932#SeeAlso clipRect clipPath clipRegion
Cary Clark8032b982017-07-28 11:04:54 -04001933
1934##
1935
Herb Derbyefe39bc2018-05-01 17:06:20 -04001936#Method void clipRRect(const SkRRect& rrect, bool doAntiAlias = false)
Cary Clark8032b982017-07-28 11:04:54 -04001937
Cary Clarkab2621d2018-01-30 10:08:57 -05001938#In Clip
Cary Clark09d80c02018-10-31 12:14:03 -04001939#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001940
1941#Example
1942#Height 128
Cary Clarkbad5ad72017-08-03 17:14:08 -04001943void draw(SkCanvas* canvas) {
1944 SkPaint paint;
1945 paint.setAntiAlias(true);
1946 auto oval = SkRRect::MakeRectXY({10, 20, 90, 100}, 9, 13);
1947 canvas->clipRRect(oval, true);
1948 canvas->drawCircle(70, 100, 60, paint);
Cary Clark8032b982017-07-28 11:04:54 -04001949}
1950##
1951
Cary Clark2ade9972017-11-02 17:49:34 -04001952#SeeAlso clipRect clipPath clipRegion
Cary Clark8032b982017-07-28 11:04:54 -04001953
1954##
1955
1956#Method void clipPath(const SkPath& path, SkClipOp op, bool doAntiAlias)
1957
Cary Clarkab2621d2018-01-30 10:08:57 -05001958#In Clip
1959#Line # combines Clip with Path ##
Cary Clark09d80c02018-10-31 12:14:03 -04001960#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001961
1962#Example
1963#Description
1964Top figure uses SkPath::kInverseWinding_FillType and SkClipOp::kDifference;
1965area outside clip is subtracted from circle.
1966
1967Bottom figure uses SkPath::kWinding_FillType and SkClipOp::kIntersect;
1968area inside clip is intersected with circle.
1969##
Cary Clarkbad5ad72017-08-03 17:14:08 -04001970void draw(SkCanvas* canvas) {
1971 SkPaint paint;
1972 paint.setAntiAlias(true);
1973 SkPath path;
1974 path.addRect({20, 30, 100, 110});
1975 path.setFillType(SkPath::kInverseWinding_FillType);
1976 canvas->save();
1977 canvas->clipPath(path, SkClipOp::kDifference, false);
1978 canvas->drawCircle(70, 100, 60, paint);
1979 canvas->restore();
1980 canvas->translate(100, 100);
1981 path.setFillType(SkPath::kWinding_FillType);
1982 canvas->clipPath(path, SkClipOp::kIntersect, false);
1983 canvas->drawCircle(70, 100, 60, paint);
Cary Clark8032b982017-07-28 11:04:54 -04001984}
1985##
1986
Cary Clark2ade9972017-11-02 17:49:34 -04001987#SeeAlso clipRect clipRRect clipRegion
Cary Clark8032b982017-07-28 11:04:54 -04001988
1989##
1990
Herb Derbyefe39bc2018-05-01 17:06:20 -04001991#Method void clipPath(const SkPath& path, SkClipOp op)
Cary Clark8032b982017-07-28 11:04:54 -04001992
Cary Clarkab2621d2018-01-30 10:08:57 -05001993#In Clip
Cary Clark09d80c02018-10-31 12:14:03 -04001994#Populate
Cary Clark8032b982017-07-28 11:04:54 -04001995
1996#Example
1997#Description
Cary Clarkbc5697d2017-10-04 14:31:33 -04001998Overlapping Rects form a clip. When clip Path_Fill_Type is set to
Herb Derbyefe39bc2018-05-01 17:06:20 -04001999SkPath::kWinding_FillType, the overlap is included. Set to
Cary Clark8032b982017-07-28 11:04:54 -04002000SkPath::kEvenOdd_FillType, the overlap is excluded and forms a hole.
2001##
Cary Clarkbad5ad72017-08-03 17:14:08 -04002002void draw(SkCanvas* canvas) {
2003 SkPaint paint;
2004 paint.setAntiAlias(true);
2005 SkPath path;
2006 path.addRect({20, 15, 100, 95});
2007 path.addRect({50, 65, 130, 135});
2008 path.setFillType(SkPath::kWinding_FillType);
2009 canvas->save();
2010 canvas->clipPath(path, SkClipOp::kIntersect);
2011 canvas->drawCircle(70, 85, 60, paint);
2012 canvas->restore();
2013 canvas->translate(100, 100);
2014 path.setFillType(SkPath::kEvenOdd_FillType);
2015 canvas->clipPath(path, SkClipOp::kIntersect);
2016 canvas->drawCircle(70, 85, 60, paint);
Cary Clark8032b982017-07-28 11:04:54 -04002017}
2018##
2019
Cary Clark2ade9972017-11-02 17:49:34 -04002020#SeeAlso clipRect clipRRect clipRegion
Cary Clark8032b982017-07-28 11:04:54 -04002021
2022##
2023
Herb Derbyefe39bc2018-05-01 17:06:20 -04002024#Method void clipPath(const SkPath& path, bool doAntiAlias = false)
Cary Clark8032b982017-07-28 11:04:54 -04002025
Cary Clarkab2621d2018-01-30 10:08:57 -05002026#In Clip
Cary Clark09d80c02018-10-31 12:14:03 -04002027#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002028
2029#Example
2030#Height 212
2031#Description
Herb Derbyefe39bc2018-05-01 17:06:20 -04002032Clip loops over itself covering its center twice. When clip Path_Fill_Type
2033is set to SkPath::kWinding_FillType, the overlap is included. Set to
Cary Clark8032b982017-07-28 11:04:54 -04002034SkPath::kEvenOdd_FillType, the overlap is excluded and forms a hole.
2035##
Cary Clarkbad5ad72017-08-03 17:14:08 -04002036void draw(SkCanvas* canvas) {
2037 SkPaint paint;
2038 paint.setAntiAlias(true);
2039 SkPath path;
2040 SkPoint poly[] = {{20, 20}, { 80, 20}, { 80, 80}, {40, 80},
2041 {40, 40}, {100, 40}, {100, 100}, {20, 100}};
2042 path.addPoly(poly, SK_ARRAY_COUNT(poly), true);
2043 path.setFillType(SkPath::kWinding_FillType);
2044 canvas->save();
2045 canvas->clipPath(path, SkClipOp::kIntersect);
2046 canvas->drawCircle(50, 50, 45, paint);
2047 canvas->restore();
2048 canvas->translate(100, 100);
2049 path.setFillType(SkPath::kEvenOdd_FillType);
2050 canvas->clipPath(path, SkClipOp::kIntersect);
2051 canvas->drawCircle(50, 50, 45, paint);
Cary Clark8032b982017-07-28 11:04:54 -04002052}
2053##
2054
Cary Clark2ade9972017-11-02 17:49:34 -04002055#SeeAlso clipRect clipRRect clipRegion
Cary Clark8032b982017-07-28 11:04:54 -04002056
2057##
2058
2059# ------------------------------------------------------------------------------
2060
Cary Clark8032b982017-07-28 11:04:54 -04002061#Method void clipRegion(const SkRegion& deviceRgn, SkClipOp op = SkClipOp::kIntersect)
2062
Cary Clarkab2621d2018-01-30 10:08:57 -05002063#In Clip
2064#Line # combines Clip with Region ##
Cary Clark09d80c02018-10-31 12:14:03 -04002065#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002066
2067#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002068#Description
Cary Clarkce101242017-09-01 15:51:02 -04002069 region is unaffected by canvas rotation; iRect is affected by canvas rotation.
2070 Both clips are Aliased; this is not noticeable on Region clip because it
Cary Clarkbad5ad72017-08-03 17:14:08 -04002071 aligns to pixel boundaries.
2072##
2073void draw(SkCanvas* canvas) {
2074 SkPaint paint;
2075 paint.setAntiAlias(true);
2076 SkIRect iRect = {30, 40, 120, 130 };
2077 SkRegion region(iRect);
2078 canvas->rotate(10);
2079 canvas->save();
2080 canvas->clipRegion(region, SkClipOp::kIntersect);
2081 canvas->drawCircle(50, 50, 45, paint);
2082 canvas->restore();
2083 canvas->translate(100, 100);
2084 canvas->clipRect(SkRect::Make(iRect), SkClipOp::kIntersect);
2085 canvas->drawCircle(50, 50, 45, paint);
Cary Clark8032b982017-07-28 11:04:54 -04002086}
2087##
2088
Cary Clark2ade9972017-11-02 17:49:34 -04002089#SeeAlso clipRect clipRRect clipPath
Cary Clark8032b982017-07-28 11:04:54 -04002090
2091##
2092
2093#Method bool quickReject(const SkRect& rect) const
2094
Cary Clarkab2621d2018-01-30 10:08:57 -05002095#In Clip
2096#Line # returns if Rect is outside Clip ##
Cary Clark09d80c02018-10-31 12:14:03 -04002097#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002098
2099#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002100void draw(SkCanvas* canvas) {
Herb Derbyefe39bc2018-05-01 17:06:20 -04002101 SkRect testRect = {30, 30, 120, 129 };
2102 SkRect clipRect = {30, 130, 120, 230 };
Cary Clarkbad5ad72017-08-03 17:14:08 -04002103 canvas->save();
2104 canvas->clipRect(clipRect);
2105 SkDebugf("quickReject %s\n", canvas->quickReject(testRect) ? "true" : "false");
2106 canvas->restore();
2107 canvas->rotate(10);
2108 canvas->clipRect(clipRect);
2109 SkDebugf("quickReject %s\n", canvas->quickReject(testRect) ? "true" : "false");
Cary Clark8032b982017-07-28 11:04:54 -04002110}
2111 #StdOut
Cary Clarkbad5ad72017-08-03 17:14:08 -04002112 quickReject true
Cary Clark8032b982017-07-28 11:04:54 -04002113 quickReject false
2114 ##
2115##
2116
Cary Clark2ade9972017-11-02 17:49:34 -04002117#SeeAlso getLocalClipBounds getTotalMatrix SkBitmap::drawsNothing
Cary Clark8032b982017-07-28 11:04:54 -04002118
2119##
2120
2121#Method bool quickReject(const SkPath& path) const
2122
Cary Clarkab2621d2018-01-30 10:08:57 -05002123#In Clip
Cary Clark09d80c02018-10-31 12:14:03 -04002124#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002125
2126#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002127void draw(SkCanvas* canvas) {
Herb Derbyefe39bc2018-05-01 17:06:20 -04002128 SkPoint testPoints[] = {{30, 30}, {120, 30}, {120, 129} };
2129 SkPoint clipPoints[] = {{30, 130}, {120, 130}, {120, 230} };
Cary Clarkbad5ad72017-08-03 17:14:08 -04002130 SkPath testPath, clipPath;
2131 testPath.addPoly(testPoints, SK_ARRAY_COUNT(testPoints), true);
2132 clipPath.addPoly(clipPoints, SK_ARRAY_COUNT(clipPoints), true);
2133 canvas->save();
2134 canvas->clipPath(clipPath);
2135 SkDebugf("quickReject %s\n", canvas->quickReject(testPath) ? "true" : "false");
2136 canvas->restore();
2137 canvas->rotate(10);
2138 canvas->clipPath(clipPath);
2139 SkDebugf("quickReject %s\n", canvas->quickReject(testPath) ? "true" : "false");
Cary Clark8032b982017-07-28 11:04:54 -04002140 #StdOut
Cary Clarkbad5ad72017-08-03 17:14:08 -04002141 quickReject true
Cary Clark8032b982017-07-28 11:04:54 -04002142 quickReject false
2143 ##
2144}
2145##
2146
Cary Clark2ade9972017-11-02 17:49:34 -04002147#SeeAlso getLocalClipBounds getTotalMatrix SkBitmap::drawsNothing
Cary Clark8032b982017-07-28 11:04:54 -04002148
2149##
2150
Herb Derbyefe39bc2018-05-01 17:06:20 -04002151#Method SkRect getLocalClipBounds() const
Cary Clark8032b982017-07-28 11:04:54 -04002152
Cary Clarkab2621d2018-01-30 10:08:57 -05002153#In Clip
2154#Line # returns Clip bounds in source coordinates ##
Cary Clark09d80c02018-10-31 12:14:03 -04002155#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002156
2157#Example
Herb Derbyefe39bc2018-05-01 17:06:20 -04002158 #Description
Cary Clark8032b982017-07-28 11:04:54 -04002159 Initial bounds is device bounds outset by 1 on all sides.
2160 Clipped bounds is clipPath bounds outset by 1 on all sides.
Cary Clark5538c132018-06-14 12:28:14 -04002161 Scaling the canvas by two on both axes scales the local bounds by 1/2
2162 on both axes.
Cary Clark8032b982017-07-28 11:04:54 -04002163 ##
Cary Clarkbad5ad72017-08-03 17:14:08 -04002164 SkCanvas local(256, 256);
2165 canvas = &local;
2166 SkRect bounds = canvas->getLocalClipBounds();
2167 SkDebugf("left:%g top:%g right:%g bottom:%g\n",
2168 bounds.fLeft, bounds.fTop, bounds.fRight, bounds.fBottom);
Herb Derbyefe39bc2018-05-01 17:06:20 -04002169 SkPoint clipPoints[] = {{30, 130}, {120, 130}, {120, 230} };
Cary Clarkbad5ad72017-08-03 17:14:08 -04002170 SkPath clipPath;
2171 clipPath.addPoly(clipPoints, SK_ARRAY_COUNT(clipPoints), true);
2172 canvas->clipPath(clipPath);
2173 bounds = canvas->getLocalClipBounds();
2174 SkDebugf("left:%g top:%g right:%g bottom:%g\n",
2175 bounds.fLeft, bounds.fTop, bounds.fRight, bounds.fBottom);
2176 canvas->scale(2, 2);
2177 bounds = canvas->getLocalClipBounds();
2178 SkDebugf("left:%g top:%g right:%g bottom:%g\n",
2179 bounds.fLeft, bounds.fTop, bounds.fRight, bounds.fBottom);
2180 #StdOut
2181 left:-1 top:-1 right:257 bottom:257
2182 left:29 top:129 right:121 bottom:231
2183 left:14.5 top:64.5 right:60.5 bottom:115.5
2184 ##
Cary Clark8032b982017-07-28 11:04:54 -04002185##
2186
2187# local canvas in example works around bug in fiddle ##
Cary Clark4855f782018-02-06 09:41:53 -05002188#Bug 6524
Cary Clark2ade9972017-11-02 17:49:34 -04002189#SeeAlso getDeviceClipBounds getBaseLayerSize quickReject
Cary Clark8032b982017-07-28 11:04:54 -04002190
2191##
2192
Herb Derbyefe39bc2018-05-01 17:06:20 -04002193#Method bool getLocalClipBounds(SkRect* bounds) const
Cary Clark8032b982017-07-28 11:04:54 -04002194
Cary Clarkab2621d2018-01-30 10:08:57 -05002195#In Clip
Cary Clark09d80c02018-10-31 12:14:03 -04002196#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002197
2198#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002199 void draw(SkCanvas* canvas) {
2200 SkCanvas local(256, 256);
2201 canvas = &local;
2202 SkRect bounds;
2203 SkDebugf("local bounds empty = %s\n", canvas->getLocalClipBounds(&bounds)
2204 ? "false" : "true");
2205 SkPath path;
2206 canvas->clipPath(path);
2207 SkDebugf("local bounds empty = %s\n", canvas->getLocalClipBounds(&bounds)
2208 ? "false" : "true");
Cary Clark8032b982017-07-28 11:04:54 -04002209 }
2210 #StdOut
Cary Clarkbad5ad72017-08-03 17:14:08 -04002211 local bounds empty = false
Cary Clark8032b982017-07-28 11:04:54 -04002212 local bounds empty = true
2213 ##
2214##
2215
2216# local canvas in example works around bug in fiddle ##
Cary Clark4855f782018-02-06 09:41:53 -05002217#Bug 6524
Cary Clark2ade9972017-11-02 17:49:34 -04002218#SeeAlso getDeviceClipBounds getBaseLayerSize quickReject
Cary Clark8032b982017-07-28 11:04:54 -04002219
2220##
2221
Herb Derbyefe39bc2018-05-01 17:06:20 -04002222#Method SkIRect getDeviceClipBounds() const
Cary Clark8032b982017-07-28 11:04:54 -04002223
Cary Clarkab2621d2018-01-30 10:08:57 -05002224#In Clip
2225#Line # returns IRect bounds of Clip ##
Cary Clark09d80c02018-10-31 12:14:03 -04002226#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002227
2228#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002229void draw(SkCanvas* canvas) {
2230 #Description
Cary Clark8032b982017-07-28 11:04:54 -04002231 Initial bounds is device bounds, not outset.
2232 Clipped bounds is clipPath bounds, not outset.
Cary Clark5538c132018-06-14 12:28:14 -04002233 Scaling the canvas by 1/2 on both axes scales the device bounds by 1/2
2234 on both axes.
Cary Clarkbad5ad72017-08-03 17:14:08 -04002235 ##
2236 SkCanvas device(256, 256);
2237 canvas = &device;
2238 SkIRect bounds = canvas->getDeviceClipBounds();
2239 SkDebugf("left:%d top:%d right:%d bottom:%d\n",
2240 bounds.fLeft, bounds.fTop, bounds.fRight, bounds.fBottom);
Herb Derbyefe39bc2018-05-01 17:06:20 -04002241 SkPoint clipPoints[] = {{30, 130}, {120, 130}, {120, 230} };
Cary Clarkbad5ad72017-08-03 17:14:08 -04002242 SkPath clipPath;
2243 clipPath.addPoly(clipPoints, SK_ARRAY_COUNT(clipPoints), true);
2244 canvas->save();
2245 canvas->clipPath(clipPath);
2246 bounds = canvas->getDeviceClipBounds();
2247 SkDebugf("left:%d top:%d right:%d bottom:%d\n",
2248 bounds.fLeft, bounds.fTop, bounds.fRight, bounds.fBottom);
2249 canvas->restore();
2250 canvas->scale(1.f/2, 1.f/2);
2251 canvas->clipPath(clipPath);
2252 bounds = canvas->getDeviceClipBounds();
2253 SkDebugf("left:%d top:%d right:%d bottom:%d\n",
2254 bounds.fLeft, bounds.fTop, bounds.fRight, bounds.fBottom);
Cary Clark8032b982017-07-28 11:04:54 -04002255 #StdOut
Cary Clarkbad5ad72017-08-03 17:14:08 -04002256 left:0 top:0 right:256 bottom:256
2257 left:30 top:130 right:120 bottom:230
Cary Clark8032b982017-07-28 11:04:54 -04002258 left:15 top:65 right:60 bottom:115
2259 ##
2260}
2261##
2262
2263#ToDo some confusion on why with an identity Matrix local and device are different ##
Cary Clark2ade9972017-11-02 17:49:34 -04002264#SeeAlso getLocalClipBounds getBaseLayerSize quickReject
Cary Clark8032b982017-07-28 11:04:54 -04002265
2266# device canvas in example works around bug in fiddle ##
Cary Clark4855f782018-02-06 09:41:53 -05002267#Bug 6524
Cary Clark8032b982017-07-28 11:04:54 -04002268
2269##
2270
Herb Derbyefe39bc2018-05-01 17:06:20 -04002271#Method bool getDeviceClipBounds(SkIRect* bounds) const
Cary Clark8032b982017-07-28 11:04:54 -04002272
Cary Clarkab2621d2018-01-30 10:08:57 -05002273#In Clip
Cary Clark09d80c02018-10-31 12:14:03 -04002274#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002275
2276#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002277 void draw(SkCanvas* canvas) {
2278 SkIRect bounds;
2279 SkDebugf("device bounds empty = %s\n", canvas->getDeviceClipBounds(&bounds)
2280 ? "false" : "true");
2281 SkPath path;
2282 canvas->clipPath(path);
2283 SkDebugf("device bounds empty = %s\n", canvas->getDeviceClipBounds(&bounds)
2284 ? "false" : "true");
Cary Clark8032b982017-07-28 11:04:54 -04002285 }
2286 #StdOut
Cary Clarkbad5ad72017-08-03 17:14:08 -04002287 device bounds empty = false
Cary Clark8032b982017-07-28 11:04:54 -04002288 device bounds empty = true
2289 ##
2290##
2291
Cary Clark2ade9972017-11-02 17:49:34 -04002292#SeeAlso getLocalClipBounds getBaseLayerSize quickReject
Cary Clark8032b982017-07-28 11:04:54 -04002293
2294##
2295
Cary Clark08895c42018-02-01 09:37:32 -05002296#Subtopic Clip ##
Cary Clark8032b982017-07-28 11:04:54 -04002297
2298# ------------------------------------------------------------------------------
Cary Clark78de7512018-02-07 07:27:09 -05002299#Subtopic Draw
Cary Clark78de7512018-02-07 07:27:09 -05002300#Line # draws into Canvas ##
2301##
Cary Clark8032b982017-07-28 11:04:54 -04002302
2303#Method void drawColor(SkColor color, SkBlendMode mode = SkBlendMode::kSrcOver)
Cary Clark78de7512018-02-07 07:27:09 -05002304#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002305#Line # fills Clip with Color and Blend_Mode ##
Cary Clark09d80c02018-10-31 12:14:03 -04002306#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002307
2308#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002309 canvas->drawColor(SK_ColorRED);
2310 canvas->clipRect(SkRect::MakeWH(150, 150));
2311 canvas->drawColor(SkColorSetARGB(0x80, 0x00, 0xFF, 0x00), SkBlendMode::kPlus);
2312 canvas->clipRect(SkRect::MakeWH(75, 75));
2313 canvas->drawColor(SkColorSetARGB(0x80, 0x00, 0x00, 0xFF), SkBlendMode::kPlus);
Cary Clark8032b982017-07-28 11:04:54 -04002314##
2315
Cary Clark2ade9972017-11-02 17:49:34 -04002316#SeeAlso clear SkBitmap::erase drawPaint
Cary Clark8032b982017-07-28 11:04:54 -04002317
2318##
2319
2320# ------------------------------------------------------------------------------
2321
Herb Derbyefe39bc2018-05-01 17:06:20 -04002322#Method void clear(SkColor color)
Cary Clark78de7512018-02-07 07:27:09 -05002323#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002324#Line # fills Clip with Color ##
Cary Clark09d80c02018-10-31 12:14:03 -04002325#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002326
2327#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002328void draw(SkCanvas* canvas) {
2329 canvas->save();
2330 canvas->clipRect(SkRect::MakeWH(256, 128));
Herb Derbyefe39bc2018-05-01 17:06:20 -04002331 canvas->clear(SkColorSetARGB(0x80, 0xFF, 0x00, 0x00));
Cary Clarkbad5ad72017-08-03 17:14:08 -04002332 canvas->restore();
2333 canvas->save();
2334 canvas->clipRect(SkRect::MakeWH(150, 192));
2335 canvas->clear(SkColorSetARGB(0x80, 0x00, 0xFF, 0x00));
2336 canvas->restore();
2337 canvas->clipRect(SkRect::MakeWH(75, 256));
2338 canvas->clear(SkColorSetARGB(0x80, 0x00, 0x00, 0xFF));
Cary Clark8032b982017-07-28 11:04:54 -04002339}
2340##
2341
Cary Clark2ade9972017-11-02 17:49:34 -04002342#SeeAlso drawColor SkBitmap::erase drawPaint
Cary Clark8032b982017-07-28 11:04:54 -04002343
2344##
2345
2346# ------------------------------------------------------------------------------
2347
Herb Derbyefe39bc2018-05-01 17:06:20 -04002348#Method void discard()
Cary Clark78de7512018-02-07 07:27:09 -05002349#In Utility
Cary Clarkab2621d2018-01-30 10:08:57 -05002350#Line # makes Canvas contents undefined ##
Cary Clark09d80c02018-10-31 12:14:03 -04002351#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002352
Herb Derbyefe39bc2018-05-01 17:06:20 -04002353#NoExample
Cary Clark8032b982017-07-28 11:04:54 -04002354##
2355
Cary Clarkabaffd82018-11-15 08:25:12 -05002356#SeeAlso flush() GrContext::abandonContext
Cary Clark09d80c02018-10-31 12:14:03 -04002357
Cary Clark8032b982017-07-28 11:04:54 -04002358##
2359
2360# ------------------------------------------------------------------------------
2361
2362#Method void drawPaint(const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05002363#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002364#Line # fills Clip with Paint ##
Cary Clark09d80c02018-10-31 12:14:03 -04002365#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002366
2367#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002368void draw(SkCanvas* canvas) {
2369 SkColor colors[] = { SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE };
2370 SkScalar pos[] = { 0, SK_Scalar1/2, SK_Scalar1 };
2371 SkPaint paint;
2372 paint.setShader(SkGradientShader::MakeSweep(256, 256, colors, pos, SK_ARRAY_COUNT(colors)));
2373 canvas->drawPaint(paint);
Cary Clark8032b982017-07-28 11:04:54 -04002374}
2375##
2376
Cary Clark2ade9972017-11-02 17:49:34 -04002377#SeeAlso clear drawColor SkBitmap::erase
Cary Clark8032b982017-07-28 11:04:54 -04002378
2379##
2380
2381# ------------------------------------------------------------------------------
2382
2383#Enum PointMode
Cary Clark08895c42018-02-01 09:37:32 -05002384#Line # sets drawPoints options ##
Cary Clark8032b982017-07-28 11:04:54 -04002385
2386#Code
Cary Clark61313f32018-10-08 14:57:48 -04002387#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002388##
2389
2390Selects if an array of points are drawn as discrete points, as lines, or as
2391an open polygon.
2392
2393#Const kPoints_PointMode 0
Cary Clark682c58d2018-05-16 07:07:07 -04002394#Line # draw each point separately ##
Cary Clark8032b982017-07-28 11:04:54 -04002395##
2396
2397#Const kLines_PointMode 1
Cary Clark682c58d2018-05-16 07:07:07 -04002398#Line # draw each pair of points as a line segment ##
Cary Clark8032b982017-07-28 11:04:54 -04002399##
2400
2401#Const kPolygon_PointMode 2
Cary Clark682c58d2018-05-16 07:07:07 -04002402#Line # draw the array of points as a open polygon ##
Cary Clark8032b982017-07-28 11:04:54 -04002403##
2404
2405#Example
Herb Derbyefe39bc2018-05-01 17:06:20 -04002406 #Description
Cary Clark8032b982017-07-28 11:04:54 -04002407 The upper left corner shows three squares when drawn as points.
2408 The upper right corner shows one line; when drawn as lines, two points are required per line.
2409 The lower right corner shows two lines; when draw as polygon, no miter is drawn at the corner.
2410 The lower left corner shows two lines with a miter when path contains polygon.
2411 ##
Cary Clarkbad5ad72017-08-03 17:14:08 -04002412void draw(SkCanvas* canvas) {
2413 SkPaint paint;
2414 paint.setStyle(SkPaint::kStroke_Style);
2415 paint.setStrokeWidth(10);
2416 SkPoint points[] = {{64, 32}, {96, 96}, {32, 96}};
2417 canvas->drawPoints(SkCanvas::kPoints_PointMode, 3, points, paint);
2418 canvas->translate(128, 0);
2419 canvas->drawPoints(SkCanvas::kLines_PointMode, 3, points, paint);
2420 canvas->translate(0, 128);
2421 canvas->drawPoints(SkCanvas::kPolygon_PointMode, 3, points, paint);
2422 SkPath path;
2423 path.addPoly(points, 3, false);
2424 canvas->translate(-128, 0);
2425 canvas->drawPath(path, paint);
Cary Clark8032b982017-07-28 11:04:54 -04002426}
2427##
2428
Cary Clark2ade9972017-11-02 17:49:34 -04002429#SeeAlso drawLine drawPoint drawPath
Cary Clark8032b982017-07-28 11:04:54 -04002430
2431##
2432
2433# ------------------------------------------------------------------------------
2434
2435#Method void drawPoints(PointMode mode, size_t count, const SkPoint pts[], const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05002436#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002437#Line # draws array as points, lines, polygon ##
Cary Clark09d80c02018-10-31 12:14:03 -04002438#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002439
2440#Example
2441#Height 200
Herb Derbyefe39bc2018-05-01 17:06:20 -04002442 #Description
Cary Clark8032b982017-07-28 11:04:54 -04002443 #List
2444 # The first column draws points. ##
2445 # The second column draws points as lines. ##
2446 # The third column draws points as a polygon. ##
2447 # The fourth column draws points as a polygonal path. ##
2448 # The first row uses a round cap and round join. ##
2449 # The second row uses a square cap and a miter join. ##
2450 # The third row uses a butt cap and a bevel join. ##
2451 ##
2452 The transparent color makes multiple line draws visible;
2453 the path is drawn all at once.
2454 ##
Cary Clarkbad5ad72017-08-03 17:14:08 -04002455void draw(SkCanvas* canvas) {
2456 SkPaint paint;
2457 paint.setAntiAlias(true);
2458 paint.setStyle(SkPaint::kStroke_Style);
2459 paint.setStrokeWidth(10);
2460 paint.setColor(0x80349a45);
2461 const SkPoint points[] = {{32, 16}, {48, 48}, {16, 32}};
Herb Derbyefe39bc2018-05-01 17:06:20 -04002462 const SkPaint::Join join[] = { SkPaint::kRound_Join,
Cary Clarkbad5ad72017-08-03 17:14:08 -04002463 SkPaint::kMiter_Join,
2464 SkPaint::kBevel_Join };
2465 int joinIndex = 0;
2466 SkPath path;
2467 path.addPoly(points, 3, false);
2468 for (const auto cap : { SkPaint::kRound_Cap, SkPaint::kSquare_Cap, SkPaint::kButt_Cap } ) {
2469 paint.setStrokeCap(cap);
2470 paint.setStrokeJoin(join[joinIndex++]);
2471 for (const auto mode : { SkCanvas::kPoints_PointMode,
2472 SkCanvas::kLines_PointMode,
2473 SkCanvas::kPolygon_PointMode } ) {
2474 canvas->drawPoints(mode, 3, points, paint);
2475 canvas->translate(64, 0);
2476 }
2477 canvas->drawPath(path, paint);
2478 canvas->translate(-192, 64);
2479 }
Cary Clark8032b982017-07-28 11:04:54 -04002480}
2481##
2482
Cary Clark2ade9972017-11-02 17:49:34 -04002483#SeeAlso drawLine drawPoint drawPath
Cary Clark8032b982017-07-28 11:04:54 -04002484
2485##
2486
2487# ------------------------------------------------------------------------------
2488
2489#Method void drawPoint(SkScalar x, SkScalar y, const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05002490#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002491#Line # draws point at (x, y) position ##
Cary Clark09d80c02018-10-31 12:14:03 -04002492#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002493
2494#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002495void draw(SkCanvas* canvas) {
2496 SkPaint paint;
2497 paint.setAntiAlias(true);
2498 paint.setColor(0x80349a45);
2499 paint.setStyle(SkPaint::kStroke_Style);
2500 paint.setStrokeWidth(100);
2501 paint.setStrokeCap(SkPaint::kRound_Cap);
2502 canvas->scale(1, 1.2f);
2503 canvas->drawPoint(64, 96, paint);
2504 canvas->scale(.6f, .8f);
2505 paint.setColor(SK_ColorWHITE);
2506 canvas->drawPoint(106, 120, paint);
Cary Clark8032b982017-07-28 11:04:54 -04002507}
2508##
2509
Cary Clark2ade9972017-11-02 17:49:34 -04002510#SeeAlso drawPoints drawCircle drawRect drawLine drawPath
Cary Clark8032b982017-07-28 11:04:54 -04002511
2512##
2513
Cary Clarkbad5ad72017-08-03 17:14:08 -04002514#Method void drawPoint(SkPoint p, const SkPaint& paint)
Cary Clark09d80c02018-10-31 12:14:03 -04002515#Populate
Cary Clarkbad5ad72017-08-03 17:14:08 -04002516
2517#Example
2518void draw(SkCanvas* canvas) {
2519 SkPaint paint;
2520 paint.setAntiAlias(true);
2521 paint.setColor(0x80349a45);
2522 paint.setStyle(SkPaint::kStroke_Style);
2523 paint.setStrokeWidth(100);
2524 paint.setStrokeCap(SkPaint::kSquare_Cap);
2525 canvas->scale(1, 1.2f);
2526 canvas->drawPoint({64, 96}, paint);
2527 canvas->scale(.6f, .8f);
2528 paint.setColor(SK_ColorWHITE);
2529 canvas->drawPoint(106, 120, paint);
2530}
2531##
2532
Cary Clark2ade9972017-11-02 17:49:34 -04002533#SeeAlso drawPoints drawCircle drawRect drawLine drawPath
Cary Clarkbad5ad72017-08-03 17:14:08 -04002534
2535##
2536
Cary Clark8032b982017-07-28 11:04:54 -04002537# ------------------------------------------------------------------------------
2538
2539#Method void drawLine(SkScalar x0, SkScalar y0, SkScalar x1, SkScalar y1, const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05002540#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002541#Line # draws line segment between two points ##
Cary Clark09d80c02018-10-31 12:14:03 -04002542#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002543
2544#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002545 SkPaint paint;
2546 paint.setAntiAlias(true);
2547 paint.setColor(0xFF9a67be);
2548 paint.setStrokeWidth(20);
2549 canvas->skew(1, 0);
2550 canvas->drawLine(32, 96, 32, 160, paint);
2551 canvas->skew(-2, 0);
2552 canvas->drawLine(288, 96, 288, 160, paint);
2553##
2554
Cary Clark2ade9972017-11-02 17:49:34 -04002555#SeeAlso drawPoint drawCircle drawRect drawPath
Cary Clarkbad5ad72017-08-03 17:14:08 -04002556
2557##
2558
2559#Method void drawLine(SkPoint p0, SkPoint p1, const SkPaint& paint)
Cary Clark09d80c02018-10-31 12:14:03 -04002560#Populate
Cary Clarkbad5ad72017-08-03 17:14:08 -04002561
2562#Example
2563 SkPaint paint;
2564 paint.setAntiAlias(true);
2565 paint.setColor(0xFF9a67be);
2566 paint.setStrokeWidth(20);
2567 canvas->skew(1, 0);
2568 canvas->drawLine({32, 96}, {32, 160}, paint);
2569 canvas->skew(-2, 0);
2570 canvas->drawLine({288, 96}, {288, 160}, paint);
Cary Clark8032b982017-07-28 11:04:54 -04002571##
2572
Cary Clark2ade9972017-11-02 17:49:34 -04002573#SeeAlso drawPoint drawCircle drawRect drawPath
Cary Clark8032b982017-07-28 11:04:54 -04002574
2575##
2576
2577# ------------------------------------------------------------------------------
2578
2579#Method void drawRect(const SkRect& rect, const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05002580#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002581#Line # draws Rect using Clip, Matrix, and Paint ##
Cary Clark09d80c02018-10-31 12:14:03 -04002582#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002583
2584#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002585void draw(SkCanvas* canvas) {
2586 SkPoint rectPts[] = { {64, 48}, {192, 160} };
2587 SkPaint paint;
2588 paint.setAntiAlias(true);
2589 paint.setStyle(SkPaint::kStroke_Style);
2590 paint.setStrokeWidth(20);
2591 paint.setStrokeJoin(SkPaint::kRound_Join);
2592 SkMatrix rotator;
2593 rotator.setRotate(30, 128, 128);
2594 for (auto color : { SK_ColorRED, SK_ColorBLUE, SK_ColorYELLOW, SK_ColorMAGENTA } ) {
2595 paint.setColor(color);
2596 SkRect rect;
2597 rect.set(rectPts[0], rectPts[1]);
2598 canvas->drawRect(rect, paint);
2599 rotator.mapPoints(rectPts, 2);
2600 }
Cary Clark8032b982017-07-28 11:04:54 -04002601}
2602##
2603
Herb Derbyefe39bc2018-05-01 17:06:20 -04002604#SeeAlso drawIRect drawRRect drawRoundRect drawRegion drawPath drawLine
Cary Clark8032b982017-07-28 11:04:54 -04002605
2606##
2607
2608# ------------------------------------------------------------------------------
2609
Herb Derbyefe39bc2018-05-01 17:06:20 -04002610#Method void drawIRect(const SkIRect& rect, const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05002611#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002612#Line # draws IRect using Clip, Matrix, and Paint ##
Cary Clark09d80c02018-10-31 12:14:03 -04002613#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002614
2615#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002616 SkIRect rect = { 64, 48, 192, 160 };
2617 SkPaint paint;
2618 paint.setAntiAlias(true);
2619 paint.setStyle(SkPaint::kStroke_Style);
2620 paint.setStrokeWidth(20);
2621 paint.setStrokeJoin(SkPaint::kRound_Join);
2622 for (auto color : { SK_ColorRED, SK_ColorBLUE, SK_ColorYELLOW, SK_ColorMAGENTA } ) {
2623 paint.setColor(color);
2624 canvas->drawIRect(rect, paint);
2625 canvas->rotate(30, 128, 128);
2626 }
Cary Clark8032b982017-07-28 11:04:54 -04002627##
2628
Cary Clark2ade9972017-11-02 17:49:34 -04002629#SeeAlso drawRect drawRRect drawRoundRect drawRegion drawPath drawLine
Cary Clark8032b982017-07-28 11:04:54 -04002630
2631##
2632
2633# ------------------------------------------------------------------------------
2634
2635#Method void drawRegion(const SkRegion& region, const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05002636#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002637#Line # draws Region using Clip, Matrix, and Paint ##
Cary Clark09d80c02018-10-31 12:14:03 -04002638#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002639
2640#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002641void draw(SkCanvas* canvas) {
2642 SkRegion region;
2643 region.op( 10, 10, 50, 50, SkRegion::kUnion_Op);
2644 region.op( 10, 50, 90, 90, SkRegion::kUnion_Op);
2645 SkPaint paint;
2646 paint.setAntiAlias(true);
2647 paint.setStyle(SkPaint::kStroke_Style);
2648 paint.setStrokeWidth(20);
2649 paint.setStrokeJoin(SkPaint::kRound_Join);
2650 canvas->drawRegion(region, paint);
Cary Clark8032b982017-07-28 11:04:54 -04002651}
2652##
2653
Cary Clark2ade9972017-11-02 17:49:34 -04002654#SeeAlso drawRect drawIRect drawPath
Cary Clark8032b982017-07-28 11:04:54 -04002655
2656##
2657
2658# ------------------------------------------------------------------------------
2659
2660#Method void drawOval(const SkRect& oval, const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05002661#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002662#Line # draws Oval using Clip, Matrix, and Paint ##
Cary Clark09d80c02018-10-31 12:14:03 -04002663#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002664
2665#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002666void draw(SkCanvas* canvas) {
2667 canvas->clear(0xFF3f5f9f);
2668 SkColor kColor1 = SkColorSetARGB(0xff, 0xff, 0x7f, 0);
2669 SkColor g1Colors[] = { kColor1, SkColorSetA(kColor1, 0x20) };
2670 SkPoint g1Points[] = { { 0, 0 }, { 0, 100 } };
2671 SkScalar pos[] = { 0.2f, 1.0f };
2672 SkRect bounds = SkRect::MakeWH(80, 70);
2673 SkPaint paint;
2674 paint.setAntiAlias(true);
2675 paint.setShader(SkGradientShader::MakeLinear(g1Points, g1Colors, pos, SK_ARRAY_COUNT(g1Colors),
2676 SkShader::kClamp_TileMode));
2677 canvas->drawOval(bounds , paint);
Cary Clark8032b982017-07-28 11:04:54 -04002678}
2679##
2680
Cary Clark2ade9972017-11-02 17:49:34 -04002681#SeeAlso drawCircle drawPoint drawPath drawRRect drawRoundRect
Cary Clark8032b982017-07-28 11:04:54 -04002682
2683##
2684
2685# ------------------------------------------------------------------------------
2686
2687#Method void drawRRect(const SkRRect& rrect, const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05002688#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002689#Line # draws Round_Rect using Clip, Matrix, and Paint ##
Cary Clark09d80c02018-10-31 12:14:03 -04002690#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002691
2692#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002693void draw(SkCanvas* canvas) {
2694 SkPaint paint;
2695 paint.setAntiAlias(true);
2696 SkRect outer = {30, 40, 210, 220};
2697 SkRect radii = {30, 50, 70, 90 };
2698 SkRRect rRect;
2699 rRect.setNinePatch(outer, radii.fLeft, radii.fTop, radii.fRight, radii.fBottom);
2700 canvas->drawRRect(rRect, paint);
2701 paint.setColor(SK_ColorWHITE);
2702 canvas->drawLine(outer.fLeft + radii.fLeft, outer.fTop,
2703 outer.fLeft + radii.fLeft, outer.fBottom, paint);
Herb Derbyefe39bc2018-05-01 17:06:20 -04002704 canvas->drawLine(outer.fRight - radii.fRight, outer.fTop,
Cary Clarkbad5ad72017-08-03 17:14:08 -04002705 outer.fRight - radii.fRight, outer.fBottom, paint);
Herb Derbyefe39bc2018-05-01 17:06:20 -04002706 canvas->drawLine(outer.fLeft, outer.fTop + radii.fTop,
Cary Clarkbad5ad72017-08-03 17:14:08 -04002707 outer.fRight, outer.fTop + radii.fTop, paint);
Herb Derbyefe39bc2018-05-01 17:06:20 -04002708 canvas->drawLine(outer.fLeft, outer.fBottom - radii.fBottom,
Cary Clarkbad5ad72017-08-03 17:14:08 -04002709 outer.fRight, outer.fBottom - radii.fBottom, paint);
2710}
Cary Clark8032b982017-07-28 11:04:54 -04002711##
2712
Cary Clark2ade9972017-11-02 17:49:34 -04002713#SeeAlso drawRect drawRoundRect drawDRRect drawCircle drawOval drawPath
Cary Clark8032b982017-07-28 11:04:54 -04002714
2715##
2716
2717# ------------------------------------------------------------------------------
2718
2719#Method void drawDRRect(const SkRRect& outer, const SkRRect& inner, const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05002720#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002721#Line # draws double Round_Rect stroked or filled ##
Cary Clark09d80c02018-10-31 12:14:03 -04002722#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002723
2724#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002725void draw(SkCanvas* canvas) {
2726 SkRRect outer = SkRRect::MakeRect({20, 40, 210, 200});
2727 SkRRect inner = SkRRect::MakeOval({60, 70, 170, 160});
2728 SkPaint paint;
2729 canvas->drawDRRect(outer, inner, paint);
Cary Clark8032b982017-07-28 11:04:54 -04002730}
2731##
2732
2733#Example
2734#Description
2735 Outer Rect has no corner radii, but stroke join is rounded.
2736 Inner Round_Rect has corner radii; outset stroke increases radii of corners.
2737 Stroke join does not affect inner Round_Rect since it has no sharp corners.
2738##
Cary Clarkbad5ad72017-08-03 17:14:08 -04002739void draw(SkCanvas* canvas) {
2740 SkRRect outer = SkRRect::MakeRect({20, 40, 210, 200});
2741 SkRRect inner = SkRRect::MakeRectXY({60, 70, 170, 160}, 10, 10);
2742 SkPaint paint;
2743 paint.setAntiAlias(true);
2744 paint.setStyle(SkPaint::kStroke_Style);
2745 paint.setStrokeWidth(20);
2746 paint.setStrokeJoin(SkPaint::kRound_Join);
2747 canvas->drawDRRect(outer, inner, paint);
2748 paint.setStrokeWidth(1);
2749 paint.setColor(SK_ColorWHITE);
2750 canvas->drawDRRect(outer, inner, paint);
Cary Clark8032b982017-07-28 11:04:54 -04002751}
2752##
2753
Cary Clark2ade9972017-11-02 17:49:34 -04002754#SeeAlso drawRect drawRoundRect drawRRect drawCircle drawOval drawPath
Cary Clark8032b982017-07-28 11:04:54 -04002755
2756##
2757
2758# ------------------------------------------------------------------------------
2759
2760#Method void drawCircle(SkScalar cx, SkScalar cy, SkScalar radius, const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05002761#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002762#Line # draws Circle using Clip, Matrix, and Paint ##
Cary Clark09d80c02018-10-31 12:14:03 -04002763#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002764
2765#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002766 void draw(SkCanvas* canvas) {
2767 SkPaint paint;
2768 paint.setAntiAlias(true);
2769 canvas->drawCircle(128, 128, 90, paint);
2770 paint.setColor(SK_ColorWHITE);
2771 canvas->drawCircle(86, 86, 20, paint);
2772 canvas->drawCircle(160, 76, 20, paint);
2773 canvas->drawCircle(140, 150, 35, paint);
2774 }
2775##
2776
Cary Clark2ade9972017-11-02 17:49:34 -04002777#SeeAlso drawOval drawRRect drawRoundRect drawPath drawArc drawPoint drawLine
Cary Clarkbad5ad72017-08-03 17:14:08 -04002778
2779##
2780
2781#Method void drawCircle(SkPoint center, SkScalar radius, const SkPaint& paint)
Cary Clark09d80c02018-10-31 12:14:03 -04002782#Populate
Cary Clarkbad5ad72017-08-03 17:14:08 -04002783
2784#Example
2785 void draw(SkCanvas* canvas) {
2786 SkPaint paint;
2787 paint.setAntiAlias(true);
2788 canvas->drawCircle(128, 128, 90, paint);
2789 paint.setColor(SK_ColorWHITE);
2790 canvas->drawCircle({86, 86}, 20, paint);
2791 canvas->drawCircle({160, 76}, 20, paint);
2792 canvas->drawCircle({140, 150}, 35, paint);
2793 }
Cary Clark8032b982017-07-28 11:04:54 -04002794##
2795
Cary Clark2ade9972017-11-02 17:49:34 -04002796#SeeAlso drawOval drawRRect drawRoundRect drawPath drawArc drawPoint drawLine
Cary Clark8032b982017-07-28 11:04:54 -04002797
2798##
2799
2800# ------------------------------------------------------------------------------
2801
2802#Method void drawArc(const SkRect& oval, SkScalar startAngle, SkScalar sweepAngle,
2803 bool useCenter, const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05002804#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002805#Line # draws Arc using Clip, Matrix, and Paint ##
Cary Clark09d80c02018-10-31 12:14:03 -04002806#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002807
2808#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04002809 void draw(SkCanvas* canvas) {
2810 SkPaint paint;
2811 paint.setAntiAlias(true);
2812 SkRect oval = { 4, 4, 60, 60};
2813 for (auto useCenter : { false, true } ) {
2814 for (auto style : { SkPaint::kFill_Style, SkPaint::kStroke_Style } ) {
2815 paint.setStyle(style);
2816 for (auto degrees : { 45, 90, 180, 360} ) {
2817 canvas->drawArc(oval, 0, degrees , useCenter, paint);
2818 canvas->translate(64, 0);
2819 }
2820 canvas->translate(-256, 64);
2821 }
2822 }
Cary Clark8032b982017-07-28 11:04:54 -04002823 }
2824##
2825
2826#Example
2827#Height 64
Cary Clarkbad5ad72017-08-03 17:14:08 -04002828 void draw(SkCanvas* canvas) {
2829 SkPaint paint;
2830 paint.setAntiAlias(true);
2831 paint.setStyle(SkPaint::kStroke_Style);
2832 paint.setStrokeWidth(4);
2833 SkRect oval = { 4, 4, 60, 60};
2834 float intervals[] = { 5, 5 };
2835 paint.setPathEffect(SkDashPathEffect::Make(intervals, 2, 2.5f));
2836 for (auto degrees : { 270, 360, 540, 720 } ) {
2837 canvas->drawArc(oval, 0, degrees, false, paint);
2838 canvas->translate(64, 0);
2839 }
Cary Clark8032b982017-07-28 11:04:54 -04002840 }
2841##
2842
Cary Clark2ade9972017-11-02 17:49:34 -04002843#SeeAlso SkPath::arcTo drawCircle drawOval drawPath
Cary Clark8032b982017-07-28 11:04:54 -04002844
2845##
2846
2847# ------------------------------------------------------------------------------
2848
2849#Method void drawRoundRect(const SkRect& rect, SkScalar rx, SkScalar ry, const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05002850#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002851#Line # draws Round_Rect using Clip, Matrix, and Paint ##
Cary Clark09d80c02018-10-31 12:14:03 -04002852#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002853
2854#Example
2855#Description
2856 Top row has a zero radius a generates a rectangle.
2857 Second row radii sum to less than sides.
2858 Third row radii sum equals sides.
2859 Fourth row radii sum exceeds sides; radii are scaled to fit.
2860##
Cary Clarkbad5ad72017-08-03 17:14:08 -04002861 void draw(SkCanvas* canvas) {
2862 SkVector radii[] = { {0, 20}, {10, 10}, {10, 20}, {10, 40} };
2863 SkPaint paint;
2864 paint.setStrokeWidth(15);
2865 paint.setStrokeJoin(SkPaint::kRound_Join);
2866 paint.setAntiAlias(true);
2867 for (auto style : { SkPaint::kStroke_Style, SkPaint::kFill_Style } ) {
2868 paint.setStyle(style );
2869 for (size_t i = 0; i < SK_ARRAY_COUNT(radii); ++i) {
2870 canvas->drawRoundRect({10, 10, 60, 40}, radii[i].fX, radii[i].fY, paint);
2871 canvas->translate(0, 60);
2872 }
2873 canvas->translate(80, -240);
2874 }
Cary Clark8032b982017-07-28 11:04:54 -04002875 }
2876##
2877
Cary Clark77b3f3a2018-11-07 14:59:03 -05002878#SeeAlso drawRRect drawRect drawDRRect drawPath drawCircle drawOval drawPoint
Cary Clark8032b982017-07-28 11:04:54 -04002879
2880##
2881
2882# ------------------------------------------------------------------------------
2883
2884#Method void drawPath(const SkPath& path, const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05002885#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002886#Line # draws Path using Clip, Matrix, and Paint ##
Cary Clark09d80c02018-10-31 12:14:03 -04002887#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002888
2889#Example
2890#Description
2891 Top rows draw stroked path with combinations of joins and caps. The open contour
2892 is affected by caps; the closed contour is affected by joins.
Herb Derbyefe39bc2018-05-01 17:06:20 -04002893 Bottom row draws fill the same for open and closed contour.
Cary Clark8032b982017-07-28 11:04:54 -04002894 First bottom column shows winding fills overlap.
2895 Second bottom column shows even odd fills exclude overlap.
2896 Third bottom column shows inverse winding fills area outside both contours.
2897##
Cary Clarkbad5ad72017-08-03 17:14:08 -04002898void draw(SkCanvas* canvas) {
2899 SkPath path;
2900 path.moveTo(20, 20);
2901 path.quadTo(60, 20, 60, 60);
2902 path.close();
2903 path.moveTo(60, 20);
2904 path.quadTo(60, 60, 20, 60);
2905 SkPaint paint;
2906 paint.setStrokeWidth(10);
2907 paint.setAntiAlias(true);
2908 paint.setStyle(SkPaint::kStroke_Style);
2909 for (auto join: { SkPaint::kBevel_Join, SkPaint::kRound_Join, SkPaint::kMiter_Join } ) {
2910 paint.setStrokeJoin(join);
2911 for (auto cap: { SkPaint::kButt_Cap, SkPaint::kSquare_Cap, SkPaint::kRound_Cap } ) {
2912 paint.setStrokeCap(cap);
2913 canvas->drawPath(path, paint);
2914 canvas->translate(80, 0);
2915 }
2916 canvas->translate(-240, 60);
2917 }
2918 paint.setStyle(SkPaint::kFill_Style);
Herb Derbyefe39bc2018-05-01 17:06:20 -04002919 for (auto fill : { SkPath::kWinding_FillType,
2920 SkPath::kEvenOdd_FillType,
Cary Clarkbad5ad72017-08-03 17:14:08 -04002921 SkPath::kInverseWinding_FillType } ) {
2922 path.setFillType(fill);
2923 canvas->save();
2924 canvas->clipRect({0, 10, 80, 70});
2925 canvas->drawPath(path, paint);
2926 canvas->restore();
2927 canvas->translate(80, 0);
2928 }
Cary Clark8032b982017-07-28 11:04:54 -04002929}
2930##
2931
Cary Clark2ade9972017-11-02 17:49:34 -04002932#SeeAlso SkPath drawLine drawArc drawRect drawPoints
Cary Clark8032b982017-07-28 11:04:54 -04002933
2934##
2935
2936# ------------------------------------------------------------------------------
Cary Clark08895c42018-02-01 09:37:32 -05002937#Subtopic Draw_Image
2938#Line # draws Image to Canvas ##
Cary Clark8032b982017-07-28 11:04:54 -04002939
Cary Clarkbad5ad72017-08-03 17:14:08 -04002940drawImage, drawImageRect, and drawImageNine can be called with a bare pointer or
2941a smart pointer as a convenience. The pairs of calls are otherwise identical.
Cary Clark8032b982017-07-28 11:04:54 -04002942
Cary Clark73fa9722017-08-29 17:36:51 -04002943#Method void drawImage(const SkImage* image, SkScalar left, SkScalar top, const SkPaint* paint = nullptr)
Cary Clarkab2621d2018-01-30 10:08:57 -05002944#In Draw_Image
Cary Clark78de7512018-02-07 07:27:09 -05002945#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05002946#Line # draws Image at (x, y) position ##
Cary Clark09d80c02018-10-31 12:14:03 -04002947#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002948
2949#Example
2950#Height 64
2951#Image 4
Cary Clarkbad5ad72017-08-03 17:14:08 -04002952void draw(SkCanvas* canvas) {
2953 // sk_sp<SkImage> image;
2954 SkImage* imagePtr = image.get();
2955 canvas->drawImage(imagePtr, 0, 0);
2956 SkPaint paint;
2957 canvas->drawImage(imagePtr, 80, 0, &paint);
2958 paint.setAlpha(0x80);
2959 canvas->drawImage(imagePtr, 160, 0, &paint);
Cary Clark8032b982017-07-28 11:04:54 -04002960}
2961##
2962
Cary Clark2ade9972017-11-02 17:49:34 -04002963#SeeAlso drawBitmap drawImageLattice drawImageNine drawImageRect SkPaint::setImageFilter
Cary Clark8032b982017-07-28 11:04:54 -04002964
2965##
2966
2967# ------------------------------------------------------------------------------
2968
2969#Method void drawImage(const sk_sp<SkImage>& image, SkScalar left, SkScalar top,
Herb Derbyefe39bc2018-05-01 17:06:20 -04002970 const SkPaint* paint = nullptr)
Cary Clark09d80c02018-10-31 12:14:03 -04002971#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002972
2973#Example
2974#Height 64
2975#Image 4
Cary Clarkbad5ad72017-08-03 17:14:08 -04002976void draw(SkCanvas* canvas) {
2977 // sk_sp<SkImage> image;
2978 canvas->drawImage(image, 0, 0);
2979 SkPaint paint;
2980 canvas->drawImage(image, 80, 0, &paint);
2981 paint.setAlpha(0x80);
2982 canvas->drawImage(image, 160, 0, &paint);
Cary Clark8032b982017-07-28 11:04:54 -04002983}
2984##
2985
Cary Clark2ade9972017-11-02 17:49:34 -04002986#SeeAlso drawBitmap drawImageLattice drawImageNine drawImageRect SkPaint::setImageFilter
Cary Clark8032b982017-07-28 11:04:54 -04002987
2988##
2989
2990# ------------------------------------------------------------------------------
2991
2992#Enum SrcRectConstraint
Cary Clark08895c42018-02-01 09:37:32 -05002993#Line # sets drawImageRect options ##
Cary Clark8032b982017-07-28 11:04:54 -04002994
2995#Code
Cary Clark61313f32018-10-08 14:57:48 -04002996#Populate
Cary Clark8032b982017-07-28 11:04:54 -04002997##
2998
Cary Clarkce101242017-09-01 15:51:02 -04002999SrcRectConstraint controls the behavior at the edge of source Rect,
3000provided to drawImageRect, trading off speed for precision.
Cary Clark8032b982017-07-28 11:04:54 -04003001
Cary Clarkce101242017-09-01 15:51:02 -04003002Image_Filter in Paint may sample multiple pixels in the image. Source Rect
Cary Clarkbad5ad72017-08-03 17:14:08 -04003003restricts the bounds of pixels that may be read. Image_Filter may slow down if
Herb Derbyefe39bc2018-05-01 17:06:20 -04003004it cannot read outside the bounds, when sampling near the edge of source Rect.
Cary Clark8032b982017-07-28 11:04:54 -04003005SrcRectConstraint specifies whether an Image_Filter is allowed to read pixels
Cary Clarkce101242017-09-01 15:51:02 -04003006outside source Rect.
Cary Clark8032b982017-07-28 11:04:54 -04003007
Cary Clark682c58d2018-05-16 07:07:07 -04003008#Const kStrict_SrcRectConstraint 0
3009#Line # sample only inside bounds; slower ##
Cary Clarkce101242017-09-01 15:51:02 -04003010 Requires Image_Filter to respect source Rect,
Cary Clark8032b982017-07-28 11:04:54 -04003011 sampling only inside of its bounds, possibly with a performance penalty.
3012##
3013
Cary Clark682c58d2018-05-16 07:07:07 -04003014#Const kFast_SrcRectConstraint 1
3015#Line # sample outside bounds; faster ##
Cary Clarkce101242017-09-01 15:51:02 -04003016 Permits Image_Filter to sample outside of source Rect
Cary Clark8032b982017-07-28 11:04:54 -04003017 by half the width of Image_Filter, permitting it to run faster but with
3018 error at the image edges.
3019##
3020
3021#Example
3022#Height 64
3023#Description
3024 redBorder contains a black and white checkerboard bordered by red.
3025 redBorder is drawn scaled by 16 on the left.
Cary Clarkce101242017-09-01 15:51:02 -04003026 The middle and right bitmaps are filtered checkerboards.
Cary Clark8032b982017-07-28 11:04:54 -04003027 Drawing the checkerboard with kStrict_SrcRectConstraint shows only a blur of black and white.
3028 Drawing the checkerboard with kFast_SrcRectConstraint allows red to bleed in the corners.
3029##
Cary Clarkbad5ad72017-08-03 17:14:08 -04003030void draw(SkCanvas* canvas) {
3031 SkBitmap redBorder;
3032 redBorder.allocPixels(SkImageInfo::MakeN32Premul(4, 4));
3033 SkCanvas checkRed(redBorder);
3034 checkRed.clear(SK_ColorRED);
3035 uint32_t checkers[][2] = { { SK_ColorBLACK, SK_ColorWHITE },
3036 { SK_ColorWHITE, SK_ColorBLACK } };
3037 checkRed.writePixels(
3038 SkImageInfo::MakeN32Premul(2, 2), (void*) checkers, sizeof(checkers[0]), 1, 1);
3039 canvas->scale(16, 16);
3040 canvas->drawBitmap(redBorder, 0, 0, nullptr);
3041 canvas->resetMatrix();
3042 sk_sp<SkImage> image = SkImage::MakeFromBitmap(redBorder);
3043 SkPaint lowPaint;
3044 lowPaint.setFilterQuality(kLow_SkFilterQuality);
3045 for (auto constraint : { SkCanvas::kStrict_SrcRectConstraint,
3046 SkCanvas::kFast_SrcRectConstraint } ) {
3047 canvas->translate(80, 0);
3048 canvas->drawImageRect(image.get(), SkRect::MakeLTRB(1, 1, 3, 3),
3049 SkRect::MakeLTRB(16, 16, 48, 48), &lowPaint, constraint);
3050 }
Cary Clark8032b982017-07-28 11:04:54 -04003051}
3052##
3053
Cary Clark2ade9972017-11-02 17:49:34 -04003054#SeeAlso drawImageRect drawImage SkPaint::setImageFilter
Cary Clark8032b982017-07-28 11:04:54 -04003055
3056##
3057
3058# ------------------------------------------------------------------------------
3059
3060#Method void drawImageRect(const SkImage* image, const SkRect& src, const SkRect& dst,
3061 const SkPaint* paint,
3062 SrcRectConstraint constraint = kStrict_SrcRectConstraint)
Cary Clarkab2621d2018-01-30 10:08:57 -05003063#In Draw_Image
Cary Clark78de7512018-02-07 07:27:09 -05003064#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05003065#Line # draws Image, source Rect to destination Rect ##
Cary Clark09d80c02018-10-31 12:14:03 -04003066#Populate
Cary Clark8032b982017-07-28 11:04:54 -04003067
3068#Example
3069#Height 64
3070#Description
3071 The left bitmap draws with Paint default kNone_SkFilterQuality, and stays within
Cary Clarkbc5697d2017-10-04 14:31:33 -04003072 its bounds; there is no bleeding with kFast_SrcRectConstraint.
Cary Clark8032b982017-07-28 11:04:54 -04003073 the middle and right bitmaps draw with kLow_SkFilterQuality; with
3074 kStrict_SrcRectConstraint, the filter remains within the checkerboard, and
3075 with kFast_SrcRectConstraint red bleeds on the edges.
3076##
Cary Clarkbad5ad72017-08-03 17:14:08 -04003077void draw(SkCanvas* canvas) {
Herb Derbyefe39bc2018-05-01 17:06:20 -04003078 uint32_t pixels[][4] = {
Cary Clarkbad5ad72017-08-03 17:14:08 -04003079 { 0xFFFF0000, 0xFFFF0000, 0xFFFF0000, 0xFFFF0000 },
3080 { 0xFFFF0000, 0xFF000000, 0xFFFFFFFF, 0xFFFF0000 },
3081 { 0xFFFF0000, 0xFFFFFFFF, 0xFF000000, 0xFFFF0000 },
3082 { 0xFFFF0000, 0xFFFF0000, 0xFFFF0000, 0xFFFF0000 } };
3083 SkBitmap redBorder;
Herb Derbyefe39bc2018-05-01 17:06:20 -04003084 redBorder.installPixels(SkImageInfo::MakeN32Premul(4, 4),
Cary Clarkbad5ad72017-08-03 17:14:08 -04003085 (void*) pixels, sizeof(pixels[0]));
3086 sk_sp<SkImage> image = SkImage::MakeFromBitmap(redBorder);
3087 SkPaint lowPaint;
3088 for (auto constraint : {
3089 SkCanvas::kFast_SrcRectConstraint,
3090 SkCanvas::kStrict_SrcRectConstraint,
3091 SkCanvas::kFast_SrcRectConstraint } ) {
3092 canvas->drawImageRect(image.get(), SkRect::MakeLTRB(1, 1, 3, 3),
3093 SkRect::MakeLTRB(16, 16, 48, 48), &lowPaint, constraint);
3094 lowPaint.setFilterQuality(kLow_SkFilterQuality);
3095 canvas->translate(80, 0);
3096 }
3097}
Cary Clark8032b982017-07-28 11:04:54 -04003098##
3099
Cary Clark2ade9972017-11-02 17:49:34 -04003100#SeeAlso SrcRectConstraint drawImage drawImageLattice drawImageNine
Cary Clark8032b982017-07-28 11:04:54 -04003101
3102##
3103
3104# ------------------------------------------------------------------------------
3105
3106#Method void drawImageRect(const SkImage* image, const SkIRect& isrc, const SkRect& dst,
3107 const SkPaint* paint, SrcRectConstraint constraint = kStrict_SrcRectConstraint)
Cary Clarkab2621d2018-01-30 10:08:57 -05003108#In Draw_Image
Cary Clark78de7512018-02-07 07:27:09 -05003109#In Draw
Cary Clark09d80c02018-10-31 12:14:03 -04003110#Populate
Cary Clark8032b982017-07-28 11:04:54 -04003111
3112#Example
3113#Image 4
Cary Clarkbad5ad72017-08-03 17:14:08 -04003114void draw(SkCanvas* canvas) {
3115 // sk_sp<SkImage> image;
3116 for (auto i : { 1, 2, 4, 8 } ) {
Herb Derbyefe39bc2018-05-01 17:06:20 -04003117 canvas->drawImageRect(image.get(), SkIRect::MakeLTRB(0, 0, 100, 100),
Cary Clarkbad5ad72017-08-03 17:14:08 -04003118 SkRect::MakeXYWH(i * 20, i * 20, i * 20, i * 20), nullptr);
3119 }
Cary Clark8032b982017-07-28 11:04:54 -04003120}
3121##
3122
Cary Clark2ade9972017-11-02 17:49:34 -04003123#SeeAlso SrcRectConstraint drawImage drawImageLattice drawImageNine
Cary Clark8032b982017-07-28 11:04:54 -04003124
3125##
3126
3127# ------------------------------------------------------------------------------
3128
Cary Clark0ac0edd2018-10-29 11:28:43 -04003129#Method void drawImageRect(const SkImage* image, const SkRect& dst, const SkPaint* paint)
Cary Clarkab2621d2018-01-30 10:08:57 -05003130#In Draw_Image
Cary Clark78de7512018-02-07 07:27:09 -05003131#In Draw
Cary Clark09d80c02018-10-31 12:14:03 -04003132#Populate
Cary Clark8032b982017-07-28 11:04:54 -04003133
3134#Example
3135#Image 4
Cary Clarkbad5ad72017-08-03 17:14:08 -04003136void draw(SkCanvas* canvas) {
3137 // sk_sp<SkImage> image;
3138 for (auto i : { 20, 40, 80, 160 } ) {
3139 canvas->drawImageRect(image.get(), SkRect::MakeXYWH(i, i, i, i), nullptr);
3140 }
Cary Clark8032b982017-07-28 11:04:54 -04003141}
3142##
3143
Cary Clark2ade9972017-11-02 17:49:34 -04003144#SeeAlso SrcRectConstraint drawImage drawImageLattice drawImageNine
Cary Clark8032b982017-07-28 11:04:54 -04003145
3146##
3147
3148# ------------------------------------------------------------------------------
3149
3150#Method void drawImageRect(const sk_sp<SkImage>& image, const SkRect& src, const SkRect& dst,
3151 const SkPaint* paint,
Herb Derbyefe39bc2018-05-01 17:06:20 -04003152 SrcRectConstraint constraint = kStrict_SrcRectConstraint)
Cary Clarkab2621d2018-01-30 10:08:57 -05003153#In Draw_Image
Cary Clark78de7512018-02-07 07:27:09 -05003154#In Draw
Cary Clark09d80c02018-10-31 12:14:03 -04003155#Populate
Cary Clark8032b982017-07-28 11:04:54 -04003156
3157#Example
3158#Height 64
3159#Description
3160 Canvas scales and translates; transformation from src to dst also scales.
3161 The two matrices are concatenated to create the final transformation.
3162##
Cary Clarkbad5ad72017-08-03 17:14:08 -04003163void draw(SkCanvas* canvas) {
3164 uint32_t pixels[][2] = { { SK_ColorBLACK, SK_ColorWHITE },
3165 { SK_ColorWHITE, SK_ColorBLACK } };
3166 SkBitmap bitmap;
Herb Derbyefe39bc2018-05-01 17:06:20 -04003167 bitmap.installPixels(SkImageInfo::MakeN32Premul(2, 2),
Cary Clarkbad5ad72017-08-03 17:14:08 -04003168 (void*) pixels, sizeof(pixels[0]));
3169 sk_sp<SkImage> image = SkImage::MakeFromBitmap(bitmap);
3170 SkPaint paint;
3171 canvas->scale(4, 4);
3172 for (auto alpha : { 50, 100, 150, 255 } ) {
3173 paint.setAlpha(alpha);
3174 canvas->drawImageRect(image, SkRect::MakeWH(2, 2), SkRect::MakeWH(8, 8), &paint);
3175 canvas->translate(8, 0);
3176 }
3177}
Cary Clark8032b982017-07-28 11:04:54 -04003178##
3179
Cary Clark2ade9972017-11-02 17:49:34 -04003180#SeeAlso SrcRectConstraint drawImage drawImageLattice drawImageNine
Cary Clark8032b982017-07-28 11:04:54 -04003181
3182##
3183
3184# ------------------------------------------------------------------------------
3185
3186#Method void drawImageRect(const sk_sp<SkImage>& image, const SkIRect& isrc, const SkRect& dst,
Herb Derbyefe39bc2018-05-01 17:06:20 -04003187 const SkPaint* paint, SrcRectConstraint constraint = kStrict_SrcRectConstraint)
Cary Clarkab2621d2018-01-30 10:08:57 -05003188#In Draw_Image
Cary Clark78de7512018-02-07 07:27:09 -05003189#In Draw
Cary Clark09d80c02018-10-31 12:14:03 -04003190#Populate
Cary Clark8032b982017-07-28 11:04:54 -04003191
3192#Example
3193#Height 64
Cary Clarkbad5ad72017-08-03 17:14:08 -04003194void draw(SkCanvas* canvas) {
3195 uint32_t pixels[][2] = { { 0x00000000, 0x55555555},
3196 { 0xAAAAAAAA, 0xFFFFFFFF} };
3197 SkBitmap bitmap;
Herb Derbyefe39bc2018-05-01 17:06:20 -04003198 bitmap.installPixels(SkImageInfo::MakeN32Premul(2, 2),
Cary Clarkbad5ad72017-08-03 17:14:08 -04003199 (void*) pixels, sizeof(pixels[0]));
3200 sk_sp<SkImage> image = SkImage::MakeFromBitmap(bitmap);
3201 SkPaint paint;
3202 canvas->scale(4, 4);
3203 for (auto color : { SK_ColorRED, SK_ColorBLUE, SK_ColorGREEN } ) {
3204 paint.setColorFilter(SkColorFilter::MakeModeFilter(color, SkBlendMode::kPlus));
3205 canvas->drawImageRect(image, SkIRect::MakeWH(2, 2), SkRect::MakeWH(8, 8), &paint);
3206 canvas->translate(8, 0);
3207 }
Cary Clark8032b982017-07-28 11:04:54 -04003208}
3209##
3210
Cary Clark2ade9972017-11-02 17:49:34 -04003211#SeeAlso SrcRectConstraint drawImage drawImageLattice drawImageNine
3212
Cary Clark8032b982017-07-28 11:04:54 -04003213##
3214
3215# ------------------------------------------------------------------------------
3216
Cary Clark0ac0edd2018-10-29 11:28:43 -04003217#Method void drawImageRect(const sk_sp<SkImage>& image, const SkRect& dst, const SkPaint* paint)
Cary Clarkab2621d2018-01-30 10:08:57 -05003218#In Draw_Image
Cary Clark78de7512018-02-07 07:27:09 -05003219#In Draw
Cary Clark09d80c02018-10-31 12:14:03 -04003220#Populate
Cary Clark8032b982017-07-28 11:04:54 -04003221
3222#Example
3223#Height 64
Cary Clarkbad5ad72017-08-03 17:14:08 -04003224void draw(SkCanvas* canvas) {
3225 uint32_t pixels[][2] = { { 0x00000000, 0x55550000},
3226 { 0xAAAA0000, 0xFFFF0000} };
3227 SkBitmap bitmap;
Herb Derbyefe39bc2018-05-01 17:06:20 -04003228 bitmap.installPixels(SkImageInfo::MakeN32Premul(2, 2),
Cary Clarkbad5ad72017-08-03 17:14:08 -04003229 (void*) pixels, sizeof(pixels[0]));
3230 sk_sp<SkImage> image = SkImage::MakeFromBitmap(bitmap);
3231 SkPaint paint;
3232 canvas->scale(4, 4);
3233 for (auto color : { SK_ColorRED, SK_ColorBLUE, SK_ColorGREEN } ) {
3234 paint.setColorFilter(SkColorFilter::MakeModeFilter(color, SkBlendMode::kPlus));
3235 canvas->drawImageRect(image, SkRect::MakeWH(8, 8), &paint);
3236 canvas->translate(8, 0);
3237 }
Cary Clark8032b982017-07-28 11:04:54 -04003238}
3239##
3240
Cary Clark2ade9972017-11-02 17:49:34 -04003241#SeeAlso SrcRectConstraint drawImage drawImageLattice drawImageNine
Cary Clark8032b982017-07-28 11:04:54 -04003242
3243##
3244
3245# ------------------------------------------------------------------------------
3246
3247#Method void drawImageNine(const SkImage* image, const SkIRect& center, const SkRect& dst,
3248 const SkPaint* paint = nullptr)
Cary Clarkab2621d2018-01-30 10:08:57 -05003249#In Draw_Image
Cary Clark78de7512018-02-07 07:27:09 -05003250#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05003251#Line # draws Nine_Patch Image ##
Cary Clark8032b982017-07-28 11:04:54 -04003252
Cary Clark80247e52018-07-11 16:18:41 -04003253Draws Image image stretched proportionally to fit into Rect dst.
Cary Clarkbad5ad72017-08-03 17:14:08 -04003254IRect center divides the image into nine sections: four sides, four corners, and
Cary Clarkce101242017-09-01 15:51:02 -04003255the center. Corners are unmodified or scaled down proportionately if their sides
Cary Clarkbad5ad72017-08-03 17:14:08 -04003256are larger than dst; center and four sides are scaled to fit remaining space, if any.
Cary Clark8032b982017-07-28 11:04:54 -04003257
Cary Clarkbad5ad72017-08-03 17:14:08 -04003258Additionally transform draw using Clip, Matrix, and optional Paint paint.
3259
Cary Clark682c58d2018-05-16 07:07:07 -04003260#paint_as_used_by_draw_lattice_or_draw_nine(image)#
Cary Clarkbad5ad72017-08-03 17:14:08 -04003261
3262If generated mask extends beyond image bounds, replicate image edge colors, just
3263as Shader made from SkImage::makeShader with SkShader::kClamp_TileMode set
Cary Clarkbc5697d2017-10-04 14:31:33 -04003264replicates the image edge color when it samples outside of its bounds.
Cary Clarkbad5ad72017-08-03 17:14:08 -04003265
3266#Param image Image containing pixels, dimensions, and format ##
3267#Param center IRect edge of image corners and sides ##
3268#Param dst destination Rect of image to draw to ##
3269#Param paint Paint containing Blend_Mode, Color_Filter, Image_Filter,
3270 and so on; or nullptr
3271##
Cary Clark8032b982017-07-28 11:04:54 -04003272
3273#Example
3274#Height 128
3275#Description
3276 The leftmost image is smaller than center; only corners are drawn, all scaled to fit.
Cary Clarkce101242017-09-01 15:51:02 -04003277 The second image equals the size of center; only corners are drawn without scaling.
3278 The remaining images are larger than center. All corners draw without scaling.
3279 The sides and center are scaled if needed to take up the remaining space.
Cary Clark8032b982017-07-28 11:04:54 -04003280##
Cary Clarkbad5ad72017-08-03 17:14:08 -04003281void draw(SkCanvas* canvas) {
3282 SkIRect center = { 20, 10, 50, 40 };
3283 SkBitmap bitmap;
3284 bitmap.allocPixels(SkImageInfo::MakeN32Premul(60, 60));
3285 SkCanvas bitCanvas(bitmap);
3286 SkPaint paint;
3287 SkColor gray = 0xFF000000;
3288 int left = 0;
3289 for (auto right: { center.fLeft, center.fRight, bitmap.width() } ) {
3290 int top = 0;
3291 for (auto bottom: { center.fTop, center.fBottom, bitmap.height() } ) {
3292 paint.setColor(gray);
3293 bitCanvas.drawIRect(SkIRect::MakeLTRB(left, top, right, bottom), paint);
3294 gray += 0x001f1f1f;
3295 top = bottom;
3296 }
Herb Derbyefe39bc2018-05-01 17:06:20 -04003297 left = right;
Cary Clarkbad5ad72017-08-03 17:14:08 -04003298 }
3299 sk_sp<SkImage> image = SkImage::MakeFromBitmap(bitmap);
3300 SkImage* imagePtr = image.get();
3301 for (auto dest: { 20, 30, 40, 60, 90 } ) {
3302 canvas->drawImageNine(imagePtr, center, SkRect::MakeWH(dest, dest), nullptr);
3303 canvas->translate(dest + 4, 0);
3304 }
Cary Clark8032b982017-07-28 11:04:54 -04003305}
3306##
3307
Cary Clark2ade9972017-11-02 17:49:34 -04003308#SeeAlso drawImage drawBitmapNine drawImageLattice drawImageRect
Cary Clark8032b982017-07-28 11:04:54 -04003309
3310##
3311
3312# ------------------------------------------------------------------------------
3313
3314#Method void drawImageNine(const sk_sp<SkImage>& image, const SkIRect& center, const SkRect& dst,
Herb Derbyefe39bc2018-05-01 17:06:20 -04003315 const SkPaint* paint = nullptr)
Cary Clarkab2621d2018-01-30 10:08:57 -05003316#In Draw_Image
Cary Clark78de7512018-02-07 07:27:09 -05003317#In Draw
Cary Clark80247e52018-07-11 16:18:41 -04003318Draws Image image stretched proportionally to fit into Rect dst.
Cary Clarkbad5ad72017-08-03 17:14:08 -04003319IRect center divides the image into nine sections: four sides, four corners, and
Cary Clarkce101242017-09-01 15:51:02 -04003320the center. Corners are not scaled, or scaled down proportionately if their sides
Cary Clarkbad5ad72017-08-03 17:14:08 -04003321are larger than dst; center and four sides are scaled to fit remaining space, if any.
Cary Clark8032b982017-07-28 11:04:54 -04003322
Cary Clarkbad5ad72017-08-03 17:14:08 -04003323Additionally transform draw using Clip, Matrix, and optional Paint paint.
3324
Cary Clark137b8742018-05-30 09:21:49 -04003325#paint_as_used_by_draw_lattice_or_draw_nine(image)#
Cary Clarkbad5ad72017-08-03 17:14:08 -04003326
3327If generated mask extends beyond image bounds, replicate image edge colors, just
3328as Shader made from SkImage::makeShader with SkShader::kClamp_TileMode set
Cary Clarkbc5697d2017-10-04 14:31:33 -04003329replicates the image edge color when it samples outside of its bounds.
Cary Clarkbad5ad72017-08-03 17:14:08 -04003330
3331#Param image Image containing pixels, dimensions, and format ##
3332#Param center IRect edge of image corners and sides ##
3333#Param dst destination Rect of image to draw to ##
3334#Param paint Paint containing Blend_Mode, Color_Filter, Image_Filter,
3335 and so on; or nullptr
3336##
Cary Clark8032b982017-07-28 11:04:54 -04003337
3338#Example
3339#Height 128
3340#Description
3341 The two leftmost images has four corners and sides to the left and right of center.
3342 The leftmost image scales the width of corners proportionately to fit.
Herb Derbyefe39bc2018-05-01 17:06:20 -04003343 The third and fourth image corners are not scaled; the sides and center are scaled to
Cary Clark8032b982017-07-28 11:04:54 -04003344 fill the remaining space.
3345 The rightmost image has four corners scaled vertically to fit, and uses sides above
3346 and below center to fill the remaining space.
3347##
Cary Clarkbad5ad72017-08-03 17:14:08 -04003348void draw(SkCanvas* canvas) {
3349 SkIRect center = { 20, 10, 50, 40 };
3350 SkBitmap bitmap;
3351 bitmap.allocPixels(SkImageInfo::MakeN32Premul(60, 60));
3352 SkCanvas bitCanvas(bitmap);
3353 SkPaint paint;
3354 SkColor gray = 0xFF000000;
3355 int left = 0;
3356 for (auto right: { center.fLeft, center.fRight, bitmap.width() } ) {
3357 int top = 0;
3358 for (auto bottom: { center.fTop, center.fBottom, bitmap.height() } ) {
3359 paint.setColor(gray);
3360 bitCanvas.drawIRect(SkIRect::MakeLTRB(left, top, right, bottom), paint);
3361 gray += 0x001f1f1f;
3362 top = bottom;
3363 }
Herb Derbyefe39bc2018-05-01 17:06:20 -04003364 left = right;
Cary Clarkbad5ad72017-08-03 17:14:08 -04003365 }
3366 sk_sp<SkImage> image = SkImage::MakeFromBitmap(bitmap);
3367 for (auto dest: { 20, 30, 40, 60, 90 } ) {
3368 canvas->drawImageNine(image, center, SkRect::MakeWH(dest, 110 - dest), nullptr);
3369 canvas->translate(dest + 4, 0);
3370 }
Cary Clark8032b982017-07-28 11:04:54 -04003371}
3372##
3373
Cary Clark2ade9972017-11-02 17:49:34 -04003374#SeeAlso drawImage drawBitmapNine drawImageLattice drawImageRect
Cary Clark8032b982017-07-28 11:04:54 -04003375
3376##
3377
3378# ------------------------------------------------------------------------------
3379
3380#Method void drawBitmap(const SkBitmap& bitmap, SkScalar left, SkScalar top,
Cary Clark73fa9722017-08-29 17:36:51 -04003381 const SkPaint* paint = nullptr)
Cary Clarkab2621d2018-01-30 10:08:57 -05003382#In Draw_Image
Cary Clark78de7512018-02-07 07:27:09 -05003383#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05003384#Line # draws Bitmap at (x, y) position ##
Cary Clark09d80c02018-10-31 12:14:03 -04003385#Populate
Cary Clark8032b982017-07-28 11:04:54 -04003386
3387#Example
3388#Height 64
Cary Clarkbad5ad72017-08-03 17:14:08 -04003389void draw(SkCanvas* canvas) {
3390 uint8_t pixels[][8] = { { 0x00, 0x00, 0x00, 0xFF, 0xFF, 0x00, 0x00, 0x00},
3391 { 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00},
3392 { 0x00, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00},
3393 { 0xFF, 0xFF, 0x00, 0xFF, 0xFF, 0x00, 0xFF, 0xFF},
3394 { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF},
3395 { 0x00, 0x00, 0xFF, 0x00, 0x00, 0xFF, 0x00, 0x00},
3396 { 0x00, 0xFF, 0x00, 0xFF, 0xFF, 0x00, 0xFF, 0x00},
3397 { 0xFF, 0x00, 0xFF, 0x00, 0x00, 0xFF, 0x00, 0xFF} };
3398 SkBitmap bitmap;
Herb Derbyefe39bc2018-05-01 17:06:20 -04003399 bitmap.installPixels(SkImageInfo::MakeA8(8, 8),
Cary Clarkbad5ad72017-08-03 17:14:08 -04003400 (void*) pixels, sizeof(pixels[0]));
3401 SkPaint paint;
3402 canvas->scale(4, 4);
3403 for (auto color : { SK_ColorRED, SK_ColorBLUE, 0xFF007F00} ) {
3404 paint.setColor(color);
3405 canvas->drawBitmap(bitmap, 0, 0, &paint);
3406 canvas->translate(12, 0);
3407 }
Cary Clark8032b982017-07-28 11:04:54 -04003408}
3409##
3410
Cary Clark2ade9972017-11-02 17:49:34 -04003411#SeeAlso drawImage drawBitmapLattice drawBitmapNine drawBitmapRect SkBitmap::readPixels SkBitmap::writePixels
Cary Clark8032b982017-07-28 11:04:54 -04003412
3413##
3414
3415# ------------------------------------------------------------------------------
3416
3417#Method void drawBitmapRect(const SkBitmap& bitmap, const SkRect& src, const SkRect& dst,
3418 const SkPaint* paint, SrcRectConstraint constraint = kStrict_SrcRectConstraint)
Cary Clarkab2621d2018-01-30 10:08:57 -05003419#In Draw_Image
Cary Clark78de7512018-02-07 07:27:09 -05003420#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05003421#Line # draws Bitmap, source Rect to destination Rect ##
Cary Clark09d80c02018-10-31 12:14:03 -04003422#Populate
Cary Clark8032b982017-07-28 11:04:54 -04003423
3424#Example
3425#Height 64
Cary Clarkbad5ad72017-08-03 17:14:08 -04003426void draw(SkCanvas* canvas) {
3427 uint8_t pixels[][8] = { { 0x00, 0x00, 0x00, 0xFF, 0xFF, 0x00, 0x00, 0x00},
3428 { 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00},
3429 { 0x00, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00},
3430 { 0xFF, 0xFF, 0x00, 0xFF, 0xFF, 0x00, 0xFF, 0xFF},
3431 { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF},
3432 { 0x00, 0x00, 0xFF, 0x00, 0x00, 0xFF, 0x00, 0x00},
3433 { 0x00, 0xFF, 0x00, 0x00, 0x00, 0x00, 0xFF, 0x00},
3434 { 0x00, 0x00, 0xFF, 0x00, 0x00, 0xFF, 0x00, 0x00} };
3435 SkBitmap bitmap;
Herb Derbyefe39bc2018-05-01 17:06:20 -04003436 bitmap.installPixels(SkImageInfo::MakeA8(8, 8),
Cary Clarkbad5ad72017-08-03 17:14:08 -04003437 (void*) pixels, sizeof(pixels[0]));
3438 SkPaint paint;
Cary Clark681287e2018-03-16 11:34:15 -04003439 paint.setMaskFilter(SkMaskFilter::MakeBlur(kSolid_SkBlurStyle, 6));
Cary Clarkbad5ad72017-08-03 17:14:08 -04003440 for (auto color : { SK_ColorRED, SK_ColorBLUE, 0xFF007F00} ) {
3441 paint.setColor(color);
3442 canvas->drawBitmapRect(bitmap, SkRect::MakeWH(8, 8), SkRect::MakeWH(32, 32), &paint);
3443 canvas->translate(48, 0);
3444 }
Cary Clark8032b982017-07-28 11:04:54 -04003445}
3446##
3447
Cary Clark2ade9972017-11-02 17:49:34 -04003448#SeeAlso drawImageRect drawBitmap drawBitmapLattice drawBitmapNine
Cary Clark8032b982017-07-28 11:04:54 -04003449
3450##
3451
3452# ------------------------------------------------------------------------------
3453
3454#Method void drawBitmapRect(const SkBitmap& bitmap, const SkIRect& isrc, const SkRect& dst,
3455 const SkPaint* paint, SrcRectConstraint constraint = kStrict_SrcRectConstraint)
Cary Clarkab2621d2018-01-30 10:08:57 -05003456#In Draw_Image
Cary Clark78de7512018-02-07 07:27:09 -05003457#In Draw
Cary Clark09d80c02018-10-31 12:14:03 -04003458#Populate
Cary Clark8032b982017-07-28 11:04:54 -04003459
3460#Example
3461#Height 64
Cary Clarkbad5ad72017-08-03 17:14:08 -04003462void draw(SkCanvas* canvas) {
3463 uint8_t pixels[][8] = { { 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00},
3464 { 0x00, 0xFF, 0x00, 0x00, 0x00, 0x00, 0xFF, 0x00},
3465 { 0xFF, 0x00, 0xFF, 0x00, 0xFF, 0x00, 0x00, 0xFF},
3466 { 0xFF, 0x00, 0xFF, 0x00, 0xFF, 0x00, 0x00, 0xFF},
3467 { 0xFF, 0x00, 0x00, 0x00, 0x00, 0xFF, 0x00, 0xFF},
3468 { 0xFF, 0x00, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0xFF},
3469 { 0x00, 0xFF, 0x00, 0x00, 0x00, 0x00, 0xFF, 0x00},
3470 { 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00} };
3471 SkBitmap bitmap;
Herb Derbyefe39bc2018-05-01 17:06:20 -04003472 bitmap.installPixels(SkImageInfo::MakeA8(8, 8),
Cary Clarkbad5ad72017-08-03 17:14:08 -04003473 (void*) pixels, sizeof(pixels[0]));
3474 SkPaint paint;
3475 paint.setFilterQuality(kHigh_SkFilterQuality);
3476 for (auto color : { SK_ColorRED, SK_ColorBLUE, 0xFF007F00, 0xFF7f007f} ) {
3477 paint.setColor(color);
3478 canvas->drawBitmapRect(bitmap, SkIRect::MakeWH(8, 8), SkRect::MakeWH(32, 32), &paint);
3479 canvas->translate(48.25f, 0);
3480 }
Cary Clark8032b982017-07-28 11:04:54 -04003481}
3482##
3483
Cary Clark2ade9972017-11-02 17:49:34 -04003484#SeeAlso drawImageRect drawBitmap drawBitmapLattice drawBitmapNine
Cary Clark8032b982017-07-28 11:04:54 -04003485
3486##
3487
3488# ------------------------------------------------------------------------------
3489
3490#Method void drawBitmapRect(const SkBitmap& bitmap, const SkRect& dst, const SkPaint* paint,
3491 SrcRectConstraint constraint = kStrict_SrcRectConstraint)
Cary Clarkab2621d2018-01-30 10:08:57 -05003492#In Draw_Image
Cary Clark78de7512018-02-07 07:27:09 -05003493#In Draw
Cary Clark09d80c02018-10-31 12:14:03 -04003494#Populate
Cary Clark8032b982017-07-28 11:04:54 -04003495
3496#Example
3497#Height 64
Cary Clarkbad5ad72017-08-03 17:14:08 -04003498void draw(SkCanvas* canvas) {
3499 uint32_t pixels[][2] = { { 0x00000000, 0x55550000},
3500 { 0xAAAA0000, 0xFFFF0000} };
3501 SkBitmap bitmap;
Herb Derbyefe39bc2018-05-01 17:06:20 -04003502 bitmap.installPixels(SkImageInfo::MakeN32Premul(2, 2),
Cary Clarkbad5ad72017-08-03 17:14:08 -04003503 (void*) pixels, sizeof(pixels[0]));
3504 SkPaint paint;
3505 canvas->scale(4, 4);
3506 for (auto color : { SK_ColorRED, SK_ColorBLUE, SK_ColorGREEN } ) {
3507 paint.setColorFilter(SkColorFilter::MakeModeFilter(color, SkBlendMode::kPlus));
3508 canvas->drawBitmapRect(bitmap, SkRect::MakeWH(8, 8), &paint);
3509 canvas->translate(8, 0);
3510 }
Cary Clark8032b982017-07-28 11:04:54 -04003511}
3512##
3513
Cary Clark2ade9972017-11-02 17:49:34 -04003514#SeeAlso drawImageRect drawBitmap drawBitmapLattice drawBitmapNine
Cary Clark8032b982017-07-28 11:04:54 -04003515
3516##
3517
3518# ------------------------------------------------------------------------------
3519
Cary Clark682c58d2018-05-16 07:07:07 -04003520#PhraseDef paint_as_used_by_draw_lattice_or_draw_nine(bitmap_or_image)
3521If Paint paint is supplied, apply Color_Filter, Color_Alpha, Image_Filter,
3522Blend_Mode, and Draw_Looper. If #bitmap_or_image# is kAlpha_8_SkColorType, apply Shader.
3523If paint contains Mask_Filter, generate mask from #bitmap_or_image# bounds. If paint
3524Filter_Quality set to kNone_SkFilterQuality, disable pixel filtering. For all
3525other values of paint Filter_Quality, use kLow_SkFilterQuality to filter pixels.
Cary Clark137b8742018-05-30 09:21:49 -04003526Any SkMaskFilter on paint is ignored as is paint Anti_Aliasing state.
Cary Clark682c58d2018-05-16 07:07:07 -04003527##
3528
Cary Clark8032b982017-07-28 11:04:54 -04003529#Method void drawBitmapNine(const SkBitmap& bitmap, const SkIRect& center, const SkRect& dst,
Cary Clark73fa9722017-08-29 17:36:51 -04003530 const SkPaint* paint = nullptr)
Cary Clarkab2621d2018-01-30 10:08:57 -05003531#In Draw_Image
Cary Clark78de7512018-02-07 07:27:09 -05003532#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05003533#Line # draws Nine_Patch Bitmap ##
Cary Clark8032b982017-07-28 11:04:54 -04003534
Cary Clark80247e52018-07-11 16:18:41 -04003535Draws Bitmap bitmap stretched proportionally to fit into Rect dst.
Cary Clarkbad5ad72017-08-03 17:14:08 -04003536IRect center divides the bitmap into nine sections: four sides, four corners,
Cary Clarkce101242017-09-01 15:51:02 -04003537and the center. Corners are not scaled, or scaled down proportionately if their
Cary Clarkbad5ad72017-08-03 17:14:08 -04003538sides are larger than dst; center and four sides are scaled to fit remaining
3539space, if any.
Cary Clark8032b982017-07-28 11:04:54 -04003540
Cary Clarkbad5ad72017-08-03 17:14:08 -04003541Additionally transform draw using Clip, Matrix, and optional Paint paint.
3542
Cary Clark682c58d2018-05-16 07:07:07 -04003543#paint_as_used_by_draw_lattice_or_draw_nine(bitmap)#
Cary Clarkbad5ad72017-08-03 17:14:08 -04003544
3545If generated mask extends beyond bitmap bounds, replicate bitmap edge colors,
3546just as Shader made from SkShader::MakeBitmapShader with
Cary Clarkbc5697d2017-10-04 14:31:33 -04003547SkShader::kClamp_TileMode set replicates the bitmap edge color when it samples
Cary Clarkbad5ad72017-08-03 17:14:08 -04003548outside of its bounds.
3549
3550#Param bitmap Bitmap containing pixels, dimensions, and format ##
3551#Param center IRect edge of image corners and sides ##
3552#Param dst destination Rect of image to draw to ##
3553#Param paint Paint containing Blend_Mode, Color_Filter, Image_Filter,
3554 and so on; or nullptr
3555##
Cary Clark8032b982017-07-28 11:04:54 -04003556
3557#Example
3558#Height 128
3559#Description
3560 The two leftmost bitmap draws has four corners and sides to the left and right of center.
3561 The leftmost bitmap draw scales the width of corners proportionately to fit.
Herb Derbyefe39bc2018-05-01 17:06:20 -04003562 The third and fourth draw corners are not scaled; the sides and center are scaled to
Cary Clark8032b982017-07-28 11:04:54 -04003563 fill the remaining space.
3564 The rightmost bitmap draw has four corners scaled vertically to fit, and uses sides above
3565 and below center to fill the remaining space.
3566##
Cary Clarkbad5ad72017-08-03 17:14:08 -04003567void draw(SkCanvas* canvas) {
3568 SkIRect center = { 20, 10, 50, 40 };
3569 SkBitmap bitmap;
3570 bitmap.allocPixels(SkImageInfo::MakeN32Premul(60, 60));
3571 SkCanvas bitCanvas(bitmap);
3572 SkPaint paint;
3573 SkColor gray = 0xFF000000;
3574 int left = 0;
3575 for (auto right: { center.fLeft, center.fRight, bitmap.width() } ) {
3576 int top = 0;
3577 for (auto bottom: { center.fTop, center.fBottom, bitmap.height() } ) {
3578 paint.setColor(gray);
3579 bitCanvas.drawIRect(SkIRect::MakeLTRB(left, top, right, bottom), paint);
3580 gray += 0x001f1f1f;
3581 top = bottom;
3582 }
Herb Derbyefe39bc2018-05-01 17:06:20 -04003583 left = right;
Cary Clarkbad5ad72017-08-03 17:14:08 -04003584 }
3585 for (auto dest: { 20, 30, 40, 60, 90 } ) {
3586 canvas->drawBitmapNine(bitmap, center, SkRect::MakeWH(dest, 110 - dest), nullptr);
3587 canvas->translate(dest + 4, 0);
3588 }
Cary Clark8032b982017-07-28 11:04:54 -04003589}
3590##
3591
Cary Clark2ade9972017-11-02 17:49:34 -04003592#SeeAlso drawImageNine drawBitmap drawBitmapLattice drawBitmapRect
Cary Clark8032b982017-07-28 11:04:54 -04003593
3594##
3595
3596# ------------------------------------------------------------------------------
Cary Clark682c58d2018-05-16 07:07:07 -04003597#Subtopic Lattice
3598#Line # divides Bitmap or Image into a rectangular grid ##
3599
Cary Clark8032b982017-07-28 11:04:54 -04003600#Struct Lattice
Cary Clark08895c42018-02-01 09:37:32 -05003601#Line # divides Bitmap or Image into a rectangular grid ##
Cary Clark682c58d2018-05-16 07:07:07 -04003602
Cary Clark8032b982017-07-28 11:04:54 -04003603#Code
Cary Clark61313f32018-10-08 14:57:48 -04003604#Populate
Cary Clark8032b982017-07-28 11:04:54 -04003605##
3606
Cary Clark137b8742018-05-30 09:21:49 -04003607Lattice divides Bitmap or Image into a rectangular grid.
3608Grid entries on even columns and even rows are fixed; these entries are
3609always drawn at their original size if the destination is large enough.
3610If the destination side is too small to hold the fixed entries, all fixed
3611entries are proportionately scaled down to fit.
3612The grid entries not on even columns and rows are scaled to fit the
3613remaining space, if any.
3614
Cary Clark2f466242017-12-11 16:03:17 -05003615 #Enum RectType
Cary Clark682c58d2018-05-16 07:07:07 -04003616 #Line # optional setting per rectangular grid entry ##
Cary Clark8032b982017-07-28 11:04:54 -04003617 #Code
Cary Clark61313f32018-10-08 14:57:48 -04003618 #Populate
Cary Clark8032b982017-07-28 11:04:54 -04003619 ##
3620
Cary Clark2f466242017-12-11 16:03:17 -05003621 Optional setting per rectangular grid entry to make it transparent,
3622 or to fill the grid entry with a color.
Cary Clark8032b982017-07-28 11:04:54 -04003623
Cary Clark2f466242017-12-11 16:03:17 -05003624 #Const kDefault 0
Cary Clark682c58d2018-05-16 07:07:07 -04003625 #Line # draws Bitmap into lattice rectangle ##
Cary Clark2f466242017-12-11 16:03:17 -05003626 ##
3627
3628 #Const kTransparent 1
Cary Clark682c58d2018-05-16 07:07:07 -04003629 #Line # skips lattice rectangle by making it transparent ##
Cary Clark2f466242017-12-11 16:03:17 -05003630 ##
3631
3632 #Const kFixedColor 2
Cary Clark682c58d2018-05-16 07:07:07 -04003633 #Line # draws one of fColors into lattice rectangle ##
Cary Clark8032b982017-07-28 11:04:54 -04003634 ##
3635 ##
3636
Cary Clark61313f32018-10-08 14:57:48 -04003637#Subtopic Members
Cary Clark682c58d2018-05-16 07:07:07 -04003638##
3639
Cary Clark8032b982017-07-28 11:04:54 -04003640 #Member const int* fXDivs
Cary Clark5538c132018-06-14 12:28:14 -04003641 #Line # x-axis values dividing bitmap ##
3642 Array of x-axis values that divide the bitmap vertically.
Cary Clarkbad5ad72017-08-03 17:14:08 -04003643 Array entries must be unique, increasing, greater than or equal to
3644 fBounds left edge, and less than fBounds right edge.
3645 Set the first element to fBounds left to collapse the left column of
3646 fixed grid entries.
Cary Clark8032b982017-07-28 11:04:54 -04003647 ##
3648
3649 #Member const int* fYDivs
Cary Clark5538c132018-06-14 12:28:14 -04003650 #Line # y-axis values dividing bitmap ##
3651 Array of y-axis values that divide the bitmap horizontally.
Cary Clarkbad5ad72017-08-03 17:14:08 -04003652 Array entries must be unique, increasing, greater than or equal to
3653 fBounds top edge, and less than fBounds bottom edge.
3654 Set the first element to fBounds top to collapse the top row of fixed
3655 grid entries.
Cary Clark8032b982017-07-28 11:04:54 -04003656 ##
3657
Cary Clark2f466242017-12-11 16:03:17 -05003658 #Member const RectType* fRectTypes
Cary Clark682c58d2018-05-16 07:07:07 -04003659 #Line # array of fill types ##
Cary Clark2f466242017-12-11 16:03:17 -05003660 Optional array of fill types, one per rectangular grid entry:
Cary Clark2be81cf2018-09-13 12:04:30 -04003661 array length must be #Formula # (fXCount + 1) * (fYCount + 1) ##.
Cary Clark6fc50412017-09-21 12:31:06 -04003662
Cary Clark2f466242017-12-11 16:03:17 -05003663 Each RectType is one of: kDefault, kTransparent, kFixedColor.
3664
Cary Clark8032b982017-07-28 11:04:54 -04003665 Array entries correspond to the rectangular grid entries, ascending
3666 left to right and then top to bottom.
3667 ##
3668
3669 #Member int fXCount
Cary Clark682c58d2018-05-16 07:07:07 -04003670 #Line # number of x-coordinates ##
Cary Clarkbad5ad72017-08-03 17:14:08 -04003671 Number of entries in fXDivs array; one less than the number of
3672 horizontal divisions.
Cary Clark8032b982017-07-28 11:04:54 -04003673 ##
3674
3675 #Member int fYCount
Cary Clark682c58d2018-05-16 07:07:07 -04003676 #Line # number of y-coordinates ##
Cary Clarkbad5ad72017-08-03 17:14:08 -04003677 Number of entries in fYDivs array; one less than the number of vertical
3678 divisions.
Cary Clark8032b982017-07-28 11:04:54 -04003679 ##
3680
3681 #Member const SkIRect* fBounds
Cary Clark682c58d2018-05-16 07:07:07 -04003682 #Line # source bounds to draw from ##
Cary Clark8032b982017-07-28 11:04:54 -04003683 Optional subset IRect source to draw from.
3684 If nullptr, source bounds is dimensions of Bitmap or Image.
3685 ##
3686
Cary Clark2f466242017-12-11 16:03:17 -05003687 #Member const SkColor* fColors
Cary Clark682c58d2018-05-16 07:07:07 -04003688 #Line # array of colors ##
Cary Clark2f466242017-12-11 16:03:17 -05003689 Optional array of colors, one per rectangular grid entry.
Cary Clark2be81cf2018-09-13 12:04:30 -04003690 Array length must be #Formula # (fXCount + 1) * (fYCount + 1) ##.
Cary Clark2f466242017-12-11 16:03:17 -05003691
3692 Array entries correspond to the rectangular grid entries, ascending
3693 left to right, then top to bottom.
3694 ##
3695
Cary Clark8032b982017-07-28 11:04:54 -04003696#Struct Lattice ##
3697
3698#Method void drawBitmapLattice(const SkBitmap& bitmap, const Lattice& lattice, const SkRect& dst,
3699 const SkPaint* paint = nullptr)
Cary Clarkab2621d2018-01-30 10:08:57 -05003700#In Draw_Image
Cary Clark78de7512018-02-07 07:27:09 -05003701#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05003702#Line # draws proportionally stretched Bitmap ##
Cary Clark8032b982017-07-28 11:04:54 -04003703
Cary Clark80247e52018-07-11 16:18:41 -04003704Draws Bitmap bitmap stretched proportionally to fit into Rect dst.
Cary Clark8032b982017-07-28 11:04:54 -04003705
3706Lattice lattice divides bitmap into a rectangular grid.
3707Each intersection of an even-numbered row and column is fixed; like the corners
Cary Clarkbad5ad72017-08-03 17:14:08 -04003708of drawBitmapNine, fixed lattice elements never scale larger than their initial
Cary Clarkbc5697d2017-10-04 14:31:33 -04003709size and shrink proportionately when all fixed elements exceed the bitmap
Cary Clarkbad5ad72017-08-03 17:14:08 -04003710dimension. All other grid elements scale to fill the available space, if any.
Cary Clark8032b982017-07-28 11:04:54 -04003711
3712Additionally transform draw using Clip, Matrix, and optional Paint paint.
Cary Clark8032b982017-07-28 11:04:54 -04003713
Cary Clark682c58d2018-05-16 07:07:07 -04003714#paint_as_used_by_draw_lattice_or_draw_nine(bitmap)#
Cary Clarkbad5ad72017-08-03 17:14:08 -04003715
3716If generated mask extends beyond bitmap bounds, replicate bitmap edge colors,
3717just as Shader made from SkShader::MakeBitmapShader with
Cary Clarkbc5697d2017-10-04 14:31:33 -04003718SkShader::kClamp_TileMode set replicates the bitmap edge color when it samples
Cary Clarkbad5ad72017-08-03 17:14:08 -04003719outside of its bounds.
3720
3721#Param bitmap Bitmap containing pixels, dimensions, and format ##
3722#Param lattice division of bitmap into fixed and variable rectangles ##
3723#Param dst destination Rect of image to draw to ##
3724#Param paint Paint containing Blend_Mode, Color_Filter, Image_Filter,
3725 and so on; or nullptr
3726##
Cary Clark8032b982017-07-28 11:04:54 -04003727
3728#Example
3729#Height 128
3730#Description
3731 The two leftmost bitmap draws has four corners and sides to the left and right of center.
3732 The leftmost bitmap draw scales the width of corners proportionately to fit.
Herb Derbyefe39bc2018-05-01 17:06:20 -04003733 The third and fourth draw corners are not scaled; the sides are scaled to
Cary Clark8032b982017-07-28 11:04:54 -04003734 fill the remaining space; the center is transparent.
3735 The rightmost bitmap draw has four corners scaled vertically to fit, and uses sides above
3736 and below center to fill the remaining space.
3737##
Cary Clarkbad5ad72017-08-03 17:14:08 -04003738void draw(SkCanvas* canvas) {
3739 SkIRect center = { 20, 10, 50, 40 };
3740 SkBitmap bitmap;
3741 bitmap.allocPixels(SkImageInfo::MakeN32Premul(60, 60));
3742 SkCanvas bitCanvas(bitmap);
3743 SkPaint paint;
3744 SkColor gray = 0xFF000000;
3745 int left = 0;
3746 for (auto right: { center.fLeft, center.fRight, bitmap.width() } ) {
3747 int top = 0;
3748 for (auto bottom: { center.fTop, center.fBottom, bitmap.height() } ) {
3749 paint.setColor(gray);
3750 bitCanvas.drawIRect(SkIRect::MakeLTRB(left, top, right, bottom), paint);
3751 gray += 0x001f1f1f;
3752 top = bottom;
3753 }
Herb Derbyefe39bc2018-05-01 17:06:20 -04003754 left = right;
Cary Clarkbad5ad72017-08-03 17:14:08 -04003755 }
3756 const int xDivs[] = { center.fLeft, center.fRight };
3757 const int yDivs[] = { center.fTop, center.fBottom };
Cary Clarkca3ebcd2017-12-12 11:22:38 -05003758 SkCanvas::Lattice::RectType fillTypes[3][3];
3759 memset(fillTypes, 0, sizeof(fillTypes));
3760 fillTypes[1][1] = SkCanvas::Lattice::kTransparent;
3761 SkColor dummy[9]; // temporary pending bug fix
3762 SkCanvas::Lattice lattice = { xDivs, yDivs, fillTypes[0], SK_ARRAY_COUNT(xDivs),
3763 SK_ARRAY_COUNT(yDivs), nullptr, dummy };
Cary Clarkbad5ad72017-08-03 17:14:08 -04003764 for (auto dest: { 20, 30, 40, 60, 90 } ) {
Cary Clarkca3ebcd2017-12-12 11:22:38 -05003765 canvas->drawBitmapLattice(bitmap, lattice, SkRect::MakeWH(dest, 110 - dest), nullptr);
Cary Clarkbad5ad72017-08-03 17:14:08 -04003766 canvas->translate(dest + 4, 0);
3767 }
Cary Clark8032b982017-07-28 11:04:54 -04003768}
3769##
3770
Cary Clark2ade9972017-11-02 17:49:34 -04003771#SeeAlso drawImageLattice drawBitmap drawBitmapNine Lattice
Cary Clark8032b982017-07-28 11:04:54 -04003772
3773##
3774
3775# ------------------------------------------------------------------------------
3776
3777#Method void drawImageLattice(const SkImage* image, const Lattice& lattice, const SkRect& dst,
3778 const SkPaint* paint = nullptr)
Cary Clarkab2621d2018-01-30 10:08:57 -05003779#In Draw_Image
Cary Clark78de7512018-02-07 07:27:09 -05003780#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05003781#Line # draws proportionally stretched Image ##
Cary Clark8032b982017-07-28 11:04:54 -04003782
Cary Clark80247e52018-07-11 16:18:41 -04003783Draws Image image stretched proportionally to fit into Rect dst.
Cary Clark8032b982017-07-28 11:04:54 -04003784
3785Lattice lattice divides image into a rectangular grid.
3786Each intersection of an even-numbered row and column is fixed; like the corners
Cary Clarkbad5ad72017-08-03 17:14:08 -04003787of drawBitmapNine, fixed lattice elements never scale larger than their initial
Cary Clarkbc5697d2017-10-04 14:31:33 -04003788size and shrink proportionately when all fixed elements exceed the bitmap
Cary Clarkbad5ad72017-08-03 17:14:08 -04003789dimension. All other grid elements scale to fill the available space, if any.
Cary Clark8032b982017-07-28 11:04:54 -04003790
3791Additionally transform draw using Clip, Matrix, and optional Paint paint.
Cary Clark8032b982017-07-28 11:04:54 -04003792
Cary Clark682c58d2018-05-16 07:07:07 -04003793#paint_as_used_by_draw_lattice_or_draw_nine(image)#
Cary Clarkbad5ad72017-08-03 17:14:08 -04003794
3795If generated mask extends beyond bitmap bounds, replicate bitmap edge colors,
3796just as Shader made from SkShader::MakeBitmapShader with
Cary Clarkbc5697d2017-10-04 14:31:33 -04003797SkShader::kClamp_TileMode set replicates the bitmap edge color when it samples
Cary Clarkbad5ad72017-08-03 17:14:08 -04003798outside of its bounds.
3799
3800#Param image Image containing pixels, dimensions, and format ##
3801#Param lattice division of bitmap into fixed and variable rectangles ##
3802#Param dst destination Rect of image to draw to ##
3803#Param paint Paint containing Blend_Mode, Color_Filter, Image_Filter,
3804 and so on; or nullptr
3805##
Cary Clark8032b982017-07-28 11:04:54 -04003806
3807#Example
3808#Height 128
3809#Description
3810 The leftmost image is smaller than center; only corners are drawn, all scaled to fit.
Cary Clarkce101242017-09-01 15:51:02 -04003811 The second image equals the size of center; only corners are drawn without scaling.
3812 The remaining images are larger than center. All corners draw without scaling. The sides
Cary Clark8032b982017-07-28 11:04:54 -04003813 are scaled if needed to take up the remaining space; the center is transparent.
3814##
Cary Clarkbad5ad72017-08-03 17:14:08 -04003815void draw(SkCanvas* canvas) {
3816 SkIRect center = { 20, 10, 50, 40 };
3817 SkBitmap bitmap;
3818 bitmap.allocPixels(SkImageInfo::MakeN32Premul(60, 60));
3819 SkCanvas bitCanvas(bitmap);
3820 SkPaint paint;
3821 SkColor gray = 0xFF000000;
3822 int left = 0;
3823 for (auto right: { center.fLeft, center.fRight, bitmap.width() } ) {
3824 int top = 0;
3825 for (auto bottom: { center.fTop, center.fBottom, bitmap.height() } ) {
3826 paint.setColor(gray);
3827 bitCanvas.drawIRect(SkIRect::MakeLTRB(left, top, right, bottom), paint);
3828 gray += 0x001f1f1f;
3829 top = bottom;
3830 }
Herb Derbyefe39bc2018-05-01 17:06:20 -04003831 left = right;
Cary Clarkbad5ad72017-08-03 17:14:08 -04003832 }
Cary Clarkbad5ad72017-08-03 17:14:08 -04003833 sk_sp<SkImage> image = SkImage::MakeFromBitmap(bitmap);
3834 SkImage* imagePtr = image.get();
3835 for (auto dest: { 20, 30, 40, 60, 90 } ) {
3836 canvas->drawImageNine(imagePtr, center, SkRect::MakeWH(dest, dest), nullptr);
3837 canvas->translate(dest + 4, 0);
3838 }
Cary Clark8032b982017-07-28 11:04:54 -04003839}
3840##
3841
Cary Clark2ade9972017-11-02 17:49:34 -04003842#SeeAlso drawBitmapLattice drawImage drawImageNine Lattice
Cary Clark8032b982017-07-28 11:04:54 -04003843
3844##
3845
Cary Clark682c58d2018-05-16 07:07:07 -04003846#Subtopic Lattice ##
3847
Cary Clark08895c42018-02-01 09:37:32 -05003848#Subtopic Draw_Image ##
Cary Clark8032b982017-07-28 11:04:54 -04003849
3850# ------------------------------------------------------------------------------
Cary Clark78de7512018-02-07 07:27:09 -05003851#Subtopic Draw_Text
Cary Clark78de7512018-02-07 07:27:09 -05003852#Line # draws text into Canvas ##
3853##
Cary Clark8032b982017-07-28 11:04:54 -04003854
Mike Reedcb268f62019-01-07 12:51:32 -05003855#Method void drawSimpleText(const void* text, size_t byteLength, SkTextEncoding encoding,
3856 SkScalar x, SkScalar y, const SkFont& font, const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05003857#In Draw_Text
3858#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05003859#Line # draws text at (x, y), using font advance ##
Cary Clark09d80c02018-10-31 12:14:03 -04003860#Populate
Cary Clark8032b982017-07-28 11:04:54 -04003861
3862#Example
3863#Height 200
3864#Description
Mike Reedcb268f62019-01-07 12:51:32 -05003865The same text is drawn varying Paint_Text_Size and varying
3866Matrix.
Cary Clark8032b982017-07-28 11:04:54 -04003867##
Cary Clarkbad5ad72017-08-03 17:14:08 -04003868void draw(SkCanvas* canvas) {
3869 SkPaint paint;
Mike Reedcb268f62019-01-07 12:51:32 -05003870 SkFont font;
Cary Clarkbad5ad72017-08-03 17:14:08 -04003871 float textSizes[] = { 12, 18, 24, 36 };
3872 for (auto size: textSizes ) {
Mike Reedcb268f62019-01-07 12:51:32 -05003873 font.setSize(size);
Hal Canary35d14702019-01-15 11:57:41 -05003874 canvas->drawSimpleText("Aa", 2, kUTF8_SkTextEncoding, 10, 20, font, paint);
Cary Clarkbad5ad72017-08-03 17:14:08 -04003875 canvas->translate(0, size * 2);
3876 }
Mike Reedcb268f62019-01-07 12:51:32 -05003877 font.setSize(12);
Cary Clarkbad5ad72017-08-03 17:14:08 -04003878 float yPos = 20;
3879 for (auto size: textSizes ) {
3880 float scale = size / 12.f;
3881 canvas->resetMatrix();
3882 canvas->translate(100, 0);
3883 canvas->scale(scale, scale);
Hal Canary35d14702019-01-15 11:57:41 -05003884 canvas->drawSimpleText("Aa", 2, kUTF8_SkTextEncoding,
3885 10 / scale, yPos / scale, font, paint);
Herb Derbyefe39bc2018-05-01 17:06:20 -04003886 yPos += size * 2;
Cary Clarkbad5ad72017-08-03 17:14:08 -04003887 }
3888}
Cary Clark8032b982017-07-28 11:04:54 -04003889##
3890
Mike Reed53a11f52019-01-03 12:18:06 -05003891#SeeAlso drawString drawTextBlob
Cary Clark8032b982017-07-28 11:04:54 -04003892
3893##
3894
Cary Clark8032b982017-07-28 11:04:54 -04003895# ------------------------------------------------------------------------------
3896
3897#Method void drawTextBlob(const SkTextBlob* blob, SkScalar x, SkScalar y, const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05003898#In Draw_Text
3899#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05003900#Line # draws text with arrays of positions and Paint ##
Cary Clark80247e52018-07-11 16:18:41 -04003901Draws Text_Blob blob at (x, y), using Clip, Matrix, and Paint paint.
Cary Clarkbad5ad72017-08-03 17:14:08 -04003902
Cary Clarkce101242017-09-01 15:51:02 -04003903blob contains Glyphs, their positions, and paint attributes specific to text:
Cary Clark3e73dcf2018-12-06 08:41:14 -05003904#font_metrics#.
Cary Clark8032b982017-07-28 11:04:54 -04003905
Mike Reed0f9d33e2018-12-05 10:54:05 -05003906Paint_Text_Encoding must be set to kGlyphID_SkTextEncoding.
Cary Clark3cd22cc2017-12-01 11:49:58 -05003907
Cary Clarkd2ca79c2018-08-10 13:09:13 -04003908Elements of paint: Anti_Alias, Blend_Mode, Color including Color_Alpha,
3909Color_Filter, Paint_Dither, Draw_Looper, Mask_Filter, Path_Effect, Shader, and
Cary Clark61313f32018-10-08 14:57:48 -04003910Paint_Style; apply to blob. If Paint contains SkPaint::kStroke_Style:
Cary Clarkd2ca79c2018-08-10 13:09:13 -04003911Paint_Miter_Limit, Paint_Stroke_Cap, Paint_Stroke_Join, and Paint_Stroke_Width;
3912apply to Path created from blob.
Cary Clark8032b982017-07-28 11:04:54 -04003913
Cary Clarkce101242017-09-01 15:51:02 -04003914#Param blob Glyphs, positions, and their paints' text size, typeface, and so on ##
Cary Clarkbad5ad72017-08-03 17:14:08 -04003915#Param x horizontal offset applied to blob ##
3916#Param y vertical offset applied to blob ##
3917#Param paint blend, color, stroking, and so on, used to draw ##
Cary Clark8032b982017-07-28 11:04:54 -04003918
3919#Example
3920#Height 120
Cary Clarke80cd442018-07-17 13:19:56 -04003921void draw(SkCanvas* canvas) {
Mike Reed0f9d33e2018-12-05 10:54:05 -05003922 SkTextBlobBuilder textBlobBuilder;
3923 const char bunny[] = "/(^x^)\\";
3924 const int len = sizeof(bunny) - 1;
3925 uint16_t glyphs[len];
Mike Reed0f9d33e2018-12-05 10:54:05 -05003926 SkFont font;
Cary Clark26d173f2019-01-07 15:16:56 -05003927 font.textToGlyphs(bunny, len, SkTextEncoding::kUTF8, glyphs, len);
Mike Reed0f9d33e2018-12-05 10:54:05 -05003928 int runs[] = { 3, 1, 3 };
3929 SkPoint textPos = { 20, 100 };
3930 int glyphIndex = 0;
3931 for (auto runLen : runs) {
3932 font.setSize(1 == runLen ? 20 : 50);
3933 const SkTextBlobBuilder::RunBuffer& run =
3934 textBlobBuilder.allocRun(font, runLen, textPos.fX, textPos.fY);
3935 memcpy(run.glyphs, &glyphs[glyphIndex], sizeof(glyphs[0]) * runLen);
Cary Clark26d173f2019-01-07 15:16:56 -05003936 font.setSize(1 == runLen ? 20 : 50);
3937 textPos.fX += font.measureText(&glyphs[glyphIndex], sizeof(glyphs[0]) * runLen,
3938 SkTextEncoding::kGlyphID);
Mike Reed0f9d33e2018-12-05 10:54:05 -05003939 glyphIndex += runLen;
3940 }
3941 sk_sp<const SkTextBlob> blob = textBlobBuilder.make();
Cary Clark26d173f2019-01-07 15:16:56 -05003942 canvas->drawTextBlob(blob.get(), 0, 0, SkPaint());
Cary Clarke80cd442018-07-17 13:19:56 -04003943}
Cary Clark8032b982017-07-28 11:04:54 -04003944##
3945
Mike Reed53a11f52019-01-03 12:18:06 -05003946#SeeAlso drawText
Cary Clark8032b982017-07-28 11:04:54 -04003947
3948##
3949
3950# ------------------------------------------------------------------------------
3951
Herb Derbyefe39bc2018-05-01 17:06:20 -04003952#Method void drawTextBlob(const sk_sp<SkTextBlob>& blob, SkScalar x, SkScalar y, const SkPaint& paint)
Cary Clark8032b982017-07-28 11:04:54 -04003953
Cary Clark80247e52018-07-11 16:18:41 -04003954Draws Text_Blob blob at (x, y), using Clip, Matrix, and Paint paint.
Cary Clarkbad5ad72017-08-03 17:14:08 -04003955
Cary Clarkce101242017-09-01 15:51:02 -04003956blob contains Glyphs, their positions, and paint attributes specific to text:
Cary Clark3e73dcf2018-12-06 08:41:14 -05003957#font_metrics#.
Cary Clark8032b982017-07-28 11:04:54 -04003958
Mike Reed0f9d33e2018-12-05 10:54:05 -05003959Paint_Text_Encoding must be set to kGlyphID_SkTextEncoding.
Cary Clark3cd22cc2017-12-01 11:49:58 -05003960
Herb Derbyefe39bc2018-05-01 17:06:20 -04003961Elements of paint: Path_Effect, Mask_Filter, Shader, Color_Filter,
Cary Clark8032b982017-07-28 11:04:54 -04003962Image_Filter, and Draw_Looper; apply to blob.
3963
Cary Clarkce101242017-09-01 15:51:02 -04003964#Param blob Glyphs, positions, and their paints' text size, typeface, and so on ##
Cary Clarkbad5ad72017-08-03 17:14:08 -04003965#Param x horizontal offset applied to blob ##
3966#Param y vertical offset applied to blob ##
3967#Param paint blend, color, stroking, and so on, used to draw ##
Cary Clark8032b982017-07-28 11:04:54 -04003968
3969#Example
3970#Height 120
3971#Description
Cary Clark8032b982017-07-28 11:04:54 -04003972Paint attributes related to text, like text size, have no effect on paint passed to drawTextBlob.
3973##
Cary Clarkbad5ad72017-08-03 17:14:08 -04003974 void draw(SkCanvas* canvas) {
3975 SkTextBlobBuilder textBlobBuilder;
Cary Clarkfc93eb02018-11-24 22:32:31 -05003976 SkFont font;
3977 font.setSize(50);
Herb Derbyefe39bc2018-05-01 17:06:20 -04003978 const SkTextBlobBuilder::RunBuffer& run =
Cary Clarkfc93eb02018-11-24 22:32:31 -05003979 textBlobBuilder.allocRun(font, 1, 20, 100);
Cary Clarkbad5ad72017-08-03 17:14:08 -04003980 run.glyphs[0] = 20;
3981 sk_sp<const SkTextBlob> blob = textBlobBuilder.make();
Cary Clarkfc93eb02018-11-24 22:32:31 -05003982 SkPaint paint;
Cary Clarkbad5ad72017-08-03 17:14:08 -04003983 paint.setTextSize(10);
3984 paint.setColor(SK_ColorBLUE);
3985 canvas->drawTextBlob(blob.get(), 0, 0, paint);
3986 }
Cary Clark8032b982017-07-28 11:04:54 -04003987##
3988
Mike Reed53a11f52019-01-03 12:18:06 -05003989#SeeAlso drawText
Cary Clark8032b982017-07-28 11:04:54 -04003990
3991##
3992
3993# ------------------------------------------------------------------------------
3994
Herb Derbyefe39bc2018-05-01 17:06:20 -04003995#Method void drawPicture(const SkPicture* picture)
Cary Clark78de7512018-02-07 07:27:09 -05003996#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05003997#Line # draws Picture using Clip and Matrix ##
Cary Clark09d80c02018-10-31 12:14:03 -04003998#Populate
Cary Clark8032b982017-07-28 11:04:54 -04003999
4000#Example
Herb Derbyefe39bc2018-05-01 17:06:20 -04004001void draw(SkCanvas* canvas) {
Cary Clarkbad5ad72017-08-03 17:14:08 -04004002 SkPictureRecorder recorder;
4003 SkCanvas* recordingCanvas = recorder.beginRecording(50, 50);
4004 for (auto color : { SK_ColorRED, SK_ColorBLUE, 0xff007f00 } ) {
4005 SkPaint paint;
4006 paint.setColor(color);
4007 recordingCanvas->drawRect({10, 10, 30, 40}, paint);
4008 recordingCanvas->translate(10, 10);
4009 recordingCanvas->scale(1.2f, 1.4f);
4010 }
4011 sk_sp<SkPicture> playback = recorder.finishRecordingAsPicture();
Cary Clarkbad5ad72017-08-03 17:14:08 -04004012 canvas->drawPicture(playback);
4013 canvas->scale(2, 2);
4014 canvas->translate(50, 0);
4015 canvas->drawPicture(playback);
Cary Clark8032b982017-07-28 11:04:54 -04004016}
4017##
4018
Cary Clark2ade9972017-11-02 17:49:34 -04004019#SeeAlso drawDrawable SkPicture SkPicture::playback
Cary Clark8032b982017-07-28 11:04:54 -04004020
4021##
4022
4023# ------------------------------------------------------------------------------
4024
Herb Derbyefe39bc2018-05-01 17:06:20 -04004025#Method void drawPicture(const sk_sp<SkPicture>& picture)
Cary Clark09d80c02018-10-31 12:14:03 -04004026#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004027
4028#Example
Herb Derbyefe39bc2018-05-01 17:06:20 -04004029void draw(SkCanvas* canvas) {
Cary Clarkbad5ad72017-08-03 17:14:08 -04004030 SkPictureRecorder recorder;
4031 SkCanvas* recordingCanvas = recorder.beginRecording(50, 50);
4032 for (auto color : { SK_ColorRED, SK_ColorBLUE, 0xff007f00 } ) {
4033 SkPaint paint;
4034 paint.setColor(color);
4035 recordingCanvas->drawRect({10, 10, 30, 40}, paint);
4036 recordingCanvas->translate(10, 10);
4037 recordingCanvas->scale(1.2f, 1.4f);
4038 }
4039 sk_sp<SkPicture> playback = recorder.finishRecordingAsPicture();
4040 canvas->drawPicture(playback);
4041 canvas->scale(2, 2);
4042 canvas->translate(50, 0);
4043 canvas->drawPicture(playback);
Cary Clark8032b982017-07-28 11:04:54 -04004044}
4045##
4046
Cary Clark2ade9972017-11-02 17:49:34 -04004047#SeeAlso drawDrawable SkPicture SkPicture::playback
Cary Clark8032b982017-07-28 11:04:54 -04004048
4049##
4050
4051# ------------------------------------------------------------------------------
4052
4053#Method void drawPicture(const SkPicture* picture, const SkMatrix* matrix, const SkPaint* paint)
Cary Clark09d80c02018-10-31 12:14:03 -04004054#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004055
4056#Example
Herb Derbyefe39bc2018-05-01 17:06:20 -04004057void draw(SkCanvas* canvas) {
Cary Clarkbad5ad72017-08-03 17:14:08 -04004058 SkPaint paint;
4059 SkPictureRecorder recorder;
4060 SkCanvas* recordingCanvas = recorder.beginRecording(50, 50);
4061 for (auto color : { SK_ColorRED, SK_ColorBLUE, 0xff007f00 } ) {
4062 paint.setColor(color);
4063 recordingCanvas->drawRect({10, 10, 30, 40}, paint);
4064 recordingCanvas->translate(10, 10);
4065 recordingCanvas->scale(1.2f, 1.4f);
4066 }
4067 sk_sp<SkPicture> playback = recorder.finishRecordingAsPicture();
4068 const SkPicture* playbackPtr = playback.get();
4069 SkMatrix matrix;
4070 matrix.reset();
4071 for (auto alpha : { 70, 140, 210 } ) {
4072 paint.setAlpha(alpha);
4073 canvas->drawPicture(playbackPtr, &matrix, &paint);
4074 matrix.preTranslate(70, 70);
4075 }
Cary Clark8032b982017-07-28 11:04:54 -04004076}
4077##
4078
Cary Clark2ade9972017-11-02 17:49:34 -04004079#SeeAlso drawDrawable SkPicture SkPicture::playback
Cary Clark8032b982017-07-28 11:04:54 -04004080
4081##
4082
4083# ------------------------------------------------------------------------------
4084
Herb Derbyefe39bc2018-05-01 17:06:20 -04004085#Method void drawPicture(const sk_sp<SkPicture>& picture, const SkMatrix* matrix, const SkPaint* paint)
Cary Clark09d80c02018-10-31 12:14:03 -04004086#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004087
4088#Example
Herb Derbyefe39bc2018-05-01 17:06:20 -04004089void draw(SkCanvas* canvas) {
Cary Clarkbad5ad72017-08-03 17:14:08 -04004090 SkPaint paint;
4091 SkPictureRecorder recorder;
4092 SkCanvas* recordingCanvas = recorder.beginRecording(50, 50);
4093 for (auto color : { SK_ColorRED, SK_ColorBLUE, 0xff007f00 } ) {
4094 paint.setColor(color);
4095 recordingCanvas->drawRect({10, 10, 30, 40}, paint);
4096 recordingCanvas->translate(10, 10);
4097 recordingCanvas->scale(1.2f, 1.4f);
4098 }
4099 sk_sp<SkPicture> playback = recorder.finishRecordingAsPicture();
4100 SkMatrix matrix;
4101 matrix.reset();
4102 for (auto alpha : { 70, 140, 210 } ) {
4103 paint.setAlpha(alpha);
4104 canvas->drawPicture(playback, &matrix, &paint);
4105 matrix.preTranslate(70, 70);
4106 }
Cary Clark8032b982017-07-28 11:04:54 -04004107}
4108##
4109
Cary Clark2ade9972017-11-02 17:49:34 -04004110#SeeAlso drawDrawable SkPicture SkPicture::playback
Cary Clark8032b982017-07-28 11:04:54 -04004111
4112##
4113
4114# ------------------------------------------------------------------------------
4115
4116#Method void drawVertices(const SkVertices* vertices, SkBlendMode mode, const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05004117#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05004118#Line # draws Vertices, a triangle mesh ##
Cary Clark09d80c02018-10-31 12:14:03 -04004119#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004120
4121#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04004122void draw(SkCanvas* canvas) {
4123 SkPaint paint;
4124 SkPoint points[] = { { 0, 0 }, { 250, 0 }, { 100, 100 }, { 0, 250 } };
4125 SkColor colors[] = { SK_ColorRED, SK_ColorBLUE, SK_ColorYELLOW, SK_ColorCYAN };
4126 auto vertices = SkVertices::MakeCopy(SkVertices::kTriangleFan_VertexMode,
4127 SK_ARRAY_COUNT(points), points, nullptr, colors);
4128 canvas->drawVertices(vertices.get(), SkBlendMode::kSrc, paint);
4129}
Cary Clark8032b982017-07-28 11:04:54 -04004130##
4131
Cary Clark2ade9972017-11-02 17:49:34 -04004132#SeeAlso drawPatch drawPicture
Cary Clark8032b982017-07-28 11:04:54 -04004133
4134##
4135
4136# ------------------------------------------------------------------------------
4137
4138#Method void drawVertices(const sk_sp<SkVertices>& vertices, SkBlendMode mode, const SkPaint& paint)
Cary Clark09d80c02018-10-31 12:14:03 -04004139#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004140
4141#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04004142void draw(SkCanvas* canvas) {
4143 SkPaint paint;
4144 SkPoint points[] = { { 0, 0 }, { 250, 0 }, { 100, 100 }, { 0, 250 } };
4145 SkPoint texs[] = { { 0, 0 }, { 0, 250 }, { 250, 250 }, { 250, 0 } };
4146 SkColor colors[] = { SK_ColorRED, SK_ColorBLUE, SK_ColorYELLOW, SK_ColorCYAN };
4147 paint.setShader(SkGradientShader::MakeLinear(points, colors, nullptr, 4,
4148 SkShader::kClamp_TileMode));
4149 auto vertices = SkVertices::MakeCopy(SkVertices::kTriangleFan_VertexMode,
4150 SK_ARRAY_COUNT(points), points, texs, colors);
Ruiqi Mao94d57c42018-07-02 15:20:10 -04004151 canvas->drawVertices(vertices, SkBlendMode::kDarken, paint);
4152}
4153##
4154
4155#SeeAlso drawPatch drawPicture
4156
4157##
4158
4159# ------------------------------------------------------------------------------
4160
Ruiqi Maoc97a3392018-08-15 10:44:19 -04004161#Method void drawVertices(const SkVertices* vertices, const SkVertices::Bone bones[],
4162 int boneCount, SkBlendMode mode, const SkPaint& paint)
Cary Clark09d80c02018-10-31 12:14:03 -04004163#Populate
Ruiqi Mao94d57c42018-07-02 15:20:10 -04004164
Ruiqi Maoc97a3392018-08-15 10:44:19 -04004165#NoExample
Ruiqi Mao94d57c42018-07-02 15:20:10 -04004166void draw(SkCanvas* canvas) {
4167 SkPaint paint;
4168 SkPoint points[] = { { 0, 0 }, { 250, 0 }, { 100, 100 }, { 0, 250 } };
4169 SkPoint texs[] = { { 0, 0 }, { 0, 250 }, { 250, 250 }, { 250, 0 } };
4170 SkColor colors[] = { SK_ColorRED, SK_ColorBLUE, SK_ColorYELLOW, SK_ColorCYAN };
4171 SkVertices::BoneIndices boneIndices[] = { { 0, 0, 0, 0 },
4172 { 1, 0, 0, 0 },
4173 { 2, 0, 0, 0 },
4174 { 3, 0, 0, 0 } };
4175 SkVertices::BoneWeights boneWeights[] = { { 0.0f, 0.0f, 0.0f, 0.0f },
4176 { 1.0f, 0.0f, 0.0f, 0.0f },
4177 { 1.0f, 0.0f, 0.0f, 0.0f },
4178 { 1.0f, 0.0f, 0.0f, 0.0f } };
Ruiqi Maoc97a3392018-08-15 10:44:19 -04004179 SkVertices::Bone bones[] = { {{ 1.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f }},
4180 {{ 1.0f, 0.0f, 0.0f, 1.0f, 0.0f, 20.0f }},
4181 {{ 1.0f, 0.0f, 0.0f, 1.0f, 50.0f, 50.0f }},
4182 {{ 1.0f, 0.0f, 0.0f, 1.0f, 20.0f, 0.0f }} };
Ruiqi Mao94d57c42018-07-02 15:20:10 -04004183 paint.setShader(SkGradientShader::MakeLinear(points, colors, nullptr, 4,
4184 SkShader::kClamp_TileMode));
4185 auto vertices = SkVertices::MakeCopy(SkVertices::kTriangleFan_VertexMode,
4186 SK_ARRAY_COUNT(points), points, texs, colors, boneIndices, boneWeights);
4187 canvas->drawVertices(vertices.get(), bones, SK_ARRAY_COUNT(bones), SkBlendMode::kDarken, paint);
4188}
4189##
4190
4191#SeeAlso drawPatch drawPicture
4192
4193##
4194
4195# ------------------------------------------------------------------------------
4196
Ruiqi Maoc97a3392018-08-15 10:44:19 -04004197#Method void drawVertices(const sk_sp<SkVertices>& vertices, const SkVertices::Bone bones[],
4198 int boneCount, SkBlendMode mode, const SkPaint& paint)
Cary Clark09d80c02018-10-31 12:14:03 -04004199#Populate
Ruiqi Mao94d57c42018-07-02 15:20:10 -04004200
Ruiqi Maoc97a3392018-08-15 10:44:19 -04004201#NoExample
Ruiqi Mao94d57c42018-07-02 15:20:10 -04004202void draw(SkCanvas* canvas) {
4203 SkPaint paint;
4204 SkPoint points[] = { { 0, 0 }, { 250, 0 }, { 100, 100 }, { 0, 250 } };
4205 SkPoint texs[] = { { 0, 0 }, { 0, 250 }, { 250, 250 }, { 250, 0 } };
4206 SkColor colors[] = { SK_ColorRED, SK_ColorBLUE, SK_ColorYELLOW, SK_ColorCYAN };
4207 SkVertices::BoneIndices boneIndices[] = { { 0, 0, 0, 0 },
4208 { 1, 0, 0, 0 },
4209 { 2, 0, 0, 0 },
4210 { 3, 0, 0, 0 } };
4211 SkVertices::BoneWeights boneWeights[] = { { 0.0f, 0.0f, 0.0f, 0.0f },
4212 { 1.0f, 0.0f, 0.0f, 0.0f },
4213 { 1.0f, 0.0f, 0.0f, 0.0f },
4214 { 1.0f, 0.0f, 0.0f, 0.0f } };
Ruiqi Maoc97a3392018-08-15 10:44:19 -04004215 SkVertices::Bone bones[] = { {{ 1.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f }},
4216 {{ 1.0f, 0.0f, 0.0f, 1.0f, 0.0f, 20.0f }},
4217 {{ 1.0f, 0.0f, 0.0f, 1.0f, 50.0f, 50.0f }},
4218 {{ 1.0f, 0.0f, 0.0f, 1.0f, 20.0f, 0.0f }} };
Ruiqi Mao94d57c42018-07-02 15:20:10 -04004219 paint.setShader(SkGradientShader::MakeLinear(points, colors, nullptr, 4,
4220 SkShader::kClamp_TileMode));
4221 auto vertices = SkVertices::MakeCopy(SkVertices::kTriangleFan_VertexMode,
4222 SK_ARRAY_COUNT(points), points, texs, colors, boneIndices, boneWeights);
4223 canvas->drawVertices(vertices, bones, SK_ARRAY_COUNT(bones), SkBlendMode::kDarken, paint);
Cary Clark8032b982017-07-28 11:04:54 -04004224}
4225##
4226
Cary Clark2ade9972017-11-02 17:49:34 -04004227#SeeAlso drawPatch drawPicture
Cary Clark8032b982017-07-28 11:04:54 -04004228
4229##
4230
4231# ------------------------------------------------------------------------------
4232
4233#Method void drawPatch(const SkPoint cubics[12], const SkColor colors[4],
4234 const SkPoint texCoords[4], SkBlendMode mode, const SkPaint& paint)
Cary Clark78de7512018-02-07 07:27:09 -05004235#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05004236#Line # draws Coons_Patch ##
Cary Clark09d80c02018-10-31 12:14:03 -04004237#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004238
4239#Example
4240#Image 5
Cary Clarkbad5ad72017-08-03 17:14:08 -04004241void draw(SkCanvas* canvas) {
4242 // SkBitmap source = cmbkygk;
4243 SkPaint paint;
4244 paint.setFilterQuality(kLow_SkFilterQuality);
4245 paint.setAntiAlias(true);
4246 SkPoint cubics[] = { { 3, 1 }, { 4, 2 }, { 5, 1 }, { 7, 3 },
4247 /* { 7, 3 }, */ { 6, 4 }, { 7, 5 }, { 5, 7 },
4248 /* { 5, 7 }, */ { 4, 6 }, { 3, 7 }, { 1, 5 },
4249 /* { 1, 5 }, */ { 2, 4 }, { 1, 3 }, /* { 3, 1 } */ };
4250 SkColor colors[] = { 0xbfff0000, 0xbf0000ff, 0xbfff00ff, 0xbf00ffff };
4251 SkPoint texCoords[] = { { -30, -30 }, { 162, -30}, { 162, 162}, { -30, 162} };
4252 paint.setShader(SkShader::MakeBitmapShader(source, SkShader::kClamp_TileMode,
4253 SkShader::kClamp_TileMode, nullptr));
4254 canvas->scale(15, 15);
4255 for (auto blend : { SkBlendMode::kSrcOver, SkBlendMode::kModulate, SkBlendMode::kXor } ) {
4256 canvas->drawPatch(cubics, colors, texCoords, blend, paint);
4257 canvas->translate(4, 4);
4258 }
Cary Clark8032b982017-07-28 11:04:54 -04004259}
4260##
4261
Cary Clark2ade9972017-11-02 17:49:34 -04004262#ToDo can patch use image filter? ##
Cary Clark77b3f3a2018-11-07 14:59:03 -05004263#SeeAlso drawVertices drawPicture
Cary Clark8032b982017-07-28 11:04:54 -04004264
4265##
4266
4267# ------------------------------------------------------------------------------
4268
4269#Method void drawPatch(const SkPoint cubics[12], const SkColor colors[4],
Herb Derbyefe39bc2018-05-01 17:06:20 -04004270 const SkPoint texCoords[4], const SkPaint& paint)
Cary Clark09d80c02018-10-31 12:14:03 -04004271#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004272
4273#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04004274void draw(SkCanvas* canvas) {
4275 SkPaint paint;
4276 paint.setAntiAlias(true);
4277 SkPoint cubics[] = { { 3, 1 }, { 4, 2 }, { 5, 1 }, { 7, 3 },
4278 /* { 7, 3 }, */ { 6, 4 }, { 7, 5 }, { 5, 7 },
4279 /* { 5, 7 }, */ { 4, 6 }, { 3, 7 }, { 1, 5 },
4280 /* { 1, 5 }, */ { 2, 4 }, { 1, 3 }, /* { 3, 1 } */ };
4281 SkColor colors[] = { SK_ColorRED, SK_ColorBLUE, SK_ColorYELLOW, SK_ColorCYAN };
4282 canvas->scale(30, 30);
4283 canvas->drawPatch(cubics, colors, nullptr, paint);
4284 SkPoint text[] = { {3,0.9f}, {4,2.5f}, {5,0.9f}, {7.5f,3.2f}, {5.5f,4.2f},
4285 {7.5f,5.2f}, {5,7.5f}, {4,5.9f}, {3,7.5f}, {0.5f,5.2f}, {2.5f,4.2f},
4286 {0.5f,3.2f} };
Mike Reed33535cb2019-01-09 11:25:18 -05004287
4288 SkFont font(nullptr, 18.f / 30);
Cary Clarkbad5ad72017-08-03 17:14:08 -04004289 for (int i = 0; i< 10; ++i) {
4290 char digit = '0' + i;
Mike Reed33535cb2019-01-09 11:25:18 -05004291 canvas->drawSimpleText(&digit, kUTF8_SkTextEncoding, 1, text[i].fX, text[i].fY, font, paint);
Cary Clarkbad5ad72017-08-03 17:14:08 -04004292 }
Mike Reed33535cb2019-01-09 11:25:18 -05004293 canvas->drawString("10", text[10].fX, text[10].fY, font, paint);
4294 canvas->drawString("11", text[11].fX, text[11].fY, font, paint);
Cary Clarkbad5ad72017-08-03 17:14:08 -04004295 paint.setStyle(SkPaint::kStroke_Style);
4296 canvas->drawPoints(SkCanvas::kPolygon_PointMode, 12, cubics, paint);
4297 canvas->drawLine(cubics[11].fX, cubics[11].fY, cubics[0].fX, cubics[0].fY, paint);
Cary Clark8032b982017-07-28 11:04:54 -04004298}
4299##
4300
4301#Example
4302#Image 6
Cary Clarkbad5ad72017-08-03 17:14:08 -04004303void draw(SkCanvas* canvas) {
4304 // SkBitmap source = checkerboard;
4305 SkPaint paint;
4306 paint.setFilterQuality(kLow_SkFilterQuality);
4307 paint.setAntiAlias(true);
4308 SkPoint cubics[] = { { 3, 1 }, { 4, 2 }, { 5, 1 }, { 7, 3 },
4309 /* { 7, 3 }, */ { 6, 4 }, { 7, 5 }, { 5, 7 },
4310 /* { 5, 7 }, */ { 4, 6 }, { 3, 7 }, { 1, 5 },
4311 /* { 1, 5 }, */ { 2, 4 }, { 1, 3 }, /* { 3, 1 } */ };
4312 SkPoint texCoords[] = { { 0, 0 }, { 0, 62}, { 62, 62}, { 62, 0 } };
4313 paint.setShader(SkShader::MakeBitmapShader(source, SkShader::kClamp_TileMode,
4314 SkShader::kClamp_TileMode, nullptr));
4315 canvas->scale(30, 30);
4316 canvas->drawPatch(cubics, nullptr, texCoords, paint);
Cary Clark8032b982017-07-28 11:04:54 -04004317}
4318##
4319
Cary Clark2ade9972017-11-02 17:49:34 -04004320#ToDo can patch use image filter? ##
Cary Clark77b3f3a2018-11-07 14:59:03 -05004321#SeeAlso drawVertices drawPicture
Cary Clark8032b982017-07-28 11:04:54 -04004322
4323##
4324
4325# ------------------------------------------------------------------------------
4326
4327#Method void drawAtlas(const SkImage* atlas, const SkRSXform xform[], const SkRect tex[],
4328 const SkColor colors[], int count, SkBlendMode mode, const SkRect* cullRect,
4329 const SkPaint* paint)
Cary Clark78de7512018-02-07 07:27:09 -05004330#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05004331#Line # draws sprites using Clip, Matrix, and Paint ##
Cary Clark09d80c02018-10-31 12:14:03 -04004332#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004333
4334#Example
4335#Image 3
Cary Clarkbad5ad72017-08-03 17:14:08 -04004336void draw(SkCanvas* canvas) {
4337 // SkBitmap source = mandrill;
4338 SkRSXform xforms[] = { { .5f, 0, 0, 0 }, {0, .5f, 200, 100 } };
4339 SkRect tex[] = { { 0, 0, 250, 250 }, { 0, 0, 250, 250 } };
4340 SkColor colors[] = { 0x7f55aa00, 0x7f3333bf };
4341 const SkImage* imagePtr = image.get();
4342 canvas->drawAtlas(imagePtr, xforms, tex, colors, 2, SkBlendMode::kSrcOver, nullptr, nullptr);
Cary Clark8032b982017-07-28 11:04:54 -04004343}
4344##
4345
Cary Clark2ade9972017-11-02 17:49:34 -04004346#SeeAlso drawBitmap drawImage
Cary Clark8032b982017-07-28 11:04:54 -04004347
4348##
4349
4350# ------------------------------------------------------------------------------
4351
4352#Method void drawAtlas(const sk_sp<SkImage>& atlas, const SkRSXform xform[], const SkRect tex[],
4353 const SkColor colors[], int count, SkBlendMode mode, const SkRect* cullRect,
Herb Derbyefe39bc2018-05-01 17:06:20 -04004354 const SkPaint* paint)
Cary Clark09d80c02018-10-31 12:14:03 -04004355#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004356
4357#Example
4358#Image 3
Cary Clarkbad5ad72017-08-03 17:14:08 -04004359void draw(SkCanvas* canvas) {
4360 // SkBitmap source = mandrill;
4361 SkRSXform xforms[] = { { .5f, 0, 0, 0 }, {0, .5f, 200, 100 } };
4362 SkRect tex[] = { { 0, 0, 250, 250 }, { 0, 0, 250, 250 } };
4363 SkColor colors[] = { 0x7f55aa00, 0x7f3333bf };
4364 SkPaint paint;
4365 paint.setAlpha(127);
4366 canvas->drawAtlas(image, xforms, tex, colors, 2, SkBlendMode::kPlus, nullptr, &paint);
Cary Clark8032b982017-07-28 11:04:54 -04004367}
4368##
4369
4370#ToDo bug in example on cpu side, gpu looks ok ##
4371
Cary Clark2ade9972017-11-02 17:49:34 -04004372#SeeAlso drawBitmap drawImage
4373
Cary Clark8032b982017-07-28 11:04:54 -04004374##
4375
4376# ------------------------------------------------------------------------------
4377
4378#Method void drawAtlas(const SkImage* atlas, const SkRSXform xform[], const SkRect tex[], int count,
Herb Derbyefe39bc2018-05-01 17:06:20 -04004379 const SkRect* cullRect, const SkPaint* paint)
Cary Clark09d80c02018-10-31 12:14:03 -04004380#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004381
4382#Example
4383#Image 3
Cary Clarkbad5ad72017-08-03 17:14:08 -04004384void draw(SkCanvas* canvas) {
4385 // sk_sp<SkImage> image = mandrill;
4386 SkRSXform xforms[] = { { .5f, 0, 0, 0 }, {0, .5f, 200, 100 } };
4387 SkRect tex[] = { { 0, 0, 250, 250 }, { 0, 0, 250, 250 } };
4388 const SkImage* imagePtr = image.get();
4389 canvas->drawAtlas(imagePtr, xforms, tex, 2, nullptr, nullptr);
Cary Clark8032b982017-07-28 11:04:54 -04004390}
4391##
4392
Cary Clark2ade9972017-11-02 17:49:34 -04004393#SeeAlso drawBitmap drawImage
Cary Clark8032b982017-07-28 11:04:54 -04004394
4395##
4396
4397# ------------------------------------------------------------------------------
4398
4399#Method void drawAtlas(const sk_sp<SkImage>& atlas, const SkRSXform xform[], const SkRect tex[],
Herb Derbyefe39bc2018-05-01 17:06:20 -04004400 int count, const SkRect* cullRect, const SkPaint* paint)
Cary Clark09d80c02018-10-31 12:14:03 -04004401#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004402
4403#Example
4404#Image 3
Cary Clarkbad5ad72017-08-03 17:14:08 -04004405void draw(SkCanvas* canvas) {
4406 // sk_sp<SkImage> image = mandrill;
4407 SkRSXform xforms[] = { { 1, 0, 0, 0 }, {0, 1, 300, 100 } };
4408 SkRect tex[] = { { 0, 0, 200, 200 }, { 200, 0, 400, 200 } };
4409 canvas->drawAtlas(image, xforms, tex, 2, nullptr, nullptr);
Cary Clark8032b982017-07-28 11:04:54 -04004410}
4411##
4412
Cary Clark2ade9972017-11-02 17:49:34 -04004413#SeeAlso drawBitmap drawImage
Cary Clark8032b982017-07-28 11:04:54 -04004414
4415##
4416
4417# ------------------------------------------------------------------------------
4418
Cary Clark73fa9722017-08-29 17:36:51 -04004419#Method void drawDrawable(SkDrawable* drawable, const SkMatrix* matrix = nullptr)
Cary Clark78de7512018-02-07 07:27:09 -05004420#In Draw
Cary Clarkab2621d2018-01-30 10:08:57 -05004421#Line # draws Drawable, encapsulated drawing commands ##
Cary Clark09d80c02018-10-31 12:14:03 -04004422#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004423
4424#Example
4425#Height 100
4426#Function
Cary Clarkbad5ad72017-08-03 17:14:08 -04004427struct MyDrawable : public SkDrawable {
4428 SkRect onGetBounds() override { return SkRect::MakeWH(50, 100); }
4429
4430 void onDraw(SkCanvas* canvas) override {
4431 SkPath path;
4432 path.conicTo(10, 90, 50, 90, 0.9f);
4433 SkPaint paint;
4434 paint.setColor(SK_ColorBLUE);
4435 canvas->drawRect(path.getBounds(), paint);
4436 paint.setAntiAlias(true);
4437 paint.setColor(SK_ColorWHITE);
4438 canvas->drawPath(path, paint);
4439 }
4440};
4441
4442#Function ##
4443void draw(SkCanvas* canvas) {
4444 sk_sp<SkDrawable> drawable(new MyDrawable);
4445 SkMatrix matrix;
4446 matrix.setTranslate(10, 10);
4447 canvas->drawDrawable(drawable.get(), &matrix);
Cary Clark8032b982017-07-28 11:04:54 -04004448}
4449##
4450
Cary Clark2ade9972017-11-02 17:49:34 -04004451#SeeAlso SkDrawable drawPicture
Cary Clark8032b982017-07-28 11:04:54 -04004452
4453##
4454
4455# ------------------------------------------------------------------------------
4456
4457#Method void drawDrawable(SkDrawable* drawable, SkScalar x, SkScalar y)
Cary Clark09d80c02018-10-31 12:14:03 -04004458#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004459
4460#Example
4461#Height 100
4462#Function
Cary Clarkbad5ad72017-08-03 17:14:08 -04004463struct MyDrawable : public SkDrawable {
4464 SkRect onGetBounds() override { return SkRect::MakeWH(50, 100); }
4465
4466 void onDraw(SkCanvas* canvas) override {
4467 SkPath path;
4468 path.conicTo(10, 90, 50, 90, 0.9f);
4469 SkPaint paint;
4470 paint.setColor(SK_ColorBLUE);
4471 canvas->drawRect(path.getBounds(), paint);
4472 paint.setAntiAlias(true);
4473 paint.setColor(SK_ColorWHITE);
4474 canvas->drawPath(path, paint);
4475 }
4476};
4477
4478#Function ##
4479void draw(SkCanvas* canvas) {
4480 sk_sp<SkDrawable> drawable(new MyDrawable);
4481 canvas->drawDrawable(drawable.get(), 10, 10);
Cary Clark8032b982017-07-28 11:04:54 -04004482}
4483##
4484
Cary Clark2ade9972017-11-02 17:49:34 -04004485#SeeAlso SkDrawable drawPicture
Cary Clark8032b982017-07-28 11:04:54 -04004486
4487##
4488
4489# ------------------------------------------------------------------------------
4490
4491#Method void drawAnnotation(const SkRect& rect, const char key[], SkData* value)
Cary Clark78de7512018-02-07 07:27:09 -05004492#In Draw
4493#In Utility
Cary Clarkab2621d2018-01-30 10:08:57 -05004494#Line # associates a Rect with a key-value pair ##
Cary Clark09d80c02018-10-31 12:14:03 -04004495#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004496
4497#Example
4498 #Height 1
4499 const char text[] = "Click this link!";
4500 SkRect bounds;
4501 SkPaint paint;
Cary Clark26d173f2019-01-07 15:16:56 -05004502 SkFont font(nullptr, 40);
4503 (void)font.measureText(text, strlen(text), SkTextEncoding::kUTF8, &bounds);
Cary Clark8032b982017-07-28 11:04:54 -04004504 const char url[] = "https://www.google.com/";
4505 sk_sp<SkData> urlData(SkData::MakeWithCString(url));
4506 canvas->drawAnnotation(bounds, "url_key", urlData.get());
4507##
4508
Cary Clark2ade9972017-11-02 17:49:34 -04004509#SeeAlso SkPicture SkDocument
Cary Clark8032b982017-07-28 11:04:54 -04004510
4511##
4512
4513# ------------------------------------------------------------------------------
4514
Herb Derbyefe39bc2018-05-01 17:06:20 -04004515#Method void drawAnnotation(const SkRect& rect, const char key[], const sk_sp<SkData>& value)
Cary Clark09d80c02018-10-31 12:14:03 -04004516#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004517
4518#Example
4519#Height 1
4520 const char text[] = "Click this link!";
4521 SkRect bounds;
4522 SkPaint paint;
Cary Clark26d173f2019-01-07 15:16:56 -05004523 SkFont font(nullptr, 40);
4524 (void)font.measureText(text, strlen(text), SkTextEncoding::kUTF8, &bounds);
Cary Clark8032b982017-07-28 11:04:54 -04004525 const char url[] = "https://www.google.com/";
4526 sk_sp<SkData> urlData(SkData::MakeWithCString(url));
4527 canvas->drawAnnotation(bounds, "url_key", urlData.get());
4528##
4529
Cary Clark2ade9972017-11-02 17:49:34 -04004530#SeeAlso SkPicture SkDocument
Cary Clark8032b982017-07-28 11:04:54 -04004531
4532##
4533
Cary Clark8032b982017-07-28 11:04:54 -04004534# ------------------------------------------------------------------------------
4535
4536#Method virtual bool isClipEmpty() const
Cary Clark78de7512018-02-07 07:27:09 -05004537#In Property
Cary Clarkab2621d2018-01-30 10:08:57 -05004538#Line # returns if Clip is empty ##
Cary Clark09d80c02018-10-31 12:14:03 -04004539#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004540
4541#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04004542 void draw(SkCanvas* canvas) {
4543 SkDebugf("clip is%s empty\n", canvas->isClipEmpty() ? "" : " not");
4544 SkPath path;
4545 canvas->clipPath(path);
4546 SkDebugf("clip is%s empty\n", canvas->isClipEmpty() ? "" : " not");
Cary Clark8032b982017-07-28 11:04:54 -04004547 }
4548 #StdOut
Cary Clarkbad5ad72017-08-03 17:14:08 -04004549 clip is not empty
Cary Clark8032b982017-07-28 11:04:54 -04004550 clip is empty
4551 ##
4552##
4553
Cary Clark2ade9972017-11-02 17:49:34 -04004554#SeeAlso isClipRect getLocalClipBounds getDeviceClipBounds
Cary Clark8032b982017-07-28 11:04:54 -04004555
4556##
4557
4558# ------------------------------------------------------------------------------
4559
4560#Method virtual bool isClipRect() const
Cary Clark78de7512018-02-07 07:27:09 -05004561#In Property
Cary Clarkab2621d2018-01-30 10:08:57 -05004562#Line # returns if Clip is Rect and not empty ##
Cary Clark09d80c02018-10-31 12:14:03 -04004563#Populate
Cary Clark8032b982017-07-28 11:04:54 -04004564
4565#Example
Cary Clarkbad5ad72017-08-03 17:14:08 -04004566 void draw(SkCanvas* canvas) {
4567 SkDebugf("clip is%s rect\n", canvas->isClipRect() ? "" : " not");
4568 canvas->clipRect({0, 0, 0, 0});
4569 SkDebugf("clip is%s rect\n", canvas->isClipRect() ? "" : " not");
Cary Clark8032b982017-07-28 11:04:54 -04004570 }
4571 #StdOut
Cary Clarkbad5ad72017-08-03 17:14:08 -04004572 clip is rect
Cary Clark8032b982017-07-28 11:04:54 -04004573 clip is not rect
4574 ##
4575##
4576
Cary Clark2ade9972017-11-02 17:49:34 -04004577#SeeAlso isClipEmpty getLocalClipBounds getDeviceClipBounds
Cary Clark8032b982017-07-28 11:04:54 -04004578
4579##
4580
4581#Class SkCanvas ##
Cary Clark884dd7d2017-10-11 10:37:52 -04004582
Cary Clark8032b982017-07-28 11:04:54 -04004583#Topic Canvas ##