关于WM_ERASEBKGND和WM_PAINT的深刻理解

关于WM_ERASEBKGND和WM_PAINT的深刻理解

 

一直以来,对于WM_PAINT和WM_ERASEBKGND消息不是很清楚,从书上和网上找了很多资料,大体上有以下几点说法:
1>WM_PAINT先产生,WM_ERASEBKGND后产生

2.WM_PAINT产生后,在调用BeginPaint时
hdc = BeginPaint(hWnd, &ps); 
如果ps.fErase为true,则BeginPaint会产生WM_ERASEBKGND消息

3.BeginPaint函数用来擦除窗口背景

4.WM_ERASEBKGND用来绘制背景

经过调试、分析,发现上面的说法并不正确。以下是一些测试代码,代码后面附上一些分析。最后总结出几点,可以解释程序中出现的所有关于窗口重绘的问题。
如有不正确的地方,大家可以指正。

为了说明问题,在此不说WM_NCPAINT消息(非客户区消息),只说WM_ERASEBKGND消息和客户区的WM_PAINT消息

//此段代码摘自vc6应用程序向导自动生成的代码,并添加了一些测试代码
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
 int wmId, wmEvent;
 PAINTSTRUCT ps;
 HDC hdc;
 TCHAR szHello[MAX_LOADSTRING];
 LoadString(hInst, IDS_HELLO, szHello, MAX_LOADSTRING);
 
 switch (message)
 {
 case WM_COMMAND:
  wmId    = LOWORD(wParam);
  wmEvent = HIWORD(wParam);
  // Parse the menu selections:
  switch (wmId)
  {
  case IDM_ABOUT:
   DialogBox(hInst, (LPCTSTR)IDD_ABOUTBOX, hWnd, (DLGPROC)About);
   break;
  case IDM_EXIT:
   DestroyWindow(hWnd);
   break;
  default:
   return DefWindowProc(hWnd, message, wParam, lParam);
  }
  break;
  case WM_ERASEBKGND: //如果处理了这个消息,则默认消息处理函数不会调用,背景就不会绘制
   {
    static int iCount=0;
    char ch[MAX_PATH];
    sprintf(ch,"%d ---------WM_ERASEBKGND\n",iCount); //这个函数需要包含#include
    OutputDebugString(ch); //调试时便于观察
    iCount++;
   break;
   }
  case WM_PAINT:
   {
    OutputDebugString("  -------------WM_PAINT\n");
    hdc = BeginPaint(hWnd, &ps); //使无效区域变得有效,并填充ps结构
    // TODO: Add any drawing code here...
    
    //绘制一个蓝色椭圆,ps.rcPaint保存了客户区矩形
    HBRUSH hbrush=::CreateSolidBrush(RGB(0,0,255)); 
    ::SelectObject(hdc,hbrush);
    ::Ellipse(hdc,ps.rcPaint.left,ps.rcPaint.top,ps.rcPaint.right,ps.rcPaint.bottom);
    ::DeleteObject(hbrush);
    
    EndPaint(hWnd, &ps);
    break;
   }
  case WM_LBUTTONDOWN: //调用DefWindowProc擦除客户区背景
   {
    HDC hdc;
    hdc=::GetDC(hWnd);
    WPARAM w=(WPARAM)hdc;
    LPARAM l=0;
    DefWindowProc(hWnd, WM_ERASEBKGND, w, l);
   }
   break;
  case WM_DESTROY:
   PostQuitMessage(0);
   break;
  default:
   return DefWindowProc(hWnd, message, wParam, lParam);
 }
 return 0;
}

 

先说一下程序运行时发现的一些现象:
1.
上面的代码:如果添加了WM_ERASEBKGND消息,里面什么也不做,如下
case WM_ERASEBKGND:
  
  break;

则当程序运行时,如果收到WM_ERASEBKGND消息,则这个switch-case结构中就不会执行默认消息处理函数DefWindowProc,运行时发现,窗口的背景就没有了,即背景为空。
这说明了窗口背景仅仅是由默认的消息处理函数DefWindowProc绘制的。
(注:注册窗口类时,背景设置为白色wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW+1);)


