打造自己的专业图像工具-Visual C++ 2005图像编程系列【五】(中)

现在,我们还没有看字体枚举的回调函数是如何工作的。回调函数里每次回调一个新的字体就需要创建一个CFontInfo对象,并写入相应的信息,然后添加到CFontComboBox的成员m_pFontVec中。在这个回调的静态函数里就需要访问CFontComboBox 对象,原来在调用这个函数的时候我们把一个CFontComboBox 对象的this指针通过LPARAM参数传入了回调函数。一切就都不是问题了。

int CFontComboBox::EnumFontProc(ENUMLOGFONTEX *lpelfe,

                                                        NEWTEXTMETRICEX *lpntme,

                                                        DWORD FontType,

                                                        LPARAM lParam)

{

       CFontComboBox *pThis = reinterpret_cast<CFontComboBox*>(lParam);

       CFontInfo *pInfo = new CFontInfo;

       pInfo->SetFontType(FontType);

       pInfo->SetFontName(lpelfe->elfLogFont.lfFaceName);

       DWORD dwFontType = FontType;

       if (FontType & TRUETYPE_FONTTYPE)

       {

              DWORD dwFontFlags = lpntme->ntmTm.ntmFlags;

              if (dwFontFlags & NTM_PS_OPENTYPE)

                     dwFontType |= PS_OPENTYPE_FONTTYPE;

              else

                     dwFontType |=0;

              if (dwFontFlags & NTM_TT_OPENTYPE)

                     dwFontType |= TT_OPENTYPE_FONTTYPE;

              else

                     dwFontType |=0;

              dwFontType |= (dwFontFlags & NTM_TYPE1 ? TYPE1_FONTTYPE : 0);

       }

       switch(dwFontType & 0x70007)

       {

       case (TRUETYPE_FONTTYPE | PS_OPENTYPE_FONTTYPE):

       case (TRUETYPE_FONTTYPE | TT_OPENTYPE_FONTTYPE):

       case (TRUETYPE_FONTTYPE | TYPE1_FONTTYPE):

              pInfo->SetImage(3);             break;

       case RASTER_FONTTYPE:

       case DEVICE_FONTTYPE:

    case NULL:    pInfo->SetImage(0XFF);      break;

       case TRUETYPE_FONTTYPE:

       default:           pInfo->SetImage(0);             break;

       }

       pThis->m_pFontVec.push_back(pInfo);

       return TRUE;

}

       注意在上面使用的字体类型定义,需要在EnumFontProc函数前面直接加上如下的预编译定义:

#ifndef   NTM_PS_OPENTYPE

#define   NTM_PS_OPENTYPE                0x00020000

#endif

#ifndef   NTM_TT_OPENTYPE

#define   NTM_TT_OPENTYPE                0x00040000

#endif

#ifndef   PS_OPENTYPE_FONTTYPE

#define   PS_OPENTYPE_FONTTYPE      0x10000

#define   TT_OPENTYPE_FONTTYPE      0x20000

#define   TYPE1_FONTTYPE                   0x40000

#endif

接下来我们看看如何设置下面列表框的宽度,当用户点击小箭头显示下拉框时,会发出CBN_DROPDOWN消息,我们需要在这个消息函数里设置下拉框的宽度。先在MFC的消息宏中添加消息映射:

BEGIN_MESSAGE_MAP(CFontComboBox, CComboBox)

       ON_CONTROL_REFLECT(CBN_DROPDOWN, OnDropdown)

       ON_WM_DESTROY()

END_MESSAGE_MAP()

       OnDropdown函数里我们要计算所有字体名称字符串的长度,用最大值做为下拉框的显示宽度,最好的显示宽度还要加上滚动条和左边的字体图片的宽度,为了方便使用,还定义了两个字体预览图的宽度、高度值:

#define           FNTIMG_X                 20

#define           FNTIMG_Y                  12

void CFontComboBox::OnDropdown()

{

       int nNumEntries = GetCount();

    int nWidth = 0;

    CString str;

    CClientDC dc(this);

    int nSave = dc.SaveDC();

    dc.SelectObject(GetFont());

    int nScrollWidth = ::GetSystemMetrics(SM_CXVSCROLL);      //取滚动条宽度

    for (int i = 0; i < nNumEntries; i++)

    {

        GetLBText(i, str);

        int nLength = dc.GetTextExtent(str).cx + nScrollWidth;

        nWidth = max(nWidth, nLength);

    }

    nWidth += dc.GetTextExtent("0").cx;

    dc.RestoreDC(nSave);

       if (!m_pFontVec.empty())

              SetDroppedWidth(nWidth + FNTIMG_XSIZE);                 //设置宽度值

}

       还记得上面枚举函数里我们用new动态申请了很多的字体信息CFontInfo对象,这些对象在窗口退出时需要释放,否则会内存泄漏。释放这些东西的位置就在窗口WM_DESTROY消息函数里。这个消息是最理想的清除地方。

应用程序关闭过程:

当用户按下菜单的close命令时,系统发出WM_CLOSE,通常程序的窗口函数不拦截这个消息,于是DefWinodwProc处理它,DefWinodwProc收到WM_CLOSE后,调用DestroyWindow把窗口清除,DestroyWindow本身会送出WM_DESTROY,程序对WM_DESTROY的标准反应就是调用PostQuitMessagePostQuitMessage没有其他的操作,就只送出WM_QUIT消息,而消息循环GetMessage得到这个消息后返回0,而结束了消息循环,再接着结束整个程序。在实际运用中,有的时候我们的程序窗口关闭了,但是在任务管理器里还有进程存在,这个问题有时候就是因为没有调用PostQuitMessage(0);引起的。

 

void CFontComboBox::OnDestroy()

{

       for (int i=0; i<m_pFontVec.size(); ++i)

              delete m_pFontVec[i];

       m_pFontVec.erase(m_pFontVec.begin(), m_pFontVec.end());

       CComboBox::OnDestroy();

}

 
在释放 STL 的很多指针容器时,容器对象的 erase 函数只是释放每个指针所占用的控件,并且释放我们 new 的指针对象,所以在 erase 之前要先 delete 每个对象指针。

你可能感兴趣的:(打造自己的专业图像工具-Visual C++ 2005图像编程系列【五】(中))