WPF:让窗口不可靠,但保持框架?

时间:2010-08-02 09:26:30

标签: c# wpf windows aero window-resize

我有一个没有标题栏的窗口(WindowStyle == WindowStyle.None)。整个窗口使用Aero玻璃效果。当我使窗口无法实现(ResizeMode == ResizeMode.NoResize)时,玻璃效果消失,我的控件只悬挂在半空中。 (基本上,窗口本身会消失,但会留下内容。)

有没有办法让我在不摆脱窗框的情况下使窗户不可靠?


我已经阅读了问题Enable Vista glass effect on a borderless WPF window,但这不是我想要的 - 我想保留窗口边框。有关我希望窗口看起来像什么的示例,请在启用Aero的情况下按Alt + Tab。


澄清一下,当我将鼠标悬停在窗口边框上时,我根本不希望显示调整大小的光标。这基本上就是我希望我的窗口看起来像:

Projector http://i37.tinypic.com/2mg4jty.png

解决方案不一定是严格的WPF - 我很好地利用Win32 API来实现这一点。

4 个答案:

答案 0 :(得分:9)

您可以挂钩wndproc并拦截WM_WINDOWPOSCHANGING消息。不是严格的WPF,但可能是你最好的选择。

如果你想隐藏调整大小的游标,那么最好的办法是拦截WM_NCHITTEST。调用DefWindowProc(获取默认行为),并测试返回值;如果它是HTBOTTOM,HTBOTTOMLEFT,HTBOTTOMRIGHT,HTTOP,HTTOPLEFT或HTTOPRIGHT,则将返回值更改为HTBORDER。

答案 1 :(得分:9)

基于Erics的答案。

Example Image

public partial class MainWindow : Window
{
    [DllImport("DwmApi.dll")]
    public static extern int DwmExtendFrameIntoClientArea(
        IntPtr hwnd,
        ref MARGINS pMarInset);

    [DllImport("user32.dll", CharSet = CharSet.Auto)]
    public static extern IntPtr DefWindowProc(
        IntPtr hWnd,
        int msg,
        IntPtr wParam,
        IntPtr lParam);

    private const int WM_NCHITTEST = 0x0084;
    private const int HTBORDER = 18;
    private const int HTBOTTOM = 15;
    private const int HTBOTTOMLEFT = 16;
    private const int HTBOTTOMRIGHT = 17;
    private const int HTLEFT = 10;
    private const int HTRIGHT = 11;
    private const int HTTOP = 12;
    private const int HTTOPLEFT = 13;
    private const int HTTOPRIGHT = 14;

    public MainWindow()
    {
        InitializeComponent();

        this.Loaded += new RoutedEventHandler(MainWindow_Loaded);
    }

    void MainWindow_Loaded(object sender, RoutedEventArgs e)
    {
        try
        {
            // Obtain the window handle for WPF application
            IntPtr mainWindowPtr = new WindowInteropHelper(this).Handle;
            HwndSource mainWindowSrc = HwndSource.FromHwnd(mainWindowPtr);
            mainWindowSrc.CompositionTarget.BackgroundColor = Color.FromArgb(0, 0, 0, 0);
            mainWindowSrc.AddHook(WndProc);

            // Set Margins
            MARGINS margins = new MARGINS();
            margins.cxLeftWidth = 10;
            margins.cxRightWidth = 10;
            margins.cyBottomHeight = 10;
            margins.cyTopHeight = 10;

            int hr = DwmExtendFrameIntoClientArea(mainWindowSrc.Handle, ref margins);
            //
            if (hr < 0)
            {
                //DwmExtendFrameIntoClientArea Failed
            }
        }
        // If not Vista, paint background white.
        catch (DllNotFoundException)
        {
            Application.Current.MainWindow.Background = Brushes.White;
        }
    }

    private IntPtr WndProc(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled)
    {
        // Override the window hit test
        // and if the cursor is over a resize border,
        // return a standard border result instead.
        if (msg == WM_NCHITTEST)
        {
            handled = true;
            var htLocation = DefWindowProc(hwnd, msg, wParam, lParam).ToInt32();
            switch (htLocation)
            {
                case HTBOTTOM:
                case HTBOTTOMLEFT:
                case HTBOTTOMRIGHT:
                case HTLEFT:
                case HTRIGHT:
                case HTTOP:
                case HTTOPLEFT:
                case HTTOPRIGHT:
                    htLocation = HTBORDER;
                    break;
            }

            return new IntPtr(htLocation);
        }

        return IntPtr.Zero;
    }

    private void Button_Click(object sender, RoutedEventArgs e)
    {
        this.Close();
    }
}

[StructLayout(LayoutKind.Sequential)]
public struct MARGINS
{
    public int cxLeftWidth;      // width of left border that retains its size
    public int cxRightWidth;     // width of right border that retains its size
    public int cyTopHeight;      // height of top border that retains its size
    public int cyBottomHeight;   // height of bottom border that retains its size
};

<Window x:Class="WpfApplication1.MainWindow"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    Title="MainWindow" Height="150" Width="200" 
    Background="Transparent"
    WindowStyle="None"
    ResizeMode="CanResize"
>
    <Grid Background="White" Margin="10,10,10,10">
        <Button Content="Go Away" Click="Button_Click" Height="20" Width="100" />
    </Grid>
</Window>

答案 2 :(得分:1)

这样做的一个hackish方法是设置MinWidth / MaxWidth和MinHeight / MaxHeight属性以有效地使其无法实现。当然,问题是你仍然可以在边界上调整大小。

答案 3 :(得分:1)

为什么不为窗口创建此窗口边框? 它使用偏移量来设置窗口的颜色。 所以,一个简单的方法就是将整个边框包裹在窗户周围,最重要的是你得到自己的颜色!