第一讲 用Create()方法新建一个窗体应用程序
一般来讲,大多数Windows应用程序的界面都是由一个或数个窗体构成。而VC++中提供了丰富的类库,用于创建Windows窗体应用程序。 我们一般可以通过CFreameWnd类中的Create()方法来创建一个窗体,Create()函数的定义如下:
BOOL Create(LPCTSTR lpszClassName, LPCTSTR lpszWindowName, DWORD dwStyle = WS_OVERLAPPEDWINDOW, const RECT& rect = rectDefault, CWnd* pParentWnd = NULL, // != NULL for popups LPCTSTR lpszMenuName = NULL, DWORD dwExStyle = 0, CCreateContext* pContext = NULL);
这是一个虚函数①,第一个参数lpszClassName是一个窗体类名字符串的指针(一个WNDCLASS②结构体)。此类名可以是任意的由全局函数AfxRegisterWndClass注册过的预定义控件类名。如果为空,则使用CWnd类的默认属性。第二个参数lpszWindowName是作为窗体标题的字符串指针。 第三个参数dwStyle是宏定义的窗体类型,具体定义如下:
WS_BORDER 创建一个有边框的窗体。
WS_CAPTION 创建一个有标题栏的窗体(隐含了WS_BORDER). 不能和WS_DLGFRAME 一起使用.
WS_CHILD 创建一个子窗体。不能和WS_POPUP一起使用。
WS_CLIPCHILDREN 不包括在父窗体中被子窗体占用的区域。用于创建父窗体。
WS_CLIPSIBLINGS 使子窗体彼此别住;就是当一个指定的子窗体接收到一个 paint消息时,WS_CLIPSIBLINGS类型将别住所有重叠的子窗 体超过区域的部分一起更新,(如果没有使用WS_CLIPSIBLINGS 并且子窗体重叠,当你在一个子窗体的客户区绘图时,可能 会绘图到邻近的子窗体的客户区。)只与WS_CHILD一起使用。
WS_DISABLED 创建一个初始不可用的窗体。
WS_DLGFRAME 创建一个有双边但无标题的窗体。
WS_GROUP 指定一个用户可以用方向键从一个控件移到另一个控件的控 件组的第一个控件。All controls defined with the WS_GROUP style FALSE after the first control belong to the same group. The next control with the WS_GROUP style starts the next group (that is, one group ends where the next begins).
WS_HSCROLL 创建一个带水平滚动条的窗体。
WS_MAXIMIZE 创建一个最大尺寸的窗体。
WS_MAXIMIZEBOX 创建一个有最大化按扭的窗体。
WS_MINIMIZE 创建一个初始最小化的窗体。只与WS_OVERLAPPED一起使用。
WS_MINIMIZEBOX 创建一个有最小化按扭的窗体。
WS_OVERLAPPED 创建一个重叠窗体。一个重叠窗体一般有标题和边框。
WS_OVERLAPPEDWINDOW 创建一个和WS_OVERLAPPED, WS_CAPTION, WS_SYSMENU, WS_THICKFRAME, WS_MINIMIZEBOX, and WS_MAXIMIZEBOX一 使用的重叠窗体。
WS_POPUP 创建一个弹出式窗体。不能和WS_CHILD一起使用。
WS_POPUPWINDOW 创建一个和WS_BORDER, WS_POPUP, and WS_SYSMENU一起使 用的弹出式窗体。WS_CAPTION必须和WS_POPUPWINDOW组合使 用才能让控件菜单可见。
WS_SYSMENU 创建一个在标题栏有控件菜单框的窗体。只能和有标题栏的 窗体一起使用。
WS_TABSTOP 指定任意数量控件中的一个可以由用户使用TAB键移动到的 控件。TAB键使用户移动到由WS_TABSTOP指定的下一个控件。
WS_THICKFRAME 创建一个有厚边框的Window,使其可以改变大小。
WS_VISIBLE 创建一个初始可见的窗体。
WS_VSCROLL 创建一个有垂直滚动条的窗体。
由于上表中常量可以进行组合,用按位或运算,所以常量名的值被定义为类似于0x00C00000L的32位16进制数型式。例如进行WS_SYSMENU|WS_MINIMIZEBOX运算时即0x00080000L|0x00020000L。数字前面的0x是16位数的标识符,L表示32位,上式的演算式可表示如下:
= 00000000000010000000000000000000 (|) 00000000000000100000000000000000------------------------------------------ 00000000000010100000000000000000 = 0X000A0000L = 655360;
当你用十进制数655360替代WS_SYSMENU|WS_MINIMIZEBOX作为实参时,你会看到相同的结果。
第四个参数rect是一个RECT结构体的对象,用于指定窗体的尺寸和位置。RECT结构体的定义如下:
typedef struct tagRECT{ LONG left; LONG top; LONG right; LONG bottom;} RECT, *PRECT, NEAR *NPRECT, FAR *LPRECT;
我们可以利用从tagRECT结构体派生出的CRect类的构造函数来初始化一个RECT结构体。CRect的一个重载构造函数定义如下:
// from left, top, right, and bottomCRect(int l, int t, int r, int b);
第五个参数pParentWnd用于指定父窗体,这是一个指向CWnd类对象的指针。第六个参数nID用于指定作为子窗体的窗体ID。最后一个参数是关于创建内容的指针,已被默认为NULL,不用理会。
注释:
① 虚函数的作用:如果没有把需要在派生类中重载的同名基类函数定义为虚函数,则当用基类定义的指针指向派生类对象的地址时(赋值兼容规则),通过此指针调用的该同名函数是在基类中定义的;反之如果定义的虚函数,则通过指针调用的该同名函数是在指针指向的对象中定义的。
第二讲 Create()方法的应用
下面是一个用CFrameWnd类的Create()函数创建一个窗体的例子:
/#include
class CMyWnd:public CFrameWnd{public: CMyWnd() { Create(AfxRegisterWndClass(CS_DBLCLKS,0,HBRUSH(COLOR_WINDOWFRAME),AfxGetApp()->LoadStandardIcon(IDI_APPLICATION)),__T("Creamdog"),WS_SYSMENU,CRect(100, 100, 500, 500),this,NULL); ShowWindow(SW_SHOWNORMAL); };};
class CMyApp:public CWinApp{public: virtual BOOL InitInstance() { m_pMainWnd=new CMyWnd; return TRUE; };};
CMyApp myApp;/
首先从其类CFrame中派生出CMyWnd类,并定义构造函数,用于生成一个新窗体。在构造函数中是使用Creat()函数创建窗体的,其中系统全局函数AfxRegisterWndClass()用于注册一个窗体类,该函数的具体定义如下:
LPCTSTR AFXAPI AfxRegisterWndClass(UINT nClassStyle, HCURSOR hCursor = 0, HBRUSH hbrBackground = 0, HICON hIcon = 0);
第一个参数nClassStyle指定了窗体类的类型,如果为NULL则所有参数都使用默认值。具体默认值为:
nClassStyle=CS_DBLCLKS;//响应双击事件hCursor=IDC_ARROW;//标准箭头hbrBackground=Null;//不更新背景hIcon=IDI_APPLICATION;//Windows徽标(在WinXP中为小窗口)
由于用默认值时窗体不会更新,一般函数的四个参数需要人为指定,第一个参数设为为CS_DBLCLKS即可。CS_DBLCLKS是窗体类型的一个宏定义,下表列出了所有窗体类型的宏定义。
CS_BYTEALIGNCLIENT 在字节边界(在X方向)上对齐窗体的客户区。此类型将影 响到在窗体显示时它的宽度和它的水平位置。
CS_BYTEALIGNWINDOW 在字节边界(在X方向)上对齐窗体。此类型将影响到在窗 体显示时它的宽度和它的水平位置。
CS_CLASSDC 分配一个设备环境并被类中的所有窗体共享。由于窗体类 被处理特化,它是可以适用于一个应用程序的若干线程创 建一个相同类的窗体。它同样适用于多个线程试图同时使 用相同的设备环境。当此种情况发生时,系统只允许一个 线程去成功的它的绘图操作。
CS_DBLCLKS 当指针在属于此类的窗体内部,并且用户双击鼠标时,将 会发送一个双击消息到窗体程序。
CS_GLOBALCLASS 指定此窗体类是一个应用程序全局类。应用程序全局类是 由一个在进程中对所有模块有效的exe或ddl注册的一个窗 体类。
CS_HREDRAW 如果窗体被移动或尺寸调整器改变了客户区的大小,重绘 全部的窗体。
CS_NOCLOSE 关闭按扭不可用。
CS_OWNDC 为此类中的每一个窗体分配唯一的设备环境。
CS_PARENTDC 设置子窗体中剪下的矩形到父窗体中,以使子窗全可以在父 窗体上绘图。一个具有CS_PARENTDC属性控制的窗体从设备 环境的系统缓存中接收到一个规则的设备环境。它不把父窗 体的设备环境或设备环境设置给予子窗体。指定CS_PARENTDC 以提高应用程序的性能。
CS_SAVEBITS 保存被此类的一个窗体摭住的屏幕图象的一部分为位图。当 窗体被移动,系统使用保存过的位图去恢复屏幕图象,包括 其它被摭住的窗体。因此如果被位图使用内存没有被释放, 并且其它的屏幕动作没有使储存的图像无效。系统不会发送 WM_PAINT消息到被摭盖的窗体。 这种类型对在其它屏幕动作发生时被暂时显示小窗体(如菜 单或对话框)很有用。这种类型增加了显示窗体所需的时间, 因为系统必须先分配内存去存储位图。 CS_VREDRAW 如果窗体被移动或尺寸调整器改变了客户区的高度,重绘 全部的窗体。
上表中的宏定义值是类似于0x0080的16位16进制数,因此它们之间可以用按位或 | 符号进行组合,原理前面以经介绍过了。
第二个参数hCursor为鼠标指针的句柄,但由于在窗体打开事件发生时,光标就会被重绘为箭头,因此在注册窗体类时对此进行设置意义不大,设为0即可。如果需要定义光标,首先应关闭数标重绘,方法是:重载窗体类基类的OnSetCursor()函数,让其返回TRUE值,这样当重绘时调用OnSetCursor()函就不起作用了。可以用下面的语句进行设置。
SetCursor(AfxGetMainWnd()->LoadStandardCursor(IDC_IBEAM));
其中::SetCursor()是全局函数,用来设置整个例程的光标参数是宏定义光标句柄。AfxGetMainWnd()是一个系统函数,它返回当前主窗体的句柄。而CWinApp的LoadStandardCursor()成员函数用来读取一个系统指针,每一种系统指针的具体宏定义如下:
IDC_APPSTARTING 带小沙漏的标准箭头IDC_ARROW 标准箭头IDC_CROSS 十字光标(用于定位)IDC_HAND Windows 2000:手型IDC_HELP 带问号的箭头IDC_IBEAM I型标IDC_ICON Obsolete for applications marked version 4.0 or later. IDC_NO 禁止符号IDC_SIZE Obsolete for applications marked version 4.0 or later. Use IDC_SIZEALL. IDC_SIZEALL 十字箭头IDC_SIZENESW 指向东北和西南的双向箭头IDC_SIZENS 指向南和北的双向箭头IDC_SIZENWSE 指向西北和东南的双向箭头IDC_SIZEWE 指向东西的双向箭头IDC_UPARROW 上箭头IDC_WAIT 沙漏
上表中宏定义值为类似于MAKEINTRESOURCE(32649)的函数,MAKEINTRESOURCE()的定义如下:
#define MAKEINTRESOURCEW(i) (LPWSTR)((DWORD)((WORD)(i))) //UNICODE#define MAKEINTRESOURCE MAKEINTRESOURCEW
将其还原为容易理解的C代码:
char *(unsigned long(unsigned short(32649)) 至于为什么系统要将其倒来倒去,是为了在重载函数中和一般的整形实参相区别(本人估计)。另外,如果想要使用自定义图标或指针文件,则牵扯到实例资源分配的问题,将在以后进行说明。
第三个参数hbrBackground,指定窗体背景的画笔资源,应该指定一个,否则背景将不会更新。此参数可使用系统颜色,定义如下:
COLOR_SCROLLBAR 0COLOR_BACKGROUND 1COLOR_ACTIVECAPTION 2COLOR_INACTIVECAPTION 3COLOR_MENU 4COLOR_WINDOW 5COLOR_WINDOWFRAME 6COLOR_MENUTEXT 7COLOR_WINDOWTEXT 8COLOR_CAPTIONTEXT 9COLOR_ACTIVEBORDER 10COLOR_INACTIVEBORDER 11COLOR_APPWORKSPACE 12COLOR_HIGHLIGHT 13COLOR_HIGHLIGHTTEXT 14COLOR_BTNFACE 15COLOR_BTNSHADOW 16COLOR_GRAYTEXT 17COLOR_BTNTEXT 18COLOR_INACTIVECAPTIONTEXT 19COLOR_BTNHIGHLIGHT 20
第四个参数为程序图标的标识符,为0时是默认的Windows徽标。与光标同样,需要自定义图标时,添加设置图标的语句,例如:
AfxGetMainWnd()->SetIcon(AfxGetApp()->LoadStandardIcon(IDI_EXCLAMATION),FALSE);
与设置光标不同的是,设置光标的函数是全局函数,而设置图标的函数是CWinApp类的成员函数(因为图标只在窗体内有效),故在函数调用之前需要用系统函数AfxGetMainWnd()来获取当前主窗体的句柄,再用CWinApp的LoadStandardIcon()成员函数来读取系统图标并返回一个图标的句柄,最后CFrameWnd类的SetIcon()成员函数将窗体的图标设置为刚才返回的图标句柄。
回到刚才的Create()语句,第一个参数用的是刚才注册的类名。第二个参数中用到了强制类型转换(__T),这个数据类型不对字符串做任何的改变,只是起到规范化编程的做用。第三个参数是前面提过的窗体类型,WS_SYSMENU是使窗体具有最大化、最小化、关闭三个按扭。第四个参数使窗体在所给出的位置和尺寸上打开。第五个参数this的意义是此窗体为父窗体。第六个参数表示无子窗体。 ShowWindow()故名思意,即显示窗体。其参数nCmdShow是确定窗体被怎样显示。它必须是下面宏定义中的一个:
SW_HIDE 0 隐藏此窗体并激活其它窗体。
SW_MAXIMIZE 3 激活并显示此窗体为最大化。
SW_MINIMIZE 6 最小化指定的窗体并激活下一个在Z顺序中位 于顶层的窗体。
SW_RESTORE 9 激活并显示此窗体。如果此窗体已被最大化或 最小化,系统将恢复此窗体至原尺寸和原位置。 一个应用程序应该在恢复一个最小化窗体时指 定这个标记。
SW_SHOW 5 在当前尺寸和位置上激活并显示此窗体。
SW_SHOWMAXIMIZED 3 同SW_MAXIMIZE
SW_SHOWMINIMIZED 2 激活并显示此窗体为最小化。
SW_SHOWMINNOACTIVE 7 显示此窗体为最小化。这个值类似于SW_SHOWMINIMIZED, 除非窗体未被激活。SW_SHOWNA 8 在最近一次的尺寸和位置显示此窗体,这个值 类似于SW_SHOWNORMAL,除非窗体未被激活。
SW_SHOWNOACTIVATE 4 在当前的尺寸和位置显示此窗体。这个值类似于SW_SHOW, 除非窗体未被激活。
SW_SHOWNORMAL 1 激活并显示一个窗体。如果此窗体已被最大化 或最小化,系统将恢复此窗体至原尺寸和原位置。 一个应用程序应该在第一次打开这个窗体时指定 这个标记。
有关窗体框架的定义就结束了,下面是由CWinApp基类派生出的类CMyApp,在类定义中对基类中的虚成员函数InitInstance()进行了重载,关于虚函数的有关概念前面以经提过了。在开始说明此函数内部的语句时首先需要对CWinApp的数据成员m_pMainWnd进行说明。 m_pMainWnd数据成员被用于储存一个指向你的线程中主窗体对向的指针。当涉及到m_pMainWnd的窗体被关闭时,MFC库将自动的终止你的线程。如果此线程为你的应用程序中的主线程,应用程序也将同样被终止。如果此数据成员为NULL,当终止此线程时,为了应用程序的CWinApp对象的主窗体也将常常被终止。m_pMainWnd是CWnd类指针的一个公有变量。 一般来讲,当你重载InitInstance()函数时设置此变量。在一个工作线程中,此数据成员的值是从它的父线程中继承来的。 正如上面所说,例子中在重载InitInstance()函数时对此变量进行了赋值,让它指向一个新的框架窗体类对象,而这个对象是由刚才定义的派生类CMyWnd实例化出来的。下来返回一个真值,表示初始化实例成功。 最后,用CMyApp类实例化出一个对象,名子随意。系统将对前面定义的类和函进行构造,这样就完成了一个简单窗体的创建。