179 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			179 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C++
		
	
	
	
| /*
 | |
|  * Copyright 2015 Google Inc.
 | |
|  *
 | |
|  * Use of this source code is governed by a BSD-style license that can be
 | |
|  * found in the LICENSE file.
 | |
|  */
 | |
| 
 | |
| #include "gm/gm.h"
 | |
| #include "include/core/SkCanvas.h"
 | |
| #include "include/core/SkColor.h"
 | |
| #include "include/core/SkColorSpace.h"
 | |
| #include "include/core/SkFont.h"
 | |
| #include "include/core/SkFontStyle.h"
 | |
| #include "include/core/SkFontTypes.h"
 | |
| #include "include/core/SkImageInfo.h"
 | |
| #include "include/core/SkPaint.h"
 | |
| #include "include/core/SkRect.h"
 | |
| #include "include/core/SkRefCnt.h"
 | |
| #include "include/core/SkScalar.h"
 | |
| #include "include/core/SkSize.h"
 | |
| #include "include/core/SkString.h"
 | |
| #include "include/core/SkSurface.h"
 | |
| #include "include/core/SkSurfaceProps.h"
 | |
| #include "include/core/SkTextBlob.h"
 | |
| #include "include/core/SkTypeface.h"
 | |
| #include "include/gpu/GrDirectContext.h"
 | |
| #include "include/gpu/GrRecordingContext.h"
 | |
| #include "tools/ToolUtils.h"
 | |
| #include "tools/fonts/RandomScalerContext.h"
 | |
| 
 | |
| #include <string.h>
 | |
| #include <utility>
 | |
| 
 | |
