OLD | NEW |
1 /* | 1 /* |
2 * Copyright 2010 Google Inc. | 2 * Copyright 2010 Google Inc. |
3 * | 3 * |
4 * Use of this source code is governed by a BSD-style license that can be | 4 * Use of this source code is governed by a BSD-style license that can be |
5 * found in the LICENSE file. | 5 * found in the LICENSE file. |
6 */ | 6 */ |
7 | 7 |
8 #include "GrTextContext.h" | 8 #include "GrTextContext.h" |
9 #include "GrContext.h" | 9 #include "GrContext.h" |
10 #include "GrDrawTarget.h" | 10 #include "GrDrawTarget.h" |
11 #include "GrFontScaler.h" | 11 #include "GrFontScaler.h" |
12 | 12 |
13 #include "SkAutoKern.h" | 13 #include "SkAutoKern.h" |
| 14 #include "SkDrawFilter.h" |
14 #include "SkDrawProcs.h" | 15 #include "SkDrawProcs.h" |
15 #include "SkGlyphCache.h" | 16 #include "SkGlyphCache.h" |
16 #include "SkGpuDevice.h" | 17 #include "SkGpuDevice.h" |
| 18 #include "SkTextBlob.h" |
17 #include "SkTextMapStateProc.h" | 19 #include "SkTextMapStateProc.h" |
18 #include "SkTextToPathIter.h" | 20 #include "SkTextToPathIter.h" |
19 | 21 |
20 GrTextContext::GrTextContext(GrContext* context, SkGpuDevice* gpuDevice, | 22 GrTextContext::GrTextContext(GrContext* context, SkGpuDevice* gpuDevice, |
21 const SkDeviceProperties& properties) | 23 const SkDeviceProperties& properties) |
22 : fFallbackTextContext(NULL) | 24 : fFallbackTextContext(NULL) |
23 , fContext(context) | 25 , fContext(context) |
24 , fGpuDevice(gpuDevice) | 26 , fGpuDevice(gpuDevice) |
25 , fDeviceProperties(properties) | 27 , fDeviceProperties(properties) |
26 , fDrawTarget(NULL) { | 28 , fDrawTarget(NULL) { |
(...skipping 57 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
84 return; | 86 return; |
85 } | 87 } |
86 textContext = textContext->fFallbackTextContext; | 88 textContext = textContext->fFallbackTextContext; |
87 } while (textContext); | 89 } while (textContext); |
88 | 90 |
89 // fall back to drawing as a path | 91 // fall back to drawing as a path |
90 this->drawPosTextAsPath(skPaint, viewMatrix, text, byteLength, pos, scalarsP
erPosition, offset, | 92 this->drawPosTextAsPath(skPaint, viewMatrix, text, byteLength, pos, scalarsP
erPosition, offset, |
91 clipBounds); | 93 clipBounds); |
92 } | 94 } |
93 | 95 |
| 96 void GrTextContext::drawTextBlob(GrRenderTarget* rt, const GrClip& clip, const S
kPaint& skPaint, |
| 97 const SkMatrix& viewMatrix, const SkTextBlob* b
lob, |
| 98 SkScalar x, SkScalar y, |
| 99 SkDrawFilter* drawFilter, const SkIRect& clipBo
unds) { |
| 100 if (!fContext->getTextTarget()) { |
| 101 return; |
| 102 } |
| 103 |
| 104 GrTextContext* textContext = this; |
| 105 do { |
| 106 if (textContext->canDraw(skPaint, viewMatrix)) { |
| 107 textContext->onDrawTextBlob(rt, clip, skPaint, viewMatrix, blob, x,
y, drawFilter, |
| 108 clipBounds); |
| 109 return; |
| 110 } |
| 111 textContext = textContext->fFallbackTextContext; |
| 112 } while (textContext); |
| 113 } |
| 114 |
94 void GrTextContext::drawTextAsPath(const SkPaint& skPaint, const SkMatrix& viewM
atrix, | 115 void GrTextContext::drawTextAsPath(const SkPaint& skPaint, const SkMatrix& viewM
atrix, |
95 const char text[], size_t byteLength, SkScala
r x, SkScalar y, | 116 const char text[], size_t byteLength, SkScala
r x, SkScalar y, |
96 const SkIRect& clipBounds) { | 117 const SkIRect& clipBounds) { |
97 SkTextToPathIter iter(text, byteLength, skPaint, true); | 118 SkTextToPathIter iter(text, byteLength, skPaint, true); |
98 | 119 |
99 SkMatrix matrix; | 120 SkMatrix matrix; |
100 matrix.setScale(iter.getPathScale(), iter.getPathScale()); | 121 matrix.setScale(iter.getPathScale(), iter.getPathScale()); |
101 matrix.postTranslate(x, y); | 122 matrix.postTranslate(x, y); |
102 | 123 |
103 const SkPath* iterPath; | 124 const SkPath* iterPath; |
(...skipping 48 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
152 | 173 |
153 matrix[SkMatrix::kMTransX] = loc.fX; | 174 matrix[SkMatrix::kMTransX] = loc.fX; |
154 matrix[SkMatrix::kMTransY] = loc.fY; | 175 matrix[SkMatrix::kMTransY] = loc.fY; |
155 fGpuDevice->internalDrawPath(*path, paint, viewMatrix, &matrix,
clipBounds, false); | 176 fGpuDevice->internalDrawPath(*path, paint, viewMatrix, &matrix,
clipBounds, false); |
156 } | 177 } |
157 } | 178 } |
158 pos += scalarsPerPosition; | 179 pos += scalarsPerPosition; |
159 } | 180 } |
160 } | 181 } |
161 | 182 |
| 183 void GrTextContext::onDrawTextBlob(GrRenderTarget* rt, const GrClip& clip, |
| 184 const SkPaint& skPaint, const SkMatrix& viewM
atrix, |
| 185 const SkTextBlob* blob, SkScalar x, SkScalar
y, |
| 186 SkDrawFilter* drawFilter, const SkIRect& clip
Bounds) { |
| 187 SkPaint runPaint = skPaint; |
| 188 |
| 189 SkTextBlob::RunIterator it(blob); |
| 190 for (;!it.done(); it.next()) { |
| 191 size_t textLen = it.glyphCount() * sizeof(uint16_t); |
| 192 const SkPoint& offset = it.offset(); |
| 193 // applyFontToPaint() always overwrites the exact same attributes, |
| 194 // so it is safe to not re-seed the paint for this reason. |
| 195 it.applyFontToPaint(&runPaint); |
| 196 |
| 197 if (drawFilter && !drawFilter->filter(&runPaint, SkDrawFilter::kText_Typ
e)) { |
| 198 // A false return from filter() means we should abort the current dr
aw. |
| 199 runPaint = skPaint; |
| 200 continue; |
| 201 } |
| 202 |
| 203 runPaint.setFlags(fGpuDevice->filterTextFlags(runPaint)); |
| 204 |
| 205 GrPaint grPaint; |
| 206 SkPaint2GrPaintShader(fContext, fRenderTarget, runPaint, viewMatrix, tru
e, &grPaint); |
| 207 |
| 208 switch (it.positioning()) { |
| 209 case SkTextBlob::kDefault_Positioning: |
| 210 this->drawText(rt, clip, grPaint, runPaint, viewMatrix, (const char
*)it.glyphs(), |
| 211 textLen, x + offset.x(), y + offset.y(), clipBounds); |
| 212 break; |
| 213 case SkTextBlob::kHorizontal_Positioning: |
| 214 this->drawPosText(rt, clip, grPaint, runPaint, viewMatrix, (const ch
ar*)it.glyphs(), |
| 215 textLen, it.pos(), 1, SkPoint::Make(x, y + offset.
y()), clipBounds); |
| 216 break; |
| 217 case SkTextBlob::kFull_Positioning: |
| 218 this->drawPosText(rt, clip, grPaint, runPaint, viewMatrix, (const ch
ar*)it.glyphs(), |
| 219 textLen, it.pos(), 2, SkPoint::Make(x, y), clipBou
nds); |
| 220 break; |
| 221 default: |
| 222 SkFAIL("unhandled positioning mode"); |
| 223 } |
| 224 |
| 225 if (drawFilter) { |
| 226 // A draw filter may change the paint arbitrarily, so we must re-see
d in this case. |
| 227 runPaint = skPaint; |
| 228 } |
| 229 } |
| 230 } |
| 231 |
162 //*** change to output positions? | 232 //*** change to output positions? |
163 int GrTextContext::MeasureText(SkGlyphCache* cache, SkDrawCacheProc glyphCachePr
oc, | 233 int GrTextContext::MeasureText(SkGlyphCache* cache, SkDrawCacheProc glyphCachePr
oc, |
164 const char text[], size_t byteLength, SkVector*
stopVector) { | 234 const char text[], size_t byteLength, SkVector*
stopVector) { |
165 SkFixed x = 0, y = 0; | 235 SkFixed x = 0, y = 0; |
166 const char* stop = text + byteLength; | 236 const char* stop = text + byteLength; |
167 | 237 |
168 SkAutoKern autokern; | 238 SkAutoKern autokern; |
169 | 239 |
170 int numGlyphs = 0; | 240 int numGlyphs = 0; |
171 while (text < stop) { | 241 while (text < stop) { |
(...skipping 24 matching lines...) Expand all Loading... |
196 if (cache->getAuxProcData(GlyphCacheAuxProc, &auxData)) { | 266 if (cache->getAuxProcData(GlyphCacheAuxProc, &auxData)) { |
197 scaler = (GrFontScaler*)auxData; | 267 scaler = (GrFontScaler*)auxData; |
198 } | 268 } |
199 if (NULL == scaler) { | 269 if (NULL == scaler) { |
200 scaler = SkNEW_ARGS(GrFontScaler, (cache)); | 270 scaler = SkNEW_ARGS(GrFontScaler, (cache)); |
201 cache->setAuxProc(GlyphCacheAuxProc, scaler); | 271 cache->setAuxProc(GlyphCacheAuxProc, scaler); |
202 } | 272 } |
203 | 273 |
204 return scaler; | 274 return scaler; |
205 } | 275 } |
OLD | NEW |