一种用户体验-显示对话框时灰化你的主窗体

袁永福 ( http://www.xdesigner.cn ) 2007-8-10

程序全部源代码下载(工程文件使用VS.NET2003格式):/Files/xdesigner/DisableMask.rar

    在一些Web程序中,有一种页面效果,当弹出一个模拟的对话框时,主页面就整体灰化了,其他的元素不能动弹,只有这个对话框能用,用户关闭了对话框,整个页面才恢复原来的操作.这种用户体验是不错的,提示了用户必须处理的对话框才能继续处理页面.如何实现的我猜想是动态生成一个大的DIV层,把它置于顶层并设置半透明的灰色.

    在WinForm程序中也需要这种用户体验,我们有时观察到主窗体显示了一个对话框,此时用户还试图用鼠标点击主窗体搞些操作,但这种操作注定是要失败,影响到软件可用性.于是我就想到把Web程序中的这种用户体验移植到WinForm程序中来帮助用户意识到主窗体的当前状态.

    实现这个用户体验有两个问题,一是如何知道主窗体弹出对话框,二是如何灰化主窗体.

    首先是解决如何知道主窗体何时弹出对话框,研究了一下,没发现窗体对象System.Windows.Forms.Form类型提供有所帮助的事件方法属性.我们可以在程序代码中,每次弹出对话框前添加灰化主窗体的代码,这样加大了程序开发量,而且代码移植性不好。后来想了又想,试了又试,发现弹出对话框时,主窗体的状态是不可用的,但此时窗体的Enabled属性不能反应这种状态,使用Win32API函数却能正确获得其状态.因此最后决定使用计时器System.Windows.Forms.Timer来不断的调用Win32API函数来测试主窗体是否可用,若可用则不必灰化主窗体,若不可用则灰化主窗体。

    第二步就是灰化主窗体了,根据WEB程序中的实现过程,我们很自然的想到用一个半透明的控件来覆盖整个主窗体,于是我们又如何创造这个半透明控件。纵观System.Windows.Forms名称空间,号称提供半透明效果效果的只有Label类型了,经过测试,发现Label类型的半透明属性是假的,是模拟出来的,它是在控件背景中模拟绘制窗体的背景来搞出半透明的效果,若Label控件背后有其他控件还是要被Label无情的覆盖掉。因此我们需要一个真正的半透明控件,于是我又想了又想,试了又试,找了又找,终于把这个真正的半透明控件搞出来了。

    基础问题解决了,然后就是代码的组成和组件化了,我定义了一个DisableMaskControl控件,实现了真正的半透明处理,里面有个定时器,不断的使用Win32API函数测试这个控件所在的窗体是否有效,若有效则隐藏控件,若无效则显示控件,把控件覆盖整个窗体并置于顶层。这样我就用一百来行的C#代码实现了这种弹出对话框灰化主窗体的用户体验,而且这个代码使用非常简单,只要在需要这种效果的主窗体上添加一行代码 this.Controls.Add( new DisableMaskControl()) 即可。

   此处或许有人提出这个定时器的效率问题,我觉得没多大问题,首先控件少,一个窗体才用一个,相对于高速的CPU,用户手动操作来显示和关闭对话框是极其缓慢的操作。而且定时器中进行的判断不多,只调用了一个API函数,无伤大雅。

   以下是程序的运行效果。

disablemask.PNG

   以下是控件 DisableMaskControl的全部代码。

  1 using System;
  2 using System.Runtime.InteropServices ;
  3 namespace DisableMask
  4 {
  5     /// <summary>
  6     /// 窗体无效时用于掩盖整个状态的半透明控件
  7     /// </summary>
  8     /// <remarks>编制 袁永福( http://www.xdesigner.cn )</remarks>
  9     public class DisableMaskControl : System.Windows.Forms.Control
 10     {
 11         /// <summary>
 12         /// 初始化对象
 13         /// </summary>
 14         public DisableMaskControl()
 15         {
 16             this.SetStyle( System.Windows.Forms.ControlStyles.SupportsTransparentBackColor , true );
 17             myTimer = new System.Windows.Forms.Timer();
 18             myTimer.Interval = 100 ;
 19             myTimer.Tick +=new EventHandler(myTimer_Tick);
 20             this.BackColor = System.Drawing.Color.FromArgb( 80 , 0 , 0 , 0 );
 21             myTimer.Start();
 22         }
 23         /// <summary>
 24         /// 内部用于定时处理的计时器
 25         /// </summary>
 26         private System.Windows.Forms.Timer myTimer = null;
 27 
 28         /// <summary>
 29         /// 已重载:返回控件创建参数
 30         /// </summary>
 31         protected override System.Windows.Forms.CreateParams CreateParams
 32         {
 33             get
 34             {
 35                 System.Windows.Forms.CreateParams ps = base.CreateParams;
 36                 ps.ExStyle = ps.ExStyle | 0x20 ;
 37                 return ps ;
 38             }
 39         }
 40 
 41         /// <summary>
 42         /// 绘制控件的背景,啥也不干.
 43         /// </summary>
 44         /// <param name="pevent">事件参数</param>
 45         protected override void OnPaintBackground(System.Windows.Forms.PaintEventArgs pevent)
 46         {
 47         }
 48         /// <summary>
 49         /// 绘制控件
 50         /// </summary>
 51         /// <param name="e">事件参数</param>
 52         protected override void OnPaint(System.Windows.Forms.PaintEventArgs e)
 53         {
 54             using( System.Drawing.SolidBrush b = new System.Drawing.SolidBrush( this.BackColor ))
 55             {
 56                 e.Graphics.FillRectangle( b , e.ClipRectangle );
 57             }
 58         }
 59 
 60         /// <summary>
 61         /// 定时器处理
 62         /// </summary>
 63         /// <param name="sender">事件参数</param>
 64         /// <param name="e">事件参数</param>
 65         private void myTimer_Tick(object sender, EventArgs e)
 66         {
 67             System.Windows.Forms.Form frm = this.FindForm();
 68             if( frm == null )
 69                 return ;
 70             if( frm.IsDisposed )
 71                 return ;
 72             ifthis.IsDisposed )
 73                 return ;
 74             ifthis.IsHandleCreated == false )
 75                 return ;
 76 
 77             // 主窗体显示对话框时窗体不可用,但此时它的Enable属性无法判断其是否真的
 78             // 不可用,因此必须调用Win32API来判断其是否真的不可用.
 79             if( IsWindowEnabled( frm.Handle ) == false )
 80             {
 81                 ifthis.Visible == false )
 82                 {
 83                     this.Dock = System.Windows.Forms.DockStyle.None ;
 84                     this.Bounds = new System.Drawing.Rectangle( 
 85                         0 , 
 86                         0 , 
 87                         frm.ClientSize.Width ,
 88                         frm.ClientSize.Height );
 89                     this.BringToFront();    
 90                     this.Visible = true;
 91                     this.Refresh();
 92                     frm.Refresh();
 93                 }
 94             }
 95             else
 96             {
 97                 ifthis.Visible )
 98                     this.Visible = false;
 99             }
