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 "SkPaint.h" 9 #include "SkPaintPriv.h" 10 #include "SkAutoKern.h" 11 #include "SkColorFilter.h" 12 #include "SkData.h" 13 #include "SkDraw.h" 14 #include "SkFontDescriptor.h" 15 #include "SkGraphics.h" 16 #include "SkGlyphCache.h" 17 #include "SkImageFilter.h" 18 #include "SkMaskFilter.h" 19 #include "SkMaskGamma.h" 20 #include "SkMutex.h" 21 #include "SkReadBuffer.h" 22 #include "SkWriteBuffer.h" 23 #include "SkOpts.h" 24 #include "SkPaintDefaults.h" 25 #include "SkPathEffect.h" 26 #include "SkSafeRange.h" 27 #include "SkScalar.h" 28 #include "SkScalerContext.h" 29 #include "SkShader.h" 30 #include "SkShaderBase.h" 31 #include "SkStringUtils.h" 32 #include "SkStroke.h" 33 #include "SkStrokeRec.h" 34 #include "SkSurfacePriv.h" 35 #include "SkTextBlob.h" 36 #include "SkTextBlobRunIterator.h" 37 #include "SkTextFormatParams.h" 38 #include "SkTextToPathIter.h" 39 #include "SkTLazy.h" 40 #include "SkTypeface.h" 41 42 static inline uint32_t set_clear_mask(uint32_t bits, bool cond, uint32_t mask) { 43 return cond ? bits | mask : bits & ~mask; 44 } 45 46 // define this to get a printf for out-of-range parameter in setters 47 // e.g. setTextSize(-1) 48 //#define SK_REPORT_API_RANGE_CHECK 49 50 SkPaint::SkPaint() { 51 fTextSize = SkPaintDefaults_TextSize; 52 fTextScaleX = SK_Scalar1; 53 fTextSkewX = 0; 54 fColor = SK_ColorBLACK; 55 fWidth = 0; 56 fMiterLimit = SkPaintDefaults_MiterLimit; 57 fBlendMode = (unsigned)SkBlendMode::kSrcOver; 58 59 // Zero all bitfields, then set some non-zero defaults. 60 fBitfieldsUInt = 0; 61 fBitfields.fFlags = SkPaintDefaults_Flags; 62 fBitfields.fCapType = kDefault_Cap; 63 fBitfields.fJoinType = kDefault_Join; 64 fBitfields.fTextAlign = kLeft_Align; 65 fBitfields.fStyle = kFill_Style; 66 fBitfields.fTextEncoding = kUTF8_TextEncoding; 67 fBitfields.fHinting = SkPaintDefaults_Hinting; 68 } 69 70 SkPaint::SkPaint(const SkPaint& src) 71 #define COPY(field) field(src.field) 72 : COPY(fTypeface) 73 , COPY(fPathEffect) 74 , COPY(fShader) 75 , COPY(fMaskFilter) 76 , COPY(fColorFilter) 77 , COPY(fDrawLooper) 78 , COPY(fImageFilter) 79 , COPY(fTextSize) 80 , COPY(fTextScaleX) 81 , COPY(fTextSkewX) 82 , COPY(fColor) 83 , COPY(fWidth) 84 , COPY(fMiterLimit) 85 , COPY(fBlendMode) 86 , COPY(fBitfields) 87 #undef COPY 88 {} 89 90 SkPaint::SkPaint(SkPaint&& src) { 91 #define MOVE(field) field = std::move(src.field) 92 MOVE(fTypeface); 93 MOVE(fPathEffect); 94 MOVE(fShader); 95 MOVE(fMaskFilter); 96 MOVE(fColorFilter); 97 MOVE(fDrawLooper); 98 MOVE(fImageFilter); 99 MOVE(fTextSize); 100 MOVE(fTextScaleX); 101 MOVE(fTextSkewX); 102 MOVE(fColor); 103 MOVE(fWidth); 104 MOVE(fMiterLimit); 105 MOVE(fBlendMode); 106 MOVE(fBitfields); 107 #undef MOVE 108 } 109 110 SkPaint::~SkPaint() {} 111 112 SkPaint& SkPaint::operator=(const SkPaint& src) { 113 if (this == &src) { 114 return *this; 115 } 116 117 #define ASSIGN(field) field = src.field 118 ASSIGN(fTypeface); 119 ASSIGN(fPathEffect); 120 ASSIGN(fShader); 121 ASSIGN(fMaskFilter); 122 ASSIGN(fColorFilter); 123 ASSIGN(fDrawLooper); 124 ASSIGN(fImageFilter); 125 ASSIGN(fTextSize); 126 ASSIGN(fTextScaleX); 127 ASSIGN(fTextSkewX); 128 ASSIGN(fColor); 129 ASSIGN(fWidth); 130 ASSIGN(fMiterLimit); 131 ASSIGN(fBlendMode); 132 ASSIGN(fBitfields); 133 #undef ASSIGN 134 135 return *this; 136 } 137 138 SkPaint& SkPaint::operator=(SkPaint&& src) { 139 if (this == &src) { 140 return *this; 141 } 142 143 #define MOVE(field) field = std::move(src.field) 144 MOVE(fTypeface); 145 MOVE(fPathEffect); 146 MOVE(fShader); 147 MOVE(fMaskFilter); 148 MOVE(fColorFilter); 149 MOVE(fDrawLooper); 150 MOVE(fImageFilter); 151 MOVE(fTextSize); 152 MOVE(fTextScaleX); 153 MOVE(fTextSkewX); 154 MOVE(fColor); 155 MOVE(fWidth); 156 MOVE(fMiterLimit); 157 MOVE(fBlendMode); 158 MOVE(fBitfields); 159 #undef MOVE 160 161 return *this; 162 } 163 164 bool operator==(const SkPaint& a, const SkPaint& b) { 165 #define EQUAL(field) (a.field == b.field) 166 return EQUAL(fTypeface) 167 && EQUAL(fPathEffect) 168 && EQUAL(fShader) 169 && EQUAL(fMaskFilter) 170 && EQUAL(fColorFilter) 171 && EQUAL(fDrawLooper) 172 && EQUAL(fImageFilter) 173 && EQUAL(fTextSize) 174 && EQUAL(fTextScaleX) 175 && EQUAL(fTextSkewX) 176 && EQUAL(fColor) 177 && EQUAL(fWidth) 178 && EQUAL(fMiterLimit) 179 && EQUAL(fBlendMode) 180 && EQUAL(fBitfieldsUInt) 181 ; 182 #undef EQUAL 183 } 184 185 #define DEFINE_REF_FOO(type) sk_sp<Sk##type> SkPaint::ref##type() const { return f##type; } 186 DEFINE_REF_FOO(ColorFilter) 187 DEFINE_REF_FOO(DrawLooper) 188 DEFINE_REF_FOO(ImageFilter) 189 DEFINE_REF_FOO(MaskFilter) 190 DEFINE_REF_FOO(PathEffect) 191 DEFINE_REF_FOO(Shader) 192 DEFINE_REF_FOO(Typeface) 193 #undef DEFINE_REF_FOO 194 195 void SkPaint::reset() { 196 SkPaint init; 197 *this = init; 198 } 199 200 void SkPaint::setFilterQuality(SkFilterQuality quality) { 201 fBitfields.fFilterQuality = quality; 202 } 203 204 void SkPaint::setHinting(Hinting hintingLevel) { 205 fBitfields.fHinting = hintingLevel; 206 } 207 208 void SkPaint::setFlags(uint32_t flags) { 209 fBitfields.fFlags = flags; 210 } 211 212 void SkPaint::setAntiAlias(bool doAA) { 213 this->setFlags(set_clear_mask(fBitfields.fFlags, doAA, kAntiAlias_Flag)); 214 } 215 216 void SkPaint::setDither(bool doDither) { 217 this->setFlags(set_clear_mask(fBitfields.fFlags, doDither, kDither_Flag)); 218 } 219 220 void SkPaint::setSubpixelText(bool doSubpixel) { 221 this->setFlags(set_clear_mask(fBitfields.fFlags, doSubpixel, kSubpixelText_Flag)); 222 } 223 224 void SkPaint::setLCDRenderText(bool doLCDRender) { 225 this->setFlags(set_clear_mask(fBitfields.fFlags, doLCDRender, kLCDRenderText_Flag)); 226 } 227 228 void SkPaint::setEmbeddedBitmapText(bool doEmbeddedBitmapText) { 229 this->setFlags(set_clear_mask(fBitfields.fFlags, doEmbeddedBitmapText, kEmbeddedBitmapText_Flag)); 230 } 231 232 void SkPaint::setAutohinted(bool useAutohinter) { 233 this->setFlags(set_clear_mask(fBitfields.fFlags, useAutohinter, kAutoHinting_Flag)); 234 } 235 236 void SkPaint::setLinearText(bool doLinearText) { 237 this->setFlags(set_clear_mask(fBitfields.fFlags, doLinearText, kLinearText_Flag)); 238 } 239 240 void SkPaint::setVerticalText(bool doVertical) { 241 this->setFlags(set_clear_mask(fBitfields.fFlags, doVertical, kVerticalText_Flag)); 242 } 243 244 void SkPaint::setFakeBoldText(bool doFakeBold) { 245 this->setFlags(set_clear_mask(fBitfields.fFlags, doFakeBold, kFakeBoldText_Flag)); 246 } 247 248 void SkPaint::setDevKernText(bool doDevKern) { 249 this->setFlags(set_clear_mask(fBitfields.fFlags, doDevKern, kDevKernText_Flag)); 250 } 251 252 void SkPaint::setStyle(Style style) { 253 if ((unsigned)style < kStyleCount) { 254 fBitfields.fStyle = style; 255 } else { 256 #ifdef SK_REPORT_API_RANGE_CHECK 257 SkDebugf("SkPaint::setStyle(%d) out of range\n", style); 258 #endif 259 } 260 } 261 262 void SkPaint::setColor(SkColor color) { 263 fColor = color; 264 } 265 266 void SkPaint::setAlpha(U8CPU a) { 267 this->setColor(SkColorSetARGB(a, SkColorGetR(fColor), 268 SkColorGetG(fColor), SkColorGetB(fColor))); 269 } 270 271 void SkPaint::setARGB(U8CPU a, U8CPU r, U8CPU g, U8CPU b) { 272 this->setColor(SkColorSetARGB(a, r, g, b)); 273 } 274 275 void SkPaint::setStrokeWidth(SkScalar width) { 276 if (width >= 0) { 277 fWidth = width; 278 } else { 279 #ifdef SK_REPORT_API_RANGE_CHECK 280 SkDebugf("SkPaint::setStrokeWidth() called with negative value\n"); 281 #endif 282 } 283 } 284 285 void SkPaint::setStrokeMiter(SkScalar limit) { 286 if (limit >= 0) { 287 fMiterLimit = limit; 288 } else { 289 #ifdef SK_REPORT_API_RANGE_CHECK 290 SkDebugf("SkPaint::setStrokeMiter() called with negative value\n"); 291 #endif 292 } 293 } 294 295 void SkPaint::setStrokeCap(Cap ct) { 296 if ((unsigned)ct < kCapCount) { 297 fBitfields.fCapType = SkToU8(ct); 298 } else { 299 #ifdef SK_REPORT_API_RANGE_CHECK 300 SkDebugf("SkPaint::setStrokeCap(%d) out of range\n", ct); 301 #endif 302 } 303 } 304 305 void SkPaint::setStrokeJoin(Join jt) { 306 if ((unsigned)jt < kJoinCount) { 307 fBitfields.fJoinType = SkToU8(jt); 308 } else { 309 #ifdef SK_REPORT_API_RANGE_CHECK 310 SkDebugf("SkPaint::setStrokeJoin(%d) out of range\n", jt); 311 #endif 312 } 313 } 314 315 /////////////////////////////////////////////////////////////////////////////// 316 317 void SkPaint::setTextAlign(Align align) { 318 if ((unsigned)align < kAlignCount) { 319 fBitfields.fTextAlign = SkToU8(align); 320 } else { 321 #ifdef SK_REPORT_API_RANGE_CHECK 322 SkDebugf("SkPaint::setTextAlign(%d) out of range\n", align); 323 #endif 324 } 325 } 326 327 void SkPaint::setTextSize(SkScalar ts) { 328 if (ts >= 0) { 329 fTextSize = ts; 330 } else { 331 #ifdef SK_REPORT_API_RANGE_CHECK 332 SkDebugf("SkPaint::setTextSize() called with negative value\n"); 333 #endif 334 } 335 } 336 337 void SkPaint::setTextScaleX(SkScalar scaleX) { 338 fTextScaleX = scaleX; 339 } 340 341 void SkPaint::setTextSkewX(SkScalar skewX) { 342 fTextSkewX = skewX; 343 } 344 345 void SkPaint::setTextEncoding(TextEncoding encoding) { 346 if ((unsigned)encoding <= kGlyphID_TextEncoding) { 347 fBitfields.fTextEncoding = encoding; 348 } else { 349 #ifdef SK_REPORT_API_RANGE_CHECK 350 SkDebugf("SkPaint::setTextEncoding(%d) out of range\n", encoding); 351 #endif 352 } 353 } 354 355 /////////////////////////////////////////////////////////////////////////////// 356 357 #define MOVE_FIELD(Field) void SkPaint::set##Field(sk_sp<Sk##Field> f) { f##Field = std::move(f); } 358 MOVE_FIELD(Typeface) 359 MOVE_FIELD(ImageFilter) 360 MOVE_FIELD(Shader) 361 MOVE_FIELD(ColorFilter) 362 MOVE_FIELD(PathEffect) 363 MOVE_FIELD(MaskFilter) 364 MOVE_FIELD(DrawLooper) 365 #undef MOVE_FIELD 366 void SkPaint::setLooper(sk_sp<SkDrawLooper> looper) { fDrawLooper = std::move(looper); } 367 368 /////////////////////////////////////////////////////////////////////////////// 369 370 static SkScalar mag2(SkScalar x, SkScalar y) { 371 return x * x + y * y; 372 } 373 374 static bool tooBig(const SkMatrix& m, SkScalar ma2max) { 375 return mag2(m[SkMatrix::kMScaleX], m[SkMatrix::kMSkewY]) > ma2max 376 || 377 mag2(m[SkMatrix::kMSkewX], m[SkMatrix::kMScaleY]) > ma2max; 378 } 379 380 bool SkPaint::TooBigToUseCache(const SkMatrix& ctm, const SkMatrix& textM, SkScalar maxLimit) { 381 SkASSERT(!ctm.hasPerspective()); 382 SkASSERT(!textM.hasPerspective()); 383 384 SkMatrix matrix; 385 matrix.setConcat(ctm, textM); 386 return tooBig(matrix, MaxCacheSize2(maxLimit)); 387 } 388 389 SkScalar SkPaint::MaxCacheSize2(SkScalar maxLimit) { 390 // we have a self-imposed maximum, just for memory-usage sanity 391 const int limit = SkMin32(SkGraphics::GetFontCachePointSizeLimit(), maxLimit); 392 const SkScalar maxSize = SkIntToScalar(limit); 393 return maxSize * maxSize; 394 } 395 396 /////////////////////////////////////////////////////////////////////////////// 397 398 #include "SkGlyphCache.h" 399 #include "SkUtils.h" 400 401 int SkPaint::textToGlyphs(const void* textData, size_t byteLength, uint16_t glyphs[]) const { 402 if (byteLength == 0) { 403 return 0; 404 } 405 406 SkASSERT(textData != nullptr); 407 408 if (nullptr == glyphs) { 409 switch (this->getTextEncoding()) { 410 case kUTF8_TextEncoding: 411 return SkUTF8_CountUnichars(textData, byteLength); 412 case kUTF16_TextEncoding: 413 return SkUTF16_CountUnichars(textData, byteLength); 414 case kUTF32_TextEncoding: 415 return SkToInt(byteLength >> 2); 416 case kGlyphID_TextEncoding: 417 return SkToInt(byteLength >> 1); 418 default: 419 SkDEBUGFAIL("unknown text encoding"); 420 } 421 return 0; 422 } 423 424 // if we get here, we have a valid glyphs[] array, so time to fill it in 425 426 // handle this encoding before the setup for the glyphcache 427 if (this->getTextEncoding() == kGlyphID_TextEncoding) { 428 // we want to ignore the low bit of byteLength 429 memcpy(glyphs, textData, byteLength >> 1 << 1); 430 return SkToInt(byteLength >> 1); 431 } 432 433 SkAutoGlyphCache autoCache(*this, nullptr, nullptr); 434 SkGlyphCache* cache = autoCache.getCache(); 435 436 const char* text = (const char*)textData; 437 const char* stop = text + byteLength; 438 uint16_t* gptr = glyphs; 439 440 switch (this->getTextEncoding()) { 441 case SkPaint::kUTF8_TextEncoding: 442 while (text < stop) { 443 SkUnichar u = SkUTF8_NextUnicharWithError(&text, stop); 444 if (u < 0) { 445 return 0; // bad UTF-8 sequence 446 } 447 *gptr++ = cache->unicharToGlyph(u); 448 } 449 break; 450 case SkPaint::kUTF16_TextEncoding: { 451 const uint16_t* text16 = (const uint16_t*)text; 452 const uint16_t* stop16 = (const uint16_t*)stop; 453 while (text16 < stop16) { 454 *gptr++ = cache->unicharToGlyph(SkUTF16_NextUnichar(&text16)); 455 } 456 break; 457 } 458 case kUTF32_TextEncoding: { 459 const int32_t* text32 = (const int32_t*)text; 460 const int32_t* stop32 = (const int32_t*)stop; 461 while (text32 < stop32) { 462 *gptr++ = cache->unicharToGlyph(*text32++); 463 } 464 break; 465 } 466 default: 467 SkDEBUGFAIL("unknown text encoding"); 468 } 469 return SkToInt(gptr - glyphs); 470 } 471 472 bool SkPaint::containsText(const void* textData, size_t byteLength) const { 473 if (0 == byteLength) { 474 return true; 475 } 476 477 SkASSERT(textData != nullptr); 478 479 // handle this encoding before the setup for the glyphcache 480 if (this->getTextEncoding() == kGlyphID_TextEncoding) { 481 const uint16_t* glyphID = static_cast<const uint16_t*>(textData); 482 size_t count = byteLength >> 1; 483 for (size_t i = 0; i < count; i++) { 484 if (0 == glyphID[i]) { 485 return false; 486 } 487 } 488 return true; 489 } 490 491 SkAutoGlyphCache autoCache(*this, nullptr, nullptr); 492 SkGlyphCache* cache = autoCache.getCache(); 493 494 switch (this->getTextEncoding()) { 495 case SkPaint::kUTF8_TextEncoding: { 496 const char* text = static_cast<const char*>(textData); 497 const char* stop = text + byteLength; 498 while (text < stop) { 499 if (0 == cache->unicharToGlyph(SkUTF8_NextUnichar(&text))) { 500 return false; 501 } 502 } 503 break; 504 } 505 case SkPaint::kUTF16_TextEncoding: { 506 const uint16_t* text = static_cast<const uint16_t*>(textData); 507 const uint16_t* stop = text + (byteLength >> 1); 508 while (text < stop) { 509 if (0 == cache->unicharToGlyph(SkUTF16_NextUnichar(&text))) { 510 return false; 511 } 512 } 513 break; 514 } 515 case SkPaint::kUTF32_TextEncoding: { 516 const int32_t* text = static_cast<const int32_t*>(textData); 517 const int32_t* stop = text + (byteLength >> 2); 518 while (text < stop) { 519 if (0 == cache->unicharToGlyph(*text++)) { 520 return false; 521 } 522 } 523 break; 524 } 525 default: 526 SkDEBUGFAIL("unknown text encoding"); 527 return false; 528 } 529 return true; 530 } 531 532 void SkPaint::glyphsToUnichars(const uint16_t glyphs[], int count, SkUnichar textData[]) const { 533 if (count <= 0) { 534 return; 535 } 536 537 SkASSERT(glyphs != nullptr); 538 SkASSERT(textData != nullptr); 539 540 SkSurfaceProps props(0, kUnknown_SkPixelGeometry); 541 SkAutoGlyphCache autoCache(*this, &props, nullptr); 542 SkGlyphCache* cache = autoCache.getCache(); 543 544 for (int index = 0; index < count; index++) { 545 textData[index] = cache->glyphToUnichar(glyphs[index]); 546 } 547 } 548 549 /////////////////////////////////////////////////////////////////////////////// 550 551 static const SkGlyph& sk_getMetrics_utf8_next(SkGlyphCache* cache, 552 const char** text) { 553 SkASSERT(cache != nullptr); 554 SkASSERT(text != nullptr); 555 556 return cache->getUnicharMetrics(SkUTF8_NextUnichar(text)); 557 } 558 559 static const SkGlyph& sk_getMetrics_utf16_next(SkGlyphCache* cache, 560 const char** text) { 561 SkASSERT(cache != nullptr); 562 SkASSERT(text != nullptr); 563 564 return cache->getUnicharMetrics(SkUTF16_NextUnichar((const uint16_t**)text)); 565 } 566 567 static const SkGlyph& sk_getMetrics_utf32_next(SkGlyphCache* cache, 568 const char** text) { 569 SkASSERT(cache != nullptr); 570 SkASSERT(text != nullptr); 571 572 const int32_t* ptr = *(const int32_t**)text; 573 SkUnichar uni = *ptr++; 574 *text = (const char*)ptr; 575 return cache->getUnicharMetrics(uni); 576 } 577 578 static const SkGlyph& sk_getMetrics_glyph_next(SkGlyphCache* cache, 579 const char** text) { 580 SkASSERT(cache != nullptr); 581 SkASSERT(text != nullptr); 582 583 const uint16_t* ptr = *(const uint16_t**)text; 584 unsigned glyphID = *ptr; 585 ptr += 1; 586 *text = (const char*)ptr; 587 return cache->getGlyphIDMetrics(glyphID); 588 } 589 590 static const SkGlyph& sk_getAdvance_utf8_next(SkGlyphCache* cache, 591 const char** text) { 592 SkASSERT(cache != nullptr); 593 SkASSERT(text != nullptr); 594 595 return cache->getUnicharAdvance(SkUTF8_NextUnichar(text)); 596 } 597 598 static const SkGlyph& sk_getAdvance_utf16_next(SkGlyphCache* cache, 599 const char** text) { 600 SkASSERT(cache != nullptr); 601 SkASSERT(text != nullptr); 602 603 return cache->getUnicharAdvance(SkUTF16_NextUnichar((const uint16_t**)text)); 604 } 605 606 static const SkGlyph& sk_getAdvance_utf32_next(SkGlyphCache* cache, 607 const char** text) { 608 SkASSERT(cache != nullptr); 609 SkASSERT(text != nullptr); 610 611 const int32_t* ptr = *(const int32_t**)text; 612 SkUnichar uni = *ptr++; 613 *text = (const char*)ptr; 614 return cache->getUnicharAdvance(uni); 615 } 616 617 static const SkGlyph& sk_getAdvance_glyph_next(SkGlyphCache* cache, 618 const char** text) { 619 SkASSERT(cache != nullptr); 620 SkASSERT(text != nullptr); 621 622 const uint16_t* ptr = *(const uint16_t**)text; 623 unsigned glyphID = *ptr; 624 ptr += 1; 625 *text = (const char*)ptr; 626 return cache->getGlyphIDAdvance(glyphID); 627 } 628 629 SkPaint::GlyphCacheProc SkPaint::GetGlyphCacheProc(TextEncoding encoding, 630 bool isDevKern, 631 bool needFullMetrics) { 632 static const GlyphCacheProc gGlyphCacheProcs[] = { 633 sk_getMetrics_utf8_next, 634 sk_getMetrics_utf16_next, 635 sk_getMetrics_utf32_next, 636 sk_getMetrics_glyph_next, 637 638 sk_getAdvance_utf8_next, 639 sk_getAdvance_utf16_next, 640 sk_getAdvance_utf32_next, 641 sk_getAdvance_glyph_next, 642 }; 643 644 unsigned index = encoding; 645 646 if (!needFullMetrics && !isDevKern) { 647 index += 4; 648 } 649 650 SkASSERT(index < SK_ARRAY_COUNT(gGlyphCacheProcs)); 651 return gGlyphCacheProcs[index]; 652 } 653 654 /////////////////////////////////////////////////////////////////////////////// 655 656 #define TEXT_AS_PATHS_PAINT_FLAGS_TO_IGNORE ( \ 657 SkPaint::kDevKernText_Flag | \ 658 SkPaint::kLinearText_Flag | \ 659 SkPaint::kLCDRenderText_Flag | \ 660 SkPaint::kEmbeddedBitmapText_Flag | \ 661 SkPaint::kAutoHinting_Flag | \ 662 SkPaint::kGenA8FromLCD_Flag ) 663 664 SkScalar SkPaint::setupForAsPaths() { 665 uint32_t flags = this->getFlags(); 666 // clear the flags we don't care about 667 flags &= ~TEXT_AS_PATHS_PAINT_FLAGS_TO_IGNORE; 668 // set the flags we do care about 669 flags |= SkPaint::kSubpixelText_Flag; 670 671 this->setFlags(flags); 672 this->setHinting(SkPaint::kNo_Hinting); 673 674 SkScalar textSize = fTextSize; 675 this->setTextSize(kCanonicalTextSizeForPaths); 676 return textSize / kCanonicalTextSizeForPaths; 677 } 678 679 class SkCanonicalizePaint { 680 public: 681 SkCanonicalizePaint(const SkPaint& paint) : fPaint(&paint), fScale(0) { 682 if (paint.isLinearText() || SkDraw::ShouldDrawTextAsPaths(paint, SkMatrix::I())) { 683 SkPaint* p = fLazy.set(paint); 684 fScale = p->setupForAsPaths(); 685 fPaint = p; 686 } 687 } 688 689 const SkPaint& getPaint() const { return *fPaint; } 690 691 /** 692 * Returns 0 if the paint was unmodified, or the scale factor need to 693 * the original textSize 694 */ 695 SkScalar getScale() const { return fScale; } 696 697 private: 698 const SkPaint* fPaint; 699 SkScalar fScale; 700 SkTLazy<SkPaint> fLazy; 701 }; 702 703 static void set_bounds(const SkGlyph& g, SkRect* bounds) { 704 bounds->set(SkIntToScalar(g.fLeft), 705 SkIntToScalar(g.fTop), 706 SkIntToScalar(g.fLeft + g.fWidth), 707 SkIntToScalar(g.fTop + g.fHeight)); 708 } 709 710 static void join_bounds_x(const SkGlyph& g, SkRect* bounds, SkScalar dx) { 711 bounds->join(SkIntToScalar(g.fLeft) + dx, 712 SkIntToScalar(g.fTop), 713 SkIntToScalar(g.fLeft + g.fWidth) + dx, 714 SkIntToScalar(g.fTop + g.fHeight)); 715 } 716 717 static void join_bounds_y(const SkGlyph& g, SkRect* bounds, SkScalar dy) { 718 bounds->join(SkIntToScalar(g.fLeft), 719 SkIntToScalar(g.fTop) + dy, 720 SkIntToScalar(g.fLeft + g.fWidth), 721 SkIntToScalar(g.fTop + g.fHeight) + dy); 722 } 723 724 typedef void (*JoinBoundsProc)(const SkGlyph&, SkRect*, SkScalar); 725 726 // xyIndex is 0 for fAdvanceX or 1 for fAdvanceY 727 static SkScalar advance(const SkGlyph& glyph, int xyIndex) { 728 SkASSERT(0 == xyIndex || 1 == xyIndex); 729 return SkFloatToScalar((&glyph.fAdvanceX)[xyIndex]); 730 } 731 732 SkScalar SkPaint::measure_text(SkGlyphCache* cache, 733 const char* text, size_t byteLength, 734 int* count, SkRect* bounds) const { 735 SkASSERT(count); 736 if (byteLength == 0) { 737 *count = 0; 738 if (bounds) { 739 bounds->setEmpty(); 740 } 741 return 0; 742 } 743 744 GlyphCacheProc glyphCacheProc = SkPaint::GetGlyphCacheProc(this->getTextEncoding(), 745 this->isDevKernText(), 746 nullptr != bounds); 747 748 int xyIndex; 749 JoinBoundsProc joinBoundsProc; 750 if (this->isVerticalText()) { 751 xyIndex = 1; 752 joinBoundsProc = join_bounds_y; 753 } else { 754 xyIndex = 0; 755 joinBoundsProc = join_bounds_x; 756 } 757 758 int n = 1; 759 const char* stop = (const char*)text + byteLength; 760 const SkGlyph* g = &glyphCacheProc(cache, &text); 761 SkScalar x = advance(*g, xyIndex); 762 763 if (nullptr == bounds) { 764 if (this->isDevKernText()) { 765 for (; text < stop; n++) { 766 const int rsb = g->fRsbDelta; 767 g = &glyphCacheProc(cache, &text); 768 x += SkAutoKern_Adjust(rsb, g->fLsbDelta) + advance(*g, xyIndex); 769 } 770 } else { 771 for (; text < stop; n++) { 772 x += advance(glyphCacheProc(cache, &text), xyIndex); 773 } 774 } 775 } else { 776 set_bounds(*g, bounds); 777 if (this->isDevKernText()) { 778 for (; text < stop; n++) { 779 const int rsb = g->fRsbDelta; 780 g = &glyphCacheProc(cache, &text); 781 x += SkAutoKern_Adjust(rsb, g->fLsbDelta); 782 joinBoundsProc(*g, bounds, x); 783 x += advance(*g, xyIndex); 784 } 785 } else { 786 for (; text < stop; n++) { 787 g = &glyphCacheProc(cache, &text); 788 joinBoundsProc(*g, bounds, x); 789 x += advance(*g, xyIndex); 790 } 791 } 792 } 793 SkASSERT(text == stop); 794 795 *count = n; 796 return x; 797 } 798 799 SkScalar SkPaint::measureText(const void* textData, size_t length, SkRect* bounds) const { 800 const char* text = (const char*)textData; 801 SkASSERT(text != nullptr || length == 0); 802 803 SkCanonicalizePaint canon(*this); 804 const SkPaint& paint = canon.getPaint(); 805 SkScalar scale = canon.getScale(); 806 807 SkAutoGlyphCache autoCache(paint, nullptr, nullptr); 808 SkGlyphCache* cache = autoCache.getCache(); 809 810 SkScalar width = 0; 811 812 if (length > 0) { 813 int tempCount; 814 815 width = paint.measure_text(cache, text, length, &tempCount, bounds); 816 if (scale) { 817 width *= scale; 818 if (bounds) { 819 bounds->fLeft *= scale; 820 bounds->fTop *= scale; 821 bounds->fRight *= scale; 822 bounds->fBottom *= scale; 823 } 824 } 825 } else if (bounds) { 826 // ensure that even if we don't measure_text we still update the bounds 827 bounds->setEmpty(); 828 } 829 return width; 830 } 831 832 size_t SkPaint::breakText(const void* textD, size_t length, SkScalar maxWidth, 833 SkScalar* measuredWidth) const { 834 if (0 == length || 0 >= maxWidth) { 835 if (measuredWidth) { 836 *measuredWidth = 0; 837 } 838 return 0; 839 } 840 841 if (0 == fTextSize) { 842 if (measuredWidth) { 843 *measuredWidth = 0; 844 } 845 return length; 846 } 847 848 SkASSERT(textD != nullptr); 849 const char* text = (const char*)textD; 850 const char* stop = text + length; 851 852 SkCanonicalizePaint canon(*this); 853 const SkPaint& paint = canon.getPaint(); 854 SkScalar scale = canon.getScale(); 855 856 // adjust max in case we changed the textSize in paint 857 if (scale) { 858 maxWidth /= scale; 859 } 860 861 SkAutoGlyphCache autoCache(paint, nullptr, nullptr); 862 SkGlyphCache* cache = autoCache.getCache(); 863 864 GlyphCacheProc glyphCacheProc = SkPaint::GetGlyphCacheProc(paint.getTextEncoding(), 865 paint.isDevKernText(), 866 false); 867 const int xyIndex = paint.isVerticalText() ? 1 : 0; 868 SkScalar width = 0; 869 870 if (this->isDevKernText()) { 871 int rsb = 0; 872 while (text < stop) { 873 const char* curr = text; 874 const SkGlyph& g = glyphCacheProc(cache, &text); 875 SkScalar x = SkAutoKern_Adjust(rsb, g.fLsbDelta) + advance(g, xyIndex); 876 if ((width += x) > maxWidth) { 877 width -= x; 878 text = curr; 879 break; 880 } 881 rsb = g.fRsbDelta; 882 } 883 } else { 884 while (text < stop) { 885 const char* curr = text; 886 SkScalar x = advance(glyphCacheProc(cache, &text), xyIndex); 887 if ((width += x) > maxWidth) { 888 width -= x; 889 text = curr; 890 break; 891 } 892 } 893 } 894 895 if (measuredWidth) { 896 if (scale) { 897 width *= scale; 898 } 899 *measuredWidth = width; 900 } 901 902 // return the number of bytes measured 903 return text - stop + length; 904 } 905 906 /////////////////////////////////////////////////////////////////////////////// 907 908 static bool FontMetricsCacheProc(const SkGlyphCache* cache, void* context) { 909 *(SkPaint::FontMetrics*)context = cache->getFontMetrics(); 910 return false; // don't detach the cache 911 } 912 913 SkScalar SkPaint::getFontMetrics(FontMetrics* metrics, SkScalar zoom) const { 914 SkCanonicalizePaint canon(*this); 915 const SkPaint& paint = canon.getPaint(); 916 SkScalar scale = canon.getScale(); 917 918 SkMatrix zoomMatrix, *zoomPtr = nullptr; 919 if (zoom) { 920 zoomMatrix.setScale(zoom, zoom); 921 zoomPtr = &zoomMatrix; 922 } 923 924 FontMetrics storage; 925 if (nullptr == metrics) { 926 metrics = &storage; 927 } 928 929 SkAutoDescriptor ad; 930 SkScalerContextEffects effects; 931 932 auto desc = SkScalerContext::CreateDescriptorAndEffectsUsingPaint( 933 paint, nullptr, SkScalerContextFlags::kNone, zoomPtr, &ad, &effects); 934 935 SkGlyphCache::VisitCache(paint.getTypeface(), effects, desc, FontMetricsCacheProc, metrics); 936 937 if (scale) { 938 SkPaintPriv::ScaleFontMetrics(metrics, scale); 939 } 940 return metrics->fDescent - metrics->fAscent + metrics->fLeading; 941 } 942 943 /////////////////////////////////////////////////////////////////////////////// 944 945 static void set_bounds(const SkGlyph& g, SkRect* bounds, SkScalar scale) { 946 bounds->set(g.fLeft * scale, 947 g.fTop * scale, 948 (g.fLeft + g.fWidth) * scale, 949 (g.fTop + g.fHeight) * scale); 950 } 951 952 int SkPaint::getTextWidths(const void* textData, size_t byteLength, 953 SkScalar widths[], SkRect bounds[]) const { 954 if (0 == byteLength) { 955 return 0; 956 } 957 958 SkASSERT(textData); 959 960 if (nullptr == widths && nullptr == bounds) { 961 return this->countText(textData, byteLength); 962 } 963 964 SkCanonicalizePaint canon(*this); 965 const SkPaint& paint = canon.getPaint(); 966 SkScalar scale = canon.getScale(); 967 968 SkAutoGlyphCache autoCache(paint, nullptr, nullptr); 969 SkGlyphCache* cache = autoCache.getCache(); 970 GlyphCacheProc glyphCacheProc = SkPaint::GetGlyphCacheProc(paint.getTextEncoding(), 971 paint.isDevKernText(), 972 nullptr != bounds); 973 974 const char* text = (const char*)textData; 975 const char* stop = text + byteLength; 976 int count = 0; 977 const int xyIndex = paint.isVerticalText() ? 1 : 0; 978 979 if (this->isDevKernText()) { 980 // we adjust the widths returned here through auto-kerning 981 SkAutoKern autokern; 982 SkScalar prevWidth = 0; 983 984 if (scale) { 985 while (text < stop) { 986 const SkGlyph& g = glyphCacheProc(cache, &text); 987 if (widths) { 988 SkScalar adjust = autokern.adjust(g); 989 990 if (count > 0) { 991 *widths++ = (prevWidth + adjust) * scale; 992 } 993 prevWidth = advance(g, xyIndex); 994 } 995 if (bounds) { 996 set_bounds(g, bounds++, scale); 997 } 998 ++count; 999 } 1000 if (count > 0 && widths) { 1001 *widths = prevWidth * scale; 1002 } 1003 } else { 1004 while (text < stop) { 1005 const SkGlyph& g = glyphCacheProc(cache, &text); 1006 if (widths) { 1007 SkScalar adjust = autokern.adjust(g); 1008 1009 if (count > 0) { 1010 *widths++ = prevWidth + adjust; 1011 } 1012 prevWidth = advance(g, xyIndex); 1013 } 1014 if (bounds) { 1015 set_bounds(g, bounds++); 1016 } 1017 ++count; 1018 } 1019 if (count > 0 && widths) { 1020 *widths = prevWidth; 1021 } 1022 } 1023 } else { // no devkern 1024 if (scale) { 1025 while (text < stop) { 1026 const SkGlyph& g = glyphCacheProc(cache, &text); 1027 if (widths) { 1028 *widths++ = advance(g, xyIndex) * scale; 1029 } 1030 if (bounds) { 1031 set_bounds(g, bounds++, scale); 1032 } 1033 ++count; 1034 } 1035 } else { 1036 while (text < stop) { 1037 const SkGlyph& g = glyphCacheProc(cache, &text); 1038 if (widths) { 1039 *widths++ = advance(g, xyIndex); 1040 } 1041 if (bounds) { 1042 set_bounds(g, bounds++); 1043 } 1044 ++count; 1045 } 1046 } 1047 } 1048 1049 SkASSERT(text == stop); 1050 return count; 1051 } 1052 1053 /////////////////////////////////////////////////////////////////////////////// 1054 1055 #include "SkDraw.h" 1056 1057 void SkPaint::getTextPath(const void* textData, size_t length, 1058 SkScalar x, SkScalar y, SkPath* path) const { 1059 SkASSERT(length == 0 || textData != nullptr); 1060 1061 const char* text = (const char*)textData; 1062 if (text == nullptr || length == 0 || path == nullptr) { 1063 return; 1064 } 1065 1066 SkTextToPathIter iter(text, length, *this, false); 1067 SkMatrix matrix; 1068 SkScalar prevXPos = 0; 1069 1070 matrix.setScale(iter.getPathScale(), iter.getPathScale()); 1071 matrix.postTranslate(x, y); 1072 path->reset(); 1073 1074 SkScalar xpos; 1075 const SkPath* iterPath; 1076 while (iter.next(&iterPath, &xpos)) { 1077 matrix.postTranslate(xpos - prevXPos, 0); 1078 if (iterPath) { 1079 path->addPath(*iterPath, matrix); 1080 } 1081 prevXPos = xpos; 1082 } 1083 } 1084 1085 void SkPaint::getPosTextPath(const void* textData, size_t length, 1086 const SkPoint pos[], SkPath* path) const { 1087 SkASSERT(length == 0 || textData != nullptr); 1088 1089 const char* text = (const char*)textData; 1090 if (text == nullptr || length == 0 || path == nullptr) { 1091 return; 1092 } 1093 1094 SkTextToPathIter iter(text, length, *this, false); 1095 SkMatrix matrix; 1096 SkPoint prevPos; 1097 prevPos.set(0, 0); 1098 1099 matrix.setScale(iter.getPathScale(), iter.getPathScale()); 1100 path->reset(); 1101 1102 unsigned int i = 0; 1103 const SkPath* iterPath; 1104 while (iter.next(&iterPath, nullptr)) { 1105 matrix.postTranslate(pos[i].fX - prevPos.fX, pos[i].fY - prevPos.fY); 1106 if (iterPath) { 1107 path->addPath(*iterPath, matrix); 1108 } 1109 prevPos = pos[i]; 1110 i++; 1111 } 1112 } 1113 1114 template <SkTextInterceptsIter::TextType TextType, typename Func> 1115 int GetTextIntercepts(const SkPaint& paint, const void* text, size_t length, 1116 const SkScalar bounds[2], SkScalar* array, Func posMaker) { 1117 SkASSERT(length == 0 || text != nullptr); 1118 if (!length) { 1119 return 0; 1120 } 1121 1122 const SkPoint pos0 = posMaker(0); 1123 SkTextInterceptsIter iter(static_cast<const char*>(text), length, paint, bounds, 1124 pos0.x(), pos0.y(), TextType); 1125 1126 int i = 0; 1127 int count = 0; 1128 while (iter.next(array, &count)) { 1129 if (TextType == SkTextInterceptsIter::TextType::kPosText) { 1130 const SkPoint pos = posMaker(++i); 1131 iter.setPosition(pos.x(), pos.y()); 1132 } 1133 } 1134 1135 return count; 1136 } 1137 1138 int SkPaint::getTextIntercepts(const void* textData, size_t length, 1139 SkScalar x, SkScalar y, const SkScalar bounds[2], 1140 SkScalar* array) const { 1141 1142 return GetTextIntercepts<SkTextInterceptsIter::TextType::kText>( 1143 *this, textData, length, bounds, array, [&x, &y] (int) -> SkPoint { 1144 return SkPoint::Make(x, y); 1145 }); 1146 } 1147 1148 int SkPaint::getPosTextIntercepts(const void* textData, size_t length, const SkPoint pos[], 1149 const SkScalar bounds[2], SkScalar* array) const { 1150 1151 return GetTextIntercepts<SkTextInterceptsIter::TextType::kPosText>( 1152 *this, textData, length, bounds, array, [&pos] (int i) -> SkPoint { 1153 return pos[i]; 1154 }); 1155 } 1156 1157 int SkPaint::getPosTextHIntercepts(const void* textData, size_t length, const SkScalar xpos[], 1158 SkScalar constY, const SkScalar bounds[2], 1159 SkScalar* array) const { 1160 1161 return GetTextIntercepts<SkTextInterceptsIter::TextType::kPosText>( 1162 *this, textData, length, bounds, array, [&xpos, &constY] (int i) -> SkPoint { 1163 return SkPoint::Make(xpos[i], constY); 1164 }); 1165 } 1166 1167 int SkPaint::getTextBlobIntercepts(const SkTextBlob* blob, const SkScalar bounds[2], 1168 SkScalar* intervals) const { 1169 int count = 0; 1170 SkPaint runPaint(*this); 1171 1172 SkTextBlobRunIterator it(blob); 1173 while (!it.done()) { 1174 it.applyFontToPaint(&runPaint); 1175 const size_t runByteCount = it.glyphCount() * sizeof(SkGlyphID); 1176 SkScalar* runIntervals = intervals ? intervals + count : nullptr; 1177 1178 switch (it.positioning()) { 1179 case SkTextBlob::kDefault_Positioning: 1180 count += runPaint.getTextIntercepts(it.glyphs(), runByteCount, it.offset().x(), 1181 it.offset().y(), bounds, runIntervals); 1182 break; 1183 case SkTextBlob::kHorizontal_Positioning: 1184 count += runPaint.getPosTextHIntercepts(it.glyphs(), runByteCount, it.pos(), 1185 it.offset().y(), bounds, runIntervals); 1186 break; 1187 case SkTextBlob::kFull_Positioning: 1188 count += runPaint.getPosTextIntercepts(it.glyphs(), runByteCount, 1189 reinterpret_cast<const SkPoint*>(it.pos()), 1190 bounds, runIntervals); 1191 break; 1192 } 1193 1194 it.next(); 1195 } 1196 1197 return count; 1198 } 1199 1200 SkRect SkPaint::getFontBounds() const { 1201 SkMatrix m; 1202 m.setScale(fTextSize * fTextScaleX, fTextSize); 1203 m.postSkew(fTextSkewX, 0); 1204 1205 SkTypeface* typeface = this->getTypeface(); 1206 if (nullptr == typeface) { 1207 typeface = SkTypeface::GetDefaultTypeface(); 1208 } 1209 1210 SkRect bounds; 1211 m.mapRect(&bounds, typeface->getBounds()); 1212 return bounds; 1213 } 1214 1215 // return true if the paint is just a single color (i.e. not a shader). If its 1216 // a shader, then we can't compute a const luminance for it :( 1217 static bool justAColor(const SkPaint& paint, SkColor* color) { 1218 SkColor c = paint.getColor(); 1219 1220 const auto* shader = as_SB(paint.getShader()); 1221 if (shader && !shader->asLuminanceColor(&c)) { 1222 return false; 1223 } 1224 if (paint.getColorFilter()) { 1225 c = paint.getColorFilter()->filterColor(c); 1226 } 1227 if (color) { 1228 *color = c; 1229 } 1230 return true; 1231 } 1232 1233 SkColor SkPaint::computeLuminanceColor() const { 1234 SkColor c; 1235 if (!justAColor(*this, &c)) { 1236 c = SkColorSetRGB(0x7F, 0x80, 0x7F); 1237 } 1238 return c; 1239 } 1240 1241 #define assert_byte(x) SkASSERT(0 == ((x) >> 8)) 1242 1243 /** 1244 * In order to call cachedDeviceLuminance, cachedPaintLuminance, or 1245 * cachedMaskGamma the caller must hold the gMaskGammaCacheMutex and continue 1246 * to hold it until the returned pointer is refed or forgotten. 1247 */ 1248 SK_DECLARE_STATIC_MUTEX(gMaskGammaCacheMutex); 1249 1250 static SkMaskGamma* gLinearMaskGamma = nullptr; 1251 static SkMaskGamma* gMaskGamma = nullptr; 1252 static SkScalar gContrast = SK_ScalarMin; 1253 static SkScalar gPaintGamma = SK_ScalarMin; 1254 static SkScalar gDeviceGamma = SK_ScalarMin; 1255 /** 1256 * The caller must hold the gMaskGammaCacheMutex and continue to hold it until 1257 * the returned SkMaskGamma pointer is refed or forgotten. 1258 */ 1259 static const SkMaskGamma& cachedMaskGamma(SkScalar contrast, SkScalar paintGamma, SkScalar deviceGamma) { 1260 gMaskGammaCacheMutex.assertHeld(); 1261 if (0 == contrast && SK_Scalar1 == paintGamma && SK_Scalar1 == deviceGamma) { 1262 if (nullptr == gLinearMaskGamma) { 1263 gLinearMaskGamma = new SkMaskGamma; 1264 } 1265 return *gLinearMaskGamma; 1266 } 1267 if (gContrast != contrast || gPaintGamma != paintGamma || gDeviceGamma != deviceGamma) { 1268 SkSafeUnref(gMaskGamma); 1269 gMaskGamma = new SkMaskGamma(contrast, paintGamma, deviceGamma); 1270 gContrast = contrast; 1271 gPaintGamma = paintGamma; 1272 gDeviceGamma = deviceGamma; 1273 } 1274 return *gMaskGamma; 1275 } 1276 1277 1278 /** 1279 * Expands fDeviceGamma, fPaintGamma, fContrast, and fLumBits into a mask pre-blend. 1280 */ 1281 //static 1282 SkMaskGamma::PreBlend SkScalerContext::GetMaskPreBlend(const SkScalerContextRec& rec) { 1283 SkAutoMutexAcquire ama(gMaskGammaCacheMutex); 1284 const SkMaskGamma& maskGamma = cachedMaskGamma(rec.getContrast(), 1285 rec.getPaintGamma(), 1286 rec.getDeviceGamma()); 1287 return maskGamma.preBlend(rec.getLuminanceColor()); 1288 } 1289 1290 size_t SkScalerContext::GetGammaLUTSize(SkScalar contrast, SkScalar paintGamma, 1291 SkScalar deviceGamma, int* width, int* height) { 1292 SkAutoMutexAcquire ama(gMaskGammaCacheMutex); 1293 const SkMaskGamma& maskGamma = cachedMaskGamma(contrast, 1294 paintGamma, 1295 deviceGamma); 1296 1297 maskGamma.getGammaTableDimensions(width, height); 1298 size_t size = (*width)*(*height)*sizeof(uint8_t); 1299 1300 return size; 1301 } 1302 1303 bool SkScalerContext::GetGammaLUTData(SkScalar contrast, SkScalar paintGamma, SkScalar deviceGamma, 1304 uint8_t* data) { 1305 SkAutoMutexAcquire ama(gMaskGammaCacheMutex); 1306 const SkMaskGamma& maskGamma = cachedMaskGamma(contrast, 1307 paintGamma, 1308 deviceGamma); 1309 const uint8_t* gammaTables = maskGamma.getGammaTables(); 1310 if (!gammaTables) { 1311 return false; 1312 } 1313 1314 int width, height; 1315 maskGamma.getGammaTableDimensions(&width, &height); 1316 size_t size = width*height * sizeof(uint8_t); 1317 memcpy(data, gammaTables, size); 1318 return true; 1319 } 1320 1321 /////////////////////////////////////////////////////////////////////////////// 1322 1323 #include "SkStream.h" 1324 1325 static uintptr_t asint(const void* p) { 1326 return reinterpret_cast<uintptr_t>(p); 1327 } 1328 1329 static uint32_t pack_4(unsigned a, unsigned b, unsigned c, unsigned d) { 1330 SkASSERT(a == (uint8_t)a); 1331 SkASSERT(b == (uint8_t)b); 1332 SkASSERT(c == (uint8_t)c); 1333 SkASSERT(d == (uint8_t)d); 1334 return (a << 24) | (b << 16) | (c << 8) | d; 1335 } 1336 1337 #ifdef SK_DEBUG 1338 static void ASSERT_FITS_IN(uint32_t value, int bitCount) { 1339 SkASSERT(bitCount > 0 && bitCount <= 32); 1340 uint32_t mask = ~0U; 1341 mask >>= (32 - bitCount); 1342 SkASSERT(0 == (value & ~mask)); 1343 } 1344 #else 1345 #define ASSERT_FITS_IN(value, bitcount) 1346 #endif 1347 1348 enum FlatFlags { 1349 kHasTypeface_FlatFlag = 0x1, 1350 kHasEffects_FlatFlag = 0x2, 1351 1352 kFlatFlagMask = 0x3, 1353 }; 1354 1355 enum BitsPerField { 1356 kFlags_BPF = 16, 1357 kHint_BPF = 2, 1358 kAlign_BPF = 2, 1359 kFilter_BPF = 2, 1360 kFlatFlags_BPF = 3, 1361 }; 1362 1363 static inline int BPF_Mask(int bits) { 1364 return (1 << bits) - 1; 1365 } 1366 1367 static uint32_t pack_paint_flags(unsigned flags, unsigned hint, unsigned align, 1368 unsigned filter, unsigned flatFlags) { 1369 ASSERT_FITS_IN(flags, kFlags_BPF); 1370 ASSERT_FITS_IN(hint, kHint_BPF); 1371 ASSERT_FITS_IN(align, kAlign_BPF); 1372 ASSERT_FITS_IN(filter, kFilter_BPF); 1373 ASSERT_FITS_IN(flatFlags, kFlatFlags_BPF); 1374 1375 // left-align the fields of "known" size, and right-align the last (flatFlags) so it can easly 1376 // add more bits in the future. 1377 return (flags << 16) | (hint << 14) | (align << 12) | (filter << 10) | flatFlags; 1378 } 1379 1380 static FlatFlags unpack_paint_flags(SkPaint* paint, uint32_t packed) { 1381 paint->setFlags(packed >> 16); 1382 paint->setHinting((SkPaint::Hinting)((packed >> 14) & BPF_Mask(kHint_BPF))); 1383 paint->setTextAlign((SkPaint::Align)((packed >> 12) & BPF_Mask(kAlign_BPF))); 1384 paint->setFilterQuality((SkFilterQuality)((packed >> 10) & BPF_Mask(kFilter_BPF))); 1385 return (FlatFlags)(packed & kFlatFlagMask); 1386 } 1387 1388 /* To save space/time, we analyze the paint, and write a truncated version of 1389 it if there are not tricky elements like shaders, etc. 1390 */ 1391 void SkPaint::flatten(SkWriteBuffer& buffer) const { 1392 SkTypeface* tf = this->getTypeface(); 1393 if (!tf) { 1394 // We force recording our typeface, even if its "default" since the receiver process 1395 // may have a different notion of default. 1396 tf = SkTypeface::GetDefaultTypeface(); 1397 SkASSERT(tf); 1398 } 1399 1400 uint8_t flatFlags = kHasTypeface_FlatFlag; 1401 1402 if (asint(this->getPathEffect()) | 1403 asint(this->getShader()) | 1404 asint(this->getMaskFilter()) | 1405 asint(this->getColorFilter()) | 1406 asint(this->getLooper()) | 1407 asint(this->getImageFilter())) { 1408 flatFlags |= kHasEffects_FlatFlag; 1409 } 1410 1411 buffer.writeScalar(this->getTextSize()); 1412 buffer.writeScalar(this->getTextScaleX()); 1413 buffer.writeScalar(this->getTextSkewX()); 1414 buffer.writeScalar(this->getStrokeWidth()); 1415 buffer.writeScalar(this->getStrokeMiter()); 1416 buffer.writeColor(this->getColor()); 1417 1418 buffer.writeUInt(pack_paint_flags(this->getFlags(), this->getHinting(), this->getTextAlign(), 1419 this->getFilterQuality(), flatFlags)); 1420 buffer.writeUInt(pack_4(this->getStrokeCap(), this->getStrokeJoin(), 1421 (this->getStyle() << 4) | this->getTextEncoding(), 1422 fBlendMode)); 1423 1424 buffer.writeTypeface(tf); 1425 1426 if (flatFlags & kHasEffects_FlatFlag) { 1427 buffer.writeFlattenable(this->getPathEffect()); 1428 buffer.writeFlattenable(this->getShader()); 1429 buffer.writeFlattenable(this->getMaskFilter()); 1430 buffer.writeFlattenable(this->getColorFilter()); 1431 buffer.write32(0); // use to be SkRasterizer 1432 buffer.writeFlattenable(this->getLooper()); 1433 buffer.writeFlattenable(this->getImageFilter()); 1434 } 1435 } 1436 1437 bool SkPaint::unflatten(SkReadBuffer& buffer) { 1438 SkSafeRange safe; 1439 1440 this->setTextSize(buffer.readScalar()); 1441 this->setTextScaleX(buffer.readScalar()); 1442 this->setTextSkewX(buffer.readScalar()); 1443 this->setStrokeWidth(buffer.readScalar()); 1444 this->setStrokeMiter(buffer.readScalar()); 1445 this->setColor(buffer.readColor()); 1446 1447 unsigned flatFlags = unpack_paint_flags(this, buffer.readUInt()); 1448 1449 uint32_t tmp = buffer.readUInt(); 1450 this->setStrokeCap(safe.checkLE((tmp >> 24) & 0xFF, kLast_Cap)); 1451 this->setStrokeJoin(safe.checkLE((tmp >> 16) & 0xFF, kLast_Join)); 1452 this->setStyle(safe.checkLE((tmp >> 12) & 0xF, kStrokeAndFill_Style)); 1453 this->setTextEncoding(safe.checkLE((tmp >> 8) & 0xF, kGlyphID_TextEncoding)); 1454 this->setBlendMode(safe.checkLE(tmp & 0xFF, SkBlendMode::kLastMode)); 1455 1456 if (flatFlags & kHasTypeface_FlatFlag) { 1457 this->setTypeface(buffer.readTypeface()); 1458 } else { 1459 this->setTypeface(nullptr); 1460 } 1461 1462 if (flatFlags & kHasEffects_FlatFlag) { 1463 this->setPathEffect(buffer.readPathEffect()); 1464 this->setShader(buffer.readShader()); 1465 this->setMaskFilter(buffer.readMaskFilter()); 1466 this->setColorFilter(buffer.readColorFilter()); 1467 (void)buffer.read32(); // use to be SkRasterizer 1468 this->setLooper(buffer.readDrawLooper()); 1469 this->setImageFilter(buffer.readImageFilter()); 1470 } else { 1471 this->setPathEffect(nullptr); 1472 this->setShader(nullptr); 1473 this->setMaskFilter(nullptr); 1474 this->setColorFilter(nullptr); 1475 this->setLooper(nullptr); 1476 this->setImageFilter(nullptr); 1477 } 1478 1479 if (!buffer.validate(safe)) { 1480 this->reset(); 1481 return false; 1482 } 1483 return true; 1484 } 1485 1486 /////////////////////////////////////////////////////////////////////////////// 1487 1488 bool SkPaint::getFillPath(const SkPath& src, SkPath* dst, const SkRect* cullRect, 1489 SkScalar resScale) const { 1490 if (!src.isFinite()) { 1491 dst->reset(); 1492 return false; 1493 } 1494 1495 SkStrokeRec rec(*this, resScale); 1496 1497 const SkPath* srcPtr = &src; 1498 SkPath tmpPath; 1499 1500 if (fPathEffect && fPathEffect->filterPath(&tmpPath, src, &rec, cullRect)) { 1501 srcPtr = &tmpPath; 1502 } 1503 1504 if (!rec.applyToPath(dst, *srcPtr)) { 1505 if (srcPtr == &tmpPath) { 1506 // If path's were copy-on-write, this trick would not be needed. 1507 // As it is, we want to save making a deep-copy from tmpPath -> dst 1508 // since we know we're just going to delete tmpPath when we return, 1509 // so the swap saves that copy. 1510 dst->swap(tmpPath); 1511 } else { 1512 *dst = *srcPtr; 1513 } 1514 } 1515 1516 if (!dst->isFinite()) { 1517 dst->reset(); 1518 return false; 1519 } 1520 return !rec.isHairlineStyle(); 1521 } 1522 1523 bool SkPaint::canComputeFastBounds() const { 1524 if (this->getLooper()) { 1525 return this->getLooper()->canComputeFastBounds(*this); 1526 } 1527 if (this->getImageFilter() && !this->getImageFilter()->canComputeFastBounds()) { 1528 return false; 1529 } 1530 return true; 1531 } 1532 1533 const SkRect& SkPaint::doComputeFastBounds(const SkRect& origSrc, 1534 SkRect* storage, 1535 Style style) const { 1536 SkASSERT(storage); 1537 1538 const SkRect* src = &origSrc; 1539 1540 if (this->getLooper()) { 1541 SkASSERT(this->getLooper()->canComputeFastBounds(*this)); 1542 this->getLooper()->computeFastBounds(*this, *src, storage); 1543 return *storage; 1544 } 1545 1546 SkRect tmpSrc; 1547 if (this->getPathEffect()) { 1548 this->getPathEffect()->computeFastBounds(&tmpSrc, origSrc); 1549 src = &tmpSrc; 1550 } 1551 1552 SkScalar radius = SkStrokeRec::GetInflationRadius(*this, style); 1553 *storage = src->makeOutset(radius, radius); 1554 1555 if (this->getMaskFilter()) { 1556 as_MFB(this->getMaskFilter())->computeFastBounds(*storage, storage); 1557 } 1558 1559 if (this->getImageFilter()) { 1560 *storage = this->getImageFilter()->computeFastBounds(*storage); 1561 } 1562 1563 return *storage; 1564 } 1565 1566 #ifndef SK_IGNORE_TO_STRING 1567 1568 void SkPaint::toString(SkString* str) const { 1569 str->append("<dl><dt>SkPaint:</dt><dd><dl>"); 1570 1571 SkTypeface* typeface = this->getTypeface(); 1572 if (typeface) { 1573 SkDynamicMemoryWStream ostream; 1574 typeface->serialize(&ostream); 1575 std::unique_ptr<SkStreamAsset> istream(ostream.detachAsStream()); 1576 1577 SkFontDescriptor descriptor; 1578 if (!SkFontDescriptor::Deserialize(istream.get(), &descriptor)) { 1579 str->append("<dt>FontDescriptor deserialization failed</dt>"); 1580 } else { 1581 str->append("<dt>Font Family Name:</dt><dd>"); 1582 str->append(descriptor.getFamilyName()); 1583 str->append("</dd><dt>Font Full Name:</dt><dd>"); 1584 str->append(descriptor.getFullName()); 1585 str->append("</dd><dt>Font PS Name:</dt><dd>"); 1586 str->append(descriptor.getPostscriptName()); 1587 str->append("</dd>"); 1588 } 1589 } 1590 1591 str->append("<dt>TextSize:</dt><dd>"); 1592 str->appendScalar(this->getTextSize()); 1593 str->append("</dd>"); 1594 1595 str->append("<dt>TextScaleX:</dt><dd>"); 1596 str->appendScalar(this->getTextScaleX()); 1597 str->append("</dd>"); 1598 1599 str->append("<dt>TextSkewX:</dt><dd>"); 1600 str->appendScalar(this->getTextSkewX()); 1601 str->append("</dd>"); 1602 1603 SkPathEffect* pathEffect = this->getPathEffect(); 1604 if (pathEffect) { 1605 str->append("<dt>PathEffect:</dt><dd>"); 1606 pathEffect->toString(str); 1607 str->append("</dd>"); 1608 } 1609 1610 if (const auto* shader = as_SB(this->getShader())) { 1611 str->append("<dt>Shader:</dt><dd>"); 1612 shader->toString(str); 1613 str->append("</dd>"); 1614 } 1615 1616 if (!this->isSrcOver()) { 1617 str->appendf("<dt>Xfermode:</dt><dd>%d</dd>", fBlendMode); 1618 } 1619 1620 SkMaskFilter* maskFilter = this->getMaskFilter(); 1621 if (maskFilter) { 1622 str->append("<dt>MaskFilter:</dt><dd>"); 1623 as_MFB(maskFilter)->toString(str); 1624 str->append("</dd>"); 1625 } 1626 1627 SkColorFilter* colorFilter = this->getColorFilter(); 1628 if (colorFilter) { 1629 str->append("<dt>ColorFilter:</dt><dd>"); 1630 colorFilter->toString(str); 1631 str->append("</dd>"); 1632 } 1633 1634 SkDrawLooper* looper = this->getLooper(); 1635 if (looper) { 1636 str->append("<dt>DrawLooper:</dt><dd>"); 1637 looper->toString(str); 1638 str->append("</dd>"); 1639 } 1640 1641 SkImageFilter* imageFilter = this->getImageFilter(); 1642 if (imageFilter) { 1643 str->append("<dt>ImageFilter:</dt><dd>"); 1644 imageFilter->toString(str); 1645 str->append("</dd>"); 1646 } 1647 1648 str->append("<dt>Color:</dt><dd>0x"); 1649 SkColor color = this->getColor(); 1650 str->appendHex(color); 1651 str->append("</dd>"); 1652 1653 str->append("<dt>Stroke Width:</dt><dd>"); 1654 str->appendScalar(this->getStrokeWidth()); 1655 str->append("</dd>"); 1656 1657 str->append("<dt>Stroke Miter:</dt><dd>"); 1658 str->appendScalar(this->getStrokeMiter()); 1659 str->append("</dd>"); 1660 1661 str->append("<dt>Flags:</dt><dd>("); 1662 if (this->getFlags()) { 1663 bool needSeparator = false; 1664 SkAddFlagToString(str, this->isAntiAlias(), "AntiAlias", &needSeparator); 1665 SkAddFlagToString(str, this->isDither(), "Dither", &needSeparator); 1666 SkAddFlagToString(str, this->isFakeBoldText(), "FakeBoldText", &needSeparator); 1667 SkAddFlagToString(str, this->isLinearText(), "LinearText", &needSeparator); 1668 SkAddFlagToString(str, this->isSubpixelText(), "SubpixelText", &needSeparator); 1669 SkAddFlagToString(str, this->isDevKernText(), "DevKernText", &needSeparator); 1670 SkAddFlagToString(str, this->isLCDRenderText(), "LCDRenderText", &needSeparator); 1671 SkAddFlagToString(str, this->isEmbeddedBitmapText(), 1672 "EmbeddedBitmapText", &needSeparator); 1673 SkAddFlagToString(str, this->isAutohinted(), "Autohinted", &needSeparator); 1674 SkAddFlagToString(str, this->isVerticalText(), "VerticalText", &needSeparator); 1675 SkAddFlagToString(str, SkToBool(this->getFlags() & SkPaint::kGenA8FromLCD_Flag), 1676 "GenA8FromLCD", &needSeparator); 1677 } else { 1678 str->append("None"); 1679 } 1680 str->append(")</dd>"); 1681 1682 str->append("<dt>FilterLevel:</dt><dd>"); 1683 static const char* gFilterQualityStrings[] = { "None", "Low", "Medium", "High" }; 1684 str->append(gFilterQualityStrings[this->getFilterQuality()]); 1685 str->append("</dd>"); 1686 1687 str->append("<dt>TextAlign:</dt><dd>"); 1688 static const char* gTextAlignStrings[SkPaint::kAlignCount] = { "Left", "Center", "Right" }; 1689 str->append(gTextAlignStrings[this->getTextAlign()]); 1690 str->append("</dd>"); 1691 1692 str->append("<dt>CapType:</dt><dd>"); 1693 static const char* gStrokeCapStrings[SkPaint::kCapCount] = { "Butt", "Round", "Square" }; 1694 str->append(gStrokeCapStrings[this->getStrokeCap()]); 1695 str->append("</dd>"); 1696 1697 str->append("<dt>JoinType:</dt><dd>"); 1698 static const char* gJoinStrings[SkPaint::kJoinCount] = { "Miter", "Round", "Bevel" }; 1699 str->append(gJoinStrings[this->getStrokeJoin()]); 1700 str->append("</dd>"); 1701 1702 str->append("<dt>Style:</dt><dd>"); 1703 static const char* gStyleStrings[SkPaint::kStyleCount] = { "Fill", "Stroke", "StrokeAndFill" }; 1704 str->append(gStyleStrings[this->getStyle()]); 1705 str->append("</dd>"); 1706 1707 str->append("<dt>TextEncoding:</dt><dd>"); 1708 static const char* gTextEncodingStrings[] = { "UTF8", "UTF16", "UTF32", "GlyphID" }; 1709 str->append(gTextEncodingStrings[this->getTextEncoding()]); 1710 str->append("</dd>"); 1711 1712 str->append("<dt>Hinting:</dt><dd>"); 1713 static const char* gHintingStrings[] = { "None", "Slight", "Normal", "Full" }; 1714 str->append(gHintingStrings[this->getHinting()]); 1715 str->append("</dd>"); 1716 1717 str->append("</dd></dl></dl>"); 1718 } 1719 #endif 1720 1721 /////////////////////////////////////////////////////////////////////////////// 1722 1723 static bool has_thick_frame(const SkPaint& paint) { 1724 return paint.getStrokeWidth() > 0 && 1725 paint.getStyle() != SkPaint::kFill_Style; 1726 } 1727 1728 SkTextBaseIter::SkTextBaseIter(const char text[], size_t length, 1729 const SkPaint& paint, 1730 bool applyStrokeAndPathEffects) 1731 : fPaint(paint) { 1732 fGlyphCacheProc = SkPaint::GetGlyphCacheProc(paint.getTextEncoding(), 1733 paint.isDevKernText(), 1734 true); 1735 1736 fPaint.setLinearText(true); 1737 fPaint.setMaskFilter(nullptr); // don't want this affecting our path-cache lookup 1738 1739 if (fPaint.getPathEffect() == nullptr && !has_thick_frame(fPaint)) { 1740 applyStrokeAndPathEffects = false; 1741 } 1742 1743 // can't use our canonical size if we need to apply patheffects 1744 if (fPaint.getPathEffect() == nullptr) { 1745 fPaint.setTextSize(SkIntToScalar(SkPaint::kCanonicalTextSizeForPaths)); 1746 fScale = paint.getTextSize() / SkPaint::kCanonicalTextSizeForPaths; 1747 if (has_thick_frame(fPaint)) { 1748 fPaint.setStrokeWidth(fPaint.getStrokeWidth() / fScale); 1749 } 1750 } else { 1751 fScale = SK_Scalar1; 1752 } 1753 1754 if (!applyStrokeAndPathEffects) { 1755 fPaint.setStyle(SkPaint::kFill_Style); 1756 fPaint.setPathEffect(nullptr); 1757 } 1758 1759 // SRGBTODO: Is this correct? 1760 fCache = SkGlyphCache::DetachCacheUsingPaint(fPaint, nullptr, 1761 SkScalerContextFlags::kFakeGammaAndBoostContrast, 1762 nullptr); 1763 1764 SkPaint::Style style = SkPaint::kFill_Style; 1765 sk_sp<SkPathEffect> pe; 1766 1767 if (!applyStrokeAndPathEffects) { 1768 style = paint.getStyle(); // restore 1769 pe = paint.refPathEffect(); // restore 1770 } 1771 fPaint.setStyle(style); 1772 fPaint.setPathEffect(pe); 1773 fPaint.setMaskFilter(paint.refMaskFilter()); // restore 1774 1775 // now compute fXOffset if needed 1776 1777 SkScalar xOffset = 0; 1778 if (paint.getTextAlign() != SkPaint::kLeft_Align) { // need to measure first 1779 int count; 1780 SkScalar width = fPaint.measure_text(fCache, text, length, &count, nullptr) * fScale; 1781 if (paint.getTextAlign() == SkPaint::kCenter_Align) { 1782 width = SkScalarHalf(width); 1783 } 1784 xOffset = -width; 1785 } 1786 fXPos = xOffset; 1787 fPrevAdvance = 0; 1788 1789 fText = text; 1790 fStop = text + length; 1791 1792 fXYIndex = paint.isVerticalText() ? 1 : 0; 1793 } 1794 1795 SkTextBaseIter::~SkTextBaseIter() { 1796 SkGlyphCache::AttachCache(fCache); 1797 } 1798 1799 bool SkTextToPathIter::next(const SkPath** path, SkScalar* xpos) { 1800 if (fText < fStop) { 1801 const SkGlyph& glyph = fGlyphCacheProc(fCache, &fText); 1802 1803 fXPos += (fPrevAdvance + fAutoKern.adjust(glyph)) * fScale; 1804 fPrevAdvance = advance(glyph, fXYIndex); // + fPaint.getTextTracking(); 1805 1806 if (glyph.fWidth) { 1807 if (path) { 1808 *path = fCache->findPath(glyph); 1809 } 1810 } else { 1811 if (path) { 1812 *path = nullptr; 1813 } 1814 } 1815 if (xpos) { 1816 *xpos = fXPos; 1817 } 1818 return true; 1819 } 1820 return false; 1821 } 1822 1823 bool SkTextInterceptsIter::next(SkScalar* array, int* count) { 1824 const SkGlyph& glyph = fGlyphCacheProc(fCache, &fText); 1825 fXPos += (fPrevAdvance + fAutoKern.adjust(glyph)) * fScale; 1826 fPrevAdvance = advance(glyph, fXYIndex); // + fPaint.getTextTracking(); 1827 if (fCache->findPath(glyph)) { 1828 fCache->findIntercepts(fBounds, fScale, fXPos, SkToBool(fXYIndex), 1829 const_cast<SkGlyph*>(&glyph), array, count); 1830 } 1831 return fText < fStop; 1832 } 1833 1834 /////////////////////////////////////////////////////////////////////////////// 1835 1836 // return true if the filter exists, and may affect alpha 1837 static bool affects_alpha(const SkColorFilter* cf) { 1838 return cf && !(cf->getFlags() & SkColorFilter::kAlphaUnchanged_Flag); 1839 } 1840 1841 // return true if the filter exists, and may affect alpha 1842 static bool affects_alpha(const SkImageFilter* imf) { 1843 // TODO: check if we should allow imagefilters to broadcast that they don't affect alpha 1844 // ala colorfilters 1845 return imf != nullptr; 1846 } 1847 1848 bool SkPaint::nothingToDraw() const { 1849 if (fDrawLooper) { 1850 return false; 1851 } 1852 switch ((SkBlendMode)fBlendMode) { 1853 case SkBlendMode::kSrcOver: 1854 case SkBlendMode::kSrcATop: 1855 case SkBlendMode::kDstOut: 1856 case SkBlendMode::kDstOver: 1857 case SkBlendMode::kPlus: 1858 if (0 == this->getAlpha()) { 1859 return !affects_alpha(fColorFilter.get()) && !affects_alpha(fImageFilter.get()); 1860 } 1861 break; 1862 case SkBlendMode::kDst: 1863 return true; 1864 default: 1865 break; 1866 } 1867 return false; 1868 } 1869 1870 uint32_t SkPaint::getHash() const { 1871 // We're going to hash 7 pointers and 7 32-bit values, finishing up with fBitfields, 1872 // so fBitfields should be 7 pointers and 6 32-bit values from the start. 1873 static_assert(offsetof(SkPaint, fBitfields) == 7 * sizeof(void*) + 7 * sizeof(uint32_t), 1874 "SkPaint_notPackedTightly"); 1875 return SkOpts::hash(reinterpret_cast<const uint32_t*>(this), 1876 offsetof(SkPaint, fBitfields) + sizeof(fBitfields)); 1877 } 1878