新建一个MFC单文档应用程序,工程名为Graphic.
1.画点、直线、矩形、椭圆
在菜单上添加一个绘图弹出菜单,并添加点、直线、矩形、椭圆菜单项,ID分别为:IDM_POINT、IDM_LINE、IDM_RECTANGLE、IDM_ELLIPS,并分别添加命令消息响应,
编辑:
void CGraphicView::OnPoint() { // TODO: Add your command handler code here m_nDrawType=1; } void CGraphicView::OnLine() { // TODO: Add your command handler code here m_nDrawType=2; } void CGraphicView::OnRectangle() { // TODO: Add your command handler code here m_nDrawType=3; } void CGraphicView::OnEllpse() { // TODO: Add your command handler code here m_nDrawType=4; }
在CGraphicView类中添加私有的成员变量:
private: CPoint m_ptOrigin; UINT m_nDrawType;
并在构造函数中初始化:
CGraphicView::CGraphicView() { // TODO: add construction code here m_nDrawType=0;//初始化为0 m_ptOrigin=0;//将原点设置成(0,0) }
再在CGraphicView类上添加两个消息处理函数,WM_LBUTTONDOWN,WM_LBUTTONUP,编辑:
void CGraphicView::OnLButtonDown(UINT nFlags, CPoint point) { // TODO: Add your message handler code here and/or call default m_ptOrigin=point;//当鼠标左键按下时,将当前点保存到成员变量当中 CView::OnLButtonDown(nFlags, point); } void CGraphicView::OnLButtonUp(UINT nFlags, CPoint point) { // TODO: Add your message handler code here and/or call default CClientDC dc(this);
CPen pen(PS_SOLID,1,RGB(255,0,0));//创建画笔
dc.SelectObject(&pen);//将画笔选到设备描述表当中
CBrush *pBrush=CBrush::FromHandle
((HBRUSH)GetStockObject(NULL_BRUSH));//创建透明的画刷
dc.SelectObject(pBrush);//将画刷选到设备描环境当中 switch(m_nDrawType) { case 1: dc.SetPixel(point,RGB(255,0,0)); break; case 2: dc.MoveTo(m_ptOrigin); dc.LineTo(point); break; case 3: dc.Rectangle(CRect(m_ptOrigin,point)); break; case 4: dc.Ellipse(CRect(m_ptOrigin,point)); break; default: break; } CView::OnLButtonUp(nFlags, point); }
运行,OK !
2.给程序添加设置对话框
在对话框资源中新建一个对话框,ID: IDD_DIALOG_SETTING ,标题:Setting ,并添加一个静态文本控件(标题:线宽)和一个编辑框控件(ID: IDC_LINE_WIDTH),双击新建一对话框控件,给它新增一个类(类名:CSettingDlg),由于VC6.0本身的问题,这里要删除工程目录下的Graphic.clw文件,然后再打开class wizard…. 重新根据源文件建立一个类向导文件,取名仍就为Grahpic.clw。再给编辑框控件添加一个关联的UINT类型的成员变量m_nLineWidth.
再给CGraphicView类添加一个UINT类型的私有成员变量(取名也叫m_nLineWidth),并在其构造方法中初始化:m_nLineWidth=0;
然后在CGraphicView类当中对话框IDM_SETTING添加一个COMMAND消息响应函数, 编辑:
void CGraphicView::OnSettiing()
{
// TODO: Add your command handler code here
CSettingDlg dlg;//设置一个CSettingDlg对象
dlg.m_nLineWidth=m_nLineWidth;
//将View类中先前保存的值传回到对话框变量
if(IDOK==dlg.DoModal())//如果选择是的OK按钮
{
m_nLineWidth=dlg.m_nLineWidth;//将用户输入的线宽保存到m_nLineWidth
}
}
并在CGraphicView::OnLButtonUp函数中把线宽改成m_nLineWidth替代:
void CGraphicView::OnLButtonUp(UINT nFlags, CPoint point) { // TODO: Add your message handler code here and/or call default CClientDC dc(this); CPen pen(PS_SOLID,m_nLineWidth,RGB(255,0,0));//创建画笔 ……………
………….
………….
}
OK ^_^ !!!
3.添加线型设置选项
在对话框中建一个组框(标题:线型),并在组框内加三个单选按钮,标题分别为:实线,虚线,点线。并选中实线单选按钮的组属性,并给它添加一个int类型的成员变量m_nStyle 。
然后在CGraphicView::OnSettiing函数中编辑:
void CGraphicView::OnSettiing() { // TODO: Add your command handler code here CSettingDlg dlg;//设置一个CSettingDlg对象 dlg.m_nLineWidth=m_nLineWidth; dlg.m_nLineStyle=m_nLineStyle; //将View类中先前保存的值传回到对话框变量 if(IDOK==dlg.DoModal())//如果选择是的OK按钮 { m_nLineWidth=dlg.m_nLineWidth;//将用户输入的线宽保存到m_nLineWidth m_nLineStyle=dlg.m_nLineStyle; } }
在CGraphicView::OnLButtonU函数中修改:
void CGraphicView::OnLButtonUp(UINT nFlags, CPoint point) { // TODO: Add your message handler code here and/or call default CClientDC dc(this); CPen pen(m_nLineStyle,m_nLineWidth,RGB(255,0,0));//创建画笔 //因为组中的单选按实线,虚线,点线的值恰好是0,1,2,所以不需要作switch语句 .............. .............. .............. CView::OnLButtonUp(nFlags, point); }
4.创建一个颜色对话框
在资源的绘图菜单下,添加一个颜色菜单选项,并在CGrahpicView类中给它添加一个命令消息响应函数OnColor. 然后再在CGrahpicView类中添加一个COLORREF类型的私有的成员变量m_clr,
并在构造函数中将其初始化:
m_clr=RGB(255,0,0);//将颜色初始化为红色
在CGraphicView::OnColor函数中编辑:
void CGraphicView::OnColor() { // TODO: Add your command handler code here CColorDialog Dlg;//创建一个颜色对话框对象 Dlg.m_cc.Flags|=CC_RGBINIT|CC_FULLOPEN;//设置Dlg的标记,与它先前的标记组合起来 //CC_FULLOPEN是把颜色对话框完全展开 Dlg.m_cc.rgbResult=m_clr;//将View类的颜色值设置回颜色对话框去 if(IDOK==Dlg.DoModal()) { m_clr=Dlg.m_cc.rgbResult; } //DoModal:Call this function to display the Windows common color dialog box and allow the user to select a color. }
在CGraphicView::OnLButtonUp中修改:
void CGraphicView::OnLButtonUp(UINT nFlags, CPoint point) { // TODO: Add your message handler code here and/or call default CClientDC dc(this); CPen pen(m_nLineStyle,m_nLineWidth,m_clr);//创建画笔 //因为组中的单选按实线,虚线,点线的值恰好是0,1,2,所以不需要作switch语句 dc.SelectObject(&pen);//将画笔选到设备描述表当中 CBrush *pBrush=CBrush::FromHandle((HBRUSH)GetStockObject(NULL_BRUSH));//创建透明的画刷 dc.SelectObject(pBrush);//将画刷选到设备描环境当中 switch(m_nDrawType) { case 1: dc.SetPixel(point,m_clr); break; .......... .......... .......... } CView::OnLButtonUp(nFlags, point); }
OK !! ^_^
5.创建字体对话框
在资源的绘图菜单下,添加一个字体菜单选项,并在CGrahpicView类中给它添加一个命令消息响应函数OnFont. 然后再在CGrahpicView类中添加一个CFont类型的私有的成员变量m_font,和一个CString类型的私有成员变量m_strFontName,并在构造函数中将其初始化:m_strFontName="";
在CGraphicView::OnFont函数中编辑:
void CGraphicView::OnFont() { // TODO: Add your command handler code here CFontDialog dlg; if(IDOK==dlg.DoModal()) { if(m_font.m_hObject)//如果字体资源对象已存在 { m_font.DeleteObject();//就释放这种字体资源对象 } m_font.CreateFontIndirect(dlg.m_cf.lpLogFont);//再创建一种字体资源,并与之相关联 //CreateFontIndirect:This method initializes a CFont object with the characteristicsgiven in a LOGFONT structure pointed to by lpLogFont. m_strFontName=dlg.m_cf.lpLogFont->lfFaceName;//将字体的名字保存下来 Invalidate();//使窗口无效,下次WM_PAINT消息发送时,窗口就进行重绘 } }
将选择的字体的名字显示出来,在CGraphicView::OnDraw函数中编辑:
void CGraphicView::OnDraw(CDC* pDC) { CGraphicDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); // TODO: add draw code for native data here CFont *pOldFont=pDC->SelectObject(&m_font);//保存原来的字体 pDC->TextOut(0,0,m_strFontName);//将选择的字体的名字输出 pDC->SelectObject(pOldFont);//将原来的字体选择回去 }
运行, OK !!
6.在设置对话框中做示例
在IDD_DIALOG_SETTING对话框中,拖入一个组框(ID: IDC_SAMPLE,标题:示例),然后再给IDC_LINE_WIDTH(编辑框)、IDC_RADIO1(实线单选按钮)、IDC_RADIO2、IDC_RADIO3分别添加EN_CHANGE、BN_CLICKED、BN_CLICKED、BN_CLICKED消息响应函数,并在这四个消息响应函数中都添加一句:Invalidate();使窗口无效,这样下次发送WM_PAINT消息,窗口便会重绘。这样就只需要在OnPaint函数中编写一处代码就行了。
接着,在CSettingDlg类上添加一个成员变量:
public:
COLORREF m_clr;//用来保存颜色值
并在构造函数中初始化:
m_clr=RGB(255,0,0);//初始化红色
同样,在CGraphicView::OnSettiing函数中也要修改:
void CGraphicView::OnSettiing() { // TODO: Add your command handler code here CSettingDlg dlg;//设置一个CSettingDlg对象 dlg.m_nLineWidth=m_nLineWidth; dlg.m_nLineStyle=m_nLineStyle; dlg.m_clr=m_clr; //将View类中先前保存的值传回到对话框变量 if(IDOK==dlg.DoModal())//如果选择是的OK按钮 { m_nLineWidth=dlg.m_nLineWidth;//将用户输入的线宽保存到m_nLineWidth m_nLineStyle=dlg.m_nLineStyle; } }
在CSettingDlg类上添加一个WM_PAINT消息响应函数,编辑:
void CSettingDlg::OnPaint() { CPaintDC dc(this); // device context for painting // TODO: Add your message handler code here //在这里完成示例线条的绘制 UpdateData();//将控件上的值反应到成员变量当中 CPen pen(m_nLineStyle,m_nLineWidth,m_clr); dc.SelectObject(&pen); CRect rect; GetDlgItem(IDC_SAMPLE)->GetWindowRect(&rect);//得到静态文本控件矩形区域的大小 ScreenToClient(rect);//将屏幕坐标转换成客户区坐标 dc.MoveTo(rect.left+12,rect.top+rect.Height()/2+3);//示例线条初始位置的坐标 dc.LineTo(rect.right-12,rect.top+rect.Height()/2+3);//示例线条末端坐标,加3是为了调整坐标 // Do not call CDialog::OnPaint() for painting messages }
7.改变对话框控件的背景色及文字颜色
修改整个对话框及其控件的背景色
设置对话框控件的背景色,在CSettingDlg类下添加一个WM_CTLCOLOR消息处理函数,并添加一个私有的成员变量:CBrush m_brush; 并在构造函数中初始化:
m_brush.CreateSolidBrush(RGB(0,0,255));//对画刷进行初始化,蓝色
然后在CSettingDlg::OnCtlColor函数中修改:
HBRUSH CSettingDlg::OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor) { HBRUSH hbr = CDialog::OnCtlColor(pDC, pWnd, nCtlColor); // TODO: Change any attributes of the DC here // TODO: Return a different brush if the default is not desired //return hbr; return m_brush; }
修改某一个控件的背景色
对线型组框修改,首先将线型组框的ID改为:IDC_LINE_STYLE,并在CSettingDlg::OnCtlColor中编辑:
HBRUSH CSettingDlg::OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor) { HBRUSH hbr = CDialog::OnCtlColor(pDC, pWnd, nCtlColor); // TODO: Change any attributes of the DC here if(pWnd->GetDlgCtrlID()==IDC_LINE_STYLE)//匹配是否是线型组框 { pDC->SetTextColor(RGB(255,0,0));//设置组框文字颜色为红色 pDC->SetBkMode(TRANSPARENT); //设置文字的背景色为透明 return m_brush; //设置相匹配的控件的背景色 } if(pWnd->GetDlgCtrlID()==IDC_LINE_WIDTH)//匹配编辑框 { pDC->SetTextColor(RGB(255,0,0));//设置编辑框文字颜色为红色 pDC->SetBkMode(TRANSPARENT); //设置文字的背景色为透明 //pDC->SetBkColor(RGB(255,0,0));//设置文字背景色为蓝色 return m_brush; } // TODO: Return a different brush if the default is not desired return hbr; //return m_brush; }
8.修改控件的字体
在对话框中插入一个静态文本控件(ID:IDC_TEXT,标题:程序员),再给CStringDlg类添加一个私有的成员变量m_font,并在构造函数中初始化:
m_font.CreatePointFont(200,"华文行楷");
在CSettingDlg::OnCtlColor添加:
if(pWnd->GetDlgCtrlID()==IDC_TEXT) //匹配程序员静态文本控件
{
pDC->SelectObject(&m_font); //设置字体
}
9.修改Button控件的背景色和文本颜色
修改OK按钮的文字颜色
先创建一个Button类,然后将CSetttingDlg这个对话框上的按钮控件与它关联。Insert -> new class -> (class type:MFC Class,name:CTestBtn,Base Class:CButton)如果这里出现问题了,可以重新建立那个Class wizard数据库文件。然后可以在CTestBtn上增加一个DrawItem虚函数,并添加以下代码:
void CTestBtn::DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct) { // TODO: Add your code to draw the specified item UINT uStyle = DFCS_BUTTONPUSH; // This code only works with buttons. ASSERT(lpDrawItemStruct->CtlType == ODT_BUTTON); // If drawing selected, add the pushed style to DrawFrameControl. if (lpDrawItemStruct->itemState & ODS_SELECTED) uStyle |= DFCS_PUSHED; // Draw the button frame. ::DrawFrameControl(lpDrawItemStruct->hDC, &lpDrawItemStruct->rcItem, DFC_BUTTON, uStyle); // Get the button's text. CString strText; GetWindowText(strText); // Draw the button text using the text color red. COLORREF crOldColor = ::SetTextColor(lpDrawItemStruct->hDC, RGB(255,0,0)); ::DrawText(lpDrawItemStruct->hDC, strText, strText.GetLength(), &lpDrawItemStruct->rcItem, DT_SINGLELINE|DT_VCENTER|DT_CENTER); ::SetTextColor(lpDrawItemStruct->hDC, crOldColor); }
接下来,给OK按钮关联一个成员变量(variable name:m_bntTesm,Category:Control,Variable type:CTestBtn),并在CSettingDlg.h头文件中包含:#include "TestBtn.h" 头文件。并将OK按钮的属性中的styles的own draw选项选中。运行,OK!
修改cancle按钮的背景色
创建一个基于CButton的类叫CSXBtn,并在这个类里插入一个DrawItem消息处理,编辑:
void CSXBtn::DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct) { UINT uStyle = BS_DEFPUSHBUTTON ;//DFCS_BUTTONPUSH; // This code only works with buttons. ASSERT(lpDrawItemStruct->CtlType == ODT_BUTTON); // If drawing selected, add the pushed style to DrawFrameControl. if (lpDrawItemStruct->itemState & ODS_SELECTED) uStyle |= DFCS_PUSHED; // Draw the button frame. ::DrawFrameControl(lpDrawItemStruct->hDC, &lpDrawItemStruct->rcItem ,DFC_BUTTON, uStyle); CDC* pDC = CDC::FromHandle(lpDrawItemStruct->hDC); // Get the button's text. CString strText; GetWindowText(strText); // Draw the button text using the text color red. CBrush B; CRect rect; CRect focusRect; focusRect.CopyRect(&lpDrawItemStruct->rcItem); DrawFocusRect(lpDrawItemStruct->hDC, (LPRECT)&focusRect); focusRect.left += 4; focusRect.right -= 4; focusRect.top += 4; focusRect.bottom -= 4; rect.CopyRect(&lpDrawItemStruct->rcItem); pDC->Draw3dRect(rect, ::GetSysColor(COLOR_BTNSHADOW), ::GetSysColor(COLOR_BTNHILIGHT)); B.CreateSolidBrush(RGB(0,255,0)); ::FillRect(lpDrawItemStruct->hDC,&rect, (HBRUSH)B.m_hObject); ::SetBkMode(lpDrawItemStruct->hDC,TRANSPARENT); COLORREF crOldColor = ::SetTextColor(lpDrawItemStruct->hDC, RGB(255,0,0)); ::DrawText(lpDrawItemStruct->hDC, strText, strText.GetLength(), &lpDrawItemStruct->rcItem, DT_SINGLELINE|DT_VCENTER|DT_CENTER); ::SetTextColor(lpDrawItemStruct->hDC, crOldColor); }
10.在窗口中贴图
步骤:1.创建位图
CBitmap bitmap;
Bitmap.LoadBitmap(IDB_BITMAP);
2.创建兼容DC
CDC dcCompatible;
dcCompatible.CreateCompatibleDC(pDC);
3.将位图选到兼容DC中
dcCompatible.SelectObject(&bitmap);
4.将兼容DC中的位图贴到当前DC中
pDC->BitBlt(rect.left,rect.top,rect.Weight(),
rect.Height(),&dcCompatible,0,0,SRCCOPY);
BitBllt: Copies a bitmap from the source device context to this current device context.
Insert -> 资源 -> Bitmap -> 引入一张位图,然后在CGraphicView类上添加一个WM_ERASEBKGND消息处理,
编辑:
BOOL CGraphicView::OnEraseBkgnd(CDC* pDC) //探除窗口 { // TODO: Add your message handler code here and/or call default CBitmap bitmap; bitmap.LoadBitmap(IDB_BITMAP1);//加载位图 BITMAP bmp;//定义一个BITMAP结构体变量 bitmap.GetBitmap(&bmp); //GetBitmap:to retrieve information about a CBitmap object. //This information is returned in the BITMAP structure referred to by pBitmap. CDC dcCompatible; dcCompatible.CreateCompatibleDC(pDC);//创建兼容DC dcCompatible.SelectObject(&bitmap);//将位图选到兼容DC CRect rect; GetClientRect(&rect); //pDC->BitBlt(0,0,rect.Width(),rect.Height(),&dcCompatible,0,0,SRCCOPY); //将兼容DC中的位图贴到当前DC中 pDC->StretchBlt(0,0,rect.Width(),rect.Height(),&dcCompatible, 0,0,bmp.bmWidth,bmp.bmHeight,SRCCOPY); //以上的代码也可以移到OnDraw函数当中,也可以实现这个功能 //OnEraseBkgnd函数: Nonzero if it erases the background; otherwise, it is zero. //return CView::OnEraseBkgnd(pDC);//不能再调用基类的擦除背景函数 return TRUE; }
OK, ^_^ !
练习:
创建一个CDialogBar,然后在CDialogBar上排放一些控件,如放置一个编辑框和列表框,改变编辑框的背景颜色和文字的颜色以及DialogBar本身的背景色;另外要求在列表当中显示一张位图。
