forked from Misaki/GhostEngine
Introduces a new Ghost.Shader.Concept project implementing a modern, data-oriented material and shader system with: - Global/local keyword bitsets (fast O(1) ops, 64 bytes) - Multi-pass shader program and per-pass render state overrides - Thread-safe, 16-byte aligned material property blocks - Material pooling to reduce GC pressure - Batch renderer for efficient PSO grouping and async variant warmup - Full demo (Program.cs) and extensive documentation (ARCHITECTURE.md, README.md, PROJECT_SUMMARY.md) - Minor integration: new enums, doc updates, and keyword handling in existing code No breaking changes to the existing engine; all new code is isolated. This serves as a reference implementation for high-performance, extensible material/shader architectures.
150 lines
5.4 KiB
C#
150 lines
5.4 KiB
C#
using Ghost.Editor.Core.SceneGraph;
|
|
using Ghost.Engine;
|
|
using Ghost.Engine.IO;
|
|
using Ghost.Engine.Utilities;
|
|
using Ghost.Entities;
|
|
using System.Text.Json;
|
|
|
|
namespace Ghost.Editor.Core.Serializer;
|
|
|
|
internal class WorldNodeSerializer : CustomSerializer<WorldNode>
|
|
{
|
|
private static class Property
|
|
{
|
|
public const string NAME = "Name";
|
|
public const string ENTITIES = "Entities";
|
|
public const string ID = "ID";
|
|
public const string ENTITY_ID = "EntityID";
|
|
public const string COMPONENTS = "Components";
|
|
public const string DATA = "Data";
|
|
public const string SYSTEMS = "Systems";
|
|
}
|
|
|
|
public override bool CanConvert(Type typeToConvert)
|
|
{
|
|
return typeToConvert == typeof(WorldNode) || typeToConvert.IsSubclassOf(typeof(WorldNode));
|
|
}
|
|
|
|
public unsafe override void SerializeJson(Utf8JsonWriter writer, WorldNode value, JsonSerializerOptions options)
|
|
{
|
|
writer.WriteObject(() =>
|
|
{
|
|
writer.WriteString(Property.NAME, value.Name);
|
|
writer.WriteStartArray(Property.ENTITIES);
|
|
|
|
for (var i = 0; i < value.World.ArchetypeCount; i++)
|
|
{
|
|
ref var archetype = ref value.World.GetArchetypeReference(i);
|
|
|
|
for (var j = 0; j < archetype.ChunkCount; j++)
|
|
{
|
|
ref var chunk = ref archetype.GetChunkReference(j);
|
|
for (var k = 0; k < chunk._count; k++)
|
|
{
|
|
foreach (var layout in archetype._layouts)
|
|
{
|
|
var type = ComponentRegistry.s_runtimeIDToType[layout.componentID];
|
|
var size = ComponentRegistry.GetComponentInfo(layout.componentID).size;
|
|
|
|
if (type.AssemblyQualifiedName == null)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
writer.WriteStartObject();
|
|
writer.WriteString("Type", type.AssemblyQualifiedName);
|
|
writer.WritePropertyName("Data");
|
|
|
|
var pComponentData = chunk.GetUnsafePtr() + layout.offset + (k * size);
|
|
ComponentSerializerRegistry.SerializeJson(layout.componentID, writer, pComponentData, options);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
writer.WriteEndArray();
|
|
|
|
writer.WriteArray(Property.SYSTEMS, value.World.SystemManager.Systems, system =>
|
|
{
|
|
var name = system.GetType().AssemblyQualifiedName;
|
|
if (name == null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
writer.WriteStringValue(name);
|
|
});
|
|
});
|
|
}
|
|
|
|
public override WorldNode? DeserializeJson(ref Utf8JsonReader reader, JsonSerializerOptions options)
|
|
{
|
|
throw new NotImplementedException();
|
|
|
|
//var element = JsonDocument.ParseValue(ref reader).RootElement;
|
|
//var name = element.GetProperty(Property.NAME).GetString() ?? "New World";
|
|
|
|
//var world = World.Create(EngineCore.JobScheduler);
|
|
//var result = new WorldNode(world, name);
|
|
|
|
//foreach (var entityElement in element.GetProperty(Property.ENTITIES).EnumerateArray())
|
|
//{
|
|
// var entityName = entityElement.GetProperty(Property.NAME).GetString() ?? "New Entity";
|
|
// var entityID = entityElement.GetProperty(Property.ID).GetInt32();
|
|
// var entity = new Entity(entityID, 0);
|
|
// var node = new EntityNode(result, entity, entityName);
|
|
|
|
// world.EntityManager.AddEntityInternal(entity);
|
|
// result.EntityNodeLookup[entity] = node;
|
|
//}
|
|
|
|
//foreach (var componentElement in element.GetProperty(Property.COMPONENTS).EnumerateObject())
|
|
//{
|
|
// var typeName = componentElement.Name;
|
|
// var space = Type.GetType(typeName) ?? throw new Exception($"Type {typeName} not found.");
|
|
|
|
// foreach (var dataElement in componentElement.Value.EnumerateArray())
|
|
// {
|
|
// var entityID = dataElement.GetProperty(Property.ENTITY_ID).GetInt32();
|
|
// var entity = new Entity(entityID, 0);
|
|
|
|
// var dataProperty = dataElement.GetProperty(Property.DATA);
|
|
// var component = JsonSerializer.Deserialize(dataProperty.GetRawText(), space, options);
|
|
// if (component is IComponent data)
|
|
// {
|
|
// world.EntityManager.AddComponent(entity, data, space);
|
|
// }
|
|
// }
|
|
//}
|
|
|
|
//foreach (var systemElement in element.GetProperty(Property.SYSTEMS).EnumerateArray())
|
|
//{
|
|
// var typeString = systemElement.GetString();
|
|
// if (string.IsNullOrEmpty(typeString))
|
|
// {
|
|
// continue;
|
|
// }
|
|
|
|
// var systemType = Type.GetType(typeString);
|
|
// if (systemType == null)
|
|
// {
|
|
// continue;
|
|
// }
|
|
|
|
// world.SystemStorage.AddSystem(systemType);
|
|
//}
|
|
|
|
//return result;
|
|
}
|
|
|
|
public override void SerializeBinary(BinaryWriter writer, WorldNode value)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
public override WorldNode? DeserializeBinary(BinaryReader reader)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
}
|