重绘函数onpaint()

2024-06-16 22:48
文章标签 函数 重绘 onpaint

本文主要是介绍重绘函数onpaint(),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

WM_PAINT是窗口每次重绘都会产生的一个消息。

OnPaint是对这个消息的反应函数

mfc CWnd::OnPaint 没做什么,只是丢给系统处理。

一:

   先执行OnEraseBkgnd,擦除背景(如果想自绘控件,这个函数直接return TRUE就可以了,这样就不会擦除背景,不会闪)

OnEraseBkGndOnPaint的区别与联系

OnEraseBkGnd,如果你不调用原来缺省的OnEraseBkGnd只是重画背景则不会有闪烁.而在OnPaint里面,由于它隐含的调用了OnEraseBkGnd,而你又没有处理OnEraseBkGnd 函数,这时就和窗口缺省的背景刷相关了.缺省的 OnEraseBkGnd操作使用窗口的缺省背景刷刷新背景(一般情况下是白刷),而随后你又自己重画背景造成屏幕闪动.

OnEraseBkGnd不是每次都会被调用的.如果你调用Invalidate的时候参数为TRUE,那么在OnPaint里面隐含调用BeginPaint的时候就产生WM_ERASEBKGND消息,如果参数是FALSE,则不会重刷背景.

ZYP解释:void Invalidate( BOOL bErase = TRUE ); 该函数的作用是使整个窗口客户区无效。窗口的客户区无效意味着需要重绘,参数bEraseTRUE时,重绘区域内的背景将被重绘即擦除,否则,背景将保持不变。调用Invalidate等函数后窗口不会立即重绘,这是由于WM_PAINT消息的优先级很低,它需要等消息队列中的其它消息发送完后才能被处理。

OnPaint里面会调用BeginPaint函数自动设置显示设备内容的剪切区域而排除任何更新区域外的区域更新区域。如果更新区域被标记为可擦除的,BeginPaint发送一个WM_ERASEBKGND消息给窗口。WM_ERASEBKGND消息的响应函数既是OnEraseBkGnd()

 

所以解决方法有三个半:

1.OnEraseBkGnd实现,不要调用原来的OnEraseBkGnd函数.

2.OnPaint实现,同时重载OnEraseBkGnd,其中直接返回.

3.OnPaint实现,创建窗口时设置背景刷为空

4.OnPaint实现,但是要求刷新时用Invalidate(FALSE)这样的函数.(不过这种情况下,窗口覆盖等造成的刷新还是要闪一下,所以不是彻底的解决方法)

都挺简单的.

MFC中任何一個window元件的繪圖都是放在這兩個member function

在設定上 OnEraseBkgnd()是用來畫底圖的而OnPaint()是用來畫主要物件的

舉例說明一個按鈕是灰色的上面還有文字

OnEraseBkgnd()所做的事就是把按鈕畫成灰色

OnPaint()所做的事就是畫上文字

既然這兩個member function都是用來畫出元件的

那為何還要分OnPaint() OnEraseBkgnd()

其實OnPaint() OnEraseBkgnd() 特性是有差的

1. OnEraseBkgnd()的要求是快速在裡面的繪圖程式最好是不要太耗時間

因為每當window元件有任何小變動都會馬上呼叫OnEraseBkgnd()

2. OnPaint() 是只有在程式有空閒的時候才會被呼叫

3. OnEraseBkgnd() 是在 OnPaint() 之前呼叫的

所以 OnPaint()被呼叫一次之前可能會呼叫OnEraseBkgnd()好幾次

 

如果我們是一個在做圖形化使用者介面的人

常會需要把一張美美的圖片設為我們dialog的底圖

把繪圖的程式碼放在OnPaint() 之中可能會常碰到一些問題

比方說拖曳一個視窗在我們做的dialog上面一直移動

dialog會變成灰色直到動作停止才恢復

這是因為每次需要重繪的時候程式都會馬上呼叫OnEraseBkgnd()

OnEraseBkgnd()就把dialog畫成灰色

而只有動作停止之後程式才會呼叫OnPaint() 這時才會把我們要畫的底圖貼上去

 

這個問題的解法比較差點的方法是把OnEraseBkgnd() 改寫成不做事的function

如下所示

BOOL CMyDlg::OnEraseBkgnd(CDC* pDC)

{

return TRUE;

}

以上本來是會呼叫CDialog::OnEraseBkgnd() 但是如果我們不呼叫的話

程式便不會畫上灰色的底色了

 

Q:基于对话框的程序中如何重载OnEraseBkGnd()函数

A:这是一个消息WM_ERASEBKWND 

 

比較好的做法是直接將繪圖的程式從OnPaint()移到OnEraseBkgnd()來做

如下所示

 

// m_bmpBKGND 為一CBitmap物件且事先早已載入我們的底圖

// 底圖的大小與我們的視窗client大小一致

 

BOOL CMyDlg::OnEraseBkgnd(CDC* pDC)