100         }
101 
102         /// <summary>
103         /// 用于判断窗体是否有效的Win32API函数
104         /// </summary>
105         /// <param name="hWnd">窗体句柄</param>
106         /// <returns>窗体是否有效</returns>
107         [DllImport("user32.dll", CharSet=CharSet.Auto, ExactSpelling=true)]
108         private static extern bool IsWindowEnabled(IntPtr hWnd);
109 
110     }//public class DisableMaskControl : System.Windows.Forms.Control
111 }

转载于:https://www.cnblogs.com/xdesigner/archive/2007/08/10/850235.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/410741.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

sql经典语句

一、基础 1、说明&#xff1a;创建数据库 CREATE DATABASE database-name 2、说明&#xff1a;删除数据库 drop database dbname 3、说明&#xff1a;备份sql server --- 创建 备份数据的 device USE master EXEC sp_addumpdevice disk, testBack, c:\mssql7backup\MyNwind_1.d…

[html] html元素哪些标签是不可替换元素?哪些是可替换元素?

[html] html元素哪些标签是不可替换元素&#xff1f;哪些是可替换元素&#xff1f; 可替换元素&#xff08;replaced element&#xff09;的展现效果不是由 CSS 来控制的。这些元素是一种外部对象&#xff0c;它们外观的渲染&#xff0c;是独立于 CSS 的。也就是说&#xff0c…