| namespace skiagm {
 | |
| class TextBlobRandomFont : public GM {
 | |
| public:
 | |
|     // This gm tests that textblobs can be translated and scaled with a font that returns random
 | |
|     // but deterministic masks
 | |
|     TextBlobRandomFont() { }
 | |
| 
 | |
| protected:
 | |
|     void onOnceBeforeDraw() override {
 | |
|         SkTextBlobBuilder builder;
 | |
| 
 | |
|         const char* text = "The quick brown fox jumps over the lazy dog.";
 | |
| 
 | |
|         SkPaint paint;
 | |
|         paint.setAntiAlias(true);
 | |
|         paint.setColor(SK_ColorMAGENTA);
 | |
| 
 | |
|         // make textbloben
 | |
|         SkFont font;
 | |
|         font.setSize(32);
 | |
|         font.setEdging(SkFont::Edging::kSubpixelAntiAlias);
 | |
| 
 | |
|         // Setup our random scaler context
 | |
|         auto typeface = ToolUtils::create_portable_typeface("sans-serif", SkFontStyle::Bold());
 | |
|         if (!typeface) {
 | |
|             typeface = SkTypeface::MakeDefault();
 | |
|         }
 | |
|         font.setTypeface(sk_make_sp<SkRandomTypeface>(std::move(typeface), paint, false));
 | |
| 
 | |
|         SkScalar y = 0;
 | |
|         SkRect bounds;
 | |
|         font.measureText(text, strlen(text), SkTextEncoding::kUTF8, &bounds);
 | |
|         y -= bounds.fTop;
 | |
|         ToolUtils::add_to_text_blob(&builder, text, font, 0, y);
 | |
|         y += bounds.fBottom;
 | |
| 
 | |
|         // A8
 | |
|         const char* bigtext1 = "The quick brown fox";
 | |
|         const char* bigtext2 = "jumps over the lazy dog.";
 | |
|         font.setSize(160);
 | |
|         font.setSubpixel(false);
 | |
|         font.setEdging(SkFont::Edging::kAntiAlias);
 | |
|         font.measureText(bigtext1, strlen(bigtext1), SkTextEncoding::kUTF8, &bounds);
 | |
|         y -= bounds.fTop;
 | |
|         ToolUtils::add_to_text_blob(&builder, bigtext1, font, 0, y);
 | |
|         y += bounds.fBottom;
 | |
| 
 | |
|         font.measureText(bigtext2, strlen(bigtext2), SkTextEncoding::kUTF8, &bounds);
 | |
|         y -= bounds.fTop;
 | |
|         ToolUtils::add_to_text_blob(&builder, bigtext2, font, 0, y);
 | |
|         y += bounds.fBottom;
 | |
| 
 | |
|         // color emoji
 | |
|         if (sk_sp<SkTypeface> origEmoji = ToolUtils::emoji_typeface()) {
 | |
|             font.setTypeface(sk_make_sp<SkRandomTypeface>(origEmoji, paint, false));
 | |
|             const char* emojiText = ToolUtils::emoji_sample_text();
 | |
|             font.measureText(emojiText, strlen(emojiText), SkTextEncoding::kUTF8, &bounds);
 | |
|             y -= bounds.fTop;
 | |
|             ToolUtils::add_to_text_blob(&builder, emojiText, font, 0, y);
 | |
|             y += bounds.fBottom;
 | |
|         }
 | |
| 
 | |
|         // build
 | |
|         fBlob = builder.make();
 | |
|     }
 | |
| 
 | |
|     SkString onShortName() override {
 | |
|         return SkString("textblobrandomfont");
 | |
|     }
 | |
| 
 | |
|     SkISize onISize() override {
 | |
|         return SkISize::Make(kWidth, kHeight);
 | |
|     }
 | |
| 
 | |
|     DrawResult onDraw(SkCanvas* canvas, SkString* errorMsg) override {
 | |
|         if (!canvas->recordingContext()) {
 | |
|             *errorMsg = "Active context required to create SkSurface";
 | |
|             return DrawResult::kSkip;
 | |
|         }
 | |
| 
 | |
|         auto dContext = GrAsDirectContext(canvas->recordingContext());
 | |
| 
 | |
|         // This GM exists to test a specific feature of the GPU backend.
 | |
|         // This GM uses ToolUtils::makeSurface which doesn't work well with vias.
 | |
|         // This GM uses SkRandomTypeface which doesn't work well with serialization.
 | |
|         canvas->drawColor(SK_ColorWHITE);
 | |
| 
 | |
|         SkImageInfo info = SkImageInfo::Make(kWidth, kHeight, canvas->imageInfo().colorType(),
 | |
|                                              kPremul_SkAlphaType,
 | |
|                                              canvas->imageInfo().refColorSpace());
 | |
|         SkSurfaceProps props(0, kUnknown_SkPixelGeometry);
 | |
|         auto           surface(ToolUtils::makeSurface(canvas, info, &props));
 | |
|         if (!surface) {
 | |
|             *errorMsg = "This test requires a surface";
 | |
|             return DrawResult::kFail;
 | |
|         }
 | |
| 
 | |
|         SkPaint paint;
 | |
|         paint.setAntiAlias(true);
 | |
| 
 | |
|         SkCanvas* surfaceCanvas = surface->getCanvas();
 | |
| 
 | |
|         SkScalar stride = SkScalarCeilToScalar(fBlob->bounds().height());
 | |
|         SkScalar yOffset = 5;
 | |
| 
 | |
|         canvas->save();
 | |
|         // Originally we would alternate between rotating and not to force blob regeneration,
 | |
|         // but that code seems to have rotted. Keeping the rotate to match the old GM as
 | |
|         // much as possible, and it seems like a reasonable stress test for transformed
 | |
|         // color emoji.
 | |
|         canvas->rotate(-0.05f);
 | |
|         canvas->drawTextBlob(fBlob, 10, yOffset, paint);
 | |
|         yOffset += stride;
 | |
|         canvas->restore();
 | |
| 
 | |
|         // Rotate in the surface canvas, not the final canvas, to avoid aliasing
 | |
|         surfaceCanvas->rotate(-0.05f);
 | |
|         surfaceCanvas->drawTextBlob(fBlob, 10, yOffset, paint);
 | |
|         surface->draw(canvas, 0, 0);
 | |
|         yOffset += stride;
 | |
| 
 | |
|         if (dContext) {
 | |
|             // free gpu resources and verify
 | |
|             dContext->freeGpuResources();
 | |
|         }
 | |
| 
 | |
|         canvas->rotate(-0.05f);
 | |
|         canvas->drawTextBlob(fBlob, 10, yOffset, paint);
 | |
|         yOffset += stride;
 | |
|         return DrawResult::kOk;
 | |
|     }
 | |
| 
 | |
| private:
 | |
|     sk_sp<SkTextBlob> fBlob;
 | |
| 
 | |
|     inline static constexpr int kWidth = 2000;
 | |
|     inline static constexpr int kHeight = 1600;
 | |
| 
 | |
|     using INHERITED = GM;
 | |
| };
 | |
| 
 | |
| //////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| DEF_GM(return new TextBlobRandomFont;)
 | |
| }  // namespace skiagm
 |