WPF 基于Adorner实现类似Popup效果
1. 什么是Adorner
装饰器是一种特殊类型的FrameworkElement,可用来向用户提供可视提示。 装饰器有很多用途,可用来向元素添加功能句柄,或者提供有关某个控件的状态信息。
2. 使用Adorner实现Popup的原因
1. 通过AdornerLayer存在与独立的布局系统,不会与界面布局环论
2. 使用过WPF中的Popup就可以知道Popup中有许多的限制(例如需要实现某些效果比较麻烦)
3. 效果
4. 主要实现
1. Popup中当StaysOpen为False的情况下,当打开Popup后如果再点击其他区域时将会关闭Popup的实现(实现方法参考自Popup, 但是不采用Mouse.Capture(element),因为会导致其他控件无法收到鼠标实现
if (StaysOpen) return; Point pos = e.GetPosition(ListenMouseElement); HitTestResult hitResult = VisualTreeHelper.HitTest(ListenMouseElement, pos); if (hitResult == null) { IsOpen = false; return; } // 如果点击对象对Child则返回 if (TreeHelper.IsDescendantOf(hitResult.VisualHit, _adorner)) { return; } // 如果点击对象PlacementTarget则返回 if (IgnoreTargetEvent && TreeHelper.IsDescendantOf(hitResult.VisualHit, PlacementTarget)) { return; } IsOpen = false;
2. 派生Adorner 将AdornerPopup的Child属性加载到Adorner
public FrameworkElementAdorner(FrameworkElement adornerChildElement, FrameworkElement adornedElement, AdornerPopup adorner, double horizontalOffset , double verticalOffset) : base(adornedElement) { this._child = adornerChildElement; this._adorner = adorner; this._bgBorder = CreateBackgroundBorder(adornerChildElement); this.HorizontalOffset = horizontalOffset; this.VerticalOffset = verticalOffset; // 通过BaseLogicalChild, AddVisualChild将Child元素加载到可视树中 base.AddLogicalChild(_bgBorder); base.AddVisualChild(_bgBorder); }
代码已开源:https://gitee.com/1Jins/WPF-AdornerPopup
5. 参考文献
装饰器概述 https://docs.microsoft.com/zh-cn/dotnet/framework/wpf/controls/adorners-overview
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步