use std::borrow::{Borrow, Cow}; use rbx_dom_weak::types::{Variant as DomValue, VariantType as DomType}; use rbx_reflection::{ClassTag, DataType}; pub(crate) struct PropertyInfo { pub enum_name: Option>, pub enum_default: Option, pub value_type: Option, pub value_default: Option<&'static DomValue>, } /** Finds the info of a property of the given class. This will also check superclasses if the property was not directly found for the given class. Returns `None` if the class or property does not exist. */ pub(crate) fn find_property_info( instance_class: impl AsRef, property_name: impl AsRef, ) -> Option { let db = rbx_reflection_database::get(); let instance_class = instance_class.as_ref(); let property_name = property_name.as_ref(); let mut current_class = Cow::Borrowed(instance_class); while let Some(class) = db.classes.get(current_class.as_ref()) { if let Some(prop_definition) = class.properties.get(property_name) { // We found a property, we should map it to a property // info containing name/type and default property value let prop_default = class.default_properties.get(property_name); return Some(match &prop_definition.data_type { DataType::Enum(enum_name) => PropertyInfo { enum_name: Some(Cow::Borrowed(enum_name)), enum_default: prop_default.and_then(|default| match default { DomValue::Enum(enum_default) => Some(enum_default.to_u32()), _ => None, }), value_type: None, value_default: None, }, DataType::Value(value_type) => PropertyInfo { enum_name: None, enum_default: None, value_type: Some(*value_type), value_default: prop_default, }, _ => PropertyInfo { enum_name: None, enum_default: None, value_type: None, value_default: None, }, }); } else if let Some(sup) = &class.superclass { // No property found, we should look at the superclass current_class = Cow::Borrowed(sup) } else { break; } } None } /** Checks if an instance class exists in the reflection database. */ pub fn class_exists(class_name: impl AsRef) -> bool { let db = rbx_reflection_database::get(); db.classes.contains_key(class_name.as_ref()) } /** Checks if an instance class matches a given class or superclass, similar to [Instance::IsA](https://create.roblox.com/docs/reference/engine/classes/Instance#IsA) from the Roblox standard library. Note that this function may return `None` if it encounters a class or superclass that does not exist in the currently known class reflection database. */ pub fn class_is_a(instance_class: impl AsRef, class_name: impl AsRef) -> Option { let mut instance_class = instance_class.as_ref(); let class_name = class_name.as_ref(); if class_name == "Instance" || instance_class == class_name { Some(true) } else { let db = rbx_reflection_database::get(); while instance_class != class_name { let class_descriptor = db.classes.get(instance_class)?; if let Some(sup) = &class_descriptor.superclass { instance_class = sup.borrow(); } else { return Some(false); } } Some(true) } } /** Checks if an instance class is a service. This is separate from [`class_is_a`] since services do not share a common base class, and are instead determined through reflection tags. Note that this function may return `None` if it encounters a class or superclass that does not exist in the currently known class reflection database. */ pub fn class_is_a_service(instance_class: impl AsRef) -> Option { let mut instance_class = instance_class.as_ref(); let db = rbx_reflection_database::get(); loop { let class_descriptor = db.classes.get(instance_class)?; if class_descriptor.tags.contains(&ClassTag::Service) { return Some(true); } else if let Some(sup) = &class_descriptor.superclass { instance_class = sup.borrow(); } else { break; } } Some(false) } #[cfg(test)] mod tests { use super::*; #[test] fn is_a_class_valid() { assert_eq!(class_is_a("Part", "Part"), Some(true)); assert_eq!(class_is_a("Part", "BasePart"), Some(true)); assert_eq!(class_is_a("Part", "PVInstance"), Some(true)); assert_eq!(class_is_a("Part", "Instance"), Some(true)); assert_eq!(class_is_a("Workspace", "Workspace"), Some(true)); assert_eq!(class_is_a("Workspace", "Model"), Some(true)); assert_eq!(class_is_a("Workspace", "Instance"), Some(true)); } #[test] fn is_a_class_invalid() { assert_eq!(class_is_a("Part", "part"), Some(false)); assert_eq!(class_is_a("Part", "Base-Part"), Some(false)); assert_eq!(class_is_a("Part", "Model"), Some(false)); assert_eq!(class_is_a("Part", "Paart"), Some(false)); assert_eq!(class_is_a("Workspace", "Service"), Some(false)); assert_eq!(class_is_a("Workspace", "."), Some(false)); assert_eq!(class_is_a("Workspace", ""), Some(false)); } #[test] fn is_a_service_valid() { assert_eq!(class_is_a_service("Workspace"), Some(true)); assert_eq!(class_is_a_service("PhysicsService"), Some(true)); assert_eq!(class_is_a_service("ReplicatedFirst"), Some(true)); assert_eq!(class_is_a_service("CSGDictionaryService"), Some(true)); } #[test] fn is_a_service_invalid() { assert_eq!(class_is_a_service("Camera"), Some(false)); assert_eq!(class_is_a_service("Terrain"), Some(false)); assert_eq!(class_is_a_service("Work-space"), None); assert_eq!(class_is_a_service("CSG Dictionary Service"), None); } }