/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 4 -*- * ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla Public License Version * 1.1 (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * http://www.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * The Original Code is Mozilla Foundation code. * * The Initial Developer of the Original Code is Mozilla Foundation. * Portions created by the Initial Developer are Copyright (C) 2005 * the Initial Developer. All Rights Reserved. * * Contributor(s): * Stuart Parmenter * Masayuki Nakano * John Daggett * * Alternatively, the contents of this file may be used under the terms of * either the GNU General Public License Version 2 or later (the "GPL"), or * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ #ifndef GFX_WINDOWSFONTS_H #define GFX_WINDOWSFONTS_H #include "prtypes.h" #include "gfxTypes.h" #include "gfxColor.h" #include "gfxFont.h" #include "gfxMatrix.h" #include "gfxFontUtils.h" #include "gfxUserFontSet.h" #include "nsDataHashtable.h" #include #include // xxx - used in FontEntry. should be trimmed, moz code doesn't use // exceptions. use gfxSparseBitSet instead? #include /** * List of different types of fonts we support on Windows. * These can generally be lumped in to 3 categories where we have to * do special things: Really old fonts bitmap and vector fonts (device * and raster), Type 1 fonts, and TrueType/OpenType fonts. * * This list is sorted in order from least prefered to most prefered. * We prefer Type1 fonts over OpenType fonts to avoid falling back to * things like Arial (opentype) when you ask for Helvetica (type1) **/ enum gfxWindowsFontType { GFX_FONT_TYPE_UNKNOWN = 0, GFX_FONT_TYPE_DEVICE, GFX_FONT_TYPE_RASTER, GFX_FONT_TYPE_TRUETYPE, GFX_FONT_TYPE_PS_OPENTYPE, GFX_FONT_TYPE_TT_OPENTYPE, GFX_FONT_TYPE_TYPE1 }; /** * FontFamily is a class that describes one of the fonts on the users system. It holds * each FontEntry (maps more directly to a font face) which holds font type, charset info * and character map info. */ class FontEntry; class FontFamily : public gfxFontFamily { public: FontFamily(const nsAString& aName) : gfxFontFamily(aName), mIsBadUnderlineFontFamily(PR_FALSE) { } FontEntry *FindFontEntry(const gfxFontStyle& aFontStyle); private: friend class gfxWindowsPlatform; void FindStyleVariations(); static int CALLBACK FamilyAddStylesProc(const ENUMLOGFONTEXW *lpelfe, const NEWTEXTMETRICEXW *nmetrics, DWORD fontType, LPARAM data); protected: PRBool FindWeightsForStyle(gfxFontEntry* aFontsForWeights[], PRBool anItalic, PRInt16 aStretch); public: PRPackedBool mIsBadUnderlineFontFamily; }; class FontEntry : public gfxFontEntry { public: FontEntry(const nsAString& aFaceName, gfxWindowsFontType aFontType, PRBool aItalic, PRUint16 aWeight, gfxUserFontData *aUserFontData) : gfxFontEntry(aFaceName), mFontType(aFontType), mForceGDI(PR_FALSE), mUnknownCMAP(PR_FALSE), mUnicodeFont(PR_FALSE), mSymbolFont(PR_FALSE), mCharset(0), mUnicodeRanges(0) { mUserFontData = aUserFontData; mItalic = aItalic; mWeight = aWeight; if (IsType1()) mForceGDI = PR_TRUE; mIsUserFont = aUserFontData != nsnull; } FontEntry(const FontEntry& aFontEntry) : gfxFontEntry(aFontEntry), mWindowsFamily(aFontEntry.mWindowsFamily), mWindowsPitch(aFontEntry.mWindowsPitch), mFontType(aFontEntry.mFontType), mForceGDI(aFontEntry.mForceGDI), mUnknownCMAP(aFontEntry.mUnknownCMAP), mUnicodeFont(aFontEntry.mUnicodeFont), mSymbolFont(aFontEntry.mSymbolFont), mCharset(aFontEntry.mCharset), mUnicodeRanges(aFontEntry.mUnicodeRanges) { } static void InitializeFontEmbeddingProcs(); // create a font entry from downloaded font data static FontEntry* LoadFont(const gfxProxyFontEntry &aProxyEntry, const PRUint8 *aFontData, PRUint32 aLength); // create a font entry for a font with a given name static FontEntry* CreateFontEntry(const nsAString& aName, gfxWindowsFontType aFontType, PRBool aItalic, PRUint16 aWeight, gfxUserFontData* aUserFontData, HDC hdc = 0, LOGFONTW *aLogFont = nsnull); // create a font entry for a font referenced by its fullname static FontEntry* LoadLocalFont(const gfxProxyFontEntry &aProxyEntry, const nsAString& aFullname); static void FillLogFont(LOGFONTW *aLogFont, const nsAString& aName, gfxWindowsFontType aFontType, PRBool aItalic, PRUint16 aWeight, gfxFloat aSize); static gfxWindowsFontType DetermineFontType(const NEWTEXTMETRICW& metrics, DWORD fontType) { gfxWindowsFontType feType; if (metrics.ntmFlags & NTM_TYPE1) feType = GFX_FONT_TYPE_TYPE1; else if (metrics.ntmFlags & NTM_PS_OPENTYPE) feType = GFX_FONT_TYPE_PS_OPENTYPE; else if (metrics.ntmFlags & NTM_TT_OPENTYPE) feType = GFX_FONT_TYPE_TT_OPENTYPE; else if (fontType == TRUETYPE_FONTTYPE) feType = GFX_FONT_TYPE_TRUETYPE; else if (fontType == RASTER_FONTTYPE) feType = GFX_FONT_TYPE_RASTER; else if (fontType == DEVICE_FONTTYPE) feType = GFX_FONT_TYPE_DEVICE; else feType = GFX_FONT_TYPE_UNKNOWN; return feType; } PRBool IsType1() const { return (mFontType == GFX_FONT_TYPE_TYPE1); } PRBool IsTrueType() const { return (mFontType == GFX_FONT_TYPE_TRUETYPE || mFontType == GFX_FONT_TYPE_PS_OPENTYPE || mFontType == GFX_FONT_TYPE_TT_OPENTYPE); } PRBool IsCrappyFont() const { /* return if it is a bitmap not a unicode font */ return (!mUnicodeFont || mSymbolFont || IsType1()); } PRBool MatchesGenericFamily(const nsACString& aGeneric) const { if (aGeneric.IsEmpty()) return PR_TRUE; // Japanese 'Mincho' fonts do not belong to FF_MODERN even if // they are fixed pitch because they have variable stroke width. if (mWindowsFamily == FF_ROMAN && mWindowsPitch & FIXED_PITCH) { return aGeneric.EqualsLiteral("monospace"); } // Japanese 'Gothic' fonts do not belong to FF_SWISS even if // they are variable pitch because they have constant stroke width. if (mWindowsFamily == FF_MODERN && mWindowsPitch & VARIABLE_PITCH) { return aGeneric.EqualsLiteral("sans-serif"); } // All other fonts will be grouped correctly using family... switch (mWindowsFamily) { case FF_DONTCARE: return PR_TRUE; case FF_ROMAN: return aGeneric.EqualsLiteral("serif"); case FF_SWISS: return aGeneric.EqualsLiteral("sans-serif"); case FF_MODERN: return aGeneric.EqualsLiteral("monospace"); case FF_SCRIPT: return aGeneric.EqualsLiteral("cursive"); case FF_DECORATIVE: return aGeneric.EqualsLiteral("fantasy"); } return PR_FALSE; } PRBool SupportsLangGroup(const nsACString& aLangGroup) const { if (aLangGroup.IsEmpty()) return PR_TRUE; PRInt16 bit = -1; /* map our langgroup names in to Windows charset bits */ if (aLangGroup.EqualsLiteral("x-western")) { bit = ANSI_CHARSET; } else if (aLangGroup.EqualsLiteral("ja")) { bit = SHIFTJIS_CHARSET; } else if (aLangGroup.EqualsLiteral("ko")) { bit = HANGEUL_CHARSET; } else if (aLangGroup.EqualsLiteral("ko-XXX")) { bit = JOHAB_CHARSET; } else if (aLangGroup.EqualsLiteral("zh-CN")) { bit = GB2312_CHARSET; } else if (aLangGroup.EqualsLiteral("zh-TW")) { bit = CHINESEBIG5_CHARSET; } else if (aLangGroup.EqualsLiteral("el")) { bit = GREEK_CHARSET; } else if (aLangGroup.EqualsLiteral("tr")) { bit = TURKISH_CHARSET; } else if (aLangGroup.EqualsLiteral("he")) { bit = HEBREW_CHARSET; } else if (aLangGroup.EqualsLiteral("ar")) { bit = ARABIC_CHARSET; } else if (aLangGroup.EqualsLiteral("x-baltic")) { bit = BALTIC_CHARSET; } else if (aLangGroup.EqualsLiteral("x-cyrillic")) { bit = RUSSIAN_CHARSET; } else if (aLangGroup.EqualsLiteral("th")) { bit = THAI_CHARSET; } else if (aLangGroup.EqualsLiteral("x-central-euro")) { bit = EASTEUROPE_CHARSET; } else if (aLangGroup.EqualsLiteral("x-symbol")) { bit = SYMBOL_CHARSET; } if (bit != -1) return mCharset[bit]; return PR_FALSE; } PRBool SupportsRange(PRUint8 range) { return mUnicodeRanges[range]; } PRBool TestCharacterMap(PRUint32 aCh); PRUint8 mWindowsFamily; PRUint8 mWindowsPitch; gfxWindowsFontType mFontType; PRPackedBool mForceGDI : 1; PRPackedBool mUnknownCMAP : 1; PRPackedBool mUnicodeFont : 1; PRPackedBool mSymbolFont : 1; std::bitset<256> mCharset; std::bitset<128> mUnicodeRanges; }; /********************************************************************** * * class gfxWindowsFont * **********************************************************************/ class gfxWindowsFont : public gfxFont { public: gfxWindowsFont(FontEntry *aFontEntry, const gfxFontStyle *aFontStyle, cairo_antialias_t anAntialiasOption = CAIRO_ANTIALIAS_DEFAULT); virtual ~gfxWindowsFont(); virtual const gfxFont::Metrics& GetMetrics(); HFONT GetHFONT() { return mFont; } cairo_font_face_t *CairoFontFace(); cairo_scaled_font_t *CairoScaledFont(); SCRIPT_CACHE *ScriptCache() { return &mScriptCache; } gfxFloat GetAdjustedSize() { MakeHFONT(); return mAdjustedSize; } virtual nsString GetUniqueName(); virtual void Draw(gfxTextRun *aTextRun, PRUint32 aStart, PRUint32 aEnd, gfxContext *aContext, PRBool aDrawToPath, gfxPoint *aBaselineOrigin, Spacing *aSpacing); virtual RunMetrics Measure(gfxTextRun *aTextRun, PRUint32 aStart, PRUint32 aEnd, BoundingBoxType aBoundingBoxType, gfxContext *aContextForTightBoundingBox, Spacing *aSpacing); virtual PRUint32 GetSpaceGlyph() { GetMetrics(); // ensure that the metrics are computed but don't recompute them return mSpaceGlyph; }; PRBool IsValid() { GetMetrics(); return mIsValid; } FontEntry *GetFontEntry(); static already_AddRefed GetOrMakeFont(FontEntry *aFontEntry, const gfxFontStyle *aStyle, PRBool aNeedsBold = PR_FALSE); protected: HFONT MakeHFONT(); void FillLogFont(gfxFloat aSize); HFONT mFont; gfxFloat mAdjustedSize; PRUint32 mSpaceGlyph; private: void ComputeMetrics(); SCRIPT_CACHE mScriptCache; cairo_font_face_t *mFontFace; cairo_scaled_font_t *mScaledFont; gfxFont::Metrics *mMetrics; LOGFONTW mLogFont; cairo_antialias_t mAntialiasOption; virtual PRBool SetupCairoFont(gfxContext *aContext); }; /********************************************************************** * * class gfxWindowsFontGroup * **********************************************************************/ class THEBES_API gfxWindowsFontGroup : public gfxFontGroup { public: gfxWindowsFontGroup(const nsAString& aFamilies, const gfxFontStyle* aStyle, gfxUserFontSet *aUserFontSet); virtual ~gfxWindowsFontGroup(); virtual gfxFontGroup *Copy(const gfxFontStyle *aStyle); virtual gfxTextRun *MakeTextRun(const PRUnichar* aString, PRUint32 aLength, const Parameters* aParams, PRUint32 aFlags); virtual gfxTextRun *MakeTextRun(const PRUint8* aString, PRUint32 aLength, const Parameters* aParams, PRUint32 aFlags); const nsACString& GetGenericFamily() const { return mGenericFamily; } const nsTArray >& GetFontList() const { return mFontEntries; } PRUint32 FontListLength() const { return mFontEntries.Length(); } FontEntry *GetFontEntryAt(PRInt32 i) { return mFontEntries[i]; } virtual gfxWindowsFont *GetFontAt(PRInt32 i); void GroupFamilyListToArrayList(nsTArray > *list, nsTArray *aNeedsBold); void FamilyListToArrayList(const nsString& aFamilies, const nsCString& aLangGroup, nsTArray > *list); void UpdateFontList(); virtual gfxFloat GetUnderlineOffset(); protected: void InitFontList(); void InitTextRunGDI(gfxContext *aContext, gfxTextRun *aRun, const char *aString, PRUint32 aLength); void InitTextRunGDI(gfxContext *aContext, gfxTextRun *aRun, const PRUnichar *aString, PRUint32 aLength); void InitTextRunUniscribe(gfxContext *aContext, gfxTextRun *aRun, const PRUnichar *aString, PRUint32 aLength); already_AddRefed WhichPrefFontSupportsChar(PRUint32 aCh); already_AddRefed WhichSystemFontSupportsChar(PRUint32 aCh); already_AddRefed WhichFontSupportsChar(const nsTArray >& fonts, PRUint32 ch); void GetPrefFonts(const char *aLangGroup, nsTArray >& array); void GetCJKPrefFonts(nsTArray >& array); private: nsCString mGenericFamily; nsTArray > mFontEntries; nsTArray mFontNeedsBold; const char *mItemLangGroup; // used by pref-lang handling code }; #endif /* GFX_WINDOWSFONTS_H */