TTF字體


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;
    }
    }

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM