2011-11-02 12:55:03 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
2012-05-21 04:12:37 -07:00
|
|
|
* 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/. */
|
2011-11-02 12:55:03 -07:00
|
|
|
|
|
|
|
#include "ScaledFontMac.h"
|
2012-01-09 10:54:44 -08:00
|
|
|
#ifdef USE_SKIA
|
2011-11-02 12:55:03 -07:00
|
|
|
#include "PathSkia.h"
|
|
|
|
#include "skia/SkPaint.h"
|
|
|
|
#include "skia/SkPath.h"
|
|
|
|
#include "skia/SkTypeface_mac.h"
|
2012-01-09 10:54:44 -08:00
|
|
|
#endif
|
|
|
|
#include "DrawTargetCG.h"
|
2011-11-02 12:55:03 -07:00
|
|
|
#include <vector>
|
2012-10-23 02:07:14 -07:00
|
|
|
#include <dlfcn.h>
|
2011-11-02 12:55:03 -07:00
|
|
|
|
2012-01-09 10:54:44 -08:00
|
|
|
// prototype for private API
|
|
|
|
extern "C" {
|
|
|
|
CGPathRef CGFontGetGlyphPath(CGFontRef fontRef, CGAffineTransform *textTransform, int unknown, CGGlyph glyph);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2011-11-02 12:55:03 -07:00
|
|
|
namespace mozilla {
|
|
|
|
namespace gfx {
|
|
|
|
|
2012-10-23 02:07:14 -07:00
|
|
|
ScaledFontMac::CTFontDrawGlyphsFuncT* ScaledFontMac::CTFontDrawGlyphsPtr = nullptr;
|
|
|
|
bool ScaledFontMac::sSymbolLookupDone = false;
|
|
|
|
|
2011-11-02 12:55:03 -07:00
|
|
|
ScaledFontMac::ScaledFontMac(CGFontRef aFont, Float aSize)
|
2012-01-09 10:54:44 -08:00
|
|
|
: ScaledFontBase(aSize)
|
2011-11-02 12:55:03 -07:00
|
|
|
{
|
2012-10-23 02:07:14 -07:00
|
|
|
if (!sSymbolLookupDone) {
|
|
|
|
CTFontDrawGlyphsPtr =
|
|
|
|
(CTFontDrawGlyphsFuncT*)dlsym(RTLD_DEFAULT, "CTFontDrawGlyphs");
|
|
|
|
sSymbolLookupDone = true;
|
|
|
|
}
|
|
|
|
|
2012-01-09 10:54:44 -08:00
|
|
|
// XXX: should we be taking a reference
|
|
|
|
mFont = CGFontRetain(aFont);
|
2012-10-23 02:07:14 -07:00
|
|
|
if (CTFontDrawGlyphsPtr != nullptr) {
|
|
|
|
// only create mCTFont if we're going to be using the CTFontDrawGlyphs API
|
|
|
|
mCTFont = CTFontCreateWithGraphicsFont(aFont, aSize, nullptr, nullptr);
|
|
|
|
} else {
|
|
|
|
mCTFont = nullptr;
|
|
|
|
}
|
2011-11-02 12:55:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ScaledFontMac::~ScaledFontMac()
|
|
|
|
{
|
2012-10-23 02:07:14 -07:00
|
|
|
if (mCTFont) {
|
|
|
|
CFRelease(mCTFont);
|
|
|
|
}
|
2012-01-09 10:54:44 -08:00
|
|
|
CGFontRelease(mFont);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef USE_SKIA
|
|
|
|
SkTypeface* ScaledFontMac::GetSkTypeface()
|
|
|
|
{
|
|
|
|
if (!mTypeface) {
|
2012-10-23 02:07:14 -07:00
|
|
|
if (mCTFont) {
|
|
|
|
mTypeface = SkCreateTypefaceFromCTFont(mCTFont);
|
|
|
|
} else {
|
|
|
|
CTFontRef fontFace = CTFontCreateWithGraphicsFont(mFont, mSize, nullptr, nullptr);
|
|
|
|
mTypeface = SkCreateTypefaceFromCTFont(fontFace);
|
|
|
|
CFRelease(fontFace);
|
|
|
|
}
|
2012-01-09 10:54:44 -08:00
|
|
|
}
|
|
|
|
return mTypeface;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// private API here are the public options on OS X
|
|
|
|
// CTFontCreatePathForGlyph
|
|
|
|
// ATSUGlyphGetCubicPaths
|
|
|
|
// we've used this in cairo sucessfully for some time.
|
|
|
|
// Note: cairo dlsyms it. We could do that but maybe it's
|
|
|
|
// safe just to use?
|
|
|
|
|
|
|
|
TemporaryRef<Path>
|
|
|
|
ScaledFontMac::GetPathForGlyphs(const GlyphBuffer &aBuffer, const DrawTarget *aTarget)
|
|
|
|
{
|
2012-07-31 08:17:43 -07:00
|
|
|
if (aTarget->GetType() == BACKEND_COREGRAPHICS || aTarget->GetType() == BACKEND_COREGRAPHICS_ACCELERATED) {
|
2012-01-09 10:54:44 -08:00
|
|
|
CGMutablePathRef path = CGPathCreateMutable();
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < aBuffer.mNumGlyphs; i++) {
|
|
|
|
// XXX: we could probably fold both of these transforms together to avoid extra work
|
|
|
|
CGAffineTransform flip = CGAffineTransformMakeScale(1, -1);
|
|
|
|
CGPathRef glyphPath = ::CGFontGetGlyphPath(mFont, &flip, 0, aBuffer.mGlyphs[i].mIndex);
|
|
|
|
|
|
|
|
CGAffineTransform matrix = CGAffineTransformMake(mSize, 0, 0, mSize,
|
|
|
|
aBuffer.mGlyphs[i].mPosition.x,
|
|
|
|
aBuffer.mGlyphs[i].mPosition.y);
|
|
|
|
CGPathAddPath(path, &matrix, glyphPath);
|
|
|
|
CGPathRelease(glyphPath);
|
|
|
|
}
|
2012-02-14 14:59:11 -08:00
|
|
|
TemporaryRef<Path> ret = new PathCG(path, FILL_WINDING);
|
|
|
|
CGPathRelease(path);
|
|
|
|
return ret;
|
2012-01-09 10:54:44 -08:00
|
|
|
} else {
|
|
|
|
return ScaledFontBase::GetPathForGlyphs(aBuffer, aTarget);
|
|
|
|
}
|
2011-11-02 12:55:03 -07:00
|
|
|
}
|
|
|
|
|
2013-07-16 08:25:49 -07:00
|
|
|
void
|
|
|
|
ScaledFontMac::CopyGlyphsToBuilder(const GlyphBuffer &aBuffer, PathBuilder *aBuilder)
|
|
|
|
{
|
|
|
|
PathBuilderCG *pathBuilderCG =
|
|
|
|
static_cast<PathBuilderCG*>(aBuilder);
|
|
|
|
// XXX: check builder type
|
|
|
|
for (unsigned int i = 0; i < aBuffer.mNumGlyphs; i++) {
|
|
|
|
// XXX: we could probably fold both of these transforms together to avoid extra work
|
|
|
|
CGAffineTransform flip = CGAffineTransformMakeScale(1, -1);
|
|
|
|
CGPathRef glyphPath = ::CGFontGetGlyphPath(mFont, &flip, 0, aBuffer.mGlyphs[i].mIndex);
|
|
|
|
|
|
|
|
CGAffineTransform matrix = CGAffineTransformMake(mSize, 0, 0, mSize,
|
|
|
|
aBuffer.mGlyphs[i].mPosition.x,
|
|
|
|
aBuffer.mGlyphs[i].mPosition.y);
|
|
|
|
CGPathAddPath(pathBuilderCG->mCGPath, &matrix, glyphPath);
|
|
|
|
CGPathRelease(glyphPath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-02 12:55:03 -07:00
|
|
|
}
|
|
|
|
}
|