WitchMendokusai DevLog 02
π _
μ΄λ² μΌμ§μμλ μμ λ€μ κ΄λ¦¬νκΈ° μν ν΄ κΈ°λ°μ λ§λ€μ΄λ³Έλ€.
π UI Toolkit, κΈ°λ³Έμ μΈ λ μ΄μμ
μ μ ν΄μ λ§λ€μ΄λ³΄λ €κ³ UI ToolKitμ μ κΉ κ³΅λΆν΄λ³Έ μ μ΄ μλ€.
μ΄μ΄μ UI ToolKitμΌλ‘ λ§λ€μ΄λ³Έλ€.
κΈ°λ³Έμ μΈ λ μ΄μμμ μ μμμ μ°Έκ³ νλ€.
μμ μ€λͺ
λμ μ ν λ§ν¬μμ μμΈν νν 리μΌμ μ 곡νλ€.
ν΄λΉ νν 리μΌμμλ λ°νμμμ μ¬μ©νκΈ° μν μΈλ²€ν 리 UIλ₯Ό λͺ©νλ‘ νλλ°, UI ToolKitμ λ°νμκ³Ό μλν° UI λͺ¨λμ μ¬μ©ν μ μκΈ° λλ¬Έμ ν¬κ² λ¬Έμ λ μλ€.
νν 리μΌμ λ³΄κ³ λ°λΌ λ§λ UIμ, λͺ κ°μ§ κΈ°λ₯μ μΆκ°ν λͺ¨μ΅μ΄λ€.
미리 InitAllList()
ν¨μμμ QuestData
νμ
μ μ€ν¬λ¦½ν°λΈ μ€λΈμ νΈ λ¦¬μ€νΈλ₯Ό μ μ₯νκ³ ,
CreateGUI()
ν¨μμμ κ° QuestData
λ₯Ό MArtifactVisual
λ‘ λ§λ€μ΄ μΆκ°νλ€.
μλκ° κ·Έ μ½λλ€.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public class MArtifactVisual : Button
{
public Artifact Artifact { get; private set;}
public MArtifactVisual(Artifact artifact)
{
this.Artifact = artifact;
name = $"{artifact.Name}";
Add(new Label(){ text = artifact.Name });
Add(new Label(){ text = artifact.ID.ToString() });
if (artifact.Sprite != null)
style.backgroundImage = artifact.Sprite.texture;
AddToClassList("slot-icons");
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
private void InitAllList()
{
const string QUEST_DIRECTORY_PATH = "Assets/_Mascari4615/ScriptableObjects/Quest/";
questDatas = new();
InitList(ref questDatas, QUEST_DIRECTORY_PATH);
static void InitList<T>(ref List<T> list, string dirPath, bool searchSubDir = true) where T : ScriptableObject
{
const string extension = ".asset";
DirectoryInfo dir = new(dirPath);
foreach (FileInfo file in dir.GetFiles())
{
if (string.Compare(file.Extension, extension, StringComparison.Ordinal) != 0)
continue;
// QuestData μ€ν¬λ¦½ν°λΈ κ°μ²΄κ° μλλ©΄ Continue
if (AssetDatabase.GetMainAssetTypeAtPath($"{dirPath}/{file.Name}") != typeof(T))
continue;
list.Add(AssetDatabase.LoadAssetAtPath<T>($"{dirPath}/{file.Name}"));
}
if (searchSubDir)
{
// dir μλ λͺ¨λ ν΄λ μμ μλ νμΌμ νμ
foreach (DirectoryInfo subDir in dir.GetDirectories())
InitList(ref list, $"{dirPath}/{subDir.Name}/");
}
}
}
1
2
3
4
5
6
7
8
9
10
11
12
public void CreateGUI()
{
// ...
VisualElement grid = rootVisualElement.Q<VisualElement>(name: "Grid");
foreach (QuestData questData in questDatas)
{
MAritifactVisual mAritifactVisual = new(questData);
mAritifactVisual.RegisterCallback<ClickEvent>(ShowArtifact);
grid.Add(mAritifactVisual);
}
}
π μμ μ λ ¬
μ μ€ν¬λ¦°μ·μ λ€μ μ΄ν΄λ³΄λ©΄, μμ μμκ° μλ§μΈκ±Έ μ μ μλ€.
κ° μμ
μ¬λ‘―μμ μ΄λ¦ ν
μ€νΈ μμ μλ κ²μ λ΄κ° μμλ‘ μ€μ ν ID
λ€.
보면 ID
1 μμ
λ€μ ID
2 μμ
μ΄ μλλΌ ID
100 μμ
μ΄ μ€κ³ μκ³ ,
λ ID
3 μμ
μ 맨 λ€μ μμΉνλ€.
μ΄λ κ² μμκ° μλ§μΈ μ΄μ λ,
νμΌμ λΆλ¬μ¬ λ νμΌ μ΄λ¦ λ¬Έμμ΄ μμλ‘ λΆλ¬μ€κ³ , (ν¨μ λ¬Έμμ κ·Έλ° μΈκΈμ μμ§λ§)
λ¨Όμ νμ¬ λλ ν 리μ μλ νμΌλ€μ 리μ€νΈμ μΆκ°ν λ€ νμ λλ ν 리λ₯Ό νμνκΈ° λλ¬Έμ΄λ€.
μ΄μ¨κ±°λ, μμ μ μ λ ¬ν΄μ€μΌ νλ€.
πΏ λμ λ리?
κ΅³μ΄ μ λ ¬νμ§ μκ³ Dictionary<int, Artifact>
λͺ¨μμΌλ‘ λμ
λ리λ₯Ό μ¨λ λ κ² κ°κΈ΄νλ€.
ID
λ₯Ό Artifact
μ κ³ μ ν κ°μΌλ‘ μ€μ ν κ²μ΄κΈ° λλ¬Έμ κ°μ ID
λ₯Ό κ°μ§ Artifact
λ μλ€.
λ§μ½ μ‘΄μ¬νλ€λ©΄ μλͺ»λ ID
μ΄κΈ° λλ¬Έμ ID
λ₯Ό μμ ν΄μΌ νλ€.
μ λ ¬ μκ³ λ¦¬λ¬μ μ΄λ€λ©΄, μ΄λ―Έ μ‘΄μ¬νλ ID
μΈμ§ νμΈνκΈ° μν΄ λ³λμ 컨ν
μ΄λλ₯Ό μ¨μΌ νλλ°,
κ·Έλ₯ λμ
λ리λ₯Ό μ΄λ€λ©΄, λ¨μν ID
λ‘ Contains
λ TryGetValue
λ₯Ό μ¨λ³΄λ©΄ μ μ μλ€.
μ¬λ‘― UI μΆκ°λ,
ID
κ° μ»€λ΄€μ 10λ§λ³΄λ€λ μμ κ±°λΌμ, for
λ¬ΈμΌλ‘ 0 ~ MAX_ID
κΉμ§ λ리면μ TryGetValue
κ° true
μΈ κ²μ λν΄ μ¬λ‘―μ μΆκ°νλ©΄ λ κ² κ°λ€.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
public void CreateGUI()
{
// ...
System.Diagnostics.Stopwatch sw = new();
sw.Start();
VisualElement grid = rootVisualElement.Q<VisualElement>(name: "Grid");
for (int i = 0; i < ID_MAX; i++)
{
if (questDataDic.TryGetValue(i, out QuestData questData))
{
MArtifactVisual mAritifactVisual = new(questData);
mAritifactVisual.RegisterCallback<ClickEvent>(ShowArtifact);
grid.Add(mAritifactVisual);
}
}
sw.Stop();
Debug.Log($"TryGetValue x {ID_MAX} = {sw.ElapsedMilliseconds}ms");
}
μ, λΉμ₯ μλλ κ±±μ μλ€!
νμ΄μ§ λ§λλ κ²λ λ¨μν μ΄ λͺ¨μμμ ν νμ΄μ§μ νμν μμ μ λ§νΌ TryGetValue
κ° true
λ©΄ μΆκ°νλ©΄ λ κ²μ΄λ€.
λ€μ λ¨κ³λ‘ λμ΄κ°λ€.
π μμ κ΄λ¦¬
μμ μ κ΄λ¦¬νλ κΈ°λ₯μ λ§λ€μ΄λ³΄μ.
μμ
μ λ§λλ κΈ°λ₯μ λ³λμ UIκ° νμν κ² κ°μμ,
λΉμ₯ λ²νΌ νλλ§ μμΌλ©΄ λλ μμ
볡μ μ μμ κΈ°λ₯λΆν° ꡬνν΄λ³Έλ€.
πΏ μμ 볡μ
μμ 볡μ κΈ°λ₯μ΄λ€.
λ μ΄μμ μ°μΈ‘μ μμΉν ν΄νμ κΈ°λ₯μ λ΄λΉνλ MArtifactDetail
ν΄λμ€λ₯Ό μλ‘ λ§λ€κ³ ,
MArtifactDetail
μμ λ²νΌμ ν΄λ¦ νμ λ, νμ¬ μ νλ Artifact
μ λν΄ MArtifact
μ DuplicateArtifact
λ₯Ό νΈμΆνλλ‘ νλ€.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public class MArtifactDetail
{
// ...
public MArtifactDetail()
{
// ...
duplicateButton = root.Q<Button>(name: "BTN_Dup");
duplicateButton.clicked += DuplicateCurArtifact;
}
public void DuplicateCurArtifact()
{
MArtifact.Instance.DuplicateArtifact(CurArtifact);
}
// ...
}
μ΄ν MArttifact
μμ μ€μ DuplicateArtifact
ν¨μλ₯Ό ꡬννλ€.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
public class MArtifact : EditorWindow
{
public void DuplicateArtifact(Artifact artifact)
{
string nName = artifact.Name + " Copy";
// μ€λ³΅λμ§ μλ IDλ₯Ό μ°Ύλλ€.
int nID = artifact.ID + 1;
while (questDataDic.ContainsKey(nID))
nID++;
string assetName = $"Q_{nID}_{nName}";
string path = AssetDatabase.GenerateUniqueAssetPath($"{QUEST_DIRECTORY_PATH}{assetName}.asset");
AssetDatabase.CopyAsset(AssetDatabase.GetAssetPath(artifact), path);
QuestData newQuestData = AssetDatabase.LoadAssetAtPath<QuestData>(path);
newQuestData.ID = nID;
newQuestData.Name = nName;
questDataDic.Add(nID, newQuestData);
VisualElement grid = rootVisualElement.Q<VisualElement>(name: "Grid");
grid.Add(new MArtifactVisual(newQuestData));
Repaint();
}
}
μ½λλ₯Ό 보면 while (questDataDic.ContainsKey(nID))
μ questDataDic.Add(nID, newQuestData);
μμ questDataDic
μ μ¬μ©νκ³ , λ μμ
μ λ€μ λ‘λν λ QuestData
νμ
μΌλ‘ λΆλ¬μ€κ³ μλ€.
νμ§λ§ μ΄ ν¨μλ λͺ¨λ Artifact
μ€ν¬λ¦½ν°λΈ μ€λΈμ νΈμ λν΄μ μ¬μ©ν κ²μ΄κΈ° λλ¬Έμ, μΌλ°ν ν΄μΌνλ€.
μ΄λ»κ² μΌλ°ν μν€λ©΄ μ’μκΉ?
μΌλ¨ νμ€νΈλ₯Ό μ μ₯ν΄λκΈ° μν questDataDic
μ΄ μλ κ² μ²λΌ, λ€λ₯Έ Artifact
λ€λ κ°μ 컨ν
μ΄λκ° μμ κ²μ΄λ€.
μ λ¬λ°μ Artifact artifact
λ₯Ό μ€μ νμ
μ λ§λ 컨ν
μ΄λμ μΆκ°ν΄ λ°μ΄ν°λ₯Ό κ°±μ ν΄μ€μΌ νλ€.
Dictionary<Type, Dictionary>
λ₯Ό λ§λ€μ΄μ Dictionary
λ€λ νλμ Dictionary
λ‘ κ΄λ¦¬ν΄λ³ΌκΉ?
Dictionary
μ Value
νμ
μ κ³ μ ν΄μΌ νκΈ°μ Dictionary<Type, Dictionary<int, Artifact>>
νμ
μΌλ‘ λ§λ€κ³ , μΈ λλ λ€μ΄μΊμ€ν
μ΄ ν΄μ μ°λ©΄ ν¬κ² λ¬Έμ κ° λ κ² κ°μ§ μλ€.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
public void DuplicateArtifact(Artifact artifact)
{
Type type = artifact.GetType();
Dictionary<int, Artifact> dic = dataDics[type];
string nName = artifact.Name + " Copy";
// μ¬μ©λμ§ μμ IDλ₯Ό μ°Ύλλ€.
int nID = artifact.ID + 1;
while (dic.ContainsKey(nID))
nID++;
string assetName = $"Q_{nID}_{nName}";
string path = AssetDatabase.GenerateUniqueAssetPath($"{QUEST_DIRECTORY_PATH}{assetName}.asset");
AssetDatabase.CopyAsset(AssetDatabase.GetAssetPath(artifact), path);
Artifact newArtifact = AssetDatabase.LoadAssetAtPath<Artifact>(path);
newArtifact.ID = nID;
newArtifact.Name = nName;
dic.Add(nID, newArtifact);
VisualElement grid = rootVisualElement.Q<VisualElement>(name: "Grid");
grid.Add(new MArtifactVisual(newArtifact));
Repaint();
}
μμ λ μ½λλ€.
μ΄λ κ² λ³΅μ κΈ°λ₯μ λ§λ€μ΄λ΄€λ€.
πΏ μμ μμ
μμ μμ κΈ°λ₯μ ꡬνν΄λ³Έλ€.
볡μ κΈ°λ₯μ²λΌ μ£Όμ΄μ§ Artifactμ λ§λ 컨ν
μ΄λλ₯Ό μ°Ύμ μμ ν΄μΌ νλ€.
.. λ νλ μ½λ©μ ν΄μΌ νλ κ²μΈκ°? κ³ λ―Όμ€λ¬μ΄ λ¬Έμ λ€.
.. κ·Έλ₯ Dictionary<Type, Dictionary<int, Artifact>>
λ₯Ό μΈκΉ?
μκ°ν΄λ³΄λ©΄ λ€μ΄μΊμ€ν
μ νμνλ€κ³ ν΄μ μ±λ₯μ΄λ ꡬνμ ν¬κ² λ¬Έμ κ° λ κ² κ°μ§ μλ€.
κ·Έλ κ² μμ νκ³ κ΅¬νν΄λ³΄μ.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
private readonly Dictionary<Type, Dictionary<int, Artifact>> dataDics = new();
public void DeleteArtifact(Artifact artifact)
{
Type type = artifact.GetType();
Dictionary<int, Artifact> dic = dataDics[type];
string assetName = $"Q_{artifact.ID}_{artifact.Name}";
string path = AssetDatabase.GenerateUniqueAssetPath($"{QUEST_DIRECTORY_PATH}{assetName}.asset");
dic.Remove(artifact.ID);
AssetDatabase.DeleteAsset(AssetDatabase.GetAssetPath(artifact));
UpdateGrid();
}
// ...
πΏ μμ μΆκ°
μ΄μ μμ μΆκ° κΈ°λ₯μ ꡬνν΄λ³Έλ€.
κ·Όλ°.. μκ°ν΄λ³΄λ©΄ μμ μΆκ°λ₯Ό μν λ³λμ UIκ° μμ΄λ λ κ² κ°λ€.
κΈ°λ³Έ μμ
μ μΆκ°νλ λ²νΌ νλλ§ κ°λ¨ν λ§λ€κ³ ,
μμ
μ 보λ₯Ό λνλ΄λ νλ©΄μμ λλ‘λ°μ€ λ±μΌλ‘ 곧λ°λ‘ μμ κ°λ₯νλλ‘ νλ©΄ λμ§ μμκΉ?
κ·Έλ¬κΈ° μν΄μλ μμ
μ μ 보λ₯Ό μμ κ°λ₯νλλ‘ UIλ₯Ό μΆκ°ν΄μΌνλ€.
.. μ΄μ°¨νΌ UIλ λ§λ€μ΄μΌνλ€.
(λͺ μκ° λ€β¦)
.. PropertyField
μ μ‘΄μ¬λ₯Ό λͺ¨λ₯΄κ³ μ§μ νλ‘νΌν° μ 보λ₯Ό κ°μ Έμμ λ§λ€κ³ μμλ€.
μ΄μ¬ν λ§λ€μλλ° κ·Έλ₯ μ§μ°κΈ° μμ¬μμ λ¨κ²¨λ³Έλ€.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
public PropertyBlock(Artifact artifact, PropertyInfo propertyInfo)
{
this.artifact = artifact;
this.propertyInfo = propertyInfo;
PropertyName = new Label(propertyInfo.Name);
PropertyName.AddToClassList("property-name");
Add(PropertyName);
SetPropertyValue();
AddToClassList("property-block");
}
private void SetPropertyValueWithType<T, U>() where U : BaseField<T>, new()
{
T value = (T)propertyInfo.GetValue(artifact);
PropertyValue = new U();
(PropertyValue as U).value = value;
}
private void SetPropertyValue()
{
Type propertyType = propertyInfo.PropertyType;
switch (propertyType)
{
case Type intType when intType == typeof(int):
SetPropertyValueWithType<int, IntegerField>();
break;
case Type stringType when stringType == typeof(string):
SetPropertyValueWithType<string, TextField>();
bool isDescription = propertyInfo.Name == nameof(Artifact.Description);
if (isDescription)
{
(PropertyValue as TextField).multiline = true;
(PropertyValue as TextField).style.minHeight = 100;
}
PropertyValue.RegisterCallback<ChangeEvent<string>>(evt =>
{
propertyInfo.SetValue(artifact, evt.newValue);
});
break;
// ...
}
}
// ...
PropertyField
λ₯Ό μ΄μ©ν μ½λλ μλμ κ°λ€.
μ΄λ, PropertyField
λ₯Ό μ΄κΈ°ννλ λΆλΆμμ λ§μ΄ μ λ₯Ό λ¨Ήμλ€.
λλ μ€ν¬λ¦½ν°λΈ μ€λΈμ νΈμ λͺ¨λ νλλ₯Ό [field: SerializeField] public int ID { get; set; }
κ°μ΄ μλꡬν νλ‘νΌν°λ‘ λ§λ€μλλ°,
CurArtifact.GetType().GetFields()
λ‘λ νλκ° κ²μλμ§ μμκ³ ,
serializedObject.FindProperty()
μμλ PropertyInfo.Name
μΌλ‘λ νλλ₯Ό μ°Ύμ μ μμλ€.
νλ μ΄λ¦μ΄ λλ체 λκΉ νκ³ sharplab.ioμμ IL μ½λλ₯Ό νμΈν΄λ³΄κ³ λμμΌ, νλ‘νΌν°μ νλ μ΄λ¦μ΄ <νλ‘νΌν°μ΄λ¦>k__BackingField
κ°μ λͺ¨μμΌλ‘ μκΈ΄λ€λ κ²μ μμλ€.
μ΄λ κ² κ΅¬νν΄λ λλ 건μ§λ μ λͺ¨λ₯΄κ² μ§λ§, μ΄μ¨λ μ λμνκΈ°μ κ·Έλλ‘ μ¨λ³΄κΈ°λ‘ νλ€.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
public void UpdateUI()
{
SerializedObject serializedObject = new(CurArtifact);
// CurArtifactμ λͺ¨λ νλ‘νΌν°λ₯Ό 리νλ μ
μΌλ‘ κ°μ Έμ€κΈ°
List<PropertyInfo> propertyInfos = CurArtifact.GetType()
.GetProperties()
.OrderBy(
p =>
{
var attribute = p.GetCustomAttribute(typeof(PropertyOrderAttribute));
if (attribute == null)
return int.MaxValue;
else
return ((PropertyOrderAttribute)attribute).Order;
}).ToList();
// CurArtifactμ λͺ¨λ νλ‘νΌν°λ₯Ό PropertyBlockμΌλ‘ λ§λ€μ΄μ artifactContentμ μΆκ°
artifactContent.Clear();
foreach (PropertyInfo propertyInfo in propertyInfos)
{
if (propertyInfo.Name == "name" || propertyInfo.Name == "hideFlags")
continue;
// HACK : μλμΌλ‘ μμ±λλ νλ‘νΌν°μ νλμ μ΄λ¦ = <νλ‘νΌν°μ΄λ¦>k__BackingField
PropertyField propertyField = new (serializedObject.FindProperty($"<{propertyInfo.Name}>k__BackingField"));
propertyField.Bind(serializedObject);
artifactContent.Add(propertyField);
}
}
μ λμνλ€.
κ·Όλ° μ΄λ΄κ±°λ©΄ κ·Έλ₯ μΈμ€νν°λ₯Ό νμ₯μν€λ©΄ λλκ±° μλμλ?
.. μΌλ¨ κ·Έλ₯ μ°μ
μ΄μ΄μ κ°λ¨ν μμ μΆκ° λ²νΌλ λ§λ€μ΄λ³Έλ€.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
private void UpdateGrid()
{
// ...
Button addButton = new()
{
text = "+",
};
addButton.AddToClassList("slot-icons");
addButton.RegisterCallback<ClickEvent>(ev =>
{
AddArtifact(MArtifactDetail.CurArtifact.GetType());
});
grid.Add(addButton);
Repaint();
}
λ¨μν λ§μ§λ§ λ²νΌμΌλ‘ μΆκ°νλ€.
π μ¬λ¬ μμ
μ§κΈμ QuestData
μ€ν¬λ¦½ν°λΈ μ€λΈμ νΈλ§ λΆλ¬μ€κ³ μλ€.
UIλ₯Ό κ°λ¨ν μμ ν ν, λ€λ₯Έ νμ
μ μ€ν¬λ¦½ν°λΈ μ€λΈμ νΈλ λΆλ¬μ€μ.
νμ¬ UIλ‘ νμν νμ
μ CurType
μΌλ‘ μ μνκ³ ,
κ·Έ CurType
μ μνλ νμ
μΌλ‘ μ€μ νλ λ²νΌλ€μ΄ λͺ¨μΈ λ©λ΄λ₯Ό λ§λ€μλ€.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
private Type CurType { get; set; } = typeof(QuestData);
public void CreateGUI()
{
// ...
VisualElement menu = rootVisualElement.Q<VisualElement>(name: "Menu");
foreach (Type type in dataDics.Keys)
{
Button button = new()
{
text = type.Name,
};
button.clicked += () =>
{
CurType = type;
UpdateGrid();
MArtifactDetail.UpdateCurArtifact(dataDics[CurType].Values.First());
};
menu.Add(button);
}
// ...
}