2

希望这不是太难遵循。

我目前正在开发一个在后台安静运行的小型时间记录应用程序。每次代码运行结束时,应用程序都会提示用户说出自上次提示以来他/她在做什么。我最终会让应用程序将数据写入电子表格。

到目前为止,我拥有的选项之一使用户能够选择他/她是否要使用默认提示设置(每次错过提示时,它都会保持可见,直到创建下一个提示,这意味着如果用户离开他/她的电脑有一段时间可能有相当多的提示坐在屏幕上等待填写)或想要合并所有提示(每次错过一个提示并弹出一个新提示时,旧的提示是关闭,新的覆盖旧提示和新提示的时间)。

用户还可以选择一个复选框来关闭提示。当他/她再次打开提示时,会弹出一个提示,要求用户填写他/她在关闭提示时所做的事情(当用户运行全屏应用程序等时很有用)。

我的问题是,当我尝试生成提示时,它们无法正确显示。我根本无法操纵它们,也没有任何控件显示。它们基本上看起来像空表格。

这是我使用代码生成提示的代码:

public void ticker(object source, System.Timers.ElapsedEventArgs e)
    {
        if (groupMissed)
        {
            incrementsMissed += 1;
            if (incrementsMissed > 1)
            {
                IncrementForm form = (IncrementForm)Application.OpenForms["IncrementForm"];
                if (form.InvokeRequired)
                {
                    form.Invoke(new MethodInvoker(delegate { form.Close(); }));
                }
            }
        }
        else
        {
            incrementsMissed = 1;
        }

        IncrementForm theIncrementForm = new IncrementForm(this, e.SignalTime);
        theIncrementForm.Show();
        latestIncrement = e.SignalTime;
    }

这是我使用“关闭提示”复选框生成提示的代码:

private void chkbxAlerts_Click(object sender, EventArgs e)
    {
        if (!chkbxAlerts.Checked)
        {
            // Ensures that the time missed is covered and restarts the timer
            DateTime now;
            now = DateTime.Now;
            if ((now - latestIncrement).TotalMinutes >= 1) // Only records time if it is equal to or greater than one minute
            {
                // TO-DO: FIX
                if (groupMissed)
                {
                    incrementsMissed += 1;
                    if (incrementsMissed > 1)
                    {
                        IncrementForm form = (IncrementForm)Application.OpenForms["IncrementForm"];
                        if (form.InvokeRequired)
                        {
                            form.Invoke(new MethodInvoker(delegate { form.Close(); }));
                        }
                    }
                }
                else
                {
                    incrementsMissed = 1;
                }
                IncrementForm theIncrementForm = new IncrementForm(this, now, latestIncrement);
                theIncrementForm.Show();
                latestIncrement = now;
            }
            timer.Enabled = true;
        }
        else
        {
            // Stops the timer
            timer.Enabled = false;
        }
    }

如果您需要任何进一步的说明,请告诉我。非常感谢您的帮助,这一直困扰着我。

4

2 回答 2

2

我认为,据我所见,不是 100%,而是您的计时器在一个单独的线程中生成您的窗口(来自计时器代码调用)。

虽然理论上可以工作(看看这个如何在线程中打开表单并强制它保持打开状态
......你最好留在主线程中。

尝试这样的事情......

yourMainWindow.Invoke(new MethodInvoker(() =>
    {
        IncrementForm theIncrementForm = new IncrementForm(this, e.SignalTime);
        theIncrementForm.Show();
        latestIncrement = e.SignalTime;
    }));

...那是你的计时器 - 这样(如我所见)你应该把它全部放在“主线程”上,让事情变得更容易。

希望这可以帮助

于 2012-04-19T01:25:10.417 回答
2

System.Timers.Timer有一个SynchronizingObject属性。如果将其设置为主窗体(或包含计时器的窗体),则将在 GUI 线程上引发计时器滴答事件。

请注意,System.Timers.Timer有吞下Elapsed事件中发生的异常的讨厌习惯。如果您的刻度处理程序抛出异常,您将永远不会看到它。这是一个讨厌的虫子隐藏器。出于这个原因,我建议使用System.Windows.Forms.TimerSystem.Threading.Timer。如果您使用 Windows 窗体计时器,则在 GUI 线程上引发 elapsed 事件。如果您使用System.Threading.Timer,则必须使用InvokeNSGaga 在他的回答中显示的内容。

有关我为什么不鼓励使用System.Timers.Timer.

于 2012-04-19T01:38:18.823 回答