Размытие фона полупрозрачной формы (например, Aero glass)

У меня есть форма WPF без границ, без изменения размера (WindowStyle=None, AllowsTransparency=True, ResizeMode=NoResize) с полупрозрачным фоном. Вот изображение того, как форма, полупрозрачный красный прямоугольник, выглядит прямо сейчас, работает поверх Блокнота:

the form as it currently appears on top of Notepad

тем не менее, я хотел бы, чтобы фон был размытым, например, как Aero glass это делает, за исключением всех причудливых границ окна и цветного фона с полосками - я хотел бы справиться с этим себя. Вот макет того, как я хочу, чтобы он выглядел:

the form as I want it to be - blur anything that's below it

Как я могу добиться этого наиболее эффективным способом?

WinForms или WPF меня устраивает. Надеюсь, он должен использовать то же самое, что использует Aero glass (я в порядке с ним, работая только с включенным Aero), а не что-то сумасшедшее, как захват области экрана ниже в качестве растрового изображения и размывание этого.

вот картина того, что я не хочу:

I don't want the entire Aero glass window chrome

Я знаю, что это возможно, и я знаю, как это сделать, но я не хочу, чтобы все окно Aero glass chrome, или границы и строка заголовка, или окно имели пользовательский цвет Aero glass, просто эффект размытия того, что находится под окном/формой.

2 ответов


Если вы хотите использовать размытие Aero, вы можете использовать DwmEnableBlurBehindWindow API-интерфейс. Вот пример производного окна, которое использует это.

public class BlurWindow : Window
{
    #region Constants

    private const int WM_DWMCOMPOSITIONCHANGED = 0x031E;
    private const int DWM_BB_ENABLE = 0x1; 

    #endregion //Constants

    #region Structures
    [StructLayout( LayoutKind.Sequential )]
    private struct DWM_BLURBEHIND
    {
        public int dwFlags;
        public bool fEnable;
        public IntPtr hRgnBlur;
        public bool fTransitionOnMaximized;
    }

    [StructLayout( LayoutKind.Sequential )]
    private struct MARGINS
    {
        public int cxLeftWidth;
        public int cxRightWidth;
        public int cyTopHeight;
        public int cyBottomHeight;
    } 
    #endregion //Structures

    #region APIs

    [DllImport( "dwmapi.dll", PreserveSig = false )]
    private static extern void DwmEnableBlurBehindWindow(IntPtr hwnd, ref DWM_BLURBEHIND blurBehind);

    [DllImport( "dwmapi.dll" )]
    private static extern int DwmExtendFrameIntoClientArea(IntPtr hWnd, ref MARGINS pMargins);

    [DllImport( "dwmapi.dll", PreserveSig = false )]
    private static extern bool DwmIsCompositionEnabled(); 

    #endregion //APIs

    #region Constructor
    public BlurWindow()
    {
        this.WindowStyle = System.Windows.WindowStyle.None;
        this.ResizeMode = System.Windows.ResizeMode.NoResize;
        this.Background = Brushes.Transparent;
    } 
    #endregion //Constructor

    #region Base class overrides
    protected override void OnSourceInitialized(EventArgs e)
    {
        base.OnSourceInitialized( e );

        if ( Environment.OSVersion.Version.Major >= 6 )
        {
            var hwnd = new WindowInteropHelper( this ).Handle;
            var hs = HwndSource.FromHwnd( hwnd );
            hs.CompositionTarget.BackgroundColor = Colors.Transparent;

            hs.AddHook( new HwndSourceHook( this.WndProc ) );
            this.InitializeGlass( hwnd );
        }
    } 
    #endregion //Base class overrides

    #region Methods

    #region InitializeGlass
    private void InitializeGlass(IntPtr hwnd)
    {
        if ( !DwmIsCompositionEnabled() )
            return;

        // fill the background with glass
        var margins = new MARGINS();
        margins.cxLeftWidth = margins.cxRightWidth = margins.cyBottomHeight = margins.cyTopHeight = -1;
        DwmExtendFrameIntoClientArea( hwnd, ref margins );

        // initialize blur for the window
        DWM_BLURBEHIND bbh = new DWM_BLURBEHIND();
        bbh.fEnable = true;
        bbh.dwFlags = DWM_BB_ENABLE;
        DwmEnableBlurBehindWindow( hwnd, ref bbh );
    }
    #endregion //InitializeGlass

    #region WndProc
    private IntPtr WndProc(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled)
    {
        if ( msg == WM_DWMCOMPOSITIONCHANGED )
        {
            this.InitializeGlass( hwnd );
            handled = false;
        }

        return IntPtr.Zero;
    } 
    #endregion //WndProc 

    #endregion //Methods
}

и вот фрагмент использования BlurWindow.

var w = new BlurWindow();
w.Width = 100;
w.Height = 100;
w.MouseLeftButtonDown += (s1, e1) => {
    ((Window)s1).DragMove();
    e1.Handled = true;
};
w.Background = new SolidColorBrush( Color.FromArgb( 75, 255, 0, 0 ) );
w.Show();

Я сделал что-то подобное один раз, но мне не нужно было следующее:

  1. мне не нужно было много двигать мою форму.
  2. никакого движения под моей формой не происходило.

что я сделал:

  1. я использовал, чтобы свернуть мое окно формы на мгновение (программно).
  2. форма используется для захвата изображения snip его размера и в тех же координатах.
  3. установите это изображение как фон после применения BlurBitmapEffect.

Не отличный ответ, я полагаю, но я просто пишу то, что я сделал!

Если вы заинтересованы в этом подходе, эта статья поможет вам: http://www.codeproject.com/Articles/91487/Screen-Capture-in-WPF-WinForms-Application