1 /*
2 * Copyright 2006 The Android Open Source Project
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 "SkDrawPaint.h"
9 #include "SkAnimateMaker.h"
10 #include "SkDrawColor.h"
11 #include "SkDrawShader.h"
12 #include "SkMaskFilter.h"
13 #include "SkPaintPart.h"
14 #include "SkPathEffect.h"
15
16 enum SkPaint_Functions {
17 SK_FUNCTION(measureText)
18 };
19
20 enum SkPaint_Properties {
21 SK_PROPERTY(ascent),
22 SK_PROPERTY(descent)
23 };
24
25 // !!! in the future, this could be compiled by build-condensed-info into an array of parameters
26 // with a lookup table to find the first parameter -- for now, it is iteratively searched through
27 const SkFunctionParamType SkDrawPaint::fFunctionParameters[] = {
28 (SkFunctionParamType) SkType_String,
29 (SkFunctionParamType) 0 // terminator for parameter list (there may be multiple parameter lists)
30 };
31
32
33 #if SK_USE_CONDENSED_INFO == 0
34
35 const SkMemberInfo SkDrawPaint::fInfo[] = {
36 SK_MEMBER(antiAlias, Boolean),
37 SK_MEMBER_PROPERTY(ascent, Float),
38 SK_MEMBER(color, Color),
39 SK_MEMBER_PROPERTY(descent, Float),
40 SK_MEMBER(fakeBold, Boolean),
41 SK_MEMBER(filterBitmap, Boolean),
42 SK_MEMBER(linearText, Boolean),
43 SK_MEMBER(maskFilter, MaskFilter),
44 SK_MEMBER_FUNCTION(measureText, Float),
45 SK_MEMBER(pathEffect, PathEffect),
46 SK_MEMBER(shader, Shader),
47 SK_MEMBER(strikeThru, Boolean),
48 SK_MEMBER(stroke, Boolean),
49 SK_MEMBER(strokeCap, Cap),
50 SK_MEMBER(strokeJoin, Join),
51 SK_MEMBER(strokeMiter, Float),
52 SK_MEMBER(strokeWidth, Float),
53 SK_MEMBER(style, Style),
54 SK_MEMBER(textAlign, Align),
55 SK_MEMBER(textScaleX, Float),
56 SK_MEMBER(textSize, Float),
57 SK_MEMBER(textSkewX, Float),
58 SK_MEMBER(typeface, Typeface),
59 SK_MEMBER(underline, Boolean),
60 SK_MEMBER(xfermode, Xfermode)
61 };
62
63 #endif
64
65 DEFINE_GET_MEMBER(SkDrawPaint);
66
SkDrawPaint()67 SkDrawPaint::SkDrawPaint() : antiAlias(-1), color(nullptr), fakeBold(-1), filterBitmap(-1),
68 linearText(-1), maskFilter((SkDrawMaskFilter*) -1), pathEffect((SkDrawPathEffect*) -1),
69 shader((SkDrawShader*) -1), strikeThru(-1), stroke(-1),
70 strokeCap((SkPaint::Cap) -1), strokeJoin((SkPaint::Join) -1), strokeMiter(SK_ScalarNaN),
71 strokeWidth(SK_ScalarNaN), style((SkPaint::Style) -1),
72 textAlign((SkPaint::Align) -1), textScaleX(SK_ScalarNaN), textSize(SK_ScalarNaN),
73 textSkewX(SK_ScalarNaN), typeface((SkDrawTypeface*) -1),
74 underline(-1), xfermode((SkXfermode::Mode) -1), fOwnsColor(false), fOwnsMaskFilter(false),
75 fOwnsPathEffect(false), fOwnsShader(false), fOwnsTypeface(false) {
76 }
77
~SkDrawPaint()78 SkDrawPaint::~SkDrawPaint() {
79 if (fOwnsColor)
80 delete color;
81 if (fOwnsMaskFilter)
82 delete maskFilter;
83 if (fOwnsPathEffect)
84 delete pathEffect;
85 if (fOwnsShader)
86 delete shader;
87 if (fOwnsTypeface)
88 delete typeface;
89 }
90
add(SkAnimateMaker * maker,SkDisplayable * child)91 bool SkDrawPaint::add(SkAnimateMaker* maker, SkDisplayable* child) {
92 SkASSERT(child && child->isPaintPart());
93 SkPaintPart* part = (SkPaintPart*) child;
94 if (part->add() && maker)
95 maker->setErrorCode(SkDisplayXMLParserError::kErrorAddingToPaint);
96 return true;
97 }
98
deepCopy(SkAnimateMaker * maker)99 SkDisplayable* SkDrawPaint::deepCopy(SkAnimateMaker* maker) {
100 SkDrawColor* tempColor = color;
101 color = nullptr;
102 SkDrawPaint* copy = (SkDrawPaint*) INHERITED::deepCopy(maker);
103 color = tempColor;
104 tempColor = (SkDrawColor*) color->deepCopy(maker);
105 tempColor->setParent(copy);
106 tempColor->add();
107 copy->fOwnsColor = true;
108 return copy;
109 }
110
draw(SkAnimateMaker & maker)111 bool SkDrawPaint::draw(SkAnimateMaker& maker) {
112 SkPaint* paint = maker.fPaint;
113 setupPaint(paint);
114 return false;
115 }
116
117 #ifdef SK_DUMP_ENABLED
dump(SkAnimateMaker * maker)118 void SkDrawPaint::dump(SkAnimateMaker* maker) {
119 dumpBase(maker);
120 dumpAttrs(maker);
121 bool closedYet = false;
122 SkDisplayList::fIndent +=4;
123 //should i say if (maskFilter && ...?
124 if (maskFilter != (SkDrawMaskFilter*)-1) {
125 SkDebugf(">\n");
126 maskFilter->dump(maker);
127 closedYet = true;
128 }
129 if (pathEffect != (SkDrawPathEffect*) -1) {
130 if (closedYet == false) {
131 SkDebugf(">\n");
132 closedYet = true;
133 }
134 pathEffect->dump(maker);
135 }
136 if (fOwnsTypeface) {
137 if (closedYet == false) {
138 SkDebugf(">\n");
139 closedYet = true;
140 }
141 typeface->dump(maker);
142 }
143 SkDisplayList::fIndent -= 4;
144 dumpChildren(maker, closedYet);
145 }
146 #endif
147
executeFunction(SkDisplayable * target,int index,SkTDArray<SkScriptValue> & parameters,SkDisplayTypes type,SkScriptValue * scriptValue)148 void SkDrawPaint::executeFunction(SkDisplayable* target, int index,
149 SkTDArray<SkScriptValue>& parameters, SkDisplayTypes type,
150 SkScriptValue* scriptValue) {
151 if (scriptValue == nullptr)
152 return;
153 SkASSERT(target == this);
154 switch (index) {
155 case SK_FUNCTION(measureText): {
156 SkASSERT(parameters.count() == 1);
157 SkASSERT(type == SkType_Float);
158 SkPaint paint;
159 setupPaint(&paint);
160 scriptValue->fType = SkType_Float;
161 SkASSERT(parameters[0].fType == SkType_String);
162 scriptValue->fOperand.fScalar = paint.measureText(parameters[0].fOperand.fString->c_str(),
163 parameters[0].fOperand.fString->size());
164 // SkDebugf("measureText: %s = %g\n", parameters[0].fOperand.fString->c_str(),
165 // scriptValue->fOperand.fScalar / 65536.0f);
166 } break;
167 default:
168 SkASSERT(0);
169 }
170 }
171
getFunctionsParameters()172 const SkFunctionParamType* SkDrawPaint::getFunctionsParameters() {
173 return fFunctionParameters;
174 }
175
getProperty(int index,SkScriptValue * value) const176 bool SkDrawPaint::getProperty(int index, SkScriptValue* value) const {
177 SkPaint::FontMetrics metrics;
178 SkPaint paint;
179 setupPaint(&paint);
180 paint.getFontMetrics(&metrics);
181 switch (index) {
182 case SK_PROPERTY(ascent):
183 value->fOperand.fScalar = metrics.fAscent;
184 break;
185 case SK_PROPERTY(descent):
186 value->fOperand.fScalar = metrics.fDescent;
187 break;
188 // should consider returning fLeading as well (or roll it into ascent/descent somehow
189 default:
190 SkASSERT(0);
191 return false;
192 }
193 value->fType = SkType_Float;
194 return true;
195 }
196
resolveIDs(SkAnimateMaker & maker,SkDisplayable * origDisp,SkApply *)197 bool SkDrawPaint::resolveIDs(SkAnimateMaker& maker, SkDisplayable* origDisp, SkApply* ) {
198 SkASSERT(origDisp->isPaint());
199 SkDrawPaint* original = (SkDrawPaint*) origDisp;
200 if (fOwnsColor && maker.resolveID(color, original->color) == false)
201 return true;
202 if (fOwnsMaskFilter && maker.resolveID(maskFilter, original->maskFilter) == false)
203 return true;
204 if (fOwnsPathEffect && maker.resolveID(pathEffect, original->pathEffect) == false)
205 return true;
206 if (fOwnsShader && maker.resolveID(shader, original->shader) == false)
207 return true;
208 if (fOwnsTypeface && maker.resolveID(typeface, original->typeface) == false)
209 return true;
210 return false; // succeeded
211 }
212
setupPaint(SkPaint * paint) const213 void SkDrawPaint::setupPaint(SkPaint* paint) const {
214 if (antiAlias != -1)
215 paint->setAntiAlias(SkToBool(antiAlias));
216 if (color != nullptr)
217 paint->setColor(color->getColor());
218 if (fakeBold != -1)
219 paint->setFakeBoldText(SkToBool(fakeBold));
220 if (filterBitmap != -1)
221 paint->setFilterQuality(filterBitmap ? kLow_SkFilterQuality : kNone_SkFilterQuality);
222 // stroke is legacy; style setting if present overrides stroke
223 if (stroke != -1)
224 paint->setStyle(SkToBool(stroke) ? SkPaint::kStroke_Style : SkPaint::kFill_Style);
225 if (style != -1)
226 paint->setStyle((SkPaint::Style) style);
227 if (linearText != -1)
228 paint->setLinearText(SkToBool(linearText));
229 if (maskFilter == nullptr)
230 paint->setMaskFilter(nullptr);
231 else if (maskFilter != (SkDrawMaskFilter*) -1)
232 SkSafeUnref(paint->setMaskFilter(maskFilter->getMaskFilter()));
233 if (pathEffect == nullptr)
234 paint->setPathEffect(nullptr);
235 else if (pathEffect != (SkDrawPathEffect*) -1)
236 SkSafeUnref(paint->setPathEffect(pathEffect->getPathEffect()));
237 if (shader == nullptr)
238 paint->setShader(nullptr);
239 else if (shader != (SkDrawShader*) -1)
240 SkSafeUnref(paint->setShader(shader->getShader()));
241 if (strikeThru != -1)
242 paint->setStrikeThruText(SkToBool(strikeThru));
243 if (strokeCap != -1)
244 paint->setStrokeCap((SkPaint::Cap) strokeCap);
245 if (strokeJoin != -1)
246 paint->setStrokeJoin((SkPaint::Join) strokeJoin);
247 if (SkScalarIsNaN(strokeMiter) == false)
248 paint->setStrokeMiter(strokeMiter);
249 if (SkScalarIsNaN(strokeWidth) == false)
250 paint->setStrokeWidth(strokeWidth);
251 if (textAlign != -1)
252 paint->setTextAlign((SkPaint::Align) textAlign);
253 if (SkScalarIsNaN(textScaleX) == false)
254 paint->setTextScaleX(textScaleX);
255 if (SkScalarIsNaN(textSize) == false)
256 paint->setTextSize(textSize);
257 if (SkScalarIsNaN(textSkewX) == false)
258 paint->setTextSkewX(textSkewX);
259 if (typeface == nullptr)
260 paint->setTypeface(nullptr);
261 else if (typeface != (SkDrawTypeface*) -1)
262 SkSafeUnref(paint->setTypeface(typeface->getTypeface()));
263 if (underline != -1)
264 paint->setUnderlineText(SkToBool(underline));
265 if (xfermode != -1)
266 paint->setXfermodeMode((SkXfermode::Mode) xfermode);
267 }
268