DragAndDrop core
This commit is contained in:
429
Lattice.UI.DragDrop.WinUI/Behaviors/WinUIDragSourceBehavior.cs
Normal file
429
Lattice.UI.DragDrop.WinUI/Behaviors/WinUIDragSourceBehavior.cs
Normal file
@@ -0,0 +1,429 @@
|
||||
using Lattice.Core.Geometry;
|
||||
using Microsoft.UI.Xaml;
|
||||
using Microsoft.UI.Xaml.Controls;
|
||||
using Microsoft.UI.Xaml.Input;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Runtime.CompilerServices;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace Lattice.UI.DragDrop.WinUI.Behaviors;
|
||||
|
||||
/// <summary>
|
||||
/// Поведение источника перетаскивания для WinUI UIElement.
|
||||
/// Исправленная версия с правильной очисткой ресурсов.
|
||||
/// </summary>
|
||||
public static class WinUIDragSourceBehavior
|
||||
{
|
||||
#region Вложенные типы
|
||||
|
||||
private sealed class DragSourceContext : IDisposable
|
||||
{
|
||||
public Point DragStartPosition { get; set; }
|
||||
public bool IsDragging { get; set; }
|
||||
public UIElement? CurrentDragElement { get; set; }
|
||||
public object? DragData { get; set; }
|
||||
public Action<UIElement>? DragStartedHandler { get; set; }
|
||||
public Action<UIElement, Core.DragDrop.Enums.DragDropEffects>? DragCompletedHandler { get; set; }
|
||||
public Core.DragDrop.Enums.DragDropEffects AllowedEffects { get; set; }
|
||||
= Core.DragDrop.Enums.DragDropEffects.Copy | Core.DragDrop.Enums.DragDropEffects.Move;
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
DragStartedHandler = null;
|
||||
DragCompletedHandler = null;
|
||||
DragData = null;
|
||||
CurrentDragElement = null;
|
||||
}
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region Прикрепленные свойства
|
||||
|
||||
/// <summary>
|
||||
/// Идентификатор свойства для привязки данных перетаскивания.
|
||||
/// </summary>
|
||||
public static readonly DependencyProperty DragDataProperty =
|
||||
DependencyProperty.RegisterAttached(
|
||||
"DragData",
|
||||
typeof(object),
|
||||
typeof(WinUIDragSourceBehavior),
|
||||
new PropertyMetadata(null, OnDragDataChanged));
|
||||
|
||||
/// <summary>
|
||||
/// Идентификатор свойства для включения перетаскивания.
|
||||
/// </summary>
|
||||
public static readonly DependencyProperty IsEnabledProperty =
|
||||
DependencyProperty.RegisterAttached(
|
||||
"IsEnabled",
|
||||
typeof(bool),
|
||||
typeof(WinUIDragSourceBehavior),
|
||||
new PropertyMetadata(false, OnIsEnabledChanged));
|
||||
|
||||
/// <summary>
|
||||
/// Идентификатор свойства для разрешенных эффектов перетаскивания.
|
||||
/// </summary>
|
||||
public static readonly DependencyProperty AllowedEffectsProperty =
|
||||
DependencyProperty.RegisterAttached(
|
||||
"AllowedEffects",
|
||||
typeof(Core.DragDrop.Enums.DragDropEffects),
|
||||
typeof(WinUIDragSourceBehavior),
|
||||
new PropertyMetadata(Core.DragDrop.Enums.DragDropEffects.Copy | Core.DragDrop.Enums.DragDropEffects.Move));
|
||||
|
||||
/// <summary>
|
||||
/// Идентификатор свойства для обработчика начала перетаскивания.
|
||||
/// </summary>
|
||||
public static readonly DependencyProperty DragStartedHandlerProperty =
|
||||
DependencyProperty.RegisterAttached(
|
||||
"DragStartedHandler",
|
||||
typeof(Action<UIElement>),
|
||||
typeof(WinUIDragSourceBehavior),
|
||||
new PropertyMetadata(null));
|
||||
|
||||
/// <summary>
|
||||
/// Идентификатор свойства для обработчика завершения перетаскивания.
|
||||
/// </summary>
|
||||
public static readonly DependencyProperty DragCompletedHandlerProperty =
|
||||
DependencyProperty.RegisterAttached(
|
||||
"DragCompletedHandler",
|
||||
typeof(Action<UIElement, Core.DragDrop.Enums.DragDropEffects>),
|
||||
typeof(WinUIDragSourceBehavior),
|
||||
new PropertyMetadata(null));
|
||||
|
||||
/// <summary>
|
||||
/// Идентификатор свойства для обработчика создания данных перетаскивания.
|
||||
/// </summary>
|
||||
public static readonly DependencyProperty DragDataFactoryProperty =
|
||||
DependencyProperty.RegisterAttached(
|
||||
"DragDataFactory",
|
||||
typeof(Func<UIElement, Task<object>>),
|
||||
typeof(WinUIDragSourceBehavior),
|
||||
new PropertyMetadata(null));
|
||||
|
||||
#endregion
|
||||
|
||||
#region Статические поля
|
||||
|
||||
private static readonly ConditionalWeakTable<UIElement, DragSourceContext> _contexts = new();
|
||||
|
||||
#endregion
|
||||
|
||||
#region Методы доступа к свойствам
|
||||
|
||||
/// <summary>
|
||||
/// Получает значение свойства DragData для указанного элемента.
|
||||
/// </summary>
|
||||
public static object GetDragData(UIElement element) =>
|
||||
element.GetValue(DragDataProperty);
|
||||
|
||||
/// <summary>
|
||||
/// Устанавливает значение свойства DragData для указанного элемента.
|
||||
/// </summary>
|
||||
public static void SetDragData(UIElement element, object value) =>
|
||||
element.SetValue(DragDataProperty, value);
|
||||
|
||||
/// <summary>
|
||||
/// Получает значение свойства IsEnabled для указанного элемента.
|
||||
/// </summary>
|
||||
public static bool GetIsEnabled(UIElement element) =>
|
||||
(bool)element.GetValue(IsEnabledProperty);
|
||||
|
||||
/// <summary>
|
||||
/// Устанавливает значение свойства IsEnabled для указанного элемента.
|
||||
/// </summary>
|
||||
public static void SetIsEnabled(UIElement element, bool value) =>
|
||||
element.SetValue(IsEnabledProperty, value);
|
||||
|
||||
/// <summary>
|
||||
/// Получает значение свойства AllowedEffects для указанного элемента.
|
||||
/// </summary>
|
||||
public static Core.DragDrop.Enums.DragDropEffects GetAllowedEffects(UIElement element) =>
|
||||
(Core.DragDrop.Enums.DragDropEffects)element.GetValue(AllowedEffectsProperty);
|
||||
|
||||
/// <summary>
|
||||
/// Устанавливает значение свойства AllowedEffects для указанного элемента.
|
||||
/// </summary>
|
||||
public static void SetAllowedEffects(UIElement element, Core.DragDrop.Enums.DragDropEffects value) =>
|
||||
element.SetValue(AllowedEffectsProperty, value);
|
||||
|
||||
/// <summary>
|
||||
/// Получает обработчик начала перетаскивания.
|
||||
/// </summary>
|
||||
public static Action<UIElement> GetDragStartedHandler(UIElement element) =>
|
||||
(Action<UIElement>)element.GetValue(DragStartedHandlerProperty);
|
||||
|
||||
/// <summary>
|
||||
/// Устанавливает обработчик начала перетаскивания.
|
||||
/// </summary>
|
||||
public static void SetDragStartedHandler(UIElement element, Action<UIElement> value) =>
|
||||
element.SetValue(DragStartedHandlerProperty, value);
|
||||
|
||||
/// <summary>
|
||||
/// Получает обработчик завершения перетаскивания.
|
||||
/// </summary>
|
||||
public static Action<UIElement, Core.DragDrop.Enums.DragDropEffects> GetDragCompletedHandler(UIElement element) =>
|
||||
(Action<UIElement, Core.DragDrop.Enums.DragDropEffects>)element.GetValue(DragCompletedHandlerProperty);
|
||||
|
||||
/// <summary>
|
||||
/// Устанавливает обработчик завершения перетаскивания.
|
||||
/// </summary>
|
||||
public static void SetDragCompletedHandler(UIElement element, Action<UIElement, Core.DragDrop.Enums.DragDropEffects> value) =>
|
||||
element.SetValue(DragCompletedHandlerProperty, value);
|
||||
|
||||
/// <summary>
|
||||
/// Получает фабрику данных перетаскивания.
|
||||
/// </summary>
|
||||
public static Func<UIElement, Task<object>> GetDragDataFactory(UIElement element) =>
|
||||
(Func<UIElement, Task<object>>)element.GetValue(DragDataFactoryProperty);
|
||||
|
||||
/// <summary>
|
||||
/// Устанавливает фабрику данных перетаскивания.
|
||||
/// </summary>
|
||||
public static void SetDragDataFactory(UIElement element, Func<UIElement, Task<object>> value) =>
|
||||
element.SetValue(DragDataFactoryProperty, value);
|
||||
|
||||
#endregion
|
||||
|
||||
#region Обработчики изменений свойств
|
||||
|
||||
private static void OnDragDataChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
|
||||
{
|
||||
if (d is UIElement element)
|
||||
{
|
||||
if (_contexts.TryGetValue(element, out var context))
|
||||
{
|
||||
context.DragData = e.NewValue;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private static void OnIsEnabledChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
|
||||
{
|
||||
if (d is UIElement element)
|
||||
{
|
||||
if ((bool)e.NewValue)
|
||||
{
|
||||
EnableDrag(element);
|
||||
}
|
||||
else
|
||||
{
|
||||
DisableDrag(element);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region Включение/выключение перетаскивания
|
||||
|
||||
private static void EnableDrag(UIElement element)
|
||||
{
|
||||
if (_contexts.TryGetValue(element, out _))
|
||||
return;
|
||||
|
||||
var context = new DragSourceContext
|
||||
{
|
||||
DragData = GetDragData(element),
|
||||
DragStartedHandler = GetDragStartedHandler(element),
|
||||
DragCompletedHandler = GetDragCompletedHandler(element),
|
||||
AllowedEffects = GetAllowedEffects(element)
|
||||
};
|
||||
|
||||
_contexts.Add(element, context);
|
||||
|
||||
element.PointerPressed += OnPointerPressed;
|
||||
element.PointerMoved += OnPointerMoved;
|
||||
element.PointerReleased += OnPointerReleased;
|
||||
element.PointerCanceled += OnPointerCanceled;
|
||||
element.PointerCaptureLost += OnPointerCaptureLost;
|
||||
element.Unloaded += OnElementUnloaded;
|
||||
}
|
||||
|
||||
private static void DisableDrag(UIElement element)
|
||||
{
|
||||
element.PointerPressed -= OnPointerPressed;
|
||||
element.PointerMoved -= OnPointerMoved;
|
||||
element.PointerReleased -= OnPointerReleased;
|
||||
element.PointerCanceled -= OnPointerCanceled;
|
||||
element.PointerCaptureLost -= OnPointerCaptureLost;
|
||||
element.Unloaded -= OnElementUnloaded;
|
||||
|
||||
if (_contexts.TryGetValue(element, out var context))
|
||||
{
|
||||
context.Dispose();
|
||||
_contexts.Remove(element);
|
||||
}
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region Обработчики событий
|
||||
|
||||
private static void OnPointerPressed(object sender, PointerRoutedEventArgs e)
|
||||
{
|
||||
if (sender is UIElement element && _contexts.TryGetValue(element, out var context))
|
||||
{
|
||||
var point = e.GetCurrentPoint(element);
|
||||
context.DragStartPosition = new Point(point.Position.X, point.Position.Y);
|
||||
context.CurrentDragElement = element;
|
||||
element.CapturePointer(e.Pointer);
|
||||
}
|
||||
}
|
||||
|
||||
private static async void OnPointerMoved(object sender, PointerRoutedEventArgs e)
|
||||
{
|
||||
if (sender is UIElement element && _contexts.TryGetValue(element, out var context))
|
||||
{
|
||||
if (context.IsDragging || context.CurrentDragElement == null)
|
||||
return;
|
||||
|
||||
var currentPoint = e.GetCurrentPoint(context.CurrentDragElement);
|
||||
var currentPosition = new Point(currentPoint.Position.X, currentPoint.Position.Y);
|
||||
|
||||
var distance = CalculateDistance(context.DragStartPosition, currentPosition);
|
||||
if (distance > 3.0) // Порог перетаскивания
|
||||
{
|
||||
context.IsDragging = true;
|
||||
await StartDragAsync(context.CurrentDragElement, currentPosition, context);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private static async Task StartDragAsync(UIElement element, Point currentPosition, DragSourceContext context)
|
||||
{
|
||||
try
|
||||
{
|
||||
object? data = context.DragData;
|
||||
|
||||
// Если есть фабрика данных, используем ее
|
||||
var factory = GetDragDataFactory(element);
|
||||
if (factory != null)
|
||||
{
|
||||
data = await factory(element);
|
||||
}
|
||||
|
||||
if (data != null)
|
||||
{
|
||||
context.DragData = data;
|
||||
|
||||
// Вызываем обработчик начала перетаскивания
|
||||
context.DragStartedHandler?.Invoke(element);
|
||||
|
||||
// Устанавливаем визуальный эффект
|
||||
SetDragVisualState(element, true);
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
System.Diagnostics.Debug.WriteLine($"Error starting drag: {ex.Message}");
|
||||
context.IsDragging = false;
|
||||
}
|
||||
}
|
||||
|
||||
private static void OnPointerReleased(object sender, PointerRoutedEventArgs e)
|
||||
{
|
||||
CompleteDrag(sender as UIElement, Core.DragDrop.Enums.DragDropEffects.Copy);
|
||||
}
|
||||
|
||||
private static void OnPointerCanceled(object sender, PointerRoutedEventArgs e)
|
||||
{
|
||||
CompleteDrag(sender as UIElement, Core.DragDrop.Enums.DragDropEffects.None);
|
||||
}
|
||||
|
||||
private static void OnPointerCaptureLost(object sender, PointerRoutedEventArgs e)
|
||||
{
|
||||
CompleteDrag(sender as UIElement, Core.DragDrop.Enums.DragDropEffects.None);
|
||||
}
|
||||
|
||||
private static void OnElementUnloaded(object sender, RoutedEventArgs e)
|
||||
{
|
||||
if (sender is UIElement element)
|
||||
{
|
||||
DisableDrag(element);
|
||||
}
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region Вспомогательные методы
|
||||
|
||||
private static void CompleteDrag(UIElement? element, Core.DragDrop.Enums.DragDropEffects effects)
|
||||
{
|
||||
if (element != null && _contexts.TryGetValue(element, out var context))
|
||||
{
|
||||
// Вызываем обработчик завершения перетаскивания
|
||||
context.DragCompletedHandler?.Invoke(element, effects);
|
||||
|
||||
// Сбрасываем визуальный эффект
|
||||
SetDragVisualState(element, false);
|
||||
|
||||
element.ReleasePointerCaptures();
|
||||
|
||||
ResetDragState(context);
|
||||
}
|
||||
}
|
||||
|
||||
private static void SetDragVisualState(UIElement element, bool isDragging)
|
||||
{
|
||||
// Для элементов, которые являются Control, используем VisualStateManager
|
||||
if (element is Control control)
|
||||
{
|
||||
// Пытаемся перейти к состоянию "Dragging" или "Normal"
|
||||
try
|
||||
{
|
||||
VisualStateManager.GoToState(control, isDragging ? "Dragging" : "Normal", true);
|
||||
}
|
||||
catch
|
||||
{
|
||||
// Если состояния не определены, меняем свойства напрямую
|
||||
control.Opacity = isDragging ? 0.7 : 1.0;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
// Для обычных UIElement меняем свойства напрямую
|
||||
element.Opacity = isDragging ? 0.7 : 1.0;
|
||||
}
|
||||
}
|
||||
|
||||
private static double CalculateDistance(Point p1, Point p2)
|
||||
{
|
||||
var dx = p2.X - p1.X;
|
||||
var dy = p2.Y - p1.Y;
|
||||
return Math.Sqrt(dx * dx + dy * dy);
|
||||
}
|
||||
|
||||
private static void ResetDragState(DragSourceContext context)
|
||||
{
|
||||
context.IsDragging = false;
|
||||
context.CurrentDragElement = null;
|
||||
context.DragStartPosition = default;
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region Методы очистки
|
||||
|
||||
/// <summary>
|
||||
/// Очищает все ресурсы, связанные с поведением перетаскивания.
|
||||
/// </summary>
|
||||
public static void Cleanup()
|
||||
{
|
||||
var elements = new List<UIElement>();
|
||||
|
||||
foreach (var kvp in _contexts)
|
||||
{
|
||||
elements.Add(kvp.Key);
|
||||
}
|
||||
|
||||
foreach (var element in elements)
|
||||
{
|
||||
DisableDrag(element);
|
||||
}
|
||||
|
||||
_contexts.Clear();
|
||||
}
|
||||
|
||||
#endregion
|
||||
}
|
||||
402
Lattice.UI.DragDrop.WinUI/Behaviors/WinUIDropTargetBehavior.cs
Normal file
402
Lattice.UI.DragDrop.WinUI/Behaviors/WinUIDropTargetBehavior.cs
Normal file
@@ -0,0 +1,402 @@
|
||||
using Microsoft.UI.Xaml;
|
||||
using Microsoft.UI.Xaml.Controls;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
|
||||
namespace Lattice.UI.DragDrop.WinUI.Behaviors;
|
||||
|
||||
/// <summary>
|
||||
/// Поведение цели сброса для WinUI UIElement.
|
||||
/// </summary>
|
||||
public static class WinUIDropTargetBehavior
|
||||
{
|
||||
/// <summary>
|
||||
/// Идентификатор свойства для включения цели сброса.
|
||||
/// </summary>
|
||||
public static readonly DependencyProperty IsEnabledProperty =
|
||||
DependencyProperty.RegisterAttached(
|
||||
"IsEnabled",
|
||||
typeof(bool),
|
||||
typeof(WinUIDropTargetBehavior),
|
||||
new PropertyMetadata(false, OnIsEnabledChanged));
|
||||
|
||||
/// <summary>
|
||||
/// Идентификатор свойства для фильтрации принимаемых данных.
|
||||
/// </summary>
|
||||
public static readonly DependencyProperty AcceptsDataTypesProperty =
|
||||
DependencyProperty.RegisterAttached(
|
||||
"AcceptsDataTypes",
|
||||
typeof(Type[]),
|
||||
typeof(WinUIDropTargetBehavior),
|
||||
new PropertyMetadata(null));
|
||||
|
||||
/// <summary>
|
||||
/// Идентификатор свойства для обработчика сброса.
|
||||
/// </summary>
|
||||
public static readonly DependencyProperty DropHandlerProperty =
|
||||
DependencyProperty.RegisterAttached(
|
||||
"DropHandler",
|
||||
typeof(Core.DragDrop.Abstractions.IDropTarget),
|
||||
typeof(WinUIDropTargetBehavior),
|
||||
new PropertyMetadata(null));
|
||||
|
||||
/// <summary>
|
||||
/// Идентификатор свойства для отображения визуальной обратной связи.
|
||||
/// </summary>
|
||||
public static readonly DependencyProperty ShowVisualFeedbackProperty =
|
||||
DependencyProperty.RegisterAttached(
|
||||
"ShowVisualFeedback",
|
||||
typeof(bool),
|
||||
typeof(WinUIDropTargetBehavior),
|
||||
new PropertyMetadata(true));
|
||||
|
||||
/// <summary>
|
||||
/// Идентификатор свойства для стиля визуальной обратной связи.
|
||||
/// </summary>
|
||||
public static readonly DependencyProperty FeedbackStyleProperty =
|
||||
DependencyProperty.RegisterAttached(
|
||||
"FeedbackStyle",
|
||||
typeof(Style),
|
||||
typeof(WinUIDropTargetBehavior),
|
||||
new PropertyMetadata(null));
|
||||
|
||||
/// <summary>
|
||||
/// Идентификатор свойства для обработчика входа перетаскивания.
|
||||
/// </summary>
|
||||
public static readonly DependencyProperty DragEnterHandlerProperty =
|
||||
DependencyProperty.RegisterAttached(
|
||||
"DragEnterHandler",
|
||||
typeof(Action<UIElement>),
|
||||
typeof(WinUIDropTargetBehavior),
|
||||
new PropertyMetadata(null));
|
||||
|
||||
/// <summary>
|
||||
/// Идентификатор свойства для обработчика выхода перетаскивания.
|
||||
/// </summary>
|
||||
public static readonly DependencyProperty DragLeaveHandlerProperty =
|
||||
DependencyProperty.RegisterAttached(
|
||||
"DragLeaveHandler",
|
||||
typeof(Action<UIElement>),
|
||||
typeof(WinUIDropTargetBehavior),
|
||||
new PropertyMetadata(null));
|
||||
|
||||
/// <summary>
|
||||
/// Идентификатор свойства для обработчика сброса.
|
||||
/// </summary>
|
||||
public static readonly DependencyProperty DropHandlerActionProperty =
|
||||
DependencyProperty.RegisterAttached(
|
||||
"DropHandlerAction",
|
||||
typeof(Action<UIElement, object>),
|
||||
typeof(WinUIDropTargetBehavior),
|
||||
new PropertyMetadata(null));
|
||||
|
||||
// Словарь для отслеживания текущих перетаскиваемых элементов
|
||||
private static readonly Dictionary<UIElement, bool> _dragOverElements = new();
|
||||
|
||||
/// <summary>
|
||||
/// Получает значение свойства IsEnabled для указанного элемента.
|
||||
/// </summary>
|
||||
public static bool GetIsEnabled(UIElement element) =>
|
||||
(bool)element.GetValue(IsEnabledProperty);
|
||||
|
||||
/// <summary>
|
||||
/// Устанавливает значение свойства IsEnabled для указанного элемента.
|
||||
/// </summary>
|
||||
public static void SetIsEnabled(UIElement element, bool value) =>
|
||||
element.SetValue(IsEnabledProperty, value);
|
||||
|
||||
/// <summary>
|
||||
/// Получает значение свойства AcceptsDataTypes для указанного элемента.
|
||||
/// </summary>
|
||||
public static Type[] GetAcceptsDataTypes(UIElement element) =>
|
||||
(Type[])element.GetValue(AcceptsDataTypesProperty);
|
||||
|
||||
/// <summary>
|
||||
/// Устанавливает значение свойства AcceptsDataTypes для указанного элемента.
|
||||
/// </summary>
|
||||
public static void SetAcceptsDataTypes(UIElement element, Type[] value) =>
|
||||
element.SetValue(AcceptsDataTypesProperty, value);
|
||||
|
||||
/// <summary>
|
||||
/// Получает значение свойства DropHandler для указанного элемента.
|
||||
/// </summary>
|
||||
public static Core.DragDrop.Abstractions.IDropTarget GetDropHandler(UIElement element) =>
|
||||
(Core.DragDrop.Abstractions.IDropTarget)element.GetValue(DropHandlerProperty);
|
||||
|
||||
/// <summary>
|
||||
/// Устанавливает значение свойства DropHandler для указанного элемента.
|
||||
/// </summary>
|
||||
public static void SetDropHandler(UIElement element, Core.DragDrop.Abstractions.IDropTarget value) =>
|
||||
element.SetValue(DropHandlerProperty, value);
|
||||
|
||||
/// <summary>
|
||||
/// Получает значение свойства ShowVisualFeedback для указанного элемента.
|
||||
/// </summary>
|
||||
public static bool GetShowVisualFeedback(UIElement element) =>
|
||||
(bool)element.GetValue(ShowVisualFeedbackProperty);
|
||||
|
||||
/// <summary>
|
||||
/// Устанавливает значение свойства ShowVisualFeedback для указанного элемента.
|
||||
/// </summary>
|
||||
public static void SetShowVisualFeedback(UIElement element, bool value) =>
|
||||
element.SetValue(ShowVisualFeedbackProperty, value);
|
||||
|
||||
/// <summary>
|
||||
/// Получает значение свойства FeedbackStyle для указанного элемента.
|
||||
/// </summary>
|
||||
public static Style GetFeedbackStyle(UIElement element) =>
|
||||
(Style)element.GetValue(FeedbackStyleProperty);
|
||||
|
||||
/// <summary>
|
||||
/// Устанавливает значение свойства FeedbackStyle для указанного элемента.
|
||||
/// </summary>
|
||||
public static void SetFeedbackStyle(UIElement element, Style value) =>
|
||||
element.SetValue(FeedbackStyleProperty, value);
|
||||
|
||||
/// <summary>
|
||||
/// Получает обработчик входа перетаскивания.
|
||||
/// </summary>
|
||||
public static Action<UIElement> GetDragEnterHandler(UIElement element) =>
|
||||
(Action<UIElement>)element.GetValue(DragEnterHandlerProperty);
|
||||
|
||||
/// <summary>
|
||||
/// Устанавливает обработчик входа перетаскивания.
|
||||
/// </summary>
|
||||
public static void SetDragEnterHandler(UIElement element, Action<UIElement> value) =>
|
||||
element.SetValue(DragEnterHandlerProperty, value);
|
||||
|
||||
/// <summary>
|
||||
/// Получает обработчик выхода перетаскивания.
|
||||
/// </summary>
|
||||
public static Action<UIElement> GetDragLeaveHandler(UIElement element) =>
|
||||
(Action<UIElement>)element.GetValue(DragLeaveHandlerProperty);
|
||||
|
||||
/// <summary>
|
||||
/// Устанавливает обработчик выхода перетаскивания.
|
||||
/// </summary>
|
||||
public static void SetDragLeaveHandler(UIElement element, Action<UIElement> value) =>
|
||||
element.SetValue(DragLeaveHandlerProperty, value);
|
||||
|
||||
/// <summary>
|
||||
/// Получает обработчик сброса.
|
||||
/// </summary>
|
||||
public static Action<UIElement, object> GetDropHandlerAction(UIElement element) =>
|
||||
(Action<UIElement, object>)element.GetValue(DropHandlerActionProperty);
|
||||
|
||||
/// <summary>
|
||||
/// Устанавливает обработчик сброса.
|
||||
/// </summary>
|
||||
public static void SetDropHandlerAction(UIElement element, Action<UIElement, object> value) =>
|
||||
element.SetValue(DropHandlerActionProperty, value);
|
||||
|
||||
private static void OnIsEnabledChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
|
||||
{
|
||||
if (d is UIElement element)
|
||||
{
|
||||
if ((bool)e.NewValue)
|
||||
{
|
||||
EnableDrop(element);
|
||||
}
|
||||
else
|
||||
{
|
||||
DisableDrop(element);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private static void EnableDrop(UIElement element)
|
||||
{
|
||||
element.AllowDrop = true;
|
||||
element.DragEnter += OnDragEnter;
|
||||
element.DragOver += OnDragOver;
|
||||
element.DragLeave += OnDragLeave;
|
||||
element.Drop += OnDrop;
|
||||
}
|
||||
|
||||
private static void DisableDrop(UIElement element)
|
||||
{
|
||||
element.AllowDrop = false;
|
||||
element.DragEnter -= OnDragEnter;
|
||||
element.DragOver -= OnDragOver;
|
||||
element.DragLeave -= OnDragLeave;
|
||||
element.Drop -= OnDrop;
|
||||
}
|
||||
|
||||
private static void OnDragEnter(object sender, DragEventArgs e)
|
||||
{
|
||||
var element = sender as UIElement;
|
||||
if (element == null) return;
|
||||
|
||||
// Проверяем, можно ли принять данные
|
||||
if (CanAcceptData(element, e.DataView))
|
||||
{
|
||||
e.AcceptedOperation = Windows.ApplicationModel.DataTransfer.DataPackageOperation.Copy;
|
||||
|
||||
// Визуальная обратная связь
|
||||
ShowVisualFeedback(element, true);
|
||||
|
||||
// Вызываем пользовательский обработчик
|
||||
GetDragEnterHandler(element)?.Invoke(element);
|
||||
|
||||
// Добавляем в словарь
|
||||
_dragOverElements[element] = true;
|
||||
}
|
||||
else
|
||||
{
|
||||
e.AcceptedOperation = Windows.ApplicationModel.DataTransfer.DataPackageOperation.None;
|
||||
}
|
||||
|
||||
e.Handled = true;
|
||||
}
|
||||
|
||||
private static void OnDragOver(object sender, DragEventArgs e)
|
||||
{
|
||||
var element = sender as UIElement;
|
||||
if (element == null) return;
|
||||
|
||||
if (CanAcceptData(element, e.DataView))
|
||||
{
|
||||
e.AcceptedOperation = Windows.ApplicationModel.DataTransfer.DataPackageOperation.Copy;
|
||||
|
||||
// Обновляем визуальную обратную связь на основе позиции
|
||||
var position = e.GetPosition(element);
|
||||
UpdateVisualFeedback(element, position);
|
||||
}
|
||||
else
|
||||
{
|
||||
e.AcceptedOperation = Windows.ApplicationModel.DataTransfer.DataPackageOperation.None;
|
||||
}
|
||||
|
||||
e.Handled = true;
|
||||
}
|
||||
|
||||
private static void OnDragLeave(object sender, DragEventArgs e)
|
||||
{
|
||||
var element = sender as UIElement;
|
||||
if (element == null) return;
|
||||
|
||||
// Скрываем визуальную обратную связь
|
||||
ShowVisualFeedback(element, false);
|
||||
|
||||
// Вызываем пользовательский обработчик
|
||||
GetDragLeaveHandler(element)?.Invoke(element);
|
||||
|
||||
// Удаляем из словаря
|
||||
_dragOverElements.Remove(element);
|
||||
}
|
||||
|
||||
private static void OnDrop(object sender, DragEventArgs e)
|
||||
{
|
||||
var element = sender as UIElement;
|
||||
if (element == null) return;
|
||||
|
||||
// Скрываем визуальную обратную связь
|
||||
ShowVisualFeedback(element, false);
|
||||
|
||||
// Получаем данные
|
||||
var data = ExtractData(e.DataView);
|
||||
|
||||
// Получаем обработчик или используем встроенный
|
||||
var handler = GetDropHandler(element);
|
||||
if (handler != null)
|
||||
{
|
||||
// Создаем DropInfo для обработчика
|
||||
var dropInfo = CreateDropInfo(element, e, data);
|
||||
handler.Drop(dropInfo);
|
||||
}
|
||||
|
||||
// Вызываем пользовательский обработчик
|
||||
var actionHandler = GetDropHandlerAction(element);
|
||||
if (actionHandler != null && data != null)
|
||||
{
|
||||
actionHandler.Invoke(element, data);
|
||||
}
|
||||
|
||||
// Удаляем из словаря
|
||||
_dragOverElements.Remove(element);
|
||||
|
||||
e.Handled = true;
|
||||
}
|
||||
|
||||
private static bool CanAcceptData(UIElement element, Windows.ApplicationModel.DataTransfer.DataPackageView dataView)
|
||||
{
|
||||
// Проверяем фильтр типов данных
|
||||
var acceptedTypes = GetAcceptsDataTypes(element);
|
||||
if (acceptedTypes != null && acceptedTypes.Length > 0)
|
||||
{
|
||||
// В реальной реализации нужно проверять доступные форматы данных
|
||||
// Здесь упрощенная проверка
|
||||
return dataView.AvailableFormats.Count > 0;
|
||||
}
|
||||
|
||||
// Если нет фильтра, принимаем все
|
||||
return true;
|
||||
}
|
||||
|
||||
private static object? ExtractData(Windows.ApplicationModel.DataTransfer.DataPackageView dataView)
|
||||
{
|
||||
// Упрощенная реализация извлечения данных
|
||||
// В реальном приложении нужно обрабатывать разные форматы данных
|
||||
try
|
||||
{
|
||||
if (dataView.Contains(Windows.ApplicationModel.DataTransfer.StandardDataFormats.Text))
|
||||
{
|
||||
// В WinUI 3 нужно использовать async/await, но это упрощенный пример
|
||||
// В реальном коде нужно использовать async методы
|
||||
return "Text data from drag";
|
||||
}
|
||||
}
|
||||
catch
|
||||
{
|
||||
// Игнорируем ошибки извлечения данных
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
private static void ShowVisualFeedback(UIElement element, bool show)
|
||||
{
|
||||
if (!GetShowVisualFeedback(element)) return;
|
||||
|
||||
// Для элементов, которые являются Control, используем VisualStateManager
|
||||
if (element is Control control)
|
||||
{
|
||||
// Пытаемся перейти к состоянию "DragOver" или "Normal"
|
||||
try
|
||||
{
|
||||
VisualStateManager.GoToState(control, show ? "DragOver" : "Normal", true);
|
||||
}
|
||||
catch
|
||||
{
|
||||
// Если состояния не определены, меняем свойства напрямую
|
||||
control.Background = show ?
|
||||
new Microsoft.UI.Xaml.Media.SolidColorBrush(Windows.UI.Color.FromArgb(30, 0, 120, 215)) :
|
||||
null;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
// Для обычных UIElement меняем свойства напрямую
|
||||
element.Opacity = show ? 0.8 : 1.0;
|
||||
}
|
||||
}
|
||||
|
||||
private static void UpdateVisualFeedback(UIElement element, Windows.Foundation.Point position)
|
||||
{
|
||||
// Можно добавить логику для различных зон сброса
|
||||
// Например, подсветка разных частей элемента
|
||||
}
|
||||
|
||||
private static Core.DragDrop.Models.DropInfo CreateDropInfo(UIElement element, DragEventArgs e, object? data)
|
||||
{
|
||||
var position = e.GetPosition(element);
|
||||
var screenPosition = element.TransformToVisual(null).TransformPoint(position);
|
||||
|
||||
return new Core.DragDrop.Models.DropInfo(
|
||||
data: data,
|
||||
position: new Core.Geometry.Point(screenPosition.X, screenPosition.Y),
|
||||
allowedEffects: Core.DragDrop.Enums.DragDropEffects.Copy | Core.DragDrop.Enums.DragDropEffects.Move,
|
||||
target: GetDropHandler(element)
|
||||
);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user