Fork me on GitHub

VC学习笔记:对话框

VC学习笔记:对话框

SkySeraph NOV.11st 2009 HQU

Email-zgzhaobo@gmail.com  QQ-452728574

Latest Modified Date:Oct.31th 2010 HQU 重新翻阅整理

//说明:孙鑫视频学习笔记 SkySeraph NOV.11st 2009 HQU

/*【对话框专题知识】*/

//知识点:创建对话框;显示对话框;

//新建Single Document程序

void CMyBoleView::OnDialog() //对话框的响应函数

{

//首先创建一个新的对话框资源,并增加一个相关的类[CTestDlg]用来操作该资源。再增加一个菜单项(view类)用来显示该对话框。【每增加一个新对话框都需增加一个相关类来做相关处理】

//▲注意:要在view类头文件里包含testdlg.h,才能在菜单项的消息响应函数使用该类。

/*■1.<<创建、显示、关闭模态对话框>>*/

CTestDlg dlg; //此处是局部变量,但不要紧,因为程序会暂时停下来等模态对话框关闭后再运行;

dlg.DoModal();//创建并显示模态对话框

 

/*■2.<<创建、显示、销毁非模态对话框>>*/

//>>错误写法:CTestDlg dlg; 

//▲不能定义成局部变量。局部变量两种处理方法:[1]定义成view类的成员变量;[2]定义成指针,在堆栈上分配内存,因为堆栈上的内存和程序的整个生命期是一致的(前提是程序不能主动销毁)

CTestDlg *pDlg=new CTestDlg;

//▲>>指针变量是一个局部变量,生命周期结束后其保存的内存地址就丢失了,程序无法再引用了,解决方法:

//[1]定义为视类的成员变量,再在其析构函数里用delete函数释放内存;

//[2]在CTestDlg类中重载PostNcDestroy虚函数,释放this指针指向的内存。

pDlg->Create(IDD_DIALOG1,this); //第一个参数可以是对话框资源ID或对话框模板名称;第二个参数指定父窗口,为NULL时即为主应用程序窗口

pDlg->ShowWindow(SW_SHOW);

//▲>>非模态与模态的区别:[1]显示 [2]局部变量的使用 [3]按下ok键后,模态的会调用OnOK虚函数,其又会调用EndDialog函数,从而终止模态对话框而非模态的并不销毁对话框,仅使其不可见,故要重写基类的OnOK虚函数(或OnCancel虚函数),在重写函数中调用DestroyWindow函数

pDlg->DestroyWindow();//销毁非模态对话框

delete pDlg;//释放指针

 

/*■3.1<<动态创建模态对话框>>*/

CTestDlg dlg;//局部变量不要紧,因为程序会暂时停下来等模态对话框关闭后再运行

dlg.DoModal();//创建并显示模态对话框

}

 

 

/*■3.<<动态创建模态对话框>>*/

//功能:单击对话框中某个按钮时(在对方框),就在对话框中动态创建一个新按钮

void CTestDlg::OnBtnAdd() //按钮Add的响应函数

