using System; using System.Collections.Generic; using UnityEngine; using UnityEngine.LowLevel; using Unity.Services.Core.Internal; using NotNull = JetBrains.Annotations.NotNullAttribute; namespace Unity.Services.Core.Scheduler.Internal { class ActionScheduler : IActionScheduler { readonly ITimeProvider m_TimeProvider; /// /// Members requiring thread safety: /// * . /// * . /// * . /// readonly object m_Lock = new object(); readonly MinimumBinaryHeap m_ScheduledActions = new MinimumBinaryHeap(new ScheduledInvocationComparer()); readonly Dictionary m_IdScheduledInvocationMap = new Dictionary(); const long k_MinimumIdValue = 1; internal readonly PlayerLoopSystem SchedulerLoopSystem; long m_NextId = k_MinimumIdValue; public ActionScheduler() : this(new UtcTimeProvider()) {} public ActionScheduler(ITimeProvider timeProvider) { m_TimeProvider = timeProvider; SchedulerLoopSystem = new PlayerLoopSystem { type = typeof(ActionScheduler), updateDelegate = ExecuteExpiredActions }; } public int ScheduledActionsCount => m_ScheduledActions.Count; public long ScheduleAction([NotNull] Action action, double delaySeconds = 0) { if (delaySeconds < 0) { throw new ArgumentException("delaySeconds can not be negative"); } if (action is null) { throw new ArgumentNullException(nameof(action)); } lock (m_Lock) { var scheduledInvocation = new ScheduledInvocation { Action = action, InvocationTime = m_TimeProvider.Now.AddSeconds(delaySeconds), ActionId = m_NextId++ }; if (m_NextId < k_MinimumIdValue) { m_NextId = k_MinimumIdValue; } m_ScheduledActions.Insert(scheduledInvocation); m_IdScheduledInvocationMap.Add(scheduledInvocation.ActionId, scheduledInvocation); return scheduledInvocation.ActionId; } } public void CancelAction(long actionId) { lock (m_Lock) { if (!m_IdScheduledInvocationMap.TryGetValue(actionId, out var scheduledInvocation)) { return; } m_ScheduledActions.Remove(scheduledInvocation); m_IdScheduledInvocationMap.Remove(scheduledInvocation.ActionId); } } internal void ExecuteExpiredActions() { var scheduledInvocationList = new List(); lock (m_Lock) { while (m_ScheduledActions.Count > 0 && m_ScheduledActions.Min?.InvocationTime <= m_TimeProvider.Now) { var scheduledInvocation = m_ScheduledActions.ExtractMin(); scheduledInvocationList.Add(scheduledInvocation); m_ScheduledActions.Remove(scheduledInvocation); m_IdScheduledInvocationMap.Remove(scheduledInvocation.ActionId); } } foreach (var scheduledInv in scheduledInvocationList) { try { scheduledInv.Action(); } catch (Exception e) { CoreLogger.LogException(e); } } } #if UNITY_EDITOR [RuntimeInitializeOnLoadMethod(RuntimeInitializeLoadType.SubsystemRegistration)] static void ClearActionSchedulerFromPlayerLoop() { var currentPlayerLoop = PlayerLoop.GetCurrentPlayerLoop(); var currentSubSystems = new List(currentPlayerLoop.subSystemList); for (var i = currentSubSystems.Count - 1; i >= 0; i--) { if (currentSubSystems[i].type == typeof(ActionScheduler)) { currentSubSystems.RemoveAt(i); } } UpdateSubSystemList(currentSubSystems, currentPlayerLoop); } #endif static void UpdateSubSystemList(List subSystemList, PlayerLoopSystem currentPlayerLoop) { currentPlayerLoop.subSystemList = subSystemList.ToArray(); PlayerLoop.SetPlayerLoop(currentPlayerLoop); } public void JoinPlayerLoopSystem() { var currentPlayerLoop = PlayerLoop.GetCurrentPlayerLoop(); var currentSubSystems = new List(currentPlayerLoop.subSystemList); if (!currentSubSystems.Contains(SchedulerLoopSystem)) { currentSubSystems.Add(SchedulerLoopSystem); UpdateSubSystemList(currentSubSystems, currentPlayerLoop); } } public void QuitPlayerLoopSystem() { var currentPlayerLoop = PlayerLoop.GetCurrentPlayerLoop(); var currentSubSystems = new List(currentPlayerLoop.subSystemList); currentSubSystems.Remove(SchedulerLoopSystem); UpdateSubSystemList(currentSubSystems, currentPlayerLoop); } } }