窗口重绘函数

在刷新窗口时经常要调用重绘函数,MFC提供了三个函数用于窗口重绘

InvalidateRect(&Rect)

Invalidate()

UpdateWindow()

当需要更新或者重绘窗口时,一般系统会发出两个消息

WM_PAINT(通知客户区有变化)和WM_NCPAINT(通知非客户区有变化)

WM_NVPAINT系统会自己搞定

WM_PAINT消息对应的函数是OnPaint(),它是系统默认的接受 WM_PAINT消息的函数,但我们一般在程序中做重绘时都在OnDraw函数中进行的,因为在ONPAIN函数中调用了ONDRAW函数。

//CView默认的标准的重画函数

void CView::OnPaint()

{ CPaintDC dc(this);

   OnPreparDC(&dc); 

   OnDraw(&dc); //调用了OnDraw

}

上面讲到 InvalidateRect(&Rect) Invalidate()两个函数形式和功能差不多但Invalidate是使得整个窗口无效,形成无效矩形,而 InvalidateRect(&Rect)是使得指定的区域无效

Invalidate()申明无效,等待WM_PAINT消息以便重绘,队列中无其他消 息时系统会自动发送 UpdateWindow()会立即发送WM_PAINT,不过在它发送前,先调用GetUpdateRect(hWnd,NULL,TRUE)看有无可 绘制区域,如果没有则不发送消息

RedrawWindow()RedrawWindow()则是具有 Invalidate()和UpdateWindow()的双特性。声明窗口的状态为无效,并立即更新窗口,立即调用WM_PAINT消息处理。

系统为什么不在调用Invalidate时发送WM_PAINT消息呢?又为什么非要等 应用消息队列为空时才发送WM_PAINT消息呢?这是因为系统把在窗口中的绘制操作当作一种低优先级的操作,于是尽 可能地推后做。不过这样也有利于提高绘制的效率:两个WM_PAINT消息之间通过InvalidateRect和InvaliateRgn使之失效的区 域就会被累加起来,然后在一个WM_PAINT消息中一次得到 更新,不仅能避免多次重复地更新同一区域,也优化了应用的更新操作。像这种通过InvalidateRect和InvalidateRgn来使窗口区域无 效,依赖于系统在合适的时机发送WM_PAINT消息的机 制实际上是一种异步工作方式,也就是说,在无效化窗口区域和发送WM_PAINT消息之间是有延迟的;有时候这种延迟并不是我们希望的,这时我们当然可以 在无效化窗口区域后利用SendMessage 发送一条WM_PAINT消息来强制立即重画,但不如使用Windows GDI为我们提供的更方便和强大的函数:UpdateWindow和RedrawWindow。UpdateWindow会检查窗口的 Update Region,当其不为空时才发送WM_PAINT消息;RedrawWindow则给我们更多的控制:是否重画非客户区和背景,是否总是发送 WM_PAINT消息而不管Update Region是否为空等。

BeginPaint和WM_PAINT消息紧密相关。试一试在WM_PAINT处理函 数中不写BeginPaint会怎样?程序会像进入了一个死循环一样达到惊人的CPU占用率,你会发现程序总在处理一个接 一个的WM_PAINT消息。这是因为在通常情况下,当应用收到WM_PAINT消息时,窗口的Update Region都是非空的(如果为空就不需要发送WM_PAINT消息了),BeginPaint的一个作用就是把该Update Region置为空,这样如果不调用BeginPaint,窗口的Update Region就一直不为空,如前所述,系统就会一直发送WM_PAINT消息。 BeginPaint和WM_ERASEBKGND消息也有关系。

当窗口的Update Region被标志为需要擦除背景时,BeginPaint会发送WM_ERASEBKGND消息来重画背景,同时在其返回信息里有一个标志表明窗口背景 是否被重画过。当我们用InvalidateRect和InvalidateRgn来把指定区域加到Update Region中时,可以设置该区域是否需要被擦除背景,这样下一个BeginPaint就知道是否需要发送WM_ERASEBKGND消息了。

另外要注意的一点是,BeginPaint只能在WM_PAINT处理函数中使用。

 

http://www.diybl.com/course/3_program/c++/cppjs/20090412/164845.html

你可能感兴趣的:(VC++)