Winform应用程序实现通用消息窗口

记得我之前发表过一篇文章《Winform应用程序实现通用遮罩层》,是实现了透明遮罩的消息窗口,功能侧重点在动图显示+消息提醒,效果看上去比较的炫,而本篇我又来重新设计通用消息窗口,功能重点在于消息提醒、进度报告,当然如果大家时间,可以将两种相结合,那样就会更完美了,我这里仍是以实现功能为主,由于代码相对简单,我就直接贴上所有代码,大家可以直接复制到本地测试,若发现问题可自行改正或反馈给我,我来完善,谢谢!

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Threading;
using System.Windows.Forms;

namespace WindowsFormsApplication1
{
    /// <summary>
    /// 等待窗口:用于处理耗时工作时,友好显示消息窗口
    /// 作者:Zuowenjun
    /// 日期:2016-1-29
    /// 网址:http://www.zuowenjun.cn
    /// </summary>
    public partial class FRM_Waitting : Form
    {
        private SynchronizationContext formContext;

        public string Message
        {
            get { return labMessage.Text; }
            set { labMessage.Text = value; }
        }

        public Action<WaittingForWorkObject> WorkAction { get; set; }

        public object WorkActionParam { get; set; }

        public Exception WorkException { get; private set; }

        public class WaittingForWorkObject
        {
            private SendOrPostCallback UpdateMessageAction = null;
            public SynchronizationContext Context { get; private set; }

            public object UserData { get; private set; }

            public void UpdateMessage(string msg)
            {
                this.Context.Post(UpdateMessageAction, msg);
            }

            public WaittingForWorkObject(FRM_Waitting parentForm)
            {
                this.Context = parentForm.formContext;
                this.UserData = parentForm.WorkActionParam;
                this.UpdateMessageAction = delegate(object o)
                {
                    parentForm.Message = o.ToString();
                };
            }
        }

        public static void WaittingForWork(Action<WaittingForWorkObject> workAction, object workParam = null, string text = "请稍候", string message = "系统处理中,请稍候...")
        {
            var waittingForm = new FRM_Waitting(text, message, workAction, workParam);
            waittingForm.ShowDialog();
            if (waittingForm.WorkException != null)
            {
                throw waittingForm.WorkException;
            }

        }

        public FRM_Waitting()
        {
            InitializeComponent();
        }

        public FRM_Waitting(string text, string message, Action<WaittingForWorkObject> workAction, object workParam = null)
            : this()
        {
            this.Text = text;
            this.Message = message;
            this.WorkAction = workAction;
            this.WorkActionParam = workParam;
        }

        private void FRM_Waitting_Load(object sender, EventArgs e)
        {

        }

        private void FRM_Waitting_Shown(object sender, EventArgs e)
        {
            formContext = SynchronizationContext.Current;
            if (WorkAction != null)
            {
                Thread workThread = new Thread(DoWork);
                workThread.IsBackground = true;
                workThread.Start();
            }
        }

        private void DoWork()
        {
            try
            {
                var wfObject = new WaittingForWorkObject(this);
                WorkAction(wfObject);
            }
            catch (Exception ex)
            {
                WorkException = ex;
            }
            formContext.Send(delegate(object o) { this.Close(); }, null);
        }

    }
}

以下是系统自动生成的代码:

namespace WindowsFormsApplication1
{
    partial class FRM_Waitting
    {
        /// <summary>
        /// Required designer variable.
        /// </summary>
        private System.ComponentModel.IContainer components = null;

        /// <summary>
        /// Clean up any resources being used.
        /// </summary>
        /// <param name="disposing">true if managed resources should be disposed; otherwise, false.</param>
        protected override void Dispose(bool disposing)
        {
            if (disposing && (components != null))
            {
                components.Dispose();
            }
            base.Dispose(disposing);
        }

        #region Windows Form Designer generated code

