侧边栏壁纸
博主头像
陌上花 博主等级

回首万事皆休

  • 累计撰写 69 篇文章
  • 累计创建 11 个标签
  • 累计收到 0 条评论

目 录CONTENT

文章目录

Dev之等待窗口

种向日葵的人
2024-08-30 / 0 评论 / 0 点赞 / 15 阅读 / 0 字

Why

  • 主要是公司需要实现的需求中有关于这种的情况,需要在后台操作的时候,显示一个页面阻塞UI主线程,然后后台继续跑算法模块。因为项目比较老,所以这一块估计经手了一些人导致内部冗余不堪,画了一两天时候看了代码我这边重新整合了一部分,同时之前未曾实现的部分也补上了。
  • 主要使用的是DevSplashScreenManager管理,省去了一部分工作,公司代码不方便展示,所以我目前这边自己剥离了一份出来。

Code

等待窗体

  • 那么对于两种窗体分别创建子,第一种必须是继承SplashScreen,以下是设计代码
public WaitScreenForm()
{
    InitializeComponent();
}

private void simpleButton1_Click(object sender, EventArgs e)
{
	//这个就是中途取消的按钮
    SplashScreenManager.CloseForm();

    SplashScreenManager.Default?.CloseWaitForm();

	//这个助手类后续会说到 表示调用所有窗体取消的事件
    WaitHelper.Cancel();
	//这个助手类后续会说到 表示重置初始状态
    WaitHelper.Reset();

}

public override void ProcessCommand(Enum cmd, object arg)
{
	//用于实时展示状态属性在变化 后续可以自行添加代码逻辑
    this.groupControl1.Text = DateTime.Now.ToString();


    base.ProcessCommand(cmd, arg);


}
  • 以下是窗体设计的界面
    WaitScreenForm.png
  • 第二种相对比较简单,只需要继承自WaitForm即可
 public partial class MyWaitForm : WaitForm
 {
     public MyWaitForm()
     {
         InitializeComponent();
     }
	
 }
  • 以下是窗体设计
    MyWaiteForm.png

助手类设计

  • 首先设计的助手类是静态类,内部有一个开始等待的函数,传入一个需要锁定的主窗体和是否是等待窗体。
public static class WaitHelper
{
	//在是第一种窗体的时候绑定的取消事件
    public static event Action OnWaitEventHandler;
    //在等待时需要锁定的父窗体
    public static Form parentForm = null;
    //开始等待
    public static void BeginWait(Form _parentForm, bool isWait = true)
    {
        parentForm = _parentForm;
        parentForm.Enabled = false;
        if (isWait)
        {
            SplashScreenManager.ShowForm(_parentForm, typeof(MyWaitForm), true, true, true, ParentFormState.Locked);
        }
        else
        {
            SplashScreenManager.ShowForm(_parentForm, typeof(WaitScreenForm), true, true, true, ParentFormState.Locked);

        }
    }

	//取消
    public static void Cancel()
    {
        OnWaitEventHandler?.Invoke();
    }
	//设置默认状态
    public static void Reset()
    {
        OnWaitEventHandler = null;
		//此处必须调用invoke
        parentForm.Invoke(new Action(() =>
        {
            parentForm.Enabled = true;
            parentForm = null;
        }));

    }
}

主窗体

  • 主窗体的测试代码,按照需求一定是需要使用线程的,我测试过thread,taskBackgroundWorker,后面两个其实都无法即时的取消线程,不管是task配合CancellationTokenSource还是BackgroundWorker的取消,都需要在循环内部使用一定的判断,因为其实并不是很推荐在运算时直接终端线程,这样会导致数据问题和未知的错误。
    //private CancellationTokenSource tokenSource = new CancellationTokenSource();
    private Thread thread = null;
    public MainForm()
    {
        InitializeComponent();
    }
    private void simpleButton1_Click(object sender, EventArgs e)
    {
        SplashScreenManager.ShowForm(this, typeof(WaitScreenForm), true, true, true, ParentFormState.Locked);

    }


    private void simpleButton2_Click(object sender, EventArgs e)
    {


        SplashScreenManager.CloseForm(true, 0, this);

    }

    private void simpleButton3_Click(object sender, EventArgs e)
    {
        //tokenSource = new CancellationTokenSource();
        WaitHelper.OnWaitEventHandler += Cancel;


        thread = new Thread(() =>
       {
           try
           {
               this.Invoke(new Action(() =>
               {
                   WaitHelper.BeginWait(this, false);

               }));
               while (true)
               {
                   Debug.WriteLine(DateTime.Now);
                   Task.Delay(1000).Wait();
                   //此处可以自定义  然后传递到ProcessCommand
                   //WaitHelper.SendCmd(WaitEnum.Wait);
               }
           }
           catch (ThreadAbortException ex)
           {

           }


       });
        thread.Start();
    }

    private void Cancel()
    {
        thread.Abort();

    }



}
  • 前面两个按钮主要是测试直接使用SplashScreenManager显示和关闭的。

Result

MainForm.png
WaitShow.png

  • 点击取消可以直接中断。

Review

  • 这个看起来是不是很简单,而且代码也少,但是你知道我这个前辈之前实现用了多少代码吗?明明一个事件可以解决的中断问题,愣是写了四五个,跳来跳去的。不得不说,如果后人看到我写的代码会不会也有这种感觉?说不定以后一个字段就可以解决了。
0
博主关闭了所有页面的评论