mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-28 23:31:56 +00:00
519 lines
15 KiB
C++
519 lines
15 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 "ScaledFontMac.h"
|
|
#include "UnscaledFontMac.h"
|
|
#ifdef USE_SKIA
|
|
#include "PathSkia.h"
|
|
#include "skia/include/core/SkPaint.h"
|
|
#include "skia/include/core/SkPath.h"
|
|
#include "skia/include/ports/SkTypeface_mac.h"
|
|
#endif
|
|
#include <vector>
|
|
#include <dlfcn.h>
|
|
#ifdef MOZ_WIDGET_UIKIT
|
|
#include <CoreFoundation/CoreFoundation.h>
|
|
#endif
|
|
#include "nsCocoaFeatures.h"
|
|
|
|
#ifdef MOZ_WIDGET_COCOA
|
|
// prototype for private API
|
|
extern "C" {
|
|
CGPathRef CGFontGetGlyphPath(CGFontRef fontRef, CGAffineTransform *textTransform, int unknown, CGGlyph glyph);
|
|
};
|
|
#endif
|
|
|
|
#ifdef USE_CAIRO_SCALED_FONT
|
|
#include "cairo-quartz.h"
|
|
#endif
|
|
|
|
namespace mozilla {
|
|
namespace gfx {
|
|
|
|
// Simple helper class to automatically release a CFObject when it goes out
|
|
// of scope.
|
|
template<class T>
|
|
class AutoRelease
|
|
{
|
|
public:
|
|
explicit AutoRelease(T aObject)
|
|
: mObject(aObject)
|
|
{
|
|
}
|
|
|
|
~AutoRelease()
|
|
{
|
|
if (mObject) {
|
|
CFRelease(mObject);
|
|
}
|
|
}
|
|
|
|
operator T()
|
|
{
|
|
return mObject;
|
|
}
|
|
|
|
T forget()
|
|
{
|
|
T obj = mObject;
|
|
mObject = nullptr;
|
|
return obj;
|
|
}
|
|
|
|
private:
|
|
T mObject;
|
|
};
|
|
|
|
ScaledFontMac::CTFontDrawGlyphsFuncT* ScaledFontMac::CTFontDrawGlyphsPtr = nullptr;
|
|
bool ScaledFontMac::sSymbolLookupDone = false;
|
|
|
|
// Helper to create a CTFont from a CGFont, copying any variations that were
|
|
// set on the original CGFont.
|
|
static CTFontRef
|
|
CreateCTFontFromCGFontWithVariations(CGFontRef aCGFont, CGFloat aSize)
|
|
{
|
|
// Avoid calling potentially buggy variation APIs on pre-Sierra macOS
|
|
// versions (see bug 1331683)
|
|
if (!nsCocoaFeatures::OnSierraOrLater()) {
|
|
return CTFontCreateWithGraphicsFont(aCGFont, aSize, nullptr, nullptr);
|
|
}
|
|
|
|
CFDictionaryRef vars = CGFontCopyVariations(aCGFont);
|
|
CTFontRef ctFont;
|
|
if (vars) {
|
|
CFDictionaryRef varAttr =
|
|
CFDictionaryCreate(nullptr,
|
|
(const void**)&kCTFontVariationAttribute,
|
|
(const void**)&vars, 1,
|
|
&kCFTypeDictionaryKeyCallBacks,
|
|
&kCFTypeDictionaryValueCallBacks);
|
|
CFRelease(vars);
|
|
|
|
CTFontDescriptorRef varDesc = CTFontDescriptorCreateWithAttributes(varAttr);
|
|
CFRelease(varAttr);
|
|
|
|
ctFont = CTFontCreateWithGraphicsFont(aCGFont, aSize, nullptr, varDesc);
|
|
CFRelease(varDesc);
|
|
} else {
|
|
ctFont = CTFontCreateWithGraphicsFont(aCGFont, aSize, nullptr, nullptr);
|
|
}
|
|
return ctFont;
|
|
}
|
|
|
|
ScaledFontMac::ScaledFontMac(CGFontRef aFont,
|
|
const RefPtr<UnscaledFont>& aUnscaledFont,
|
|
Float aSize,
|
|
bool aOwnsFont)
|
|
: ScaledFontBase(aUnscaledFont, aSize)
|
|
, mFont(aFont)
|
|
{
|
|
if (!sSymbolLookupDone) {
|
|
CTFontDrawGlyphsPtr =
|
|
(CTFontDrawGlyphsFuncT*)dlsym(RTLD_DEFAULT, "CTFontDrawGlyphs");
|
|
sSymbolLookupDone = true;
|
|
}
|
|
|
|
if (!aOwnsFont) {
|
|
// XXX: should we be taking a reference
|
|
CGFontRetain(aFont);
|
|
}
|
|
|
|
if (CTFontDrawGlyphsPtr != nullptr) {
|
|
// only create mCTFont if we're going to be using the CTFontDrawGlyphs API
|
|
mCTFont = CreateCTFontFromCGFontWithVariations(aFont, aSize);
|
|
} else {
|
|
mCTFont = nullptr;
|
|
}
|
|
}
|
|
|
|
ScaledFontMac::~ScaledFontMac()
|
|
{
|
|
if (mCTFont) {
|
|
CFRelease(mCTFont);
|
|
}
|
|
CGFontRelease(mFont);
|
|
}
|
|
|
|
#ifdef USE_SKIA
|
|
SkTypeface* ScaledFontMac::GetSkTypeface()
|
|
{
|
|
if (!mTypeface) {
|
|
if (mCTFont) {
|
|
mTypeface = SkCreateTypefaceFromCTFont(mCTFont);
|
|
} else {
|
|
CTFontRef fontFace = CreateCTFontFromCGFontWithVariations(mFont, mSize);
|
|
mTypeface = SkCreateTypefaceFromCTFont(fontFace);
|
|
CFRelease(fontFace);
|
|
}
|
|
}
|
|
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?
|
|
|
|
already_AddRefed<Path>
|
|
ScaledFontMac::GetPathForGlyphs(const GlyphBuffer &aBuffer, const DrawTarget *aTarget)
|
|
{
|
|
return ScaledFontBase::GetPathForGlyphs(aBuffer, aTarget);
|
|
}
|
|
|
|
uint32_t
|
|
CalcTableChecksum(const uint32_t *tableStart, uint32_t length, bool skipChecksumAdjust = false)
|
|
{
|
|
uint32_t sum = 0L;
|
|
const uint32_t *table = tableStart;
|
|
const uint32_t *end = table+((length+3) & ~3) / sizeof(uint32_t);
|
|
while (table < end) {
|
|
if (skipChecksumAdjust && (table - tableStart) == 2) {
|
|
table++;
|
|
} else {
|
|
sum += CFSwapInt32BigToHost(*table++);
|
|
}
|
|
}
|
|
return sum;
|
|
}
|
|
|
|
struct TableRecord {
|
|
uint32_t tag;
|
|
uint32_t checkSum;
|
|
uint32_t offset;
|
|
uint32_t length;
|
|
CFDataRef data;
|
|
};
|
|
|
|
int maxPow2LessThan(int a)
|
|
{
|
|
int x = 1;
|
|
int shift = 0;
|
|
while ((x<<(shift+1)) < a) {
|
|
shift++;
|
|
}
|
|
return shift;
|
|
}
|
|
|
|
struct writeBuf
|
|
{
|
|
explicit writeBuf(int size)
|
|
{
|
|
this->data = new unsigned char [size];
|
|
this->offset = 0;
|
|
}
|
|
~writeBuf() {
|
|
delete this->data;
|
|
}
|
|
|
|
template <class T>
|
|
void writeElement(T a)
|
|
{
|
|
*reinterpret_cast<T*>(&this->data[this->offset]) = a;
|
|
this->offset += sizeof(T);
|
|
}
|
|
|
|
void writeMem(const void *data, unsigned long length)
|
|
{
|
|
memcpy(&this->data[this->offset], data, length);
|
|
this->offset += length;
|
|
}
|
|
|
|
void align()
|
|
{
|
|
while (this->offset & 3) {
|
|
this->data[this->offset] = 0;
|
|
this->offset++;
|
|
}
|
|
}
|
|
|
|
unsigned char *data;
|
|
int offset;
|
|
};
|
|
|
|
static void CollectVariationSetting(const void *key, const void *value, void *context)
|
|
{
|
|
auto keyPtr = static_cast<const CFTypeRef>(key);
|
|
auto valuePtr = static_cast<const CFTypeRef>(value);
|
|
auto vpp = static_cast<FontVariation**>(context);
|
|
if (CFGetTypeID(keyPtr) == CFNumberGetTypeID() &&
|
|
CFGetTypeID(valuePtr) == CFNumberGetTypeID()) {
|
|
uint64_t t;
|
|
double v;
|
|
if (CFNumberGetValue(static_cast<CFNumberRef>(keyPtr), kCFNumberSInt64Type, &t) &&
|
|
CFNumberGetValue(static_cast<CFNumberRef>(valuePtr), kCFNumberDoubleType, &v)) {
|
|
(*vpp)->mTag = t;
|
|
(*vpp)->mValue = v;
|
|
(*vpp)++;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool
|
|
UnscaledFontMac::GetFontFileData(FontFileDataOutput aDataCallback, void *aBaton)
|
|
{
|
|
// We'll reconstruct a TTF font from the tables we can get from the CGFont
|
|
CFArrayRef tags = CGFontCopyTableTags(mFont);
|
|
CFIndex count = CFArrayGetCount(tags);
|
|
|
|
TableRecord *records = new TableRecord[count];
|
|
uint32_t offset = 0;
|
|
offset += sizeof(uint32_t)*3;
|
|
offset += sizeof(uint32_t)*4*count;
|
|
bool CFF = false;
|
|
for (CFIndex i = 0; i<count; i++) {
|
|
uint32_t tag = (uint32_t)(uintptr_t)CFArrayGetValueAtIndex(tags, i);
|
|
if (tag == 0x43464620) // 'CFF '
|
|
CFF = true;
|
|
CFDataRef data = CGFontCopyTableForTag(mFont, tag);
|
|
records[i].tag = tag;
|
|
records[i].offset = offset;
|
|
records[i].data = data;
|
|
records[i].length = CFDataGetLength(data);
|
|
bool skipChecksumAdjust = (tag == 0x68656164); // 'head'
|
|
records[i].checkSum = CalcTableChecksum(reinterpret_cast<const uint32_t*>(CFDataGetBytePtr(data)),
|
|
records[i].length, skipChecksumAdjust);
|
|
offset += records[i].length;
|
|
// 32 bit align the tables
|
|
offset = (offset + 3) & ~3;
|
|
}
|
|
CFRelease(tags);
|
|
|
|
struct writeBuf buf(offset);
|
|
// write header/offset table
|
|
if (CFF) {
|
|
buf.writeElement(CFSwapInt32HostToBig(0x4f54544f));
|
|
} else {
|
|
buf.writeElement(CFSwapInt32HostToBig(0x00010000));
|
|
}
|
|
buf.writeElement(CFSwapInt16HostToBig(count));
|
|
buf.writeElement(CFSwapInt16HostToBig((1<<maxPow2LessThan(count))*16));
|
|
buf.writeElement(CFSwapInt16HostToBig(maxPow2LessThan(count)));
|
|
buf.writeElement(CFSwapInt16HostToBig(count*16-((1<<maxPow2LessThan(count))*16)));
|
|
|
|
// write table record entries
|
|
for (CFIndex i = 0; i<count; i++) {
|
|
buf.writeElement(CFSwapInt32HostToBig(records[i].tag));
|
|
buf.writeElement(CFSwapInt32HostToBig(records[i].checkSum));
|
|
buf.writeElement(CFSwapInt32HostToBig(records[i].offset));
|
|
buf.writeElement(CFSwapInt32HostToBig(records[i].length));
|
|
}
|
|
|
|
// write tables
|
|
int checkSumAdjustmentOffset = 0;
|
|
for (CFIndex i = 0; i<count; i++) {
|
|
if (records[i].tag == 0x68656164) {
|
|
checkSumAdjustmentOffset = buf.offset + 2*4;
|
|
}
|
|
buf.writeMem(CFDataGetBytePtr(records[i].data), CFDataGetLength(records[i].data));
|
|
buf.align();
|
|
CFRelease(records[i].data);
|
|
}
|
|
delete[] records;
|
|
|
|
// clear the checksumAdjust field before checksumming the whole font
|
|
memset(&buf.data[checkSumAdjustmentOffset], 0, sizeof(uint32_t));
|
|
uint32_t fontChecksum = CFSwapInt32HostToBig(0xb1b0afba - CalcTableChecksum(reinterpret_cast<const uint32_t*>(buf.data), offset));
|
|
// set checkSumAdjust to the computed checksum
|
|
memcpy(&buf.data[checkSumAdjustmentOffset], &fontChecksum, sizeof(fontChecksum));
|
|
|
|
// we always use an index of 0
|
|
aDataCallback(buf.data, buf.offset, 0, aBaton);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
ScaledFontMac::GetFontInstanceData(FontInstanceDataOutput aCb, void* aBaton)
|
|
{
|
|
// Collect any variation settings that were incorporated into the CTFont.
|
|
uint32_t variationCount = 0;
|
|
FontVariation* variations = nullptr;
|
|
// Avoid calling potentially buggy variation APIs on pre-Sierra macOS
|
|
// versions (see bug 1331683)
|
|
if (nsCocoaFeatures::OnSierraOrLater()) {
|
|
if (mCTFont) {
|
|
CFDictionaryRef dict = CTFontCopyVariation(mCTFont);
|
|
if (dict) {
|
|
CFIndex count = CFDictionaryGetCount(dict);
|
|
if (count > 0) {
|
|
variations = new FontVariation[count];
|
|
FontVariation* vPtr = variations;
|
|
CFDictionaryApplyFunction(dict, CollectVariationSetting, &vPtr);
|
|
variationCount = vPtr - variations;
|
|
}
|
|
CFRelease(dict);
|
|
}
|
|
}
|
|
}
|
|
|
|
aCb(reinterpret_cast<uint8_t*>(variations), variationCount * sizeof(FontVariation), aBaton);
|
|
delete[] variations;
|
|
|
|
return true;
|
|
}
|
|
|
|
static CFDictionaryRef
|
|
CreateVariationDictionaryOrNull(CGFontRef aCGFont, uint32_t aVariationCount,
|
|
const FontVariation* aVariations)
|
|
{
|
|
// Avoid calling potentially buggy variation APIs on pre-Sierra macOS
|
|
// versions (see bug 1331683)
|
|
if (!nsCocoaFeatures::OnSierraOrLater()) {
|
|
return nullptr;
|
|
}
|
|
|
|
AutoRelease<CTFontRef>
|
|
ctFont(CTFontCreateWithGraphicsFont(aCGFont, 0, nullptr, nullptr));
|
|
AutoRelease<CFArrayRef> axes(CTFontCopyVariationAxes(ctFont));
|
|
if (!axes) {
|
|
return nullptr;
|
|
}
|
|
|
|
CFIndex axisCount = CFArrayGetCount(axes);
|
|
AutoRelease<CFMutableDictionaryRef>
|
|
dict(CFDictionaryCreateMutable(kCFAllocatorDefault, axisCount,
|
|
&kCFTypeDictionaryKeyCallBacks,
|
|
&kCFTypeDictionaryValueCallBacks));
|
|
|
|
// Number of variation settings passed in the aVariations parameter.
|
|
// This will typically be a very low value, so we just linear-search them.
|
|
bool allDefaultValues = true;
|
|
|
|
for (CFIndex i = 0; i < axisCount; ++i) {
|
|
// We sanity-check the axis info found in the CTFont, and bail out
|
|
// (returning null) if it doesn't have the expected types.
|
|
CFTypeRef axisInfo = CFArrayGetValueAtIndex(axes, i);
|
|
if (CFDictionaryGetTypeID() != CFGetTypeID(axisInfo)) {
|
|
return nullptr;
|
|
}
|
|
CFDictionaryRef axis = static_cast<CFDictionaryRef>(axisInfo);
|
|
|
|
CFTypeRef axisTag =
|
|
CFDictionaryGetValue(axis, kCTFontVariationAxisIdentifierKey);
|
|
if (!axisTag || CFGetTypeID(axisTag) != CFNumberGetTypeID()) {
|
|
return nullptr;
|
|
}
|
|
int64_t tagLong;
|
|
if (!CFNumberGetValue(static_cast<CFNumberRef>(axisTag),
|
|
kCFNumberSInt64Type, &tagLong)) {
|
|
return nullptr;
|
|
}
|
|
|
|
CFTypeRef axisName =
|
|
CFDictionaryGetValue(axis, kCTFontVariationAxisNameKey);
|
|
if (!axisName || CFGetTypeID(axisName) != CFStringGetTypeID()) {
|
|
return nullptr;
|
|
}
|
|
|
|
// Clamp axis values to the supported range.
|
|
CFTypeRef min = CFDictionaryGetValue(axis, kCTFontVariationAxisMinimumValueKey);
|
|
CFTypeRef max = CFDictionaryGetValue(axis, kCTFontVariationAxisMaximumValueKey);
|
|
CFTypeRef def = CFDictionaryGetValue(axis, kCTFontVariationAxisDefaultValueKey);
|
|
if (!min || CFGetTypeID(min) != CFNumberGetTypeID() ||
|
|
!max || CFGetTypeID(max) != CFNumberGetTypeID() ||
|
|
!def || CFGetTypeID(def) != CFNumberGetTypeID()) {
|
|
return nullptr;
|
|
}
|
|
double minDouble;
|
|
double maxDouble;
|
|
double defDouble;
|
|
if (!CFNumberGetValue(static_cast<CFNumberRef>(min), kCFNumberDoubleType,
|
|
&minDouble) ||
|
|
!CFNumberGetValue(static_cast<CFNumberRef>(max), kCFNumberDoubleType,
|
|
&maxDouble) ||
|
|
!CFNumberGetValue(static_cast<CFNumberRef>(def), kCFNumberDoubleType,
|
|
&defDouble)) {
|
|
return nullptr;
|
|
}
|
|
|
|
double value = defDouble;
|
|
for (uint32_t j = 0; j < aVariationCount; ++j) {
|
|
if (aVariations[j].mTag == tagLong) {
|
|
value = std::min(std::max<double>(aVariations[j].mValue,
|
|
minDouble),
|
|
maxDouble);
|
|
if (value != defDouble) {
|
|
allDefaultValues = false;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
AutoRelease<CFNumberRef> valueNumber(CFNumberCreate(kCFAllocatorDefault,
|
|
kCFNumberDoubleType,
|
|
&value));
|
|
CFDictionaryAddValue(dict, axisName, valueNumber);
|
|
}
|
|
|
|
if (allDefaultValues) {
|
|
// We didn't actually set any non-default values, so throw away the
|
|
// variations dictionary and just use the default rendering.
|
|
return nullptr;
|
|
}
|
|
|
|
return dict.forget();
|
|
}
|
|
|
|
CGFontRef
|
|
UnscaledFontMac::CreateCGFontWithVariations(CGFontRef aFont,
|
|
uint32_t aVariationCount,
|
|
const FontVariation* aVariations)
|
|
{
|
|
MOZ_ASSERT(aVariationCount > 0);
|
|
MOZ_ASSERT(aVariations);
|
|
|
|
AutoRelease<CFDictionaryRef>
|
|
varDict(CreateVariationDictionaryOrNull(aFont, aVariationCount, aVariations));
|
|
if (!varDict) {
|
|
return nullptr;
|
|
}
|
|
|
|
return CGFontCreateCopyWithVariations(aFont, varDict);
|
|
}
|
|
|
|
already_AddRefed<ScaledFont>
|
|
UnscaledFontMac::CreateScaledFont(Float aGlyphSize,
|
|
const uint8_t* aInstanceData,
|
|
uint32_t aInstanceDataLength)
|
|
{
|
|
uint32_t variationCount =
|
|
aInstanceDataLength / sizeof(FontVariation);
|
|
const FontVariation* variations =
|
|
reinterpret_cast<const FontVariation*>(aInstanceData);
|
|
|
|
CGFontRef fontRef = mFont;
|
|
if (variationCount > 0) {
|
|
CGFontRef varFont =
|
|
CreateCGFontWithVariations(mFont, variationCount, variations);
|
|
if (varFont) {
|
|
fontRef = varFont;
|
|
}
|
|
}
|
|
|
|
RefPtr<ScaledFontMac> scaledFont =
|
|
new ScaledFontMac(fontRef, this, aGlyphSize, fontRef != mFont);
|
|
|
|
if (!scaledFont->PopulateCairoScaledFont()) {
|
|
gfxWarning() << "Unable to create cairo scaled Mac font.";
|
|
return nullptr;
|
|
}
|
|
|
|
return scaledFont.forget();
|
|
}
|
|
|
|
#ifdef USE_CAIRO_SCALED_FONT
|
|
cairo_font_face_t*
|
|
ScaledFontMac::GetCairoFontFace()
|
|
{
|
|
MOZ_ASSERT(mFont);
|
|
return cairo_quartz_font_face_create_for_cgfont(mFont);
|
|
}
|
|
#endif
|
|
|
|
} // namespace gfx
|
|
} // namespace mozilla
|