2.
如上面的代码,因为有WM_ERASEBKGND消息,则程序运行时窗口背景为NULL,
但是如果添加了WM_LBUTTONDOWN消息,从里面调用默认窗口消息处理函数,如下
case WM_LBUTTONDOWN:
  {
   HDC hdc;
   hdc=::GetDC(hWnd);
   WPARAM w=(WPARAM)hdc; //变量w作为WM_ERASEBKGND消息的wParam参数,保存了设备环境句柄
   LPARAM l=0;
   DefWindowProc(hWnd, WM_ERASEBKGND, w, l); //调用默认消息处理函数DefWindowProc
  }
  break;
程序运行时,如果用鼠标单击一下窗口客户区,则窗口的背景就会显示!这进一步说明了窗口的背景色是由默认消息处理函数DefWindowProc绘制的。

3.
以上代码,因为添加了WM_ERASEBKGND消息,所以窗口背景是空。
虽然在WM_PAINT消息中有
hdc = BeginPaint(hWnd, &ps);函数的调用,但是窗口背景仍然是空,这说明了BeginPaint函数并不会擦除背景(即用默认画刷绘制窗口背景)。

BeginPaint函数只做了两件事情:
1》使窗口无效区域变得有效,从而使Windows不再发送WM_PAINT消息(直到窗口大小改变等,使窗口再次变得无效)。
(如果窗口一直无效,则Windows会不停地发送WM_PAINT消息)

1》填充PAINTSTRUCT结构。填充这个结构的目的,是让程序员可以根据ps变量中的标志值进行某些操作

4.
调试的时候,发现:当窗口改变大小,或者其它操作使窗口变得无效时,WM_ERASEBKGND消息总是先于WM_PAINT消息发出,而且如果产生WM_ERASEBKGND消息,
则最后一个WM_ERASEBKGND的下一条消息一定是WM_PAINT消息(WM_ERASEBKGND可能会连续产生几次)。WM_ERASEBKGND消息和WM_PAINT消息之间没有其它消息


--------------------------------------------
以下是一些总结

1.窗口背景的擦除(即绘制)

窗口的背景色是由默认的消息处理函数DefWindowProc擦除的(即这个函数使用注册窗口类时使用的背景刷擦除窗口背景)。
什么时候绘制?在窗口函数收到WM_ERASEBKGND消息,DefWindowProc函数以WM_ERASEBKGND为参数,才会绘制窗口背景
(注:当WM_ERASEBKGND消息产生后,窗口一定有一部分变得无效)

2.窗口的无效:

当拖动窗口的一个顶点改变了窗口的大小、窗口由最小化恢复到最大化、窗口的一部分被其它窗口遮住又重新显示、调用MoveWindow函数改变了窗口大小、窗口移动到桌面之外的

部分被拖回重新显示时,窗口就会变得无效。 无效区域是整个客户区,因此默认窗口处理函数DefWindowProc会擦除整个客户区。
(注:拖动窗口标题栏移动窗口,只要窗口没有移动到屏幕之外,那么这两个消息都不产生)

当窗口无效时,Windows会给窗口发出WM_ERASEBKGND消息和WM_PAINT消息,而且WM_ERASEBKGND先发出一次或者几次,紧接着是WM_PAINT


例外:InvalidateRect函数的调用会使窗口变得无效,并产生WM_ERASEBKGND消息和WM_PAINT消息,而WM_ERASEBKGND是否产生取决于参数bErase

void InvalidateRect (
LPCRECT lpRect,
BOOL bErase = TRUE );
当参数bErase为true时,WM_ERASEBKGND消息产生,当bErase为false时WM_ERASEBKGND消息不产生


3.消息的处理过程
当窗口无效时,

先发出WM_ERASEBKGND消息若干次-----------再发出WM_PAINT消息,WM_ERASEBKGND和WM_PAINT之间没有其它消息
WM_ERASEBKGND消息的后面一定是WM_PAINT

