638 lines
22 KiB
C#
638 lines
22 KiB
C#
#pragma warning disable CS1591 // Missing XML comment for publicly visible type or member
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Threading;
|
|
using Cysharp.Threading.Tasks.Internal;
|
|
|
|
namespace Cysharp.Threading.Tasks
|
|
{
|
|
public partial struct UniTask
|
|
{
|
|
public static UniTask WaitUntil(Func<bool> predicate, PlayerLoopTiming timing = PlayerLoopTiming.Update, CancellationToken cancellationToken = default(CancellationToken), bool cancelImmediately = false)
|
|
{
|
|
return new UniTask(WaitUntilPromise.Create(predicate, timing, cancellationToken, cancelImmediately, out var token), token);
|
|
}
|
|
|
|
public static UniTask WaitWhile(Func<bool> predicate, PlayerLoopTiming timing = PlayerLoopTiming.Update, CancellationToken cancellationToken = default(CancellationToken), bool cancelImmediately = false)
|
|
{
|
|
return new UniTask(WaitWhilePromise.Create(predicate, timing, cancellationToken, cancelImmediately, out var token), token);
|
|
}
|
|
|
|
public static UniTask WaitUntilCanceled(CancellationToken cancellationToken, PlayerLoopTiming timing = PlayerLoopTiming.Update, bool completeImmediately = false)
|
|
{
|
|
return new UniTask(WaitUntilCanceledPromise.Create(cancellationToken, timing, completeImmediately, out var token), token);
|
|
}
|
|
|
|
public static UniTask<U> WaitUntilValueChanged<T, U>(T target, Func<T, U> monitorFunction, PlayerLoopTiming monitorTiming = PlayerLoopTiming.Update, IEqualityComparer<U> equalityComparer = null, CancellationToken cancellationToken = default(CancellationToken), bool cancelImmediately = false)
|
|
where T : class
|
|
{
|
|
var unityObject = target as UnityEngine.Object;
|
|
var isUnityObject = target is UnityEngine.Object; // don't use (unityObject == null)
|
|
|
|
return new UniTask<U>(isUnityObject
|
|
? WaitUntilValueChangedUnityObjectPromise<T, U>.Create(target, monitorFunction, equalityComparer, monitorTiming, cancellationToken, cancelImmediately, out var token)
|
|
: WaitUntilValueChangedStandardObjectPromise<T, U>.Create(target, monitorFunction, equalityComparer, monitorTiming, cancellationToken, cancelImmediately, out token), token);
|
|
}
|
|
|
|
sealed class WaitUntilPromise : IUniTaskSource, IPlayerLoopItem, ITaskPoolNode<WaitUntilPromise>
|
|
{
|
|
static TaskPool<WaitUntilPromise> pool;
|
|
WaitUntilPromise nextNode;
|
|
public ref WaitUntilPromise NextNode => ref nextNode;
|
|
|
|
static WaitUntilPromise()
|
|
{
|
|
TaskPool.RegisterSizeGetter(typeof(WaitUntilPromise), () => pool.Size);
|
|
}
|
|
|
|
Func<bool> predicate;
|
|
CancellationToken cancellationToken;
|
|
CancellationTokenRegistration cancellationTokenRegistration;
|
|
|
|
UniTaskCompletionSourceCore<object> core;
|
|
|
|
WaitUntilPromise()
|
|
{
|
|
}
|
|
|
|
public static IUniTaskSource Create(Func<bool> predicate, PlayerLoopTiming timing, CancellationToken cancellationToken, bool cancelImmediately, out short token)
|
|
{
|
|
if (cancellationToken.IsCancellationRequested)
|
|
{
|
|
return AutoResetUniTaskCompletionSource.CreateFromCanceled(cancellationToken, out token);
|
|
}
|
|
|
|
if (!pool.TryPop(out var result))
|
|
{
|
|
result = new WaitUntilPromise();
|
|
}
|
|
|
|
result.predicate = predicate;
|
|
result.cancellationToken = cancellationToken;
|
|
|
|
if (cancelImmediately && cancellationToken.CanBeCanceled)
|
|
{
|
|
result.cancellationTokenRegistration = cancellationToken.RegisterWithoutCaptureExecutionContext(state =>
|
|
{
|
|
var promise = (WaitUntilPromise)state;
|
|
promise.core.TrySetCanceled(promise.cancellationToken);
|
|
}, result);
|
|
}
|
|
|
|
TaskTracker.TrackActiveTask(result, 3);
|
|
|
|
PlayerLoopHelper.AddAction(timing, result);
|
|
|
|
token = result.core.Version;
|
|
return result;
|
|
}
|
|
|
|
public void GetResult(short token)
|
|
{
|
|
try
|
|
{
|
|
core.GetResult(token);
|
|
}
|
|
finally
|
|
{
|
|
TryReturn();
|
|
}
|
|
}
|
|
|
|
public UniTaskStatus GetStatus(short token)
|
|
{
|
|
return core.GetStatus(token);
|
|
}
|
|
|
|
public UniTaskStatus UnsafeGetStatus()
|
|
{
|
|
return core.UnsafeGetStatus();
|
|
}
|
|
|
|
public void OnCompleted(Action<object> continuation, object state, short token)
|
|
{
|
|
core.OnCompleted(continuation, state, token);
|
|
}
|
|
|
|
public bool MoveNext()
|
|
{
|
|
if (cancellationToken.IsCancellationRequested)
|
|
{
|
|
core.TrySetCanceled(cancellationToken);
|
|
return false;
|
|
}
|
|
|
|
try
|
|
{
|
|
if (!predicate())
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
core.TrySetException(ex);
|
|
return false;
|
|
}
|
|
|
|
core.TrySetResult(null);
|
|
return false;
|
|
}
|
|
|
|
bool TryReturn()
|
|
{
|
|
TaskTracker.RemoveTracking(this);
|
|
core.Reset();
|
|
predicate = default;
|
|
cancellationToken = default;
|
|
cancellationTokenRegistration.Dispose();
|
|
return pool.TryPush(this);
|
|
}
|
|
}
|
|
|
|
sealed class WaitWhilePromise : IUniTaskSource, IPlayerLoopItem, ITaskPoolNode<WaitWhilePromise>
|
|
{
|
|
static TaskPool<WaitWhilePromise> pool;
|
|
WaitWhilePromise nextNode;
|
|
public ref WaitWhilePromise NextNode => ref nextNode;
|
|
|
|
static WaitWhilePromise()
|
|
{
|
|
TaskPool.RegisterSizeGetter(typeof(WaitWhilePromise), () => pool.Size);
|
|
}
|
|
|
|
Func<bool> predicate;
|
|
CancellationToken cancellationToken;
|
|
CancellationTokenRegistration cancellationTokenRegistration;
|
|
|
|
UniTaskCompletionSourceCore<object> core;
|
|
|
|
WaitWhilePromise()
|
|
{
|
|
}
|
|
|
|
public static IUniTaskSource Create(Func<bool> predicate, PlayerLoopTiming timing, CancellationToken cancellationToken, bool cancelImmediately, out short token)
|
|
{
|
|
if (cancellationToken.IsCancellationRequested)
|
|
{
|
|
return AutoResetUniTaskCompletionSource.CreateFromCanceled(cancellationToken, out token);
|
|
}
|
|
|
|
if (!pool.TryPop(out var result))
|
|
{
|
|
result = new WaitWhilePromise();
|
|
}
|
|
|
|
result.predicate = predicate;
|
|
result.cancellationToken = cancellationToken;
|
|
|
|
if (cancelImmediately && cancellationToken.CanBeCanceled)
|
|
{
|
|
result.cancellationTokenRegistration = cancellationToken.RegisterWithoutCaptureExecutionContext(state =>
|
|
{
|
|
var promise = (WaitWhilePromise)state;
|
|
promise.core.TrySetCanceled(promise.cancellationToken);
|
|
}, result);
|
|
}
|
|
|
|
TaskTracker.TrackActiveTask(result, 3);
|
|
|
|
PlayerLoopHelper.AddAction(timing, result);
|
|
|
|
token = result.core.Version;
|
|
return result;
|
|
}
|
|
|
|
public void GetResult(short token)
|
|
{
|
|
try
|
|
{
|
|
core.GetResult(token);
|
|
}
|
|
finally
|
|
{
|
|
TryReturn();
|
|
}
|
|
}
|
|
|
|
public UniTaskStatus GetStatus(short token)
|
|
{
|
|
return core.GetStatus(token);
|
|
}
|
|
|
|
public UniTaskStatus UnsafeGetStatus()
|
|
{
|
|
return core.UnsafeGetStatus();
|
|
}
|
|
|
|
public void OnCompleted(Action<object> continuation, object state, short token)
|
|
{
|
|
core.OnCompleted(continuation, state, token);
|
|
}
|
|
|
|
public bool MoveNext()
|
|
{
|
|
if (cancellationToken.IsCancellationRequested)
|
|
{
|
|
core.TrySetCanceled(cancellationToken);
|
|
return false;
|
|
}
|
|
|
|
try
|
|
{
|
|
if (predicate())
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
core.TrySetException(ex);
|
|
return false;
|
|
}
|
|
|
|
core.TrySetResult(null);
|
|
return false;
|
|
}
|
|
|
|
bool TryReturn()
|
|
{
|
|
TaskTracker.RemoveTracking(this);
|
|
core.Reset();
|
|
predicate = default;
|
|
cancellationToken = default;
|
|
cancellationTokenRegistration.Dispose();
|
|
return pool.TryPush(this);
|
|
}
|
|
}
|
|
|
|
sealed class WaitUntilCanceledPromise : IUniTaskSource, IPlayerLoopItem, ITaskPoolNode<WaitUntilCanceledPromise>
|
|
{
|
|
static TaskPool<WaitUntilCanceledPromise> pool;
|
|
WaitUntilCanceledPromise nextNode;
|
|
public ref WaitUntilCanceledPromise NextNode => ref nextNode;
|
|
|
|
static WaitUntilCanceledPromise()
|
|
{
|
|
TaskPool.RegisterSizeGetter(typeof(WaitUntilCanceledPromise), () => pool.Size);
|
|
}
|
|
|
|
CancellationToken cancellationToken;
|
|
CancellationTokenRegistration cancellationTokenRegistration;
|
|
|
|
UniTaskCompletionSourceCore<object> core;
|
|
|
|
WaitUntilCanceledPromise()
|
|
{
|
|
}
|
|
|
|
public static IUniTaskSource Create(CancellationToken cancellationToken, PlayerLoopTiming timing, bool completeImmediately, out short token)
|
|
{
|
|
if (cancellationToken.IsCancellationRequested)
|
|
{
|
|
return AutoResetUniTaskCompletionSource.CreateFromCanceled(cancellationToken, out token);
|
|
}
|
|
|
|
if (!pool.TryPop(out var result))
|
|
{
|
|
result = new WaitUntilCanceledPromise();
|
|
}
|
|
|
|
result.cancellationToken = cancellationToken;
|
|
|
|
if (completeImmediately && cancellationToken.CanBeCanceled)
|
|
{
|
|
result.cancellationTokenRegistration = cancellationToken.RegisterWithoutCaptureExecutionContext(state =>
|
|
{
|
|
var promise = (WaitUntilCanceledPromise)state;
|
|
promise.core.TrySetResult(null);
|
|
}, result);
|
|
}
|
|
|
|
TaskTracker.TrackActiveTask(result, 3);
|
|
|
|
PlayerLoopHelper.AddAction(timing, result);
|
|
|
|
token = result.core.Version;
|
|
return result;
|
|
}
|
|
|
|
public void GetResult(short token)
|
|
{
|
|
try
|
|
{
|
|
core.GetResult(token);
|
|
}
|
|
finally
|
|
{
|
|
TryReturn();
|
|
}
|
|
}
|
|
|
|
public UniTaskStatus GetStatus(short token)
|
|
{
|
|
return core.GetStatus(token);
|
|
}
|
|
|
|
public UniTaskStatus UnsafeGetStatus()
|
|
{
|
|
return core.UnsafeGetStatus();
|
|
}
|
|
|
|
public void OnCompleted(Action<object> continuation, object state, short token)
|
|
{
|
|
core.OnCompleted(continuation, state, token);
|
|
}
|
|
|
|
public bool MoveNext()
|
|
{
|
|
if (cancellationToken.IsCancellationRequested)
|
|
{
|
|
core.TrySetResult(null);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool TryReturn()
|
|
{
|
|
TaskTracker.RemoveTracking(this);
|
|
core.Reset();
|
|
cancellationToken = default;
|
|
cancellationTokenRegistration.Dispose();
|
|
return pool.TryPush(this);
|
|
}
|
|
}
|
|
|
|
// where T : UnityEngine.Object, can not add constraint
|
|
sealed class WaitUntilValueChangedUnityObjectPromise<T, U> : IUniTaskSource<U>, IPlayerLoopItem, ITaskPoolNode<WaitUntilValueChangedUnityObjectPromise<T, U>>
|
|
{
|
|
static TaskPool<WaitUntilValueChangedUnityObjectPromise<T, U>> pool;
|
|
WaitUntilValueChangedUnityObjectPromise<T, U> nextNode;
|
|
public ref WaitUntilValueChangedUnityObjectPromise<T, U> NextNode => ref nextNode;
|
|
|
|
static WaitUntilValueChangedUnityObjectPromise()
|
|
{
|
|
TaskPool.RegisterSizeGetter(typeof(WaitUntilValueChangedUnityObjectPromise<T, U>), () => pool.Size);
|
|
}
|
|
|
|
T target;
|
|
UnityEngine.Object targetAsUnityObject;
|
|
U currentValue;
|
|
Func<T, U> monitorFunction;
|
|
IEqualityComparer<U> equalityComparer;
|
|
CancellationToken cancellationToken;
|
|
CancellationTokenRegistration cancellationTokenRegistration;
|
|
|
|
UniTaskCompletionSourceCore<U> core;
|
|
|
|
WaitUntilValueChangedUnityObjectPromise()
|
|
{
|
|
}
|
|
|
|
public static IUniTaskSource<U> Create(T target, Func<T, U> monitorFunction, IEqualityComparer<U> equalityComparer, PlayerLoopTiming timing, CancellationToken cancellationToken, bool cancelImmediately, out short token)
|
|
{
|
|
if (cancellationToken.IsCancellationRequested)
|
|
{
|
|
return AutoResetUniTaskCompletionSource<U>.CreateFromCanceled(cancellationToken, out token);
|
|
}
|
|
|
|
if (!pool.TryPop(out var result))
|
|
{
|
|
result = new WaitUntilValueChangedUnityObjectPromise<T, U>();
|
|
}
|
|
|
|
result.target = target;
|
|
result.targetAsUnityObject = target as UnityEngine.Object;
|
|
result.monitorFunction = monitorFunction;
|
|
result.currentValue = monitorFunction(target);
|
|
result.equalityComparer = equalityComparer ?? UnityEqualityComparer.GetDefault<U>();
|
|
result.cancellationToken = cancellationToken;
|
|
|
|
if (cancelImmediately && cancellationToken.CanBeCanceled)
|
|
{
|
|
result.cancellationTokenRegistration = cancellationToken.RegisterWithoutCaptureExecutionContext(state =>
|
|
{
|
|
var promise = (WaitUntilValueChangedUnityObjectPromise<T, U>)state;
|
|
promise.core.TrySetCanceled(promise.cancellationToken);
|
|
}, result);
|
|
}
|
|
|
|
TaskTracker.TrackActiveTask(result, 3);
|
|
|
|
PlayerLoopHelper.AddAction(timing, result);
|
|
|
|
token = result.core.Version;
|
|
return result;
|
|
}
|
|
|
|
public U GetResult(short token)
|
|
{
|
|
try
|
|
{
|
|
return core.GetResult(token);
|
|
}
|
|
finally
|
|
{
|
|
TryReturn();
|
|
}
|
|
}
|
|
|
|
void IUniTaskSource.GetResult(short token)
|
|
{
|
|
GetResult(token);
|
|
}
|
|
|
|
public UniTaskStatus GetStatus(short token)
|
|
{
|
|
return core.GetStatus(token);
|
|
}
|
|
|
|
public UniTaskStatus UnsafeGetStatus()
|
|
{
|
|
return core.UnsafeGetStatus();
|
|
}
|
|
|
|
public void OnCompleted(Action<object> continuation, object state, short token)
|
|
{
|
|
core.OnCompleted(continuation, state, token);
|
|
}
|
|
|
|
public bool MoveNext()
|
|
{
|
|
if (cancellationToken.IsCancellationRequested || targetAsUnityObject == null) // destroyed = cancel.
|
|
{
|
|
core.TrySetCanceled(cancellationToken);
|
|
return false;
|
|
}
|
|
|
|
U nextValue = default(U);
|
|
try
|
|
{
|
|
nextValue = monitorFunction(target);
|
|
if (equalityComparer.Equals(currentValue, nextValue))
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
core.TrySetException(ex);
|
|
return false;
|
|
}
|
|
|
|
core.TrySetResult(nextValue);
|
|
return false;
|
|
}
|
|
|
|
bool TryReturn()
|
|
{
|
|
TaskTracker.RemoveTracking(this);
|
|
core.Reset();
|
|
target = default;
|
|
currentValue = default;
|
|
monitorFunction = default;
|
|
equalityComparer = default;
|
|
cancellationToken = default;
|
|
cancellationTokenRegistration.Dispose();
|
|
return pool.TryPush(this);
|
|
}
|
|
}
|
|
|
|
sealed class WaitUntilValueChangedStandardObjectPromise<T, U> : IUniTaskSource<U>, IPlayerLoopItem, ITaskPoolNode<WaitUntilValueChangedStandardObjectPromise<T, U>>
|
|
where T : class
|
|
{
|
|
static TaskPool<WaitUntilValueChangedStandardObjectPromise<T, U>> pool;
|
|
WaitUntilValueChangedStandardObjectPromise<T, U> nextNode;
|
|
public ref WaitUntilValueChangedStandardObjectPromise<T, U> NextNode => ref nextNode;
|
|
|
|
static WaitUntilValueChangedStandardObjectPromise()
|
|
{
|
|
TaskPool.RegisterSizeGetter(typeof(WaitUntilValueChangedStandardObjectPromise<T, U>), () => pool.Size);
|
|
}
|
|
|
|
WeakReference<T> target;
|
|
U currentValue;
|
|
Func<T, U> monitorFunction;
|
|
IEqualityComparer<U> equalityComparer;
|
|
CancellationToken cancellationToken;
|
|
CancellationTokenRegistration cancellationTokenRegistration;
|
|
|
|
UniTaskCompletionSourceCore<U> core;
|
|
|
|
WaitUntilValueChangedStandardObjectPromise()
|
|
{
|
|
}
|
|
|
|
public static IUniTaskSource<U> Create(T target, Func<T, U> monitorFunction, IEqualityComparer<U> equalityComparer, PlayerLoopTiming timing, CancellationToken cancellationToken, bool cancelImmediately, out short token)
|
|
{
|
|
if (cancellationToken.IsCancellationRequested)
|
|
{
|
|
return AutoResetUniTaskCompletionSource<U>.CreateFromCanceled(cancellationToken, out token);
|
|
}
|
|
|
|
if (!pool.TryPop(out var result))
|
|
{
|
|
result = new WaitUntilValueChangedStandardObjectPromise<T, U>();
|
|
}
|
|
|
|
result.target = new WeakReference<T>(target, false); // wrap in WeakReference.
|
|
result.monitorFunction = monitorFunction;
|
|
result.currentValue = monitorFunction(target);
|
|
result.equalityComparer = equalityComparer ?? UnityEqualityComparer.GetDefault<U>();
|
|
result.cancellationToken = cancellationToken;
|
|
|
|
if (cancelImmediately && cancellationToken.CanBeCanceled)
|
|
{
|
|
result.cancellationTokenRegistration = cancellationToken.RegisterWithoutCaptureExecutionContext(state =>
|
|
{
|
|
var promise = (WaitUntilValueChangedStandardObjectPromise<T, U>)state;
|
|
promise.core.TrySetCanceled(promise.cancellationToken);
|
|
}, result);
|
|
}
|
|
|
|
TaskTracker.TrackActiveTask(result, 3);
|
|
|
|
PlayerLoopHelper.AddAction(timing, result);
|
|
|
|
token = result.core.Version;
|
|
return result;
|
|
}
|
|
|
|
public U GetResult(short token)
|
|
{
|
|
try
|
|
{
|
|
return core.GetResult(token);
|
|
}
|
|
finally
|
|
{
|
|
TryReturn();
|
|
}
|
|
}
|
|
|
|
void IUniTaskSource.GetResult(short token)
|
|
{
|
|
GetResult(token);
|
|
}
|
|
|
|
public UniTaskStatus GetStatus(short token)
|
|
{
|
|
return core.GetStatus(token);
|
|
}
|
|
|
|
public UniTaskStatus UnsafeGetStatus()
|
|
{
|
|
return core.UnsafeGetStatus();
|
|
}
|
|
|
|
public void OnCompleted(Action<object> continuation, object state, short token)
|
|
{
|
|
core.OnCompleted(continuation, state, token);
|
|
}
|
|
|
|
public bool MoveNext()
|
|
{
|
|
if (cancellationToken.IsCancellationRequested || !target.TryGetTarget(out var t)) // doesn't find = cancel.
|
|
{
|
|
core.TrySetCanceled(cancellationToken);
|
|
return false;
|
|
}
|
|
|
|
U nextValue = default(U);
|
|
try
|
|
{
|
|
nextValue = monitorFunction(t);
|
|
if (equalityComparer.Equals(currentValue, nextValue))
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
core.TrySetException(ex);
|
|
return false;
|
|
}
|
|
|
|
core.TrySetResult(nextValue);
|
|
return false;
|
|
}
|
|
|
|
bool TryReturn()
|
|
{
|
|
TaskTracker.RemoveTracking(this);
|
|
core.Reset();
|
|
target = default;
|
|
currentValue = default;
|
|
monitorFunction = default;
|
|
equalityComparer = default;
|
|
cancellationToken = default;
|
|
cancellationTokenRegistration.Dispose();
|
|
return pool.TryPush(this);
|
|
}
|
|
}
|
|
}
|
|
}
|