html文档中用于表示页面标题的标记对是,汽车发动机拆装与检修实训超星尔雅答案...

摘要&#xff1a;汽车生活事件也称为应激源( )在教师的直接领导下整个班级的学生一起进行的学习&#xff0c;机拆检修称为( )实训教学二字连用最早出自( )...汽车生活事件也称为应激源( )发动下列不属于课程与教学关系观的是( )在教师的直接领导下整个班级的学生一起进行的学习…

一些类的概念

1、异常处理 try: pass except: pass2、面向对象 怎么用类 面向过程#1、4s店#2、保险公司&#xff0c;上保险#3、税务局交税#4、车管所 面向对象1&#xff1a;买车处 #1、4s店 #2、保险公司&#xff0c;上保险 #3、税务局交税…

[html] 请使用canvas画一个椭圆

[html] 请使用canvas画一个椭圆 <script> var cdocument.getElementById("myCanvas"); var ctxc.getContext("2d"); ctx.beginPath(); ctx.arc(95,50,40,0,2*Math.PI); ctx.stroke(); </script> 个人简介 我是歌谣&#xff0c;欢迎和大家一起…

html 跟随鼠标移动线条,canvas跟随鼠标移动的随机线条

/** type {HTMLCanvasElement} */let canvas document.querySelector(#canvas);let cxt canvas.getContext(2d);let w canvas.width window.innerWidth;let h canvas.height window.innerHeight;mousePosition {x: Number,y: Number}//随机颜色function ranColor() {ret…

go 优秀文档

go语言资料汇总 &#xff1a; https://blog.zhnytech.com/articles/2016/07/15/Golang%E5%AD%A6%E4%B9%A0%E8%B5%84%E6%96%99%E6%94%B6%E9%9B%86.html github资料基础&#xff1a;https://github.com/Unknwon/the-way-to-go_ZH_CN/blob/master/eBook/directory.md k8s国外镜像 …

Python 程序 运行过程

一. Python 解释器 Python 是一门编程语言&#xff0c;同时也是一个解释器的软件包。 解释器是一种让其他程序运行起来的程序。 当你编写了一段Python 程序&#xff0c;Python 解释器将读取程序&#xff0c;并按照其中的命令执行&#xff0c;得出结果。解释器是代码与计算机硬…

[html] 在H5中如何预加载音频?

[html] 在H5中如何预加载音频&#xff1f; audio标签默认会预加载个人简介 我是歌谣&#xff0c;欢迎和大家一起交流前后端知识。放弃很容易&#xff0c; 但坚持一定很酷。欢迎大家一起讨论 主目录 与歌谣一起通关前端面试题

H5|web移动前端自适应适配布局解决方案

方案&#xff1a; 固定一个某些宽度&#xff0c;使用一个模式&#xff0c;加上少许的媒体查询方案使用flexbox解决方案使用百分比加媒体查询使用rem1. 简单问题简单解决 我觉得有些web app并一定很复杂&#xff0c;比如拉勾网&#xff0c;你看看它的页面在iphone4,iphone6,ipad…

[转] 用Diff和Patch工具维护源码

在Unix系统下&#xff0c;维护源码版本可以使用很多方法&#xff0c;其中最常用的当然是大名鼎鼎的CVS&#xff0c;但实际上&#xff0c;简单的版本维护工作并没有必要使用复杂的CVS等专门的版本维护工具&#xff0c;Unix标配中的diff和patch工具就完全可以完成代码的简单备份和…

[html] 当页面中使用application/xhtml+xml会出现什么问题吗?

[html] 当页面中使用application/xhtmlxml会出现什么问题吗&#xff1f; 要求比较严格&#xff0c;必须有head、body标签且每个元素必须是关闭的。一些老的浏览器不支持&#xff0c;实际上&#xff0c;任何最新的浏览器都将支持application/xhtmlxml媒体类型。大多数浏览器也接…

html5 本地保存大数据库,HTML5本地储存 - jellydd的个人空间 - OSCHINA - 中文开源技术交流社区...

传统方式我们用document.cookie来储存&#xff0c;但是储存大小较小&#xff0c;而且解析复杂&#xff0c;HTML5提出新的解决方案&#xff0c;使用sessionStorage和localStorage存储数据。localStorage&#xff1a;1. 永久生效2. 多窗口共享3. 容量大约为20M◆window.localStor…

[html] H5的Web Storage带来什么好处?

[html] H5的Web Storage带来什么好处&#xff1f; 存储的数据量更大&#xff0c;可以达到 5M&#xff1b;减少不必要的数据请求&#xff0c;不会自动把数据发送给服务器localStorage 可以把数据永久保存在本地&#xff0c;除非显示的清除或删除数据支持事件通知机制API 更加方便…

「BZOJ2654」tree

「BZOJ2654」tree 最小生成树二分答案。 最开始并没有觉得可以二分答案&#xff0c;因为答案并不单调啊。 其实根据题意&#xff0c;白边的数目肯定大于need条&#xff0c;而最小生成树的白边数并不等于need&#xff08;废话&#xff09;&#xff0c;可以二分将每条白边的权值m…

jQuery 结合 Json 提交数据到Webservice,并接收从Webservice返回的Json数据

简单的Json数据提交 jQuery ajax webservice&#xff1a;get 和 post 一、GET 方式客户端代码vardata { classCode: "0001"}; //这里要直接使用JOSN对象$.ajax({ type: "GET", contentType: "application/json; chars…

html服务器框架,一种类似http/html的分布式GUI程序设计框架

笔者构想了一种类似http/html的分布式GUI程序设计框架&#xff0c;适用于WIndows、桌面Linux、Mac OS以及C&#xff0c;Java&#xff0c;Python等多种支持GUI编程的程序设计系统。但是对于手机小屏幕&#xff0c;可能不合适。本系统包括窗口描述规则;gui.exe -- 一个命令行程序…

X86逆向教程10:学会使用硬件断点

本节课我们将学习硬件断点的使用技巧&#xff0c;硬件断点是由硬件提供给我们的一组寄存器&#xff0c;我们可以对这些硬件寄存器设置相应的值&#xff0c;然后让硬件帮我们断在需要下断点的地址上面&#xff0c;这就是硬件断点&#xff0c;硬件断点依赖于寄存器&#xff0c;这…

PHP获取IP的多种方式解析

转自&#xff1a;http://developer.51cto.com/art/200911/164514.htm PHP获取IP的方法有许多种&#xff0c;我们今天向大家总结了六种方法&#xff0c;希望通过对这六种方法的学习&#xff0c;能够加深我们对PHP语言的进一步了解&#xff0c;巩固我们所掌握的知识。 PHP获取IP方…

html 监控键盘,后台监控鼠标和键盘(可监听全局的鼠标以及键盘按键)

资源下载此资源下载价格为3D币&#xff0c;请先登录资源文件列表MouseKeyboardLibrary/MouseKeyboardLibrary/app.config , 134MouseKeyboardLibrary/MouseKeyboardLibrary/bin/Debug/MouseKeyboardLibrary.exe , 23040MouseKeyboardLibrary/MouseKeyboardLibrary/bin/Debug/Mo…