mirror of
https://github.com/GeWuYou/GFramework.git
synced 2026-03-24 12:33:30 +08:00
- 实现 Store 类作为集中式状态容器,默认支持状态归约和订阅通知 - 添加 IReadonlyStore、IStore、IReducer 等状态管理相关抽象接口 - 实现 StoreExtensions 扩展方法,提供 Select 和 ToBindableProperty 选择器功能 - 添加 StoreSelection 类,支持从完整状态树中投影局部状态视图 - 实现 StoreDispatchContext 和 StoreDispatchRecord 用于分发过程诊断 - 添加 IStoreMiddleware 中间件接口,支持在分发过程中插入日志和审计逻辑 - 实现完整的状态选择器和绑定属性桥接功能,便于现有 UI 代码复用 - 添加 Store 相关单元测试,覆盖状态归约、订阅通知和选择器桥接场景
230 lines
7.7 KiB
C#
230 lines
7.7 KiB
C#
using GFramework.Core.Abstractions.Events;
|
|
using GFramework.Core.Abstractions.Property;
|
|
using GFramework.Core.Abstractions.StateManagement;
|
|
using GFramework.Core.Events;
|
|
|
|
namespace GFramework.Core.StateManagement;
|
|
|
|
/// <summary>
|
|
/// Store 选择结果的只读绑定视图。
|
|
/// 该类型将整棵状态树上的订阅转换为局部状态片段的订阅,
|
|
/// 使现有依赖 IReadonlyBindableProperty 的 UI 代码能够平滑复用到 Store 场景中。
|
|
/// </summary>
|
|
/// <typeparam name="TState">源状态类型。</typeparam>
|
|
/// <typeparam name="TSelected">投影后的局部状态类型。</typeparam>
|
|
public class StoreSelection<TState, TSelected> : IReadonlyBindableProperty<TSelected>
|
|
{
|
|
/// <summary>
|
|
/// 用于判断选择结果是否真正变化的比较器。
|
|
/// </summary>
|
|
private readonly IEqualityComparer<TSelected> _comparer;
|
|
|
|
/// <summary>
|
|
/// 当前监听器列表。
|
|
/// </summary>
|
|
private readonly List<Action<TSelected>> _listeners = [];
|
|
|
|
/// <summary>
|
|
/// 保护监听器集合和底层 Store 订阅句柄的同步锁。
|
|
/// </summary>
|
|
private readonly object _lock = new();
|
|
|
|
/// <summary>
|
|
/// 负责从完整状态中投影出局部状态的选择器。
|
|
/// </summary>
|
|
private readonly Func<TState, TSelected> _selector;
|
|
|
|
/// <summary>
|
|
/// 源 Store。
|
|
/// </summary>
|
|
private readonly IReadonlyStore<TState> _store;
|
|
|
|
/// <summary>
|
|
/// 当前已缓存的选择结果。
|
|
/// 该缓存仅在存在监听器时用于变化比较和事件通知,直接读取 Value 时始终以 Store 当前状态为准。
|
|
/// </summary>
|
|
private TSelected _currentValue = default!;
|
|
|
|
/// <summary>
|
|
/// 连接到底层 Store 的订阅句柄。
|
|
/// 仅当当前存在至少一个监听器时才会建立该订阅,以减少长期闲置对象造成的引用链。
|
|
/// </summary>
|
|
private IUnRegister? _storeSubscription;
|
|
|
|
/// <summary>
|
|
/// 初始化一个新的 Store 选择视图。
|
|
/// </summary>
|
|
/// <param name="store">源 Store。</param>
|
|
/// <param name="selector">状态选择器。</param>
|
|
/// <param name="comparer">选择结果比较器;未提供时使用 <see cref="EqualityComparer{T}.Default"/>。</param>
|
|
/// <exception cref="ArgumentNullException">
|
|
/// 当 <paramref name="store"/> 或 <paramref name="selector"/> 为 <see langword="null"/> 时抛出。
|
|
/// </exception>
|
|
public StoreSelection(
|
|
IReadonlyStore<TState> store,
|
|
Func<TState, TSelected> selector,
|
|
IEqualityComparer<TSelected>? comparer = null)
|
|
{
|
|
_store = store ?? throw new ArgumentNullException(nameof(store));
|
|
_selector = selector ?? throw new ArgumentNullException(nameof(selector));
|
|
_comparer = comparer ?? EqualityComparer<TSelected>.Default;
|
|
}
|
|
|
|
/// <summary>
|
|
/// 获取当前选择结果。
|
|
/// </summary>
|
|
public TSelected Value => _selector(_store.State);
|
|
|
|
/// <summary>
|
|
/// 将无参事件监听适配为带选择结果参数的监听。
|
|
/// </summary>
|
|
/// <param name="onEvent">无参事件监听器。</param>
|
|
/// <returns>用于取消订阅的句柄。</returns>
|
|
IUnRegister IEvent.Register(Action onEvent)
|
|
{
|
|
ArgumentNullException.ThrowIfNull(onEvent);
|
|
return Register(_ => onEvent());
|
|
}
|
|
|
|
/// <summary>
|
|
/// 注册选择结果变化监听器。
|
|
/// </summary>
|
|
/// <param name="onValueChanged">选择结果变化时的回调。</param>
|
|
/// <returns>用于取消订阅的句柄。</returns>
|
|
/// <exception cref="ArgumentNullException">当 <paramref name="onValueChanged"/> 为 <see langword="null"/> 时抛出。</exception>
|
|
public IUnRegister Register(Action<TSelected> onValueChanged)
|
|
{
|
|
ArgumentNullException.ThrowIfNull(onValueChanged);
|
|
|
|
var shouldAttach = false;
|
|
|
|
lock (_lock)
|
|
{
|
|
if (_listeners.Count == 0)
|
|
{
|
|
_currentValue = Value;
|
|
shouldAttach = true;
|
|
}
|
|
|
|
_listeners.Add(onValueChanged);
|
|
}
|
|
|
|
if (shouldAttach)
|
|
{
|
|
AttachToStore();
|
|
}
|
|
|
|
return new DefaultUnRegister(() => UnRegister(onValueChanged));
|
|
}
|
|
|
|
/// <summary>
|
|
/// 注册选择结果变化监听器,并立即回放当前值。
|
|
/// </summary>
|
|
/// <param name="action">选择结果变化时的回调。</param>
|
|
/// <returns>用于取消订阅的句柄。</returns>
|
|
/// <exception cref="ArgumentNullException">当 <paramref name="action"/> 为 <see langword="null"/> 时抛出。</exception>
|
|
public IUnRegister RegisterWithInitValue(Action<TSelected> action)
|
|
{
|
|
ArgumentNullException.ThrowIfNull(action);
|
|
|
|
var currentValue = Value;
|
|
action(currentValue);
|
|
return Register(action);
|
|
}
|
|
|
|
/// <summary>
|
|
/// 取消注册选择结果变化监听器。
|
|
/// </summary>
|
|
/// <param name="onValueChanged">需要移除的监听器。</param>
|
|
/// <exception cref="ArgumentNullException">当 <paramref name="onValueChanged"/> 为 <see langword="null"/> 时抛出。</exception>
|
|
public void UnRegister(Action<TSelected> onValueChanged)
|
|
{
|
|
ArgumentNullException.ThrowIfNull(onValueChanged);
|
|
|
|
IUnRegister? storeSubscription = null;
|
|
|
|
lock (_lock)
|
|
{
|
|
_listeners.Remove(onValueChanged);
|
|
if (_listeners.Count == 0 && _storeSubscription != null)
|
|
{
|
|
storeSubscription = _storeSubscription;
|
|
_storeSubscription = null;
|
|
}
|
|
}
|
|
|
|
storeSubscription?.UnRegister();
|
|
}
|
|
|
|
/// <summary>
|
|
/// 将当前选择视图连接到底层 Store。
|
|
/// </summary>
|
|
private void AttachToStore()
|
|
{
|
|
var subscription = _store.Subscribe(OnStoreChanged);
|
|
Action<TSelected>[] listenersSnapshot = Array.Empty<Action<TSelected>>();
|
|
var latestValue = Value;
|
|
var shouldNotify = false;
|
|
|
|
lock (_lock)
|
|
{
|
|
// 如果在建立底层订阅期间所有监听器都已被移除,则立即释放刚刚建立的订阅,
|
|
// 避免选择视图在无人监听时继续被 Store 保持引用。
|
|
if (_listeners.Count == 0)
|
|
{
|
|
subscription.UnRegister();
|
|
return;
|
|
}
|
|
|
|
if (_storeSubscription != null)
|
|
{
|
|
subscription.UnRegister();
|
|
return;
|
|
}
|
|
|
|
_storeSubscription = subscription;
|
|
if (!_comparer.Equals(_currentValue, latestValue))
|
|
{
|
|
_currentValue = latestValue;
|
|
listenersSnapshot = _listeners.ToArray();
|
|
shouldNotify = listenersSnapshot.Length > 0;
|
|
}
|
|
}
|
|
|
|
if (!shouldNotify)
|
|
{
|
|
return;
|
|
}
|
|
|
|
foreach (var listener in listenersSnapshot)
|
|
{
|
|
listener(latestValue);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// 响应底层 Store 的状态变化,并在选中片段真正变化时通知监听器。
|
|
/// </summary>
|
|
/// <param name="state">新的完整状态。</param>
|
|
private void OnStoreChanged(TState state)
|
|
{
|
|
var selectedValue = _selector(state);
|
|
Action<TSelected>[] listenersSnapshot = Array.Empty<Action<TSelected>>();
|
|
|
|
lock (_lock)
|
|
{
|
|
if (_listeners.Count == 0 || _comparer.Equals(_currentValue, selectedValue))
|
|
{
|
|
return;
|
|
}
|
|
|
|
_currentValue = selectedValue;
|
|
listenersSnapshot = _listeners.ToArray();
|
|
}
|
|
|
|
foreach (var listener in listenersSnapshot)
|
|
{
|
|
listener(selectedValue);
|
|
}
|
|
}
|
|
} |