using System; using System.Collections.Generic; using System.Linq; namespace ET { public static class WaitTypeError { public const int Success = 0; public const int Destroy = 1; public const int Cancel = 2; public const int Timeout = 3; } public interface IWaitType { int Error { get; set; } } [FriendOf(typeof(ObjectWait))] public static class ObjectWaitSystem { [ObjectSystem] public class ObjectWaitAwakeSystem: AwakeSystem { protected override void Awake(ObjectWait self) { self.tcss.Clear(); } } [ObjectSystem] public class ObjectWaitDestroySystem: DestroySystem { protected override void Destroy(ObjectWait self) { foreach (object v in self.tcss.Values.ToArray()) { ((IDestroyRun) v).SetResult(); } } } private interface IDestroyRun { void SetResult(); } private class ResultCallback: IDestroyRun where K : struct, IWaitType { private ETTask tcs; public ResultCallback() { this.tcs = ETTask.Create(true); } public bool IsDisposed { get { return this.tcs == null; } } public ETTask Task => this.tcs; public void SetResult(K k) { var t = tcs; this.tcs = null; t.SetResult(k); } public void SetResult() { var t = tcs; this.tcs = null; t.SetResult(new K() { Error = WaitTypeError.Destroy }); } } public static async ETTask Wait(this ObjectWait self, ETCancellationToken cancellationToken = null) where T : struct, IWaitType { ResultCallback tcs = new ResultCallback(); Type type = typeof (T); self.tcss.Add(type, tcs); void CancelAction() { self.Notify(new T() { Error = WaitTypeError.Cancel }); } T ret; try { cancellationToken?.Add(CancelAction); ret = await tcs.Task; } finally { cancellationToken?.Remove(CancelAction); } return ret; } public static async ETTask Wait(this ObjectWait self, int timeout, ETCancellationToken cancellationToken = null) where T : struct, IWaitType { ResultCallback tcs = new ResultCallback(); async ETTask WaitTimeout() { await TimerComponent.Instance.WaitAsync(timeout, cancellationToken); if (cancellationToken.IsCancel()) { return; } if (tcs.IsDisposed) { return; } self.Notify(new T() { Error = WaitTypeError.Timeout }); } WaitTimeout().Coroutine(); self.tcss.Add(typeof (T), tcs); void CancelAction() { self.Notify(new T() { Error = WaitTypeError.Cancel }); } T ret; try { cancellationToken?.Add(CancelAction); ret = await tcs.Task; } finally { cancellationToken?.Remove(CancelAction); } return ret; } public static void Notify(this ObjectWait self, T obj) where T : struct, IWaitType { Type type = typeof (T); if (!self.tcss.TryGetValue(type, out object tcs)) { return; } self.tcss.Remove(type); ((ResultCallback) tcs).SetResult(obj); } } [ComponentOf] public class ObjectWait: Entity, IAwake, IDestroy { public Dictionary tcss = new Dictionary(); } }