WPF自定义窗口和最大化的大小

时间:2013-02-21 13:28:16

标签: wpf winapi window maximize

我正在开发一个完全自定义的窗口控件并处理它的最大化,我已经覆盖了WindowProc函数并自己处理了WM_GETMINMAXINFO消息。

我填充MINMAXINFO结构并将其发送到窗口句柄并完成。

它适用于我的主显示器,但在我的第二台显示器上,窗户太大了。

我不明白的是,我在第二台显示器上最大化时发送的MINMAXINFO结构具有正确的尺寸信息(1920x1080),但一旦最大化,窗口结束时的尺寸为2160x1100 ..

任何人都知道发生了什么事?

非常感谢。

3 个答案:

答案 0 :(得分:1)

@Karnalta - 我一段时间做了类似的事。看看我的an old blog post。我在上面的链接中考虑了可下载样本中的任务栏和任务栏的自动隐藏。我的两个显示器都是相同的尺寸,所以它似乎对我来说很完美,但尝试一下。以下是您可能最感兴趣的两种方法。

private static void WmGetMinMaxInfo(IntPtr hwnd, IntPtr lParam)
{
    MINMAXINFO mmi = (MINMAXINFO)Marshal.PtrToStructure(lParam, typeof(MINMAXINFO));
    IntPtr monitorContainingApplication = MonitorFromWindow(hwnd, MONITOR_DEFAULTTONEAREST);

    if (monitorContainingApplication != System.IntPtr.Zero)
    {
        MONITORINFO monitorInfo = new MONITORINFO();
        GetMonitorInfo(monitorContainingApplication, monitorInfo);
        RECT rcWorkArea = monitorInfo.rcWork;
        RECT rcMonitorArea = monitorInfo.rcMonitor;
        mmi.ptMaxPosition.x = Math.Abs(rcWorkArea.left - rcMonitorArea.left);
        mmi.ptMaxPosition.y = Math.Abs(rcWorkArea.top - rcMonitorArea.top);
        mmi.ptMaxSize.x = Math.Abs(rcWorkArea.right - rcWorkArea.left);
        mmi.ptMaxSize.y = Math.Abs(rcWorkArea.bottom - rcWorkArea.top);
        mmi.ptMaxTrackSize.x = mmi.ptMaxSize.x;                                                 //maximum drag X size for the window
        mmi.ptMaxTrackSize.y = mmi.ptMaxSize.y;                                                 //maximum drag Y size for the window
        mmi.ptMinTrackSize.x = 800;                                                             //minimum drag X size for the window
        mmi.ptMinTrackSize.y = 600;                                                             //minimum drag Y size for the window
        mmi = AdjustWorkingAreaForAutoHide(monitorContainingApplication, mmi);                  //need to adjust sizing if taskbar is set to autohide
    }
    Marshal.StructureToPtr(mmi, lParam, true);
}

private static MINMAXINFO AdjustWorkingAreaForAutoHide(IntPtr monitorContainingApplication, MINMAXINFO mmi)
{
    IntPtr hwnd = FindWindow("Shell_TrayWnd", null);
    if (hwnd == null) return mmi;
    IntPtr monitorWithTaskbarOnIt = MonitorFromWindow(hwnd, MONITOR_DEFAULTTONEAREST);
    if (!monitorContainingApplication.Equals(monitorWithTaskbarOnIt)) return mmi;
    APPBARDATA abd = new APPBARDATA();
    abd.cbSize = Marshal.SizeOf(abd);
    abd.hWnd = hwnd;
    SHAppBarMessage((int)ABMsg.ABM_GETTASKBARPOS, ref abd);
    int uEdge = GetEdge(abd.rc);
    bool autoHide = System.Convert.ToBoolean(SHAppBarMessage((int)ABMsg.ABM_GETSTATE, ref abd));

    if (!autoHide) return mmi;

    switch (uEdge)
    {
        case (int)ABEdge.ABE_LEFT:
            mmi.ptMaxPosition.x += 2;
            mmi.ptMaxTrackSize.x -= 2;
            mmi.ptMaxSize.x -= 2;
            break;
        case (int)ABEdge.ABE_RIGHT:
            mmi.ptMaxSize.x -= 2;
            mmi.ptMaxTrackSize.x -= 2;
            break;
        case (int)ABEdge.ABE_TOP:
            mmi.ptMaxPosition.y += 2;
            mmi.ptMaxTrackSize.y -= 2;
            mmi.ptMaxSize.y -= 2;
            break;
        case (int)ABEdge.ABE_BOTTOM:
            mmi.ptMaxSize.y -= 2;
            mmi.ptMaxTrackSize.y -= 2;
            break;
        default:
            return mmi;
    }
    return mmi;
}

答案 1 :(得分:0)

不是使用win32事件,而是让WPF自己最大化窗口(使用WindowState = System.Windows.WindowState.Maximized;)并修改窗口内容模板以适合您想要的大小,这可能更容易。

例如,您可以将窗口内容封装在网格中,并将边距设置为此网格。 最后,然后将窗口的背景属性(如果有)移动到网格,并为窗口背景设置透明颜色。

答案 2 :(得分:0)

听起来你根据第二台显示器的尺寸提供尺寸。 documentation for WM_GETMINMAXINFO表示宽度/高度是根据主监视器的大小表示的,即使窗口将显示在具有不同分辨率的不同监视器中。