Local Time Scale Scripting
v1.0
Check the LocalTimeScale
class to view all of the methods available.
Getting the Time Values
The LocalTimeScale
will have its own time values similar to the Time
class. You can get these using DeltaTime
, UnscaledDeltaTime
, and FixedDeltaTime
.
LocalTimeScale localTimeScale;
float _counter = 0f;
float _fixedDeltaTime;
_counter += localTimeScale.DeltaTime;
_fixedCounter += localTimeScale.FixedDeltaTime;
Getting & Setting the Value
The "Value" is a float
representing the percent of "real time". 1f
is essentially 100% of Time.deltaTime
. The Value
property will return the current value, and you can also specifically set this value as well using SetValue(float newValue)
.
If you're using the MagicTimeUser
or IHaveLocalTime
interface, often you'll want to know the value of the combined time scales affecting that object. In that case, use the TimeScale
property.
// Get the value
LocalTimeScale localTimeScale;
var currentValue = localTimeScale.Value;
// Set the value
float newValue = 0.5f;
localTimeScale.SetValue(newValue);
When the value is changed, the UnityEvent OnTimeScaleChangedUnityEvent
and Event OnTimeScaleChanged
will be triggered for any subscribers.
If the value is changed to 0f
(with or without using the Pause()
method), the OnPaused
and OnPausedUnityEvent
events will be called.
Similarly, if the value changes from 0f
to something else, even without using the Resume()
method, the OnResumed
and OnResumedUnityEvent
will be called.
Pausing and Resuming
The Pause()
and Resume()
methods will set value to 0f
, and then return the value to the last cached value. If you know the resumed value should be the same as what it was prior to the "pause" action, then this is often a simpler way of handling this logic. Otherwise, you can always just SetValue(0f)
manually.
// Pause this specific localTimeScale
localTimeScale.Pause(); // Will cache the current Value before setting it to 0f.
// Resume this specific localTimeScale
localTimeScale.Resume(); // This will reset to the value when Pause() was called.
Subscribing
Any IHaveLocalTime
object can subscribe to a LocalTimeScale
object. This will ensure the object gets event notifications from the LocalTimeScale
.
Remember, the actual calculations for combined time scale will be handled by the IHaveLocalTime
object.
// This is from the "SubscribeToLocalTimeScale(LocalTimeScale myTimeScale)"
// method on MagicTimeUser.cs
// Can't subscribe to the same timeScale twice.
if (SubscribedTimeScales.Contains(myTimeScale)) return;
SubscribedTimeScales.Add(myTimeScale);
myTimeScale.Subscribe(this);
myTimeScale.OnTimeScaleChanged += OnTimeScaleChanged;
// In this example, we keep track of the timeScales we're subscribed to, and subscribe
// to them at the same time.
When subscribing, the OnSubscribed
and OnSubscribedUnityEvent
will be called.=
Information About Subscribers
Subscribers are stored in a HashSet<IHaveLocalTime>
so there is no index
available. There are a few helpful properties that provide more information about subscribers to a specific Local Time Scale.
LocalTimeScale localTimeScale;
// Check the number of subscribers
if (localTimeScale.SubscriberCount == 0)
return;
// Get the string name of those subscribed
foreach(var subscriberName in localTimeScale.SubscriberNames)
{
Debug.Log($"{subscriberName} is subscribed");
}
// Get the GameObject of those subscribed
foreach(var subscriberObject in SubscriberGameObjects)
{
DoSomethingWithSubscriber(subscriberObject);
}
Unsubscribing
You can use the UnsubscribeAll()
method to call Unsubscribe()
on all current subscribers.
The Unsubscribe(IHaveLocalTime subscriber)
method does a few things, if the subscriber
passed in is actually subscribed.
First, it will remove the subscriber
from the Subscribers
list, as you might expect. But it also will call subscriber.UnsubscribeFromLocalTimeScale(this)
, which allows the subscriber to take any actions as needed.
// This is the method from MagicTimeUser.cs
public virtual void UnsubscribeFromLocalTimeScale(LocalTimeScale timeScale)
{
if (timeScale == null)
return;
// If it is in our list, then unsubscribe and remove it.
if (SubscribedTimeScales.Contains(timeScale))
{
timeScale.Unsubscribe(this);
SubscribedTimeScales.Remove(timeScale);
}
// This can be called from the LocalTimeScale, so we may need to reset the desired timeScale,
// even if the timeScale is not in our list.
ResetDesiredTimeScale();
}
Next, the OnUnsubscribed
and OnUnsubscribedUnityEvent
will be called as well, on the Local Time Scale object. Both of these will pass the subscriber (IHaveLocalTime
) object in as a parameter.
Finally, if AutoRemoveWhenEmpty
is true
, the time scale will be removed. See details below.
Automatically Removing the LocalTimeScale
LocalTimeScale
In some cases, you may wish to automatically remove a LocalTimeScale when there are no subscribers.
localTimeScale.AutoRemoveWhenEmpty = true;
Set AutoRemoveWhenEmpty
to true
, and then whenever subscribers Unsubscribe()
from the Local Time Scale, if there are no subscribers left, then the MagicTimeManager
method RemoveTimeScale()
method will be called, removing this object.
The check only happens when the Unsubscribe()
method is called, so you can safely set this value to true
before any subscribers are active.
Subscribing to Events
Classes can subscribe to individual events on the LocalTimeScale
class. This is useful to perform actions based on changes to the Local Time Scale.
In the example below, from ZoneColorChanger.cs
in the demo, we subscribe and unsubscribe to events on the LocalTimeScale
attached to a specific time zone.
In this case, the LocalTimeScale
instance is created at runtime, so it will not be availalbe immediately. To handle this, we can subscribe via a Coroutine, which is started OnEnable()
.
private void OnEnable() => StartCoroutine(SubscribeToTimeScale());
private void OnDisable()
{
timeZone.TimeScale.OnSubscribed -= OnSubscribed;
timeZone.TimeScale.OnUnsubscribed -= OnUnsubscribed;
}
private IEnumerator SubscribeToTimeScale()
{
yield return new WaitUntil(() => timeZone.TimeScale != null);
timeZone.TimeScale.OnSubscribed += OnSubscribed;
timeZone.TimeScale.OnUnsubscribed += OnUnsubscribed;
}
// Handle the events -- make sure the object passed is the right type
private void OnSubscribed(object go)
{
var newObject = (IHaveLocalTime)go;
if (newObject == null) return;
var newGameObject = newObject.GameObject;
if (newGameObject == null) return;
AddColor(newGameObject);
}
private void OnUnsubscribed(object go)
{
var newObject = (IHaveLocalTime)go;
if (newObject == null) return;
var newGameObject = newObject.GameObject;
if (newGameObject == null) return;
RemoveColor(newGameObject);
}
Last updated
Was this helpful?