WS_EX_NOACTIVATE 和 WinForms 的两个问题
Two issues with WS_EX_NOACTIVATE and WinForms
在我的一些应用程序中,我使用 WS_EX_NOACTIVATE extended window style value(例如创建虚拟键盘或在另一个程序中托管的表单)。此值可防止表单获得焦点。
以下是我的处理方式:
protected override CreateParams CreateParams
{
get
{
CreateParams p = base.CreateParams;
p.ExStyle |= Win32.WS_EX_NOACTIVATE;
return p;
}
}
它运行良好,但我注意到此解决方案有两个问题:
- 用户无法通过按 "Tab"
在控件之间移动
- 当用户拖动窗体时,我们看不到位移(移动过程中窗体不重绘)
那么,是否可以解决这些问题,或者至少实施一些替代方案?也许 WS_EX_NOACTIVATE 不是最好的解决方案?
非常感谢!
以下是我为每个问题找到的解决方案:
问题 #1:
protected override bool ProcessKeyPreview(ref Message m)
{
if (m.Msg == Win32.WM_KEYDOWN && (Keys)m.WParam == Keys.Tab)
{
if (Control.ModifierKeys == Keys.Shift)
{
this.SelectNextControl(ActiveControl, false, true, true, true); // Bring focus to previous control
}
else
{
this.SelectNextControl(ActiveControl, true, true, true, true); // Bring focus to next control
}
}
return base.ProcessKeyPreview(ref m);
}
问题 #2:您必须拦截从系统(WM_SIZING
和 WM_MOVING
)接收到的适当消息,并使用 SetWindowPos()
- 这将迫使它移动!
在您的表单中 class:
public const int WM_SIZING = 0x0214;
public const int WM_MOVING = 0x0216;
[StructLayout(LayoutKind.Sequential)]
public struct RECT
{
public int Left;
public int Top;
public int Right;
public int Bottom;
}
[DllImport("user32.dll", SetLastError = true)]
private static extern bool SetWindowPos(IntPtr hWnd, IntPtr hWndInstertAfter, int x, int y, int cx, int cy, uint flags);
protected override void WndProc(ref Message m)
{
if (m.Msg == WM_SIZING || m.Msg == WM_MOVING)
{
RECT rect = (RECT)Marshal.PtrToStructure(m.LParam, typeof(RECT));
SetWindowPos(this.Handle, IntPtr.Zero, rect.Left, rect.Top, rect.Right - rect.Left, rect.Bottom - rect.Top, 0);
}
else
{
base.WndProc(ref m);
}
}
在我的一些应用程序中,我使用 WS_EX_NOACTIVATE extended window style value(例如创建虚拟键盘或在另一个程序中托管的表单)。此值可防止表单获得焦点。
以下是我的处理方式:
protected override CreateParams CreateParams
{
get
{
CreateParams p = base.CreateParams;
p.ExStyle |= Win32.WS_EX_NOACTIVATE;
return p;
}
}
它运行良好,但我注意到此解决方案有两个问题:
- 用户无法通过按 "Tab" 在控件之间移动
- 当用户拖动窗体时,我们看不到位移(移动过程中窗体不重绘)
那么,是否可以解决这些问题,或者至少实施一些替代方案?也许 WS_EX_NOACTIVATE 不是最好的解决方案?
非常感谢!
以下是我为每个问题找到的解决方案:
问题 #1:
protected override bool ProcessKeyPreview(ref Message m)
{
if (m.Msg == Win32.WM_KEYDOWN && (Keys)m.WParam == Keys.Tab)
{
if (Control.ModifierKeys == Keys.Shift)
{
this.SelectNextControl(ActiveControl, false, true, true, true); // Bring focus to previous control
}
else
{
this.SelectNextControl(ActiveControl, true, true, true, true); // Bring focus to next control
}
}
return base.ProcessKeyPreview(ref m);
}
问题 #2:您必须拦截从系统(WM_SIZING
和 WM_MOVING
)接收到的适当消息,并使用 SetWindowPos()
- 这将迫使它移动!
在您的表单中 class:
public const int WM_SIZING = 0x0214;
public const int WM_MOVING = 0x0216;
[StructLayout(LayoutKind.Sequential)]
public struct RECT
{
public int Left;
public int Top;
public int Right;
public int Bottom;
}
[DllImport("user32.dll", SetLastError = true)]
private static extern bool SetWindowPos(IntPtr hWnd, IntPtr hWndInstertAfter, int x, int y, int cx, int cy, uint flags);
protected override void WndProc(ref Message m)
{
if (m.Msg == WM_SIZING || m.Msg == WM_MOVING)
{
RECT rect = (RECT)Marshal.PtrToStructure(m.LParam, typeof(RECT));
SetWindowPos(this.Handle, IntPtr.Zero, rect.Left, rect.Top, rect.Right - rect.Left, rect.Bottom - rect.Top, 0);
}
else
{
base.WndProc(ref m);
}
}