CompositionAnimationTests.cs 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Reflection;
  4. using System.Threading.Tasks;
  5. using Avalonia.Animation.Easings;
  6. using Avalonia.Controls;
  7. using Avalonia.Rendering;
  8. using Avalonia.Rendering.Composition;
  9. using Avalonia.Rendering.Composition.Expressions;
  10. using Avalonia.Rendering.Composition.Server;
  11. using Avalonia.Threading;
  12. using Avalonia.UnitTests;
  13. using Xunit;
  14. using Xunit.Sdk;
  15. using Xunit.v3;
  16. namespace Avalonia.Base.UnitTests.Composition;
  17. public class CompositionAnimationTests : ScopedTestBase
  18. {
  19. class AnimationDataProvider : DataAttribute
  20. {
  21. IEnumerable<AnimationData> Generate() =>
  22. new AnimationData[]
  23. {
  24. new("3 frames starting from 0")
  25. {
  26. Frames =
  27. {
  28. (0f, 10f),
  29. (0.5f, 30f),
  30. (1f, 20f)
  31. },
  32. Checks =
  33. {
  34. (0.25f, 20f),
  35. (0.5f, 30f),
  36. (0.75f, 25f),
  37. (1f, 20f)
  38. }
  39. },
  40. new("1 final frame")
  41. {
  42. Frames =
  43. {
  44. (1f, 10f)
  45. },
  46. Checks =
  47. {
  48. (0f, 0f),
  49. (0.5f, 5f),
  50. (1f, 10f)
  51. }
  52. }
  53. };
  54. public override ValueTask<IReadOnlyCollection<ITheoryDataRow>> GetData(MethodInfo testMethod, DisposalTracker disposalTracker)
  55. {
  56. var list = new List<ITheoryDataRow>();
  57. foreach (var ani in Generate())
  58. {
  59. list.Add(new TheoryDataRow<AnimationData>(ani));
  60. }
  61. return ValueTask.FromResult<IReadOnlyCollection<ITheoryDataRow>>(list);
  62. }
  63. public override bool SupportsDiscoveryEnumeration()
  64. => true;
  65. }
  66. class DummyDispatcher : IDispatcher
  67. {
  68. public bool CheckAccess() => true;
  69. public void VerifyAccess()
  70. {
  71. }
  72. public void Post(Action action, DispatcherPriority priority = default) => throw new NotSupportedException();
  73. }
  74. [AnimationDataProvider]
  75. [Theory]
  76. public void GenericCheck(AnimationData data)
  77. {
  78. using var scope = AvaloniaLocator.EnterScope();
  79. var compositor =
  80. new Compositor(RenderLoop.FromTimer(new CompositorTestServices.ManualRenderTimer()), null);
  81. var target = compositor.CreateSolidColorVisual();
  82. var ani = new ScalarKeyFrameAnimation(compositor);
  83. foreach (var frame in data.Frames)
  84. ani.InsertKeyFrame(frame.key, frame.value, new LinearEasing());
  85. ani.Duration = TimeSpan.FromSeconds(1);
  86. var instance = ani.CreateInstance(target.Server, null);
  87. instance.Initialize(TimeSpan.Zero, data.StartingValue, ServerCompositionVisual.s_IdOfRotationAngleProperty);
  88. var currentValue = ExpressionVariant.Create(data.StartingValue);
  89. foreach (var check in data.Checks)
  90. {
  91. currentValue = instance.Evaluate(TimeSpan.FromSeconds(check.time), currentValue);
  92. Assert.Equal(check.value, currentValue.Double);
  93. }
  94. }
  95. public class AnimationData
  96. {
  97. public AnimationData(string name)
  98. {
  99. Name = name;
  100. }
  101. public string Name { get; }
  102. public List<(float key, float value)> Frames { get; set; } = new();
  103. public List<(float time, float value)> Checks { get; set; } = new();
  104. public float StartingValue { get; set; }
  105. public float Duration { get; set; } = 1;
  106. public override string ToString()
  107. {
  108. return Name;
  109. }
  110. }
  111. [Theory]
  112. [InlineData("Color")]
  113. [InlineData("Offset")]
  114. public void GetCompositionProperty_ReturnsRegisteredProperties(string propName)
  115. {
  116. using var scope = AvaloniaLocator.EnterScope();
  117. var compositor = new Compositor(RenderLoop.FromTimer(new CompositorTestServices.ManualRenderTimer()), null);
  118. var target = compositor.CreateSolidColorVisual();
  119. var property = target.Server.GetCompositionProperty(propName);
  120. Assert.NotNull(property);
  121. Assert.Equal(propName, property.Name);
  122. Assert.NotNull(property.GetVariant);
  123. }
  124. [Fact]
  125. public void ExpressionAnimation_Operations_WorksCorrectly()
  126. {
  127. using var scope = AvaloniaLocator.EnterScope();
  128. var compositor = new Compositor(RenderLoop.FromTimer(new CompositorTestServices.ManualRenderTimer()), null);
  129. var target = compositor.CreateSolidColorVisual();
  130. target.Server.Offset = new Vector3D(100, 200, 0);
  131. var ani = compositor.CreateExpressionAnimation("this.Target.Offset.X * 0.5 + 10");
  132. var instance = ani.CreateInstance(target.Server, null);
  133. instance.Initialize(TimeSpan.Zero, ExpressionVariant.Create(0f),
  134. ServerCompositionVisual.s_IdOfRotationAngleProperty);
  135. var result = instance.Evaluate(TimeSpan.Zero, ExpressionVariant.Create(0f));
  136. Assert.Equal(VariantType.Double, result.Type);
  137. Assert.Equal(60.0, result.Double);
  138. }
  139. [Fact]
  140. public void ExpressionAnimation_Tracks_ReferenceParameter()
  141. {
  142. using var scope = AvaloniaLocator.EnterScope();
  143. var compositor = new Compositor(RenderLoop.FromTimer(new CompositorTestServices.ManualRenderTimer()), null);
  144. var target = compositor.CreateSolidColorVisual();
  145. var obj = compositor.CreateSolidColorVisual();
  146. obj.Server.Offset = new Vector3D(100, 200, 0);
  147. var ani = compositor.CreateExpressionAnimation("obj.Offset.X * 0.5 + 10");
  148. ani.SetReferenceParameter("obj", obj);
  149. var instance = ani.CreateInstance(target.Server, null);
  150. target.Server.Activate();
  151. // Invoke OnSetAnimatedValue manually to create ServerObjectAnimationInstance.
  152. target.Server.GetOrCreateAnimations();
  153. var tmp = 0f;
  154. target.Server.Animations!.OnSetAnimatedValue(ServerCompositionVisual.s_IdOfRotationAngleProperty, ref tmp, TimeSpan.Zero, instance);
  155. var initialResult = instance.Evaluate(TimeSpan.Zero, ExpressionVariant.Create(0f));
  156. Assert.Equal(60.0, initialResult.Double);
  157. obj.Server.Offset = new Vector3D(200, 300, 0);
  158. var updatedResult = instance.Evaluate(TimeSpan.Zero, ExpressionVariant.Create(0f));
  159. Assert.Equal(110.0, updatedResult.Double);
  160. }
  161. [Fact]
  162. public void ExpressionAnimation_Tracks_Target()
  163. {
  164. using var scope = AvaloniaLocator.EnterScope();
  165. var compositor = new Compositor(RenderLoop.FromTimer(new CompositorTestServices.ManualRenderTimer()), null);
  166. var target = compositor.CreateSolidColorVisual();
  167. target.Server.Offset = new Vector3D(100, 200, 0);
  168. var ani = compositor.CreateExpressionAnimation("this.Target.Offset.X * 0.5 + 10");
  169. var instance = ani.CreateInstance(target.Server, null);
  170. target.Server.Activate();
  171. // Invoke OnSetAnimatedValue manually to create ServerObjectAnimationInstance.
  172. target.Server.GetOrCreateAnimations();
  173. var tmp = 0f;
  174. target.Server.Animations!.OnSetAnimatedValue(ServerCompositionVisual.s_IdOfRotationAngleProperty, ref tmp, TimeSpan.Zero, instance);
  175. var initialResult = instance.Evaluate(TimeSpan.Zero, ExpressionVariant.Create(0f));
  176. Assert.Equal(60, initialResult.Double);
  177. target.Server.Offset = new Vector3D(200, 300, 0);
  178. var updatedResult = instance.Evaluate(TimeSpan.Zero, ExpressionVariant.Create(0f));
  179. Assert.Equal(110.0, updatedResult.Double);
  180. }
  181. [Fact]
  182. public void ExpressionAnimation_Requeues_Target_When_Another_Animation_Is_Invalidated_During_Evaluation()
  183. {
  184. using var services = new CompositorTestServices();
  185. var border = new Border
  186. {
  187. Width = 10,
  188. Height = 10
  189. };
  190. services.TopLevel.Content = border;
  191. services.RunJobs();
  192. var visual = ElementComposition.GetElementVisual(border)!;
  193. var opacityAnimation = visual.Compositor.CreateExpressionAnimation("this.Target.RotationAngle * 0.1");
  194. var rotationAnimation = visual.Compositor.CreateExpressionAnimation("this.Target.Offset.X * 0.5");
  195. visual.StartAnimation("Opacity", opacityAnimation);
  196. visual.StartAnimation("RotationAngle", rotationAnimation);
  197. services.RunJobs();
  198. visual.Offset = new Vector3D(100, 0, 0);
  199. services.RunJobs();
  200. Assert.Equal(50, visual.Server.RotationAngle);
  201. Assert.Equal(5, visual.Server.Opacity);
  202. }
  203. }