        /// <summary>
        /// Required method for Designer support - do not modify
        /// the contents of this method with the code editor.
        /// </summary>
        private void InitializeComponent()
        {
            this.labMessage = new System.Windows.Forms.Label();
            this.SuspendLayout();
            //
            // labMessage
            //
            this.labMessage.Dock = System.Windows.Forms.DockStyle.Fill;
            this.labMessage.Font = new System.Drawing.Font("宋体", 9.75F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(134)));
            this.labMessage.Location = new System.Drawing.Point(0, 0);
            this.labMessage.Name = "labMessage";
            this.labMessage.Padding = new System.Windows.Forms.Padding(5);
            this.labMessage.Size = new System.Drawing.Size(453, 125);
            this.labMessage.TabIndex = 0;
            this.labMessage.Text = "Message";
            this.labMessage.TextAlign = System.Drawing.ContentAlignment.MiddleCenter;
            this.labMessage.UseWaitCursor = true;
            //
            // FRM_Waitting
            //
            this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 12F);
            this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
            this.ClientSize = new System.Drawing.Size(453, 125);
            this.ControlBox = false;
            this.Controls.Add(this.labMessage);
            this.FormBorderStyle = System.Windows.Forms.FormBorderStyle.FixedSingle;
            this.Name = "FRM_Waitting";
            this.ShowIcon = false;
            this.ShowInTaskbar = false;
            this.StartPosition = System.Windows.Forms.FormStartPosition.CenterParent;
            this.Text = "FRM_Waitting";
            this.UseWaitCursor = true;
            this.Load += new System.EventHandler(this.FRM_Waitting_Load);
            this.Shown += new System.EventHandler(this.FRM_Waitting_Shown);
            this.ResumeLayout(false);

        }

        #endregion

        private System.Windows.Forms.Label labMessage;
    }
}

上述代码比较简单,我这里对消息窗口的实现原理作一个简要的说明:

1.将耗时处理逻辑代码封装到一个委托中(Aciton<FRM_Waitting.WaittingForWorkObject>);

2.获取当前同步上下文并保存,以便可以跨线程操作UI;

3.创建并运行一个后台线程,同时将该线程指定到DoWork(工作方法);

4.在DoWork方法中实例化WaittingForWorkObject对象,并传给1中委托,然后执行委托,这样耗时的操作都在后台线程中处理了;

5.在DoWork方法使用try catch捕获可能存在的异常,若发生异常则保存到WorkException属性中;

6.执行完成后(无论是否报错),通过同上下文发送关闭消息窗口指令,使消息窗口关闭;

7.在静态方法WaittingForWork中判断WorkException属性是否不为空,若不为空则重新抛出错误,这样主线程就知道发生了什么异常;

 说明:为了能够兼容.NET 2.0及以上版本,代码中采用了匿名方法,而非Lambada表达式,实际使用时则可以任意选择,下面的测试示例中均提供了新旧两种代码写法,以供大家比较。

以下是各种测试示例:

        /// <summary>
        /// 测试:普通显示一个消息窗口
        /// </summary>
        private void Test1()
        {
            //旧方式(兼容.NET2.0及以上)
            FRM_Waitting.WaittingForWork(delegate(FRM_Waitting.WaittingForWorkObject o)
            {
                //在这里面写耗时处理逻辑代码,以下是模拟耗时
                Thread.Sleep(10 * 1000);
            });

            //新方式(.NET4.0及以上)
            //FRM_Waitting.WaittingForWork((o) =>
            //{
                //在这里面写耗时处理逻辑代码,以下是模拟耗时
            //    Thread.Sleep(10 * 1000);
            //});

        }

效果如下:

        /// <summary>
        /// 测试:普通显示一个消息窗口,并自定义提示消息并窗口标题
        /// </summary>
        private void Test1_1()
        {
            //旧方式(兼容.NET2.0及以上)
            FRM_Waitting.WaittingForWork(delegate(FRM_Waitting.WaittingForWorkObject o)
            {
                //在这里面写耗时处理逻辑代码,以下是模拟耗时
                Thread.Sleep(10 * 1000);
            },null,"客官请稍候","客官,店小二正在为您拼命处理中,请稍等片刻...");

            //新方式(.NET4.0及以上)
            //FRM_Waitting.WaittingForWork((o) =>
            //{
                //在这里面写耗时处理逻辑代码,以下是模拟耗时
            //    Thread.Sleep(10 * 1000);
            //},null,"客官请稍候","客官,店小二正在为您拼命处理中,请稍等片刻...");

        }

