| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122 |
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using System.Windows.Controls.Primitives;
- using System.Runtime.InteropServices;
- using System.Windows.Interop;
- using System.Windows;
- using System.Windows.Media;
- namespace Muchinfo.WPF.Controls
- {
- public class PopubEx : 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(PopubEx), new PropertyMetadata(true, new PropertyChangedCallback(IsPositionUpdateChanged)));
- private static void IsPositionUpdateChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
- {
- (d as PopubEx).pup_Loaded(d as PopubEx, null);
- }
- /// <summary>
- /// 加载窗口随动事件
- /// </summary>
- public PopubEx()
- {
- 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;
- if (IsPositionUpdate)
- (win as Window).LocationChanged += PositionChanged;
- }
- }
- /// <summary>
- /// 刷新位置
- /// </summary>
- private void PositionChanged(object sender, EventArgs e)
- {
- var method = typeof(Popup).GetMethod("UpdatePosition", System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance);
- if (this.IsOpen) method.Invoke(this, null);
- }
- //是否最前默认为非最前(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 PopubEx).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
- }
- }
|