2025-10-23 00:49:44 -04:00
|
|
|
|
#if UNITY_EDITOR
|
2025-10-24 09:11:22 -04:00
|
|
|
|
using System;
|
2026-03-20 11:56:50 -04:00
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
|
using System.IO;
|
|
|
|
|
|
using System.Linq;
|
|
|
|
|
|
using Sirenix.OdinInspector;
|
2025-10-23 00:49:44 -04:00
|
|
|
|
using UnityEditor;
|
|
|
|
|
|
using UnityEngine;
|
|
|
|
|
|
|
|
|
|
|
|
namespace Continentis.MainGame.Equipment
|
|
|
|
|
|
{
|
2026-03-20 11:56:50 -04:00
|
|
|
|
// =========================================================================
|
|
|
|
|
|
// EquipmentDataEditor
|
|
|
|
|
|
//
|
|
|
|
|
|
// 注意:此类已不再使用 [CustomEditor] 注解。
|
|
|
|
|
|
// Odin Inspector 会自动接管 EquipmentData(ScriptableObject)的 Inspector 渲染,
|
|
|
|
|
|
// 无需任何自定义 Editor 类。本文件现主要用于承载 EquipmentData 的编辑器扩展分部类。
|
|
|
|
|
|
// =========================================================================
|
|
|
|
|
|
internal static class EquipmentDataEditorPlaceholder
|
|
|
|
|
|
{
|
|
|
|
|
|
// 保留此类以维持文件存在意义,可在此添加未来的编辑器工具方法。
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// =========================================================================
|
|
|
|
|
|
// partial class EquipmentData — 编辑器专属扩展
|
|
|
|
|
|
//
|
|
|
|
|
|
// 包含:
|
|
|
|
|
|
// 1. GetEquipmentLogicDropdownItems() — 装备逻辑类下拉列表(静态)
|
|
|
|
|
|
// 2. OnEquipmentLogicClassSelected() — 选中逻辑类后自动填充字段的回调
|
|
|
|
|
|
// 3. GetAvailable*() — References 列表的资产名称选择器
|
|
|
|
|
|
// 4. 内部共享辅助方法(AssetDatabase 查询)
|
|
|
|
|
|
// =========================================================================
|
|
|
|
|
|
public partial class EquipmentData
|
|
|
|
|
|
{
|
|
|
|
|
|
// ── 1. 装备逻辑类选择器 ────────────────────────────────────────────────
|
|
|
|
|
|
// 替代旧 EquipmentDataEditor 中的 DrawSearchableTypeSelector()
|
|
|
|
|
|
// 配合 EquipmentData.cs 中 _classSelector 字段上的
|
|
|
|
|
|
// [ValueDropdown("@EquipmentData.GetEquipmentLogicDropdownItems()")]
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 为 [ValueDropdown] 提供所有 EquipmentBase 子类的层级下拉项。
|
|
|
|
|
|
/// 路径格式为 "ModName/Category/ClassName",与旧 TypeSelectorWindow 分组逻辑一致。
|
|
|
|
|
|
/// 此方法为 static,故在 [ValueDropdown] 中使用 @ 表达式语法引用。
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public static IEnumerable<ValueDropdownItem<string>> GetEquipmentLogicDropdownItems()
|
2025-10-23 00:49:44 -04:00
|
|
|
|
{
|
2026-03-20 11:56:50 -04:00
|
|
|
|
const string namespacePrefix = "Continentis.Mods";
|
|
|
|
|
|
const string namespaceToRemove = "Equipments";
|
|
|
|
|
|
|
|
|
|
|
|
IEnumerable<Type> types = AppDomain.CurrentDomain.GetAssemblies()
|
|
|
|
|
|
.SelectMany(a =>
|
|
|
|
|
|
{
|
|
|
|
|
|
try { return a.GetTypes(); }
|
|
|
|
|
|
catch { return Type.EmptyTypes; }
|
|
|
|
|
|
})
|
|
|
|
|
|
.Where(t => typeof(EquipmentBase).IsAssignableFrom(t)
|
|
|
|
|
|
&& !t.IsAbstract
|
|
|
|
|
|
&& !t.IsInterface
|
|
|
|
|
|
&& t != typeof(EquipmentBase));
|
|
|
|
|
|
|
|
|
|
|
|
foreach (Type type in types.OrderBy(t => t.FullName))
|
|
|
|
|
|
{
|
|
|
|
|
|
string path;
|
|
|
|
|
|
|
|
|
|
|
|
if (type.Namespace != null && type.Namespace.StartsWith(namespacePrefix))
|
|
|
|
|
|
{
|
|
|
|
|
|
List<string> segments = type.Namespace
|
|
|
|
|
|
.Substring(namespacePrefix.Length)
|
|
|
|
|
|
.Split('.')
|
|
|
|
|
|
.Where(s => !string.IsNullOrEmpty(s) && s != namespaceToRemove)
|
|
|
|
|
|
.ToList();
|
|
|
|
|
|
|
|
|
|
|
|
path = segments.Count > 0
|
|
|
|
|
|
? string.Join("/", segments) + "/" + type.Name
|
|
|
|
|
|
: type.Name;
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
path = "Uncategorized/" + type.Name;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// value 存储类型的完整名称(FullName),便于在 OnValueChanged 中精确反查
|
|
|
|
|
|
yield return new ValueDropdownItem<string>(path, type.FullName ?? type.Name);
|
|
|
|
|
|
}
|
2025-10-23 00:49:44 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-03-20 11:56:50 -04:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 当 classFullName 通过下拉菜单改变后,自动填充 className / modName / displayName。
|
|
|
|
|
|
/// 被 EquipmentData.cs 中 classFullName 字段上的 [OnValueChanged("OnEquipmentLogicClassSelected")] 触发。
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
private void OnEquipmentLogicClassSelected()
|
2025-10-23 00:49:44 -04:00
|
|
|
|
{
|
2026-03-20 11:56:50 -04:00
|
|
|
|
if (string.IsNullOrEmpty(classFullName)) return;
|
2025-10-23 00:49:44 -04:00
|
|
|
|
|
2026-03-20 11:56:50 -04:00
|
|
|
|
// 根据 FullName 反查对应 Type
|
|
|
|
|
|
Type selectedType = AppDomain.CurrentDomain.GetAssemblies()
|
|
|
|
|
|
.SelectMany(a =>
|
|
|
|
|
|
{
|
|
|
|
|
|
try { return a.GetTypes(); }
|
|
|
|
|
|
catch { return Type.EmptyTypes; }
|
|
|
|
|
|
})
|
|
|
|
|
|
.FirstOrDefault(t =>
|
|
|
|
|
|
(t.FullName == classFullName || t.Name == classFullName)
|
|
|
|
|
|
&& typeof(EquipmentBase).IsAssignableFrom(t)
|
|
|
|
|
|
&& !t.IsAbstract);
|
2025-10-23 00:49:44 -04:00
|
|
|
|
|
2026-03-20 11:56:50 -04:00
|
|
|
|
if (selectedType == null) return;
|
|
|
|
|
|
|
|
|
|
|
|
const string prefix = "Continentis.Mods";
|
|
|
|
|
|
string ns = selectedType.Namespace ?? string.Empty;
|
|
|
|
|
|
|
|
|
|
|
|
if (ns.StartsWith(prefix))
|
2025-10-23 00:49:44 -04:00
|
|
|
|
{
|
2026-03-20 11:56:50 -04:00
|
|
|
|
// 例:ns = "Continentis.Mods.Basic.Equipments.Sword"
|
|
|
|
|
|
// → afterPrefix = ".Basic.Equipments.Sword" (if prefix is without dot)
|
|
|
|
|
|
// 统一处理点号
|
|
|
|
|
|
string afterPrefix = ns.Substring(prefix.Length).TrimStart('.');
|
|
|
|
|
|
string[] parts = afterPrefix.Split('.');
|
|
|
|
|
|
string resolvedMod = parts.Length > 0 ? parts[0] : string.Empty;
|
|
|
|
|
|
|
|
|
|
|
|
// 移除 "Equipments" 层级并计算 className(短路径模式)
|
|
|
|
|
|
string equipmentsSegment = "Equipments";
|
|
|
|
|
|
string resolvedCategory = ns.Contains(equipmentsSegment)
|
|
|
|
|
|
? ns.Split(new[] { equipmentsSegment }, StringSplitOptions.None).Last().TrimStart('.')
|
|
|
|
|
|
: string.Empty;
|
|
|
|
|
|
|
|
|
|
|
|
modName = resolvedMod;
|
|
|
|
|
|
className = string.IsNullOrEmpty(resolvedCategory)
|
|
|
|
|
|
? selectedType.Name
|
|
|
|
|
|
: resolvedCategory + "." + selectedType.Name;
|
2025-10-24 09:11:22 -04:00
|
|
|
|
|
2026-03-20 11:56:50 -04:00
|
|
|
|
displayName = $"Equipment_{resolvedMod}_{selectedType.Name}_DisplayName";
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
modName = string.Empty;
|
|
|
|
|
|
className = selectedType.Name;
|
|
|
|
|
|
displayName = $"Equipment_{selectedType.Name}_DisplayName";
|
2025-10-23 00:49:44 -04:00
|
|
|
|
}
|
2025-10-24 09:11:22 -04:00
|
|
|
|
|
2026-03-20 11:56:50 -04:00
|
|
|
|
EditorUtility.SetDirty(this);
|
|
|
|
|
|
|
|
|
|
|
|
Debug.Log($"[EquipmentData] 已自动填充 → modName: {modName}, className: {className}, " +
|
|
|
|
|
|
$"displayName: {displayName}, classFullName: {classFullName}");
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ── 2. References 列表的资产名称选择器 ──────────────────────────────────
|
|
|
|
|
|
// 替代旧 DrawCharacterListGUI<T>(prop)
|
|
|
|
|
|
|
|
|
|
|
|
private IEnumerable<ValueDropdownItem<string>> GetAvailablePrefabs()
|
|
|
|
|
|
=> GetAssetNameDropdown("t:Prefab");
|
|
|
|
|
|
|
|
|
|
|
|
private IEnumerable<ValueDropdownItem<string>> GetAvailableCardData()
|
|
|
|
|
|
=> GetAssetNameDropdown("t:CardData");
|
|
|
|
|
|
|
|
|
|
|
|
private IEnumerable<ValueDropdownItem<string>> GetAvailableCharacterData()
|
|
|
|
|
|
=> GetAssetNameDropdown("t:CharacterData");
|
|
|
|
|
|
|
|
|
|
|
|
// ── 内部共享辅助方法 ─────────────────────────────────────────────────
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 通过 AssetDatabase 搜索特定类型/标签的资产,将文件名(无扩展名)作为下拉项返回。
|
|
|
|
|
|
/// 用于 References 列表的字符串引用选择。
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
private static IEnumerable<ValueDropdownItem<string>> GetAssetNameDropdown(string searchFilter)
|
|
|
|
|
|
{
|
|
|
|
|
|
string[] guids = AssetDatabase.FindAssets(searchFilter);
|
|
|
|
|
|
foreach (string guid in guids)
|
2025-10-23 00:49:44 -04:00
|
|
|
|
{
|
2026-03-20 11:56:50 -04:00
|
|
|
|
string path = AssetDatabase.GUIDToAssetPath(guid);
|
|
|
|
|
|
string name = Path.GetFileNameWithoutExtension(path);
|
|
|
|
|
|
yield return new ValueDropdownItem<string>(name, name);
|
2025-10-23 00:49:44 -04:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2026-03-20 11:56:50 -04:00
|
|
|
|
}
|
2025-10-23 00:49:44 -04:00
|
|
|
|
}
|
|
|
|
|
|
#endif
|