效果如下:

        /// <summary>
        /// 测试:普通显示一个消息窗口,并在后台线程中抛出错误,前台显示错误信息
        /// </summary>
         private void Test1_2()
        {
             try
             {
                 //旧方式(兼容.NET2.0及以上)
                 //FRM_Waitting.WaittingForWork(delegate(FRM_Waitting.WaittingForWorkObject o)
                 //{
                 //    //在这里面写耗时处理逻辑代码,以下是模拟耗时
                 //    Thread.Sleep(10 * 1000);
                 //    throw new Exception("这里后台线程里抛出的错误!");
                 //});

                 //新方式(.NET4.0及以上)
                 FRM_Waitting.WaittingForWork((o) =>
                 {
                    //在这里面写耗时处理逻辑代码,以下是模拟耗时
                     Thread.Sleep(10 * 1000);
                     throw new Exception("这里后台线程里抛出的错误!");
                 });

             }
             catch(Exception ex)
             {
                 MessageBox.Show("发生异常:" + ex.Message);
             }
        }

效果如下:

        /// <summary>
        /// 测试:在消息窗口上显示加载进度
        /// </summary>
        private void Test2()
        {

            //旧方式(兼容.NET2.0及以上)
            //FRM_Waitting.WaittingForWork(delegate(FRM_Waitting.WaittingForWorkObject o)
            //{
                    //在这里面写耗时处理逻辑代码,以下是模拟耗时
            //    for (int i = 1; i <= 10; i++)
            //    {
            //        Thread.Sleep(1000);
            //        o.UpdateMessage(string.Format("共{0}项,当前已加载{1}项", 10, i));
            //    }

            //});

            //新方式(.NET4.0及以上)
            FRM_Waitting.WaittingForWork((o) =>
            {
                //在这里面写耗时处理逻辑代码,以下是模拟耗时
                for (int i = 1; i <= 10; i++)
                {
                    Thread.Sleep(1000);
                    o.UpdateMessage(string.Format("共{0}项,当前已加载{1}项", 10, i));
                }

            });

        }

效果如下:

        /// <summary>
        /// 测试:在消息窗口上显示加载进度,并同时在主窗口(非消息窗口都可以)上更新控件内容
        /// </summary>
        private void Test3()
        {

            //旧方式(兼容.NET2.0及以上)
            FRM_Waitting.WaittingForWork(delegate(FRM_Waitting.WaittingForWorkObject o)
            {
                //在这里面写耗时处理逻辑代码,以下是模拟耗时
                for (int i = 1; i <= 10; i++)
                {
                    Thread.Sleep(1000);
                    o.UpdateMessage(string.Format("共{0}项,当前已加载{1}项", 10, i));
                    o.Context.Send(delegate(object d) { this.listBox1.Items.Add(d); }, string.Format("共{0}项,当前已加载{1}项", 10, i));
                }

            });

            //新方式(.NET4.0及以上)
            //FRM_Waitting.WaittingForWork((o) =>
            //{
                //在这里面写耗时处理逻辑代码,以下是模拟耗时
            //    for (int i = 1; i <= 10; i++)
            //    {
            //        Thread.Sleep(1000);
            //        o.UpdateMessage(string.Format("共{0}项,当前已加载{1}项", 10, i));
            //        o.Context.Send(d => this.listBox1.Items.Add(d), string.Format("共{0}项,当前已加载{1}项", 10, i));
            //    }

            //});

        }

效果如下:

看完上面的测试效果,大家觉得如何,能否满足你的日常要求呢,我认为基本都可以满足的,当然如果发现更多的情况,欢迎在下方评论留言。

时间: 2024-10-02 06:24:26

Winform应用程序实现通用消息窗口的相关文章

Winform应用程序实现通用遮罩层二

之前先后发表过:<Winform应用程序实现通用遮罩层>.<Winform应用程序实现通用消息窗口>,这两款遮罩层其实都是基于弹出窗口的,今天为大家分享一个比较简单但界面相对友好的另一种实现方案,废话不多说,直接进入主题. 一.实现思路(解决问题顺序): 透明遮罩: 1.实现可设置透明的Panel控件(MaskPanel): 2.Panel控件(MaskPanel)能够覆盖父容器(一般是当前窗体form对象)客户区区域(即:与父容器客户区区域大小相同),并处于最上层,保证父容器上的

Winform应用程序实现通用遮罩层

Winform应用程序实现通用遮罩层 在WEB上,我们在需要进行大数据或复杂逻辑处理时,由于耗时较长,一般我们会在处理过程中的页面上显示一个半透明的遮罩层,上面放个图标或提示:正在处理中...等字样,这样用户体验就比较好了,然而如果在Winform客户端程序,通常遮罩层的处理就显得不那么简单或不那么好看,而我今天要说明的是,我实现的这个Winform通用遮罩层,却可以实现类似WEB上的遮罩层,既可以透明,而且还可以显示动态图片以及文字,那如何实现的呢,我现在一一讲解. 首先要明确我们要实现的效果

