blob: 3164fb3ba3454a536f284232a3c4098f06475b1f [file] [log] [blame]
/*
* Copyright 2014 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "src/utils/win/SkDWriteNTDDI_VERSION.h"
#include "include/core/SkTypes.h"
#if defined(SK_BUILD_FOR_WIN)
#include "src/base/SkLeanWindows.h"
// SkLeanWindows will include Windows.h, which will pull in all of the GDI defines.
// GDI #defines GetGlyphIndices to GetGlyphIndicesA or GetGlyphIndicesW, but
// IDWriteFontFace has a method called GetGlyphIndices. Since this file does
// not use GDI, undefing GetGlyphIndices makes things less confusing.
#undef GetGlyphIndices
#include "include/core/SkData.h"
#include "include/core/SkFontTypes.h"
#include "include/private/base/SkTo.h"
#include "src/core/SkFontDescriptor.h"
#include "src/core/SkFontStream.h"
#include "src/core/SkScalerContext.h"
#include "src/ports/SkScalerContext_win_dw.h"
#include "src/ports/SkTypeface_win_dw.h"
#include "src/sfnt/SkOTTable_OS_2.h"
#include "src/sfnt/SkOTTable_fvar.h"
#include "src/sfnt/SkOTTable_head.h"
#include "src/sfnt/SkOTTable_hhea.h"
#include "src/sfnt/SkOTTable_post.h"
#include "src/sfnt/SkOTUtils.h"
#include "src/utils/win/SkDWrite.h"
#include "src/utils/win/SkDWriteFontFileStream.h"
using namespace skia_private;
SkFontStyle DWriteFontTypeface::GetStyle(IDWriteFont* font, IDWriteFontFace* fontFace) {
int weight = font->GetWeight();
int width = font->GetStretch();
SkFontStyle::Slant slant = SkFontStyle::kUpright_Slant;
switch (font->GetStyle()) {
case DWRITE_FONT_STYLE_NORMAL: slant = SkFontStyle::kUpright_Slant; break;
case DWRITE_FONT_STYLE_OBLIQUE: slant = SkFontStyle::kOblique_Slant; break;
case DWRITE_FONT_STYLE_ITALIC: slant = SkFontStyle::kItalic_Slant; break;
default: SkASSERT(false); break;
}
#if defined(NTDDI_WIN10_RS3) && NTDDI_VERSION >= NTDDI_WIN10_RS3
[&weight, &width, &slant, fontFace]() -> void {
SkTScopedComPtr<IDWriteFontFace5> fontFace5;
if (FAILED(fontFace->QueryInterface(&fontFace5))) {
return;
}
if (!fontFace5->HasVariations()) {
return;
}
UINT32 fontAxisCount = fontFace5->GetFontAxisValueCount();
SkTScopedComPtr<IDWriteFontResource> fontResource;
HRV(fontFace5->GetFontResource(&fontResource));
AutoSTMalloc<8, DWRITE_FONT_AXIS_VALUE> fontAxisValue(fontAxisCount);
HRV(fontFace5->GetFontAxisValues(fontAxisValue.get(), fontAxisCount));
for (UINT32 axisIndex = 0; axisIndex < fontAxisCount; ++axisIndex) {
if (fontAxisValue[axisIndex].axisTag == DWRITE_FONT_AXIS_TAG_WEIGHT) {
weight = fontAxisValue[axisIndex].value;
}
if (fontAxisValue[axisIndex].axisTag == DWRITE_FONT_AXIS_TAG_WIDTH) {
SkScalar wdthValue = fontAxisValue[axisIndex].value;
width = SkFontDescriptor::SkFontStyleWidthForWidthAxisValue(wdthValue);
}
if (fontAxisValue[axisIndex].axisTag == DWRITE_FONT_AXIS_TAG_SLANT &&
slant != SkFontStyle::kItalic_Slant)
{
if (fontAxisValue[axisIndex].value == 0) {
slant = SkFontStyle::kUpright_Slant;
} else {
slant = SkFontStyle::kOblique_Slant;
}
}
}
}();
#endif
return SkFontStyle(weight, width, slant);
}
sk_sp<DWriteFontTypeface> DWriteFontTypeface::Make(
IDWriteFactory* factory,
IDWriteFontFace* fontFace,
IDWriteFont* font,
IDWriteFontFamily* fontFamily,
sk_sp<Loaders> loaders,
const SkFontArguments::Palette& palette)
{
return sk_sp<DWriteFontTypeface>(new DWriteFontTypeface(
GetStyle(font, fontFace), factory, fontFace, font, fontFamily, std::move(loaders), palette));
}
HRESULT DWriteFontTypeface::initializePalette() {
if (!fIsColorFont) {
return S_OK;
}
UINT32 dwPaletteCount = fDWriteFontFace2->GetColorPaletteCount();
if (dwPaletteCount == 0) {
return S_OK;
}
// Treat out of range palette index values as 0. Still apply overrides.
// https://www.w3.org/TR/css-fonts-4/#base-palette-desc
UINT32 basePaletteIndex = 0;
if (SkTFitsIn<UINT32>(fRequestedPalette.index) &&
SkTo<UINT32>(fRequestedPalette.index) < dwPaletteCount)
{
basePaletteIndex = fRequestedPalette.index;
}
UINT32 dwPaletteEntryCount = fDWriteFontFace2->GetPaletteEntryCount();
AutoSTMalloc<8, DWRITE_COLOR_F> dwPaletteEntry(dwPaletteEntryCount);
HRM(fDWriteFontFace2->GetPaletteEntries(basePaletteIndex,
0, dwPaletteEntryCount,
dwPaletteEntry),
"Could not retrieve palette entries.");
fPalette.reset(new SkColor[dwPaletteEntryCount]);
fDWPalette.reset(new DWRITE_COLOR_F[dwPaletteEntryCount]);
for (UINT32 i = 0; i < dwPaletteEntryCount; ++i) {
fPalette[i] = SkColorSetARGB(sk_float_round2int(dwPaletteEntry[i].a * 255),
sk_float_round2int(dwPaletteEntry[i].r * 255),
sk_float_round2int(dwPaletteEntry[i].g * 255),
sk_float_round2int(dwPaletteEntry[i].b * 255));
fDWPalette[i] = dwPaletteEntry[i];
}
for (int i = 0; i < fRequestedPalette.overrideCount; ++i) {
const SkFontArguments::Palette::Override& paletteOverride = fRequestedPalette.overrides[i];
if (SkTFitsIn<UINT32>(paletteOverride.index) &&
SkTo<UINT32>(paletteOverride.index) < dwPaletteEntryCount)
{
fPalette[paletteOverride.index] = paletteOverride.color;
// Avoid brace initialization as DWRITE_COLOR_F can be defined as four floats
// (dxgitype.h, d3d9types.h) or four unions of two floats (dwrite_2.h, d3dtypes.h).
// The type changed in Direct3D 10, but the change does not appear to be documented.
const SkColor4f skColor = SkColor4f::FromColor(paletteOverride.color);
DWRITE_COLOR_F& dwColor = fDWPalette[paletteOverride.index];
dwColor.r = skColor.fR;
dwColor.g = skColor.fG;
dwColor.b = skColor.fB;
dwColor.a = skColor.fA;
}
}
fPaletteEntryCount = dwPaletteEntryCount;
return S_OK;
}
DWriteFontTypeface::DWriteFontTypeface(const SkFontStyle& style,
IDWriteFactory* factory,
IDWriteFontFace* fontFace,
IDWriteFont* font,
IDWriteFontFamily* fontFamily,
sk_sp<Loaders> loaders,
const SkFontArguments::Palette& palette)
: SkTypeface(style, false)
, fFactory(SkRefComPtr(factory))
, fDWriteFontFamily(SkRefComPtr(fontFamily))
, fDWriteFont(SkRefComPtr(font))
, fDWriteFontFace(SkRefComPtr(fontFace))
, fRequestedPaletteEntryOverrides(palette.overrideCount
? (SkFontArguments::Palette::Override*)memcpy(
new SkFontArguments::Palette::Override[palette.overrideCount],
palette.overrides,
palette.overrideCount * sizeof(palette.overrides[0]))
: nullptr)
, fRequestedPalette{palette.index,
fRequestedPaletteEntryOverrides.get(), palette.overrideCount }
, fPaletteEntryCount(0)
, fLoaders(std::move(loaders))
{
if (!SUCCEEDED(fDWriteFontFace->QueryInterface(&fDWriteFontFace1))) {
// IUnknown::QueryInterface states that if it fails, punk will be set to nullptr.
// http://blogs.msdn.com/b/oldnewthing/archive/2004/03/26/96777.aspx
SkASSERT_RELEASE(nullptr == fDWriteFontFace1.get());
}
if (!SUCCEEDED(fDWriteFontFace->QueryInterface(&fDWriteFontFace2))) {
SkASSERT_RELEASE(nullptr == fDWriteFontFace2.get());
}
if (!SUCCEEDED(fDWriteFontFace->QueryInterface(&fDWriteFontFace4))) {
SkASSERT_RELEASE(nullptr == fDWriteFontFace4.get());
}
#if DWRITE_CORE || (defined(NTDDI_WIN11_ZN) && NTDDI_VERSION >= NTDDI_WIN11_ZN)
if (!SUCCEEDED(fDWriteFontFace->QueryInterface(&fDWriteFontFace7))) {
SkASSERT_RELEASE(nullptr == fDWriteFontFace7/*.get()*/);
}
#endif
if (!SUCCEEDED(fFactory->QueryInterface(&fFactory2))) {
SkASSERT_RELEASE(nullptr == fFactory2.get());
}
if (fDWriteFontFace1 && fDWriteFontFace1->IsMonospacedFont()) {
this->setIsFixedPitch(true);
}
fIsColorFont = fFactory2 && fDWriteFontFace2 && fDWriteFontFace2->IsColorFont();
this->initializePalette();
}
DWriteFontTypeface::~DWriteFontTypeface() {
#if DWRITE_CORE || (defined(NTDDI_WIN11_ZN) && NTDDI_VERSION >= NTDDI_WIN11_ZN)
if (fDWriteFontFace7) {
fDWriteFontFace7->Release();
}
#endif
}
DWriteFontTypeface::Loaders::~Loaders() {
// Don't return if any fail, just keep going to free up as much as possible.
HRESULT hr;
hr = fFactory->UnregisterFontCollectionLoader(fDWriteFontCollectionLoader.get());
if (FAILED(hr)) {
SK_TRACEHR(hr, "FontCollectionLoader");
}
hr = fFactory->UnregisterFontFileLoader(fDWriteFontFileLoader.get());
if (FAILED(hr)) {
SK_TRACEHR(hr, "FontFileLoader");
}
}
void DWriteFontTypeface::onGetFamilyName(SkString* familyName) const {
SkTScopedComPtr<IDWriteLocalizedStrings> familyNames;
HRV(fDWriteFontFamily->GetFamilyNames(&familyNames));
sk_get_locale_string(familyNames.get(), nullptr/*fMgr->fLocaleName.get()*/, familyName);
}
bool DWriteFontTypeface::onGetPostScriptName(SkString* skPostScriptName) const {
SkString localSkPostScriptName;
SkTScopedComPtr<IDWriteLocalizedStrings> postScriptNames;
BOOL exists = FALSE;
if (FAILED(fDWriteFont->GetInformationalStrings(
DWRITE_INFORMATIONAL_STRING_POSTSCRIPT_NAME,
&postScriptNames,
&exists)) ||
!exists ||
FAILED(sk_get_locale_string(postScriptNames.get(), nullptr, &localSkPostScriptName)))
{
return false;
}
if (skPostScriptName) {
*skPostScriptName = localSkPostScriptName;
}
return true;
}
void DWriteFontTypeface::onGetFontDescriptor(SkFontDescriptor* desc,
bool* serialize) const {
// Get the family name.
SkTScopedComPtr<IDWriteLocalizedStrings> familyNames;
HRV(fDWriteFontFamily->GetFamilyNames(&familyNames));
SkString utf8FamilyName;
sk_get_locale_string(familyNames.get(), nullptr/*fMgr->fLocaleName.get()*/, &utf8FamilyName);
desc->setFamilyName(utf8FamilyName.c_str());
desc->setStyle(this->fontStyle());
desc->setPaletteIndex(fRequestedPalette.index);
sk_careful_memcpy(desc->setPaletteEntryOverrides(fRequestedPalette.overrideCount),
fRequestedPalette.overrides,
fRequestedPalette.overrideCount * sizeof(fRequestedPalette.overrides[0]));
desc->setFactoryId(FactoryId);
*serialize = SkToBool(fLoaders);
}
void DWriteFontTypeface::onCharsToGlyphs(const SkUnichar* uni, int count,
SkGlyphID glyphs[]) const {
fDWriteFontFace->GetGlyphIndices((const UINT32*)uni, count, glyphs);
}
int DWriteFontTypeface::onCountGlyphs() const {
return fDWriteFontFace->GetGlyphCount();
}
void DWriteFontTypeface::getPostScriptGlyphNames(SkString*) const {}
int DWriteFontTypeface::onGetUPEM() const {
DWRITE_FONT_METRICS metrics;
fDWriteFontFace->GetMetrics(&metrics);
return metrics.designUnitsPerEm;
}
class LocalizedStrings_IDWriteLocalizedStrings : public SkTypeface::LocalizedStrings {
public:
/** Takes ownership of the IDWriteLocalizedStrings. */
explicit LocalizedStrings_IDWriteLocalizedStrings(IDWriteLocalizedStrings* strings)
: fIndex(0), fStrings(strings)
{ }
bool next(SkTypeface::LocalizedString* localizedString) override {
if (fIndex >= fStrings->GetCount()) {
return false;
}
// String
UINT32 stringLen;
HRBM(fStrings->GetStringLength(fIndex, &stringLen), "Could not get string length.");
SkSMallocWCHAR wString(static_cast<size_t>(stringLen)+1);
HRBM(fStrings->GetString(fIndex, wString.get(), stringLen+1), "Could not get string.");
HRB(sk_wchar_to_skstring(wString.get(), stringLen, &localizedString->fString));
// Locale
UINT32 localeLen;
HRBM(fStrings->GetLocaleNameLength(fIndex, &localeLen), "Could not get locale length.");
SkSMallocWCHAR wLocale(static_cast<size_t>(localeLen)+1);
HRBM(fStrings->GetLocaleName(fIndex, wLocale.get(), localeLen+1), "Could not get locale.");
HRB(sk_wchar_to_skstring(wLocale.get(), localeLen, &localizedString->fLanguage));
++fIndex;
return true;
}
private:
UINT32 fIndex;
SkTScopedComPtr<IDWriteLocalizedStrings> fStrings;
};
SkTypeface::LocalizedStrings* DWriteFontTypeface::onCreateFamilyNameIterator() const {
sk_sp<SkTypeface::LocalizedStrings> nameIter =
SkOTUtils::LocalizedStrings_NameTable::MakeForFamilyNames(*this);
if (!nameIter) {
SkTScopedComPtr<IDWriteLocalizedStrings> familyNames;
HRNM(fDWriteFontFamily->GetFamilyNames(&familyNames), "Could not obtain family names.");
nameIter = sk_make_sp<LocalizedStrings_IDWriteLocalizedStrings>(familyNames.release());
}
return nameIter.release();
}
bool DWriteFontTypeface::onGlyphMaskNeedsCurrentColor() const {
return fDWriteFontFace2 && fDWriteFontFace2->GetColorPaletteCount() > 0;
}
int DWriteFontTypeface::onGetVariationDesignPosition(
SkFontArguments::VariationPosition::Coordinate coordinates[], int coordinateCount) const
{
#if defined(NTDDI_WIN10_RS3) && NTDDI_VERSION >= NTDDI_WIN10_RS3
SkTScopedComPtr<IDWriteFontFace5> fontFace5;
if (FAILED(fDWriteFontFace->QueryInterface(&fontFace5))) {
return -1;
}
// Return 0 if the font is not variable font.
if (!fontFace5->HasVariations()) {
return 0;
}
UINT32 fontAxisCount = fontFace5->GetFontAxisValueCount();
SkTScopedComPtr<IDWriteFontResource> fontResource;
HR_GENERAL(fontFace5->GetFontResource(&fontResource), nullptr, -1);
UINT32 variableAxisCount = 0;
for (UINT32 i = 0; i < fontAxisCount; ++i) {
if (fontResource->GetFontAxisAttributes(i) & DWRITE_FONT_AXIS_ATTRIBUTES_VARIABLE) {
++variableAxisCount;
}
}
if (!coordinates || coordinateCount < 0 || (unsigned)coordinateCount < variableAxisCount) {
return SkTo<int>(variableAxisCount);
}
AutoSTMalloc<8, DWRITE_FONT_AXIS_VALUE> fontAxisValue(fontAxisCount);
HR_GENERAL(fontFace5->GetFontAxisValues(fontAxisValue.get(), fontAxisCount), nullptr, -1);
UINT32 coordIndex = 0;
for (UINT32 axisIndex = 0; axisIndex < fontAxisCount; ++axisIndex) {
if (fontResource->GetFontAxisAttributes(axisIndex) & DWRITE_FONT_AXIS_ATTRIBUTES_VARIABLE) {
coordinates[coordIndex].axis = SkEndian_SwapBE32(fontAxisValue[axisIndex].axisTag);
coordinates[coordIndex].value = fontAxisValue[axisIndex].value;
++coordIndex;
}
}
SkASSERT(coordIndex == variableAxisCount);
return SkTo<int>(variableAxisCount);
#else
return -1;
#endif
}
int DWriteFontTypeface::onGetVariationDesignParameters(
SkFontParameters::Variation::Axis parameters[], int parameterCount) const
{
#if defined(NTDDI_WIN10_RS3) && NTDDI_VERSION >= NTDDI_WIN10_RS3
SkTScopedComPtr<IDWriteFontFace5> fontFace5;
if (FAILED(fDWriteFontFace->QueryInterface(&fontFace5))) {
return -1;
}
// Return 0 if the font is not variable font.
if (!fontFace5->HasVariations()) {
return 0;
}
UINT32 fontAxisCount = fontFace5->GetFontAxisValueCount();
SkTScopedComPtr<IDWriteFontResource> fontResource;
HR_GENERAL(fontFace5->GetFontResource(&fontResource), nullptr, -1);
int variableAxisCount = 0;
for (UINT32 i = 0; i < fontAxisCount; ++i) {
if (fontResource->GetFontAxisAttributes(i) & DWRITE_FONT_AXIS_ATTRIBUTES_VARIABLE) {
variableAxisCount++;
}
}
if (!parameters || parameterCount < variableAxisCount) {
return variableAxisCount;
}
AutoSTMalloc<8, DWRITE_FONT_AXIS_RANGE> fontAxisRange(fontAxisCount);
HR_GENERAL(fontResource->GetFontAxisRanges(fontAxisRange.get(), fontAxisCount), nullptr, -1);
AutoSTMalloc<8, DWRITE_FONT_AXIS_VALUE> fontAxisDefaultValue(fontAxisCount);
HR_GENERAL(fontResource->GetDefaultFontAxisValues(fontAxisDefaultValue.get(), fontAxisCount),
nullptr, -1);
UINT32 coordIndex = 0;
for (UINT32 axisIndex = 0; axisIndex < fontAxisCount; ++axisIndex) {
if (fontResource->GetFontAxisAttributes(axisIndex) & DWRITE_FONT_AXIS_ATTRIBUTES_VARIABLE) {
parameters[coordIndex].tag = SkEndian_SwapBE32(fontAxisDefaultValue[axisIndex].axisTag);
parameters[coordIndex].min = fontAxisRange[axisIndex].minValue;
parameters[coordIndex].def = fontAxisDefaultValue[axisIndex].value;
parameters[coordIndex].max = fontAxisRange[axisIndex].maxValue;
parameters[coordIndex].setHidden(fontResource->GetFontAxisAttributes(axisIndex) &
DWRITE_FONT_AXIS_ATTRIBUTES_HIDDEN);
++coordIndex;
}
}
return variableAxisCount;
#else
return -1;
#endif
}
int DWriteFontTypeface::onGetTableTags(SkFontTableTag tags[]) const {
DWRITE_FONT_FACE_TYPE type = fDWriteFontFace->GetType();
if (type != DWRITE_FONT_FACE_TYPE_CFF &&
type != DWRITE_FONT_FACE_TYPE_TRUETYPE &&
type != DWRITE_FONT_FACE_TYPE_TRUETYPE_COLLECTION)
{
return 0;
}
int ttcIndex;
std::unique_ptr<SkStreamAsset> stream = this->openStream(&ttcIndex);
return stream.get() ? SkFontStream::GetTableTags(stream.get(), ttcIndex, tags) : 0;
}
size_t DWriteFontTypeface::onGetTableData(SkFontTableTag tag, size_t offset,
size_t length, void* data) const
{
AutoDWriteTable table(fDWriteFontFace.get(), SkEndian_SwapBE32(tag));
if (!table.fExists) {
return 0;
}
if (offset > table.fSize) {
return 0;
}
size_t size = std::min(length, table.fSize - offset);
if (data) {
memcpy(data, table.fData + offset, size);
}
return size;
}
sk_sp<SkData> DWriteFontTypeface::onCopyTableData(SkFontTableTag tag) const {
const uint8_t* data;
UINT32 size;
void* lock;
BOOL exists;
fDWriteFontFace->TryGetFontTable(SkEndian_SwapBE32(tag),
reinterpret_cast<const void **>(&data), &size, &lock, &exists);
if (!exists) {
return nullptr;
}
struct Context {
Context(void* lock, IDWriteFontFace* face) : fLock(lock), fFontFace(SkRefComPtr(face)) {}
~Context() { fFontFace->ReleaseFontTable(fLock); }
void* fLock;
SkTScopedComPtr<IDWriteFontFace> fFontFace;
};
return SkData::MakeWithProc(data, size,
[](const void*, void* ctx) { delete (Context*)ctx; },
new Context(lock, fDWriteFontFace.get()));
}
sk_sp<SkTypeface> DWriteFontTypeface::onMakeClone(const SkFontArguments& args) const {
// Skip if the current face index does not match the ttcIndex
if (fDWriteFontFace->GetIndex() != SkTo<UINT32>(args.getCollectionIndex())) {
return sk_ref_sp(this);
}
#if defined(NTDDI_WIN10_RS3) && NTDDI_VERSION >= NTDDI_WIN10_RS3
SkTScopedComPtr<IDWriteFontFace5> fontFace5;
if (SUCCEEDED(fDWriteFontFace->QueryInterface(&fontFace5)) && fontFace5->HasVariations()) {
UINT32 fontAxisCount = fontFace5->GetFontAxisValueCount();
UINT32 argsCoordCount = args.getVariationDesignPosition().coordinateCount;
AutoSTMalloc<8, DWRITE_FONT_AXIS_VALUE> fontAxisValue(fontAxisCount);
HRN(fontFace5->GetFontAxisValues(fontAxisValue.get(), fontAxisCount));
for (UINT32 fontIndex = 0; fontIndex < fontAxisCount; ++fontIndex) {
for (UINT32 argsIndex = 0; argsIndex < argsCoordCount; ++argsIndex) {
if (SkEndian_SwapBE32(fontAxisValue[fontIndex].axisTag) ==
args.getVariationDesignPosition().coordinates[argsIndex].axis) {
fontAxisValue[fontIndex].value =
args.getVariationDesignPosition().coordinates[argsIndex].value;
}
}
}
SkTScopedComPtr<IDWriteFontResource> fontResource;
HRN(fontFace5->GetFontResource(&fontResource));
SkTScopedComPtr<IDWriteFontFace5> newFontFace5;
HRN(fontResource->CreateFontFace(fDWriteFont->GetSimulations(),
fontAxisValue.get(),
fontAxisCount,
&newFontFace5));
SkTScopedComPtr<IDWriteFontFace> newFontFace;
HRN(newFontFace5->QueryInterface(&newFontFace));
return DWriteFontTypeface::Make(fFactory.get(),
newFontFace.get(),
fDWriteFont.get(),
fDWriteFontFamily.get(),
fLoaders,
args.getPalette());
}
#endif
// If the palette args have changed, a new font will need to be created.
if (args.getPalette().index != fRequestedPalette.index ||
args.getPalette().overrideCount != fRequestedPalette.overrideCount ||
memcmp(args.getPalette().overrides, fRequestedPalette.overrides,
fRequestedPalette.overrideCount * sizeof(fRequestedPalette.overrides[0])))
{
return DWriteFontTypeface::Make(fFactory.get(),
fDWriteFontFace.get(),
fDWriteFont.get(),
fDWriteFontFamily.get(),
fLoaders,
args.getPalette());
}
return sk_ref_sp(this);
}
std::unique_ptr<SkStreamAsset> DWriteFontTypeface::onOpenStream(int* ttcIndex) const {
*ttcIndex = fDWriteFontFace->GetIndex();
UINT32 numFiles = 0;
HRNM(fDWriteFontFace->GetFiles(&numFiles, nullptr),
"Could not get number of font files.");
if (numFiles != 1) {
return nullptr;
}
SkTScopedComPtr<IDWriteFontFile> fontFile;
HRNM(fDWriteFontFace->GetFiles(&numFiles, &fontFile), "Could not get font files.");
const void* fontFileKey;
UINT32 fontFileKeySize;
HRNM(fontFile->GetReferenceKey(&fontFileKey, &fontFileKeySize),
"Could not get font file reference key.");
SkTScopedComPtr<IDWriteFontFileLoader> fontFileLoader;
HRNM(fontFile->GetLoader(&fontFileLoader), "Could not get font file loader.");
SkTScopedComPtr<IDWriteFontFileStream> fontFileStream;
HRNM(fontFileLoader->CreateStreamFromKey(fontFileKey, fontFileKeySize,
&fontFileStream),
"Could not create font file stream.");
return std::unique_ptr<SkStreamAsset>(new SkDWriteFontFileStream(fontFileStream.get()));
}
std::unique_ptr<SkScalerContext> DWriteFontTypeface::onCreateScalerContext(
const SkScalerContextEffects& effects, const SkDescriptor* desc) const
{
return std::make_unique<SkScalerContext_DW>(
sk_ref_sp(const_cast<DWriteFontTypeface*>(this)), effects, desc);
}
void DWriteFontTypeface::onFilterRec(SkScalerContextRec* rec) const {
if (rec->fFlags & SkScalerContext::kLCD_Vertical_Flag) {
rec->fMaskFormat = SkMask::kA8_Format;
rec->fFlags |= SkScalerContext::kGenA8FromLCD_Flag;
}
unsigned flagsWeDontSupport = SkScalerContext::kForceAutohinting_Flag |
SkScalerContext::kEmbolden_Flag |
SkScalerContext::kLCD_Vertical_Flag;
rec->fFlags &= ~flagsWeDontSupport;
SkFontHinting h = rec->getHinting();
// DirectWrite2 allows for hinting to be turned off. Force everything else to normal.
if (h != SkFontHinting::kNone || !fFactory2 || !fDWriteFontFace2) {
h = SkFontHinting::kNormal;
}
rec->setHinting(h);
#if defined(SK_FONT_HOST_USE_SYSTEM_SETTINGS)
IDWriteFactory* factory = sk_get_dwrite_factory();
if (factory != nullptr) {
SkTScopedComPtr<IDWriteRenderingParams> defaultRenderingParams;
if (SUCCEEDED(factory->CreateRenderingParams(&defaultRenderingParams))) {
float gamma = defaultRenderingParams->GetGamma();
rec->setDeviceGamma(gamma);
rec->setPaintGamma(gamma);
rec->setContrast(defaultRenderingParams->GetEnhancedContrast());
}
}
#endif
}
///////////////////////////////////////////////////////////////////////////////
//PDF Support
static void glyph_to_unicode_map(IDWriteFontFace* fontFace, DWRITE_UNICODE_RANGE range,
UINT32* remainingGlyphCount, UINT32 numGlyphs,
SkUnichar* glyphToUnicode)
{
constexpr const int batchSize = 128;
UINT32 codepoints[batchSize];
UINT16 glyphs[batchSize];
for (UINT32 c = range.first; c <= range.last && *remainingGlyphCount != 0; c += batchSize) {
UINT32 numBatchedCodePoints = std::min<UINT32>(range.last - c + 1, batchSize);
for (UINT32 i = 0; i < numBatchedCodePoints; ++i) {
codepoints[i] = c + i;
}
HRVM(fontFace->GetGlyphIndices(codepoints, numBatchedCodePoints, glyphs),
"Failed to get glyph indexes.");
for (UINT32 i = 0; i < numBatchedCodePoints; ++i) {
UINT16 glyph = glyphs[i];
// Intermittent DW bug on Windows 10. See crbug.com/470146.
if (glyph >= numGlyphs) {
return;
}
if (0 < glyph && glyphToUnicode[glyph] == 0) {
glyphToUnicode[glyph] = c + i; // Always use lowest-index unichar.
--*remainingGlyphCount;
}
}
}
}
void DWriteFontTypeface::getGlyphToUnicodeMap(SkUnichar* glyphToUnicode) const {
IDWriteFontFace* face = fDWriteFontFace.get();
UINT32 numGlyphs = face->GetGlyphCount();
sk_bzero(glyphToUnicode, sizeof(SkUnichar) * numGlyphs);
UINT32 remainingGlyphCount = numGlyphs;
if (fDWriteFontFace1) {
IDWriteFontFace1* face1 = fDWriteFontFace1.get();
UINT32 numRanges = 0;
HRESULT hr = face1->GetUnicodeRanges(0, nullptr, &numRanges);
if (hr != E_NOT_SUFFICIENT_BUFFER && FAILED(hr)) {
HRVM(hr, "Failed to get number of ranges.");
}
std::unique_ptr<DWRITE_UNICODE_RANGE[]> ranges(new DWRITE_UNICODE_RANGE[numRanges]);
HRVM(face1->GetUnicodeRanges(numRanges, ranges.get(), &numRanges), "Failed to get ranges.");
for (UINT32 i = 0; i < numRanges; ++i) {
glyph_to_unicode_map(face1, ranges[i], &remainingGlyphCount, numGlyphs, glyphToUnicode);
}
} else {
glyph_to_unicode_map(face, {0, 0x10FFFF}, &remainingGlyphCount, numGlyphs, glyphToUnicode);
}
}
std::unique_ptr<SkAdvancedTypefaceMetrics> DWriteFontTypeface::onGetAdvancedMetrics() const {
std::unique_ptr<SkAdvancedTypefaceMetrics> info(nullptr);
DWRITE_FONT_METRICS dwfm;
fDWriteFontFace->GetMetrics(&dwfm);
info.reset(new SkAdvancedTypefaceMetrics);
info->fAscent = SkToS16(dwfm.ascent);
info->fDescent = SkToS16(dwfm.descent);
info->fCapHeight = SkToS16(dwfm.capHeight);
{
SkTScopedComPtr<IDWriteLocalizedStrings> postScriptNames;
BOOL exists = FALSE;
if (FAILED(fDWriteFont->GetInformationalStrings(
DWRITE_INFORMATIONAL_STRING_POSTSCRIPT_NAME,
&postScriptNames,
&exists)) ||
!exists ||
FAILED(sk_get_locale_string(postScriptNames.get(), nullptr, &info->fPostScriptName)))
{
SkDEBUGF("Unable to get postscript name for typeface %p\n", this);
}
}
DWRITE_FONT_FACE_TYPE fontType = fDWriteFontFace->GetType();
if (fontType != DWRITE_FONT_FACE_TYPE_TRUETYPE &&
fontType != DWRITE_FONT_FACE_TYPE_CFF &&
fontType != DWRITE_FONT_FACE_TYPE_TRUETYPE_COLLECTION &&
fontType != DWRITE_FONT_FACE_TYPE_OPENTYPE_COLLECTION)
{
return info;
}
// Simulated fonts aren't really TrueType fonts.
if (fDWriteFontFace->GetSimulations() == DWRITE_FONT_SIMULATIONS_NONE) {
info->fType = SkAdvancedTypefaceMetrics::kTrueType_Font;
}
AutoTDWriteTable<SkOTTableHead> headTable(fDWriteFontFace.get());
AutoTDWriteTable<SkOTTablePostScript> postTable(fDWriteFontFace.get());
AutoTDWriteTable<SkOTTableHorizontalHeader> hheaTable(fDWriteFontFace.get());
AutoTDWriteTable<SkOTTableOS2> os2Table(fDWriteFontFace.get());
if (!headTable.fExists || !postTable.fExists || !hheaTable.fExists || !os2Table.fExists) {
return info;
}
SkOTUtils::SetAdvancedTypefaceFlags(os2Table->version.v4.fsType, info.get());
// There are versions of DirectWrite which support named instances for system variation fonts,
// but no means to indicate that such a typeface is a variation.
AutoTDWriteTable<SkOTTableFontVariations> fvarTable(fDWriteFontFace.get());
if (fvarTable.fExists) {
info->fFlags |= SkAdvancedTypefaceMetrics::kVariable_FontFlag;
}
//There exist CJK fonts which set the IsFixedPitch and Monospace bits,
//but have full width, latin half-width, and half-width kana.
bool fixedWidth = (postTable->isFixedPitch &&
(1 == SkEndian_SwapBE16(hheaTable->numberOfHMetrics)));
//Monospace
if (fixedWidth) {
info->fStyle |= SkAdvancedTypefaceMetrics::kFixedPitch_Style;
}
//Italic
if (os2Table->version.v0.fsSelection.field.Italic) {
info->fStyle |= SkAdvancedTypefaceMetrics::kItalic_Style;
}
//Serif
using SerifStyle = SkPanose::Data::TextAndDisplay::SerifStyle;
SerifStyle serifStyle = os2Table->version.v0.panose.data.textAndDisplay.bSerifStyle;
if (SkPanose::FamilyType::TextAndDisplay == os2Table->version.v0.panose.bFamilyType) {
if (SerifStyle::Cove == serifStyle ||
SerifStyle::ObtuseCove == serifStyle ||
SerifStyle::SquareCove == serifStyle ||
SerifStyle::ObtuseSquareCove == serifStyle ||
SerifStyle::Square == serifStyle ||
SerifStyle::Thin == serifStyle ||
SerifStyle::Bone == serifStyle ||
SerifStyle::Exaggerated == serifStyle ||
SerifStyle::Triangle == serifStyle)
{
info->fStyle |= SkAdvancedTypefaceMetrics::kSerif_Style;
}
//Script
} else if (SkPanose::FamilyType::Script == os2Table->version.v0.panose.bFamilyType) {
info->fStyle |= SkAdvancedTypefaceMetrics::kScript_Style;
}
info->fItalicAngle = SkEndian_SwapBE32(postTable->italicAngle) >> 16;
info->fBBox = SkIRect::MakeLTRB((int32_t)SkEndian_SwapBE16((uint16_t)headTable->xMin),
(int32_t)SkEndian_SwapBE16((uint16_t)headTable->yMax),
(int32_t)SkEndian_SwapBE16((uint16_t)headTable->xMax),
(int32_t)SkEndian_SwapBE16((uint16_t)headTable->yMin));
return info;
}
class StreamFontFileLoader : public IDWriteFontFileLoader {
public:
// IUnknown methods
SK_STDMETHODIMP QueryInterface(REFIID iid, void** ppvObject) override;
SK_STDMETHODIMP_(ULONG) AddRef() override;
SK_STDMETHODIMP_(ULONG) Release() override;
// IDWriteFontFileLoader methods
SK_STDMETHODIMP CreateStreamFromKey(
void const* fontFileReferenceKey,
UINT32 fontFileReferenceKeySize,
IDWriteFontFileStream** fontFileStream) override;
// Takes ownership of stream.
static HRESULT Create(std::unique_ptr<SkStreamAsset> stream,
StreamFontFileLoader** streamFontFileLoader) {
*streamFontFileLoader = new StreamFontFileLoader(std::move(stream));
if (nullptr == *streamFontFileLoader) {
return E_OUTOFMEMORY;
}
return S_OK;
}
private:
StreamFontFileLoader(std::unique_ptr<SkStreamAsset> stream)
: fStream(std::move(stream)), fRefCount(1)
{}
virtual ~StreamFontFileLoader() { }
std::unique_ptr<SkStreamAsset> fStream;
ULONG fRefCount;
};
SK_STDMETHODIMP StreamFontFileLoader::QueryInterface(REFIID iid, void** ppvObject) {
if (iid == IID_IUnknown || iid == __uuidof(IDWriteFontFileLoader)) {
*ppvObject = this;
AddRef();
return S_OK;
} else {
*ppvObject = nullptr;
return E_NOINTERFACE;
}
}
SK_STDMETHODIMP_(ULONG) StreamFontFileLoader::AddRef() {
return InterlockedIncrement(&fRefCount);
}
SK_STDMETHODIMP_(ULONG) StreamFontFileLoader::Release() {
ULONG newCount = InterlockedDecrement(&fRefCount);
if (0 == newCount) {
delete this;
}
return newCount;
}
SK_STDMETHODIMP StreamFontFileLoader::CreateStreamFromKey(
void const* fontFileReferenceKey,
UINT32 fontFileReferenceKeySize,
IDWriteFontFileStream** fontFileStream)
{
SkTScopedComPtr<SkDWriteFontFileStreamWrapper> stream;
HR(SkDWriteFontFileStreamWrapper::Create(fStream->duplicate().release(), &stream));
*fontFileStream = stream.release();
return S_OK;
}
class StreamFontFileEnumerator : public IDWriteFontFileEnumerator {
public:
// IUnknown methods
SK_STDMETHODIMP QueryInterface(REFIID iid, void** ppvObject) override;
SK_STDMETHODIMP_(ULONG) AddRef() override;
SK_STDMETHODIMP_(ULONG) Release() override;
// IDWriteFontFileEnumerator methods
SK_STDMETHODIMP MoveNext(BOOL* hasCurrentFile) override;
SK_STDMETHODIMP GetCurrentFontFile(IDWriteFontFile** fontFile) override;
static HRESULT Create(IDWriteFactory* factory, IDWriteFontFileLoader* fontFileLoader,
StreamFontFileEnumerator** streamFontFileEnumerator) {
*streamFontFileEnumerator = new StreamFontFileEnumerator(factory, fontFileLoader);
if (nullptr == *streamFontFileEnumerator) {
return E_OUTOFMEMORY;
}
return S_OK;
}
private:
StreamFontFileEnumerator(IDWriteFactory* factory, IDWriteFontFileLoader* fontFileLoader);
virtual ~StreamFontFileEnumerator() { }
ULONG fRefCount;
SkTScopedComPtr<IDWriteFactory> fFactory;
SkTScopedComPtr<IDWriteFontFile> fCurrentFile;
SkTScopedComPtr<IDWriteFontFileLoader> fFontFileLoader;
bool fHasNext;
};
StreamFontFileEnumerator::StreamFontFileEnumerator(IDWriteFactory* factory,
IDWriteFontFileLoader* fontFileLoader)
: fRefCount(1)
, fFactory(SkRefComPtr(factory))
, fCurrentFile()
, fFontFileLoader(SkRefComPtr(fontFileLoader))
, fHasNext(true)
{ }
SK_STDMETHODIMP StreamFontFileEnumerator::QueryInterface(REFIID iid, void** ppvObject) {
if (iid == IID_IUnknown || iid == __uuidof(IDWriteFontFileEnumerator)) {
*ppvObject = this;
AddRef();
return S_OK;
} else {
*ppvObject = nullptr;
return E_NOINTERFACE;
}
}
SK_STDMETHODIMP_(ULONG) StreamFontFileEnumerator::AddRef() {
return InterlockedIncrement(&fRefCount);
}
SK_STDMETHODIMP_(ULONG) StreamFontFileEnumerator::Release() {
ULONG newCount = InterlockedDecrement(&fRefCount);
if (0 == newCount) {
delete this;
}
return newCount;
}
SK_STDMETHODIMP StreamFontFileEnumerator::MoveNext(BOOL* hasCurrentFile) {
*hasCurrentFile = FALSE;
if (!fHasNext) {
return S_OK;
}
fHasNext = false;
UINT32 fontFileReferenceKey = 0;
HR(fFactory->CreateCustomFontFileReference(
&fontFileReferenceKey, //cannot be nullptr
sizeof(fontFileReferenceKey), //even if this is 0
fFontFileLoader.get(),
&fCurrentFile));
*hasCurrentFile = TRUE;
return S_OK;
}
SK_STDMETHODIMP StreamFontFileEnumerator::GetCurrentFontFile(IDWriteFontFile** fontFile) {
if (fCurrentFile.get() == nullptr) {
*fontFile = nullptr;
return E_FAIL;
}
*fontFile = SkRefComPtr(fCurrentFile.get());
return S_OK;
}
class StreamFontCollectionLoader : public IDWriteFontCollectionLoader {
public:
// IUnknown methods
SK_STDMETHODIMP QueryInterface(REFIID iid, void** ppvObject) override;
SK_STDMETHODIMP_(ULONG) AddRef() override;
SK_STDMETHODIMP_(ULONG) Release() override;
// IDWriteFontCollectionLoader methods
SK_STDMETHODIMP CreateEnumeratorFromKey(
IDWriteFactory* factory,
void const* collectionKey,
UINT32 collectionKeySize,
IDWriteFontFileEnumerator** fontFileEnumerator) override;
static HRESULT Create(IDWriteFontFileLoader* fontFileLoader,
StreamFontCollectionLoader** streamFontCollectionLoader) {
*streamFontCollectionLoader = new StreamFontCollectionLoader(fontFileLoader);
if (nullptr == *streamFontCollectionLoader) {
return E_OUTOFMEMORY;
}
return S_OK;
}
private:
StreamFontCollectionLoader(IDWriteFontFileLoader* fontFileLoader)
: fRefCount(1)
, fFontFileLoader(SkRefComPtr(fontFileLoader))
{ }
virtual ~StreamFontCollectionLoader() { }
ULONG fRefCount;
SkTScopedComPtr<IDWriteFontFileLoader> fFontFileLoader;
};
SK_STDMETHODIMP StreamFontCollectionLoader::QueryInterface(REFIID iid, void** ppvObject) {
if (iid == IID_IUnknown || iid == __uuidof(IDWriteFontCollectionLoader)) {
*ppvObject = this;
AddRef();
return S_OK;
} else {
*ppvObject = nullptr;
return E_NOINTERFACE;
}
}
SK_STDMETHODIMP_(ULONG) StreamFontCollectionLoader::AddRef() {
return InterlockedIncrement(&fRefCount);
}
SK_STDMETHODIMP_(ULONG) StreamFontCollectionLoader::Release() {
ULONG newCount = InterlockedDecrement(&fRefCount);
if (0 == newCount) {
delete this;
}
return newCount;
}
template <typename T> class SkAutoIDWriteUnregister {
public:
SkAutoIDWriteUnregister(IDWriteFactory* factory, T* unregister)
: fFactory(factory), fUnregister(unregister)
{ }
SkAutoIDWriteUnregister(const SkAutoIDWriteUnregister&) = delete;
SkAutoIDWriteUnregister& operator=(const SkAutoIDWriteUnregister&) = delete;
SkAutoIDWriteUnregister(SkAutoIDWriteUnregister&&) = delete;
SkAutoIDWriteUnregister& operator=(SkAutoIDWriteUnregister&&) = delete;
~SkAutoIDWriteUnregister() {
if (fUnregister) {
unregister(fFactory, fUnregister);
}
}
T* detatch() {
T* old = fUnregister;
fUnregister = nullptr;
return old;
}
private:
HRESULT unregister(IDWriteFactory* factory, IDWriteFontFileLoader* unregister) {
return factory->UnregisterFontFileLoader(unregister);
}
HRESULT unregister(IDWriteFactory* factory, IDWriteFontCollectionLoader* unregister) {
return factory->UnregisterFontCollectionLoader(unregister);
}
IDWriteFactory* fFactory;
T* fUnregister;
};
SK_STDMETHODIMP StreamFontCollectionLoader::CreateEnumeratorFromKey(
IDWriteFactory* factory,
void const* collectionKey,
UINT32 collectionKeySize,
IDWriteFontFileEnumerator** fontFileEnumerator)
{
SkTScopedComPtr<StreamFontFileEnumerator> enumerator;
HR(StreamFontFileEnumerator::Create(factory, fFontFileLoader.get(), &enumerator));
*fontFileEnumerator = enumerator.release();
return S_OK;
}
static HRESULT apply_fontargument_variation(SkTScopedComPtr<IDWriteFontFace>& fontFace,
const SkFontArguments& args)
{
#if defined(NTDDI_WIN10_RS3) && NTDDI_VERSION >= NTDDI_WIN10_RS3
SkTScopedComPtr<IDWriteFontFace5> fontFace5;
if (FAILED(fontFace->QueryInterface(&fontFace5)) || !fontFace5->HasVariations()) {
return S_OK;
}
UINT32 fontAxisCount = fontFace5->GetFontAxisValueCount();
UINT32 argsCoordCount = args.getVariationDesignPosition().coordinateCount;
AutoSTMalloc<8, DWRITE_FONT_AXIS_VALUE> variation(fontAxisCount);
SkTScopedComPtr<IDWriteFontResource> fontResource;
HR(fontFace5->GetFontResource(&fontResource));
HR(fontResource->GetDefaultFontAxisValues(variation, fontAxisCount));
for (UINT32 fontAxisIndex = 0; fontAxisIndex < fontAxisCount; ++fontAxisIndex) {
DWRITE_FONT_AXIS_VALUE& fontCoordinate = variation[fontAxisIndex];
for (UINT32 argsCoordIndex = argsCoordCount; argsCoordIndex --> 0;) {
const SkFontArguments::VariationPosition::Coordinate& argsCoordinate =
args.getVariationDesignPosition().coordinates[argsCoordIndex];
if (SkEndian_SwapBE32(fontCoordinate.axisTag) == argsCoordinate.axis) {
fontCoordinate.value = argsCoordinate.value;
break;
}
}
}
SkTScopedComPtr<IDWriteFontFace5> fontFace5_Out;
HR(fontResource->CreateFontFace(DWRITE_FONT_SIMULATIONS_NONE,
variation.get(), fontAxisCount,
&fontFace5_Out));
fontFace.reset();
HR(fontFace5_Out->QueryInterface(&fontFace));
#endif
return S_OK;
}
sk_sp<SkTypeface> DWriteFontTypeface::MakeFromStream(std::unique_ptr<SkStreamAsset> stream,
const SkFontArguments& args) {
// TODO: do we need to use some user provided factory?
IDWriteFactory* factory = sk_get_dwrite_factory();
if (nullptr == factory) {
return nullptr;
}
SkTScopedComPtr<StreamFontFileLoader> fontFileLoader;
HRN(StreamFontFileLoader::Create(std::move(stream), &fontFileLoader));
HRN(factory->RegisterFontFileLoader(fontFileLoader.get()));
SkAutoIDWriteUnregister<StreamFontFileLoader> autoUnregisterFontFileLoader(
factory, fontFileLoader.get());
SkTScopedComPtr<StreamFontCollectionLoader> fontCollectionLoader;
HRN(StreamFontCollectionLoader::Create(fontFileLoader.get(), &fontCollectionLoader));
HRN(factory->RegisterFontCollectionLoader(fontCollectionLoader.get()));
SkAutoIDWriteUnregister<StreamFontCollectionLoader> autoUnregisterFontCollectionLoader(
factory, fontCollectionLoader.get());
SkTScopedComPtr<IDWriteFontCollection> fontCollection;
HRN(factory->CreateCustomFontCollection(fontCollectionLoader.get(), nullptr, 0,
&fontCollection));
// Find the first non-simulated font which has the given ttc index.
UINT32 familyCount = fontCollection->GetFontFamilyCount();
for (UINT32 familyIndex = 0; familyIndex < familyCount; ++familyIndex) {
SkTScopedComPtr<IDWriteFontFamily> fontFamily;
HRN(fontCollection->GetFontFamily(familyIndex, &fontFamily));
UINT32 fontCount = fontFamily->GetFontCount();
for (UINT32 fontIndex = 0; fontIndex < fontCount; ++fontIndex) {
SkTScopedComPtr<IDWriteFont> font;
HRN(fontFamily->GetFont(fontIndex, &font));
// Skip if the current font is simulated
if (font->GetSimulations() != DWRITE_FONT_SIMULATIONS_NONE) {
continue;
}
SkTScopedComPtr<IDWriteFontFace> fontFace;
HRN(font->CreateFontFace(&fontFace));
int faceIndex = fontFace->GetIndex();
int ttcIndex = args.getCollectionIndex();
// Skip if the current face index does not match the ttcIndex
if (faceIndex != ttcIndex) {
continue;
}
apply_fontargument_variation(fontFace, args);
return DWriteFontTypeface::Make(
factory, fontFace.get(), font.get(), fontFamily.get(),
sk_make_sp<DWriteFontTypeface::Loaders>(
factory,
autoUnregisterFontFileLoader.detatch(),
autoUnregisterFontCollectionLoader.detatch()),
args.getPalette());
}
}
return nullptr;
}
#endif//defined(SK_BUILD_FOR_WIN)