1》WM_ERASEBKGND消息的处理:
上面的代码,如果没有添加WM_ERASEBKGND,则默认的消息处理函数DefWindowProc会被调用,此时的DefWindowProc会擦除窗口背景(即绘制背景),并且ps.fErase会为FALSE

如果添加了WM_ERASEBKGND消息,DefWindowProc就不会被调用,则无法擦除窗口背景,并且ps.fErase会为true

2》WM_PAINT的处理
在这个消息中如果调用了hdc = BeginPaint(hWnd, &ps);函数,则此函数只做了两件事:填充ps结构、使窗口重新变得有效

另外DefWindowProc函数也会使窗口变得有效

关于ps.fErase;
这个参数和窗口函数WndProc的返回值有关:
当窗口函数WndProc返回true;则产生WM_PAINT消息时,ps.fErase就为false;表明系统擦除了背景
当窗口函数WndProc返回false;则产生WM_PAINT消息时,ps.fErase就为true;表明系统没有擦除背景

设想一下,当上面的代码中添加了WM_ERASEBKGND消息并在其中直接返回true(这表明系统已经绘制了窗口背景),则ps.fErase就为false
case WM_ERASEBKGND: 
return true; //窗口函数WndProc返回true;

注意返回的真或者假只是让程序员可以看见ps.fErase,并作出自己的代码,与窗口的显示即背景没有关系
有些人说当ps.fErase==true,BeginPaint函数会发送一个WM_ERASEBKGND消息,其实BeginPaint并未发出WM_ERASEBKGND消息

4.自己绘制背景或者系统绘制背景。
如果程序员不想系统擦除背景,而自己想绘制背景,怎么办呢?方法是在WM_ERASEBKGND消息处理中添加自己的绘制代码。
对于WM_ERASEBKGND消息,wParam参数保存了用于绘制的设备环境,lParam不使用。

如上面的示例代码,当添加了WM_ERASEBKGND消息,则switch---case中就不会调用DefWindowProc函数绘制背景。这时,程序员自己就可以添加绘制代码
而在基于MFC的程序中,是这样处理自绘代码的:

BOOL CCeDlg::OnEraseBkgnd(CDC* pDC) //这个函数就是WM_ERASEBKGND的消息处理函数
{
 // TODO: Add your message handler code here and/or call default
//添加自绘代码
...
return TRUE; //返回真,代表着窗口函数的返回值。以便于程序员在WM_PAINT消息中作出相应处理(如果需要)。这里返回时就不会调用下面的默认处理

//下面将调用系统默认的消息处理函数DefWindowProc进行背景的默认绘制。
 return CDialog::OnEraseBkgnd(pDC); //不执行自动生成的这个函数
}

执行这个函数时,提示用户绘制背景,如果用户没有绘制背景,则return CDialog::OnEraseBkgnd(pDC);调用默认的窗口处理函数进行背景的擦除


BOOL CCeDlg::OnEraseBkgnd(CDC* pDC)
{
 // TODO: Add your message handler code here and/or call default
//添加自绘背景代码
 CBitmap m_bitmap;
 BITMAP m_bmInfo;
 m_bitmap.LoadBitmap(IDB_BITMAP1);
 m_bitmap.GetObject(sizeof(m_bmInfo),&m_bmInfo);

 CDC memDC;
 memDC.CreateCompatibleDC(pDC);
 memDC.SelectObject(&m_bitmap);

 GetClientRect(m_rect);
 pDC->StretchBlt(0,0,m_rect.Width(),m_rect.Height(),

&memDC,0,0,m_bmInfo.bmWidth,m_bmInfo.bmHeight,SRCCOPY); //内存拷贝函数。绘制背景
 memDC.DeleteDC();

 return true;//返回真,代表着窗口函数的返回值。以便于程序员在WM_PAINT消息中作出相应处理(如果需要)。这里返回时就不会调用下面的默认处理

//下面将调用系统默认的消息处理函数DefWindowProc进行背景的默认绘制。
 return CDialog::OnEraseBkgnd(pDC); //不执行自动生成的这个函数
}

