(一)UpdateAllViews() 与 Invalidate()的区别
UpdateAllViews()是在DOC/VIEW结构中,当一个视图的数据改变后,通知所有视图作相应的改变,和重画毫无关系。
Invalidate()是使窗口无效,使系统向其发WM_PAINT消息,使的程序的OnPaint被调用重画客户区。
而UpdateAllViews()是文档与视之间的联系,调用从它会使程序与此文档相关的所有视的UpdateView被调用至于是否重画以及怎么画是由各视的UpdateView来决定的。
Invalidate()是Cwnd的成员函数,与DOC-VIEW无关;
UpdateAllViews是CDocument的成员函数,具体体现DOC-VIEW的精神。
如果仅重画当前窗口用 this->Invalidate();
如果通知所有和当前文档相关的窗口重画用GetDocument()->UpdateAllViews(this)(在View中)或this->UpdateAllViews(NULL)(在Doc中)。
Invalidate()函数产生一条WM_PAINT消息,并送入windows消息队列中,是窗口产生重画。
UpdateAllViews 并不进入windows消息队列中,直接产生重画
(二) CDocument::UpdateAllViews
void UpdateAllViews(CView* pSender,LPARAM lHint = 0L,CObject* pHint = NULL);
参数:
pSender
指向修改文档的视图,如果所有视图被更新,则设为NULL;
lHint
包含文档被修改的信息;
pHint
指向一个存储修改信息的的对象。
备注:
在调用SetModifiedFlag 成员函数之后,应该调用这个函数。该函数把文档被修改的信息通知给每个视图, 视图被参数pSender指定的情况除外。通常在用户已经通过视图改变文档之后,从视图类中调用此函数,来获取文档被修改的信息。
这个函数为文档类的视图调用 CView::OnUpdate成员函数传递 pHint 和 lHint。用这些参数传递文档被修改的信息给视图。可以通过lHint编码信息,或者可以定义一个 CObject 派生类去存储修改信息,而且通过使用pHint传递该类的一个对象。在CView派生类中,基于传递的信息,重载CView::OnUpdate 成员函数去优化视图显示的更新。
通过MFC中Document/View结构我们知道,同一份Document产生一个以上的Views视图是MDI程序的天赋,MDI程序标准的【Window/new window】项目就是为达此目的而设计的。
让所有的Views同步更新数据的关键在于两个函数:
1. CDocument::UpdateAllViews 这个函数会巡防所有隶属同一份Document的各个Views,找到一个就通知一个,这里所谓“通知”就是调用其Update函数。
2. CView::OnUpdate 我们可以在这个函数中设计绘图操作,实质上是根据UpdateAllViews传递进来的pHint参数设计View视图中的无效区域Invalidate()或者InvalidateRect(),用以产生WM_PAINT消息,进而调用OnDraw函数重绘无效绘图区域。
virtual void UpdateAllViews(CView* pSender, LPAEAM lHint, CObject* pHint);
> 第一个参数代表发出这一通牒的始作俑者。因为始作俑者自己已经把画面更新过了,不需要在被通知OnUpdate。
> 后面两个参数lHint和pHint是所谓的提示参数(Hint),他们会被传送到同一Document所对应的每一个Views的OnUpdate函数中去。lHint可以使一些特殊的提示值,pHint则是一个派生自CObject的对象指针。
virtual void OnUpdate(CView* pSender, LPAEAM lHint, CObject* pHint);
OnUpdate收到三个参数(由CDocument::UpdateAllViews发出的)。
void CDocument::UpdateAllViews(CView* pSender, LPARAM lHint, CObject* pHint) // walk through all views { ASSERT(pSender == NULL || !m_viewList.IsEmpty()); // must have views if sent by one of them
POSITION pos = GetFirstViewPosition(); while (pos != NULL) { CView* pView = GetNextView(pos); ASSERT_VALID(pView); if (pView != pSender) pView->OnUpdate(pSender, lHint, pHint); } }
......
void CView::OnUpdate(CView* pSender, LPARAM /*lHint*/, CObject* /*pHint*/) { ASSERT(pSender != this); UNUSED(pSender); // unused in release builds // invalidate the entire pane, erase background too Invalidate(TRUE); }
在设计这一步同一份Document的所有Views视图同步更新的过程中,我们可能还需要利用的几个重要函数也在此列出:
OnPrepareDC(); LPtoDP(); DPtoLP(); GetClipBox(); IntersectRect();
OnPrepareDC(); LPtoDP(); DPtoLP()这三个函数会在下一节CScrollView(可滚动的窗口)中介绍到,这里简单介绍后两个函数的作用:
GetClipBox()
函数功能:该函数得到一个能够完全包含当前可见区域的最小矩形的大小。该可见区域由当前的剪切区域定义或由剪切路径所定义或者由任何重叠的窗口所定义。
函数原型:int GetClipBox(HDC hdc, LPRECT lprc);
> hdc: 设备环境句柄。
> lprc:RECT结构的一个指针,用来接收矩形的大小。
返回值:如果该函数执行成功,那么它的返回值定义了剪切区域的复杂性,返回值的意义为:
NULLREGIOW:区域为空;
SimpIEREGZO:区域为一单个矩形;
CompLEXREGIOW:区域为多个矩形;
ERROR:发生错误;
GetclipBox:返回基于给定设备一下文环境的逻辑坐标。
Windows若想获得更多的错误信息,调用GetLastError函数。
IntersectRect()是CRect类的一个成员函数,用来计算两个矩形的交集。
参考:http://blog.sina.com.cn/s/blog_ce4139180101c0a6.html
http://technet.microsoft.com/zh-cn/library/w1z3te7e.aspx