summaryrefslogtreecommitdiff
path: root/Xamarin.Forms.Platform.MacOS/CADisplayLinkTicker.cs
blob: 1e965ff5d3577b7452013cc7084bc3115d31e492 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
using System;
using System.Collections.Concurrent;
using System.Threading;
using Foundation;
using Xamarin.Forms.Internals;
using CoreVideo;
using AppKit;
using CoreAnimation;

namespace Xamarin.Forms.Platform.MacOS
{
	// ReSharper disable once InconsistentNaming
	internal class CADisplayLinkTicker : Ticker
	{
		readonly BlockingCollection<Action> _queue = new BlockingCollection<Action>();
		CVDisplayLink _link;

		public CADisplayLinkTicker()
		{
			var thread = new Thread(StartThread);
			thread.Start();
		}

		internal new static CADisplayLinkTicker Default => Ticker.Default as CADisplayLinkTicker;

		public void Invoke(Action action)
		{
			_queue.Add(action);
		}

		protected override void DisableTimer()
		{
			_link?.Stop();
			_link?.Dispose();
			_link = null;
		}

		protected override void EnableTimer()
		{
			_link = new CVDisplayLink();
			_link.SetOutputCallback(DisplayLinkOutputCallback);
			_link.Start();
		}

		public CVReturn DisplayLinkOutputCallback(CVDisplayLink displayLink, ref CVTimeStamp inNow,
			ref CVTimeStamp inOutputTime, CVOptionFlags flagsIn, ref CVOptionFlags flagsOut)
		{
			// There is no autorelease pool when this method is called because it will be called from a background thread
			// It's important to create one or you will leak objects
			// ReSharper disable once UnusedVariable
			using (var pool = new NSAutoreleasePool())
			{
				Device.BeginInvokeOnMainThread(() => SendSignals());
			}
			return CVReturn.Success;
		}

		void StartThread()
		{
			while (true)
			{
				Action action = _queue.Take();
				bool previous = NSApplication.CheckForIllegalCrossThreadCalls;
				NSApplication.CheckForIllegalCrossThreadCalls = false;

				CATransaction.Begin();
				action.Invoke();

				while (_queue.TryTake(out action))
					action.Invoke();
				CATransaction.Commit();

				NSApplication.CheckForIllegalCrossThreadCalls = previous;
			}
			// ReSharper disable once FunctionNeverReturns
		}
	}
}