Roblox-File-Format/Tree/Instance.cs

604 lines
20 KiB
C#
Raw Normal View History

using System;
using System.Collections.Generic;
2020-08-17 05:33:59 +00:00
using System.Diagnostics.Contracts;
using System.IO;
using System.Linq;
using System.Reflection;
using System.Text;
namespace RobloxFiles
{
/// <summary>
/// Describes an object in Roblox's DataModel hierarchy.
/// Instances can have sets of properties loaded from *.rbxl/*.rbxm files.
/// </summary>
public class Instance
{
public Instance()
{
Name = ClassName;
}
/// <summary>The ClassName of this Instance.</summary>
public string ClassName => GetType().Name;
/// <summary>Internal list of properties that are under this Instance.</summary>
2019-05-19 04:44:51 +00:00
private Dictionary<string, Property> props = new Dictionary<string, Property>();
/// <summary>A list of properties that are defined under this Instance.</summary>
2019-05-19 04:44:51 +00:00
public IReadOnlyDictionary<string, Property> Properties => props;
/// <summary>The raw list of children for this Instance.</summary>
2019-12-23 03:23:06 +00:00
internal HashSet<Instance> Children = new HashSet<Instance>();
/// <summary>The raw value of the Instance's parent.</summary>
private Instance RawParent;
/// <summary>The name of this Instance.</summary>
public string Name;
/// <summary>Indicates whether this Instance should be serialized.</summary>
public bool Archivable = true;
/// <summary>The name of this Instance, if a Name property is defined.</summary>
public override string ToString() => Name;
/// <summary>A unique identifier for this instance when being serialized.</summary>
public string Referent { get; internal set; }
2019-05-19 04:44:51 +00:00
/// <summary>Indicates whether the parent of this object is locked.</summary>
public bool ParentLocked { get; internal set; }
/// <summary>Indicates whether this Instance is a Service.</summary>
public bool IsService { get; internal set; }
/// <summary>A list of CollectionService tags assigned to this Instance.</summary>
public List<string> Tags { get; } = new List<string>();
/// <summary>The attributes defined for this Instance.</summary>
public Attributes Attributes { get; private set; }
2019-10-30 23:33:00 +00:00
/// <summary>The internal serialized data of this Instance's attributes</summary>
internal byte[] AttributesSerialize
{
get
{
2020-08-17 05:33:59 +00:00
return Attributes?.Serialize() ?? Array.Empty<byte>();
}
set
{
MemoryStream data = new MemoryStream(value);
Attributes = new Attributes(data);
}
}
/// <summary>
/// Internal format of the Instance's CollectionService tags.
/// Property objects will look to this member for serializing the Tags property.
/// </summary>
internal byte[] SerializedTags
{
get
{
string fullString = string.Join("\0", Tags.ToArray());
byte[] buffer = fullString.ToCharArray()
.Select(ch => (byte)ch)
.ToArray();
return buffer;
}
set
{
int length = value.Length;
List<byte> buffer = new List<byte>();
Tags.Clear();
for (int i = 0; i < length; i++)
{
byte id = value[i];
if (id != 0)
buffer.Add(id);
if (id == 0 || i == (length - 1))
{
byte[] data = buffer.ToArray();
buffer.Clear();
string tag = Encoding.UTF8.GetString(data);
Tags.Add(tag);
}
}
}
}
/// <summary>Returns true if this Instance is an ancestor to the provided Instance.</summary>
/// <param name="descendant">The instance whose descendance will be tested against this Instance.</param>
public bool IsAncestorOf(Instance descendant)
{
while (descendant != null)
{
if (descendant == this)
return true;
descendant = descendant.Parent;
}
return false;
}
/// <summary>Returns true if this Instance is a descendant of the provided Instance.</summary>
/// <param name="ancestor">The instance whose ancestry will be tested against this Instance.</param>
public bool IsDescendantOf(Instance ancestor)
{
2020-08-17 05:33:59 +00:00
Contract.Requires(ancestor != null);
return ancestor.IsAncestorOf(this);
}
/// <summary>
/// Returns true if the provided instance inherits from the provided instance type.
/// </summary>
[Obsolete("Use the `is` operator instead.")]
public bool IsA<T>() where T : Instance
2019-05-19 04:44:51 +00:00
{
Type myType = GetType();
Type classType = typeof(T);
return classType.IsAssignableFrom(myType);
2019-05-19 04:44:51 +00:00
}
/// <summary>
/// Attempts to cast this Instance to an inherited class of type '<typeparamref name="T"/>'.
/// Returns null if the instance cannot be casted to the provided type.
/// </summary>
/// <typeparam name="T">The type of Instance to cast to.</typeparam>
/// <returns>The instance as the type '<typeparamref name="T"/>' if it can be converted, or null.</returns>
public T Cast<T>() where T : Instance
{
T result = null;
if (this is T)
result = this as T;
return result;
}
/// <summary>
/// The parent of this Instance, or null if the instance is the root of a tree.<para/>
/// Setting the value of this property will throw an exception if:<para/>
/// - The parent is currently locked.<para/>
/// - The value is set to itself.<para/>
/// - The value is a descendant of the Instance.
/// </summary>
public Instance Parent
{
get
{
return RawParent;
}
set
{
2019-05-19 04:44:51 +00:00
if (ParentLocked)
throw new Exception("The Parent property of this instance is locked.");
if (IsAncestorOf(value))
throw new Exception("Parent would result in circular reference.");
if (Parent == this)
throw new Exception("Attempt to set parent to self.");
RawParent?.Children.Remove(this);
value?.Children.Add(this);
RawParent = value;
}
}
/// <summary>
/// Returns an array containing all the children of this Instance.
/// </summary>
public Instance[] GetChildren()
{
return Children.ToArray();
}
/// <summary>
/// Returns an array containing all the children of this Instance, whose type is '<typeparamref name="T"/>'.
/// </summary>
public T[] GetChildrenOfType<T>() where T : Instance
{
T[] ofType = GetChildren()
.Where(child => child is T)
.Cast<T>()
.ToArray();
return ofType;
}
/// <summary>
/// Returns an array containing all the descendants of this Instance.
/// </summary>
public Instance[] GetDescendants()
{
var results = new List<Instance>();
foreach (Instance child in Children)
{
// Add this child to the results.
results.Add(child);
// Add its descendants to the results.
Instance[] descendants = child.GetDescendants();
results.AddRange(descendants);
}
return results.ToArray();
}
/// <summary>
/// Returns an array containing all the descendants of this Instance, whose type is '<typeparamref name="T"/>'.
/// </summary>
public T[] GetDescendantsOfType<T>() where T : Instance
{
T[] ofType = GetDescendants()
.Where(desc => desc is T)
.Cast<T>()
.ToArray();
return ofType;
}
/// <summary>
/// Returns the first child of this Instance whose Name is the provided string name.
/// If the instance is not found, this returns null.
/// </summary>
/// <param name="name">The Name of the Instance to find.</param>
/// <param name="recursive">Indicates if we should search descendants as well.</param>
public T FindFirstChild<T>(string name, bool recursive = false) where T : Instance
{
T result = null;
var query = Children
.Where(child => child is T)
.Where(child => name == child.Name)
.Cast<T>();
2020-08-17 05:33:59 +00:00
if (query.Any())
{
result = query.First();
}
else if (recursive)
{
foreach (Instance child in Children)
{
T found = child.FindFirstChild<T>(name, true);
if (found != null)
{
result = found;
break;
}
}
}
return result;
}
/// <summary>
/// Returns the first child of this Instance whose Name is the provided string name.
/// If the instance is not found, this returns null.
/// </summary>
/// <param name="name">The Name of the Instance to find.</param>
/// <param name="recursive">Indicates if we should search descendants as well.</param>
public Instance FindFirstChild(string name, bool recursive = false)
{
return FindFirstChild<Instance>(name, recursive);
}
/// <summary>
/// Returns the first ancestor of this Instance whose Name is the provided string name.
/// If the instance is not found, this returns null.
/// </summary>
/// <param name="name">The Name of the Instance to find.</param>
public T FindFirstAncestor<T>(string name) where T : Instance
{
Instance ancestor = Parent;
while (ancestor != null)
{
if (ancestor is T && ancestor.Name == name)
2020-08-17 05:33:59 +00:00
return ancestor as T;
ancestor = ancestor.Parent;
}
return null;
}
/// <summary>
/// Returns the first ancestor of this Instance whose Name is the provided string name.
/// If the instance is not found, this returns null.
/// </summary>
/// <param name="name">The Name of the Instance to find.</param>
public Instance FindFirstAncestor(string name)
{
return FindFirstAncestor<Instance>(name);
}
2019-05-19 04:44:51 +00:00
/// <summary>
/// Returns the first ancestor of this Instance whose ClassName is the provided string className.
/// If the instance is not found, this returns null.
/// </summary>
/// <param name="name">The Name of the Instance to find.</param>
public T FindFirstAncestorOfClass<T>() where T : Instance
{
Instance ancestor = Parent;
while (ancestor != null)
{
if (ancestor is T)
2020-08-17 05:33:59 +00:00
return ancestor as T;
ancestor = ancestor.Parent;
}
return null;
}
/// <summary>
/// Returns the first ancestor of this Instance which derives from the provided type <typeparamref name="T"/>.
/// If the instance is not found, this returns null.
/// </summary>
/// <param name="name">The Name of the Instance to find.</param>
public T FindFirstAncestorWhichIsA<T>() where T : Instance
{
T ancestor = null;
Instance check = Parent;
while (check != null)
{
if (check is T)
{
ancestor = check as T;
break;
}
check = check.Parent;
}
return ancestor;
}
/// <summary>
2019-05-19 04:44:51 +00:00
/// Returns the first Instance whose ClassName is the provided string className.
/// If the instance is not found, this returns null.
/// </summary>
/// <param name="className">The ClassName of the Instance to find.</param>
public T FindFirstChildOfClass<T>(bool recursive = false) where T : Instance
{
var query = Children
.Where(child => child is T)
.Cast<T>();
T result = null;
2020-08-17 05:33:59 +00:00
if (query.Any())
{
result = query.First();
}
else if (recursive)
{
foreach (Instance child in Children)
{
T found = child.FindFirstChildOfClass<T>(true);
if (found != null)
{
result = found;
break;
}
}
}
return result;
}
/// <summary>
/// Returns the first child of this Instance which derives from the provided type <typeparamref name="T"/>.
/// If the instance is not found, this returns null.
/// </summary>
/// <param name="recursive">Whether this should search descendants as well.</param>
public T FindFirstChildWhichIsA<T>(bool recursive = false) where T : Instance
{
var query = Children
.Where(child => child is T)
.Cast<T>();
2020-08-17 05:33:59 +00:00
if (query.Any())
return query.First();
if (recursive)
{
foreach (Instance child in Children)
{
T found = child.FindFirstChildWhichIsA<T>(true);
if (found == null)
continue;
return found;
}
}
return null;
}
/// <summary>
2019-10-06 09:08:46 +00:00
/// Returns a string describing the index traversal of this Instance, starting from its root ancestor.
/// </summary>
2019-12-23 03:23:06 +00:00
public string GetFullName(string separator = ".")
{
string fullName = Name;
Instance at = Parent;
while (at != null)
{
2019-12-23 03:23:06 +00:00
fullName = at.Name + separator + fullName;
at = at.Parent;
}
return fullName;
}
/// <summary>
/// Returns a Property object whose name is the provided string name.
2019-05-19 04:44:51 +00:00
/// </summary>
public Property GetProperty(string name)
{
Property result = null;
if (props.ContainsKey(name))
result = props[name];
2019-05-19 04:44:51 +00:00
return result;
}
2019-05-19 04:44:51 +00:00
/// <summary>
/// Adds a property by reference to this Instance's property list.
2019-05-19 04:44:51 +00:00
/// </summary>
/// <param name="prop">A reference to the property that will be added.</param>
internal void AddProperty(ref Property prop)
2019-05-19 04:44:51 +00:00
{
prop.Instance = this;
2019-05-19 04:44:51 +00:00
if (props.ContainsKey(prop.Name))
props.Remove(prop.Name);
2019-05-19 04:44:51 +00:00
props.Add(prop.Name, prop);
}
/// <summary>
/// Removes a property with the provided name if a property with the provided name exists.
/// </summary>
/// <param name="name">The name of the property to be removed.</param>
/// <returns>True if a property with the provided name was removed.</returns>
internal bool RemoveProperty(string name)
{
if (props.ContainsKey(name))
{
Property prop = Properties[name];
prop.Instance = null;
}
return props.Remove(name);
}
/// <summary>
/// Ensures that all serializable properties of this Instance have
/// a registered Property object with the correct PropertyType.
/// </summary>
internal IReadOnlyDictionary<string, Property> RefreshProperties()
{
Type instType = GetType();
FieldInfo[] fields = instType.GetFields(Property.BindingFlags);
foreach (FieldInfo field in fields)
{
string fieldName = field.Name;
Type fieldType = field.FieldType;
if (field.GetCustomAttribute<ObsoleteAttribute>() != null)
continue;
2020-09-13 01:16:19 +00:00
if (fieldName == "Archivable" || fieldName.EndsWith("k__BackingField"))
continue;
else if (fieldName == "Bevel_Roundness")
fieldName = "Bevel Roundness";
2019-07-04 23:26:53 +00:00
PropertyType propType = PropertyType.Unknown;
if (Property.Types.ContainsKey(fieldType))
2019-07-04 23:26:53 +00:00
propType = Property.Types[fieldType];
else if (fieldType.IsEnum)
propType = PropertyType.Enum;
2019-07-04 23:26:53 +00:00
if (propType != PropertyType.Unknown)
{
if (fieldName.EndsWith("_"))
fieldName = instType.Name;
2020-08-18 01:12:24 +00:00
string xmlToken = fieldType.Name;
if (fieldType.IsEnum)
xmlToken = "token";
switch (xmlToken)
{
case "String":
case "Double":
case "Int64":
2020-08-18 01:12:24 +00:00
xmlToken = xmlToken.ToLowerInvariant();
break;
case "Boolean":
xmlToken = "bool";
break;
case "Single":
xmlToken = "float";
break;
case "Int32":
xmlToken = "int";
break;
case "Rect":
xmlToken = "Rect2D";
break;
case "CFrame":
xmlToken = "CoordinateFrame";
break;
default: break;
}
if (!props.ContainsKey(fieldName))
{
var newProp = new Property()
{
Value = field.GetValue(this),
2020-08-18 01:12:24 +00:00
XmlToken = xmlToken,
Name = fieldName,
2019-07-04 23:26:53 +00:00
Type = propType,
Instance = this
};
AddProperty(ref newProp);
}
else
{
Property prop = props[fieldName];
prop.Value = field.GetValue(this);
2020-08-18 01:12:24 +00:00
prop.XmlToken = xmlToken;
2019-07-04 23:26:53 +00:00
prop.Type = propType;
}
2019-05-19 04:44:51 +00:00
}
}
Property tags = GetProperty("Tags");
Property attributes = GetProperty("AttributesSerialize");
if (tags == null)
{
tags = new Property("Tags", PropertyType.String);
AddProperty(ref tags);
}
if (attributes == null)
{
attributes = new Property("AttributesSerialize", PropertyType.String);
AddProperty(ref attributes);
}
return Properties;
}
}
2019-10-06 09:08:46 +00:00
}