(51) 如何訪(fǎng)問(wèn)預(yù)定義的GDI對(duì)象
可以通過(guò)調(diào)用CDC:: SlectStockObject使用Windows的幾個(gè)預(yù)定義的對(duì)象,諸如刷子、筆以及字體。下例使用了Windows預(yù)定義的筆和刷子GDI對(duì)象在視窗中畫(huà)一個(gè)橢圓。
//Draw ellipse using stock black pen and gray brush.
void CSampleView:: OnDraw (CDC* pDC)
{
//Determine size of view.
CRect rcView
GetClientRect (rcView)
//Use stock black pen and stock gray brush to draw ellipse.
pDC->SelectStockObject (BLACK_PEN)
pDC->SelectStockObject (GRAY_BRUSH)
//Draw the ellipse.
pDC->Ellipse (reView)
}
也可以調(diào)用新的SDK函數(shù)GetSysColorBrush獲取一個(gè)系統(tǒng)顏色刷子,下例用背景色在視窗中畫(huà)一個(gè)橢圓:
void CsampleView:: OnDraw (CDC* pDC)
{
//Determine size of view.
CRect rcView
GetClientRect (rcView)
//Use background color for tooltips brush.
CBrush * pOrgBrush=pDC->SelectObject ( CBrush ::FromHandle( ::GetSysColorBrush (COLOR_INFOBK)))
//Draw the ellipse.
pDC->Ellipse (rcView)
//Restore original brush.
pDC->SelectObject (pOrgBrush)
}
(52) 如何獲取GDI對(duì)象的屬性信息
可以調(diào)用GDIObject:: GetObject。這個(gè)函數(shù)將指定圖表設(shè)備的消息寫(xiě)入到緩沖區(qū)。下例創(chuàng)建了幾個(gè)有用的輔助函數(shù)。
//Determine if font is bold.
BOOL IsFontBold (const CFont&font)
{
LOGFONT stFont
font.GetObject (sizeof (LOGFONT), &stFont)
return (stFont.lfBold)? TRUE: FALSE
}
//Return the size of a bitmap.
CSize GetBitmapSize (const CBitmap&bitmap)
{
BITMAP stBitmap
bitmap.GetObject (sizeof (BITMAP), &stBitmap)
return CSize (stBitmap.bmWidth, stBitmap.bmHeight)
}
//Create a pen with the same color as a brush.
BOOL CreatePenFromBrush (Cpen&pen, cost Cbrush&brush)
{
LOGBRUSH stBrush
brush.Getobject (sizeof (LOGBRUSH), &stBrush)
return pen. Createpen (PS_SOLID, 0, stBrush.ibColor)
}
(53) 如何實(shí)現(xiàn)一個(gè)橡皮區(qū)矩形
CRectTracker是一個(gè)很有用的類(lèi),可以通過(guò)調(diào)用CRectTracker::TrackRubberBand 響應(yīng)WM_LBUTTONDOWN消息來(lái)創(chuàng)建一個(gè)橡皮區(qū)矩形。
下例表明使用CRectTracker移動(dòng)和重置視窗中的藍(lán)色橢圓的大小是很容易的事情。
首先,在文件檔中聲明一個(gè)CRectTracker數(shù)據(jù)成員:
class CSampleView : Public CView
{
…
public :
CrectTracker m_tracker
…
}
其次,在文檔類(lèi)的構(gòu)造函數(shù)中初始化CRectTracker 對(duì)象:
CSampleDoc:: CSampleDOC ()
{
//Initialize tracker position, size and style.
m_tracker.m_rect.SetRect (0, 0, 10, 10)
m_tracker.m_nStyle=CRectTracker:: resizeInside | CRectTracker ::dottedLine
}
然后,在OnDraw函數(shù)中畫(huà)橢圓和蹤跡矩形:
void CSampleView:: OnDraw (CDC* pDC)
{
CSampleDoc* pDoc=GetDocument ()
ASSERT_VALID (pDoc)
//Select blue brush into device context.
CBrush brush (RGB (0, 0, 255))
CBrush* pOldBrush=pDC->SelectObject (&brush)
//draw ellipse in tracking rectangle.
Crect rcEllipse
pDoc->m_tracker.GetTrueRect (rcEllipse)
pDC->Ellipse (rcEllipse)
//Draw tracking rectangle.
pDoc->m_tracker.Draw (pDC)
//Select blue brush out of device context.
pDC->Selectobject (pOldBrush)
}
最后,使用ClassWizard處理WM_LBUTTONDOWN消息,并增加下述代碼。該段代碼根據(jù)鼠標(biāo)擊鍵情況可以拖放、移動(dòng)或者重置橢圓的大小。
void CSampleView::OnLButtonDown (UINT nFlags, CPoint point)
{
//Get pointer to document.
CSampleDoc* pDoc=GetDocument ()
ASSERT_VALID (pDoc)
//If clicked on ellipse, drag or resize it.Otherwise create a
//rubber-band rectangle nd create a new ellipse.
BOOL bResult=pDoc->m_tracker.HitTest (point)!= CRectTracker::hitNothing
//Tracker rectangle changed so update views.
if (bResult)
{
pDoc->m_tracker.Track (this,point,TRue)
pDoc->SetModifiedFlag ()
pDoc->UpdateAllViews (NULL)
}
else
pDoc->m-tracker.TrackRubberBand(this,point,TRUE)
CView:: onLButtonDown (nFlags,point)
}
(54) 如何更新翻轉(zhuǎn)背景顏色的文本
調(diào) 用CDC:: SetBkmode并傳送OPAQUE用當(dāng)前的背景顏色填充背景,或者調(diào)用CDC::SetBkMode并傳送TRANSPAARENT使背景保持不變, 這兩種方法都可以設(shè)置背景模式。下例設(shè)置背景模式為T(mén)RANSPARENT,可以?xún)纱胃麓没ㄉ珟Ш陉幱案挛谋尽:谏诩t色串之后,但由于設(shè)置了 背景模式仍然可見(jiàn)。
void CSampleView:: OnDraw (CDC* pDC)
{
//Determint size of view.
CRect rcView
GetClientRect (rcVieew)
//Create sample string to display.
CString str (_T ("Awesome Shadow Text..."))
//Set the background mode to transparent.
pDC->SetBKMode (TRANSPARENT)
//Draw black shadow text.
rcView.OffsetRect (1, 1)
pDc->SetTextColor (RGB (0, 0, 0))
pDC->DrawText (str, str.GetLength (), rcView, DT_SINGLELINE | DT_CENTER | DT_VCENTER)
//Draw red text.
rcView.OffsetRect (-1,-1)
pDc->SetTextColor (RGB (255, 0, 0))
pDC->DrawText (str, str.GetLength (), rcView, DT_SINGLELINE | DT_CENTER | DT_VCENTER)
}
(55) 如何創(chuàng)建一個(gè)具有特定點(diǎn)大小的字體
可以指定字體邏輯單位的大小,但有時(shí)指定字體的點(diǎn)的大小可能會(huì)更方便一些。可以如下將字體的點(diǎn)轉(zhuǎn)換為字體的高度:
int nHeigth=mulDiv (nPointSize, -dc.GetDeviceCaps (LOGPIXELSY), 72)
下例創(chuàng)建了一個(gè)8點(diǎn)的Apial字體:
…
CClientDC dc (AqfxGetMainWnd ())
m_font. CreateFont (MulDiv (8, -dc.GetDeviceCaps (LOGPIXELSY), 72), 0, 0, 0, FW_NORMAL, 0, 0, 0, ANSI_CHARSET, OUT_STROKE_PRECIS, CLIP_STROKE_PRECIS, DRAFT_QUALITY, VARIABLE_PITCH | FF-SWISS,_T("Arial"))
(56) 如何計(jì)算一個(gè)串的大小
函數(shù)CDC:: Det text Extent 根據(jù)當(dāng)前選擇的字體計(jì)算一個(gè)串的高度和寬度。如果使用的不是系統(tǒng)字體而是其他字體,則在調(diào)用GetTextExtent之前將字體選進(jìn)設(shè)備上下文中是很重 要的,否則計(jì)算高度和寬度時(shí)將依據(jù)系統(tǒng)字體,由此得出的結(jié)果當(dāng)然是不正確的。下述樣板程序當(dāng)改變下壓按鈕的標(biāo)題時(shí)動(dòng)態(tài)調(diào)整按鈕的大小,按鈕的大小由按鈕的 字體和標(biāo)題的大小而定。響應(yīng)消息WM_SETTEXT時(shí)調(diào)用OnSetText,該消息使用ON_MESSAE宏指令定義的用戶(hù)自定義消息。
LRESULT CMyButton:: OnSettext (WPARAM wParam, LPARAM lParam)
{
//Pass message to window procedure.
LRESULT bResult=CallWindowProc (*GetSuperWndProcAddr(), m_hWnd, GetCurrentMessage() ->message,wParam,lParam)
//Get title of push button.
CString strTitle
GetWindowText (strTitle)
//Select current font into device context.
CDC* pDC=GetDc ()
CFont*pFont=GetFont ()
CFont*pOldFont=pDC->SelectObject (pFont)
//Calculate size of title.
CSize size=pDC->GetTextExent (strTitle,strTitle.GetLength())
//Adjust the button's size based on its title.
//Add a 5-pixel border around the button.
SetWindowPos (NULL, 0, 0, size.cx+10, size.cy+10, SWP_NOMOVE | SWP_NOZORDER | SWP_NOACTIVATE)
//Clean up.
pDC->SelectFont (pOldFont)
ReleaseDC (pDC)
return bResult
}
(57) 如何顯示旋轉(zhuǎn)文本
只 要用戶(hù)使用TrueType或者GDI筆或字體就可以顯示旋轉(zhuǎn)文本(有些硬件設(shè)備也支持旋轉(zhuǎn)光柵字體)。LOGFONT結(jié)構(gòu)中的ifEscapement 成員指定了文本行和x軸的角度,角度的單位是十分之一度而不是度,例如,ifEscapement為450表示字體旋轉(zhuǎn)45度。為確保所有的字體沿坐標(biāo)系 統(tǒng)的同一方向旋轉(zhuǎn),一定要設(shè)置ifEscapement成員的CLIP_LH_ANGLES位,否則,有些字體可能反向旋轉(zhuǎn)。下例使用了14點(diǎn)Arial 字體每間隔15度畫(huà)一個(gè)串。
void CSampleView:: OnDraw (CDC* pDC)
{
//Determine the size of the window.
CRect rcClient
GetClientRect (rcClient)
//Create sample string.
CString str (_T ("Wheeee...I am rotating!"))
//Draw transparent, red text.
pDC->SetBkMode (TRANSPARENT)
pDC->SetTextColor (RGB (255,0,0))
CFont font
//font object
LOGFONT stFont //font definition
//Set font attributes that will not change.
memset (&stFont, 0, sizeof (LOGFONT))
stFont.ifheight=MulDiv (14, -pDC->GetDeviceCaps(LOGPIXELSY), 72)
stFont.ifWeight=FW_NORMAL
stFont.ifClipPrecision=LCIP_LH_ANGLES
strcpy (stFont.lfFaceName, "Arial")
//Draw text at 15degree intervals.
for (int nAngle=0 nAngle<3600 nAngle+=150)
{
//Specify new angle.
stFont.lfEscapement=nAngle
//Create and select font into dc.
font.CreateFontIndirect(&stfont)
CFont* pOldFont=pDC ->SelectObject(&font)
//Draw the text.
pDC->SelectObject(pOldFont)
font.DelectObjext()
}
}
(58) 如何正確顯示包含標(biāo)簽字符的串
調(diào) 用GDI文本繪畫(huà)函數(shù)時(shí)需要展開(kāi)標(biāo)簽字符,這可以通過(guò)調(diào)用CDC:: TabbedTextOut或者CDC:: DrawText并指定DT_EXPANDTABS標(biāo)志來(lái)完成。TabbedTextOut函數(shù)允許指定標(biāo)簽位的數(shù)組,下例指定每20設(shè)備單位展開(kāi)一個(gè)標(biāo) 簽:
void CSampleView:: OnDraw (CDC* pDC)
{
CTestDoc* pDoc=GetDocument ()
ASSERT_VALID (pDoC)
CString str
str.Format (_T ("Cathy\tNorman\tOliver"))
int nTabStop=20 //tabs are every 20 pixels
pDC->TabbedtextOut (10, 10, str, 1, &nTabStop, 10)
}
(59) 如何快速地格式化一個(gè)CString對(duì)象
調(diào)用CString:: Format,該函數(shù)和printf函數(shù)具有相同的參數(shù),下例說(shuō)明了如何使用Format函數(shù):
//Get size of window.
CRect rcWindow
GetWindowRect (rcWindow)
//Format message string.
CString strMessage
strMessage.Format (_T ("Window Size (%d, %d)"),
rcWindow.Width (), rcWindow.Height ())
//Display the message.
MessageBox (strmessage)
(60) 串太長(zhǎng)時(shí)如何在其末尾顯示一個(gè)省略號(hào)
調(diào)用CDC:: DrawText并指定DT_END_ELLIPSIS標(biāo)志,這樣就可以用小略號(hào)取代串末尾的字符使其適合于指定的邊界矩形。如果要顯示路徑信息,指定DT_END_ELLIPSIS標(biāo)志并省略號(hào)取代串中間的字符。
void CSampleView:: OnDraw (CDC* pDC)
{
CTestDoc* pDoc=GetDocument ()
ASSERT_VALID (pDoc)
//Add ellpsis to end of string if it does not fit
pDC->Drawtext (CString ("This is a long string"), CRect (10, 10, 80, 30), DT_LEFT | DT_END_ELLIPSIS)
//Add ellpsis to middle of string if it does not fit
pDC->DrawText (AfxgetApp () ->m_pszhelpfilePath, CRect (10, 40, 200, 60), DT_LEFT | DT_PATH_ELLIPSIS)
}
(61) 為什么即使調(diào)用EnableMenuItem菜單項(xiàng)后,菜單項(xiàng)還處于禁止?fàn)顟B(tài)
需要將CFrameWnd:: m_bAutomenuEnable設(shè)置為FALSE,如果該數(shù)據(jù)成員為T(mén)RUE(缺省值),工作框?qū)⒆詣?dòng)地禁止沒(méi)有ON_UPDATE_COMMAND_UI或者ON_COMMAND的菜單項(xiàng)。
//Disable MFC from automatically disabling menu items.
m_bAuoMenuEnable=FALSE
//Now enable the menu item.
CMenu* pMenu=GetMenu ()
ASSERT_VALID (pMenu)
pMenu->EnableMenuItem (ID_MENU_ITEM,MF_BYCOMMAND | MF_ENABLED)
(62) 如何給系統(tǒng)菜單添加一個(gè)菜單項(xiàng)
給系統(tǒng)菜單添加一個(gè)菜單項(xiàng)需要進(jìn)行下述三個(gè)步驟:
首先,使用Resource Symbols對(duì)話(huà)(在View菜單中選擇Resource Symbols...可以顯示該對(duì)話(huà))定義菜單項(xiàng)ID,該ID應(yīng)大于0x0F而小于0xF000;
其次,調(diào)用CWnd::GetSystemMenu獲取系統(tǒng)菜單的指針并調(diào)用CWnd:: Appendmenu將菜單項(xiàng)添加到菜單中。下例給系統(tǒng)菜單添加兩個(gè)新的
int CMainFrame:: OnCreate (LPCREATESTRUCT lpCreateStruct)
{
…
//Make sure system menu item is in the right range.
ASSERT (IDM_MYSYSITEM &0xFFF0)==IDM_MYSYSITEM)
ASSERT (IDM-MYSYSITEM<0xF000)
//Get pointer to system menu.
CMenu* pSysmenu=GetSystemmenu (FALSE)
ASSERT_VALID (pSysMenu)
//Add a separator and our menu item to system menu.
CString StrMenuItem (_T ("New menu item"))
pSysMenu->Appendmenu (MF_SEPARATOR)
pSysMenu->AppendMenu (MF_STRING, IDM_MYSYSITEM, strMenuitem)
…
}
現(xiàn)在,選擇系統(tǒng)菜單項(xiàng)時(shí)用戶(hù)應(yīng)進(jìn)行檢測(cè)。使用ClassWizard處理WM_SYSCOMMAND消息并檢測(cè)用戶(hù)菜單的nID參數(shù):
void CMainFrame:: OnSysCommand (UINT nID,LPARAM lParam)
{
//Determine if our system menu item was selected.
if ( (nID & 0xFFF0)==IDM_MYSYSITEM)
{
//TODO-process system menu item
}
else
CMDIFrameWnd ::OnSysCommand (nID, lParam)
}
最后,一個(gè)設(shè)計(jì)良好的UI應(yīng)用程序應(yīng)當(dāng)在系統(tǒng)菜單項(xiàng)加亮?xí)r在狀態(tài)條顯示一個(gè)幫助信息,這可以通過(guò)增加一個(gè)包含系統(tǒng)菜單基ID的串表的入口來(lái)實(shí)現(xiàn)。
(63) 如何確定頂層菜單所占據(jù)的菜單行數(shù)
這可以通過(guò)簡(jiǎn)單的減法和除法來(lái)實(shí)現(xiàn)。首先,用戶(hù)需要計(jì)算主框窗口的高度和客戶(hù)區(qū);其次,從主框窗口的高度中減去客戶(hù)區(qū)、框邊界以及標(biāo)題的高度;最后,除以菜單欄的高度。下例成員函數(shù)是一個(gè)計(jì)算主框菜單所占據(jù)的行數(shù)的代碼實(shí)現(xiàn)。
int CMainFrame:: GetMenuRows ()
{
CRect rcFrame,rcClient
GetWindowRect (rcFrame)
GetClientRect (rcClient)
return (rcFrame.Height () -rcClient.Height () - :: GetSystemMetrics(SM_CYCAPTION) - (:: getSystemMetrics(SM_CYFRAME) *2)) / :: GetSystemMetrics(SM_CYMENU)
}
(64) 在用戶(hù)環(huán)境中如何確定系統(tǒng)顯示元素的顏色
調(diào)用SDK函數(shù)GetSysColor可以獲取一個(gè)特定顯示元素的顏色。下例說(shuō)明了如何在MFC函數(shù)CMainFrameWnd:: OnNcPaint中調(diào)用該函數(shù)設(shè)置窗口標(biāo)題顏色。
void CMiniFrameWnd:: OnNcPaint ()
{
…
dc.SetTextColor (:: GetSysColor (m_bActive ? COLOR_CAPTIONTEXT : COLOR_INACTIVECAPTIONTEXT))
…
(65) 如何查詢(xún)和設(shè)置系統(tǒng)參數(shù)
在Windows 3.1 SDK中介紹過(guò)SDK函數(shù)SystemParametersInfo,調(diào)用該函數(shù)可以查詢(xún)和設(shè)置系統(tǒng)參數(shù),諸如按鍵的重復(fù)速率設(shè)置、鼠標(biāo)雙擊延遲時(shí)間、圖標(biāo)字體以及桌面覆蓋位圖等等。
//Create a font that is used for icon titles.
LOGFONT stFont
∶: SystemParametersInfo (SPIF_GETICONTITLELOGFONT, sizeof (LOGFONT), &stFont, SPIF_SENDWININICHANGE)
m_font.CreateFontIndirect (&stFont)
//Change the wallpaper to leaves.bmp.
∶ : SystemParametersInfo (SPI_SETDESKWALLPAPER, 0, _T (" forest.bmp"), SPIF_UPDATEINIFILE)
(66) 如何確定當(dāng)前屏幕分辨率
調(diào)用SDK函數(shù)GetSystemMetrics,該函數(shù)可以檢索有關(guān)windows顯示信息,諸如標(biāo)題大小、邊界大小以及滾動(dòng)條大小等等。
//Initialize CSize object with screen size.
CSize sizeScreen (GetSystemMetrics (SM_CXSCREEN),
GetSystemMetrics (SM_CYSCREEN))
(67) 如何使用一個(gè)預(yù)定義的Windows光標(biāo)
調(diào)用CWinApp:: LoadStandardCursor并傳送光標(biāo)標(biāo)識(shí)符。
BOOL CSampleDialog:: OnSetCursor (CWnd* pWnd,
UINT nHitTest, UINT
message)
{
//Display wait cursor if busy.
if (m_bBusy)
{
SetCursor (AfxGetApp () ->LoadStandardCursor (IDC_WAIT))
return TRUE
}
return CDialog:: OnSetCursor (pWnd. nHitTest,message)
}
(68) 如何檢索原先的Task Manager應(yīng)用程序使用的任務(wù)列表
原 先的Task Manager應(yīng)用程序顯示頂層窗口的列表。為了顯示該列表,窗口必須可見(jiàn)、包含一個(gè)標(biāo)題以及不能被其他窗口擁有。調(diào)用CWnd:: GetWindow可以檢索頂層窗口的列表,調(diào)用IsWindowVisible、GetWindowTextLength以及GetOwner可以確定 窗口是否應(yīng)該在列表中。下例將把TaskManager窗口的標(biāo)題填充到列表中。
void GetTadkList (CListBox&list)
{
CString strCaption
//Caption of window.
list.ResetContent ()
//Clear list box.
//Get first Window in window list.
ASSERT_VALID (AfxGetMainWnd ())
CWnd* pWnd=AfxGetMainWnd () ->GetWindow (GW_HWNDFIRST)
//Walk window list.
while (pWnd)
{
// I window visible, has a caption, and does not have an owner?
if (pWnd ->IsWindowVisible()
&& pWnd ->GetWindowTextLength ()
&&! pWnd ->GetOwner ())
{
//Add caption o window to list box.
pWnd ->GetWindowText (strCaption)
list.AddString (strCaption)
}
//Get next window in window list.
pWnd=pWnd ->GetWindow(GW_HWNDNEXT)
}
}
(69) 如何確定Windows和Windows系統(tǒng)目錄
有兩個(gè)SDK函數(shù)可以完成該功能。GetWindowsDirectory和GetSystemDirectory,下例說(shuō)明了如何使用這兩個(gè)函數(shù):
TCHAR szDir [MAX_PATH]
//Get the full path of the windows directory.
∶ : GetWindowsDirectory (szDir, MAX_PATH)
TRACE ("Windows directory %s\n", szDir)
//Get the full path of the windows system directory.
∶ : GetSystemDirectory (szDir, MAX_PATH)
TRACE ("Windows system directory %s\n", szDir)
(70) 在哪兒創(chuàng)建臨文件
調(diào)用SDK函數(shù)GetTemPath可以確定臨時(shí)文件的目錄,該函數(shù)首先為臨時(shí)路徑檢測(cè)TMP環(huán)境變量:如果沒(méi)有指定TMP,檢測(cè)TMP環(huán)境變量,然后返回到當(dāng)前目錄。下例說(shuō)明了如何創(chuàng)建一個(gè)臨時(shí)文件。
…
//get unique temporary file.
CString strFile
GetUniqueTempName (strFile)
TRY
{
//Create file and write data.Note that file is closed
//in the destructor of the CFile object.
CFile file (strFile,CFile ::modeCreate | Cfile:: modeWrite)
//write data
}
CATCH (CFileException, e)
{
//error opening file
}
END_CATCH
…
Void GetuniqueTempName (CString& strTempName)
{
//Get the temporary files directory.
TCHAR szTempPath [MAX_PATH]
DWORD dwResult=:: GetTempPath (MAX_PATH, szTempPath)
ASSERT (dwResult)
//Create a unique temporary file.
TCHAR szTempFile [MAX_PATH]
UINT nResult=GetTempFileName (szTempPath, _T ("~ex"),0,szTempfile)
ASSERT (nResult)
strTempName=szTempFile
}
(71) 我怎樣才能建立一個(gè)等待光標(biāo)?
調(diào) 用 BeginWaitCursor 函 數(shù) 來(lái) 啟 動(dòng) 等 待 光 標(biāo),調(diào) 用 EndWaitCursor 函 數(shù) 來(lái) 結(jié) 束 等 待 光 標(biāo)。要 注 意,二 者 都 要 調(diào) 用 app 的 成 員 函 數(shù),如 下 所 示:
AfxGetApp()->BeginWaitCursor();
// 要做的事
AfxGetApp()->EndWaitCursor();
(72) 我在MDI框架中有個(gè) form 視窗。它有個(gè)取消按鈕,我需要當(dāng)用戶(hù)按取消按鈕時(shí)可關(guān)閉form視窗。我應(yīng)該如何關(guān)閉該文檔?
調(diào) 用 OnCloseDocument 函 數(shù)。
(73) 如何訪(fǎng)問(wèn)桌面窗口
靜態(tài)函數(shù)CWnd:: GetDesktopWindow 返回桌面窗口的指針。下例說(shuō)明了MFC函數(shù)CFrameWnd::BeginModalStae是如何使用該函數(shù)進(jìn)入內(nèi)部窗口列表的。
void CFrameWnd::BeginModalState ()
{
…
//first count all windows that need to be disabled
UINT nCount=0
HWND hWnd= :: GetWindow (:: GetDesktopWindow(), GW_CHILD)
while (hWnd!=NULL)
{
if (:: IsWindowEnabled (hwnd)
&& CWnd::FromHandlePermanent (hWnd)!=NULL
&& AfxIsDescendant (pParent->m_hWnd, hWnd)
&& :: SendMessage (hWnd, WM_DISABLEMODAL, 0, 0)==0)
{
++nCount
}
hWnd=:: GetWindow (hWnd, GW_HWNDNEXT)
}
…
(74) 什么是COLORREF? 我該怎樣用它?
COLORREF 是 一 個(gè) 32-bit 整 型 數(shù) 值,它 代 表 了 一 種 顏 色。你 可 以 使 用 RGB 函 數(shù) 來(lái) 初 始 化 COLORREF。例 如:
COLORREF color = RGB(0, 255, 0);
RGB 函 數(shù) 接 收 三 個(gè) 0-255 數(shù) 值,一 個(gè) 代 表 紅 色, 一 個(gè) 代 表 綠 色, 一 個(gè) 代 表 藍(lán) 色。在 上 面的 例 子 中, 紅 色 和 藍(lán) 色 值 都 為 0,所 以 在 該 顏 色 中 沒(méi) 有 紅 色 和 藍(lán) 色。綠 色 為 最 大 值 255。所 以 該 顏 色 為 綠 色。0,0,0 為 黑 色,255,255,255 為 白 色。
另 一 種 初 始 化 COLORREF 的 方 法 如 下 所 示:
CColorDialog colorDialog;
COLORREF color;
if( colorDialog.DoModal() == IDOK )
{
color = colorDialog.GetColor();
}
這 段 代 碼 使 用 了 MFC 中 的 顏 色 對(duì) 話(huà) 框,它 需 要 文 件。
(75) AppWizard所產(chǎn)生的STDAFX文件是干什么用的?
它 主 要 是 協(xié) 助 產(chǎn) 生 預(yù) 編 譯 頭 文 件 的。通 常 你 是 不 需 要 修 改 它 的。
(76) 我在我的程序中是了CDWordArray。我向它添加了約10,000個(gè)整數(shù),這使得它變得非常非常慢。為什么會(huì)這么糟?
CDWordArray 是 很 好 用 的,只 是 因 為 你 沒(méi) 有 指 定 數(shù) 組 的最大尺寸。因 此,當(dāng) 你 添 加 新 元 素 時(shí),該 類(lèi) 會(huì) 從 堆 中 重 新 分 配 空 間。不 幸 的 是,該 類(lèi) 會(huì) 在 每 次 插 入 新 元 素 時(shí) 都 為 數(shù) 組 重 新 分 配 空 間。如 果 你 向 它 添 加 了 很 多 新 元 素,所 有 這 些 分 配 和 復(fù) 制 數(shù) 組 的 操 作 會(huì) 就 會(huì) 使 它 變 慢。解 決 該 問(wèn) 題 的 方 法 是,你 可 以 使 用 SetSize 函 數(shù) 的 第 二 個(gè) 參 數(shù) 來(lái) 改 變 這 種 重 新 分 配 的 頻 率。例 如,如 果 你 把 該 參 數(shù) 設(shè) 置 為 500,則 每 次 數(shù) 組 空 間 超 出 時(shí) 它 才 重 新 分 配 并 添 加 500 個(gè) 新 空 間,而 不 是 1 個(gè)。這 樣 一 來(lái),你 就 可 以 不 用 重 新 分 配 而 添 加 了 另 外 499 個(gè) 元 素 空 間,這 也 會(huì) 大 大 提 高 程 序 的 運(yùn) 行 速 度。
(77) 我該如何改變MDI框架窗口的子窗口的大小以使在窗口以一定的大小打開(kāi)?
在 視 中 的 OnInitialUpdate 函 數(shù) 中 調(diào) 用 GetParentFrame 函 數(shù)。GetParentFrame 會(huì) 返 回 一 指 向 一 保 存 有 該 視 的 框 架 窗 口 的 指 針。然 后 調(diào) 用 在 框 架 窗 口 上 調(diào) 用 MoveWindow。
(78) 在我的程序的某些部分,我可以調(diào)用 MessageBox 函數(shù)來(lái)建立一個(gè)信息對(duì)話(huà)框,例如在視類(lèi)中。但是,在其它部分我卻不能,如文檔類(lèi)中。為什么?我怎樣才能在我的應(yīng)用程序類(lèi)中建立一個(gè)信息對(duì)話(huà)框?
MessageBox 函 數(shù) 來(lái) 自 CWnd 類(lèi),所 以 你 只 能 在 從 CWnd 繼 承 的 類(lèi) ( 如 CView ) 中 調(diào) 用 它。但 是,MFC 也 提 供 了 AfxMessageBox 函 數(shù),你 可 以 在 任 何 地 方 調(diào) 用 它。
(79) 我需要在我的程序中設(shè)置全局變量,以使文檔中的所有類(lèi)都能訪(fǎng)問(wèn)。我應(yīng)該吧它放到哪兒?
把 該 變 量 放 到 該 應(yīng) 用 程 序 類(lèi) 的 頭 文 件 中 的 attribute 處。然 后,在 程 序 的 任 何 地 方,你 都 可 以 用 下 面 的 方 法 來(lái) 訪(fǎng) 問(wèn) 該 變 量:
CMyApp *app = (CMyApp *)AfxGetApp();
app->MyGlobalVariable = ...
(80) 我聽(tīng)說(shuō)MFC可以發(fā)現(xiàn)內(nèi)存漏洞,我怎樣使用該特性?
如 果 你 在 Debug 菜 單 中 的 Go 選 項(xiàng) ( 不 是 Project 菜 單 中 的 Execute 選 項(xiàng) ) 來(lái) 運(yùn) 行 你 的 應(yīng) 用 程 序,MFC 應(yīng) 該 在 程 序 終 止 時(shí) 報(bào) 告 內(nèi) 存 漏 洞。如 果 沒(méi) 有,那 么 試 試 運(yùn) 行 MFC Tracer 工 具 程 序 ( 在 VC++ 程 序 組 中 ),并 啟 動(dòng) 跟 蹤。然 后 返 回 應(yīng) 用 程 序。
(81) 我怎樣才能在我的應(yīng)用程序中循環(huán)瀏覽已經(jīng)打開(kāi)的文檔?
使用CDocTemplate中未公開(kāi)的GetFirstDocPosition()和GetNextDoc()函數(shù)。
(82)才能在我的應(yīng)用程序中循環(huán)瀏覽已經(jīng)打開(kāi)的視?
使 用 CDocument 中 未 公 開(kāi) 的 GetFirstViewPosition() 和 GetNextView() 函 數(shù)。
(83)數(shù)PreCreateWindow是干什么用的?
PreCreateWindow 允 許 你 在 調(diào) 用 CreateWindow 之 前 來(lái) 改 變 窗 口 屬 性。
(84)該怎樣防止MFC在窗口標(biāo)題欄上把文檔名預(yù)置成應(yīng)用程序名?
在 PreCreateWindow 函 數(shù) 中 刪 除 FWS_PREFIXTITLE 標(biāo) 志 的 窗 口 樣 式:
cs.style &= ~FWS_PREFIXTITLE;
(85) 我應(yīng)該怎樣防止MFC在窗口標(biāo)題欄上添加文檔名?
在 PreCreateWindow 函 數(shù) 中 刪 除 FWS_ADDTOTITLE 標(biāo) 志 的 窗 口 樣 式:
cs.style &= ~FWS_ADDTOTITLE ;
(86) 我應(yīng)該如何改變視窗口的大小?
因 為 視 窗 口 實(shí) 際 上 是 框 架 窗 口 的 子 窗 口,所 以 你 必 須 改 變 框 架 窗 口 的 大 小,而 不 是 改 變 視 窗 口。使 用 CView 類(lèi) 中 的 GetParentFrame() 函 數(shù) 獲 得 指 向 框 架 窗 口 的 指 針,然 后 調(diào) 用 MoveWindow() 函 數(shù) 來(lái) 改 變 框 架 的 大 小。這 會(huì) 使 變 尺 寸 的 視 充 滿(mǎn) 框 架 窗 口。
(87) 我有一無(wú)模式對(duì)話(huà)框。我怎樣才能在窗口退出時(shí)刪除CDialog對(duì)象?
把“delete this”加 到 PostNcDestroy 中。這 主 要 用 在 需 要 自 動(dòng) 刪 除 對(duì) 象 的 場(chǎng) 合。
(88) 為什么把“delete this”放在PostNcDestroy中而不是OnNcDestroy?
OnNcDestroy 只 被 已 建 立 的 窗 口 調(diào) 用。如 果 建 立 窗 口 失 敗 ( 如 PreCreateWindow ),則 沒(méi) 有 窗 口 處 來(lái) 發(fā) 送 WM_NCDESTROY 消 息。PostNcDestroy 是 在 對(duì) 象 窗 口 被 完 全 刪 除,在 OnNcDestroy 后,甚 至 在 窗 口 建 立 失 敗 之 后 調(diào) 用 的。
(89) File菜單中的MRU列表是從哪兒來(lái)的?列表中的名字放在哪兒了?我怎樣才能改變列表中項(xiàng)目的最大值?
在 應(yīng) 用 程 序 類(lèi) 的 InitInstance 函 數(shù) 中 對(duì) LoadStdProfileSettings 的 調(diào) 用 中。該 調(diào) 用 接 受 一 個(gè) 參 數(shù) ( 在 缺 省 情 況 下 如 果 沒(méi) 有 傳 遞 值 則 為 4 )。MRU 文 件 名 是 從 INI 文 件 中 調(diào) 用 的。如 果 你 有 帶 有 ID_FILE_MRU_FILE1 的 ID 的 菜 單 選 項(xiàng),它 會(huì) 為 調(diào) 入 的 MRU 列 表 所 替 換。如 果 你 改 變 傳 遞 給 LoadStdProfileSettings 的 數(shù) 值 ( 最 大 為 16 ),則 你 就 改 變 了 所 裝 如 文 件 名 的 最 大 值。
(90) 我在菜單中添加了新的項(xiàng)。但是,當(dāng)我選該項(xiàng)時(shí),在狀態(tài)欄上沒(méi)有出現(xiàn)任何提示信息。為什么?
打 開(kāi) 資 源 文 件 中 的 菜 單 模 板。打 開(kāi) 新 菜 單 選 項(xiàng) 的 屬 性 對(duì) 話(huà) 框。在 對(duì) 話(huà) 框 的 底 部 的 Prompt 編 輯 框 中 ,你 可 以 如 下 指 定 狀 態(tài) 欄 上 的 提 示 信 息 和 工 具 欄 上 的 提 示 信 息 ( 如 果 你 已 經(jīng) 建 立 的 工 具 欄 按 鈕 ):
Status bar string\nFlying tag
(91) 我怎樣才能在應(yīng)用程序的缺省系統(tǒng)菜單中加上一些東西?
系 統(tǒng) 菜 單 與 其 它 菜 單 類(lèi) 似,你 可 以 添 加 或 刪 除 項(xiàng) 目,這 需 要 使 用 CMenu 類(lèi) 的 成 員 函 數(shù)。下 面 的 代 碼 在 你 的 系 統(tǒng) 菜 單 后 面 添 加 一 個(gè) 新 菜 單 項(xiàng):
CMenu *sysmenu;
sysmenu = m_pMainWnd->GetSystemMenu(FALSE);
sysmenu->AppendMenu(MF_STRING, 1000, "xxx");
參 見(jiàn) MFC 幫 助 文 件 中 的 CMenu 類(lèi)。
(92) 我建立了一個(gè)對(duì)話(huà)框。但是當(dāng)我顯示該對(duì)話(huà)框時(shí),第一個(gè)編輯框總是不能獲得焦點(diǎn),我必須單擊它來(lái)使它獲得焦點(diǎn)。我怎樣才能使第一個(gè)編輯框在對(duì)話(huà)框打開(kāi)時(shí)就獲得焦點(diǎn)?
打 開(kāi) 資 源 編 輯 器 中 的 對(duì) 話(huà) 框 模 板。在 Layout 菜單 中 選 擇 Tab Order 選 項(xiàng)。按 你 的 需 求 單 擊 對(duì) 話(huà) 框 中 的 控 制 來(lái) 重 新 排 列 這 些 控 制 的 tab 順 序。
(93) 我怎樣才能使一個(gè)窗口具有“always on top”特性?
在 調(diào) 用 OnFileNew 后,在 你 的 InitInstance 函 數(shù) 中 加 上 下 面 的 代 碼:
m_pMainWnd->SetWindowPos(&CWnd::wndTopMost,0,0,0,0, SWP_NOMOVE | SWP_NOSIZE);
(94) 我要為我的form view添加文檔模板。我先建立了對(duì)話(huà)框模板,然后使用ClassWizard建立了基于CFormView的新類(lèi),它也是從CDocument繼承來(lái) 的。我還建立了相應(yīng)的資源并在InitInstance中添加了新的文檔模板。但是,當(dāng)我試圖運(yùn)行該程序時(shí),出現(xiàn)了Assertion信息。為什么?
form 的 對(duì) 話(huà) 框 模 板 需 要 些 特 殊 設(shè) 置 以 便 可 用 于 CFromView。確 保 這 些 設(shè) 置 的 最 簡(jiǎn) 單 方 法 是 使 用 AppWizard 來(lái) 建 立 CFormView 應(yīng) 用 程 序,并 查 看 AppWizard 所 建 立 的 對(duì) 話(huà) 框 模 板 所 選 擇 的Styles Properties。你 會(huì) 發(fā) 現(xiàn) 該 對(duì) 話(huà) 框 模 板 具 有 下 列 樣 式:沒(méi) 有 標(biāo) 題 欄、不 可 見(jiàn) 和“Child”。把 你 的 form view 的 對(duì) 話(huà) 框 屬 性 變 成 這 樣 就 可 以 了。
(95) 我在一對(duì)話(huà)框中有一列表框,我需要tabbed列表框中的項(xiàng)目。但是,當(dāng)我處理含有tab字符(用AddString添加的)的列表項(xiàng)時(shí),tab被顯示成小黑塊而沒(méi)有展開(kāi)。哪兒出錯(cuò)了?
在 對(duì) 話(huà) 框 模 版 中,打 開(kāi) 列 表 框 的 屬 性。確 保 選 擇 了“Use Tabstops” 樣 式。然 后,確 保 在 對(duì) 話(huà) 框 類(lèi) 中 OnInitDialog 函 數(shù) 中 調(diào) 用 SetTabStops。
(96) 我建立了一個(gè)應(yīng)用程序,并使用了CRecordset類(lèi)。但是,當(dāng)我運(yùn)行該程序時(shí),它試圖要訪(fǎng)問(wèn)數(shù)據(jù)庫(kù),并給出“Internal Application Error”對(duì)話(huà)框。我應(yīng)該怎樣做?
通 常 情 況 下,當(dāng) 你 的 程 序 中 向 數(shù) 據(jù) 庫(kù) 發(fā) 送 信 息 的 SQL 語(yǔ) 句 出 現(xiàn) 問(wèn) 題 時(shí) 才 出 現(xiàn) 該 對(duì) 話(huà) 框。例 如,參 見(jiàn) 下 面 的 例 子:
set.m_strFilter = "(ZipCode = '27111')";
如 果 ZipCode 列 被 定 義 為 字 符 串 時(shí) 不 會(huì) 出 現(xiàn) 問(wèn) 題,如 果 定 義 為 long,則 會(huì) 出 現(xiàn)“Internal Application Error”對(duì) 話(huà) 框,這 是 由 于 類(lèi) 型 不 匹 配 的 緣 故。如 果 你 刪 除 27111 的 單 引 號(hào),則 不 會(huì) 出 現(xiàn) 問(wèn) 題。當(dāng) 你 看 到“Internal Application Error”時(shí),最 好 檢 查 一 下 試 圖 要 發(fā) 送 給 數(shù) 據(jù) 庫(kù) 的 SQL 語(yǔ) 句。
(97) 我用ClassWizard建立了一個(gè)類(lèi)。但是,我把名字取錯(cuò)了,我想把它從項(xiàng)目中刪除,應(yīng)該如何做?
在 ClassWizard 對(duì) 話(huà) 框 關(guān) 閉 后,用 文 件 管 理 器 刪 除 新 類(lèi) 的 H 和 CPP 文 件。然 后 打 開(kāi) ClassWizard,它 會(huì) 提 示 丟 失 了 兩 個(gè) 文 件,并 詢(xún) 問(wèn) 你 該 如 何 做。你 可 以 選 擇 從 項(xiàng) 目 中 刪 除 這 兩 個(gè) 問(wèn) 的 按 鈕。
(98) 當(dāng)我打開(kāi)應(yīng)用程序中的窗口時(shí),我要傳遞該窗口的矩形尺寸。該矩形指定了窗口的外圍大小,但是當(dāng)我調(diào)用GetClientRect時(shí),所得到的尺寸要比所希望的值要小(因?yàn)楣ぞ邫诤痛翱谶吙虻木壒?。有其它方法來(lái)計(jì)算窗口的尺寸嗎?
參 見(jiàn) CWnd::CalcWindowRect。
(99) 我在文檔類(lèi)中設(shè)置了一個(gè)整型變量。但是,當(dāng)我試圖把該變量寫(xiě)入Serialize函數(shù)中的archive文件中時(shí),出現(xiàn)了類(lèi)型錯(cuò)誤。而文檔中的其它變量沒(méi)有問(wèn)題。為什么?
archive 類(lèi) 只 重 載 某 些 類(lèi) 型 的 >> 和 << 操 作 符。“int”類(lèi) 型 沒(méi) 有 在 其 中,也 許 是 因 為 int 變 量 在 Windows 3.1 與 Windows NT/95 有 所 不 同 的 緣 故 吧。“long”類(lèi) 型 得 到 了 支 持,所 以 你 可 以 把 int 類(lèi) 型 改 成 long 型。參 見(jiàn) MFC 幫 助 文 件 中 CArchive 類(lèi)。
(100) 如何控制菜單的大小?
我用MFC的CMenu生成了一個(gè)動(dòng)態(tài)菜單(例如File,Edit,View...Help), 我想控制這個(gè)菜單的大小(長(zhǎng)+高).
方法一:查找 WM_MEASUREITEM 和 MEASUREITEMSTRUCT.
方法二:查詢(xún)系統(tǒng)::GetSystemMetric(SM_CXMENUSIZE).
/* 你可以通過(guò)如下代碼來(lái)獲得文本的大小:
(A)獲得被使用的字體 */
NONCLIENTMETRICS ncm;
HFONT hFontMenu;
SIZE size;
size.cy = size.cy = 0;
memset(&ncm, 0, sizeof(NONCLIENTMETRICS));
ncm.cbSize = sizeof(NONCLIENTMETRICS);
if (SystemParametersInfo(SPI_GETNONCLIENTMETRICS, sizeof(NONCLIENTMETRICS), &ncm, 0))
{
hFontMenu = CreateFontIndirect(&ncm.lfMenuFont);
/*
(B) 獲得菜單項(xiàng)的文本: */
char szText[_MAX_PATH];
pMenu->GetMenuString(0, szText, _MAX_PATH, MF_BYPOSITION);
/*
然后,獲得菜單項(xiàng)文本的高度: */
HFONT hFontOld;
HDC hDC;
hDC = ::GetDC(NULL);
hFontOld = (HFONT) ::SelectObject(hDC, hFontMenu);
GetTextExtentPoint32(hDC, szText, lstrlen(szText), &size);
SelectObject(hDC, hFontOld);
::ReleaseDC(NULL, hDC);
}
/*此時(shí),size.cy即為高度,size.cx為寬度,你可以給菜單加上自定義的高度和寬度,通過(guò)比較,我發(fā)現(xiàn)寬度為4
比較合適。*/