收藏 分享(赏)

第6章-文档与视图.ppt

上传人:天天快乐 文档编号:968731 上传时间:2018-05-09 格式:PPT 页数:84 大小:1.16MB
下载 相关 举报
第6章-文档与视图.ppt_第1页
第1页 / 共84页
第6章-文档与视图.ppt_第2页
第2页 / 共84页
第6章-文档与视图.ppt_第3页
第3页 / 共84页
第6章-文档与视图.ppt_第4页
第4页 / 共84页
第6章-文档与视图.ppt_第5页
第5页 / 共84页
点击查看更多>>
资源描述

1、第6章,Visual C+面向对象编程 Visual C+ Object Oriented Programming,第6章 文档与视图,第6章 文档与视图,内容提要,文档与视图结构原理 菜单编程 鼠标消息处理 工具栏和状态栏编程 读写操作机制,通过本章的学习,可以利用 Visual C+创建一个完整的文档/视图结构应用程序。,文档与视图结构是MFC应用程序最基本的程序结构,适用于大多数Windows应用程序。文档和视图完成了程序的大部分功能,它们是MFC应用程序的核心。文档与视图结构是MFC的基石,掌握文档与视图结构对于利用MFC编程有着至关重要的意义。本章对文档与视图结构进行更深入的讨论。,

2、信息管理是计算机的一个主要应用,而信息是用数据表示的,因此数据的处理是一般软件都要完成的一项主要工作。 采用传统的编程方法,数据处理是一项复杂的任务,并且每一个程序员都可能有不同的处理方法。为了统一和简化数据处理方法,Microsoft公司在MFC中提出了文档/视图结构的概念,其产品Word就是典型的文档/视图结构应用程序。,6.1 文档与视图结构,标题栏主菜单工具栏客户区状态栏不同程序的相同菜单项和工具栏按钮表示相同的操作,6.1.1 文档与视图结构概述,Windows应用程序界面特点:,分为数据的管理和显示 文档用于管理和维护数据 视图用来显示和编辑数据 MFC通过其文档类和视图类提供了大

3、量有关数据处理的方法。,MFC文档/视图结构数据处理工作分工:,文档的概念在MFC应用程序中的适用范围很广,一般说来,文档是能够被逻辑地组合的一系列数据,包括文本、图形、图象和表格数据。 一个文档代表了用户存储或打开的一个文件单位。文档的主要作用是把对数据的处理从对用户界面的处理中分离出来,集中处理数据,同时提供了一个与其它类交互的接口。,什么是文档?,视图是文档在屏幕上的一个映像,它就像一个观景器,用户通过视图看到文档,也是通过视图来改变文档,视图充当了文档与用户之间的媒介物。 应用程序通过视图向用户显示文档中的数据,并把用户的输入解释为对文档的操作。 一个视图总是与一个文档对象相关联,用户

4、通过与文档相关联的视图与文档进行交互。当用户打开一个文档时,应用程序就会创建一个与之相关联的视图。,什么是视图?,视图负责显示和编辑文档数据,但不负责存储。用户对数据的编辑需要依靠窗口上的鼠标与键盘操作才得以完成,这些消息都是由视图类接收后进行处理或通知文档类,如收到窗口刷新消息时调用视图类的成员函数OnDraw()显示文档内容。视图还可在打印机上输出。文档负责数据的读写操作,数据通常被保存在文档类的成员变量中,文档类通过一个称为序列化的成员函数将成员变量的数据保存到磁盘文件中。MFC应用程序为数据的序列化提供了默认支持。,视图和文档的功能:,文档、视图、框架窗口之间的关系,一个视图是一个没有

