2020-07-28 15:59:24 +02:00
|
|
|
|
using QSB.Messaging;
|
2020-02-28 22:07:39 +01:00
|
|
|
|
using UnityEngine;
|
2020-02-24 19:55:16 +01:00
|
|
|
|
using UnityEngine.Networking;
|
2020-02-28 22:07:39 +01:00
|
|
|
|
using UnityEngine.SceneManagement;
|
2020-02-14 22:14:24 +01:00
|
|
|
|
|
2020-02-24 19:55:16 +01:00
|
|
|
|
namespace QSB.TimeSync
|
2020-02-15 20:48:02 +01:00
|
|
|
|
{
|
2020-02-24 19:55:16 +01:00
|
|
|
|
public class WakeUpSync : NetworkBehaviour
|
2020-02-15 20:48:02 +01:00
|
|
|
|
{
|
2020-02-28 22:07:39 +01:00
|
|
|
|
private const float TimeThreshold = 0.5f;
|
2020-03-14 15:42:51 +01:00
|
|
|
|
private const float MaxFastForwardSpeed = 60f;
|
|
|
|
|
private const float MaxFastForwardDiff = 20f;
|
|
|
|
|
private const float MinFastForwardSpeed = 2f;
|
2020-02-28 22:07:39 +01:00
|
|
|
|
|
2020-03-13 20:44:32 +01:00
|
|
|
|
private enum State { NotLoaded, Loaded, FastForwarding, Pausing }
|
2020-02-28 22:07:39 +01:00
|
|
|
|
private State _state = State.NotLoaded;
|
|
|
|
|
|
2020-02-21 23:36:07 +01:00
|
|
|
|
private MessageHandler<WakeUpMessage> _wakeUpHandler;
|
2020-02-28 22:07:39 +01:00
|
|
|
|
|
|
|
|
|
private float _sendTimer;
|
|
|
|
|
private float _serverTime;
|
2020-02-29 22:12:09 +01:00
|
|
|
|
private float _timeScale;
|
|
|
|
|
private bool _isInputEnabled = true;
|
2020-03-13 20:44:32 +01:00
|
|
|
|
private int _localLoopCount;
|
|
|
|
|
private int _serverLoopCount;
|
2020-02-15 20:48:02 +01:00
|
|
|
|
|
|
|
|
|
private void Start()
|
|
|
|
|
{
|
2020-02-24 19:55:16 +01:00
|
|
|
|
if (!isLocalPlayer)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
2020-02-28 22:07:39 +01:00
|
|
|
|
|
2020-08-09 11:11:45 +02:00
|
|
|
|
_wakeUpHandler = new MessageHandler<WakeUpMessage>(MessageType.WakeUp);
|
2020-02-21 23:36:07 +01:00
|
|
|
|
_wakeUpHandler.OnClientReceiveMessage += OnClientReceiveMessage;
|
2020-03-13 20:44:32 +01:00
|
|
|
|
|
|
|
|
|
var sceneName = SceneManager.GetActiveScene().name;
|
|
|
|
|
if (sceneName == "SolarSystem" || sceneName == "EyeOfTheUniverse")
|
|
|
|
|
{
|
|
|
|
|
Init();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
SceneManager.sceneLoaded += OnSceneLoaded;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
GlobalMessenger.AddListener("RestartTimeLoop", OnLoopStart);
|
2020-02-14 22:14:24 +01:00
|
|
|
|
}
|
|
|
|
|
|
2020-02-28 22:07:39 +01:00
|
|
|
|
private void OnSceneLoaded(Scene scene, LoadSceneMode mode)
|
|
|
|
|
{
|
2020-03-13 20:44:32 +01:00
|
|
|
|
if (scene.name == "SolarSystem" || scene.name == "EyeOfTheUniverse")
|
|
|
|
|
{
|
|
|
|
|
Init();
|
|
|
|
|
}
|
|
|
|
|
else
|
2020-02-28 22:07:39 +01:00
|
|
|
|
{
|
2020-03-13 20:44:32 +01:00
|
|
|
|
Reset();
|
2020-02-28 22:07:39 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-13 20:44:32 +01:00
|
|
|
|
private void OnLoopStart()
|
|
|
|
|
{
|
|
|
|
|
_localLoopCount++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Init()
|
2020-02-15 20:48:02 +01:00
|
|
|
|
{
|
2020-03-13 20:44:32 +01:00
|
|
|
|
_state = State.Loaded;
|
2020-02-29 09:39:15 +01:00
|
|
|
|
gameObject.AddComponent<PreserveTimeScale>();
|
2020-02-28 22:07:39 +01:00
|
|
|
|
if (isServer)
|
2020-02-15 20:48:02 +01:00
|
|
|
|
{
|
2020-02-28 22:07:39 +01:00
|
|
|
|
SendServerTime();
|
2020-02-14 22:14:24 +01:00
|
|
|
|
}
|
2020-02-28 22:07:39 +01:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
WakeUpOrSleep();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-13 20:44:32 +01:00
|
|
|
|
private void Reset()
|
|
|
|
|
{
|
|
|
|
|
_state = State.NotLoaded;
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-28 22:07:39 +01:00
|
|
|
|
private void SendServerTime()
|
|
|
|
|
{
|
|
|
|
|
var message = new WakeUpMessage
|
|
|
|
|
{
|
2020-03-13 20:44:32 +01:00
|
|
|
|
ServerTime = Time.timeSinceLevelLoad,
|
|
|
|
|
LoopCount = _localLoopCount
|
2020-02-28 22:07:39 +01:00
|
|
|
|
};
|
|
|
|
|
_wakeUpHandler.SendToAll(message);
|
2020-02-14 22:14:24 +01:00
|
|
|
|
}
|
|
|
|
|
|
2020-02-21 23:36:07 +01:00
|
|
|
|
private void OnClientReceiveMessage(WakeUpMessage message)
|
2020-02-15 20:48:02 +01:00
|
|
|
|
{
|
2020-02-24 19:55:16 +01:00
|
|
|
|
if (isServer)
|
2020-02-15 20:48:02 +01:00
|
|
|
|
{
|
2020-02-14 22:14:24 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
2020-02-28 22:07:39 +01:00
|
|
|
|
_serverTime = message.ServerTime;
|
2020-03-13 20:44:32 +01:00
|
|
|
|
_serverLoopCount = message.LoopCount;
|
2020-02-28 22:07:39 +01:00
|
|
|
|
WakeUpOrSleep();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void WakeUpOrSleep()
|
|
|
|
|
{
|
2020-03-13 20:44:32 +01:00
|
|
|
|
if (_state == State.NotLoaded || _localLoopCount != _serverLoopCount)
|
2020-02-28 22:07:39 +01:00
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var myTime = Time.timeSinceLevelLoad;
|
|
|
|
|
var diff = myTime - _serverTime;
|
|
|
|
|
|
|
|
|
|
if (diff > TimeThreshold)
|
|
|
|
|
{
|
|
|
|
|
StartPausing();
|
|
|
|
|
return;
|
|
|
|
|
}
|
2020-02-14 22:14:24 +01:00
|
|
|
|
|
2020-02-28 22:07:39 +01:00
|
|
|
|
if (diff < -TimeThreshold)
|
|
|
|
|
{
|
2020-07-28 15:59:24 +02:00
|
|
|
|
StartFastForwarding();
|
2020-02-28 22:07:39 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-28 15:59:24 +02:00
|
|
|
|
private void StartFastForwarding()
|
2020-02-28 22:07:39 +01:00
|
|
|
|
{
|
2020-02-29 22:12:09 +01:00
|
|
|
|
if (_state == State.FastForwarding)
|
2020-02-28 22:07:39 +01:00
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
2020-03-14 15:42:51 +01:00
|
|
|
|
_timeScale = MaxFastForwardSpeed;
|
2020-02-29 22:12:09 +01:00
|
|
|
|
_state = State.FastForwarding;
|
2020-02-28 22:07:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void StartPausing()
|
|
|
|
|
{
|
|
|
|
|
if (_state == State.Pausing)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
2020-02-29 22:12:09 +01:00
|
|
|
|
_timeScale = 0f;
|
2020-02-28 22:07:39 +01:00
|
|
|
|
_state = State.Pausing;
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-29 22:12:09 +01:00
|
|
|
|
private void ResetTimeScale()
|
2020-02-28 22:07:39 +01:00
|
|
|
|
{
|
2020-02-29 22:12:09 +01:00
|
|
|
|
_timeScale = 1f;
|
2020-03-13 20:44:32 +01:00
|
|
|
|
_state = State.Loaded;
|
2020-02-29 22:12:09 +01:00
|
|
|
|
|
|
|
|
|
if (!_isInputEnabled)
|
2020-02-28 22:07:39 +01:00
|
|
|
|
{
|
2020-02-29 22:12:09 +01:00
|
|
|
|
EnableInput();
|
2020-02-28 22:07:39 +01:00
|
|
|
|
}
|
2020-02-29 22:12:09 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void DisableInput()
|
|
|
|
|
{
|
|
|
|
|
_isInputEnabled = false;
|
|
|
|
|
OWInput.ChangeInputMode(InputMode.None);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void EnableInput()
|
|
|
|
|
{
|
|
|
|
|
_isInputEnabled = true;
|
|
|
|
|
OWInput.ChangeInputMode(InputMode.Character);
|
2020-02-28 22:07:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Update()
|
|
|
|
|
{
|
|
|
|
|
if (isServer)
|
|
|
|
|
{
|
|
|
|
|
UpdateServer();
|
|
|
|
|
}
|
|
|
|
|
else if (isLocalPlayer)
|
|
|
|
|
{
|
|
|
|
|
UpdateLocal();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void UpdateServer()
|
|
|
|
|
{
|
2020-03-13 20:44:32 +01:00
|
|
|
|
if (_state != State.Loaded)
|
2020-02-28 22:07:39 +01:00
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_sendTimer += Time.unscaledDeltaTime;
|
|
|
|
|
if (_sendTimer > 1)
|
|
|
|
|
{
|
|
|
|
|
SendServerTime();
|
|
|
|
|
_sendTimer = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void UpdateLocal()
|
|
|
|
|
{
|
|
|
|
|
_serverTime += Time.unscaledDeltaTime;
|
|
|
|
|
|
2020-03-13 20:44:32 +01:00
|
|
|
|
if (_state == State.NotLoaded)
|
2020-02-28 22:07:39 +01:00
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-14 15:42:51 +01:00
|
|
|
|
if (_state == State.FastForwarding)
|
|
|
|
|
{
|
|
|
|
|
var diff = _serverTime - Time.timeSinceLevelLoad;
|
|
|
|
|
Time.timeScale = Mathf.Lerp(MinFastForwardSpeed, MaxFastForwardSpeed, Mathf.Abs(diff) / MaxFastForwardDiff);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
Time.timeScale = _timeScale;
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-28 15:59:24 +02:00
|
|
|
|
var isDoneFastForwarding = _state == State.FastForwarding && Time.timeSinceLevelLoad >= _serverTime;
|
|
|
|
|
var isDonePausing = _state == State.Pausing && Time.timeSinceLevelLoad < _serverTime;
|
2020-02-29 22:12:09 +01:00
|
|
|
|
|
|
|
|
|
if (isDoneFastForwarding || isDonePausing)
|
2020-02-28 22:07:39 +01:00
|
|
|
|
{
|
2020-02-29 22:12:09 +01:00
|
|
|
|
ResetTimeScale();
|
2020-02-28 22:07:39 +01:00
|
|
|
|
}
|
2020-02-29 22:12:09 +01:00
|
|
|
|
|
|
|
|
|
if (!_isInputEnabled && OWInput.GetInputMode() != InputMode.None)
|
2020-02-28 22:07:39 +01:00
|
|
|
|
{
|
2020-02-29 22:12:09 +01:00
|
|
|
|
DisableInput();
|
2020-02-28 22:07:39 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-14 22:14:24 +01:00
|
|
|
|
}
|
|
|
|
|
}
|