summaryrefslogtreecommitdiff
path: root/Xamarin.Forms.Core/Ticker.cs
blob: ed828f8ad81140cff974ef3b9d9948edbca0c845 (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
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Threading;

namespace Xamarin.Forms
{
	internal class Ticker
	{
		static Ticker s_ticker;
		readonly Stopwatch _stopwatch;
		readonly SynchronizationContext _sync;
		readonly List<Tuple<int, Func<long, bool>>> _timeouts;

		readonly ITimer _timer;
		int _count;
		bool _enabled;

		internal Ticker()
		{
			_sync = SynchronizationContext.Current;
			_count = 0;
			_timer = Device.PlatformServices.CreateTimer(HandleElapsed, null, Timeout.Infinite, Timeout.Infinite);
			_timeouts = new List<Tuple<int, Func<long, bool>>>();

			_stopwatch = new Stopwatch();
		}

		public static Ticker Default
		{
			internal set { s_ticker = value; }
			get { return s_ticker ?? (s_ticker = new Ticker()); }
		}

		public virtual int Insert(Func<long, bool> timeout)
		{
			_count++;
			_timeouts.Add(new Tuple<int, Func<long, bool>>(_count, timeout));

			if (!_enabled)
			{
				_enabled = true;
				Enable();
			}

			return _count;
		}

		public virtual void Remove(int handle)
		{
			Device.BeginInvokeOnMainThread(() =>
			{
				_timeouts.RemoveAll(t => t.Item1 == handle);

				if (!_timeouts.Any())
				{
					_enabled = false;
					Disable();
				}
			});
		}

		protected virtual void DisableTimer()
		{
			_timer.Change(Timeout.Infinite, Timeout.Infinite);
		}

		protected virtual void EnableTimer()
		{
			_timer.Change(16, 16);
		}

		protected void SendSignals(int timestep = -1)
		{
			long step = timestep >= 0 ? timestep : _stopwatch.ElapsedMilliseconds;
			_stopwatch.Reset();
			_stopwatch.Start();

			var localCopy = new List<Tuple<int, Func<long, bool>>>(_timeouts);
			foreach (Tuple<int, Func<long, bool>> timeout in localCopy)
			{
				bool remove = !timeout.Item2(step);
				if (remove)
					_timeouts.RemoveAll(t => t.Item1 == timeout.Item1);
			}

			if (!_timeouts.Any())
			{
				_enabled = false;
				Disable();
			}
		}

		void Disable()
		{
			_stopwatch.Reset();
			DisableTimer();
		}

		void Enable()
		{
			_stopwatch.Start();
			EnableTimer();
		}

		void HandleElapsed(object state)
		{
			if (_timeouts.Count > 0)
			{
				_sync.Post(o => SendSignals(), null);
				_stopwatch.Reset();
				_stopwatch.Start();
			}
		}
	}
}