forked from Shardstone/trail-into-darkness
141 lines
5.3 KiB
C#
141 lines
5.3 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using UnityEngine;
|
|
using UnityEngine.AddressableAssets;
|
|
|
|
namespace Jovian.EncounterSystem {
|
|
/// <summary>id → encounter cache. Editor auto-repopulates on asset changes; runtime must call <see cref="PopulateEncounters"/>.</summary>
|
|
[CreateAssetMenu(fileName = "EncounterRegistry", menuName = "Jovian/Encounter System/Encounter Registry")]
|
|
public class EncounterRegistry : ScriptableObject {
|
|
public EncountersCollection[] encounterCollections = Array.Empty<EncountersCollection>();
|
|
|
|
private readonly Dictionary<string, IEncounter> encounters = new();
|
|
|
|
public Dictionary<string, IEncounter> GetEncounters() => encounters;
|
|
|
|
public void RegisterEncounter(IEncounter encounter) {
|
|
var id = encounter?.EncounterDefinition?.internalId;
|
|
if(string.IsNullOrEmpty(id)) {
|
|
return;
|
|
}
|
|
encounters.TryAdd(id, encounter);
|
|
}
|
|
|
|
public void UnregisterEncounter(IEncounter encounter) {
|
|
var id = encounter?.EncounterDefinition?.internalId;
|
|
if(string.IsNullOrEmpty(id)) {
|
|
return;
|
|
}
|
|
encounters.Remove(id);
|
|
}
|
|
|
|
public void ClearEncounters() {
|
|
encounters.Clear();
|
|
}
|
|
|
|
public void PopulateEncounters() {
|
|
if(encounterCollections == null) {
|
|
return;
|
|
}
|
|
|
|
for(int c = 0; c < encounterCollections.Length; c++) {
|
|
var collection = encounterCollections[c];
|
|
if(collection?.encounterTables == null) {
|
|
continue;
|
|
}
|
|
|
|
for(int t = 0; t < collection.encounterTables.Length; t++) {
|
|
var table = collection.encounterTables[t];
|
|
if(table?.encounters == null) {
|
|
continue;
|
|
}
|
|
|
|
for(int i = 0; i < table.encounters.Count; i++) {
|
|
RegisterEncounter(table.encounters[i]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
public EncounterTable GetEncounterTable(string id) {
|
|
foreach(var collection in encounterCollections) {
|
|
foreach(var table in collection.encounterTables) {
|
|
if(table.id == id) {
|
|
return table;
|
|
}
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
public IEncounter GetRandomEncounter(string encounterTableId) {
|
|
var table = GetEncounterTable(encounterTableId);
|
|
if(!table) {
|
|
return null;
|
|
}
|
|
var ec = table.encounters;
|
|
return ec[UnityEngine.Random.Range(0, ec.Count)];
|
|
}
|
|
|
|
/// <summary>Random pick restricted to encounters whose runtime type matches <paramref name="type"/>.
|
|
/// Zero-alloc — uses reservoir sampling instead of building an intermediate filtered list.</summary>
|
|
public IEncounter GetRandomEncounter(string encounterTableId, IEncounterKind encounterKind) {
|
|
var table = GetEncounterTable(encounterTableId);
|
|
if(!table || encounterKind == null) {
|
|
return null;
|
|
}
|
|
|
|
IEncounter selected = null;
|
|
var seen = 0;
|
|
for(var i = 0; i < encounters.Count; i++) {
|
|
var encounter = table.encounters[i];
|
|
if(encounter == null || encounter.GetType() != encounterKind.GetType()) {
|
|
continue;
|
|
}
|
|
seen++;
|
|
if(UnityEngine.Random.Range(0, seen) == 0) {
|
|
selected = encounter;
|
|
}
|
|
}
|
|
return selected;
|
|
}
|
|
|
|
/// <summary>Random pick restricted by <paramref name="filter"/>. Used with
|
|
/// <see cref="QuestProgress.IsGated"/> to exclude gated encounters. Zero-alloc via reservoir sampling.</summary>
|
|
public IEncounter GetRandomEncounter(Predicate<IEncounter> filter, string encounterTableId) {
|
|
var table = GetEncounterTable(encounterTableId);
|
|
if(!table) {
|
|
return null;
|
|
}
|
|
if(filter == null) {
|
|
return GetRandomEncounter(encounterTableId);
|
|
}
|
|
|
|
IEncounter selected = null;
|
|
var seen = 0;
|
|
foreach(var encounter in table.encounters) {
|
|
if(encounter == null || !filter(encounter)) {
|
|
continue;
|
|
}
|
|
seen++;
|
|
if(UnityEngine.Random.Range(0, seen) == 0) {
|
|
selected = encounter;
|
|
}
|
|
}
|
|
return selected;
|
|
}
|
|
}
|
|
|
|
#if UNITY_EDITOR
|
|
/// <summary>Rebuilds the registry (Addressables key "EncounterRegistry") on any asset import/move/delete.</summary>
|
|
public class EncounterRegister : UnityEditor.AssetPostprocessor {
|
|
private static EncounterRegistry registryCache;
|
|
|
|
private static void OnPostprocessAllAssets(string[] importedAssets, string[] deletedAssets, string[] movedAssets, string[] movedFromAssetPaths) {
|
|
registryCache ??= Addressables.LoadAssetAsync<EncounterRegistry>("EncounterRegistry").WaitForCompletion();
|
|
registryCache.ClearEncounters();
|
|
registryCache.PopulateEncounters();
|
|
}
|
|
}
|
|
#endif
|
|
}
|