Net.Like.Xue.Tokyo/Assets/BITKit/Unity/Scripts/Tick/GameTickService.cs

140 lines
3.4 KiB
C#
Raw Normal View History

2024-11-03 16:42:23 +08:00
using System;
using System.Collections;
using System.Collections.Generic;
using System.Globalization;
using System.Timers;
using Cysharp.Threading.Tasks;
#if UNITY_EDITOR
using UnityEditor;
#endif
using UnityEngine;
2025-04-17 19:36:08 +08:00
using UnityEngine.LowLevel;
2024-11-03 16:42:23 +08:00
namespace BITKit
{
[Serializable]
public sealed class GameTick:ITicker
{
public ulong TickCount => GameTickService.TickCount;
public void Add(Action action)=>GameTickService.Add(action);
public void Add(Action<float> action)=>GameTickService.Add(action);
public void Remove(Action<float> action)=>GameTickService.Remove(action);
public void ManualTick(float delta)
{
throw new NotImplementedException();
}
}
[CustomType(typeof(ITicker))]
public class GameTickService : MonoBehaviour,ITicker
{
[RuntimeInitializeOnLoadMethod]
private static void Reload()
{
_ActionQueue.Clear();
//_TickActions.Clear();
TickCount = ulong.MinValue;
}
private static readonly Queue<Action> _ActionQueue = new();
// private static readonly CacheList<Action<float>> _TickActions = new();
// public static void Add(Action<float> action) => _TickActions.Add(action);
// public static void Remove(Action<float> action) => _TickActions.Remove(action);
private static event Action<float> _TickEvents;
public static void Add( Action<float> action)=>_TickEvents += action;
public static void Remove( Action<float> action)=>_TickEvents -= action;
public static void Add(Action action)=>_ActionQueue.Enqueue(action);
public static ulong TickCount { get; private set; }
ulong ITicker.TickCount => TickCount;
void ITicker.Add(Action action)=>Add(action);
void ITicker.Add(Action<float> action)=>Add(action);
void ITicker.Remove(Action<float> action)=>Remove(action);
public void ManualTick(float delta)
{
throw new NotImplementedException();
}
[SerializeField] private int tickRate = 32;
[SerializeField] private bool isConcurrent;
[SerializeField] private string lastTickTime="0";
private readonly Timer _timer = new();
2025-04-17 19:36:08 +08:00
private double _deltaTime;
2024-11-03 16:42:23 +08:00
2025-04-17 19:36:08 +08:00
private PlayerLoopSystem _playerLoop;
private int _update;
private float _lastTickTime;
2024-11-03 16:42:23 +08:00
private void Start()
{
2025-04-17 19:36:08 +08:00
_playerLoop = PlayerLoop.GetCurrentPlayerLoop();
2024-11-03 16:42:23 +08:00
tickRate = Mathf.Clamp(tickRate, 1, 128);
2025-04-17 19:36:08 +08:00
_deltaTime = 1d / tickRate;
2024-11-03 16:42:23 +08:00
_timer.Elapsed += Tick;
_timer.AutoReset = isConcurrent;
_timer.Interval = TimeSpan.FromSeconds(_deltaTime).TotalMilliseconds;
_timer.Start();
2025-04-17 19:36:08 +08:00
destroyCancellationToken.Register(_timer.Stop);
destroyCancellationToken.Register(_timer.Dispose);
2024-11-03 16:42:23 +08:00
}
2025-04-17 19:36:08 +08:00
private void Update()
2024-11-03 16:42:23 +08:00
{
#if UNITY_EDITOR
2025-04-17 19:36:08 +08:00
if (EditorApplication.isPlaying is false)
{
_update = 0;
_lastTickTime = Time.time;
return;
}
2024-11-03 16:42:23 +08:00
#endif
2025-04-17 19:36:08 +08:00
for (var i = 0; i < _update; i++)
{
var delta = Time.time - _lastTickTime;
_lastTickTime = Time.time;
2024-11-03 16:42:23 +08:00
while (_ActionQueue.TryDequeue(out var action))
{
action?.Invoke();
}
2025-04-17 19:36:08 +08:00
_TickEvents?.Invoke(delta);
}
_update = 0;
}
private void Tick(object sender, ElapsedEventArgs e)
{
TickCount++;
try
{
_update++;
2024-11-03 16:42:23 +08:00
}
catch (OperationCanceledException)
{
return;
}
catch (Exception exception)
{
BIT4Log.LogException(exception);
}
lastTickTime = BITApp.Time.TimeAsDouble.ToString(CultureInfo.InvariantCulture);
Restart();
return;
void Restart()
{
if (isConcurrent is false && destroyCancellationToken.IsCancellationRequested is false)
{
_timer.Start();
}
}
}
}
}