C#程序员开发WinForm必须知道的 Window 消息大全

不要以为下面的东西只有C++中才会用到哦! 消息,就是指Windows发出的一个通知,告诉应用程序某个事情发生了.例如,单击鼠标.改变窗口尺寸.按下键盘上的一个键都会使Windows发送一个消息给应用程序. 消息本身是作为一个记录传递给应用程序的,这个记录中包含了消息的类型以及其他信息.例如,对于单击鼠标所产生的消息来说,这个记录中包含了单击鼠标时的坐标.这个记录类型叫做TMsg,它在Windows单元中是这样声明的: type TMsg = packed record hwnd: HWND;

日积(Running)月累(ZSSURE):WCF学习之“通过事件绑定控制WinForm宿主程序主界面控件”

背景: WCF服务需要寄宿到相应的可运行进程中执行,常见的有四种寄宿,分别是控制台程序.WinForm程序.IIS和Windows服务.之前学习老A博客和<WCF全面解析>时最常用到的是控制台寄宿,近期由于项目需求,需要在WinForm程序中调用WCF服务,本博文通过一个简单的实例来演示WCF在WinForm中的寄宿.并着重介绍如何利用事件绑定控制宿主主UI界面控件. 题记: 之前一直坚守在C++阵地,对于新语言.新技术(诸如Python.J2EE.Bigdata.AI)不甚感冒.自以为&qu

WinForm应用程序中实现自动更新功能

WinForm应用程序中实现自动更新功能 编写人:左丘文 2015-4-20 近来在给一客户实施ECM系统,但他们使用功能并不是我们ECM制造版提供的标准功能,他们要求对系统作一些定制功能,为了避免因程序的bug而带来频繁让用户更新程序的不良影响,就想给ECM增加一个winform自动更新功能,今天在这里,我想与大家一起分享代码,在此做个小结,以供参考.有兴趣的同学,可以一同探讨与学习一下,否则就略过吧.   1. 首先我们在这里先分析一下其它程序猿的一些基本情况: 相信有许多程序猿都喜欢用Wi

.net remoting 实现通用消息处理窗口

.net remoting 实现通用消息处理窗口 实现机制是制作一个cmd窗口作为信息展示窗口,主程序将需要展示的信息抛出到cmd窗口显示,以此方式做到消息的展示. 以下是cmd窗口的代码,cmd窗体作为一个消息接收的server 1 static void Main(string[] args) 2 { 3 try 4 { 5 //声明一个TCP通道指定端口8085 6 TcpChannel channel = new TcpChannel(8085); 7 //注册通道 8 ChannelS

PyQt4消息窗口

默认情况下,如果我们单击了窗口标题栏上的X标记,窗口就会被关闭.但是有些时候我们想要改变这一默认行为.比如,我们正在编辑的文件内容发生了变化,这时若单击X标记关闭窗口,编辑器就应当但出确认窗口. #!/usr/bin/python # -*- coding:utf-8 -*- import sys from PyQt4 import QtGui class MessageBox(QtGui.QWidget): def __init__(self, parent = None): QtGui.QW

.NET 基础 一步步 一幕幕[Winform应用程序]

时隔半载,重回博客园,一切从头再来,今天只是开始,原谅我这一生放荡不羁爱自由. 进入今天得主题曲:Winform应用程序(简介) 1.      winform应用程序是一种智能客户端技术,我们可以使用winform应用程序,帮助我们获得信息或者传输信息等. 2.      winform控件得属性: Name:在后台要获得前台的控件对象,需要使用Name属性. visible:指示一个控件是否可见. Enabled:指示一个控件是否可用. 3. 在Main函数当中创建的窗体对象,我们称之为这个

VC++ 对话框程序响应键盘消息的处理方法的说明(非常重要)

基于MFC对话框的应用程序在响应按键消息和热键方面都力不从心,CDialog类的消息循环中去掉了TranslateAccelerator函数,因此不能响应热键:同时由于对话框上可能有很多控件,且默认情况下这些子窗口已经截获了焦点,因此键盘消息已经被控件捕获了:同时为了实现控件焦点切换和对话框默认行为,  VK_TAB.VK_LEFT.VK_RIGHT.VK_UP.VK_DOWN. VK_RETURN.VK_ESCAPE 等键已经被截获处理,因此对话框没有控件时仍然不能完全响应按键消息. 关于热键