WPF 实现Popup不在最上层显示、随窗口移动

文章目录


前言

由于WPF 默认的Popup总是显示在所有窗口的前面,如何让popup 层只显示在该父级之上,并随着父级而动呢?下面来看实现。


一、定义类继承Popup类

c 复制代码
public class PopupEx : Popup
{
    /// <summary>  
    /// 是否窗口随动,默认为随动(true)  
    /// </summary>  
    public bool IsPositionUpdate
    {
        get { return (bool)GetValue(IsPositionUpdateProperty); }
        set { SetValue(IsPositionUpdateProperty, value); }
    }

    public static readonly DependencyProperty IsPositionUpdateProperty =
        DependencyProperty.Register("IsPositionUpdate", typeof(bool), typeof(PopupEx), new PropertyMetadata(true, new PropertyChangedCallback(IsPositionUpdateChanged)));

    private static void IsPositionUpdateChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
    {
        (d as PopupEx).pup_Loaded(d as PopupEx, null);
    }

    /// <summary>  
    /// 加载窗口随动事件  
    /// </summary>  
    public PopupEx()
    {
        this.Loaded += pup_Loaded;
    }

    /// <summary>  
    /// 加载窗口随动事件  
    /// </summary>  
    private void pup_Loaded(object sender, RoutedEventArgs e)
    {
        Popup pup = sender as Popup;
        var win = VisualTreeHelper.GetParent(pup);
        while (win != null && (win as Window) == null)
        {
            win = VisualTreeHelper.GetParent(win);
        }
        if ((win as Window) != null)
        {
            (win as Window).LocationChanged -= PositionChanged;
            (win as Window).SizeChanged -= PositionChanged;
            if (IsPositionUpdate)
            {
                (win as Window).LocationChanged += PositionChanged;
                (win as Window).SizeChanged += PositionChanged;
            }
        }
    }

    /// <summary>  
    /// 刷新位置  
    /// </summary>  
    private void PositionChanged(object sender, EventArgs e)
    {
        try
        {
            var method = typeof(Popup).GetMethod("UpdatePosition", System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance);
            if (this.IsOpen)
            {
                method.Invoke(this, null);
            }
        }
        catch
        {
            return;
        }
    }

    //是否最前默认为非最前(false)  
    public static DependencyProperty TopmostProperty = Window.TopmostProperty.AddOwner(typeof(Popup), new FrameworkPropertyMetadata(false, OnTopmostChanged));
    public bool Topmost
    {
        get { return (bool)GetValue(TopmostProperty); }
        set { SetValue(TopmostProperty, value); }
    }
    private static void OnTopmostChanged(DependencyObject obj, DependencyPropertyChangedEventArgs e)
    {
        (obj as PopupEx).UpdateWindow();
    }

    /// <summary>  
    /// 重写拉开方法,置于非最前  
    /// </summary>  
    /// <param name="e"></param>  
    protected override void OnOpened(EventArgs e)
    {
        UpdateWindow();
    }

    /// <summary>  
    /// 刷新Popup层级  
    /// </summary>  
    private void UpdateWindow()
    {
        var hwnd = ((HwndSource)PresentationSource.FromVisual(this.Child)).Handle;
        RECT rect;
        if (NativeMethods.GetWindowRect(hwnd, out rect))
        {
            NativeMethods.SetWindowPos(hwnd, Topmost ? -1 : -2, rect.Left, rect.Top, (int)this.Width, (int)this.Height, 0);
        }
    }

    [StructLayout(LayoutKind.Sequential)]
    public struct RECT
    {
        public int Left;
        public int Top;
        public int Right;
        public int Bottom;
    }
    #region P/Invoke imports & definitions  
    public static class NativeMethods
    {


        [DllImport("user32.dll")]
        [return: MarshalAs(UnmanagedType.Bool)]
        internal static extern bool GetWindowRect(IntPtr hWnd, out RECT lpRect);
        [DllImport("user32", EntryPoint = "SetWindowPos")]
        internal static extern int SetWindowPos(IntPtr hWnd, int hwndInsertAfter, int x, int y, int cx, int cy, int wFlags);
    }
    #endregion
}

二、使用

1.在XAML头部加入链接

xml 复制代码
 xmlns:Component="clr-namespace:ZzgkChatRoom.Component"

2. 在XAML文件使用

代码如下(示例):

xml 复制代码
  <Button
      x:Name="EmojiBtn"
      Grid.Column="0"
      Margin="0,0,0,0"
      HorizontalAlignment="Left"
      VerticalAlignment="Center"
      Click="EmojiBtn_Click"
      Content="0 2 0 0"
      Style="{StaticResource ChatBottomBtn}"
      Tag="{x:Static Icon:PackIconVaadinIconsKind.SmileyOutline}" />
	<Component:PopupEx
    	x:Name="EmojiPop"
    	AllowsTransparency="True"
   	 	HorizontalOffset="-40"
    	IsOpen="False"
    	Placement="Top"
   	 	PlacementTarget="{Binding ElementName=EmojiBtn}"
    	VerticalOffset="-5">
    <Border
        Width="400"
        Height="200"
        Background="#ffffff"
        BorderBrush="#646465"
        BorderThickness="1"
        CornerRadius="3" />
	</Component:PopupEx>

总结

相关推荐
小二·19 小时前
微服务架构设计与实践
微服务·架构·wpf
暖馒21 小时前
WPF-Prism学习入门步骤记录
学习·wpf
baivfhpwxf202321 小时前
雷赛(Leadshine)EtherCAT 数字 I/O 模块(如 EMC-E5064-8)的状态指示灯(I/O 状态)说明
c#·wpf
故渊at2 天前
第二板块:Android 四大组件标准化学理 | 第十二篇:四大组件全景总结与系统服务(System Server)架构
android·架构·wpf·四大组件·system service
伶俜662 天前
# [特殊字符] 零基础学 ArkUI 数据持久化(专题三):5 种存储方案深度对比
学习·华为·wpf·harmonyos
IT策士2 天前
Redis 从入门到精通:数据结构String 与键管理
数据结构·redis·wpf
AC赳赳老秦2 天前
技术文章素材收集自动化:用 OpenClaw 自动爬取行业资讯、技术热点、优质文章
运维·开发语言·python·自动化·wpf·deepseek·openclaw
加号32 天前
【WPF】 Storyboard 故事板动画设计深度解析
wpf
xiaoshuaishuai82 天前
C# Avalonia 依赖属性与WPF的区别
开发语言·c#·wpf
大G的笔记本2 天前
生产级 Spring Boot 网关简单实现方案
wpf