AdaptiveBehavior.cs 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. using System;
  2. using Avalonia.Collections;
  3. using Avalonia.Controls;
  4. using Avalonia.Metadata;
  5. using Avalonia.Reactive;
  6. using Avalonia.Xaml.Interactivity;
  7. namespace Avalonia.Xaml.Interactions.Responsive;
  8. /// <summary>
  9. /// Observes <see cref="StyledElementBehavior{T}.AssociatedObject"/> control or <see cref="SourceControl"/> control <see cref="Visual.Bounds"/> property changes and if triggered sets or removes style classes when conditions from <see cref="AdaptiveClassSetter"/> are met.
  10. /// </summary>
  11. public class AdaptiveBehavior : StyledElementBehavior<Control>
  12. {
  13. private IDisposable? _disposable;
  14. private AvaloniaList<AdaptiveClassSetter>? _setters;
  15. /// <summary>
  16. /// Identifies the <seealso cref="SourceControl"/> avalonia property.
  17. /// </summary>
  18. public static readonly StyledProperty<Control?> SourceControlProperty =
  19. AvaloniaProperty.Register<AdaptiveBehavior, Control?>(nameof(SourceControl));
  20. /// <summary>
  21. /// Identifies the <seealso cref="TargetControl"/> avalonia property.
  22. /// </summary>
  23. public static readonly StyledProperty<Control?> TargetControlProperty =
  24. AvaloniaProperty.Register<AdaptiveBehavior, Control?>(nameof(TargetControl));
  25. /// <summary>
  26. /// Identifies the <seealso cref="Setters"/> avalonia property.
  27. /// </summary>
  28. public static readonly DirectProperty<AdaptiveBehavior, AvaloniaList<AdaptiveClassSetter>> SettersProperty =
  29. AvaloniaProperty.RegisterDirect<AdaptiveBehavior, AvaloniaList<AdaptiveClassSetter>>(nameof(Setters), t => t.Setters);
  30. /// <summary>
  31. /// Gets or sets the the source control that <see cref="Visual.BoundsProperty"/> property are observed from, if not set <see cref="StyledElementBehavior{T}.AssociatedObject"/> is used. This is a avalonia property.
  32. /// </summary>
  33. [ResolveByName]
  34. public Control? SourceControl
  35. {
  36. get => GetValue(SourceControlProperty);
  37. set => SetValue(SourceControlProperty, value);
  38. }
  39. /// <summary>
  40. /// Gets or sets the target control that class name that should be added or removed when triggered, if not set <see cref="StyledElementBehavior{T}.AssociatedObject"/> is used or <see cref="AdaptiveClassSetter.TargetControl"/> from <see cref="AdaptiveClassSetter"/>. This is a avalonia property.
  41. /// </summary>
  42. [ResolveByName]
  43. public Control? TargetControl
  44. {
  45. get => GetValue(TargetControlProperty);
  46. set => SetValue(TargetControlProperty, value);
  47. }
  48. /// <summary>
  49. /// Gets adaptive class setters collection. This is a avalonia property.
  50. /// </summary>
  51. [Content]
  52. public AvaloniaList<AdaptiveClassSetter> Setters => _setters ??= [];
  53. /// <inheritdoc/>
  54. protected override void OnAttachedToVisualTree()
  55. {
  56. base.OnAttachedToVisualTree();
  57. StopObserving();
  58. StartObserving();
  59. }
  60. /// <inheritdoc/>
  61. protected override void OnDetachedFromVisualTree()
  62. {
  63. base.OnDetachedFromVisualTree();
  64. StopObserving();
  65. }
  66. private void StartObserving()
  67. {
  68. var sourceControl = GetValue(SourceControlProperty) is not null
  69. ? SourceControl
  70. : AssociatedObject;
  71. if (sourceControl is not null)
  72. {
  73. _disposable = ObserveBounds(sourceControl);
  74. }
  75. }
  76. private void StopObserving()
  77. {
  78. _disposable?.Dispose();
  79. }
  80. private IDisposable ObserveBounds(Control sourceControl)
  81. {
  82. if (sourceControl is null)
  83. {
  84. throw new ArgumentNullException(nameof(sourceControl));
  85. }
  86. Execute(sourceControl, Setters, sourceControl.GetValue(Visual.BoundsProperty));
  87. return sourceControl.GetObservable(Visual.BoundsProperty)
  88. .Subscribe(new AnonymousObserver<Rect>(bounds =>
  89. {
  90. Execute(sourceControl, Setters, bounds);
  91. }));
  92. }
  93. private void Execute(Control? sourceControl, AvaloniaList<AdaptiveClassSetter>? setters, Rect bounds)
  94. {
  95. if (sourceControl is null || setters is null)
  96. {
  97. return;
  98. }
  99. foreach (var setter in setters)
  100. {
  101. var isMinOrMaxWidthSet = setter.IsSet(AdaptiveClassSetter.MinWidthProperty)
  102. || setter.IsSet(AdaptiveClassSetter.MaxWidthProperty);
  103. var widthConditionTriggered = GetResult(setter.MinWidthOperator, bounds.Width, setter.MinWidth)
  104. && GetResult(setter.MaxWidthOperator, bounds.Width, setter.MaxWidth);
  105. var isMinOrMaxHeightSet = setter.IsSet(AdaptiveClassSetter.MinHeightProperty)
  106. || setter.IsSet(AdaptiveClassSetter.MaxHeightProperty);
  107. var heightConditionTriggered = GetResult(setter.MinHeightOperator, bounds.Height, setter.MinHeight)
  108. && GetResult(setter.MaxHeightOperator, bounds.Height, setter.MaxHeight);
  109. var isAddClassTriggered = isMinOrMaxWidthSet switch
  110. {
  111. true when !isMinOrMaxHeightSet => widthConditionTriggered,
  112. false when isMinOrMaxHeightSet => heightConditionTriggered,
  113. true when isMinOrMaxHeightSet => widthConditionTriggered && heightConditionTriggered,
  114. _ => false
  115. };
  116. var targetControl = setter.GetValue(AdaptiveClassSetter.TargetControlProperty) is not null
  117. ? setter.TargetControl
  118. : GetValue(TargetControlProperty) is not null
  119. ? TargetControl
  120. : AssociatedObject;
  121. if (targetControl is not null)
  122. {
  123. var className = setter.ClassName;
  124. var isPseudoClass = setter.IsPseudoClass;
  125. if (isAddClassTriggered)
  126. {
  127. Add(targetControl, className, isPseudoClass);
  128. }
  129. else
  130. {
  131. Remove(targetControl, className, isPseudoClass);
  132. }
  133. }
  134. else
  135. {
  136. throw new ArgumentNullException(nameof(targetControl));
  137. }
  138. }
  139. }
  140. private bool GetResult(ComparisonConditionType comparisonConditionType, double property, double value)
  141. {
  142. return comparisonConditionType switch
  143. {
  144. // ReSharper disable once CompareOfFloatsByEqualityOperator
  145. ComparisonConditionType.Equal => property == value,
  146. // ReSharper disable once CompareOfFloatsByEqualityOperator
  147. ComparisonConditionType.NotEqual => property != value,
  148. ComparisonConditionType.LessThan => property < value,
  149. ComparisonConditionType.LessThanOrEqual => property <= value,
  150. ComparisonConditionType.GreaterThan => property > value,
  151. ComparisonConditionType.GreaterThanOrEqual => property >= value,
  152. _ => throw new ArgumentOutOfRangeException()
  153. };
  154. }
  155. private static void Add(Control targetControl, string? className, bool isPseudoClass)
  156. {
  157. if (className is null || string.IsNullOrEmpty(className) || targetControl.Classes.Contains(className))
  158. {
  159. return;
  160. }
  161. if (isPseudoClass)
  162. {
  163. ((IPseudoClasses) targetControl.Classes).Add(className);
  164. }
  165. else
  166. {
  167. targetControl.Classes.Add(className);
  168. }
  169. }
  170. private static void Remove(Control targetControl, string? className, bool isPseudoClass)
  171. {
  172. if (className is null || string.IsNullOrEmpty(className) || !targetControl.Classes.Contains(className))
  173. {
  174. return;
  175. }
  176. if (isPseudoClass)
  177. {
  178. ((IPseudoClasses) targetControl.Classes).Remove(className);
  179. }
  180. else
  181. {
  182. targetControl.Classes.Remove(className);
  183. }
  184. }
  185. }