blob: a266bf90864aae9028c50354c1d850c861003699 [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.
*/
#ifndef SkTypeface_win_dw_DEFINED
#define SkTypeface_win_dw_DEFINED
#include "include/core/SkTypeface.h"
#include "src/core/SkAdvancedTypefaceMetrics.h"
#include "src/core/SkLeanWindows.h"
#include "src/core/SkTypefaceCache.h"
#include "src/utils/win/SkDWrite.h"
#include "src/utils/win/SkHRESULT.h"
#include "src/utils/win/SkTScopedComPtr.h"
#include <dwrite.h>
#include <dwrite_1.h>
#include <dwrite_2.h>
#include <dwrite_3.h>
class SkFontDescriptor;
struct SkScalerContextRec;
static SkFontStyle get_style(IDWriteFont* font) {
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;
}
return SkFontStyle(weight, width, slant);
}
class DWriteFontTypeface : public SkTypeface {
public:
struct Loaders : public SkNVRefCnt<Loaders> {
Loaders(IDWriteFactory* factory,
IDWriteFontFileLoader* fontFileLoader,
IDWriteFontCollectionLoader* fontCollectionLoader)
: fFactory(SkRefComPtr(factory))
, fDWriteFontFileLoader(SkRefComPtr(fontFileLoader))
, fDWriteFontCollectionLoader(SkRefComPtr(fontCollectionLoader))
{}
Loaders(const Loaders&) = delete;
Loaders& operator=(const Loaders&) = delete;
Loaders(Loaders&&) = delete;
Loaders& operator=(Loaders&&) = delete;
~Loaders();
SkTScopedComPtr<IDWriteFactory> fFactory;
SkTScopedComPtr<IDWriteFontFileLoader> fDWriteFontFileLoader;
SkTScopedComPtr<IDWriteFontCollectionLoader> fDWriteFontCollectionLoader;
};
private:
DWriteFontTypeface(const SkFontStyle& style,
IDWriteFactory* factory,
IDWriteFontFace* fontFace,
IDWriteFont* font,
IDWriteFontFamily* fontFamily,
sk_sp<Loaders> loaders)
: SkTypeface(style, false)
, fFactory(SkRefComPtr(factory))
, fDWriteFontFamily(SkRefComPtr(fontFamily))
, fDWriteFont(SkRefComPtr(font))
, fDWriteFontFace(SkRefComPtr(fontFace))
, 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 (!SUCCEEDED(fFactory->QueryInterface(&fFactory2))) {
SkASSERT_RELEASE(nullptr == fFactory2.get());
}
if (fDWriteFontFace1 && fDWriteFontFace1->IsMonospacedFont()) {
this->setIsFixedPitch(true);
}
}
public:
SkTScopedComPtr<IDWriteFactory> fFactory;
SkTScopedComPtr<IDWriteFactory2> fFactory2;
SkTScopedComPtr<IDWriteFontFamily> fDWriteFontFamily;
SkTScopedComPtr<IDWriteFont> fDWriteFont;
SkTScopedComPtr<IDWriteFontFace> fDWriteFontFace;
SkTScopedComPtr<IDWriteFontFace1> fDWriteFontFace1;
SkTScopedComPtr<IDWriteFontFace2> fDWriteFontFace2;
SkTScopedComPtr<IDWriteFontFace4> fDWriteFontFace4;
static sk_sp<DWriteFontTypeface> Make(
IDWriteFactory* factory,
IDWriteFontFace* fontFace,
IDWriteFont* font,
IDWriteFontFamily* fontFamily,
sk_sp<Loaders> loaders)
{
return sk_sp<DWriteFontTypeface>(new DWriteFontTypeface(
get_style(font), factory, fontFace, font, fontFamily, std::move(loaders)));
}
protected:
void weak_dispose() const override {
fLoaders.reset();
//SkTypefaceCache::Remove(this);
INHERITED::weak_dispose();
}
sk_sp<SkTypeface> onMakeClone(const SkFontArguments&) const override;
std::unique_ptr<SkStreamAsset> onOpenStream(int* ttcIndex) const override;
SkScalerContext* onCreateScalerContext(const SkScalerContextEffects&,
const SkDescriptor*) const override;
void onFilterRec(SkScalerContextRec*) const override;
void getGlyphToUnicodeMap(SkUnichar* glyphToUnicode) const override;
std::unique_ptr<SkAdvancedTypefaceMetrics> onGetAdvancedMetrics() const override;
void onGetFontDescriptor(SkFontDescriptor*, bool*) const override;
void onCharsToGlyphs(const SkUnichar* chars, int count, SkGlyphID glyphs[]) const override;
int onCountGlyphs() const override;
void getPostScriptGlyphNames(SkString*) const override;
int onGetUPEM() const override;
void onGetFamilyName(SkString* familyName) const override;
SkTypeface::LocalizedStrings* onCreateFamilyNameIterator() const override;
int onGetVariationDesignPosition(SkFontArguments::VariationPosition::Coordinate coordinates[],
int coordinateCount) const override;
int onGetVariationDesignParameters(SkFontParameters::Variation::Axis parameters[],
int parameterCount) const override;
int onGetTableTags(SkFontTableTag tags[]) const override;
size_t onGetTableData(SkFontTableTag, size_t offset, size_t length, void* data) const override;
sk_sp<SkData> onCopyTableData(SkFontTableTag) const override;
private:
mutable sk_sp<Loaders> fLoaders;
typedef SkTypeface INHERITED;
};
#endif