{

//>>先创建模态对话框(view类Ondialog中,见上3.1),再在对话框IDD_DIALOG1添加一按钮Add并对其添加消息处理函数(CTestDlg类,也即此函数)

//>>再在CTestDlg类中添加CButton型的私有成员变量m_bt。为m_bt对象是否已经与窗口关联,避免重复关联而出错,有如下三种实现方法:

//▲法一:BOOL型私有成员变量m_bIsCreated(在其构造函数中初始化为FALSE),用来标识是否已经创建过按钮窗口了

if(m_bIsCreated==FALSE) //确保m_bt没有与某窗口绑定了

{

m_bt.Create("New",BS_DEFPUSHBUTTON | WS_CHILD | WS_VISIBLE,CRect(0,0,100,200),this,123); //CButton类的Create用来创建按钮。若无WS_VISIBLE,则要加ShowWindow

m_bIsCreated=TRUE;

}

else

{

m_bt.DestroyWindow();

m_bIsCreated=FALSE;

}        

//▲法二:若不用成员变量m_bISCreated,也可以用下方法

    static BOOL bIsCreated=FALSE;//注意:必须是static,否则每次调用OnBtAdd函数时都会被重新定义,实现不了所需功能;而对于static,第一次加载时,就会为其分配内存空间并初始化为FALSE,以后再进入OnBtnAdd时,就不会再为这个静态分配内存空间并初始化了!

//关于static更详细说明 可参考http://www.cnblogs.com/skyseraph/archive/2010/10/11/1848115.html

if(bIsCreated==FALSE) //确保m_bt没有与某窗口绑定了

{

m_bt.Create("New",BS_DEFPUSHBUTTON | WS_CHILD | WS_VISIBLE,CRect(0,0,100,200),this,123);//若无WS_VISIBLE,则要加ShowWindow

bIsCreated=TRUE;

}

else

{

m_bt.DestroyWindow();

bIsCreated=FALSE;

}*/

//▲法三:由于CWnd类有一个成员变量m_hWnd用来保存与窗口对象相关联的窗口句柄,如果窗口句柄没有任一窗口相关联,则句柄为NULL

if(!m_bt.m_hWnd)

{

m_bt.Create("New",BS_DEFPUSHBUTTON | WS_CHILD | WS_VISIBLE,CRect(0,0,100,200),this,123);

}

else

{

m_bt.DestroyWindow();

}

}

 

  • /*■4.<<静态文本控件的访问>>*/

//功能:单击一个静态文本(Number1:)时,把文本变成“数值1:”

void CTestDlg::OnNumber1() //静态文本Number1的响应函数

{

//先添加一个对话框及类,并添加菜单来显示(同前)

     //再添加静态文本Number1:,改变其ID(由于静态文本一般只其标签的作用,其ID默认都相同,特殊情况下需要操作它时需改变它)

//注意:静态文本控件默认是不发送通告消息的。需要勾上Notify选项

CString str;

if(GetDlgItem(IDC_NUMBER1)->GetWindowText(str), str == "Number1:") //if语句的一种使用方法

//控件实际上也是窗口,要获取静态文本控件上显示的文本,可以利用CWnd类GetWindowText来实现,之前需由GetDlgItem来获取静态文本框件对象

//HWND GetDlgItem( HWND hDlg,int nIDDlgItem);   //virtual CWnd* GetDlgItem(int nID) const;

{

GetDlgItem(IDC_NUMBER1)->SetWindowText("数值1:");

}

else

{

GetDlgItem(IDC_NUMBER1)->SetWindowText("Number1:");

}

}

 

  • /*■5.<<编辑框控件>>*/

//功能:在前两个编辑框中分别输入数字,然后单击Add按钮,把其和显示在第三个编辑框

void CTestDlg::OnBtnAdd()

