1

我有一个带有两个子无模式表单的主表单,例如所有表单都可以同时处于活动状态:

class MainForm : Form
{
    Form child1;
    Form child2;

    public MainForm()
    {
        Text = "MainForm";
        child1 = new Form { Text = "Child1" };
        child2 = new Form { Text = "Child2" };
        child1.Show(this);
        child2.Show(this);
    }
}

我想让用户Alt+Tab进入所有这些,但令人惊讶的是,我发现如果任何子表单处于活动状态,则无法从Alt+Tab菜单中选择所有者表单。

所有三个表单都显示在列表中,但显然当您选择所有者窗口并且有一个活动的孩子时,选择的是孩子而不是所有者。在任务栏中选择表单时也会发生同样的事情。

我错过了什么吗?我开始考虑显式配置快捷键以允许从无模式子窗体导航到所有者窗口,但在此之前我想确认是否已经有一些内置键盘快捷键可以执行此操作,因为我不想打破用户的期望。

令人惊讶的是,我找不到任何提到这种行为的问题,我也觉得这很奇怪。

4

2 回答 2

2

设置窗体的所有者,使该窗体作为非模式窗口保持在其所有者之上。
如果拥有的Form 的ShowInTaskbar属性设置为true,则用于在系统中迭代打开的 Windows 的标准键ALT+TABWIN+TAB键组合将显示(激活)下一个拥有的 Form 而不是 Owner。
激活哪个窗体,取决于窗体在任务栏中的当前位置。

如果ShowInTaskbar子项的属性设置为false,则激活所有者表单。
需要注意的是,如果可以最小化子窗体,可以观察到一些尴尬的行为:Alt 或 Control-tabbing,导致子窗体以不愉快的方式出现和消失。

无论如何,可以使用标准组合CONTROL+F6键将焦点移到打开的子窗体(以及在这种布局中的所有者窗体)上,如果需要(如果窗体最小化),也可以将它们放在前面。
在这里,覆盖ProcessCmdKey ,因此无论哪个子控件捕获了光标 ,都会拦截键的组合。

此处的代码激活一个同时按下CONTROL+F6和的表单CONTROL+SHIFT+F6,将焦点移动到每个打开的子表单和所有者。当最小化子表单(或所有子表单)时,它也可以工作。

在所有者表格中:

protected override bool ProcessCmdKey(ref Message msg, Keys keyData)
{
    bool isControlF6 = keyData == (Keys.Control | Keys.F6);
    bool isCtrlShiftF6 = keyData == (Keys.Control | Keys.Shift | Keys.F6);

    if (isControlF6 || isCtrlShiftF6)
    {
        Form frm = isCtrlShiftF6 
                 ? Application.OpenForms.OfType<Form>().LastOrDefault(f => f.Owner == this)
                 : Application.OpenForms.OfType<Form>().FirstOrDefault(f => f.Owner == this);
        if (frm is null) return true;
        frm.WindowState = FormWindowState.Normal;
        frm.Focus();
        return true;
    }
    return base.ProcessCmdKey(ref msg, keyData);
}

在子窗体中:

protected override bool ProcessCmdKey(ref Message msg, Keys keyData)
{
    bool isControlF6 = keyData == (Keys.Control | Keys.F6);
    bool isCtrlShiftF6 = keyData == (Keys.Control | Keys.Shift | Keys.F6);
    if (isControlF6 || isCtrlShiftF6) {
        int frmNext = 0;
        var formsList = Application.OpenForms.OfType<Form>()
                                   .Where(f => (f.Owner == this.Owner) || (f == this.Owner)).ToList();
        for (int i = 0; i < formsList.Count; i++) {
            if (formsList[i] == this) {
                if (isCtrlShiftF6) { frmNext = i == 0 ? formsList.Count - 1 : i - 1; }
                if (isControlF6) { frmNext = i == formsList.Count - 1 ? 0 : i + 1; }
                formsList[frmNext].WindowState = FormWindowState.Normal;
                formsList[frmNext].Focus();
                return true;
            }
        }
    }
    return base.ProcessCmdKey(ref msg, keyData);
}
于 2019-02-25T01:17:31.263 回答
0

编辑答案:

我不知道为什么,但我就是放不下。似乎应该有一个简单的解决方案。

@glopes 我相信这是您根据您的评论正在寻找的东西。

此代码将在子窗口失去焦点之前将焦点设置回父窗口。这类似于单击窗口,并允许您使用 Alt-Tab 键切换到您想要的任何窗口。

public class MainForm : Form
{
    Form child1;
    Form child2;

    public MainForm()
    {
        Text = "MainForm";
        child1 = new ChildForm { Text = "Child1", ParentPtr = Handle };
        child2 = new ChildForm { Text = "Child2", ParentPtr = Handle };
        child1.Show(this);
        child2.Show(this);
    }
}

public class ChildForm : Form
{
    [DllImport("user32.dll")]
    public static extern bool SetFocus(IntPtr hWnd);

    private const int WM_KILLFOCUS = 0x0008;

    public IntPtr ParentPtr { get; set; }

    protected override void WndProc(ref Message m)
    {
        if (m.Msg == WM_KILLFOCUS) SetFocus(ParentPtr);

        base.WndProc(ref m);
    }
}
于 2019-02-24T17:30:02.397 回答