5.WM_ERASEBKGND消息和WM_PAINT消息的另外一种含义:背景色与前景色

可以这样理解WM_ERASEBKGND消息和WM_PAINT消息:

WM_ERASEBKGND消息用于通知系统或者程序员绘制背景色
WM_PAINT消息用于通知程序员绘制前景色,比如在WM_PAINT中调用TextOut函数输出文本

 

 

 

 

 

 

发布了1 篇原创文章 · 获赞 2 · 访问量 1万+

大学四年,我把私藏的自学「学习网站/实用工具」都贡献出来了

10-29 阅读数 9万+

在分享之前,先说说初学者如何学习编程,这个话题想必非常的重要,要学好编程,给你一些学习网站也好、实用工具也好,但前提是你知道如何去学习它。 见过很多初学者,以及小鹿我刚开始学习的时候,也是自己瞎摸索,... 博文

中国麻将:世界上最早的区块链项目

10-29 阅读数 5万+

中国麻将:世界上最早的区块链项目

最近区块链这个玩意又被市场搞的很是火热,相信大部分人都不太清楚这玩意到底是怎么样的一个概念,它来了,它来了,它到底是啥~ 国家都开始发文支持了,下面是一个通俗易懂的…


博文



比特币原理详解

10-29 阅读数 4万+

一、什么是比特币

比特币是一种电子货币,是一种基于密码学的货币,在2008年11月1日由中本聪发表比特币白皮书,文中提出了一种去中心化的电子记账系统,我们平时的电子现金是银行来记账,因为银行的背后是…


博文



python学习方法总结(内附python全套学习资料)

11-03 阅读数 1万+

不要再问我python好不好学了 我之前做过半年少儿编程老师,一个小学四年级的小孩子都能在我的教学下独立完成python游戏,植物大战僵尸简单版,如果要肯花时间,接下来的网络开发也不是问题,人工智能也... 博文

兼职程序员一般可以从什么平台接私活?

10-31 阅读数 8万+

这个问题我进行了系统性的总结,以下将进行言简意赅的说明和渠道提供,希望对各位小猿/小媛们有帮助~

根据我们的经验,程序员兼职主要分为三种:兼职职位众包、项目整包和自由职业者驻场。

所谓的兼职职位众…


博文



程序员接私活怎样防止做完了不给钱?

10-31 阅读数 2万+

首先跟大家说明一点,我们做 IT 类的外包开发,是非标品开发,所以很有可能在开发过程中会有这样那样的需求修改,而这种需求修改很容易造成扯皮,进而影响到费用支付,甚至出现做完了项目收不到钱的情况。

那…


博文



Python十大装B语法

11-02 阅读数 9万+

Python 是一种代表简单思想的语言,其语法相对简单,很容易上手。不过,如果就此小视 Python 语法的精妙和深邃,那就大错特错了。本文精心筛选了最能展现 Python 语法之精妙的十个知识点,并... 博文

数据库优化 - SQL优化

11-01 阅读数 1万+

从一个示例入手,带你一步一步掌握SQL优化的技巧! 博文

通俗易懂地给女朋友讲:线程池的内部原理

11-04 阅读数 8869

餐盘在灯光的照耀下格外晶莹洁白,女朋友拿起红酒杯轻轻地抿了一小口,对我说:“经常听你说线程池,到底线程池到底是个什么原理?”... 博文

C/C++/Java/Go/Rust,Python喊你来打擂:3秒钟内统计出小于1亿的素数个数

11-04 阅读数 9623

前几天,有个非计算机专业的同学问我,如何快速找出1亿之内的孪生素数——所谓孪生素数,就是差值为2的两个素数。原本以为这是一个很简单的问题,随便用python写了一个方法,没想到却要跑17分钟左右。改用... 博文

Java中List集合介绍(炒鸡详细呦)

11-04 阅读数 1342

Java中List集合介绍(炒鸡详细呦) 1,Java集合介绍

作为一个程序猿,Java集合类可以说是我们在工作中运用最多、最频繁的类。相比于数组(Array)来说,集合类的长度可变,更加方便开发。…


