DropShadow para ventana sin bordes WPF

Tengo una ventana de WPF con WindowStyle configurado en none. ¿Hay alguna forma en que pueda forzar a esta ventana a soltar una sombra (como la que obtienes cuando WindowStyle no es ninguna)? No quiero establecer AllowTransparency en verdadero, porque afecta el rendimiento. Y tampoco quiero deshabilitar la representación de hardware (leí en alguna parte que la transparencia funciona mejor con ella deshabilitada).

He escrito una pequeña clase de utilidad que es capaz de hacer exactamente lo que quiere: colocar una sombra estándar sobre una Window sin AllowsTransparency pero con AllowsTransparency establecido en false .

Solo tiene que llamar al DropShadowToWindow(Window window) . Se prefiere que realice esta llamada justo después del InitializeComponent() del constructor de la ventana, pero funcionará incluso si la llama después de que se muestre la ventana.

 using System; using System.Drawing.Printing; using System.Runtime.InteropServices; using System.Windows; using System.Windows.Interop; public static class DwmDropShadow { [DllImport("dwmapi.dll", PreserveSig = true)] private static extern int DwmSetWindowAttribute(IntPtr hwnd, int attr, ref int attrValue, int attrSize); [DllImport("dwmapi.dll")] private static extern int DwmExtendFrameIntoClientArea(IntPtr hWnd, ref Margins pMarInset); ///  /// Drops a standard shadow to a WPF Window, even if the window is borderless. Only works with DWM (Windows Vista or newer). /// This method is much more efficient than setting AllowsTransparency to true and using the DropShadow effect, /// as AllowsTransparency involves a huge performance issue (hardware acceleration is turned off for all the window). ///  /// Window to which the shadow will be applied public static void DropShadowToWindow(Window window) { if (!DropShadow(window)) { window.SourceInitialized += new EventHandler(window_SourceInitialized); } } private static void window_SourceInitialized(object sender, EventArgs e) { Window window = (Window)sender; DropShadow(window); window.SourceInitialized -= new EventHandler(window_SourceInitialized); } ///  /// The actual method that makes API calls to drop the shadow to the window ///  /// Window to which the shadow will be applied /// True if the method succeeded, false if not private static bool DropShadow(Window window) { try { WindowInteropHelper helper = new WindowInteropHelper(window); int val = 2; int ret1 = DwmSetWindowAttribute(helper.Handle, 2, ref val, 4); if (ret1 == 0) { Margins m = new Margins { Bottom = 0, Left = 0, Right = 0, Top = 0 }; int ret2 = DwmExtendFrameIntoClientArea(helper.Handle, ref m); return ret2 == 0; } else { return false; } } catch (Exception ex) { // Probably dwmapi.dll not found (incompatible OS) return false; } } } 

Utilice la biblioteca de integración de Microsoft WPF Shell , más fácil y con un mejor rendimiento

La respuesta de Patrick funciona muy bien, excepto cuando se aloja una ventana win32. Cuando eso ocurre, observa que la ventana alojada está “lavada” (parece que Windows está aplicando el efecto de “hoja de vidrio” a toda la ventana alojada). Este comportamiento extraño se soluciona cuando se define la estructura localmente, por ejemplo

 [StructLayout(LayoutKind.Sequential)] public struct Margins { public int Left; public int Right; public int Top; public int Bottom; } 

Si permite que la ventana ResizeMode tamaño de los bordes, configurando ResizeMode en CanResize , obtendrá la sombra CanResize del sistema operativo. Luego puede establecer MaxWidth , MinWidth , MaxHeight y MinHeight en valores que evitarán el cambio de tamaño.

Si tiene una ventana sin bordes sin un estilo, tendrá que proporcionar toda la apariencia de la ventana en su propio árbol visual, incluida una sombra paralela, ya que esta combinación de configuraciones equivale a decir que no quiere lo que el sistema operativo proporciona.

EDITAR:

Desde ese punto, si el tamaño de su ventana es fijo, simplemente agregue la sombra de gotas, tal vez como un como el primer elemento en el contenido de un

algo como esto:

       

Tenga en cuenta que la propiedad Fill de ese primer Rectangle es parcialmente transparente, lo que también podría hacer con la propiedad Opacity del Rectangle . Puede usar un gráfico propio o una forma diferente para personalizar el aspecto de la sombra paralela.

Tenga en cuenta que esto infringe su requisito de que AllowsTransparency sea False , pero no tiene otra opción: si desea transparencia, debe permitirla.

¿Por qué no crear la sombra con el mismo objeto que su “ventana” pero más grande y detrás de ella?

          

O si necesita una barra de título transparente, podría reemplazarse por un

          

Editar: Me acabo de dar cuenta de que OP quiere que AllowsTransparency se establezca en False. No puedo ver que una sombra funcione sin que sea “Verdadera”, por decirlo de alguna manera.

    Intereting Posts