153 lines
5.6 KiB
C++
153 lines
5.6 KiB
C++
/*
|
|
* Copyright 2017 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/SkImageInfo.h"
|
|
#include "include/core/SkMatrix.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/SkTypes.h"
|
|
#include "include/private/SkTArray.h"
|
|
#include "tools/ToolUtils.h"
|
|
|
|
#include <initializer_list>
|
|
|
|
/**
|
|
* This GM tests reusing the same text blobs with distance fields rendering using various
|
|
* combinations of perspective and non-perspetive matrices, scissor clips, and different x,y params
|
|
* passed to the draw.
|
|
*/
|
|
class DFTextBlobPerspGM : public skiagm::GM {
|
|
public:
|
|
DFTextBlobPerspGM() { this->setBGColor(0xFFFFFFFF); }
|
|
|
|
protected:
|
|
SkString onShortName() override {
|
|
return SkString("dftext_blob_persp");
|
|
}
|
|
|
|
SkISize onISize() override { return SkISize::Make(900, 350); }
|
|
|
|
void onOnceBeforeDraw() override {
|
|
for (int i = 0; i < 3; ++i) {
|
|
SkFont font;
|
|
font.setSize(32);
|
|
font.setEdging(i == 0 ? SkFont::Edging::kAlias :
|
|
(i == 1 ? SkFont::Edging::kAntiAlias :
|
|
SkFont::Edging::kSubpixelAntiAlias));
|
|
font.setSubpixel(true);
|
|
SkTextBlobBuilder builder;
|
|
ToolUtils::add_to_text_blob(&builder, "SkiaText", font, 0, 0);
|
|
fBlobs.emplace_back(builder.make());
|
|
}
|
|
}
|
|
|
|
void onDraw(SkCanvas* inputCanvas) override {
|
|
// set up offscreen rendering with distance field text
|
|
auto ctx = inputCanvas->recordingContext();
|
|
SkISize size = this->onISize();
|
|
if (!inputCanvas->getBaseLayerSize().isEmpty()) {
|
|
size = inputCanvas->getBaseLayerSize();
|
|
}
|
|
SkImageInfo info = SkImageInfo::MakeN32(size.width(), size.height(), kPremul_SkAlphaType,
|
|
inputCanvas->imageInfo().refColorSpace());
|
|
SkSurfaceProps inputProps;
|
|
inputCanvas->getProps(&inputProps);
|
|
SkSurfaceProps props(SkSurfaceProps::kUseDeviceIndependentFonts_Flag | inputProps.flags(),
|
|
inputProps.pixelGeometry());
|
|
auto surface = SkSurface::MakeRenderTarget(ctx, SkBudgeted::kNo, info, 0, &props);
|
|
SkCanvas* canvas = surface ? surface->getCanvas() : inputCanvas;
|
|
// init our new canvas with the old canvas's matrix
|
|
canvas->setMatrix(inputCanvas->getLocalToDeviceAs3x3());
|
|
SkScalar x = 0, y = 0;
|
|
SkScalar maxH = 0;
|
|
for (auto twm : {TranslateWithMatrix::kNo, TranslateWithMatrix::kYes}) {
|
|
for (auto pm : {PerspMode::kNone, PerspMode::kX, PerspMode::kY, PerspMode::kXY}) {
|
|
for (auto& blob : fBlobs) {
|
|
for (bool clip : {false, true}) {
|
|
canvas->save();
|
|
SkScalar w = blob->bounds().width();
|
|
SkScalar h = blob->bounds().height();
|
|
if (clip) {
|
|
auto rect =
|
|
SkRect::MakeXYWH(x + 5, y + 5, w * 3.f / 4.f, h * 3.f / 4.f);
|
|
canvas->clipRect(rect, false);
|
|
}
|
|
this->drawBlob(canvas, blob.get(), SK_ColorBLACK, x, y + h, pm, twm);
|
|
x += w + 20.f;
|
|
maxH = std::max(h, maxH);
|
|
canvas->restore();
|
|
}
|
|
}
|
|
x = 0;
|
|
y += maxH + 20.f;
|
|
maxH = 0;
|
|
}
|
|
}
|
|
// render offscreen buffer
|
|
if (surface) {
|
|
SkAutoCanvasRestore acr(inputCanvas, true);
|
|
// since we prepended this matrix already, we blit using identity
|
|
inputCanvas->resetMatrix();
|
|
inputCanvas->drawImage(surface->makeImageSnapshot().get(), 0, 0);
|
|
}
|
|
}
|
|
|
|
private:
|
|
enum class PerspMode { kNone, kX, kY, kXY };
|
|
|
|
enum class TranslateWithMatrix : bool { kNo, kYes };
|
|
|
|
void drawBlob(SkCanvas* canvas, SkTextBlob* blob, SkColor color, SkScalar x, SkScalar y,
|
|
PerspMode perspMode, TranslateWithMatrix translateWithMatrix) {
|
|
canvas->save();
|
|
SkMatrix persp = SkMatrix::I();
|
|
switch (perspMode) {
|
|
case PerspMode::kNone:
|
|
break;
|
|
case PerspMode::kX:
|
|
persp.setPerspX(0.005f);
|
|
break;
|
|
case PerspMode::kY:
|
|
persp.setPerspY(00.005f);
|
|
break;
|
|
case PerspMode::kXY:
|
|
persp.setPerspX(-0.001f);
|
|
persp.setPerspY(-0.0015f);
|
|
break;
|
|
}
|
|
persp = SkMatrix::Concat(persp, SkMatrix::Translate(-x, -y));
|
|
persp = SkMatrix::Concat(SkMatrix::Translate(x, y), persp);
|
|
canvas->concat(persp);
|
|
if (TranslateWithMatrix::kYes == translateWithMatrix) {
|
|
canvas->translate(x, y);
|
|
x = 0;
|
|
y = 0;
|
|
}
|
|
SkPaint paint;
|
|
paint.setColor(color);
|
|
canvas->drawTextBlob(blob, x, y, paint);
|
|
canvas->restore();
|
|
}
|
|
|
|
SkTArray<sk_sp<SkTextBlob>> fBlobs;
|
|
using INHERITED = skiagm::GM;
|
|
};
|
|
|
|
DEF_GM(return new DFTextBlobPerspGM;)
|