博文



送给单身猿们的表白神器

11-07 阅读数 2万+

问天下男生,有谁想单身?又有谁想单身一辈子? 虽然本人也是单身狗,但是也是有一个远大的理想,哈哈,大白天的我又开始做梦了 原网址:http://wfhuang.coding.me/LoveJuan... 博文

刷了几千道算法题,这些我私藏的刷题网站都在这里了!

11-08 阅读数 4万+

遥想当年,机缘巧合入了 ACM 的坑,周边巨擘林立,从此过上了"天天被虐似死狗"的生活…

然而我是谁,我可是死狗中的战斗鸡,智力不够那刷题来凑,开始了夜以继日哼哧哼哧刷题的日子,从此"读题与提交…


博文



2019工程伦理慕课答案(2019秋)习题及期末答案

11-09 阅读数 1万+

第一章习题(下) 单选题 (1/1 point) 下列哪一项不是工程与技术的区别

内容和性质
目的
活动主体
任务、对象和思维方式

单选题 (1/1 point)
下列…


博文



JavaScript 为什么能活到现在?

11-08 阅读数 1万+

作者 | 司徒正美

责编 |郭芮

出品 | CSDN(ID:CSDNnews)

JavaScript能发展到现在的程度已经经历不少的坎坷,早产带来的某些缺陷是永久性的,因此浏览器才有禁用Ja…


博文



别翻了,这篇文章绝对让你深刻理解java类的加载以及ClassLoader源码分析【JVM篇二】

12-01 阅读数 1万+

点进文章的盆友不如先来做一道非常常见的面试题,如果你能做出来,可能你早已掌握并理解了java的类加载机制,若结果出乎你的意料,那就很有必要来了解了解java的类加载机制了。代码如下嗯哼?其实上面程序并... 博文

Nginx 原理和架构

11-09 阅读数 1万+

Nginx 是一个免费的,开源的,高性能的 HTTP 服务器和反向代理,以及 IMAP / POP3 代理服务器。Nginx 以其高性能,稳定性,丰富的功能,简单的配置和低资源消耗而闻名。 Nginx... 博文

致 Python 初学者

11-13 阅读数 5万+

欢迎来到“Python进阶”专栏!来到这里的每一位同学,应该大致上学习了很多 Python 的基础知识,正在努力成长的过程中。在此期间,一定遇到了很多的困惑,对未来的学习方向感到迷茫。我非常理解你们所... 博文

《吊打面试官》系列-Redis终章_凛冬将至 FPX_新王登基

11-11 阅读数 4127

你知道的越多,你不知道的越多 点赞再看,养成习惯

前言
Redis在互联网技术存储方面使用如此广泛,几乎所有的后端技术面试官都要在Redis的使用和原理方面对小伙伴们进行360°的刁难。作为一个…


博文



YouTube排名第一的励志英文演讲《Dream(梦想)》

11-12 阅读数 2万+

Idon’t know what that dream is that you have, I don't care how disappointing it might have been as y... 博文

“狗屁不通文章生成器”登顶GitHub热榜,分分钟写出万字形式主义大作

11-13 阅读数 4万+

一、垃圾文字生成器介绍

最近在浏览GitHub的时候,发现了这样一个骨骼清奇的雷人项目,而且热度还特别高。

项目中文名:狗屁不通文章生成器
项目英文名:BullshitGenerator
根据作…


博文



程序员:我终于知道post和get的区别

11-14 阅读数 5万+

是一个老生常谈的话题,然而随着不断的学习,对于以前的认识有很多误区,所以还是需要不断地总结的,学而时习之,不亦说乎... 博文

Java世界最常用的工具类库

11-20 阅读数 1万+

Apache Commons Apache Commons有很多子项目 Google Guava 参考博客

博文

程序员把地府后台管理系统做出来了,还有3.0版本!12月7号最新消息:已在开发中有github地址

11-17 阅读数 4万+

第一幕:缘起

听说阎王爷要做个生死簿后台管理系统,我们派去了一个程序员……

996程序员做的梦:

