using ProjectM.Simulation;
using Unity.Burst;
using Unity.Collections;
using Unity.Entities;
using Unity.NetCode;
namespace ProjectM.Client
{
///
/// Client-side connection handshake: for every connection that has been assigned a
/// but is not yet , mark it in-game and
/// fire a RPC so the server spawns this client's player ghost.
/// Adding NetworkStreamInGame is what gates snapshot/command flow on. Mirrors the netcode
/// "networked-cube" go-in-game sample.
///
[BurstCompile]
[WorldSystemFilter(WorldSystemFilterFlags.ClientSimulation | WorldSystemFilterFlags.ThinClientSimulation)]
public partial struct GoInGameClientSystem : ISystem
{
[BurstCompile]
public void OnCreate(ref SystemState state)
{
var builder = new EntityQueryBuilder(Allocator.Temp)
.WithAll()
.WithNone();
state.RequireForUpdate(state.GetEntityQuery(builder));
}
[BurstCompile]
public void OnUpdate(ref SystemState state)
{
// A FULL client must not go in-game until the gameplay subscene's ghost prefabs have streamed in.
// Otherwise the server's first ghost snapshot arrives before the client can resolve those prefabs
// ("ghost ... ENTITY_NOT_FOUND" -> the server disconnects the connection -> "nothing loads"). On
// loopback / fast LAN the connect+go-in-game handshake easily beats the ~0.5s entity-subscene stream.
// PlayerSpawner is a subscene-baked singleton that co-loads with the ghost prefabs, so its presence
// is a sound "subscene ready" gate. Thin clients never instantiate ghosts (and don't stream the
// subscene), so they skip the gate and connect immediately.
bool isThinClient = (state.WorldUnmanaged.Flags & WorldFlags.GameThinClient) == WorldFlags.GameThinClient;
if (!isThinClient && !SystemAPI.HasSingleton())
return;
var ecb = new EntityCommandBuffer(Allocator.Temp);
foreach (var (_, connection) in
SystemAPI.Query>().WithNone().WithEntityAccess())
{
ecb.AddComponent(connection);
byte classId = SystemAPI.HasSingleton() ? SystemAPI.GetSingleton().ClassId : (byte)0;
var request = ecb.CreateEntity();
ecb.AddComponent(request, new GoInGameRequest { ClassId = classId }); // Slice 2: carry the chosen class
ecb.AddComponent(request, new SendRpcCommandRequest { TargetConnection = connection });
}
ecb.Playback(state.EntityManager);
}
}
}