blob: a98b176a376674f04d787062635931d2a20d26ed [file] [log] [blame]
reed71c3c762015-06-24 10:29:17 -07001/*
2 * Copyright 2015 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 */
7
8#include "gm.h"
9#include "SkCanvas.h"
10#include "SkRSXform.h"
11#include "SkSurface.h"
12
13class DrawAtlasGM : public skiagm::GM {
reed9ce9d672016-03-17 10:51:11 -070014 static sk_sp<SkImage> MakeAtlas(SkCanvas* caller, const SkRect& target) {
reed71c3c762015-06-24 10:29:17 -070015 SkImageInfo info = SkImageInfo::MakeN32Premul(100, 100);
reede8f30622016-03-23 18:59:25 -070016 auto surface(caller->makeSurface(info));
halcanary96fcdcc2015-08-27 07:41:13 -070017 if (nullptr == surface) {
reede8f30622016-03-23 18:59:25 -070018 surface = SkSurface::MakeRaster(info);
reed71c3c762015-06-24 10:29:17 -070019 }
20 SkCanvas* canvas = surface->getCanvas();
21 // draw red everywhere, but we don't expect to see it in the draw, testing the notion
22 // that drawAtlas draws a subset-region of the atlas.
23 canvas->clear(SK_ColorRED);
24
25 SkPaint paint;
26 paint.setXfermodeMode(SkXfermode::kClear_Mode);
27 SkRect r(target);
28 r.inset(-1, -1);
29 // zero out a place (with a 1-pixel border) to land our drawing.
30 canvas->drawRect(r, paint);
halcanary96fcdcc2015-08-27 07:41:13 -070031 paint.setXfermode(nullptr);
reed71c3c762015-06-24 10:29:17 -070032 paint.setColor(SK_ColorBLUE);
33 paint.setAntiAlias(true);
34 canvas->drawOval(target, paint);
reed9ce9d672016-03-17 10:51:11 -070035 return surface->makeImageSnapshot();
reed71c3c762015-06-24 10:29:17 -070036 }
37
reed9ce9d672016-03-17 10:51:11 -070038 sk_sp<SkImage> fAtlas;
reed71c3c762015-06-24 10:29:17 -070039
40public:
41 DrawAtlasGM() {}
halcanary9d524f22016-03-29 09:03:52 -070042
reed71c3c762015-06-24 10:29:17 -070043protected:
halcanary9d524f22016-03-29 09:03:52 -070044
reed71c3c762015-06-24 10:29:17 -070045 SkString onShortName() override {
46 return SkString("draw-atlas");
47 }
halcanary9d524f22016-03-29 09:03:52 -070048
reed71c3c762015-06-24 10:29:17 -070049 SkISize onISize() override {
50 return SkISize::Make(640, 480);
51 }
halcanary9d524f22016-03-29 09:03:52 -070052
reed71c3c762015-06-24 10:29:17 -070053 void onDraw(SkCanvas* canvas) override {
54 const SkRect target = { 50, 50, 80, 90 };
halcanary96fcdcc2015-08-27 07:41:13 -070055 if (nullptr == fAtlas) {
reed9ce9d672016-03-17 10:51:11 -070056 fAtlas = MakeAtlas(canvas, target);
reed71c3c762015-06-24 10:29:17 -070057 }
58
59 const struct {
60 SkScalar fScale;
61 SkScalar fDegrees;
62 SkScalar fTx;
63 SkScalar fTy;
halcanary9d524f22016-03-29 09:03:52 -070064
reed71c3c762015-06-24 10:29:17 -070065 void apply(SkRSXform* xform) const {
66 const SkScalar rad = SkDegreesToRadians(fDegrees);
67 xform->fSCos = fScale * SkScalarCos(rad);
68 xform->fSSin = fScale * SkScalarSin(rad);
69 xform->fTx = fTx;
70 xform->fTy = fTy;
71 }
72 } rec[] = {
73 { 1, 0, 10, 10 }, // just translate
74 { 2, 0, 110, 10 }, // scale + translate
75 { 1, 30, 210, 10 }, // rotate + translate
76 { 2, -30, 310, 30 }, // scale + rotate + translate
77 };
78
79 const int N = SK_ARRAY_COUNT(rec);
80 SkRSXform xform[N];
81 SkRect tex[N];
82 SkColor colors[N];
83
84 for (int i = 0; i < N; ++i) {
85 rec[i].apply(&xform[i]);
86 tex[i] = target;
87 colors[i] = 0x80FF0000 + (i * 40 * 256);
88 }
89
90 SkPaint paint;
91 paint.setFilterQuality(kLow_SkFilterQuality);
92 paint.setAntiAlias(true);
93
reed9ce9d672016-03-17 10:51:11 -070094 canvas->drawAtlas(fAtlas.get(), xform, tex, N, nullptr, &paint);
reed71c3c762015-06-24 10:29:17 -070095 canvas->translate(0, 100);
reed9ce9d672016-03-17 10:51:11 -070096 canvas->drawAtlas(fAtlas.get(), xform, tex, colors, N, SkXfermode::kSrcIn_Mode, nullptr, &paint);
reed71c3c762015-06-24 10:29:17 -070097 }
halcanary9d524f22016-03-29 09:03:52 -070098
reed71c3c762015-06-24 10:29:17 -070099private:
100 typedef GM INHERITED;
101};
102DEF_GM( return new DrawAtlasGM; )
reed45561a02016-07-07 12:47:17 -0700103
104///////////////////////////////////////////////////////////////////////////////////////////////////
105#include "SkPath.h"
106#include "SkPathMeasure.h"
107
108static void draw_text_on_path_rigid(SkCanvas* canvas, const void* text, size_t length,
109 const SkPoint xy[], const SkPath& path, const SkPaint& paint) {
110 SkPathMeasure meas(path, false);
111
112 int count = paint.countText(text, length);
reed7c70d7c2016-07-12 15:06:33 -0700113 size_t size = count * (sizeof(SkRSXform) + sizeof(SkScalar));
114 SkAutoSMalloc<512> storage(size);
115 SkRSXform* xform = (SkRSXform*)storage.get();
116 SkScalar* widths = (SkScalar*)(xform + count);
117
118 paint.getTextWidths(text, length, widths);
reed45561a02016-07-07 12:47:17 -0700119
120 for (int i = 0; i < count; ++i) {
reed7c70d7c2016-07-12 15:06:33 -0700121 // we want to position each character on the center of its advance
122 const SkScalar offset = SkScalarHalf(widths[i]);
reed45561a02016-07-07 12:47:17 -0700123 SkPoint pos;
124 SkVector tan;
reed7c70d7c2016-07-12 15:06:33 -0700125 if (!meas.getPosTan(xy[i].x() + offset, &pos, &tan)) {
reed45561a02016-07-07 12:47:17 -0700126 pos = xy[i];
127 tan.set(1, 0);
128 }
129 xform[i].fSCos = tan.x();
130 xform[i].fSSin = tan.y();
reed7c70d7c2016-07-12 15:06:33 -0700131 xform[i].fTx = pos.x() - tan.y() * xy[i].y() - tan.x() * offset;
132 xform[i].fTy = pos.y() + tan.x() * xy[i].y() - tan.y() * offset;
reed45561a02016-07-07 12:47:17 -0700133 }
134
reed7c70d7c2016-07-12 15:06:33 -0700135 // Compute a conservative bounds so we can cull the draw
136 const SkRect font = paint.getFontBounds();
137 const SkScalar max = SkTMax(SkTMax(SkScalarAbs(font.fLeft), SkScalarAbs(font.fRight)),
138 SkTMax(SkScalarAbs(font.fTop), SkScalarAbs(font.fBottom)));
139 const SkRect bounds = path.getBounds().makeOutset(max, max);
140
141 canvas->drawTextRSXform(text, length, &xform[0], &bounds, paint);
142
143 if (true) {
144 SkPaint p;
145 p.setStyle(SkPaint::kStroke_Style);
146 canvas->drawRect(bounds, p);
147 }
reed45561a02016-07-07 12:47:17 -0700148}
149
reed7c70d7c2016-07-12 15:06:33 -0700150DEF_SIMPLE_GM(drawTextRSXform, canvas, 860, 860) {
reeddde2b1f2016-07-08 03:31:09 -0700151 const char text0[] = "ABCDFGHJKLMNOPQRSTUVWXYZ";
reeddde2b1f2016-07-08 03:31:09 -0700152 const int N = sizeof(text0) - 1;
reed45561a02016-07-07 12:47:17 -0700153 SkPoint pos[N];
reed45561a02016-07-07 12:47:17 -0700154
155 SkPaint paint;
156 paint.setAntiAlias(true);
reed7c70d7c2016-07-12 15:06:33 -0700157 paint.setTextSize(100);
158
159 SkScalar x = 0;
160 for (int i = 0; i < N; ++i) {
161 pos[i].set(x, 0);
162 x += paint.measureText(&text0[i], 1);
163 }
reed45561a02016-07-07 12:47:17 -0700164
165 SkPath path;
reed7c70d7c2016-07-12 15:06:33 -0700166 path.addOval(SkRect::MakeXYWH(160, 160, 540, 540));
reed45561a02016-07-07 12:47:17 -0700167
reed7c70d7c2016-07-12 15:06:33 -0700168 draw_text_on_path_rigid(canvas, text0, N, pos, path, paint);
reed45561a02016-07-07 12:47:17 -0700169
170 paint.setStyle(SkPaint::kStroke_Style);
171 canvas->drawPath(path, paint);
172}
173
174