blob: 8bc1f59d053a067f25361ef2bb738b243a942bdd [file] [log] [blame]
Mike Reede4ca3622017-01-11 21:18:25 -05001/*
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
reed@google.com55b8e8c2011-01-13 16:22:35 +00008#include "SampleCode.h"
9#include "SkView.h"
10#include "SkCanvas.h"
11#include "SkGradientShader.h"
Mike Reed356f7c22017-01-10 11:58:39 -050012#include "SkMakeUnique.h"
reed@google.com55b8e8c2011-01-13 16:22:35 +000013
reed8a21c9f2016-03-08 18:50:00 -080014static sk_sp<SkShader> make_grad(SkScalar w, SkScalar h) {
reed@google.com55b8e8c2011-01-13 16:22:35 +000015 SkColor colors[] = { 0xFF000000, 0xFF333333 };
16 SkPoint pts[] = { { 0, 0 }, { w, h } };
reed8a21c9f2016-03-08 18:50:00 -080017 return SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkShader::kClamp_TileMode);
reed@google.com55b8e8c2011-01-13 16:22:35 +000018}
19
reed@google.com0faac1e2011-05-11 05:58:58 +000020class BigGradientView : public SampleView {
reed@google.com55b8e8c2011-01-13 16:22:35 +000021public:
rmistry@google.comae933ce2012-08-23 18:19:56 +000022 BigGradientView() {}
reed@google.com55b8e8c2011-01-13 16:22:35 +000023
24protected:
reed8a21c9f2016-03-08 18:50:00 -080025 bool onQuery(SkEvent* evt) override {
reed@google.com55b8e8c2011-01-13 16:22:35 +000026 if (SampleCode::TitleQ(*evt)) {
27 SampleCode::TitleR(evt, "BigGradient");
28 return true;
29 }
30 return this->INHERITED::onQuery(evt);
31 }
32
reed8a21c9f2016-03-08 18:50:00 -080033 void onDrawContent(SkCanvas* canvas) override {
reed@google.com55b8e8c2011-01-13 16:22:35 +000034 SkRect r;
35 r.set(0, 0, this->width(), this->height());
36 SkPaint p;
reed8a21c9f2016-03-08 18:50:00 -080037 p.setShader(make_grad(this->width(), this->height()));
reed@google.com55b8e8c2011-01-13 16:22:35 +000038 canvas->drawRect(r, p);
39 }
40
41private:
reed@google.com0faac1e2011-05-11 05:58:58 +000042 typedef SampleView INHERITED;
reed@google.com55b8e8c2011-01-13 16:22:35 +000043};
44
45///////////////////////////////////////////////////////////////////////////////
46
47static SkView* MyFactory() { return new BigGradientView; }
48static SkViewRegister reg(MyFactory);
Mike Reed356f7c22017-01-10 11:58:39 -050049
50///////////////////////////////////////////////////////////////////////////////
51
Mike Reed356f7c22017-01-10 11:58:39 -050052#include "SkRasterHandleAllocator.h"
53
54class GraphicsPort {
55protected:
56 SkCanvas* fCanvas;
57
58public:
59 GraphicsPort(SkCanvas* canvas) : fCanvas(canvas) {}
60 virtual ~GraphicsPort() {}
61
62 void save() { fCanvas->save(); }
63 void saveLayer(const SkRect& bounds, SkAlpha alpha) {
64 fCanvas->saveLayerAlpha(&bounds, alpha);
65 }
66 void restore() { fCanvas->restore(); }
67
68 void translate(float x, float y) { fCanvas->translate(x, y); }
69 void scale(float s) { fCanvas->scale(s, s); }
70
71 void drawOval(const SkRect& r, SkColor c) {
72 SkPaint p;
73 p.setColor(c);
74 fCanvas->drawOval(r, p);
75 }
76
77 virtual void drawRect(const SkRect& r, SkColor c) {
78 SkPaint p;
79 p.setColor(c);
80 fCanvas->drawRect(r, p);
81 }
82};
83
Mike Reede4ca3622017-01-11 21:18:25 -050084#ifdef SK_BUILD_FOR_MAC
85
86#include "SkCGUtils.h"
Mike Reed356f7c22017-01-10 11:58:39 -050087class CGGraphicsPort : public GraphicsPort {
88public:
89 CGGraphicsPort(SkCanvas* canvas) : GraphicsPort(canvas) {}
90
91 void drawRect(const SkRect& r, SkColor c) override {
92 CGContextRef cg = (CGContextRef)fCanvas->accessTopRasterHandle();
93
94 CGColorRef color = CGColorCreateGenericRGB(SkColorGetR(c)/255.f,
95 SkColorGetG(c)/255.f,
96 SkColorGetB(c)/255.f,
97 SkColorGetA(c)/255.f);
98
99 CGContextSetFillColorWithColor(cg, color);
100 CGContextFillRect(cg, CGRectMake(r.x(), r.y(), r.width(), r.height()));
101 }
102};
103
104static CGAffineTransform matrix_to_transform(CGContextRef cg, const SkMatrix& ctm) {
105 SkMatrix matrix;
106 matrix.setScale(1, -1);
107 matrix.postTranslate(0, SkIntToScalar(CGBitmapContextGetHeight(cg)));
108 matrix.preConcat(ctm);
109
110 return CGAffineTransformMake(matrix[SkMatrix::kMScaleX],
111 matrix[SkMatrix::kMSkewY],
112 matrix[SkMatrix::kMSkewX],
113 matrix[SkMatrix::kMScaleY],
114 matrix[SkMatrix::kMTransX],
115 matrix[SkMatrix::kMTransY]);
116}
117
118class Allocator_CG : public SkRasterHandleAllocator {
119public:
120 Allocator_CG() {}
121
122 bool allocHandle(const SkImageInfo& info, Rec* rec) override {
123 // let CG allocate the pixels
124 CGContextRef cg = SkCreateCGContext(SkPixmap(info, nullptr, 0));
125 if (!cg) {
126 return false;
127 }
128 rec->fReleaseProc = [](void* pixels, void* ctx){ CGContextRelease((CGContextRef)ctx); };
129 rec->fReleaseCtx = cg;
130 rec->fPixels = CGBitmapContextGetData(cg);
131 rec->fRowBytes = CGBitmapContextGetBytesPerRow(cg);
132 rec->fHandle = cg;
133 CGContextSaveGState(cg); // balanced each time updateContext is called
134 return true;
135 }
136
137 void updateHandle(Handle hndl, const SkMatrix& ctm, const SkIRect& clip) override {
138 CGContextRef cg = (CGContextRef)hndl;
139
140 CGContextRestoreGState(cg);
141 CGContextSaveGState(cg);
142 CGContextClearRect(cg, CGRectMake(clip.x(), clip.y(), clip.width(), clip.height()));
143 CGContextConcatCTM(cg, matrix_to_transform(cg, ctm));
144 }
145};
146
Mike Reede4ca3622017-01-11 21:18:25 -0500147#define MyPort CGGraphicsPort
148#define MyAllocator Allocator_CG
149
150#elif defined(WIN32)
151
152class GDIGraphicsPort : public GraphicsPort {
153public:
154 GDIGraphicsPort(SkCanvas* canvas) : GraphicsPort(canvas) {}
155
156 void drawRect(const SkRect& r, SkColor c) override {
157 HDC hdc = (HDC)fCanvas->accessTopRasterHandle();
158
159 COLORREF cr = RGB(SkColorGetR(c), SkColorGetG(c), SkColorGetB(c));// SkEndian_Swap32(c) >> 8;
160 SkIRect ir = r.round();
161 RECT rect = { ir.left(), ir.top(), ir.right(), ir.bottom() };
162 FillRect(hdc, &rect, CreateSolidBrush(cr));
163
164 // Assuming GDI wrote zeros for alpha, this will or-in 0xFF for alpha
165 SkPaint paint;
166 paint.setBlendMode(SkBlendMode::kDstATop);
167 fCanvas->drawRect(r, paint);
168 }
169};
170
171static void DeleteHDCCallback(void*, void* context) {
172 HDC hdc = static_cast<HDC>(context);
173 HBITMAP hbitmap = static_cast<HBITMAP>(SelectObject(hdc, nullptr));
174 DeleteObject(hbitmap);
175 DeleteDC(hdc);
176}
177
178// We use this static factory function instead of the regular constructor so
179// that we can create the pixel data before calling the constructor. This is
180// required so that we can call the base class' constructor with the pixel
181// data.
182static bool Create(int width, int height, bool is_opaque, SkRasterHandleAllocator::Rec* rec) {
183 BITMAPINFOHEADER hdr = { 0 };
184 hdr.biSize = sizeof(BITMAPINFOHEADER);
185 hdr.biWidth = width;
186 hdr.biHeight = -height; // Minus means top-down bitmap.
187 hdr.biPlanes = 1;
188 hdr.biBitCount = 32;
189 hdr.biCompression = BI_RGB; // No compression.
190 hdr.biSizeImage = 0;
191 hdr.biXPelsPerMeter = 1;
192 hdr.biYPelsPerMeter = 1;
193 void* pixels;
194 HBITMAP hbitmap = CreateDIBSection(nullptr, (const BITMAPINFO*)&hdr, 0, &pixels, 0, 0);
195 if (!hbitmap) {
196 return false;
197 }
198
199 size_t row_bytes = width * sizeof(SkPMColor);
200 sk_bzero(pixels, row_bytes * height);
201
202 HDC hdc = CreateCompatibleDC(nullptr);
203 if (!hdc) {
204 DeleteObject(hbitmap);
205 return false;
206 }
207 SetGraphicsMode(hdc, GM_ADVANCED);
208 SelectObject(hdc, hbitmap);
209
210 rec->fReleaseProc = DeleteHDCCallback;
211 rec->fReleaseCtx = hdc;
212 rec->fPixels = pixels;
213 rec->fRowBytes = row_bytes;
214 rec->fHandle = hdc;
215 return true;
216}
217
218/**
219* Subclass of SkRasterHandleAllocator that returns an HDC as its "handle".
220*/
221class GDIAllocator : public SkRasterHandleAllocator {
222public:
223 GDIAllocator() {}
224
225 bool allocHandle(const SkImageInfo& info, Rec* rec) override {
226 SkASSERT(info.colorType() == kN32_SkColorType);
227 return Create(info.width(), info.height(), info.isOpaque(), rec);
228 }
229
230 void updateHandle(Handle handle, const SkMatrix& ctm, const SkIRect& clip_bounds) override {
231 HDC hdc = static_cast<HDC>(handle);
232
233 XFORM xf;
234 xf.eM11 = ctm[SkMatrix::kMScaleX];
235 xf.eM21 = ctm[SkMatrix::kMSkewX];
236 xf.eDx = ctm[SkMatrix::kMTransX];
237 xf.eM12 = ctm[SkMatrix::kMSkewY];
238 xf.eM22 = ctm[SkMatrix::kMScaleY];
239 xf.eDy = ctm[SkMatrix::kMTransY];
240 SetWorldTransform(hdc, &xf);
241
242#if 0
243 HRGN hrgn = CreateRectRgnIndirect(&skia::SkIRectToRECT(clip_bounds));
244 int result = SelectClipRgn(hdc, hrgn);
245 SkASSERT(result != ERROR);
246 result = DeleteObject(hrgn);
247 SkASSERT(result != 0);
248#endif
249 }
250};
251
252#define MyPort GDIGraphicsPort
253#define MyAllocator GDIAllocator
254
255#endif
256
257#ifdef MyAllocator
Mike Reed356f7c22017-01-10 11:58:39 -0500258class RasterAllocatorSample : public SampleView {
259public:
260 RasterAllocatorSample() {}
261
262protected:
263 bool onQuery(SkEvent* evt) override {
264 if (SampleCode::TitleQ(*evt)) {
265 SampleCode::TitleR(evt, "raster-allocator");
266 return true;
267 }
268 return this->INHERITED::onQuery(evt);
269 }
270
271 void doDraw(GraphicsPort* port) {
272 port->drawRect({0, 0, 256, 256}, SK_ColorRED);
273 port->save();
274 port->translate(30, 30);
275 port->drawRect({0, 0, 30, 30}, SK_ColorBLUE);
276 port->drawOval({10, 10, 20, 20}, SK_ColorWHITE);
277 port->restore();
278
279 port->saveLayer({50, 50, 100, 100}, 0x80);
280 port->drawRect({55, 55, 95, 95}, SK_ColorGREEN);
281 port->restore();
282 }
283
284 void onDrawContent(SkCanvas* canvas) override {
285 GraphicsPort skp(canvas);
286 doDraw(&skp);
287
288 const SkImageInfo info = SkImageInfo::MakeN32Premul(256, 256);
289 std::unique_ptr<SkCanvas> c2 =
Mike Reede4ca3622017-01-11 21:18:25 -0500290 SkRasterHandleAllocator::MakeCanvas(skstd::make_unique<MyAllocator>(), info);
291 MyPort cgp(c2.get());
Mike Reed356f7c22017-01-10 11:58:39 -0500292 doDraw(&cgp);
293
294 SkPixmap pm;
295 c2->peekPixels(&pm);
296 SkBitmap bm;
297 bm.installPixels(pm);
298 canvas->drawBitmap(bm, 280, 0, nullptr);
299 }
300
301private:
302 typedef SampleView INHERITED;
303};
304DEF_SAMPLE( return new RasterAllocatorSample; )
305#endif