1

我有一个带有 TabStrip 控件的 Winforms 应用程序。在运行时,用户控件将被动态加载到不同的选项卡中。

我想在加载 UserControl 之前向用户显示“用户控件 xyz 正在加载”消息(将现有标签设置为可见并更改其文本),直到加载完全完成。

到目前为止我的方法:

  1. 试图在 BackgroundWorker 线程中加载用户控件。这失败了,因为我必须在加载 UserControl 期间访问 Gui-Controls
  2. 试图在 BackgroundWorker 线程中显示消息。这显然失败了,因为 BackgroundWorker 线程不是 UI 线程 ;-)
  3. 显示消息,调用 DoEvents(),加载 UserControl。每次我加载 UserControl 时,这都会导致不同的行为(闪烁,...),并且我无法控制何时以及如何将其再次设置为不可见。

总结一下,我有两个问题:

  1. 在加载用户控件之前如何确保消息直接可见
  2. 如何确保消息再次设置为不可见,就在 UserControl 完全加载的那一刻(包括所有 DataBindings、网格格式等)
4

3 回答 3

3

我们使用的类似这样:

  1. 创建一个新表单,其中包含您想向用户展示的任何内容,
  2. 实现一个静态方法,您可以在其中调用此表单以在其内部创建,以防止内存泄漏
  3. 在此表单中创建一个新线程,以便表单在单独的线程中运行并保持响应;我们使用一个显示进度条填充的ajax 控件。

在用于启动线程的方法中,将其属性设置为 topmost true 以确保它保持在顶部。

例如,在您的主要形式中执行此操作:

loadingForm.ShowLoadingScreen("usercontrollname");
//do something
loadingform.CloseLoadingScreen();

在加载表单类中;

public LoadingScreen()
{
  InitializeComponent();
}

public static void ShowLoadingScreen(string usercontrollname)
{
  // do something with the usercontroll name if desired
  if (_LoadingScreenThread == null)
  {
    _LoadingScreenThread = new Thread(new ThreadStart(DoShowLoadingScreen));
    _LoadingScreenThread.IsBackground = true;
    _LoadingScreenThread.Start();
  }
}

public static void CloseLoadingScreen()
{
  if (_ls.InvokeRequired)
  {
    _ls.Invoke(new MethodInvoker(CloseLoadingScreen));
  }
  else
  {
    Application.ExitThread();
    _ls.Dispose();
    _LoadingScreenThread = null;
  }
}

private static void DoShowLoadingScreen()
{
    _ls = new LoadingScreen();
    _ls.FormBorderStyle = FormBorderStyle.None;
    _ls.MinimizeBox = false;
    _ls.ControlBox = false;
    _ls.MaximizeBox = false;
    _ls.TopMost = true;
    _ls.StartPosition = FormStartPosition.CenterScreen;

  Application.Run(_ls);
}
于 2012-08-08T09:49:47.327 回答
1

根据@wterbeek 的示例,我出于自己的目的修改了该类:

  • 将其居中加载表格
  • 修改其不透明度
  • 将其调整为父大小
  • 将其显示为对话框并阻止所有用户交互
  • 我被要求展示一个颤抖的人

我在线收到一个空错误:

if (_ls.InvokeRequired)

所以我添加了一个 _shown 条件(如果动作完成得如此之快以至于 _LoadingScreenThread 线程甚至没有运行)来检查表单是否存在。

此外,如果 _LoadingScreenThread 没有启动,Application.Exit 将关闭主线程。

我想发布它可能会帮助其他人。代码中的注释将解释更多。

public partial class LoadingScreen : Form {

    private static Thread _LoadingScreenThread;

    private static LoadingScreen _ls;

    //condition required to check if the form has been loaded
    private static bool _shown = false;

    private static Form _parent;

    public LoadingScreen() {
        InitializeComponent();
    }
    //added the parent to the initializer
    //CHECKS FOR NULL HAVE NOT BEEN IMPLEMENTED
    public static void ShowLoadingScreen(string usercontrollname, Form parent) {
        // do something with the usercontroll name if desired
        _parent = parent;
        if (_LoadingScreenThread == null) {
            _LoadingScreenThread = new Thread(new ThreadStart(DoShowLoadingScreen));
            _LoadingScreenThread.SetApartmentState(ApartmentState.STA);
            _LoadingScreenThread.IsBackground = true;
            _LoadingScreenThread.Start();
        }
    }

    public static void CloseLoadingScreen() {
        //if the operation is too short, the _ls is not correctly initialized and it throws 
        //a null error
        if (_ls!=null && _ls.InvokeRequired) {
            _ls.Invoke(new MethodInvoker(CloseLoadingScreen));
        } else {
            if (_shown)
            {
                //if the operation is too short and the thread is not started
                //this would close the main thread
                _shown = false;
                Application.ExitThread();
            }
            if (_LoadingScreenThread != null)
                _LoadingScreenThread.Interrupt();
            //this check prevents the appearance of the loader
            //or its closing/disposing if shown
            //have not found the answer
            //if (_ls !=null)
            //{
               _ls.Close();
               _ls.Dispose();
            //}
            _LoadingScreenThread = null;
        }
    }

    private static void DoShowLoadingScreen() {
        _ls = new LoadingScreen();
        _ls.FormBorderStyle = FormBorderStyle.None;
        _ls.MinimizeBox = false;
        _ls.ControlBox = false;
        _ls.MaximizeBox = false;
        _ls.TopMost = true;
        //get the parent size
        _ls.Size = _parent.Size; 
        //get the location of the parent in order to show the form over the 
        //target form
        _ls.Location = _parent.Location;     
        //in order to use the size and the location specified above
        //we need to set the start position to "Manual"
        _ls.StartPosition =FormStartPosition.Manual;
        //set the opacity
        _ls.Opacity = 0.5;
        _shown = true;
        //Replaced Application.Run with ShowDialog to show as dialog
        //Application.Run(_ls);
        _ls.ShowDialog();
    }
}
于 2013-03-28T07:27:42.343 回答
1

再次尝试第二种方法:

试图在 BackgroundWorker 线程中显示消息。这显然失败了,因为 BackgroundWorker 线程不是 UI 线程 ;-)

但是这一次,在后台线程中使用以下代码来更新您的标签:

label.Invoke((MethodInvoker) delegate {
    label.Text = "User Control xyz is loading";
    label.Visible = true;
});
// Load your user control
// ...
label.Invoke((MethodInvoker) delegate {
    label.Visible = false;
});

Invoke允许您在另一个线程中更新您的 UI。

于 2012-08-08T09:53:45.640 回答