refactor: Round 53 remove all legacy backward-compatibility code
- QuestSO: remove giverNpcId, prerequisiteQuests/Flags/FlagsLogic, failCondition, conditionFlags, npcDialogueKey fields; simplify GiverNpcId property to giverNpc?.npcId; clean ValidatePrerequisiteCycles/HasPrerequisiteCycle to use prerequisites.questDependencies; remove ValidateBranchDialogueKeys migration warning block; clean QuestPrerequisite doc - QuestManager: remove OnLoad DataVersion 1/2 migration paths (ProgressCounts, hasNewFormat/ useNewFormat); remove CheckQuestDepsAndFlags old-field fallback (prerequisiteQuests/Flags); remove UnlockBranches conditionFlags fallback; remove DispatchEvent failCondition fallback; fix ValidatePrerequisites DFS to scan prerequisites.questDependencies - SaveData: remove ProgressCounts (Obsolete), ObjectiveIndex (unused), GiverNpcId (never written) fields from QuestState; simplify DataVersion doc comment - QuestSOEditor: replace migration-only editor with minimal DrawDefaultInspector - QuestModule: update all prerequisiteQuests/conditionFlags/npcDialogueKey/failCondition references to canonical new fields; update ValidateBranchFlags check 10 - FlagAuditModule: replace conditionFlags/prerequisiteFlags scans with conditionFlagEntries/ prerequisites.flagCondition.flags - NpcSO: remove QuestSO.giverNpcId reference from npcId tooltip - NpcAffinityEvent/RewardSO: update doc comments to reference giverNpc instead of giverNpcId Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
This commit is contained in:
@@ -10,7 +10,8 @@ namespace BaseGames.Quest
|
||||
[System.Serializable]
|
||||
public struct NpcAffinityEvent
|
||||
{
|
||||
/// <summary>发生好感度变化的 NPC ID(与 QuestSO.giverNpcId 保持一致)。</summary>
|
||||
/// <summary>
|
||||
/// NPC 好感度变化的 NPC ID(与 QuestSO.giverNpc.npcId 保持一致)。</summary>
|
||||
public string npcId;
|
||||
/// <summary>好感度变化量(正值=增加,负值=减少)。</summary>
|
||||
public int delta;
|
||||
|
||||
@@ -438,59 +438,34 @@ namespace BaseGames.Quest
|
||||
if (!conditionMet) continue;
|
||||
|
||||
// 世界状态标志条件(And/Or 由 conditionFlagsLogic 决定)
|
||||
// 优先用新版 conditionFlagEntries(支持 invert/NOT 取反),若为空则回退到旧版 conditionFlags
|
||||
// saveService 未注入时降级:跳过标志检查,仅由 conditionQuest 决定分支
|
||||
bool hasFlagEntries = branch.conditionFlagEntries != null && branch.conditionFlagEntries.Length > 0;
|
||||
bool hasLegacyFlags = branch.conditionFlags != null && branch.conditionFlags.Length > 0;
|
||||
bool hasFlagConds = hasFlagEntries || hasLegacyFlags;
|
||||
bool hasFlagEntries = branch.conditionFlagEntries != null && branch.conditionFlagEntries.Length > 0;
|
||||
|
||||
if (hasFlagConds && saveService != null)
|
||||
if (hasFlagEntries && saveService != null)
|
||||
{
|
||||
if (branch.conditionFlagsLogic == BaseGames.Core.WorldStateFlagLogic.Or)
|
||||
{
|
||||
conditionMet = false;
|
||||
if (hasFlagEntries)
|
||||
foreach (var entry in branch.conditionFlagEntries)
|
||||
{
|
||||
foreach (var entry in branch.conditionFlagEntries)
|
||||
{
|
||||
if (string.IsNullOrEmpty(entry.flagId)) continue;
|
||||
bool raw = saveService.GetFlag(entry.flagId);
|
||||
if (entry.invert ? !raw : raw) { conditionMet = true; break; }
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
foreach (var flag in branch.conditionFlags)
|
||||
{
|
||||
if (!string.IsNullOrEmpty(flag) && saveService.GetFlag(flag))
|
||||
{ conditionMet = true; break; }
|
||||
}
|
||||
if (string.IsNullOrEmpty(entry.flagId)) continue;
|
||||
bool raw = saveService.GetFlag(entry.flagId);
|
||||
if (entry.invert ? !raw : raw) { conditionMet = true; break; }
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
// AND(默认):全部标志均须满足(支持 invert 取反)
|
||||
if (hasFlagEntries)
|
||||
foreach (var entry in branch.conditionFlagEntries)
|
||||
{
|
||||
foreach (var entry in branch.conditionFlagEntries)
|
||||
{
|
||||
if (string.IsNullOrEmpty(entry.flagId)) continue;
|
||||
bool raw = saveService.GetFlag(entry.flagId);
|
||||
if (entry.invert ? raw : !raw) { conditionMet = false; break; }
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
foreach (var flag in branch.conditionFlags)
|
||||
{
|
||||
if (string.IsNullOrEmpty(flag)) continue;
|
||||
if (!saveService.GetFlag(flag)) { conditionMet = false; break; }
|
||||
}
|
||||
if (string.IsNullOrEmpty(entry.flagId)) continue;
|
||||
bool raw = saveService.GetFlag(entry.flagId);
|
||||
if (entry.invert ? raw : !raw) { conditionMet = false; break; }
|
||||
}
|
||||
}
|
||||
}
|
||||
#if UNITY_EDITOR || DEVELOPMENT_BUILD
|
||||
else if (hasFlagConds && saveService == null)
|
||||
else if (hasFlagEntries && saveService == null)
|
||||
{
|
||||
Debug.LogWarning(
|
||||
$"[QuestManager] 任务 '{questId}' 分支配置了标志条件,但 ISaveService 未注册," +
|
||||
@@ -700,14 +675,8 @@ namespace BaseGames.Quest
|
||||
var quest = GetQuestSO(id);
|
||||
if (quest?.objectives == null) continue;
|
||||
|
||||
bool hasNewFormat = saved.ObjectiveProgress != null && saved.ObjectiveProgress.Count > 0;
|
||||
// DataVersion >= 2:新格式(objectiveId 键值对);DataVersion <= 1 或遗留存档:旧格式(按索引)
|
||||
// Count > 0 作为无 DataVersion 字段时的兼容兜底
|
||||
bool useNewFormat = saved.DataVersion >= 2 || hasNewFormat;
|
||||
|
||||
if (useNewFormat && saved.ObjectiveProgress != null)
|
||||
if (saved.ObjectiveProgress != null)
|
||||
{
|
||||
// 新格式:objectiveId → count,重排顺序后仍可正确恢复
|
||||
foreach (var obj in quest.objectives)
|
||||
{
|
||||
if (obj == null) continue;
|
||||
@@ -716,29 +685,12 @@ namespace BaseGames.Quest
|
||||
if (!_objectiveStates.TryGetValue(compositeKey, out var os))
|
||||
os = _objectiveStates[compositeKey] = new QuestObjectiveState();
|
||||
os.progressCount = count;
|
||||
// DataVersion >= 3:从存档恢复 completed 标志(防止 GetRequiredCount 变更后判定漂移)
|
||||
if (saved.ObjectiveCompleted != null &&
|
||||
saved.ObjectiveCompleted.TryGetValue(obj.objectiveId, out bool done))
|
||||
os.completed = done;
|
||||
}
|
||||
}
|
||||
else if (saved.ProgressCounts != null
|
||||
#pragma warning disable CS0618 // ProgressCounts 弃用字段:仅在此处读取用于旧存档迁移,不再写入
|
||||
&& saved.ProgressCounts.Count > 0)
|
||||
{
|
||||
// 旧格式兼容(按数组索引):迁移旧存档用,不再写入新存档
|
||||
for (int i = 0; i < quest.objectives.Length && i < saved.ProgressCounts.Count; i++)
|
||||
{
|
||||
var obj = quest.objectives[i];
|
||||
if (obj == null) continue;
|
||||
string compositeKey = GetCompositeKey(id, obj.objectiveId);
|
||||
if (!_objectiveStates.TryGetValue(compositeKey, out var os))
|
||||
os = _objectiveStates[compositeKey] = new QuestObjectiveState();
|
||||
os.progressCount = saved.ProgressCounts[i];
|
||||
}
|
||||
}
|
||||
#pragma warning restore CS0618
|
||||
// DataVersion >= 3:恢复任务开始 / 完成时间戳(0 = 旧存档未记录,跳过)
|
||||
|
||||
if (saved.StartedAtUtc != 0) _startedAtUtc[id] = saved.StartedAtUtc;
|
||||
if (saved.CompletedAtUtc != 0) _completedAtUtc[id] = saved.CompletedAtUtc;
|
||||
}
|
||||
@@ -775,7 +727,7 @@ namespace BaseGames.Quest
|
||||
/// <summary>
|
||||
/// 初始化(或修正)所有任务的 Available/Unavailable 状态。
|
||||
/// 在 Awake(冷启动)和 OnLoad(存档恢复)后调用。
|
||||
/// OnLoad 后 ISaveService 已就绪,会重新评估 prerequisiteFlags,
|
||||
/// OnLoad 后 ISaveService 已就绪,会重新评估 prerequisites.flagCondition.flags,
|
||||
/// 修正 Awake 期间因服务未就绪而被跳过的标志检查。
|
||||
/// Active/Completed/Failed 状态来自存档,不重置。
|
||||
/// </summary>
|
||||
@@ -794,7 +746,7 @@ namespace BaseGames.Quest
|
||||
// _affinityInitialized 为 true 说明是 OnLoad 后调用,Awake 期间不打此日志
|
||||
bool isNewToSave = !_questStates.ContainsKey(q.questId) && _affinityInitialized;
|
||||
#endif
|
||||
// Available/Unavailable 均重新评估,确保 prerequisiteFlags 变更后状态正确
|
||||
// Available/Unavailable 均重新评估,确保 prerequisites.flagCondition.flags 变更后状态正确
|
||||
_questStates[q.questId] = MeetsPrerequisites(q) ? QuestStateEnum.Available : QuestStateEnum.Unavailable;
|
||||
#if UNITY_EDITOR || DEVELOPMENT_BUILD
|
||||
if (isNewToSave)
|
||||
@@ -824,64 +776,28 @@ namespace BaseGames.Quest
|
||||
{
|
||||
if (quest == null) return new QuestLockInfo { Reason = QuestLockReason.NotFound };
|
||||
|
||||
if (quest.prerequisites.HasAny)
|
||||
{
|
||||
if (quest.prerequisites.questDependencies != null)
|
||||
foreach (var dep in quest.prerequisites.questDependencies)
|
||||
{
|
||||
if (dep == null) continue;
|
||||
if (string.IsNullOrEmpty(dep.questId))
|
||||
{
|
||||
#if UNITY_EDITOR || DEVELOPMENT_BUILD
|
||||
Debug.LogWarning($"[QuestManager] 任务 '{quest.questId}' 的 prerequisites.questDependencies 含 questId 为空的条目,已跳过。");
|
||||
#endif
|
||||
continue;
|
||||
}
|
||||
if (GetState(dep.questId) != QuestStateEnum.Completed)
|
||||
return new QuestLockInfo { Reason = QuestLockReason.RequiresQuest, Param = dep.questId };
|
||||
}
|
||||
|
||||
var fc = quest.prerequisites.flagCondition;
|
||||
if (fc.flags != null && fc.flags.Length > 0)
|
||||
if (quest.prerequisites.questDependencies != null)
|
||||
foreach (var dep in quest.prerequisites.questDependencies)
|
||||
{
|
||||
var svc = BaseGames.Core.ServiceLocator.GetOrDefault<ISaveService>();
|
||||
// ISaveService 未就绪(Awake 阶段)→ 保守跳过;OnLoad 后重新评估
|
||||
if (svc != null && !EvaluateFlagPrerequisites(fc.flags, fc.logic, svc))
|
||||
return new QuestLockInfo { Reason = QuestLockReason.FlagConditionNotMet };
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
// 旧版字段回退(兼容现有资产)
|
||||
#pragma warning disable CS0618
|
||||
if (quest.prerequisiteQuests != null)
|
||||
foreach (var pre in quest.prerequisiteQuests)
|
||||
if (dep == null) continue;
|
||||
if (string.IsNullOrEmpty(dep.questId))
|
||||
{
|
||||
if (pre == null) continue;
|
||||
if (string.IsNullOrEmpty(pre.questId))
|
||||
{
|
||||
#if UNITY_EDITOR || DEVELOPMENT_BUILD
|
||||
Debug.LogWarning($"[QuestManager] 任务 '{quest.questId}' 的 prerequisiteQuests 含 questId 为空的条目,已跳过该前置条件。");
|
||||
Debug.LogWarning($"[QuestManager] 任务 '{quest.questId}' 的 prerequisites.questDependencies 含 questId 为空的条目,已跳过。");
|
||||
#endif
|
||||
continue;
|
||||
}
|
||||
if (GetState(pre.questId) != QuestStateEnum.Completed)
|
||||
return new QuestLockInfo { Reason = QuestLockReason.RequiresQuest, Param = pre.questId };
|
||||
continue;
|
||||
}
|
||||
if (GetState(dep.questId) != QuestStateEnum.Completed)
|
||||
return new QuestLockInfo { Reason = QuestLockReason.RequiresQuest, Param = dep.questId };
|
||||
}
|
||||
|
||||
if (quest.prerequisiteFlags != null && quest.prerequisiteFlags.Length > 0)
|
||||
{
|
||||
var svc = BaseGames.Core.ServiceLocator.GetOrDefault<ISaveService>();
|
||||
if (svc != null)
|
||||
{
|
||||
if (!EvaluateFlagPrerequisites(quest.prerequisiteFlags, quest.prerequisiteFlagsLogic, svc))
|
||||
return new QuestLockInfo { Reason = QuestLockReason.FlagConditionNotMet };
|
||||
}
|
||||
#if UNITY_EDITOR || DEVELOPMENT_BUILD
|
||||
else Debug.LogWarning($"[QuestManager] 任务 '{quest.questId}' 的 prerequisiteFlags 需要 ISaveService,但服务未注册,标志检查已跳过。");
|
||||
#endif
|
||||
}
|
||||
#pragma warning restore CS0618
|
||||
var fc = quest.prerequisites.flagCondition;
|
||||
if (fc.flags != null && fc.flags.Length > 0)
|
||||
{
|
||||
var svc = BaseGames.Core.ServiceLocator.GetOrDefault<ISaveService>();
|
||||
// ISaveService 未就绪(Awake 阶段)→ 保守跳过;OnLoad 后重新评估
|
||||
if (svc != null && !EvaluateFlagPrerequisites(fc.flags, fc.logic, svc))
|
||||
return new QuestLockInfo { Reason = QuestLockReason.FlagConditionNotMet };
|
||||
}
|
||||
|
||||
return new QuestLockInfo { Reason = QuestLockReason.None };
|
||||
@@ -1065,22 +981,14 @@ namespace BaseGames.Quest
|
||||
// 设计意图:暂停期间目标冻结,失败条件也不判定,恢复后再继续检查。
|
||||
if (quest.canFail)
|
||||
{
|
||||
// P3-A:多失败条件支持——failConditions 数组中任意一个达成即失败
|
||||
bool triggered = false;
|
||||
if (quest.failConditions != null && quest.failConditions.Length > 0)
|
||||
if (quest.failConditions != null)
|
||||
{
|
||||
foreach (var fc in quest.failConditions)
|
||||
{
|
||||
if (fc != null && CheckObjective(qid, fc)) { triggered = true; break; }
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
// 向后兼容旧版单一 failCondition 字段(Obsolete,将在后续版本移除)
|
||||
#pragma warning disable CS0618
|
||||
triggered = quest.failCondition != null && CheckObjective(qid, quest.failCondition);
|
||||
#pragma warning restore CS0618
|
||||
}
|
||||
if (triggered)
|
||||
{
|
||||
toFail ??= new List<string>();
|
||||
@@ -1181,7 +1089,7 @@ namespace BaseGames.Quest
|
||||
|
||||
#if UNITY_EDITOR || DEVELOPMENT_BUILD
|
||||
/// <summary>
|
||||
/// 通过 DFS 后序遍历检测 prerequisiteQuests 中是否存在循环引用。
|
||||
/// 通过 DFS 后序遍历检测 prerequisites.questDependencies 中是否存在循环引用。
|
||||
/// 在编辑器 OnValidate 及开发构建 Awake 时调用,发现问题立即打 LogError。
|
||||
/// </summary>
|
||||
[UnityEngine.ContextMenu("校验前置任务循环引用")]
|
||||
@@ -1219,10 +1127,10 @@ namespace BaseGames.Quest
|
||||
color[startId] = 1;
|
||||
path.Add(startId);
|
||||
|
||||
var prereqs = index[startId].prerequisiteQuests;
|
||||
if (prereqs != null)
|
||||
var prereqDeps = index[startId].prerequisites.questDependencies;
|
||||
if (prereqDeps != null)
|
||||
{
|
||||
foreach (var pre in prereqs)
|
||||
foreach (var pre in prereqDeps)
|
||||
{
|
||||
if (pre == null || string.IsNullOrEmpty(pre.questId)) continue;
|
||||
if (HasCycle(pre.questId, path)) return true;
|
||||
|
||||
@@ -16,21 +16,13 @@ namespace BaseGames.Quest
|
||||
[Tooltip("任务唯一 ID,如 \"Quest_FindMushroom\"。运行时由 QuestManager 以此为键索引,必须全局唯一。")]
|
||||
public string questId;
|
||||
|
||||
[Tooltip("发布/完成该任务的 NPC(直接引用 NpcSO 资产,推荐)。\n" +
|
||||
[Tooltip("发布/完成该任务的 NPC(直接引用 NpcSO 资产)。\n" +
|
||||
"用于完成任务后向该 NPC 应用 affinityBonus,及 CanAccept 好感度门槛检查。\n" +
|
||||
"留空时跳过好感度相关逻辑;与旧字段 giverNpcId 同时有值时以此 SO 为准。")]
|
||||
"留空时跳过好感度相关逻辑。")]
|
||||
public NpcSO giverNpc;
|
||||
|
||||
[System.Obsolete("已废弃,请改用 giverNpc(NpcSO 直接引用)。保留以兼容现有资产序列化。")]
|
||||
[HideInInspector]
|
||||
public string giverNpcId;
|
||||
|
||||
/// <summary>运行时使用的 NPC ID:giverNpc 优先,回退到旧字段 giverNpcId。</summary>
|
||||
public string GiverNpcId => (giverNpc != null && !string.IsNullOrEmpty(giverNpc.npcId))
|
||||
? giverNpc.npcId
|
||||
#pragma warning disable CS0618
|
||||
: giverNpcId;
|
||||
#pragma warning restore CS0618
|
||||
/// <summary>运行时使用的 NPC ID(来自 giverNpc.npcId)。</summary>
|
||||
public string GiverNpcId => giverNpc != null ? giverNpc.npcId : string.Empty;
|
||||
|
||||
[Tooltip("本地化 Key,格式如 \"Quest_FindMushroom_Name\"。通过 LocalizationManager.Get(displayNameKey, \"Quest\") 显示。")]
|
||||
public string displayNameKey;
|
||||
@@ -52,28 +44,9 @@ namespace BaseGames.Quest
|
||||
public QuestObjectiveSO[] objectives;
|
||||
|
||||
[Header("前置条件")]
|
||||
[Tooltip("任务前置条件(统一配置版)。将前置任务依赖和世界标志依赖合并为单一结构,便于 Inspector 管理。\n" +
|
||||
"如旧版字段(prerequisiteQuests / prerequisiteFlags)已有数据,运行时将自动回退使用旧版字段,无需手动迁移。")]
|
||||
[Tooltip("任务前置条件(前置任务依赖 + 世界标志依赖)。留空表示无前置限制。")]
|
||||
public QuestPrerequisite prerequisites = new QuestPrerequisite();
|
||||
|
||||
// ── 旧版前置字段(向后兼容,新配置请改用 prerequisites)────────────────
|
||||
[HideInInspector]
|
||||
[Tooltip("【已归入 prerequisites.questDependencies,此字段仅用于旧资产兼容】\n" +
|
||||
"所有前置任务必须处于 Completed 状态,本任务才能被接取。\n" +
|
||||
"⚠ 此字段计划在 v2.0 移除,请尽快通过 QuestSOEditor 迁移至 prerequisites。")]
|
||||
public QuestSO[] prerequisiteQuests;
|
||||
|
||||
[HideInInspector]
|
||||
[Tooltip("【已归入 prerequisites.flagCondition.logic,此字段仅用于旧资产兼容】\n" +
|
||||
"⚠ 此字段计划在 v2.0 移除,请尽快通过 QuestSOEditor 迁移至 prerequisites。")]
|
||||
public BaseGames.Core.WorldStateFlagLogic prerequisiteFlagsLogic = BaseGames.Core.WorldStateFlagLogic.And;
|
||||
|
||||
[HideInInspector]
|
||||
[Tooltip("【已归入 prerequisites.flagCondition.flags,此字段仅用于旧资产兼容】\n" +
|
||||
"⚠ 此字段计划在 v2.0 移除,请尽快通过 QuestSOEditor 迁移至 prerequisites。")]
|
||||
[BaseGames.Core.WorldStateFlag]
|
||||
public string[] prerequisiteFlags;
|
||||
|
||||
[Tooltip("接取本任务所需的 NPC 好感度下限(0 = 无限制)。由好感度系统提供实际数值。")]
|
||||
public int minAffinityToAccept;
|
||||
|
||||
@@ -88,10 +61,6 @@ namespace BaseGames.Quest
|
||||
[Tooltip("失败判定目标列表(任意一个达成即失败)。canFail=true 时有效。\n" +
|
||||
"支持多个失败条件(如「BOSS 在限时内未被击败」OR「关键 NPC 死亡」)。")]
|
||||
public QuestObjectiveSO[] failConditions;
|
||||
[System.Obsolete("已废弃,请改用 failConditions(数组,支持多个失败条件)。保留以兼容现有资产序列化。")]
|
||||
[HideInInspector]
|
||||
[Tooltip("(旧版单一失败条件,已被 failConditions 数组取代。保留以兼容现有资产。)")]
|
||||
public QuestObjectiveSO failCondition;
|
||||
|
||||
[Header("接取/完成对话")]
|
||||
[Tooltip("玩家接取任务时自动触发的 NPC 对话序列(如 NPC 委托台词)。\n" +
|
||||
@@ -182,10 +151,7 @@ namespace BaseGames.Quest
|
||||
var visited = new System.Collections.Generic.HashSet<string>(System.StringComparer.Ordinal);
|
||||
visited.Add(questId);
|
||||
|
||||
#pragma warning disable CS0618
|
||||
QuestSO[] deps = prerequisites.HasAny ? prerequisites.questDependencies : prerequisiteQuests;
|
||||
#pragma warning restore CS0618
|
||||
|
||||
QuestSO[] deps = prerequisites.questDependencies;
|
||||
if (deps == null) return;
|
||||
|
||||
foreach (var dep in deps)
|
||||
@@ -196,7 +162,7 @@ namespace BaseGames.Quest
|
||||
Debug.LogError(
|
||||
$"[QuestSO] '{name}'(questId='{questId}')的前置任务链存在循环依赖!" +
|
||||
$"前置任务 '{dep.name}' 最终指回自身或已访问任务," +
|
||||
"运行时将导致任务无法被接取。请检查 prerequisites/prerequisiteQuests 配置。", this);
|
||||
"运行时将导致任务无法被接取。请检查 prerequisites 配置。", this);
|
||||
return;
|
||||
}
|
||||
}
|
||||
@@ -218,9 +184,7 @@ namespace BaseGames.Quest
|
||||
if (string.IsNullOrEmpty(quest.questId)) return false;
|
||||
if (!visited.Add(quest.questId)) return true; // 已在当前路径上 = 环路
|
||||
|
||||
#pragma warning disable CS0618
|
||||
QuestSO[] deps = quest.prerequisites.HasAny ? quest.prerequisites.questDependencies : quest.prerequisiteQuests;
|
||||
#pragma warning restore CS0618
|
||||
QuestSO[] deps = quest.prerequisites.questDependencies;
|
||||
|
||||
if (deps != null)
|
||||
{
|
||||
@@ -288,23 +252,7 @@ namespace BaseGames.Quest
|
||||
|
||||
private void ValidateBranchDialogueKeys()
|
||||
{
|
||||
if (branches == null || branches.Length == 0) return;
|
||||
|
||||
foreach (var branch in branches)
|
||||
{
|
||||
if (branch == null) continue;
|
||||
|
||||
// npcDialogueSequence 是 SO 直接引用,无需字符串校验。
|
||||
// 旧字段 npcDialogueKey(Obsolete)有值时提示迁移。
|
||||
#pragma warning disable CS0618
|
||||
if (!string.IsNullOrEmpty(branch.npcDialogueKey) && branch.npcDialogueSequence == null)
|
||||
{
|
||||
Debug.LogWarning(
|
||||
$"[QuestSO] '{name}' 分支仍使用旧字段 npcDialogueKey='{branch.npcDialogueKey}'," +
|
||||
"请迁移至 npcDialogueSequence(直接拖入 DialogueSequenceSO)。", this);
|
||||
}
|
||||
#pragma warning restore CS0618
|
||||
}
|
||||
// npcDialogueSequence 是 SO 直接引用;旧字符串字段已移除。
|
||||
}
|
||||
#endif
|
||||
}
|
||||
@@ -318,20 +266,12 @@ namespace BaseGames.Quest
|
||||
" And(默认)= 全部 conditionFlagEntries 均满足才走本分支\n" +
|
||||
" Or = 任意一个 conditionFlagEntry 满足即可走本分支")]
|
||||
public BaseGames.Core.WorldStateFlagLogic conditionFlagsLogic = BaseGames.Core.WorldStateFlagLogic.And;
|
||||
[Tooltip("世界状态标志条件(支持 invert 取反)。按 conditionFlagsLogic 逻辑与 conditionQuest 共同决定分支是否激活。\n" +
|
||||
"优先使用此字段;若为空则自动回退到旧版 conditionFlags 以保证兼容性。")]
|
||||
[Tooltip("世界状态标志条件(支持 invert 取反)。按 conditionFlagsLogic 逻辑与 conditionQuest 共同决定分支是否激活。")]
|
||||
public BranchFlagEntry[] conditionFlagEntries;
|
||||
[Tooltip("(旧版兼容字段,已被 conditionFlagEntries 取代。如 conditionFlagEntries 不为空则本字段被忽略。)")]
|
||||
[HideInInspector]
|
||||
public string[] conditionFlags;
|
||||
[Tooltip("本分支解锁的后续任务。满足所有条件后,此任务将被设为 Available。")]
|
||||
public QuestSO nextQuest;
|
||||
[Tooltip("完成本任务后触发的 NPC 对话序列(直接引用 DialogueSequenceSO 资产,无需手写 ID)。")]
|
||||
public DialogueSequenceSO npcDialogueSequence;
|
||||
|
||||
[System.Obsolete("已废弃,请改用 npcDialogueSequence(直接 SO 引用)。保留字段以兼容现有资产序列化。")]
|
||||
[HideInInspector]
|
||||
public string npcDialogueKey;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
@@ -365,10 +305,7 @@ namespace BaseGames.Quest
|
||||
// =========================================================================
|
||||
|
||||
/// <summary>
|
||||
/// 任务前置条件统一配置结构。
|
||||
/// 将旧版三个独立字段(prerequisiteQuests / prerequisiteFlags / prerequisiteFlagsLogic)
|
||||
/// 合并为单一可序列化类,便于 Inspector 统一管理与代码维护。
|
||||
/// 运行时通过 <see cref="HasAny"/> 判断是否启用新格式;若未配置则自动回退到旧版字段。
|
||||
/// 任务前置条件统一配置结构,将前置任务依赖和世界状态标志依赖合并为单一可序列化类。
|
||||
/// </summary>
|
||||
[Serializable]
|
||||
public class QuestPrerequisite
|
||||
@@ -379,7 +316,7 @@ namespace BaseGames.Quest
|
||||
[Tooltip("世界状态标志前置条件(支持 And / Or 逻辑)。")]
|
||||
public FlagCondition flagCondition;
|
||||
|
||||
/// <summary>此前置结构是否配置了任何条件(用于判断是否启用新格式,回退到旧字段)。</summary>
|
||||
/// <summary>此前置结构是否配置了任何条件。</summary>
|
||||
public bool HasAny =>
|
||||
(questDependencies != null && questDependencies.Length > 0) ||
|
||||
(flagCondition.flags != null && flagCondition.flags.Length > 0);
|
||||
|
||||
@@ -25,7 +25,7 @@ namespace BaseGames.Quest
|
||||
public string[] itemIds;
|
||||
|
||||
[Header("NPC 关系")]
|
||||
[Tooltip("完成任务后对 giverNpcId(QuestSO 中配置)的好感度增量。\n" +
|
||||
[Tooltip("完成任务后对发布 NPC(QuestSO.giverNpc)的好感度增量。\n" +
|
||||
"正值=好感增加,负值=好感降低。0 = 不影响好感度。\n" +
|
||||
"增量以强类型 NpcAffinityEvent(npcId + delta + newTotal)广播至 EVT_NpcAffinityChanged,\n" +
|
||||
"并持久化到 SaveData.World.NpcRelations。接收方无需字符串解析。")]
|
||||
|
||||
Reference in New Issue
Block a user