blob: 27683ef787eede99ee60b2e1f1b95d7af1cddf0f [file] [log] [blame]
reed@android.com8a1c16f2008-12-17 15:59:43 +00001/* libs/graphics/effects/SkLayerRasterizer.cpp
2**
3** Copyright 2006, The Android Open Source Project
4**
reed@google.com82065d62011-02-07 15:30:46 +00005** Licensed under the Apache License, Version 2.0 (the "License");
6** you may not use this file except in compliance with the License.
7** You may obtain a copy of the License at
reed@android.com8a1c16f2008-12-17 15:59:43 +00008**
reed@google.com82065d62011-02-07 15:30:46 +00009** http://www.apache.org/licenses/LICENSE-2.0
reed@android.com8a1c16f2008-12-17 15:59:43 +000010**
reed@google.com82065d62011-02-07 15:30:46 +000011** Unless required by applicable law or agreed to in writing, software
12** distributed under the License is distributed on an "AS IS" BASIS,
13** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14** See the License for the specific language governing permissions and
reed@android.com8a1c16f2008-12-17 15:59:43 +000015** limitations under the License.
16*/
17
18#include "SkLayerRasterizer.h"
19#include "SkBuffer.h"
20#include "SkDraw.h"
21#include "SkMask.h"
22#include "SkMaskFilter.h"
23#include "SkPaint.h"
24#include "SkPath.h"
25#include "SkRegion.h"
26#include "SkXfermode.h"
27#include <new>
28
29struct SkLayerRasterizer_Rec {
30 SkPaint fPaint;
31 SkVector fOffset;
32};
33
34SkLayerRasterizer::SkLayerRasterizer() : fLayers(sizeof(SkLayerRasterizer_Rec))
35{
36}
37
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +000038SkLayerRasterizer::~SkLayerRasterizer() {
reed@google.com4c09d5c2011-02-22 13:16:38 +000039 SkDeque::F2BIter iter(fLayers);
reed@android.com8a1c16f2008-12-17 15:59:43 +000040 SkLayerRasterizer_Rec* rec;
41
42 while ((rec = (SkLayerRasterizer_Rec*)iter.next()) != NULL)
43 rec->fPaint.~SkPaint();
44}
45
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +000046void SkLayerRasterizer::addLayer(const SkPaint& paint, SkScalar dx,
47 SkScalar dy) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000048 SkLayerRasterizer_Rec* rec = (SkLayerRasterizer_Rec*)fLayers.push_back();
49
50 new (&rec->fPaint) SkPaint(paint);
51 rec->fOffset.set(dx, dy);
52}
53
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +000054static bool compute_bounds(const SkDeque& layers, const SkPath& path,
55 const SkMatrix& matrix,
56 const SkIRect* clipBounds, SkIRect* bounds) {
reed@google.com4c09d5c2011-02-22 13:16:38 +000057 SkDeque::F2BIter iter(layers);
reed@android.com8a1c16f2008-12-17 15:59:43 +000058 SkLayerRasterizer_Rec* rec;
59
60 bounds->set(SK_MaxS32, SK_MaxS32, SK_MinS32, SK_MinS32);
reed@google.com82065d62011-02-07 15:30:46 +000061
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +000062 while ((rec = (SkLayerRasterizer_Rec*)iter.next()) != NULL) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000063 const SkPaint& paint = rec->fPaint;
64 SkPath fillPath, devPath;
65 const SkPath* p = &path;
66
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +000067 if (paint.getPathEffect() || paint.getStyle() != SkPaint::kFill_Style) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000068 paint.getFillPath(path, &fillPath);
69 p = &fillPath;
70 }
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +000071 if (p->isEmpty()) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000072 continue;
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +000073 }
reed@android.com8a1c16f2008-12-17 15:59:43 +000074
75 // apply the matrix and offset
76 {
77 SkMatrix m = matrix;
78 m.preTranslate(rec->fOffset.fX, rec->fOffset.fY);
79 p->transform(m, &devPath);
80 }
81
82 SkMask mask;
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +000083 if (!SkDraw::DrawToMask(devPath, clipBounds, paint.getMaskFilter(),
84 &matrix, &mask,
85 SkMask::kJustComputeBounds_CreateMode)) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000086 return false;
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +000087 }
reed@android.com8a1c16f2008-12-17 15:59:43 +000088
89 bounds->join(mask.fBounds);
90 }
91 return true;
92}
93
94bool SkLayerRasterizer::onRasterize(const SkPath& path, const SkMatrix& matrix,
95 const SkIRect* clipBounds,
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +000096 SkMask* mask, SkMask::CreateMode mode) {
97 if (fLayers.empty()) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000098 return false;
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +000099 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000100
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +0000101 if (SkMask::kJustRenderImage_CreateMode != mode) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000102 if (!compute_bounds(fLayers, path, matrix, clipBounds, &mask->fBounds))
103 return false;
104 }
105
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +0000106 if (SkMask::kComputeBoundsAndRenderImage_CreateMode == mode) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000107 mask->fFormat = SkMask::kA8_Format;
reed@android.com49f0ff22009-03-19 21:52:42 +0000108 mask->fRowBytes = mask->fBounds.width();
reed@android.com543ed932009-04-24 12:43:40 +0000109 size_t size = mask->computeImageSize();
110 if (0 == size) {
111 return false; // too big to allocate, abort
112 }
113 mask->fImage = SkMask::AllocImage(size);
114 memset(mask->fImage, 0, size);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000115 }
116
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +0000117 if (SkMask::kJustComputeBounds_CreateMode != mode) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000118 SkBitmap device;
119 SkDraw draw;
120 SkMatrix translatedMatrix; // this translates us to our local pixels
121 SkMatrix drawMatrix; // this translates the path by each layer's offset
122 SkRegion rectClip;
reed@google.com82065d62011-02-07 15:30:46 +0000123
reed@android.com8a1c16f2008-12-17 15:59:43 +0000124 rectClip.setRect(0, 0, mask->fBounds.width(), mask->fBounds.height());
125
126 translatedMatrix = matrix;
127 translatedMatrix.postTranslate(-SkIntToScalar(mask->fBounds.fLeft),
128 -SkIntToScalar(mask->fBounds.fTop));
129
130 device.setConfig(SkBitmap::kA8_Config, mask->fBounds.width(), mask->fBounds.height(), mask->fRowBytes);
131 device.setPixels(mask->fImage);
132
133 draw.fBitmap = &device;
134 draw.fMatrix = &drawMatrix;
135 draw.fClip = &rectClip;
136 // we set the matrixproc in the loop, as the matrix changes each time (potentially)
137 draw.fBounder = NULL;
reed@google.com82065d62011-02-07 15:30:46 +0000138
reed@google.com4c09d5c2011-02-22 13:16:38 +0000139 SkDeque::F2BIter iter(fLayers);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000140 SkLayerRasterizer_Rec* rec;
141
142 while ((rec = (SkLayerRasterizer_Rec*)iter.next()) != NULL) {
143 drawMatrix = translatedMatrix;
144 drawMatrix.preTranslate(rec->fOffset.fX, rec->fOffset.fY);
145 draw.drawPath(path, rec->fPaint);
146 }
147 }
148 return true;
149}
150
151/////////// Routines for flattening /////////////////
152
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +0000153static void paint_read(SkPaint* paint, SkFlattenableReadBuffer& buffer) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000154 paint->setAntiAlias(buffer.readBool());
155 paint->setStyle((SkPaint::Style)buffer.readU8());
156 paint->setAlpha(buffer.readU8());
reed@google.com82065d62011-02-07 15:30:46 +0000157
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +0000158 if (paint->getStyle() != SkPaint::kFill_Style) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000159 paint->setStrokeWidth(buffer.readScalar());
160 paint->setStrokeMiter(buffer.readScalar());
161 paint->setStrokeCap((SkPaint::Cap)buffer.readU8());
162 paint->setStrokeJoin((SkPaint::Join)buffer.readU8());
163 }
164
reed@google.com82065d62011-02-07 15:30:46 +0000165 SkSafeUnref(paint->setMaskFilter((SkMaskFilter*)buffer.readFlattenable()));
166 SkSafeUnref(paint->setPathEffect((SkPathEffect*)buffer.readFlattenable()));
167 SkSafeUnref(paint->setRasterizer((SkRasterizer*)buffer.readFlattenable()));
168 SkSafeUnref(paint->setXfermode((SkXfermode*)buffer.readFlattenable()));
reed@android.com8a1c16f2008-12-17 15:59:43 +0000169}
170
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +0000171static void paint_write(const SkPaint& paint, SkFlattenableWriteBuffer& buffer) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000172 buffer.writeBool(paint.isAntiAlias());
173 buffer.write8(paint.getStyle());
174 buffer.write8(paint.getAlpha());
reed@google.com82065d62011-02-07 15:30:46 +0000175
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +0000176 if (paint.getStyle() != SkPaint::kFill_Style) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000177 buffer.writeScalar(paint.getStrokeWidth());
178 buffer.writeScalar(paint.getStrokeMiter());
179 buffer.write8(paint.getStrokeCap());
180 buffer.write8(paint.getStrokeJoin());
181 }
reed@google.com82065d62011-02-07 15:30:46 +0000182
reed@android.com8a1c16f2008-12-17 15:59:43 +0000183 buffer.writeFlattenable(paint.getMaskFilter());
184 buffer.writeFlattenable(paint.getPathEffect());
185 buffer.writeFlattenable(paint.getRasterizer());
186 buffer.writeFlattenable(paint.getXfermode());
187}
188
189SkLayerRasterizer::SkLayerRasterizer(SkFlattenableReadBuffer& buffer)
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +0000190 : SkRasterizer(buffer), fLayers(sizeof(SkLayerRasterizer_Rec)) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000191 int count = buffer.readS32();
reed@google.com82065d62011-02-07 15:30:46 +0000192
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +0000193 for (int i = 0; i < count; i++) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000194 SkLayerRasterizer_Rec* rec = (SkLayerRasterizer_Rec*)fLayers.push_back();
reed@google.com82065d62011-02-07 15:30:46 +0000195
reed@android.com8a1c16f2008-12-17 15:59:43 +0000196#if 0
197 new (&rec->fPaint) SkPaint(buffer);
198#else
199 new (&rec->fPaint) SkPaint;
200 paint_read(&rec->fPaint, buffer);
201#endif
202 rec->fOffset.fX = buffer.readScalar();
203 rec->fOffset.fY = buffer.readScalar();
204 }
205}
206
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +0000207void SkLayerRasterizer::flatten(SkFlattenableWriteBuffer& buffer) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000208 this->INHERITED::flatten(buffer);
209
210 buffer.write32(fLayers.count());
211
reed@google.com4c09d5c2011-02-22 13:16:38 +0000212 SkDeque::F2BIter iter(fLayers);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000213 const SkLayerRasterizer_Rec* rec;
214
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +0000215 while ((rec = (const SkLayerRasterizer_Rec*)iter.next()) != NULL) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000216#if 0
217 rec->fPaint.flatten(buffer);
218#else
219 paint_write(rec->fPaint, buffer);
220#endif
221 buffer.writeScalar(rec->fOffset.fX);
222 buffer.writeScalar(rec->fOffset.fY);
223 }
224}
225
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +0000226SkFlattenable* SkLayerRasterizer::CreateProc(SkFlattenableReadBuffer& buffer) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000227 return SkNEW_ARGS(SkLayerRasterizer, (buffer));
228}
229
mike@reedtribe.org3334c3a2011-04-20 11:39:28 +0000230SkFlattenable::Factory SkLayerRasterizer::getFactory() {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000231 return CreateProc;
232}
233