5、边框的窗口,它位于主框架窗口中的客户区。视图是文档对外显示的窗口,但它并不能完全独立,它必须依存在一个框架窗口内。一个视图只能拥有一个文档,但一个文档可以同时拥有多个视图。,视图是文档在屏幕上的一个映像,它就像一个观景器,文档/视图结构的优点:,把数据处理类从用户界面处理类中分离出来,使得每一个类都能集中地执行一项工作。把Windows程序通常要做的工作分成若干定义好的类,这样有助于应用程序的模块化,程序也易于扩展,编程时只需修改所涉及的类。虽然文档/视图结构牵涉到许多类,其中的也关系比较复杂,但MFC AppWizard向导建立的MFC应用程序框架已经把程序的主要结构完成了,模块间的消息传递

6、以及各函数的功能都已确定。MFC应用程序框架起到了穿针引线的作用,按照消息处理函数功能的不同,将不同消息的响应分别分布在文档类和视图类中。,文档/视图结构并没有完全要求所有数据都属于文档类,视图类也可以有自己的数据。如果在视图类中不定义任何数据,在需要时都从文档类中获取,这样做会影响程序的效率。 例如,在文本编辑程序中,往往在视图中缓存部分数据,这样可以避免对文档的频繁访问,提高运行效率。,在视图类中定义数据,包含多个类的MFC文档/视图结构应用程序要管理这些类中的数据,除了考虑在程序的哪一部分拥有数据和在哪一部分显示数据,一个主要的问题是文档数据更改后如何保持视图显示的同步,即文档与视图如何

7、进行交互。在文档、视图和应用程序框架之间包含了一系列复杂的相互作用过程,文档与视图的交互是通过类的公有成员变量和成员函数实现的。,6.1.2 文档与视图之间的相互作用,1视图类的成员函数GetDocument() 一个视图对象只有一个与之相关联的文档对象。在MFC应用程序中,视图对象通过调用成员函数函数GetDocument()得到当前文档。GetDocument()是视图类的成员函数,调用它可以返回与视图相关联的文档对象的指针,利用这个指针可以访问文档类及其派生类的公有成员。 当利用MFC AppWizard向导创建一个SDI单文档应用程序Mysdi时,生成了视图类的一个派生类,并在派生类中

