292 lines
7.6 KiB
C#
292 lines
7.6 KiB
C#
using System;
|
|
using System.Threading;
|
|
|
|
namespace Cysharp.Threading.Tasks
|
|
{
|
|
public interface ITriggerHandler<T>
|
|
{
|
|
void OnNext(T value);
|
|
void OnError(Exception ex);
|
|
void OnCompleted();
|
|
void OnCanceled(CancellationToken cancellationToken);
|
|
|
|
// set/get from TriggerEvent<T>
|
|
ITriggerHandler<T> Prev { get; set; }
|
|
ITriggerHandler<T> Next { get; set; }
|
|
}
|
|
|
|
// be careful to use, itself is struct.
|
|
public struct TriggerEvent<T>
|
|
{
|
|
ITriggerHandler<T> head; // head.prev is last
|
|
ITriggerHandler<T> iteratingHead;
|
|
ITriggerHandler<T> iteratingNode;
|
|
|
|
void LogError(Exception ex)
|
|
{
|
|
#if UNITY_2018_3_OR_NEWER
|
|
UnityEngine.Debug.LogException(ex);
|
|
#else
|
|
Console.WriteLine(ex);
|
|
#endif
|
|
}
|
|
|
|
public void SetResult(T value)
|
|
{
|
|
if (iteratingNode != null)
|
|
{
|
|
throw new InvalidOperationException("Can not trigger itself in iterating.");
|
|
}
|
|
|
|
var h = head;
|
|
while (h != null)
|
|
{
|
|
iteratingNode = h;
|
|
|
|
try
|
|
{
|
|
h.OnNext(value);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
LogError(ex);
|
|
Remove(h);
|
|
}
|
|
|
|
// If `h` itself is removed by OnNext, h.Next is null.
|
|
// Therefore, instead of looking at h.Next, the `iteratingNode` reference itself is replaced.
|
|
h = h == iteratingNode ? h.Next : iteratingNode;
|
|
}
|
|
|
|
iteratingNode = null;
|
|
if (iteratingHead != null)
|
|
{
|
|
Add(iteratingHead);
|
|
iteratingHead = null;
|
|
}
|
|
}
|
|
|
|
public void SetCanceled(CancellationToken cancellationToken)
|
|
{
|
|
if (iteratingNode != null)
|
|
{
|
|
throw new InvalidOperationException("Can not trigger itself in iterating.");
|
|
}
|
|
|
|
var h = head;
|
|
while (h != null)
|
|
{
|
|
iteratingNode = h;
|
|
try
|
|
{
|
|
h.OnCanceled(cancellationToken);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
LogError(ex);
|
|
}
|
|
|
|
var next = h == iteratingNode ? h.Next : iteratingNode;
|
|
iteratingNode = null;
|
|
Remove(h);
|
|
h = next;
|
|
}
|
|
|
|
iteratingNode = null;
|
|
if (iteratingHead != null)
|
|
{
|
|
Add(iteratingHead);
|
|
iteratingHead = null;
|
|
}
|
|
}
|
|
|
|
public void SetCompleted()
|
|
{
|
|
if (iteratingNode != null)
|
|
{
|
|
throw new InvalidOperationException("Can not trigger itself in iterating.");
|
|
}
|
|
|
|
var h = head;
|
|
while (h != null)
|
|
{
|
|
iteratingNode = h;
|
|
try
|
|
{
|
|
h.OnCompleted();
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
LogError(ex);
|
|
}
|
|
|
|
var next = h == iteratingNode ? h.Next : iteratingNode;
|
|
iteratingNode = null;
|
|
Remove(h);
|
|
h = next;
|
|
}
|
|
|
|
iteratingNode = null;
|
|
if (iteratingHead != null)
|
|
{
|
|
Add(iteratingHead);
|
|
iteratingHead = null;
|
|
}
|
|
}
|
|
|
|
public void SetError(Exception exception)
|
|
{
|
|
if (iteratingNode != null)
|
|
{
|
|
throw new InvalidOperationException("Can not trigger itself in iterating.");
|
|
}
|
|
|
|
var h = head;
|
|
while (h != null)
|
|
{
|
|
iteratingNode = h;
|
|
try
|
|
{
|
|
h.OnError(exception);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
LogError(ex);
|
|
}
|
|
|
|
var next = h == iteratingNode ? h.Next : iteratingNode;
|
|
iteratingNode = null;
|
|
Remove(h);
|
|
h = next;
|
|
}
|
|
|
|
iteratingNode = null;
|
|
if (iteratingHead != null)
|
|
{
|
|
Add(iteratingHead);
|
|
iteratingHead = null;
|
|
}
|
|
}
|
|
|
|
public void Add(ITriggerHandler<T> handler)
|
|
{
|
|
if (handler == null) throw new ArgumentNullException(nameof(handler));
|
|
|
|
// zero node.
|
|
if (head == null)
|
|
{
|
|
head = handler;
|
|
return;
|
|
}
|
|
|
|
if (iteratingNode != null)
|
|
{
|
|
if (iteratingHead == null)
|
|
{
|
|
iteratingHead = handler;
|
|
return;
|
|
}
|
|
|
|
var last = iteratingHead.Prev;
|
|
if (last == null)
|
|
{
|
|
// single node.
|
|
iteratingHead.Prev = handler;
|
|
iteratingHead.Next = handler;
|
|
handler.Prev = iteratingHead;
|
|
}
|
|
else
|
|
{
|
|
// multi node
|
|
iteratingHead.Prev = handler;
|
|
last.Next = handler;
|
|
handler.Prev = last;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
var last = head.Prev;
|
|
if (last == null)
|
|
{
|
|
// single node.
|
|
head.Prev = handler;
|
|
head.Next = handler;
|
|
handler.Prev = head;
|
|
}
|
|
else
|
|
{
|
|
// multi node
|
|
head.Prev = handler;
|
|
last.Next = handler;
|
|
handler.Prev = last;
|
|
}
|
|
}
|
|
}
|
|
|
|
public void Remove(ITriggerHandler<T> handler)
|
|
{
|
|
if (handler == null) throw new ArgumentNullException(nameof(handler));
|
|
|
|
var prev = handler.Prev;
|
|
var next = handler.Next;
|
|
|
|
if (next != null)
|
|
{
|
|
next.Prev = prev;
|
|
}
|
|
|
|
if (handler == head)
|
|
{
|
|
head = next;
|
|
}
|
|
// when handler is head, prev indicate last so don't use it.
|
|
else if (prev != null)
|
|
{
|
|
prev.Next = next;
|
|
}
|
|
|
|
if (handler == iteratingNode)
|
|
{
|
|
iteratingNode = next;
|
|
}
|
|
if (handler == iteratingHead)
|
|
{
|
|
iteratingHead = next;
|
|
}
|
|
|
|
if (head != null)
|
|
{
|
|
if (head.Prev == handler)
|
|
{
|
|
if (prev != head)
|
|
{
|
|
head.Prev = prev;
|
|
}
|
|
else
|
|
{
|
|
head.Prev = null;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (iteratingHead != null)
|
|
{
|
|
if (iteratingHead.Prev == handler)
|
|
{
|
|
if (prev != iteratingHead.Prev)
|
|
{
|
|
iteratingHead.Prev = prev;
|
|
}
|
|
else
|
|
{
|
|
iteratingHead.Prev = null;
|
|
}
|
|
}
|
|
}
|
|
|
|
handler.Prev = null;
|
|
handler.Next = null;
|
|
}
|
|
}
|
|
}
|