ChoreographerTimer.cs 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Reactive.Disposables;
  4. using System.Threading.Tasks;
  5. using Android.OS;
  6. using Android.Views;
  7. using Avalonia.Rendering;
  8. using Java.Lang;
  9. namespace Avalonia.Android
  10. {
  11. internal sealed class ChoreographerTimer : Java.Lang.Object, IRenderTimer, Choreographer.IFrameCallback
  12. {
  13. private readonly object _lock = new object();
  14. private readonly Thread _thread;
  15. private readonly TaskCompletionSource<Choreographer> _choreographer = new TaskCompletionSource<Choreographer>();
  16. private readonly ISet<AvaloniaView> _views = new HashSet<AvaloniaView>();
  17. private Action<TimeSpan> _tick;
  18. private int _count;
  19. public ChoreographerTimer()
  20. {
  21. _thread = new Thread(Loop);
  22. _thread.Start();
  23. }
  24. public event Action<TimeSpan> Tick
  25. {
  26. add
  27. {
  28. lock (_lock)
  29. {
  30. _tick += value;
  31. _count++;
  32. if (_count == 1)
  33. {
  34. _choreographer.Task.Result.PostFrameCallback(this);
  35. }
  36. }
  37. }
  38. remove
  39. {
  40. lock (_lock)
  41. {
  42. _tick -= value;
  43. _count--;
  44. }
  45. }
  46. }
  47. internal IDisposable SubscribeView(AvaloniaView view)
  48. {
  49. lock (_lock)
  50. {
  51. _views.Add(view);
  52. if (_views.Count == 1)
  53. {
  54. _choreographer.Task.Result.PostFrameCallback(this);
  55. }
  56. }
  57. return Disposable.Create(
  58. () =>
  59. {
  60. lock (_lock)
  61. {
  62. _views.Remove(view);
  63. }
  64. }
  65. );
  66. }
  67. private void Loop()
  68. {
  69. Looper.Prepare();
  70. _choreographer.SetResult(Choreographer.Instance);
  71. Looper.Loop();
  72. }
  73. public void DoFrame(long frameTimeNanos)
  74. {
  75. _tick?.Invoke(TimeSpan.FromTicks(frameTimeNanos / 100));
  76. lock (_lock)
  77. {
  78. if (_count > 0 && _views.Count > 0)
  79. {
  80. Choreographer.Instance.PostFrameCallback(this);
  81. }
  82. }
  83. }
  84. }
  85. }