8、定义了函数GetDocument()。,文档和视图类常用的成员函数,CMysdiDoc* CMysdiView:GetDocument() ASSERT(m_pDocument-IsKindOf(RUNTIME_CLASS(CMysdiDoc);return (CMysdiDoc*)m_pDocument; / m_pDocument是CArchive类的数据成员,/ 指向当前文档对象,GetDocument()的Debug版函数代码:,一个文档对象可以有多个与之相关联的视图对象,当一个文档的数据通过某个视图被修改后,与它关联的每一个视图都必须反映出这些修改。因此,视图在需要时必须进行重绘,即

9、当文档数据发生改变时,必须通知到所有相关联的视图对象,以便更新所显示的数据。 更新与该文档有关的所有视图的方法是调用成员函数CDocument:UpdateAllViews()。,2CDocument类的成员函数UpdateAllViews(),如果在文档派生类的成员函数中调用UpdateAllViews()函数,其第一个参数pSender设为NULL,表示所有与当前文档相关的视图都要重绘(参见例6-3)。 如果在视图派生类的成员函数中通过当前文档指针调用UpdateAllViews()函数,其第一个参数pSender设为当前视图,如下形式: GetDocument()-UpdateAllVi

10、ews(this),函数声明:void UpdateAllViews(CView* pSender, LPARAM lHint = 0L, CObject* pHint=NULL );,当程序调用CDocument:UpdateAllViews()函数时,实际上是调用了所有相关视图的OnUpdate()函数,以更新相关的视图。需要时,可以直接在视图派生类的成员函数中调用该函数刷新当前视图。,3视图类的成员函数OnUpdate(),void CView:OnUpdate(CView* pSender, LPARAM /*lHint*/, CObject* /*pHint*/)ASSERT(pSe

11、nder != this);UNUSED(pSender); / unused in release builds/ invalidate the entire pane, erase background tooInvalidate(TRUE); / 使整个窗口矩形无效,通过调 / 用OnDraw()更新整个视图窗口,基类CView的成员函数,在OnUpdate()中通过调用函数CWnd:Invalidate()刷新整个客户区,我们也可以在自己的CWnd派生类中直接调用函数Invalidate()。,总结:刷新视图时默认的函数调用过程: CDocument:UpdateAllViews()

12、CView:OnUpdate() CWnd:Invalidate() OnPaint() OnDraw(),MFC基于文档/视图结构的应用程序分为单文档和多文档两种类型,一个多文档应用程序有一个主窗口,但在主窗口中可以同时打开多个子窗口,每一个子窗口对应一个不同的文档。利用MFC AppWizardexe向导可以很方便地建立一个多文档应用程序,只需在MFC AppWizard向导第1步选择Multiple documents程序类型。SDI和MDI使用不同框架窗口。SDI的框架窗口是唯一的主框架窗口,窗口类是CMainFrame,由CFrameWnd派生而来。,6.1.3 多文档,MDI的框架

13、窗口分为主框架窗口和子框架窗口,区别于SDI,MDI的主框架窗口不包含视图,分别由每个子框架窗口包含一个视图。MDI的主框架窗口类不与某个打开的文档相关联,而只与子框架窗口相关联。MDI主框架窗口类CMainFrame由CMDIFrameWnd派生而来,而MDI子框架窗口类CChildFrame由CMDIChildWnd派生而来。,在文档/视图结构中,数据以文档类对象的形式存在。文档对象通过视图对象显示,而视图对象又是主框架窗口的一个子窗口,并且涉及文档操作的菜单和工具栏等资源也是建立在主框架窗口上。这样,文档、视图、框架类和所涉及的资源形成了一种固定的联系,这种固定的联系就称为文档模板。也就

14、是说,文档模板描述了相对应每一种类型文档的视图和窗口的风格类型。当打开某种类型的文件时,应用程序必须确定那一种文档模板用于解释这种文件。在初始化程序时,必须首先注册文档模板,以便程序利用这个模板来完成主框架窗口、视图、文档对象的创建和资源的装入。,文档模板的概念:,标准Windows应用程序界面窗口组成: 客户区 非客户区:,6.2 菜单设计,窗口的边框 标题栏 菜单栏 工具栏 状态栏 滚动条 菜单、工具栏、状态栏是用户与应用程序进行交互的重要工具。菜单和工具栏为应用程序提供了传递用户命令的选择区域,而状态栏是提示信息的输出区域。,6.2.1 建立菜单资源,使用MFC AppWizard向导创

15、建文档/视图结构应用程序时,向导将自动生成Windows标准的菜单资源和命令处理函数。但这个默认生成的主框架菜单资源往往不能满足实际的需要,因此我们需要利用菜单资源编辑器对其进行修改和添加。,例 编写一个单文档应用程序DrawCoin,为程序添加一个“画硬币”主菜单,并在其中添加“增加硬币”和“减少硬币”两个菜单项。,1利用MFC AppWizardexe向导创建SDI应用程序。在项目工作区的ResourceView页面中选择Menu并展开它,双击下面的IDR_MAINFRAME项弹出菜单资源编辑器,显示应用程序向导所创建的菜单资源。 2为程序添加主菜单。双击菜单栏右边虚空白框,弹出属性对话框

16、,在Caption框输入主菜单的标题“画硬币(&C)”,字符&用于在显示C时加上下划线,并表示其快捷键为Alt+C。 3在主菜单“画硬币(C)”下方双击带虚框的空白菜单项,弹出属性对话框。在ID栏输入ID_COIN_ADD。在Caption框输入菜单项的标题“增加硬币(&A)tCtrl+A”。,6.2.2 添加菜单命令处理函数,菜单实际上是一系列命令的列表,当一个菜单项被选中后,一个含有该菜单项ID标识的WM_COMMAND命令消息将发送到应用程序窗口,应用程序将该消息转换为一个函数(命令消息处理函数)调用。命令消息来自于用户界面对象,是由菜单项、工具栏按钮和快捷键等程序界面元素发送的WM_C

17、OMMAND消息。 在MFC应用程序中,许多类都能接收菜单被选中而引发的消息。总的来说,从类CCmdTarget派生出来的类都可以加入应用程序的消息循环。,应该将菜单命令映射到哪个类中,需要由该命令的功能决定。如果一个命令同视图的显示有关,就应该将其映射到视图类;如果同文档的读写有关,就映射到文档类中;如果命令完成通用功能,一般映射到窗口框架类。有时无法对功能进行准确分类,则可以将菜单命令映射到任意一个类,看看是否能够完成指定的功能。,将菜单命令映射到哪个类?,利用ClassWizard类向导添加菜单命令WM_ COMMAND消息处理函数后,向导将自动添加一个如下格式消息映射: ON_COMM

18、AND(MenuItemID, MemberFuntion) 其中参数MenuItemID是菜单项的ID标识,参数MemberFuntion是处理该消息的成员函数名。一个菜单项的WM_COMMAND消息意味着选择了该菜单项,或选择了对应的工具栏按钮、键盘快捷键。向导还生成了消息处理函数的框架代码。 例:,利用ClassWizard添加菜单命令处理函数,例 为程序DrawCoin添加菜单命令处理函数。,1. 启动ClassWizard类向导,在Class Name栏选择类CDrawCoinDoc,在Object IDs栏选择ID_COIN_ADD,在Messages栏选择COMMAND,单击Ad

19、d Funtion按钮。同样为ID_COIN_SUB添加命令处理函数。void CDrawCoinDoc:OnCoinAdd() m_nCoins+; / 硬币数量加一UpdateAllViews(NULL); / 刷新视图 void CDrawCoinDoc:OnCoinSub() if(m_nCoins0) m_nCoins-; / 硬币数量减一UpdateAllViews(NULL);,定义成员变量m_nCoins并初始化:,2. 为文档派生类CDrawCoinDoc添加一个类型为int、属性为public的成员变量m_nCoins。 按下Ctrl+W启动ClassWizard类向导,在

20、Class Name栏和Object IDs栏选择类CDrawCoinDoc,在Messages栏选择DeleteContents,单击Add Funtion按钮和Edit Code按钮,在生成的虚函数中添加初始化成员变量m_nCoins的代码,该函数在用户重新使用(打开或创建)一个文档时调用。,void CDrawCoinDoc:DeleteContents() / TODO: Add your . . . m_nCoins=1; / 初始化成员变量CDocument:DeleteContents();,在客户区画出硬币:,3. 在OnDraw()函数中根据m_nCoins画出指定个数的硬币

21、。void CDrawCoinView:OnDraw(CDC* pDC)CDrawCoinDoc* pDoc = GetDocument();ASSERT_VALID(pDoc);/ TODO: add draw code for native data herefor(int i=0; im_nCoins; i+)int y=200-10*i;pDC-Ellipse(200,y,300,y-30); / 用两个偏移的椭圆表示一枚硬币pDC-Ellipse(200,y-10,300,y-35);,利用ClassWizard类向导为菜单项添加命令处理函数时,在Messages栏除了WM_COMM

22、AND消息,还有一个UPDATE_COMMAND_UI消息,它称为更新命令用户界面消息。有时一个菜单项可以有可用和不可用两种状态,即允许或禁止菜单项的使用(处于灰色状态)。例如,初始状态下,菜单项“减少硬币”不可用,因为开始时客户区一个硬币也没有画出。UPDATE_COMMAND_UI消息为程序员根据程序当前运行情况对菜单项的状态进行动态设置而提供了一个简便的方法。参阅例6-4。,更新命令用户界面消息,菜单分为两类: 依附于框架窗口的固定菜单 浮动的弹出式菜单,快捷菜单,上下文菜单 当用户单击鼠标右键,弹出式菜单出现在光标所在位置。弹出式菜单是通过利用CMenu类和其成员函数,在程序运行过程中

23、动态建立的。 一般弹出式菜单是利用现有的菜单项来进行创建,也可以为弹出式菜单专门建立一个菜单资源,然后通过调用函数CMenu:LoadMenu()装入所创建的菜单资源。,6.2.3 弹出式菜单,当右击鼠标并释放后,WM_CONTEXTMENU消息将发给应用程序。因此,在程序中可通过为WM_CON TEXTMENU添加消息处理函数来实现弹出式菜单。WM_CONTEXTMENU消息是在收到WM_RBUTTON UP消息后,由Windows产生的。但如果在WM_RBUT TONUP的消息处理函数中没有调用基类的处理函数,那么应用程序将不会收到WM_CONTEXTMENU消息。 例:,有关弹出式菜单的

24、消息处理,利用ClassWizard为视图类添加WM_CONTEXTMENU的消息处理函数,添加如下代码:void CDrawCoinView:OnContextMenu(CWnd* pWnd, CPoint point) CMenu menuPopup; / 声明菜单对象 if (menuPopup.CreatePopupMenu() / 创建弹出式菜单 menuPopup.AppendMenu(MF_STRING, ID_COIN_ADD, 增加硬币tCtrl+A);menuPopup.AppendMenu(MF_STRING, ID_COIN_SUB, 减少硬币tCtrl+B);menu

25、Popup.TrackPopupMenu(TPM_LEFTALIGN,point.x, point.y, this); ,例 为程序DrawCoin添加弹出式菜单。,函数TrackPopupMenu()用于在指定位置显示弹出式菜单,并响应用户的菜单项选择。函数第1个参数是位置标记,TPM_LEFTALIGN表示以x坐标为标准左对齐显示菜单;第2、3个参数指定弹出式菜单的屏幕坐标;第4个参数指定拥有此弹出式菜单的窗口。,函数AppendMenu()用于向菜单menuPopup添加菜单项,函数第1个参数指定加入菜单项的风格,值MF_ STRING表示菜单项是一个字符串;第2个参数指定要加入菜单项的

26、ID,如ID_COIN_ADD;第3个参数指定菜单项的显示文本。,Windows是基于事件驱动、消息传递的操作系统。用户所有的输入都是以事件或消息的形式传递给应用程序的,鼠标也不例外。鼠标驱动程序将鼠标硬件信号转换成Windows可以识别的信息,Windows根据这些信息构造鼠标消息,并将它们发送到应用程序的消息队列中。,6.3 鼠标消息处理,鼠标构成:左键、右键(中键和滚动滑轮)鼠标操作:单击、双击、释放和移动 主要鼠标消息: WM_MOUSEMOVE:移动 WM_LBUTTONDOWN:按下左键 WM_LBUTTONUP:释放左键 WM_RBUTTONDOWN:按下右键 WM_RBUTTO

27、NUP:释放右键 WM_LBUTTONDBLCLK:双击左键,6.3.1 鼠标消息,鼠标消息分为两类:,客户区鼠标消息 非客户区鼠标消息:,WM_NCLBUTTONDOWN:按下鼠标左键消 WM_NCRBUTTONDOWN:按下鼠标右键 . . . . . .,非客户区鼠标消息由Windows操作系统处理,应用程序一般不需要处理。客户区鼠标消息发送到应用程序后,可以由应用程序自己处理。 通过消息结构中的消息参数wParam来区分客户区鼠标消息和非客户区鼠标消息。,利用MFC ClassWizard类向导生成的鼠标消息处理函数一般都有两个参数: nFlags:类型为UINT,表示鼠标按键和 键盘

28、上控制键的状态。 point:类型为CPoint,表示鼠标当前所 在位置坐标。,鼠标消息处理函数参数,使用鼠标的一个典型例子就是绘图程序,鼠标被用作画笔,绘图过程中要进行不同鼠标消息的处理,如按下鼠标、释放鼠标和移动鼠标等。当用户按下鼠标左键时必须记录下当前鼠标的位置,当用户移动鼠标时,如果鼠标左键被按住,则从上一个鼠标位置到当前位置画一段直线,并保存当前鼠标的位置,供下一次画线用。当用户弹起鼠标左键时释放鼠标。,6.3.2 鼠标消息处理举例:一个简单的绘图程序,例 编写一个绘图程序,程序运行后,当用户在客户区窗口按下鼠标左键并移动时,根据鼠标移动的轨迹绘制出指定的线段。,1利用MFC App

29、Wizardexe向导创建一个SDI应用程序MyDraw,为视图类CMyDrawView添加成员变量: protected: / 定义有关鼠标作图的成员变量CPoint m_ptOrigin; / 起始点坐标bool m_bDragging; / 拖拽标记HCURSOR m_hCross; / 光标句柄,2在视图类CMyDrawView构造函数中设置拖拽标记和十字光标。CMyDrawView:CMyDrawView()/ TODO: add construction code herem_bDragging=false; / 初始化拖拽标记/ 获得十字光标句柄m_hCross=AfxGetAp

30、p()- LoadStandardCursor(IDC_CROSS);,3利用ClassWizard类向导为视图类添加按下鼠标左键WM_LBUTTONDOWN的消息处理函数。,void CMyDrawView:OnLButtonDown(UINT nFlags, CPoint point) / TODO: Add your message . . . . . .SetCapture(); / 捕捉鼠标:SetCursor(m_hCross); / 设置十字光标m_ptOrigin=point;m_bDragging=TRUE; / 设置拖拽标记/ CView:OnLButtonDown(nFl

31、ags, point);,利用ClassWizard类向导为视图类添加鼠标移动WM_MOUSEMOVE的消息处理函数。,void CMyDrawView:OnMouseMove(UINT nFlags, CPoint point) / TODO: Add your message . . . . . . if(m_bDragging)CClientDC dc(this);dc.MoveTo(m_ptOrigin); dc.LineTo(point); / 绘制线段m_ptOrigin=point; / 新的起始点,void CMyDrawView:OnLButtonUp(UINT nFlags

32、, CPoint point) if(m_bDragging) m_bDragging=false; / 清拖拽标记 ReleaseCapture(); / 释放鼠标,还原鼠标形状,系统中任一时刻只有当前窗口才能捕获鼠标。在程序中需要时通过调用函数CWnd:SetCapture()捕获鼠标,使用鼠标画图结束后应该调用函数ReleaseCapture()释放鼠标。,利用ClassWizard类向导为视图类添加左键释放WM_LBUTTONUP的消息处理函数。,MyDraw程序有一个缺陷,当改变窗口大小或将窗口最小化后再重新打开,原来的线段没有显示出来。其原因是此时调用的是视图类的刷新函数OnDra

33、w(),而在该函数中并没有实现绘制线段的功能。,MyDraw程序存在的问题,完善绘图程序MyDraw,1为线段定义新类CLine。选择“Insert|New Class”菜单命令,弹出New Class对话框中,在Class type栏选择Generic Class,在类名Name栏输入CLine,在类名Base classes栏输入CObject,单击OK按钮,自动生成了类CLine的头文件Line.h和实现文件Line.cpp的框架。,例 完善绘图程序MyDraw,在重绘窗口时能够显示已绘制的线段。,2为类CLine定义成员变量和成员函数。class Cline : CObjectpriv

34、ate:/ 定义成员变量,表示一条直线起点和终点的坐标CPoint m_pt1;CPoint m_pt2;public:CLine();virtual CLine();CLine(CPoint pt1, CPoint pt2); / 构造函数void DrawLine(CDC *pDC); / 绘制线段;,CLine:CLine(CPoint pt1, CPoint pt2)m_pt1=pt1;m_pt2=pt2;void CLine:DrawLine(CDC* pDC)pDC-MoveTo(m_pt1);pDC-LineTo(m_pt2);,在Line.cpp中编写成员函数的实现代码:,3一

35、般都使用数组来保存多条线段的数据,而且MFC提供了实现动态数组的类模板。类CObArray支持CObject指针数组,用它定义的对象可以动态生成。这样,可将存放每条线段数据的变量的指针存到CObArray类的对象中。为此在文档类CMyDrawDoc中定义有关的成员变量和成员函数,需要包含CLine类定义的头文件。,存储线段采用什么数据结构?,#include Line.h#include / 使用MFC类模板class CMyDrawDoc : public CDocument . . . . . .protected:CTypedPtrArray m_LineArray; / 存放线段对象指

36、针的动态数组public:CLine* GetLine(int nIndex); / 获取指定序号线段对象的指针void AddLine(CPoint pt1, CPoint pt2); / 向动态数组中添加新的线段对象的指针int GetNumLines(); / 获取线段的数量. . . . . .;,成员变量m_LineArray是类模板CTypedPtrArray的对象。使用数组类模板CTypedPtrArray需要指定两个模板参数: CTypedPtrArray 参数BASE_CLASS指定基类,可以是CObArray或CPtrArray;参数TYPE指定存储在基类数组中元素的类型。

37、本例中,这两个参数分别为CObArray和CLine*,表示m_LineArray是CObArray的派生类对象,用来存放CLine对象的指针。 为了使用MFC类模板,须包含头文件afxtempl.h。,使用MFC数组类模板CTypedPtrArray,其他主要成员函数:void CMyDrawDoc:AddLine(CPoint pt1, CPoint pt2)CLine* pLine=new CLine(pt1, pt2); / 新建一条线段对象m_LineArray.Add(pLine); / 将该线段加到动态数组CLine* CMyDrawDoc:GetLine(int nIndex)

38、if(nIndexm_LineArray.GetUpperBound() / 判断是否越界return NULL;return m_LineArray.GetAt(nIndex); / 返回给定序号线段对象的指针,int CMyDrawDoc:GetNumLines()return m_LineArray.GetSize(); / 返回线段的数量,4鼠标移动时,除了绘制线段,还要保存当前线段的起点和终点坐标。需要在视图类CMyDrawView的鼠标移动消息处理函数OnMouseMove() 中添加有关代码。,void CMyDrawView:OnMouseMove(UINT nFlags, C

39、Point point) / TODO: Add your message handler code here and/or call defaultif(m_bDragging)CMyDrawDoc *pDoc=GetDocument(); /获得文档对象指针ASSERT_VALID(pDoc); /测试文档对象是否运行有效pDoc-AddLine(m_ptOrigin, point); /加入线段到指针数组CClientDC dc(this);dc.MoveTo(m_ptOrigin); dc.LineTo(point); / 绘制线段m_ptOrigin=point; / 新的起始点/

40、CView:OnMouseMove(nFlags, point);,为了在改变程序窗口大小或最小化窗口后重新打开窗口时保留窗口中原有的图形,必须在OnDraw()函数中重新绘制前面利用鼠标所绘制的线段。这些线段的坐标作为类CLine对象的成员变量,所有CLine对象的指针已保存在动态数组m_LineArray中。,5. 修改OnDraw()函数,void CMyDrawView:OnDraw(CDC* pDC)CMyDrawDoc* pDoc = GetDocument();ASSERT_VALID(pDoc);/ TODO: add draw code for native data her

41、eint nIndex=pDoc-GetNumLines(); / 取得线段的数量/ 循环画出每一段线段while(nIndex-) / 数组下标从0到nIndex-1pDoc-GetLine(nIndex)-DrawLine(pDC); / 类CLine的成员函数,6.4 工具栏和状态栏设计6.4.1 添加工具栏按钮 我们知道单击工具栏按钮也产生命令消息。而且,事实上工具栏按钮和菜单项的功能往往是一致的。为了使工具栏上某个按钮的功能与某个菜单命令的功能相同,只需让该按钮的ID值与对应菜单命令的ID值相同即可。参看例6-10。,状态栏用于显示当前操作的提示信息和程序的运行状态。 MFC应用程序

42、默认的状态栏分为四个部分: 第1部分:显示菜单或工具栏的提示信息 第2部分: Caps Lock,显示键盘的大小写状态 第3部分: Num Lock,显示键盘的数字状态 第4部分: Scroll Lock,显示键盘的滚动状态 状态栏上的每个部分称为一个面板(pane)。,6.4.2 定制状态栏,static UINT indicators =ID_SEPARATOR, / 定义分隔符,作为提示信息行的面板标识ID_INDICATOR_CAPS, / 大写指示器面板标识ID_INDICATOR_NUM, / 数字指示器面板标识ID_INDICATOR_SCRL, / 滚动指示器面板标识;,利用M

43、FC AppWizard向导创建应用程序时,在CMainFrame类中定义了一个成员变量m_wndStatusBar,它是状态栏CStatusBar类的对象。 在MFC应用程序框架的实现文件MainFrm.cpp中,为状态栏定义了一个静态数组indicators。,为了显示硬币数量,添加一个ID为ID_INDICATOR _COIN的指示器面板,将数组indicators作如下修改:static UINT indicators =ID_SEPARATOR, / status line indicatorID_INDICATOR_COIN, / 显示硬币数量指示器ID_INDICATOR_CAP

44、S,ID_INDICATOR_NUM,ID_INDICATOR_SCRL,;,例 修改程序DrawCoin,在状态栏显示硬币的数量。,修改OnDraw()函数,添加显示硬币数量的代码: . . . . . . CString strCoins;/ 先获得主窗口, 再获得状态栏的指针CStatusBar* pStatus=(CStatusBar*)AfxGetApp()-m_pMainWnd-GetDescendantWindow(ID_VIEW_STATUS_BAR); if(pStatus)strCoins.Format(硬币:%d, pDoc-m_nCoins); / 设置要显示的信息pS

45、tatus-SetPaneText(1, strCoins); / 显示硬币数量,面板编号从0开始 . . . . . .,涉及到数据处理的应用程序一般都要考虑文档数据的永久保存。虽然可以直接利用类CFile来实现文件的读写操功能,但在MFC应用程序中,序列化(Serialize)使得程序员可以不直接面对一个物理文件而进行文档的读写。序列化实现了文档数据的保存和装入的幕后工作,MFC通过序列化实现应用程序的文档读写功能。,6.5 文档的读写,1. 序列化的基本思想: 一个类应该能够对自己的成员变量的数据进行读写操作,对象可以通过读操作而重新创建。即对象可以将其当前状态(由其成员变量的值表示)写

46、入永久性存储体(通常是指磁盘)中,以后可以从永久性存储体中读取(载入)对象的状态,从而重建对象。类的对象自己应该具备将状态值写入磁盘或从磁盘中读出的方法(即成员函数),这种对象的保存和恢复的过程称为序列化。,6.5.1 序列化工作原理,一个可序列化的类必须有一个称作为序列化的成员函数Serialize(),文档的序列化在文档类的成员函数Serialize()中进行。MFC AppWizard应用程序向导在生成应用程序时只创建了文档派生类序列化Serialize()函数的框架,由于不同程序的数据结构各不相同,可序列化的类应该重载Serialize()函数,使其支持对特定数据的序列化。并且,任何需要保存的变量(数据)都应该在文档派生类中声明。,

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > 企业管理 > 经营企划

本站链接:文库   一言   我酷   合作


客服QQ:2549714901微博号:道客多多官方知乎号:道客多多

经营许可证编号: 粤ICP备2021046453号世界地图

道客多多©版权所有2020-2025营业执照举报