{

//▲法一:GetDlgItem->GetWindowText

int num1, num2, num3;

char ch1[10], ch2[10], ch3[10];

 

GetDlgItem(IDC_EDIT1)->GetWindowText(ch1,10);

GetDlgItem(IDC_EDIT2)->GetWindowText(ch2,10);

 

num1 = atoi(ch1);//atoi将数字组成的字符串转换成数值

//int atoi(const char *str);  Convert a string to integer

num2 = atoi(ch2);

num3 = num1 + num2;

 

itoa(num3,ch3,10);//以十进制的形式把数值转换成文本

//char *_itoa(int value,char *str,int radix);

GetDlgItem(IDC_EDIT3)->SetWindowText(ch3);

 

//法二:GetDlgItemText

int num1, num2, num3;

char ch1[10], ch2[10], ch3[10];

 

GetDlgItemText(IDC_EDIT1,ch1,10);//GetDlgItem和GetWindowText的组合

GetDlgItemText(IDC_EDIT2,ch2,10);

//virtual int GetDlgItemText(int nID,LPTSTR lpStr,int nMaxCount) const;

 

num1 = atoi(ch1);

num2 = atoi(ch2);

num3 = num1 + num2;

 

itoa(num3,ch3,10);

SetDlgItemText(IDC_EDIT3,ch3);

 

//法三:GetDlgItemInt

int num1, num2, num3;

num1 = GetDlgItemInt(IDC_EDIT1);//返回指定控件的文本,并将其转换为一个整型数值。可以实现负值的操作

// virtual UINT GetDlgItemInt(int nID,BOOL* lpTrans=NULL,BOOL bSigned=TRUE) const;

 

num2 = GetDlgItemInt(IDC_EDIT2);

num3 = num1 + num2;

SetDlgItemInt(IDC_EDIT3,num3);

 

//▲法四:将控件与整型变量关联

//将三个编辑框分别与对话框类的三个成员变量相关联,然后再通过这些成员变量来检索和设置编辑框的文本

//说明:关联变量成员时,变量类别为value,类型为int[视实际需求而定],可以浏览头文件和源文件查看发生了哪些变化?

//[共三处:头文件中注视宏中添加定义,源文件的构造函数中初始化、DoDataExchange中实现对话框控件与类成员变量的关联,如DDX_Text(pDX,IDC_EDIT1,m_num1) ...] //DDX_为前缀的函数用于不同控件的数据交换

//注意:在程序中从来都不直接调用DoDateExchange函数,而是通过CWnd的另外一个成员函数UpdataData()来调用

UpdateData();//获取数据

//BOOL UpdateData(BOOL bSaveAndValidate = TRUE); 参数为TRUE说明该函数正在获取对话框的数据,为FALSE说明该函数正在初始化对话框控件

//对于模态对话框,它创建时,自动以参数FALSE来调用UpdateData函数来初始化控件

m_num3 = m_num1 + m_num2;

UpdateData(FALSE);//初始化对话框

//还可以通过ClassWizard中的成员变量选项卡中,选择IDC_EDIT1控件,在对话框底部编辑框中的min..何max..分别输入0、100即可设定编辑框输入范围

//此时在DoDataExchange中增加了DDV_MinMaxInt(pDXp,n_num1,0,100);

//DDX:Dialog Data Exchange对话框数据交换      DDV:Dialog Data Validation对话框数据校验

 

 

//▲法五:将控件与控件变量关联

//编辑框与控件变量关联,这些控件变量就是控件本身 ,即变量类别选control,类型为CEdit/派生于CWnd

int num1, num2, num3;

char ch1[10], ch2[10], ch3[10];

 

m_edit1.GetWindowText(ch1,10);

m_edit2.GetWindowText(ch2,10);

 

num1 = atoi(ch1);

num2 = atoi(ch2);

num3 = num1 + num2;

 

itoa(num3,ch3,10);

m_edit3.SetWindowText(ch3);

 

//法六:SendMessage

//机理:windows程序都是基于消息的,故要获取或设置窗口的文本,只要知道获取或设置窗口文本的消息,就可以通过sendmessage来发送,从而获取或设置窗口的文本。windows中,获取窗口文本的消息是WM_GETTEXT,发送该消息后,系统把指定的窗口文本复制到调用者提供的一个缓存中。这个消息的两个参数中,wParam指定将复制的字节数,lParam就是调用者提供的用来保存窗口文本的缓存地址。而设置窗口文本消息是WM_SETTEXT,此消息wParam消息没有被使用,lParam消息用来指定设置窗口文本的字符串地址。

int num1, num2, num3;

char ch1[10], ch2[10], ch3[10];

 

::SendMessage(GetDlgItem(IDC_EDIT1)->m_hWnd, WM_GETTEXT, 10,(LPARAM)ch1);

//Platform SDK和CWnd类都提供SendMessage函数,加上::调用Platform SDK的函数

::SendMessage(m_edit2.m_hWnd, WM_GETTEXT, 10,(LPARAM)ch2);

 

num1 = atoi(ch1);

num2 = atoi(ch2);

num3 = num1 + num2;

 

itoa(num3,ch3,10);

m_edit3.SendMessage(WM_SETTEXT, 0, (LPARAM)ch3);        

 

//法七:SendDlgItemMessage 直接给对话框的子控件发送消息

int num1, num2, num3;

char ch1[10], ch2[10], ch3[10];

 

SendDlgItemMessage(IDC_EDIT1, WM_GETTEXT, 10,(LPARAM)ch1);//相当于组合了GetDlgItem和SendMessage,无需先获得子控件对象再发送消息,因为其本身就是对对话框中给它的子控件发送消息用的

SendDlgItemMessage(IDC_EDIT2, WM_GETTEXT, 10,(LPARAM)ch2);

//LRESULT SendDlgItemMessage(HWND hDlg, int nIDDlgItem, UINT Msg,WPARAM wParam,LPARAM lParam);

 

num1 = atoi(ch1);

num2 = atoi(ch2);

num3 = num1 + num2;

 

itoa(num3,ch3,10);

SendDlgItemMessage(IDC_EDIT3, WM_SETTEXT, 0, (LPARAM)ch3);

 

SendDlgItemMessage(IDC_EDIT3, EM_SETSEL,1,3); //设置/获得[EM_GETSEL]编辑框复选的内容,wParam表示开始位置,lParam表示结束位置

//特殊情况:当wParam为0。lParam为-1时,那么编辑框中所有内容都将被复选,此特例是非常有用的

//EM_开头的消息是指编辑框控件消息(Edit Control Message)

m_edit3.SetFocus();//为实现复选,需将当前焦点[单击后,在按钮上]转移到编辑框上

}

  • /*■6.<<对话框伸缩功能的实现>>*/

