| /* |
| * Copyright 2014 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "gm.h" |
| #include "SkCanvas.h" |
| #include "SkPath.h" |
| #include "SkTypeface.h" |
| #include "SkRandom.h" |
| |
| /** |
| * Draws text with random parameters. The text draws each get their own clip rect. It is also |
| * used as a bench to measure how well the GPU backend batches text draws. |
| */ |
| |
| class VariedTextGM : public skiagm::GM { |
| public: |
| VariedTextGM(bool effectiveClip, bool lcd) |
| : fEffectiveClip(effectiveClip) |
| , fLCD(lcd) { |
| memset(fTypefacesToUnref, 0, sizeof(fTypefacesToUnref)); |
| } |
| |
| ~VariedTextGM() { |
| for (size_t i = 0; i < SK_ARRAY_COUNT(fTypefacesToUnref); ++i) { |
| SkSafeUnref(fTypefacesToUnref[i]); |
| } |
| } |
| |
| protected: |
| SkString onShortName() override { |
| SkString name("varied_text"); |
| if (fEffectiveClip) { |
| name.append("_clipped"); |
| } else { |
| name.append("_ignorable_clip"); |
| } |
| if (fLCD) { |
| name.append("_lcd"); |
| } else { |
| name.append("_no_lcd"); |
| } |
| return name; |
| } |
| |
| SkISize onISize() override { |
| return SkISize::Make(640, 480); |
| } |
| |
| void onOnceBeforeDraw() override { |
| fPaint.setAntiAlias(true); |
| fPaint.setLCDRenderText(fLCD); |
| |
| SkISize size = this->getISize(); |
| SkScalar w = SkIntToScalar(size.fWidth); |
| SkScalar h = SkIntToScalar(size.fHeight); |
| |
| SK_COMPILE_ASSERT(4 == SK_ARRAY_COUNT(fTypefacesToUnref), typeface_cnt); |
| fTypefacesToUnref[0] = sk_tool_utils::create_portable_typeface("sans-serif", SkTypeface::kNormal); |
| fTypefacesToUnref[1] = sk_tool_utils::create_portable_typeface("sans-serif", SkTypeface::kBold); |
| fTypefacesToUnref[2] = sk_tool_utils::create_portable_typeface("serif", SkTypeface::kNormal); |
| fTypefacesToUnref[3] = sk_tool_utils::create_portable_typeface("serif", SkTypeface::kBold); |
| |
| SkRandom random; |
| for (int i = 0; i < kCnt; ++i) { |
| int length = random.nextRangeU(kMinLength, kMaxLength); |
| char text[kMaxLength]; |
| for (int j = 0; j < length; ++j) { |
| text[j] = (char)random.nextRangeU('!', 'z'); |
| } |
| fStrings[i].set(text, length); |
| |
| fColors[i] = random.nextU(); |
| fColors[i] |= 0xFF000000; |
| fColors[i] = sk_tool_utils::color_to_565(fColors[i]); |
| |
| static const SkScalar kMinPtSize = 8.f; |
| static const SkScalar kMaxPtSize = 32.f; |
| |
| fPtSizes[i] = random.nextRangeScalar(kMinPtSize, kMaxPtSize); |
| |
| fTypefaces[i] = fTypefacesToUnref[ |
| random.nextULessThan(SK_ARRAY_COUNT(fTypefacesToUnref))]; |
| |
| SkRect r; |
| fPaint.setColor(fColors[i]); |
| fPaint.setTypeface(fTypefaces[i]); |
| fPaint.setTextSize(fPtSizes[i]); |
| |
| fPaint.measureText(fStrings[i].c_str(), fStrings[i].size(), &r); |
| // safeRect is set of x,y positions where we can draw the string without hitting |
| // the GM's border. |
| SkRect safeRect = SkRect::MakeLTRB(-r.fLeft, -r.fTop, w - r.fRight, h - r.fBottom); |
| if (safeRect.isEmpty()) { |
| // If we don't fit then just don't worry about how we get cliped to the device |
| // border. |
| safeRect = SkRect::MakeWH(w, h); |
| } |
| fPositions[i].fX = random.nextRangeScalar(safeRect.fLeft, safeRect.fRight); |
| fPositions[i].fY = random.nextRangeScalar(safeRect.fTop, safeRect.fBottom); |
| |
| fClipRects[i] = r; |
| fClipRects[i].offset(fPositions[i].fX, fPositions[i].fY); |
| fClipRects[i].outset(2.f, 2.f); |
| |
| if (fEffectiveClip) { |
| fClipRects[i].fRight -= 0.25f * fClipRects[i].width(); |
| } |
| } |
| } |
| |
| void onDraw(SkCanvas* canvas) override { |
| for (int i = 0; i < kCnt; ++i) { |
| fPaint.setColor(fColors[i]); |
| fPaint.setTextSize(fPtSizes[i]); |
| fPaint.setTypeface(fTypefaces[i]); |
| |
| canvas->save(); |
| canvas->clipRect(fClipRects[i]); |
| canvas->translate(fPositions[i].fX, fPositions[i].fY); |
| canvas->drawText(fStrings[i].c_str(), fStrings[i].size(), 0, 0, fPaint); |
| canvas->restore(); |
| } |
| |
| // Visualize the clips, but not in bench mode. |
| if (kBench_Mode != this->getMode()) { |
| SkPaint wirePaint; |
| wirePaint.setAntiAlias(true); |
| wirePaint.setStrokeWidth(0); |
| wirePaint.setStyle(SkPaint::kStroke_Style); |
| for (int i = 0; i < kCnt; ++i) { |
| canvas->drawRect(fClipRects[i], wirePaint); |
| } |
| } |
| } |
| |
| bool runAsBench() const override { return true; } |
| |
| private: |
| static const int kCnt = 30; |
| static const int kMinLength = 15; |
| static const int kMaxLength = 40; |
| |
| bool fEffectiveClip; |
| bool fLCD; |
| SkTypeface* fTypefacesToUnref[4]; |
| SkPaint fPaint; |
| |
| // precomputed for each text draw |
| SkString fStrings[kCnt]; |
| SkColor fColors[kCnt]; |
| SkScalar fPtSizes[kCnt]; |
| SkTypeface* fTypefaces[kCnt]; |
| SkPoint fPositions[kCnt]; |
| SkRect fClipRects[kCnt]; |
| |
| typedef skiagm::GM INHERITED; |
| }; |
| |
| DEF_GM( return SkNEW(VariedTextGM(false, false)); ) |
| DEF_GM( return SkNEW(VariedTextGM(true, false)); ) |
| DEF_GM( return SkNEW(VariedTextGM(false, true)); ) |
| DEF_GM( return SkNEW(VariedTextGM(true, true)); ) |