2025-03-14 16:53:25 +08:00
|
|
|
|
#if DISABLE_DEBUG
|
|
|
|
|
|
#undef DEBUG
|
|
|
|
|
|
#endif
|
|
|
|
|
|
using DCFApixels.DragonECS.Core;
|
2024-11-01 12:41:10 +08:00
|
|
|
|
using DCFApixels.DragonECS.Internal;
|
2024-10-11 18:28:46 +08:00
|
|
|
|
using System;
|
2024-08-24 12:29:58 +08:00
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
|
using System.Runtime.CompilerServices;
|
2023-12-24 15:40:19 +08:00
|
|
|
|
|
|
|
|
|
|
namespace DCFApixels.DragonECS
|
2023-05-07 00:50:02 +08:00
|
|
|
|
{
|
2024-08-24 12:29:58 +08:00
|
|
|
|
public partial class EcsWorld
|
|
|
|
|
|
{
|
2024-11-05 17:16:50 +08:00
|
|
|
|
private readonly Dictionary<(Type, object), IQueryExecutorImplementation> _executorCoures;
|
2024-11-07 08:22:10 +08:00
|
|
|
|
|
2024-11-05 15:50:03 +08:00
|
|
|
|
public TExecutor GetExecutorForMask<TExecutor>(IComponentMask gmask)
|
|
|
|
|
|
where TExecutor : MaskQueryExecutor, new()
|
2024-08-24 12:29:58 +08:00
|
|
|
|
{
|
2024-11-07 08:22:10 +08:00
|
|
|
|
var executorType = typeof(TExecutor);
|
2024-11-05 15:50:03 +08:00
|
|
|
|
//проверяет ключ по абстрактной маске
|
2024-11-07 08:22:10 +08:00
|
|
|
|
if (_executorCoures.TryGetValue((executorType, gmask), out IQueryExecutorImplementation executor) == false)
|
2024-08-24 12:29:58 +08:00
|
|
|
|
{
|
2024-11-05 15:50:03 +08:00
|
|
|
|
var mask = gmask.ToMask(this);
|
|
|
|
|
|
//проверяет ключ по конкретной маске, или что конкретная и абстрактая одна и таже
|
|
|
|
|
|
if (mask == gmask ||
|
2024-11-07 08:22:10 +08:00
|
|
|
|
_executorCoures.TryGetValue((executorType, mask), out executor) == false)
|
2024-11-05 15:50:03 +08:00
|
|
|
|
{
|
2024-11-07 08:22:10 +08:00
|
|
|
|
TExecutor executorCore = new TExecutor();
|
|
|
|
|
|
executorCore.Initialize(this, mask);
|
|
|
|
|
|
executor = executorCore;
|
2024-11-05 15:50:03 +08:00
|
|
|
|
}
|
2024-11-07 08:22:10 +08:00
|
|
|
|
_executorCoures.Add((executorType, gmask), executor);
|
2024-08-24 12:29:58 +08:00
|
|
|
|
}
|
2024-11-05 15:50:03 +08:00
|
|
|
|
return (TExecutor)executor;
|
2024-10-03 08:20:22 +08:00
|
|
|
|
}
|
2024-11-07 10:19:44 +08:00
|
|
|
|
|
|
|
|
|
|
public void GetMaskQueryExecutors(List<MaskQueryExecutor> result, ref int version)
|
|
|
|
|
|
{
|
2024-11-07 16:21:51 +08:00
|
|
|
|
if (_executorCoures == null || version == _executorCoures.Count)
|
2024-11-07 10:19:44 +08:00
|
|
|
|
{
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
result.Clear();
|
|
|
|
|
|
|
|
|
|
|
|
foreach (var item in _executorCoures)
|
|
|
|
|
|
{
|
2024-11-07 16:21:51 +08:00
|
|
|
|
if (item.Value is MaskQueryExecutor x)
|
2024-11-07 10:19:44 +08:00
|
|
|
|
{
|
|
|
|
|
|
result.Add(x);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
version = _executorCoures.Count;
|
|
|
|
|
|
}
|
2024-08-24 12:29:58 +08:00
|
|
|
|
}
|
2024-11-05 15:50:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
namespace DCFApixels.DragonECS.Core
|
|
|
|
|
|
{
|
|
|
|
|
|
public interface IQueryExecutorImplementation
|
|
|
|
|
|
{
|
|
|
|
|
|
EcsWorld World { get; }
|
|
|
|
|
|
long Version { get; }
|
|
|
|
|
|
bool IsCached { get; }
|
2024-11-07 08:22:10 +08:00
|
|
|
|
int LastCachedCount { get; }
|
2024-11-05 15:50:03 +08:00
|
|
|
|
void Destroy();
|
|
|
|
|
|
}
|
|
|
|
|
|
public abstract class MaskQueryExecutor : IQueryExecutorImplementation
|
2024-08-24 12:29:58 +08:00
|
|
|
|
{
|
|
|
|
|
|
private EcsWorld _source;
|
2024-08-24 13:10:50 +08:00
|
|
|
|
private EcsMask _mask;
|
2024-08-24 12:29:58 +08:00
|
|
|
|
public short WorldID
|
|
|
|
|
|
{
|
|
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
2024-10-31 16:27:53 +08:00
|
|
|
|
get { return _source.ID; }
|
2024-08-24 12:29:58 +08:00
|
|
|
|
}
|
|
|
|
|
|
public EcsWorld World
|
|
|
|
|
|
{
|
|
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
|
|
|
|
get { return _source; }
|
|
|
|
|
|
}
|
2024-11-07 16:21:51 +08:00
|
|
|
|
public EcsMask Mask
|
2024-08-24 13:10:50 +08:00
|
|
|
|
{
|
|
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
|
|
|
|
get { return _mask; }
|
|
|
|
|
|
}
|
2024-10-03 08:20:22 +08:00
|
|
|
|
public abstract long Version { get; }
|
2024-11-05 15:50:03 +08:00
|
|
|
|
public abstract bool IsCached { get; }
|
2024-11-07 08:22:10 +08:00
|
|
|
|
public abstract int LastCachedCount { get; }
|
2024-08-24 13:10:50 +08:00
|
|
|
|
internal void Initialize(EcsWorld world, EcsMask mask)
|
2024-08-24 12:29:58 +08:00
|
|
|
|
{
|
|
|
|
|
|
_source = world;
|
2024-08-24 13:10:50 +08:00
|
|
|
|
_mask = mask;
|
2024-08-24 12:29:58 +08:00
|
|
|
|
OnInitialize();
|
|
|
|
|
|
}
|
2024-11-05 15:50:03 +08:00
|
|
|
|
void IQueryExecutorImplementation.Destroy()
|
2024-08-24 12:29:58 +08:00
|
|
|
|
{
|
|
|
|
|
|
OnDestroy();
|
|
|
|
|
|
_source = null;
|
2023-12-24 15:40:19 +08:00
|
|
|
|
}
|
2023-06-08 04:04:39 +08:00
|
|
|
|
protected abstract void OnInitialize();
|
2023-05-26 06:18:09 +08:00
|
|
|
|
protected abstract void OnDestroy();
|
2023-05-07 00:50:02 +08:00
|
|
|
|
}
|
2024-08-23 22:31:43 +08:00
|
|
|
|
|
2024-10-11 18:28:46 +08:00
|
|
|
|
public readonly unsafe struct WorldStateVersionsChecker : IDisposable
|
2024-08-23 22:31:43 +08:00
|
|
|
|
{
|
2024-10-11 18:28:46 +08:00
|
|
|
|
private readonly EcsWorld _world;
|
|
|
|
|
|
private readonly int[] _maskInc;
|
|
|
|
|
|
private readonly int[] _maskExc;
|
|
|
|
|
|
// [0] world version
|
|
|
|
|
|
// [-> _maskInc.Length] inc versions
|
|
|
|
|
|
// [-> _maskExc.Length] exc versions
|
|
|
|
|
|
private readonly long* _versions;
|
2025-03-13 16:41:23 +08:00
|
|
|
|
private readonly int _count;
|
2024-10-11 18:28:46 +08:00
|
|
|
|
public long Version
|
|
|
|
|
|
{
|
|
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
|
|
|
|
get { return _versions[0]; }
|
|
|
|
|
|
}
|
2024-08-23 22:31:43 +08:00
|
|
|
|
|
2024-10-11 18:28:46 +08:00
|
|
|
|
public WorldStateVersionsChecker(EcsMask mask)
|
2024-08-23 22:31:43 +08:00
|
|
|
|
{
|
2024-10-11 18:28:46 +08:00
|
|
|
|
_world = mask.World;
|
2024-11-01 12:41:10 +08:00
|
|
|
|
_maskInc = mask._incs;
|
|
|
|
|
|
_maskExc = mask._excs;
|
2025-03-13 16:41:23 +08:00
|
|
|
|
_count = 1 + mask._incs.Length + mask._excs.Length;
|
|
|
|
|
|
_versions = UnmanagedArrayUtility.NewAndInit<long>(_count);
|
2024-10-11 18:28:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
public bool Check()
|
|
|
|
|
|
{
|
|
|
|
|
|
if (*_versions == _world.Version)
|
|
|
|
|
|
{
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-10-31 14:46:21 +08:00
|
|
|
|
long* versionsPtr = _versions;
|
2024-10-11 18:28:46 +08:00
|
|
|
|
var slots = _world._poolSlots;
|
|
|
|
|
|
foreach (var slotIndex in _maskInc)
|
|
|
|
|
|
{
|
2024-10-31 14:46:21 +08:00
|
|
|
|
versionsPtr++;
|
|
|
|
|
|
if (*versionsPtr != slots[slotIndex].version)
|
2024-10-11 18:28:46 +08:00
|
|
|
|
{
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
foreach (var slotIndex in _maskExc)
|
|
|
|
|
|
{
|
2024-10-31 14:46:21 +08:00
|
|
|
|
versionsPtr++;
|
|
|
|
|
|
if (*versionsPtr != slots[slotIndex].version)
|
2024-10-11 18:28:46 +08:00
|
|
|
|
{
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return true;
|
2024-08-23 22:31:43 +08:00
|
|
|
|
}
|
2024-10-11 18:28:46 +08:00
|
|
|
|
public void Next()
|
|
|
|
|
|
{
|
|
|
|
|
|
*_versions = _world.Version;
|
2024-08-23 22:31:43 +08:00
|
|
|
|
|
2025-03-30 22:40:39 +08:00
|
|
|
|
long* versionsPtr = _versions;
|
2024-10-11 18:28:46 +08:00
|
|
|
|
var slots = _world._poolSlots;
|
|
|
|
|
|
foreach (var slotIndex in _maskInc)
|
|
|
|
|
|
{
|
2025-03-30 22:40:39 +08:00
|
|
|
|
versionsPtr++;
|
|
|
|
|
|
*versionsPtr = slots[slotIndex].version;
|
2024-10-11 18:28:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
foreach (var slotIndex in _maskExc)
|
|
|
|
|
|
{
|
2025-03-30 22:40:39 +08:00
|
|
|
|
versionsPtr++;
|
|
|
|
|
|
*versionsPtr = slots[slotIndex].version;
|
2024-10-11 18:28:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
public bool CheckAndNext()
|
2024-08-23 22:31:43 +08:00
|
|
|
|
{
|
2024-10-11 18:28:46 +08:00
|
|
|
|
if (*_versions == _world.Version)
|
|
|
|
|
|
{
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
*_versions = _world.Version;
|
|
|
|
|
|
|
2025-03-30 22:40:39 +08:00
|
|
|
|
long* versionsPtr = _versions;
|
2024-10-11 18:28:46 +08:00
|
|
|
|
var slots = _world._poolSlots;
|
2025-03-13 16:41:23 +08:00
|
|
|
|
bool result = _count != 1;
|
2024-10-11 18:28:46 +08:00
|
|
|
|
foreach (var slotIndex in _maskInc)
|
2024-08-23 22:31:43 +08:00
|
|
|
|
{
|
2025-03-30 22:40:39 +08:00
|
|
|
|
versionsPtr++;
|
|
|
|
|
|
if (*versionsPtr != slots[slotIndex].version)
|
2024-08-23 22:31:43 +08:00
|
|
|
|
{
|
|
|
|
|
|
result = false;
|
2025-03-30 22:40:39 +08:00
|
|
|
|
*versionsPtr = slots[slotIndex].version;
|
2024-08-23 22:31:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-10-11 18:28:46 +08:00
|
|
|
|
foreach (var slotIndex in _maskExc)
|
2024-08-23 22:31:43 +08:00
|
|
|
|
{
|
2025-03-30 22:40:39 +08:00
|
|
|
|
return false; //TODO hotfix, не правильная логика проверки версия для EXC, потому сейчас она скипается
|
|
|
|
|
|
|
|
|
|
|
|
//versionsPtr++;
|
|
|
|
|
|
//if (*versionsPtr != slots[slotIndex].version)
|
|
|
|
|
|
//{
|
|
|
|
|
|
// result = false;
|
|
|
|
|
|
// *versionsPtr = slots[slotIndex].version;
|
|
|
|
|
|
//}
|
2024-08-23 22:31:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
return result;
|
|
|
|
|
|
}
|
2024-10-11 18:28:46 +08:00
|
|
|
|
|
|
|
|
|
|
public void Dispose()
|
|
|
|
|
|
{
|
2024-11-17 21:43:50 +08:00
|
|
|
|
UnmanagedArrayUtility.Free(_versions);
|
2024-10-11 18:28:46 +08:00
|
|
|
|
}
|
2024-08-23 22:31:43 +08:00
|
|
|
|
}
|
2023-12-24 15:40:19 +08:00
|
|
|
|
}
|