-
Notifications
You must be signed in to change notification settings - Fork 396
Expand file tree
/
Copy pathReClassNetProject.cs
More file actions
135 lines (105 loc) · 2.97 KB
/
ReClassNetProject.cs
File metadata and controls
135 lines (105 loc) · 2.97 KB
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
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
using System;
using System.Collections.Generic;
using System.Diagnostics.Contracts;
using System.Linq;
using ReClassNET.Extensions;
using ReClassNET.Nodes;
using ReClassNET.Util;
namespace ReClassNET
{
public class ReClassNetProject : IDisposable
{
public delegate void ClassesChangedEvent(ClassNode sender);
public event ClassesChangedEvent ClassAdded;
public event ClassesChangedEvent ClassRemoved;
private readonly List<ClassNode> classes = new List<ClassNode>();
public IEnumerable<ClassNode> Classes => classes;
public string Path { get; set; }
/// <summary>
/// Key-Value map with custom data for plugins to store project related data.
/// The preferred key format is {Plugin Name}_{Key Name}.
/// </summary>
public Dictionary<string, string> CustomData { get; } = new Dictionary<string, string>();
public void Dispose()
{
Clear();
ClassAdded = null;
ClassRemoved = null;
}
public void AddClass(ClassNode node)
{
Contract.Requires(node != null);
classes.Add(node);
node.NodesChanged += NodesChanged_Handler;
ClassAdded?.Invoke(node);
}
public bool ContainsClass(NodeUuid uuid)
{
Contract.Requires(uuid != null);
return classes.Any(c => c.Uuid.Equals(uuid));
}
public ClassNode GetClassByUuid(NodeUuid uuid)
{
Contract.Requires(uuid != null);
return classes.First(c => c.Uuid.Equals(uuid));
}
private void NodesChanged_Handler(BaseNode sender)
{
classes.ForEach(c => c.UpdateOffsets());
}
public void Clear()
{
var temp = classes.ToList();
classes.Clear();
foreach (var node in temp)
{
node.NodesChanged -= NodesChanged_Handler;
ClassRemoved?.Invoke(node);
}
}
private IEnumerable<ClassNode> GetClassReferences(ClassNode node)
{
Contract.Requires(node != null);
return classes.Where(c => c != node).Where(c => c.Descendants().Any(n => (n as BaseReferenceNode)?.InnerNode == node));
}
public void Remove(ClassNode node)
{
Contract.Requires(node != null);
var references = GetClassReferences(node).ToList();
if (references.Any())
{
throw new ClassReferencedException(references);
}
if (classes.Remove(node))
{
node.NodesChanged -= NodesChanged_Handler;
ClassRemoved?.Invoke(node);
}
}
public void RemoveUnusedClasses()
{
var toRemove = classes
.Except(classes.Where(x => GetClassReferences(x).Any())) // check for references
.Where(c => c.Nodes.All(n => n is BaseHexNode)) // check if only hex nodes are present
.ToList();
foreach (var node in toRemove)
{
if (classes.Remove(node))
{
ClassRemoved?.Invoke(node);
}
}
}
}
public class ClassReferencedException : Exception
{
public IEnumerable<ClassNode> References { get; }
public ClassReferencedException(IEnumerable<ClassNode> references)
: base("This class has references.")
{
Contract.Requires(references != null);
Contract.Requires(Contract.ForAll(references, c => c != null));
References = references;
}
}
}