第一场:团队招募

为了应对地府管理危机,阎王打算找“人”开发一套地府后台管理系统,于是…


博文



网易云6亿用户音乐推荐算法

11-17 阅读数 1万+

网易云音乐是音乐爱好者的集聚地,云音乐推荐系统致力于通过 AI 算法的落地,实现用户千人千面的个性化推荐,为用户带来不一样的听歌体验。

本次分享重点介绍 AI 算法在音乐推荐中的应用实践,以及在算法…


博文



小白都能看得懂的java虚拟机内存模型

11-26 阅读数 7749

目录

一、虚拟机

二、虚拟机组成

1.栈

栈帧

2.程序计数器

3.方法区

对象组成

4.本地方法栈

5.堆

GC

GC案例

一、虚拟机

同样的java代码在不…


博文



深入理解HashMap

11-20 阅读数 9815

什么是 HashMap? ​ HashMap 是基于哈希表的 Map 接口是实现的。此实现提供所有可选操作,并允许使用 null 做为值(key)和键(value)。HashMap 不保证映射的顺序... 博文

《吊打面试官》系列-消息队列基础

11-20 阅读数 1万+

你知道的越多,你不知道的越多 点赞再看,养成习惯 GitHub上已经开源 https://github.com/JavaFamily 有一线大厂面试点脑图、个人联系方式和人才交流群,欢迎Sta... 博文

碎片化的时代,如何学习

11-20 阅读数 4354

今天周末,和大家聊聊学习这件事情。 在如今这个社会,我们的时间被各类 APP 撕的粉碎。 刷知乎、刷微博、刷朋友圈; 看论坛、看博客、看公号; 等等形形色色的信息和知识获取方式一个都不错过。 貌似学了... 博文

【Java面试官】史上最全的JAVA专业术语面试100问

11-30 阅读数 2万+

春风如贵客,一到便繁华。各位看官点赞再看,养成好习惯(●´∀`●)

gitee上已经开源https://gitee.com/Li-Ren/blog里面有一线大厂面试点脑图,欢迎Star和PR你认为…


博文



27 个提升开发幸福度的 VsCode 插件

11-20 阅读数 1万+

作者:Jsmanifest 译者:前端小智 来源:Medium

Visual Studio Code(也称为VSCode)是一种轻量级但功能强大的跨平台源代码编辑器, 借助对TypeScript …


博文



腾讯“疯狂”开源!

11-20 阅读数 1万+

作者 | 马超

责编 | 胡巍巍

出品 | CSDN(ID:CSDNnews)

近日,腾讯自研的万亿级分布式消息中间件TubeMQ正式开源,并捐赠给Apache基金会,成为基金会官方认可的Inc…


博文



so easy! 10行代码写个"狗屁不通"文章生成器

11-20 阅读数 2万+

前几天,GitHub 有个开源项目特别火,只要输入标题就可以生成一篇长长的文章。

背后实现代码一定很复杂吧,里面一定有很多高深莫测的机器学习等复杂算法

不过,当我看了源代码之后…


博文



MySQL数据库总结

11-25 阅读数 9616

一、数据库简介

数据库(Database,DB)是按照数据结构来组织,存储和管理数据的仓库。
典型特征:数据的结构化、数据间的共享、减少数据的冗余度,数据的独立性。
关系型数据库:使用关系模型把数据…


博文



深入理解ArrayList

11-21 阅读数 1万+

什么是ArrayList? ArrayList的实现原理其实就是数组(动态数组),ArrayList的介绍及简单使用方法 动态数组与一般数组有什么区别? 与Java中的数组相比,ArrayList的容... 博文

Java生成随机图片验证码

11-21 阅读数 5307

今天跟大佬学了java随机生成验证码,

开心Ing,

激动ing,

前台html代码

<div style=“margin-top: 50px;”>



博文



                
                                

没有更多推荐了,返回首页

©️2019 CSDN 皮肤主题: 大白 设计师: CSDN官方博客

你可能感兴趣的:(关于WM_ERASEBKGND和WM_PAINT的深刻理解)