2019-01-26 00:39:37 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
|
|
|
|
|
namespace Roblox
|
|
|
|
|
{
|
2019-01-29 09:50:55 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Describes an object in Roblox's Parent->Child hierarchy.
|
|
|
|
|
/// Instances can have sets of properties loaded from *.rbxl/*.rbxm files.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public class Instance
|
2019-01-26 00:39:37 +00:00
|
|
|
|
{
|
2019-01-29 09:50:55 +00:00
|
|
|
|
public string ClassName = "";
|
|
|
|
|
public List<Property> Properties = new List<Property>();
|
|
|
|
|
|
|
|
|
|
private List<Instance> Children = new List<Instance>();
|
|
|
|
|
private Instance rawParent;
|
|
|
|
|
|
|
|
|
|
public string Name => ReadProperty("Name", ClassName);
|
|
|
|
|
public override string ToString() => Name;
|
|
|
|
|
|
|
|
|
|
/// <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)
|
2019-01-26 00:39:37 +00:00
|
|
|
|
{
|
2019-01-29 09:50:55 +00:00
|
|
|
|
while (descendant != null)
|
2019-01-26 00:39:37 +00:00
|
|
|
|
{
|
2019-01-29 09:50:55 +00:00
|
|
|
|
if (descendant == this)
|
2019-01-26 00:39:37 +00:00
|
|
|
|
return true;
|
|
|
|
|
|
2019-01-29 09:50:55 +00:00
|
|
|
|
descendant = descendant.Parent;
|
2019-01-26 00:39:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-29 09:50:55 +00:00
|
|
|
|
/// <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)
|
2019-01-26 00:39:37 +00:00
|
|
|
|
{
|
2019-01-29 09:50:55 +00:00
|
|
|
|
return ancestor.IsAncestorOf(this);
|
2019-01-26 00:39:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-01-29 09:50:55 +00:00
|
|
|
|
/// <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 value is set to itself.<para/>
|
|
|
|
|
/// - The value is a descendant of the Instance.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public Instance Parent
|
2019-01-26 00:39:37 +00:00
|
|
|
|
{
|
2019-01-29 09:50:55 +00:00
|
|
|
|
get { return rawParent; }
|
2019-01-26 00:39:37 +00:00
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
if (IsAncestorOf(value))
|
|
|
|
|
throw new Exception("Parent would result in circular reference.");
|
|
|
|
|
|
|
|
|
|
if (Parent == this)
|
|
|
|
|
throw new Exception("Attempt to set parent to self");
|
|
|
|
|
|
2019-01-29 09:50:55 +00:00
|
|
|
|
if (rawParent != null)
|
|
|
|
|
rawParent.Children.Remove(this);
|
2019-01-26 00:39:37 +00:00
|
|
|
|
|
2019-01-29 09:50:55 +00:00
|
|
|
|
value.Children.Add(this);
|
|
|
|
|
rawParent = value;
|
2019-01-26 00:39:37 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-29 09:50:55 +00:00
|
|
|
|
public IEnumerable<Instance> GetChildren()
|
2019-01-26 00:39:37 +00:00
|
|
|
|
{
|
2019-01-29 09:50:55 +00:00
|
|
|
|
var current = Children.ToArray();
|
|
|
|
|
return current.AsEnumerable();
|
2019-01-26 00:39:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-01-29 09:50:55 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Returns the first 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>
|
|
|
|
|
/// <returns>The instance that was found with this name, or null.</returns>
|
|
|
|
|
public Instance FindFirstChild(string name)
|
|
|
|
|
{
|
|
|
|
|
Instance result = null;
|
|
|
|
|
|
|
|
|
|
var query = Children.Where(child => child.Name == name);
|
|
|
|
|
if (query.Count() > 0)
|
|
|
|
|
result = query.First();
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Looks for a property with the specified property name, and returns it as an object.
|
|
|
|
|
/// <para/>The resulting value may be null if the property is not serialized.
|
|
|
|
|
/// <para/>You can use the templated ReadProperty overload to fetch it as a specific type with a default value provided.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="propertyName">The name of the property to be fetched from this Instance.</param>
|
|
|
|
|
/// <returns>An object reference to the value of the specified property, if it exists.</returns>
|
|
|
|
|
///
|
2019-01-26 00:39:37 +00:00
|
|
|
|
public object ReadProperty(string propertyName)
|
|
|
|
|
{
|
2019-01-29 09:50:55 +00:00
|
|
|
|
Property property = null;
|
|
|
|
|
|
|
|
|
|
if (query.Count() > 0)
|
|
|
|
|
property = query.First();
|
|
|
|
|
|
|
|
|
|
return (property != null ? property.Value : null);
|
|
|
|
|
}
|
2019-01-26 00:39:37 +00:00
|
|
|
|
|
2019-01-29 09:50:55 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Looks for a property with the specified property name, and returns it as the specified type.<para/>
|
|
|
|
|
/// If it cannot be converted, the provided nullFallback value will be returned instead.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <typeparam name="T">The value type to convert to when finding the specified property name.</typeparam>
|
|
|
|
|
/// <param name="propertyName">The name of the property to be fetched from this Instance.</param>
|
|
|
|
|
/// <param name="nullFallback">A fallback value to be returned if casting to T fails, or the property is not found.</param>
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
public T ReadProperty<T>(string propertyName, T nullFallback)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
object result = ReadProperty(propertyName);
|
|
|
|
|
return (T)result;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
return nullFallback;
|
|
|
|
|
}
|
2019-01-26 00:39:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-01-29 09:50:55 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Looks for a property with the specified property name. If found, it will try to set the value of the referenced outValue to its value.<para/>
|
|
|
|
|
/// Returns true if the property was found and its value was casted to the referenced outValue. If it returns false, the outValue has not been set.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <typeparam name="T">The value type to convert to when finding the specified property name.</typeparam>
|
|
|
|
|
/// <param name="propertyName">The name of the property to be fetched from this Instance.</param>
|
|
|
|
|
/// <param name="outValue">The value to write to if the property can be casted to T correctly.</param>
|
|
|
|
|
public bool TryReadProperty<T>(string propertyName, ref T outValue)
|
2019-01-26 00:39:37 +00:00
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
object result = ReadProperty(propertyName);
|
2019-01-29 09:50:55 +00:00
|
|
|
|
outValue = (T)result;
|
2019-01-26 00:39:37 +00:00
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
catch
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-29 09:50:55 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Adds a property by reference to this Instance's property list.
|
|
|
|
|
/// This is used during the file loading procedure.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="prop">A reference to the property that will be added.</param>
|
|
|
|
|
public void AddProperty(ref Property prop)
|
2019-01-26 00:39:37 +00:00
|
|
|
|
{
|
2019-01-29 09:50:55 +00:00
|
|
|
|
Properties.Add(prop);
|
2019-01-26 00:39:37 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2019-01-29 09:50:55 +00:00
|
|
|
|
}
|