blob: fa4488bb3ae82c54d5fdf6294b44671eba5ac736 [file] [log] [blame]
Mike Reedfae8fce2019-04-03 10:27:45 -04001/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00002 * Copyright 2011 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
reed8a21c9f2016-03-08 18:50:00 -08007
Ben Wagner2b4dcd32019-06-06 14:39:52 -04008#include "gm/gm.h"
9#include "include/core/SkBitmap.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050010#include "include/core/SkCanvas.h"
Ben Wagner2b4dcd32019-06-06 14:39:52 -040011#include "include/core/SkPixmap.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050012#include "include/core/SkRasterHandleAllocator.h"
Ben Wagner9f02d2d2019-08-14 15:38:43 -040013#include "include/core/SkSurface.h"
Mike Reed356f7c22017-01-10 11:58:39 -050014
15class GraphicsPort {
16protected:
17 SkCanvas* fCanvas;
18
19public:
20 GraphicsPort(SkCanvas* canvas) : fCanvas(canvas) {}
21 virtual ~GraphicsPort() {}
22
23 void save() { fCanvas->save(); }
24 void saveLayer(const SkRect& bounds, SkAlpha alpha) {
25 fCanvas->saveLayerAlpha(&bounds, alpha);
26 }
27 void restore() { fCanvas->restore(); }
28
29 void translate(float x, float y) { fCanvas->translate(x, y); }
30 void scale(float s) { fCanvas->scale(s, s); }
Mike Reed108f55e2017-01-12 11:28:01 -050031 void clip(const SkRect& r) { fCanvas->clipRect(r); }
Mike Reed356f7c22017-01-10 11:58:39 -050032
33 void drawOval(const SkRect& r, SkColor c) {
34 SkPaint p;
35 p.setColor(c);
36 fCanvas->drawOval(r, p);
37 }
38
39 virtual void drawRect(const SkRect& r, SkColor c) {
40 SkPaint p;
41 p.setColor(c);
42 fCanvas->drawRect(r, p);
43 }
Mike Reed108f55e2017-01-12 11:28:01 -050044
45 SkCanvas* peekCanvas() const { return fCanvas; }
Mike Reed356f7c22017-01-10 11:58:39 -050046};
47
Ben Wagner9f02d2d2019-08-14 15:38:43 -040048class SkiaGraphicsPort : public GraphicsPort {
49public:
50 SkiaGraphicsPort(SkCanvas* canvas) : GraphicsPort(canvas) {}
51
52 void drawRect(const SkRect& r, SkColor c) override {
53 SkCanvas* canvas = (SkCanvas*)fCanvas->accessTopRasterHandle();
54 canvas->drawRect(r, SkPaint(SkColor4f::FromColor(c)));
55 }
56};
57
58class SkiaAllocator : public SkRasterHandleAllocator {
59public:
60 SkiaAllocator() {}
61
62 bool allocHandle(const SkImageInfo& info, Rec* rec) override {
63 sk_sp<SkSurface> surface = SkSurface::MakeRaster(info);
64 if (!surface) {
65 return false;
66 }
67 SkCanvas* canvas = surface->getCanvas();
68 SkPixmap pixmap;
69 canvas->peekPixels(&pixmap);
70
71 rec->fReleaseProc = [](void* pixels, void* ctx){ SkSafeUnref((SkSurface*)ctx); };
72 rec->fReleaseCtx = surface.release();
73 rec->fPixels = pixmap.writable_addr();
74 rec->fRowBytes = pixmap.rowBytes();
75 rec->fHandle = canvas;
76 canvas->save(); // balanced each time updateHandle is called
77 return true;
78 }
79
80 void updateHandle(Handle hndl, const SkMatrix& ctm, const SkIRect& clip) override {
81 SkCanvas* canvas = (SkCanvas*)hndl;
82 canvas->restore();
83 canvas->save();
84 canvas->clipRect(SkRect::Make(clip));
85 canvas->concat(ctm);
86 }
87};
88
Mike Reede4ca3622017-01-11 21:18:25 -050089#ifdef SK_BUILD_FOR_MAC
90
Mike Kleinc0bd9f92019-04-23 12:05:21 -050091#include "include/utils/mac/SkCGUtils.h"
Mike Reed356f7c22017-01-10 11:58:39 -050092class CGGraphicsPort : public GraphicsPort {
93public:
94 CGGraphicsPort(SkCanvas* canvas) : GraphicsPort(canvas) {}
95
96 void drawRect(const SkRect& r, SkColor c) override {
97 CGContextRef cg = (CGContextRef)fCanvas->accessTopRasterHandle();
Ben Wagner63fd7602017-10-09 15:45:33 -040098
Mike Reed356f7c22017-01-10 11:58:39 -050099 CGColorRef color = CGColorCreateGenericRGB(SkColorGetR(c)/255.f,
100 SkColorGetG(c)/255.f,
101 SkColorGetB(c)/255.f,
102 SkColorGetA(c)/255.f);
103
104 CGContextSetFillColorWithColor(cg, color);
105 CGContextFillRect(cg, CGRectMake(r.x(), r.y(), r.width(), r.height()));
106 }
107};
108
109static CGAffineTransform matrix_to_transform(CGContextRef cg, const SkMatrix& ctm) {
110 SkMatrix matrix;
111 matrix.setScale(1, -1);
112 matrix.postTranslate(0, SkIntToScalar(CGBitmapContextGetHeight(cg)));
113 matrix.preConcat(ctm);
114
115 return CGAffineTransformMake(matrix[SkMatrix::kMScaleX],
116 matrix[SkMatrix::kMSkewY],
117 matrix[SkMatrix::kMSkewX],
118 matrix[SkMatrix::kMScaleY],
119 matrix[SkMatrix::kMTransX],
120 matrix[SkMatrix::kMTransY]);
121}
122
Ben Wagner9f02d2d2019-08-14 15:38:43 -0400123class CGAllocator : public SkRasterHandleAllocator {
Mike Reed356f7c22017-01-10 11:58:39 -0500124public:
Ben Wagner9f02d2d2019-08-14 15:38:43 -0400125 CGAllocator() {}
Ben Wagner63fd7602017-10-09 15:45:33 -0400126
Mike Reed356f7c22017-01-10 11:58:39 -0500127 bool allocHandle(const SkImageInfo& info, Rec* rec) override {
128 // let CG allocate the pixels
129 CGContextRef cg = SkCreateCGContext(SkPixmap(info, nullptr, 0));
130 if (!cg) {
131 return false;
132 }
133 rec->fReleaseProc = [](void* pixels, void* ctx){ CGContextRelease((CGContextRef)ctx); };
134 rec->fReleaseCtx = cg;
135 rec->fPixels = CGBitmapContextGetData(cg);
136 rec->fRowBytes = CGBitmapContextGetBytesPerRow(cg);
137 rec->fHandle = cg;
Ben Wagner9f02d2d2019-08-14 15:38:43 -0400138 CGContextSaveGState(cg); // balanced each time updateHandle is called
Mike Reed356f7c22017-01-10 11:58:39 -0500139 return true;
140 }
141
142 void updateHandle(Handle hndl, const SkMatrix& ctm, const SkIRect& clip) override {
143 CGContextRef cg = (CGContextRef)hndl;
Ben Wagner63fd7602017-10-09 15:45:33 -0400144
Mike Reed356f7c22017-01-10 11:58:39 -0500145 CGContextRestoreGState(cg);
146 CGContextSaveGState(cg);
Mike Reed108f55e2017-01-12 11:28:01 -0500147 CGContextClipToRect(cg, CGRectMake(clip.x(), clip.y(), clip.width(), clip.height()));
Mike Reed356f7c22017-01-10 11:58:39 -0500148 CGContextConcatCTM(cg, matrix_to_transform(cg, ctm));
149 }
150};
151
Ben Wagner9f02d2d2019-08-14 15:38:43 -0400152using MyPort = CGGraphicsPort;
153using MyAllocator = CGAllocator;
Mike Reede4ca3622017-01-11 21:18:25 -0500154
Ben Wagner2b4dcd32019-06-06 14:39:52 -0400155#elif defined(SK_BUILD_FOR_WIN)
156
157#include "src/core/SkLeanWindows.h"
Mike Reede4ca3622017-01-11 21:18:25 -0500158
Mike Reed9e937af2017-01-12 14:18:39 -0500159static RECT toRECT(const SkIRect& r) {
160 return { r.left(), r.top(), r.right(), r.bottom() };
161}
162
Mike Reede4ca3622017-01-11 21:18:25 -0500163class GDIGraphicsPort : public GraphicsPort {
164public:
165 GDIGraphicsPort(SkCanvas* canvas) : GraphicsPort(canvas) {}
166
167 void drawRect(const SkRect& r, SkColor c) override {
168 HDC hdc = (HDC)fCanvas->accessTopRasterHandle();
169
170 COLORREF cr = RGB(SkColorGetR(c), SkColorGetG(c), SkColorGetB(c));// SkEndian_Swap32(c) >> 8;
Mike Kleinc722f792017-07-31 11:57:21 -0400171 RECT rounded = toRECT(r.round());
172 FillRect(hdc, &rounded, CreateSolidBrush(cr));
Mike Reede4ca3622017-01-11 21:18:25 -0500173
174 // Assuming GDI wrote zeros for alpha, this will or-in 0xFF for alpha
175 SkPaint paint;
176 paint.setBlendMode(SkBlendMode::kDstATop);
177 fCanvas->drawRect(r, paint);
178 }
179};
180
Mike Reede4ca3622017-01-11 21:18:25 -0500181// We use this static factory function instead of the regular constructor so
182// that we can create the pixel data before calling the constructor. This is
183// required so that we can call the base class' constructor with the pixel
184// data.
185static bool Create(int width, int height, bool is_opaque, SkRasterHandleAllocator::Rec* rec) {
Chris Dalton1ef80942017-12-04 12:01:30 -0700186 BITMAPINFOHEADER hdr;
187 memset(&hdr, 0, sizeof(hdr));
Mike Reede4ca3622017-01-11 21:18:25 -0500188 hdr.biSize = sizeof(BITMAPINFOHEADER);
189 hdr.biWidth = width;
190 hdr.biHeight = -height; // Minus means top-down bitmap.
191 hdr.biPlanes = 1;
192 hdr.biBitCount = 32;
193 hdr.biCompression = BI_RGB; // No compression.
194 hdr.biSizeImage = 0;
195 hdr.biXPelsPerMeter = 1;
196 hdr.biYPelsPerMeter = 1;
197 void* pixels;
198 HBITMAP hbitmap = CreateDIBSection(nullptr, (const BITMAPINFO*)&hdr, 0, &pixels, 0, 0);
199 if (!hbitmap) {
200 return false;
201 }
202
203 size_t row_bytes = width * sizeof(SkPMColor);
204 sk_bzero(pixels, row_bytes * height);
205
206 HDC hdc = CreateCompatibleDC(nullptr);
207 if (!hdc) {
208 DeleteObject(hbitmap);
209 return false;
210 }
211 SetGraphicsMode(hdc, GM_ADVANCED);
Ben Wagner2b4dcd32019-06-06 14:39:52 -0400212 HGDIOBJ origBitmap = SelectObject(hdc, hbitmap);
Mike Reede4ca3622017-01-11 21:18:25 -0500213
Ben Wagner2b4dcd32019-06-06 14:39:52 -0400214 struct ReleaseContext {
215 HDC hdc;
216 HGDIOBJ hbitmap;
217 };
218 rec->fReleaseProc = [](void*, void* context) {
219 ReleaseContext* ctx = static_cast<ReleaseContext*>(context);
220 HBITMAP hbitmap = static_cast<HBITMAP>(SelectObject(ctx->hdc, ctx->hbitmap));
221 DeleteObject(hbitmap);
222 DeleteDC(ctx->hdc);
223 delete ctx;
224 };
225 rec->fReleaseCtx = new ReleaseContext{hdc, origBitmap};
Mike Reede4ca3622017-01-11 21:18:25 -0500226 rec->fPixels = pixels;
227 rec->fRowBytes = row_bytes;
228 rec->fHandle = hdc;
229 return true;
230}
231
232/**
233* Subclass of SkRasterHandleAllocator that returns an HDC as its "handle".
234*/
235class GDIAllocator : public SkRasterHandleAllocator {
236public:
237 GDIAllocator() {}
238
239 bool allocHandle(const SkImageInfo& info, Rec* rec) override {
240 SkASSERT(info.colorType() == kN32_SkColorType);
241 return Create(info.width(), info.height(), info.isOpaque(), rec);
242 }
243
244 void updateHandle(Handle handle, const SkMatrix& ctm, const SkIRect& clip_bounds) override {
245 HDC hdc = static_cast<HDC>(handle);
246
247 XFORM xf;
248 xf.eM11 = ctm[SkMatrix::kMScaleX];
249 xf.eM21 = ctm[SkMatrix::kMSkewX];
250 xf.eDx = ctm[SkMatrix::kMTransX];
251 xf.eM12 = ctm[SkMatrix::kMSkewY];
252 xf.eM22 = ctm[SkMatrix::kMScaleY];
253 xf.eDy = ctm[SkMatrix::kMTransY];
254 SetWorldTransform(hdc, &xf);
255
Mike Kleinc722f792017-07-31 11:57:21 -0400256 RECT clip_bounds_RECT = toRECT(clip_bounds);
257 HRGN hrgn = CreateRectRgnIndirect(&clip_bounds_RECT);
Brian Salomon54fd96b2021-08-05 12:16:07 -0400258 SK_MAYBE_UNUSED int result = SelectClipRgn(hdc, hrgn);
Mike Reede4ca3622017-01-11 21:18:25 -0500259 SkASSERT(result != ERROR);
260 result = DeleteObject(hrgn);
261 SkASSERT(result != 0);
Mike Reede4ca3622017-01-11 21:18:25 -0500262 }
263};
264
Ben Wagner9f02d2d2019-08-14 15:38:43 -0400265using MyPort = GDIGraphicsPort;
266using MyAllocator = GDIAllocator;
267
268#else
269
270using MyPort = SkiaGraphicsPort;
271using MyAllocator = SkiaAllocator;
Mike Reede4ca3622017-01-11 21:18:25 -0500272
273#endif
274
Ben Wagner2b4dcd32019-06-06 14:39:52 -0400275DEF_SIMPLE_GM(rasterallocator, canvas, 600, 300) {
276 auto doDraw = [](GraphicsPort* port) {
Mike Reed108f55e2017-01-12 11:28:01 -0500277 SkAutoCanvasRestore acr(port->peekCanvas(), true);
278
Mike Reed356f7c22017-01-10 11:58:39 -0500279 port->drawRect({0, 0, 256, 256}, SK_ColorRED);
280 port->save();
281 port->translate(30, 30);
282 port->drawRect({0, 0, 30, 30}, SK_ColorBLUE);
283 port->drawOval({10, 10, 20, 20}, SK_ColorWHITE);
284 port->restore();
Ben Wagner63fd7602017-10-09 15:45:33 -0400285
Mike Reed356f7c22017-01-10 11:58:39 -0500286 port->saveLayer({50, 50, 100, 100}, 0x80);
287 port->drawRect({55, 55, 95, 95}, SK_ColorGREEN);
288 port->restore();
Mike Reed108f55e2017-01-12 11:28:01 -0500289
290 port->clip({150, 50, 200, 200});
291 port->drawRect({0, 0, 256, 256}, 0xFFCCCCCC);
Ben Wagner2b4dcd32019-06-06 14:39:52 -0400292 };
Mike Reed356f7c22017-01-10 11:58:39 -0500293
Ben Wagner53ae9362019-06-07 14:16:54 -0400294 // TODO: this common code fails pic-8888 and serialize-8888
Ben Wagner53ae9362019-06-07 14:16:54 -0400295 //GraphicsPort skiaPort(canvas);
296 //doDraw(&skiaPort);
Mike Reed356f7c22017-01-10 11:58:39 -0500297
Ben Wagner2b4dcd32019-06-06 14:39:52 -0400298 const SkImageInfo info = SkImageInfo::MakeN32Premul(256, 256);
Ben Wagner53ae9362019-06-07 14:16:54 -0400299 std::unique_ptr<SkCanvas> nativeCanvas =
Mike Kleinf46d5ca2019-12-11 10:45:01 -0500300 SkRasterHandleAllocator::MakeCanvas(std::make_unique<MyAllocator>(), info);
Ben Wagner53ae9362019-06-07 14:16:54 -0400301 MyPort nativePort(nativeCanvas.get());
302 doDraw(&nativePort);
Mike Reed356f7c22017-01-10 11:58:39 -0500303
Ben Wagner2b4dcd32019-06-06 14:39:52 -0400304 SkPixmap pm;
Ben Wagner53ae9362019-06-07 14:16:54 -0400305 nativeCanvas->peekPixels(&pm);
Mike Reed34a0c972021-01-25 17:49:32 -0500306 canvas->drawImage(SkImage::MakeRasterCopy(pm), 280, 0);
Ben Wagner53ae9362019-06-07 14:16:54 -0400307}