mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-12-02 01:48:05 +00:00
190 lines
4.8 KiB
C++
190 lines
4.8 KiB
C++
/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "ScaledFontBase.h"
|
|
|
|
#ifdef USE_SKIA
|
|
#include "PathSkia.h"
|
|
#include "skia/SkPaint.h"
|
|
#endif
|
|
|
|
#ifdef USE_CAIRO
|
|
#include "PathCairo.h"
|
|
#include "DrawTargetCairo.h"
|
|
#include "HelpersCairo.h"
|
|
#endif
|
|
|
|
#include <vector>
|
|
#include <cmath>
|
|
|
|
using namespace std;
|
|
|
|
namespace mozilla {
|
|
namespace gfx {
|
|
|
|
ScaledFontBase::~ScaledFontBase()
|
|
{
|
|
#ifdef USE_SKIA
|
|
SkSafeUnref(mTypeface);
|
|
#endif
|
|
#ifdef USE_CAIRO_SCALED_FONT
|
|
cairo_scaled_font_destroy(mScaledFont);
|
|
#endif
|
|
}
|
|
|
|
ScaledFontBase::ScaledFontBase(Float aSize)
|
|
: mSize(aSize)
|
|
{
|
|
#ifdef USE_SKIA
|
|
mTypeface = nullptr;
|
|
#endif
|
|
#ifdef USE_CAIRO_SCALED_FONT
|
|
mScaledFont = nullptr;
|
|
#endif
|
|
}
|
|
|
|
#ifdef USE_SKIA
|
|
SkPath
|
|
ScaledFontBase::GetSkiaPathForGlyphs(const GlyphBuffer &aBuffer)
|
|
{
|
|
SkTypeface *typeFace = GetSkTypeface();
|
|
MOZ_ASSERT(typeFace);
|
|
|
|
SkPaint paint;
|
|
paint.setTypeface(typeFace);
|
|
paint.setTextEncoding(SkPaint::kGlyphID_TextEncoding);
|
|
paint.setTextSize(SkFloatToScalar(mSize));
|
|
|
|
std::vector<uint16_t> indices;
|
|
std::vector<SkPoint> offsets;
|
|
indices.resize(aBuffer.mNumGlyphs);
|
|
offsets.resize(aBuffer.mNumGlyphs);
|
|
|
|
for (unsigned int i = 0; i < aBuffer.mNumGlyphs; i++) {
|
|
indices[i] = aBuffer.mGlyphs[i].mIndex;
|
|
offsets[i].fX = SkFloatToScalar(aBuffer.mGlyphs[i].mPosition.x);
|
|
offsets[i].fY = SkFloatToScalar(aBuffer.mGlyphs[i].mPosition.y);
|
|
}
|
|
|
|
SkPath path;
|
|
paint.getPosTextPath(&indices.front(), aBuffer.mNumGlyphs*2, &offsets.front(), &path);
|
|
return path;
|
|
}
|
|
#endif
|
|
|
|
TemporaryRef<Path>
|
|
ScaledFontBase::GetPathForGlyphs(const GlyphBuffer &aBuffer, const DrawTarget *aTarget)
|
|
{
|
|
#ifdef USE_SKIA
|
|
if (aTarget->GetBackendType() == BackendType::SKIA) {
|
|
SkPath path = GetSkiaPathForGlyphs(aBuffer);
|
|
return new PathSkia(path, FillRule::FILL_WINDING);
|
|
}
|
|
#endif
|
|
#ifdef USE_CAIRO
|
|
if (aTarget->GetBackendType() == BackendType::CAIRO) {
|
|
MOZ_ASSERT(mScaledFont);
|
|
|
|
DrawTarget *dt = const_cast<DrawTarget*>(aTarget);
|
|
cairo_t *ctx = static_cast<cairo_t*>(dt->GetNativeSurface(NativeSurfaceType::CAIRO_CONTEXT));
|
|
|
|
bool isNewContext = !ctx;
|
|
if (!ctx) {
|
|
ctx = cairo_create(DrawTargetCairo::GetDummySurface());
|
|
cairo_matrix_t mat;
|
|
GfxMatrixToCairoMatrix(aTarget->GetTransform(), mat);
|
|
cairo_set_matrix(ctx, &mat);
|
|
}
|
|
|
|
cairo_set_scaled_font(ctx, mScaledFont);
|
|
|
|
// Convert our GlyphBuffer into an array of Cairo glyphs.
|
|
std::vector<cairo_glyph_t> glyphs(aBuffer.mNumGlyphs);
|
|
for (uint32_t i = 0; i < aBuffer.mNumGlyphs; ++i) {
|
|
glyphs[i].index = aBuffer.mGlyphs[i].mIndex;
|
|
glyphs[i].x = aBuffer.mGlyphs[i].mPosition.x;
|
|
glyphs[i].y = aBuffer.mGlyphs[i].mPosition.y;
|
|
}
|
|
|
|
cairo_new_path(ctx);
|
|
|
|
cairo_glyph_path(ctx, &glyphs[0], aBuffer.mNumGlyphs);
|
|
|
|
RefPtr<PathCairo> newPath = new PathCairo(ctx);
|
|
if (isNewContext) {
|
|
cairo_destroy(ctx);
|
|
}
|
|
|
|
return newPath.forget();
|
|
}
|
|
#endif
|
|
return nullptr;
|
|
}
|
|
|
|
void
|
|
ScaledFontBase::CopyGlyphsToBuilder(const GlyphBuffer &aBuffer, PathBuilder *aBuilder, BackendType aBackendType, const Matrix *aTransformHint)
|
|
{
|
|
#ifdef USE_SKIA
|
|
if (aBackendType == BackendType::SKIA) {
|
|
PathBuilderSkia *builder = static_cast<PathBuilderSkia*>(aBuilder);
|
|
builder->AppendPath(GetSkiaPathForGlyphs(aBuffer));
|
|
return;
|
|
}
|
|
#endif
|
|
#ifdef USE_CAIRO
|
|
if (aBackendType == BackendType::CAIRO) {
|
|
MOZ_ASSERT(mScaledFont);
|
|
|
|
PathBuilderCairo* builder = static_cast<PathBuilderCairo*>(aBuilder);
|
|
cairo_t *ctx = cairo_create(DrawTargetCairo::GetDummySurface());
|
|
|
|
if (aTransformHint) {
|
|
cairo_matrix_t mat;
|
|
GfxMatrixToCairoMatrix(*aTransformHint, mat);
|
|
cairo_set_matrix(ctx, &mat);
|
|
}
|
|
|
|
// Convert our GlyphBuffer into an array of Cairo glyphs.
|
|
std::vector<cairo_glyph_t> glyphs(aBuffer.mNumGlyphs);
|
|
for (uint32_t i = 0; i < aBuffer.mNumGlyphs; ++i) {
|
|
glyphs[i].index = aBuffer.mGlyphs[i].mIndex;
|
|
glyphs[i].x = aBuffer.mGlyphs[i].mPosition.x;
|
|
glyphs[i].y = aBuffer.mGlyphs[i].mPosition.y;
|
|
}
|
|
|
|
cairo_set_scaled_font(ctx, mScaledFont);
|
|
cairo_glyph_path(ctx, &glyphs[0], aBuffer.mNumGlyphs);
|
|
|
|
RefPtr<PathCairo> cairoPath = new PathCairo(ctx);
|
|
cairo_destroy(ctx);
|
|
|
|
cairoPath->AppendPathToBuilder(builder);
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
MOZ_CRASH("The specified backend type is not supported by CopyGlyphsToBuilder");
|
|
}
|
|
|
|
#ifdef USE_CAIRO_SCALED_FONT
|
|
void
|
|
ScaledFontBase::SetCairoScaledFont(cairo_scaled_font_t* font)
|
|
{
|
|
MOZ_ASSERT(!mScaledFont);
|
|
|
|
if (font == mScaledFont)
|
|
return;
|
|
|
|
if (mScaledFont)
|
|
cairo_scaled_font_destroy(mScaledFont);
|
|
|
|
mScaledFont = font;
|
|
cairo_scaled_font_reference(mScaledFont);
|
|
}
|
|
#endif
|
|
|
|
}
|
|
}
|