为对话框添加一个按钮,属性设为“收缩<<”,并为其添加消息响应函数

void CTestDlg::OnButton1()

{

// TODO: Add your control notification handler code here

//①按钮文本变换功能

CString str;

if(GetDlgItemText(IDC_BUTTON1, str), str == "收缩<<")

{

SetDlgItemText(IDC_BUTTON1,"扩展>>");

}

else

{

SetDlgItemText(IDC_BUTTON1,"收缩<<");

}

 

//②切割的实现:先放置分隔条,用图片控件实现,ID为IDC_SEPERATOR,选用Sunken风格(下陷状态),去掉visibale,不可见

static CRect rectLarge;//第一次获得该变量后,以后不需要再次改变,故设为static

static CRect rectSmall;

 

//判断矩形为空:CRect有两个成员函数,IsRectNull和IsRectEmpty,两者不同,参考MSDN

if(rectLarge.IsRectNull())

{

CRect rectSeparator;

GetWindowRect(&rectLarge);//获取原始尺寸

GetDlgItem(IDC_SEPARATOR)->GetWindowRect(&rectSeparator);

 

rectSmall.left=rectLarge.left;

rectSmall.top=rectLarge.top;

rectSmall.right=rectLarge.right;

rectSmall.bottom=rectSeparator.bottom;

}

if(str=="收缩<<")

{

SetWindowPos(NULL,0,0,rectSmall.Width(),rectSmall.Height(),

SWP_NOMOVE | SWP_NOZORDER);

//virtual BOOL SetWindowPos(const CWnd* pWndInsertAfter,int x,int y,int cx,int cy,UINT nFlags); 

//Sets the size, position, and Z order of the control site.

//窗口Z次序:表明重叠堆中窗口的位置

}

else

{

SetWindowPos(NULL,0,0,rectLarge.Width(),rectLarge.Height(),

SWP_NOMOVE | SWP_NOZORDER);

}

}

总结:①实现按钮变化功能  ②实现窗口分割

 

Author:         SKySeraph

Email/GTalk: zgzhaobo@gmail.com    QQ:452728574

From:         http://www.cnblogs.com/skyseraph/

本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,请尊重作者的劳动成果。

posted @ 2010-10-31 21:34  SkySeraph  阅读(2012)  评论(0编辑  收藏  举报