using System; using System.Collections.Generic; using System.Reflection; using RobloxFiles.BinaryFormat; using RobloxFiles.BinaryFormat.Chunks; using RobloxFiles.DataTypes; namespace RobloxFiles { public enum PropertyType { Unknown, String, Bool, Int, Float, Double, UDim, UDim2, Ray, Faces, Axes, BrickColor, Color3, Vector2, Vector3, CFrame = 16, Quaternion, Enum, Ref, Vector3int16, NumberSequence, ColorSequence, NumberRange, Rect, PhysicalProperties, Color3uint8, Int64, SharedString, } public class Property { public string Name { get; internal set; } public Instance Instance { get; internal set; } public PropertyType Type { get; internal set; } public string XmlToken { get; internal set; } public byte[] RawBuffer { get; internal set; } internal object RawValue; internal BinaryRobloxFileWriter CurrentWriter; internal static BindingFlags BindingFlags = BindingFlags.Instance | BindingFlags.Public | BindingFlags.FlattenHierarchy | BindingFlags.IgnoreCase; public static IReadOnlyDictionary Types = new Dictionary() { { typeof(Axes), PropertyType.Axes }, { typeof(Faces), PropertyType.Faces }, { typeof(int), PropertyType.Int }, { typeof(bool), PropertyType.Bool }, { typeof(long), PropertyType.Int64 }, { typeof(float), PropertyType.Float }, { typeof(double), PropertyType.Double }, { typeof(string), PropertyType.String }, { typeof(Ray), PropertyType.Ray }, { typeof(Rect), PropertyType.Rect }, { typeof(UDim), PropertyType.UDim }, { typeof(UDim2), PropertyType.UDim2 }, { typeof(CFrame), PropertyType.CFrame }, { typeof(Color3), PropertyType.Color3 }, { typeof(Content), PropertyType.String }, { typeof(Vector2), PropertyType.Vector2 }, { typeof(Vector3), PropertyType.Vector3 }, { typeof(BrickColor), PropertyType.BrickColor }, { typeof(Quaternion), PropertyType.Quaternion }, { typeof(Color3uint8), PropertyType.Color3uint8 }, { typeof(NumberRange), PropertyType.NumberRange }, { typeof(SharedString), PropertyType.SharedString }, { typeof(Vector3int16), PropertyType.Vector3int16 }, { typeof(ColorSequence), PropertyType.ColorSequence }, { typeof(NumberSequence), PropertyType.NumberSequence }, { typeof(ProtectedString), PropertyType.String }, { typeof(PhysicalProperties), PropertyType.PhysicalProperties }, }; private void ImproviseRawBuffer() { if (RawValue is byte[]) { RawBuffer = RawValue as byte[]; return; } else if (RawValue is SharedString) { var sharedString = CastValue(); RawBuffer = sharedString.SharedValue; return; } switch (Type) { case PropertyType.Int: RawBuffer = BitConverter.GetBytes((int)Value); break; case PropertyType.Int64: RawBuffer = BitConverter.GetBytes((long)Value); break; case PropertyType.Bool: RawBuffer = BitConverter.GetBytes((bool)Value); break; case PropertyType.Float: RawBuffer = BitConverter.GetBytes((float)Value); break; case PropertyType.Double: RawBuffer = BitConverter.GetBytes((double)Value); break; // } } private string ImplicitName { get { if (Instance != null) { Type instType = Instance.GetType(); string typeName = instType.Name; if (typeName == Name) { FieldInfo directField = instType.GetField(typeName, BindingFlags.DeclaredOnly); if (directField != null) { var implicitName = Name + '_'; return implicitName; } } } if (Name.Contains(" ")) return Name.Replace(' ', '_'); return Name; } } public object Value { get { if (Instance != null) { if (Name == "Tags") { byte[] data = Instance.SerializedTags; RawValue = data; } else { FieldInfo field = Instance.GetType() .GetField(ImplicitName, BindingFlags); if (field != null) { object value = field.GetValue(Instance); RawValue = value; } else { Console.WriteLine($"RobloxFiles.Property - No defined field for {Instance.ClassName}.{Name}"); } } } return RawValue; } set { if (Instance != null) { if (Name == "Tags" && value is byte[]) { byte[] data = value as byte[]; Instance.SerializedTags = data; } else { FieldInfo field = Instance.GetType() .GetField(ImplicitName, BindingFlags); if (field != null) { Type fieldType = field.FieldType; Type valueType = value?.GetType(); if (fieldType == valueType || value == null) { try { field.SetValue(Instance, value); } catch { Console.WriteLine($"RobloxFiles.Property - Failed to cast value {value} into property {Instance.ClassName}.{Name}"); } } else if (valueType != null) { var typeWrapper = new Type[] { valueType }; MethodInfo implicitCast = fieldType.GetMethod("op_Implicit", typeWrapper); if (implicitCast != null) { var valueWrapper = new object[] { value }; try { object castedValue = implicitCast.Invoke(null, valueWrapper); field.SetValue(Instance, castedValue); } catch { Console.WriteLine($"RobloxFiles.Property - Failed to implicitly cast value {value} into property {Instance.ClassName}.{Name}"); } } } } } } RawValue = value; RawBuffer = null; ImproviseRawBuffer(); } } public bool HasRawBuffer { get { // Improvise what the buffer should be if this is a primitive. if (RawBuffer == null && Value != null) ImproviseRawBuffer(); return (RawBuffer != null); } } public Property(string name = "", PropertyType type = PropertyType.Unknown, Instance instance = null) { Instance = instance; Name = name; Type = type; } public Property(Instance instance, PROP property) { Instance = instance; Name = property.Name; Type = property.Type; } public string GetFullName() { string result = Name; if (Instance != null) result = Instance.GetFullName() + '.' + result; return result; } public override string ToString() { string typeName = Enum.GetName(typeof(PropertyType), Type); string valueLabel = (Value != null ? Value.ToString() : "null"); if (Type == PropertyType.String) valueLabel = '"' + valueLabel + '"'; return string.Join(" ", typeName, Name, '=', valueLabel); } public T CastValue() { object result; if (typeof(T) == typeof(string)) result = Value?.ToString() ?? ""; else if (Value is T) result = (T)Value; else result = default(T); return (T)result; } internal void WriteValue() where T : struct { if (CurrentWriter == null) throw new Exception("Property.CurrentWriter must be set to use WriteValue"); T value = CastValue(); byte[] bytes = BinaryRobloxFileWriter.GetBytes(value); CurrentWriter.Write(bytes); } } }