331 lines
9 KiB
C++
331 lines
9 KiB
C++
/*
|
|
* @doc INTERNAL
|
|
*
|
|
* @module _FONT.H -- Declaration of classes comprising font caching |
|
|
*
|
|
* Purpose:
|
|
* Font cache
|
|
*
|
|
* Owner: <nl>
|
|
* David R. Fulmer (original RE 1.0 code)<nl>
|
|
* Christian Fortini (initial conversion to C++)<nl>
|
|
* Jon Matousek <nl>
|
|
*
|
|
* History: <nl>
|
|
* 8/6/95 jonmat Devised dynamic expanding cache for widths.
|
|
*
|
|
* Copyright (c) 1995-2000 Microsoft Corporation. All rights reserved.
|
|
*/
|
|
|
|
#ifndef _FONT_H
|
|
#define _FONT_H
|
|
|
|
#include "_kern.h"
|
|
|
|
// Forwards
|
|
class CFontCache;
|
|
class CDevDesc;
|
|
class CDisplay;
|
|
// ============================= CCcs ========================
|
|
// CCcs - caches font metrics and character size for one font
|
|
|
|
#define DEFAULTCACHESIZE 0 // size - 1
|
|
#define INITIALCACHESIZE 7 // size - 1 = 7; 2^n-1; size = 8
|
|
#define PERFCHECKEPOCH 64 // If changed, you must recalc
|
|
// and change COLLISION_SHIFT below.
|
|
|
|
static const INT maxCacheSize = 511;
|
|
|
|
SHORT GetFontNameIndex(const WCHAR *pFontName);
|
|
BYTE GetFontLegitimateSize(LONG iFont, BOOL fUIFont, int iCharRep);
|
|
BOOL SetFontLegitimateSize(LONG iFont, BOOL fUIFont, BYTE bSize, BOOL fFEcpg);
|
|
void SetFontSignature(LONG iFont, QWORD qwFontSig);
|
|
const WCHAR *GetFontName(LONG iFont);
|
|
UINT GetTextCharsetInfoPri(HDC hdc, FONTSIGNATURE* pcsi, DWORD dwFlags);
|
|
QWORD GetFontSignatureFromFace(int ifont, QWORD* pqwFontSig = NULL);
|
|
void FreeFontNames();
|
|
|
|
enum FONTINDEX
|
|
{
|
|
IFONT_ARIAL = 0,
|
|
IFONT_TMSNEWRMN = 1,
|
|
IFONT_SYMBOL = 2,
|
|
IFONT_SYSTEM = 3
|
|
};
|
|
|
|
typedef unsigned int CCSHASHKEY;
|
|
|
|
extern const WCHAR *szArial;
|
|
extern const WCHAR *szTimesNewRoman;
|
|
extern const WCHAR *szSymbol;
|
|
extern const WCHAR *szSystem;
|
|
extern const WCHAR *szWingdings;
|
|
|
|
//Not automatically added to font table
|
|
extern const WCHAR *szMicrosSansSerif;
|
|
extern const WCHAR *szMSSansSerif;
|
|
extern const WCHAR *szMangal;
|
|
extern const WCHAR *szLatha;
|
|
extern const WCHAR *szRaavi;
|
|
extern const WCHAR *szShruti;
|
|
extern const WCHAR *szTunga;
|
|
extern const WCHAR *szGautami;
|
|
extern const WCHAR *szCordiaNew;
|
|
extern const WCHAR *szTahoma;
|
|
extern const WCHAR *szArialUnicode;
|
|
extern const WCHAR *szSylfaen;
|
|
extern const WCHAR *szSyriac;
|
|
extern const WCHAR *szThaana;
|
|
|
|
/*
|
|
* CWidthCache
|
|
*
|
|
* @class Lightweight Unicode width cache.
|
|
*
|
|
*
|
|
* Owner: <nl>
|
|
* Jon Matousek (jonmat) <nl>
|
|
*/
|
|
struct CacheEntry
|
|
{
|
|
WCHAR ch;
|
|
SHORT width;
|
|
};
|
|
|
|
class CWidthCache
|
|
{
|
|
//@access Private methods and data
|
|
friend class CCcs;
|
|
private:
|
|
|
|
INT _cacheSize; //@cmember size is total cache slots - 1.
|
|
|
|
INT _cacheUsed; //@cmember for statistics, num slots in use.
|
|
INT _collisions; //@cmember for statistics, num fetches required.
|
|
INT _accesses; //@cmember for statistics, total num accesses.
|
|
BOOL _fMaxPerformance; //@cmember for statistics, TRUE if grown to max.
|
|
|
|
SHORT _dupCJK;
|
|
//@cmember default storage for widths.
|
|
CacheEntry _defaultWidthCache[DEFAULTCACHESIZE+1];
|
|
//@cmember pointers to storage for widths.
|
|
CacheEntry *(_pWidthCache);
|
|
|
|
__forceinline BOOL FLookasideCharacter(WCHAR ch)
|
|
{
|
|
if (ch < 0x3400)
|
|
return FALSE;
|
|
|
|
if (IN_RANGE(0x3400, ch, 0x9FFF) || // CJK ideograph
|
|
IN_RANGE(0xF900, ch, 0xFAFF) || // CJK compatibility ideograph
|
|
IN_RANGE(0xAC00, ch, 0xD7FF)) // Hangul
|
|
return TRUE;
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
//@cmember Get location where width is stored.
|
|
inline CacheEntry * GetEntry( const WCHAR ch )
|
|
{ // logical & is really a MOD, as all of the bits
|
|
// of cacheSize are turned on; the value of cacheSize is
|
|
// required to be of the form 2^n-1.
|
|
return &_pWidthCache[ ch & _cacheSize ];
|
|
}
|
|
|
|
//@cmember See if cache is performing within spec.
|
|
void CheckPerformance();
|
|
//@cmember Increase width cache size.
|
|
BOOL GrowCache( CacheEntry **widthCache, INT *cacheSize, INT *cacheUsed);
|
|
|
|
//@access Public Methods
|
|
public:
|
|
//@cmember Called before GetWidth
|
|
BOOL CheckWidth (const WCHAR ch, LONG &dup);
|
|
//@cmember Fetch width if CheckWidth ret FALSE.
|
|
|
|
void Free(); //@cmember Recycle width cache
|
|
|
|
CWidthCache(); //@cmember Construct width cache
|
|
~CWidthCache(); //@cmember Free dynamic mem
|
|
};
|
|
|
|
|
|
class CCcs
|
|
{
|
|
friend class CFontCache;
|
|
|
|
private:
|
|
CCSHASHKEY _ccshashkey; // Hash key
|
|
DWORD _dwAge; // for LRU algorithm
|
|
SHORT _iFont; // Index into FONTNAME table
|
|
SHORT _cRefs; // ref. count
|
|
|
|
class CWidthCache _widths;
|
|
|
|
public:
|
|
QWORD _qwFontSig; // Internal font signature flags
|
|
|
|
HDC _hdc; // HDC font is selected into
|
|
HFONT _hfont; // Windows font handle
|
|
|
|
void* _sc; // A handle to the Uniscribe glyph width/font cmap information
|
|
|
|
//REVIEW (keithcu) We should make these into at least 24 bit or possibly 32 bit values,
|
|
//or at least use unsigned values so that we don't overflow as easily.
|
|
SHORT _yHeightRequest;// Font height requested (logical units)
|
|
SHORT _yHeight; // Total height of char cell (logical units)
|
|
SHORT _yDescent; // Distance from baseline to char cell bottom (logical units)
|
|
|
|
SHORT _xAveCharWidth; // Average character width in logical units
|
|
SHORT _xOverhangAdjust;// Overhang for synthesized fonts in logical units
|
|
|
|
SHORT _dyULOffset; // Underline offset
|
|
SHORT _dyULWidth; // Underline width
|
|
SHORT _dySOOffset; // Strikeout offset
|
|
SHORT _dySOWidth; // Strikeout width
|
|
|
|
USHORT _weight; // Font weight
|
|
USHORT _wCodePage; // Font code page
|
|
|
|
BYTE _bCharSetRequest; //Requested charset
|
|
BYTE _bCharSet; // Font CharSet
|
|
BYTE _bCMDefault; // Used in calculation of _bConvertMode
|
|
BYTE _bConvertMode; // CONVERTMODE: CVT_NONE, CVT_WCTMB, CVT_LOWBYTE
|
|
|
|
BYTE _bPitchAndFamily;// Font pitch and family
|
|
TFLOW _tflow; // Tflow of _hfont
|
|
BYTE _bQuality; // LOGFONT quality
|
|
|
|
BYTE _fValid:1; // CCcs is valid
|
|
BYTE _fFixPitchFont:1; // Font has fixed character width
|
|
BYTE _fItalic:1; // Font is italic
|
|
BYTE _fFECharSet:1; // Font has FE charset
|
|
BYTE _fForceTrueType:1; // Font has been forced to be truetype
|
|
BYTE _fCustomTextOut:1; // Should we use the ICustomTextOut handlers?
|
|
BYTE _fUseAtFont:1; // Switch to @ font
|
|
|
|
private:
|
|
|
|
BOOL Compare (const CCharFormat * const pCF, HDC hdc, DWORD dwFlags);
|
|
BOOL MakeFont(const CCharFormat * const pCF);
|
|
void DestroyFont();
|
|
BOOL GetMetrics(WCHAR *szNewFaceName = 0);
|
|
HFONT GetFontWithMetrics(LOGFONT *plf, WCHAR* szNewFaceName);
|
|
BOOL FillWidth(WCHAR ch, LONG &dup);
|
|
|
|
public:
|
|
CCcs () {_fValid = FALSE;}
|
|
~CCcs () {if(_fValid) Free();}
|
|
|
|
void GetFontOverhang(LONG *pdupOverhang, LONG *pdupUnderhang);
|
|
void GetOffset(const CCharFormat * const pCF, LONG dvpInch,
|
|
LONG *pyOffset, LONG *pyAdjust);
|
|
|
|
BOOL Init(const CCharFormat * const pCF);
|
|
void Free();
|
|
void AddRef() {_cRefs++;}
|
|
void Release() {if(_cRefs) _cRefs--;}
|
|
|
|
BOOL Include(WCHAR ch, LONG &dup)
|
|
{
|
|
if(!_widths.CheckWidth(ch, dup))
|
|
return FillWidth(ch, dup);
|
|
return TRUE;
|
|
}
|
|
BYTE BestCharRep(BYTE iCharRep, BYTE iCharRepDefault, int fFontMatching);
|
|
|
|
SHORT AdjustFEHeight(BOOL fAjdust)
|
|
{return ((fAjdust && _fFECharSet) ? W32MulDiv(_yHeight, 15, 100) : 0);}
|
|
};
|
|
|
|
|
|
// FONTINFO cache
|
|
|
|
typedef union
|
|
{
|
|
WORD wFlags;
|
|
struct
|
|
{
|
|
WORD fCached :1; // Font signature was already cached
|
|
WORD fBadFaceName :1; // Face is junk or doesnt exist in the system
|
|
WORD fTrueType :1; // Font is TrueType
|
|
WORD fBitmap :1; // Font is Bitmap
|
|
WORD fNonBiDiAscii :1; // Font is non-BiDi, single charset and support ASCII
|
|
WORD fScaleByCpg :1; // Scale the font based on given codepage
|
|
WORD fThaiDTP :1; // Thai DTP font
|
|
};
|
|
} FONTINFO_FLAGS;
|
|
|
|
class CFontFamilyMember
|
|
{
|
|
public:
|
|
CFontFamilyMember(LONG weight, BOOL fItalic)
|
|
{
|
|
_weight = weight; _fItalic = fItalic;
|
|
}
|
|
void Free() {_kc.Free();}
|
|
CKernCache* GetKernCache() {return &_kc;}
|
|
|
|
LONG _weight;
|
|
BOOL _fItalic;
|
|
CKernCache _kc;
|
|
};
|
|
|
|
class CFontFamilyMgr
|
|
{
|
|
public:
|
|
CFontFamilyMgr::~CFontFamilyMgr();
|
|
CFontFamilyMember *GetFontFamilyMember(LONG weight, BOOL fItalic);
|
|
CArray <CFontFamilyMember> _rgf;
|
|
};
|
|
|
|
|
|
struct FONTINFO
|
|
{
|
|
const WCHAR *szFontName;
|
|
QWORD qwFontSig; // Font signature
|
|
BYTE bSizeUI; // UI font legitimate size (in point)
|
|
BYTE bSizeNonUI; // Non-UI font legitimate size
|
|
FONTINFO_FLAGS ff; // flags
|
|
CFontFamilyMgr* _pffm; // Information which is different for
|
|
}; // bold/italic variants of a font
|
|
|
|
|
|
|
|
// ============================= CFontCache =====================================================
|
|
// CFontCache - maintains up to FONTCACHESIZE font caches
|
|
|
|
//The low 2 bits are reserved for passing down the TFLOW of the text
|
|
const DWORD FGCCSUSETRUETYPE = 0x04;
|
|
const DWORD FGCCSUSEATFONT = 0x08;
|
|
|
|
class CFontCache
|
|
{
|
|
friend class CCcs;
|
|
|
|
private:
|
|
CCcs _rgccs[FONTCACHESIZE];
|
|
DWORD _dwAgeNext;
|
|
struct {
|
|
CCSHASHKEY ccshashkey;
|
|
CCcs *pccs;
|
|
} quickHashSearch[CCSHASHSEARCHSIZE+1];
|
|
|
|
private:
|
|
CCcs* GrabInitNewCcs(const CCharFormat * const pCF, HDC hdc, DWORD dwFlags);
|
|
CCSHASHKEY MakeHashKey(const CCharFormat *pCF);
|
|
public:
|
|
void Init();
|
|
|
|
CFontFamilyMgr * GetFontFamilyMgr(LONG iFont);
|
|
CKernCache * GetKernCache(LONG iFont, LONG weight, BOOL fItalic);
|
|
|
|
CCcs* GetCcs(CCharFormat *pCF, const LONG dvpInch, DWORD dwFlags, HDC hdc = 0);
|
|
FONTINFO_FLAGS GetInfoFlags(int ifont);
|
|
};
|
|
|
|
|
|
extern CFontCache & fc(); // font cache manager
|
|
|
|
#endif
|