TTF字體:TrueType Font ,由Apple和微軟公司合作推出的文字文件格式;
一個TTF字符由許多輪廓組成,每一個輪廓由一個名叫TTPOLYGONHEADER的數據結構開始,跟在TTPOLYGONHEADER后面的是一系列的TTPOLYCURVE數據結構體。輪廓的頂點是POINTFX類型的數據結構點。
TTPOLYGONHEADER的結構如下:
1 typedef struct _TTPOLYGONHEADER { 2 DWORD cb; //描述字符要求的字節數 3 DWORD dwType; //指定返回的字符輪廓類型:TT_POLYGON_TYPE 4 POINTFX pfxStart; //指定字符輪廓的起點 5 } TTPOLYGONHEADER, *LPTTPOLYGONHEADER;
TTPOLYCURVE的結構如下:
typedef struct tagTTPOLYCURVE { WORD wType; //指定描述字符輪廓的線類型:TT_PRIM_LINE, TT_PRIM_QSPLINE, TT_PRIM_CSPLINE WORD cpfx; //指定結構中含有POINTFX的個數 POINTFX apfx[1]; //指定字體中保留的polyline和Bzier spline } TTPOLYCURVE, *LPTTPOLYCURVE;
POINTFX的結構如下:
typedef struct tagPOINTFX { FIXED x; //x-component of a point on the outline of a TrueType character FIXED y; //y-component of a point on the outline of a TrueType character } POINTFX, *LPPOINTFX;
對文字輪廓的獲取通過API函數GetGlyphOutline()進行獲取,在進行繪制。
在TTF字體結構中描述了TTF字體數據組成,在這里通過GLYPHMETRICS來了解TTF字體結構。理解可能有所偏差,詳細情況參考MSDN。
GLYPHMETRICS的結構如下:
typedef struct _GLYPHMETRICS { UINT gmBlackBoxX; //指定完全包圍字體結構的最小矩陣的寬度 UINT gmBlackBoxY; //指定完全包圍字體結構的最小矩陣的高度 POINT gmptGlyphOrigin; //指定完全包圍字體結構的最小矩陣左上角的點坐標 short gmCellIncX; //指定當前的起點到下一個字符的起點的水平距離 short gmCellIncY; //...垂直距離 } GLYPHMETRICS, *LPGLYPHMETRICS;
結構GLYPHMETRICS的使用主要在API函數GetGlyphOutline()來保存字符單元在字形的位置。
TrueType字體中的字符輪廓由直線和二次貝塞爾曲線片段所構成。Windows系統提供了API函數GetGlyphOutline()來獲取字符的輪廓。函數通過字符Code確定一個字符在TTF字庫中的結構,並取出字符Code所對應的數據。下面簡單說明下函數的結構和字符輪廓獲取的實現代碼。
GetGlyphOutline()函數結構:
DWORD GetGlyphOutline(UINT nChar, //需要獲取字符 UINT nFormat, //獲取字符的格式 LPGLYPHMETRICS lpgm,//獲取字符的相關信息 DWORD cbBuffer, //保存字符數據緩沖區的大小 LPVOID lpBuffer, //保存字符數據的緩沖區 const MAT2* lpmat2 //變換矩陣 )const;
使用GetGlyphOutline()函數獲取字符輪廓實現:
HDC hDC = pDC->GetSafeHdc(); //創建字體 CFont font; VERIFY(font.CreateFont(m_iFontHeight, 0, 0, 0,FW_NORMAL, FALSE, FALSE, 0, ANSI_CHARSET,OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, DEFAULT_PITCH | FF_SWISS,m_sFontFaceName)); CFont *pOldFont = pDC->SelectObject(&font); //定義並初始化變換矩陣 MAT2 mat2; memset(&mat2, 0, sizeof(mat2)); mat2.eM11 = 1; mat2.eM22 = 1; GLYPHMETRICS metrics; //保存字符相關信息 DWORD dwDataSize = 0; //初始化字符數據緩沖區大小 //…… //通過函數GetGlyphOutline()確定存儲字符結構的空間 dwDataSize = pDC->GetGlyphOutline(nChar, GGO_NATIVE, &metrics, 0, NULL, &mat2); if ((dwDataSize != 0) && (dwDataSize != GDI_ERROR)) { //創建保存字符數據緩沖區大小 LPBYTE pPixels = new BYTE[dwDataSize]; ASSERT( pPixels != NULL ); TTPOLYGONHEADER *pHeader = (TTPOLYGONHEADER*)pPixels; dwDataSize = pDC->GetGlyphOutline(nChar,GGO_NATIVE,&metrics, dwDataSize, pPixels, &mat2); while(dwDataSize > 0) { //計算字符輪廓的起點,轉換坐標 int xOld = MapFXY(pHeader->pfxStart.x); int yOld = MapFXY(pHeader->pfxStart.y); //根據TTF字體結構獲取字符輪廓 ::MoveToEx(hDC,iXpos + xOld,iYpos - yOld,NULL); TTPOLYCURVE *pCurrentCurve = (TTPOLYCURVE*)(pHeader + 1); int remainByte = pHeader->cb - sizeof(TTPOLYGONHEADER); while (remainByte > 0) { CPoint lpPoint[1000]; CPoint bezi[2]; int index; for (index = 0; index < pCurrentCurve->cpfx; ++index) { lpPoint[index].x = iXpos + MapFXY(pCurrentCurve->apfx[index].x); lpPoint[index].y = iYpos - MapFXY(pCurrentCurve->apfx[index].y); } switch (pCurrentCurve->wType) { case TT_PRIM_LINE: case TT_PRIM_QSPLINE: for (index =0; index < pCurrentCurve->cpfx; index++) { ::LineTo(hDC,lpPoint[index].x,lpPoint[index].y); } break; default: MessageBox(_T("字體不支持")); break; } int count = sizeof(TTPOLYCURVE) + (pCurrentCurve->cpfx -1)*sizeof(POINTFX); pCurrentCurve = (TTPOLYCURVE*)((char*)pCurrentCurve + count); remainByte -= count; } ::LineTo(hDC,iXpos + xOld, iYpos - yOld); dwDataSize -= pHeader->cb; pHeader = (TTPOLYGONHEADER*)((char*)pHeader + pHeader->cb); } delete [] pPixels; } }