{

CRect rc;

GetUpdateRect(&rc);

CDC srcDC;

srcDC.CreateCompatibleDC(pDC);

srcDC.SelectObject(m_bmpBKGND);

 

pDC->BitBlt(rc.left,rc.top,rc.GetWidth(),

rc.GetHeight(),&srcDC,rc.left,rc.top,SRCCOPY);

return TRUE;

}

 

特別要注意的是取得重畫大小是使用GetUpdateRect() 而不是GetClientRect()

如果使用GetClientRect() 會把不該重畫的地方重畫

二:

     系统的Onpaint中调用了OnDraw,但如果我们自己继承了一个OnPaint函数又没有显式调用OnDraw,则OnDraw就不会被调用,OnInitialUpdateOnDraw之前,是窗口被创建以后调用的第一个函数。

这篇关于重绘函数onpaint()的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1067776

相关文章

hdu1171(母函数或多重背包)

题意:把物品分成两份,使得价值最接近 可以用背包,或者是母函数来解,母函数(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v) 其中指数为价值,每一项的数目为(该物品数+1)个 代码如下: #include<iostream>#include<algorithm>

C++操作符重载实例(独立函数)

C++操作符重载实例,我们把坐标值CVector的加法进行重载,计算c3=c1+c2时,也就是计算x3=x1+x2,y3=y1+y2,今天我们以独立函数的方式重载操作符+(加号),以下是C++代码: c1802.cpp源代码: D:\YcjWork\CppTour>vim c1802.cpp #include <iostream>using namespace std;/*** 以独立函数

函数式编程思想

我们经常会用到各种各样的编程思想,例如面向过程、面向对象。不过笔者在该博客简单介绍一下函数式编程思想. 如果对函数式编程思想进行概括,就是f(x) = na(x) , y=uf(x)…至于其他的编程思想,可能是y=a(x)+b(x)+c(x)…,也有可能是y=f(x)=f(x)/a + f(x)/b+f(x)/c… 面向过程的指令式编程 面向过程,简单理解就是y=a(x)+b(x)+c(x)

利用matlab bar函数绘制较为复杂的柱状图,并在图中进行适当标注

示例代码和结果如下:小疑问:如何自动选择合适的坐标位置对柱状图的数值大小进行标注?😂 clear; close all;x = 1:3;aa=[28.6321521955954 26.2453660695847 21.69102348512086.93747104431360 6.25442246899816 3.342835958564245.51365061796319 4.87

OpenCV结构分析与形状描述符(11)椭圆拟合函数fitEllipse()的使用

操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C++11 算法描述 围绕一组2D点拟合一个椭圆。 该函数计算出一个椭圆,该椭圆在最小二乘意义上最好地拟合一组2D点。它返回一个内切椭圆的旋转矩形。使用了由[90]描述的第一个算法。开发者应该注意,由于数据点靠近包含的 Mat 元素的边界,返回的椭圆/旋转矩形数据

Unity3D 运动之Move函数和translate

CharacterController.Move 移动 function Move (motion : Vector3) : CollisionFlags Description描述 A more complex move function taking absolute movement deltas. 一个更加复杂的运动函数,每次都绝对运动。 Attempts to

✨机器学习笔记(二)—— 线性回归、代价函数、梯度下降

1️⃣线性回归(linear regression) f w , b ( x ) = w x + b f_{w,b}(x) = wx + b fw,b​(x)=wx+b 🎈A linear regression model predicting house prices: 如图是机器学习通过监督学习运用线性回归模型来预测房价的例子,当房屋大小为1250 f e e t 2 feet^

JavaSE(十三)——函数式编程(Lambda表达式、方法引用、Stream流)

函数式编程 函数式编程 是 Java 8 引入的一个重要特性,它允许开发者以函数作为一等公民(first-class citizens)的方式编程,即函数可以作为参数传递给其他函数,也可以作为返回值。 这极大地提高了代码的可读性、可维护性和复用性。函数式编程的核心概念包括高阶函数、Lambda 表达式、函数式接口、流(Streams)和 Optional 类等。 函数式编程的核心是Lambda

PHP APC缓存函数使用教程

APC,全称是Alternative PHP Cache,官方翻译叫”可选PHP缓存”。它为我们提供了缓存和优化PHP的中间代码的框架。 APC的缓存分两部分:系统缓存和用户数据缓存。(Linux APC扩展安装) 系统缓存 它是指APC把PHP文件源码的编译结果缓存起来,然后在每次调用时先对比时间标记。如果未过期,则使用缓存的中间代码运行。默认缓存 3600s(一小时)。但是这样仍会浪费大量C

PHP7扩展开发之函数方式使用lib库

前言 首先说下什么是lib库。lib库就是一个提供特定功能的一个文件。可以把它看成是PHP的一个文件,这个文件提供一些函数方法。只是这个lib库是用c或者c++写的。 使用lib库的场景。一些软件已经提供了lib库,我们就没必要再重复实现一次。如,原先的mysql扩展,就是使用mysql官方的lib库进行的封装。 在本文,我们将建立一个简单的lib库,并在扩展中进行封装调用。 代码 基础