use rbx_types::Ref; use rbx_dom_weak::WeakDom; use super::vector3::Vector3; // LMAO look at this function! fn dom(lua:&mlua::Lua,mut f:impl FnMut(&mut WeakDom)->mlua::Result)->mlua::Result{ let mut dom=lua.app_data_mut::<&'static mut WeakDom>().ok_or(mlua::Error::runtime("DataModel missing"))?; f(&mut *dom) } fn coerce_float32(value:&mlua::Value)->Option{ match value{ &mlua::Value::Integer(i)=>Some(i as f32), &mlua::Value::Number(f)=>Some(f as f32), _=>None, } } trait Referent{ fn referent(&self)->Ref; fn get<'a>(&self,dom:&'a WeakDom)->mlua::Result<&'a rbx_dom_weak::Instance>{ dom.get_by_ref(self.referent()).ok_or(mlua::Error::runtime("Instance missing")) } fn get_mut<'a>(&self,dom:&'a mut WeakDom)->mlua::Result<&'a mut rbx_dom_weak::Instance>{ dom.get_by_ref_mut(self.referent()).ok_or(mlua::Error::runtime("Instance missing")) } } pub struct Instance{ referent:Ref, } impl Referent for Instance{ fn referent(&self)->Ref{ self.referent } } impl From for Instance{ fn from(value:crate::script::Script)->Self{ Self{referent:value.script} } } impl Instance{ pub const fn new(referent:Ref)->Self{ Self{referent} } fn add_fields<'lua,T:Referent,F:mlua::UserDataFields<'lua,T>>(fields:&mut F){ fields.add_field_method_get("Parent",|lua,this|{ dom(lua,|dom|{ let instance=this.get(dom)?; Ok(Instance::new(instance.parent())) }) }); fields.add_field_method_set("Parent",|lua,this,val:Self|{ dom(lua,|dom|{ dom.transfer_within(this.referent(),val.referent); Ok(()) }) }); fields.add_field_method_get("Name",|lua,this|{ dom(lua,|dom|{ let instance=this.get(dom)?; Ok(instance.name.clone()) }) }); fields.add_field_method_set("Name",|lua,this,val:String|{ dom(lua,move|dom|{ let instance=this.get_mut(dom)?; //Why does this need to be cloned? instance.name=val.clone(); Ok(()) }) }); } fn add_methods<'lua,T:Referent,M:mlua::UserDataMethods<'lua,T>>(methods:&mut M){ methods.add_method("GetChildren",|lua,this,_:()| dom(lua,|dom|{ let instance=this.get(dom)?; let children:Vec<_>=instance .children() .iter() .copied() .map(Instance::new) .collect(); Ok(children) }) ); methods.add_method("GetDescendants",|lua,this,_:()| dom(lua,|dom|{ let children:Vec<_>=dom .descendants_of(this.referent()) .map(|instance| Instance::new(instance.referent()) ) .collect(); Ok(children) }) ); methods.add_method("IsA",|lua,this,classname:mlua::String| dom(lua,|dom|{ let instance=this.get(dom)?; Ok(crate::context::class_is_a(instance.class.as_str(),classname.to_str()?)) }) ); methods.add_meta_function(mlua::MetaMethod::NewIndex,|lua,(this,index,value):(Self,mlua::String,mlua::Value)| dom(lua,|dom|{ //println!("__newindex t={this:?} i={index:?} v={value:?}"); let instance=this.get_mut(dom)?; let index_str=index.to_str()?; let db=rbx_reflection_database::get(); let class=db.classes.get(instance.class.as_str()).ok_or(mlua::Error::runtime("Class missing"))?; let property=db.find_default_property(class,index_str).ok_or(mlua::Error::runtime(format!("Property '{index_str}' missing on class '{}'",class.name)))?; match property{ rbx_types::Variant::Vector3(_)=>{ let typed_value:Vector3=value.as_userdata().ok_or(mlua::Error::runtime("Expected Userdata"))?.take()?; instance.properties.insert(index_str.to_owned(),rbx_types::Variant::Vector3(typed_value.into())); }, rbx_types::Variant::Float32(_)=>{ let typed_value:f32=coerce_float32(&value).ok_or(mlua::Error::runtime("Expected f32"))?; instance.properties.insert(index_str.to_owned(),rbx_types::Variant::Float32(typed_value)); }, other=>println!("Unimplemented property type: {other:?}"), } Ok(()) }) ); } } impl mlua::UserData for Instance{ fn add_fields<'lua,F:mlua::UserDataFields<'lua,Self>>(fields:&mut F){ Instance::add_fields(fields); } fn add_methods<'lua,M:mlua::UserDataMethods<'lua,Self>>(methods:&mut M){ Instance::add_methods(methods); } } impl<'lua> mlua::FromLua<'lua> for Instance{ fn from_lua(value:mlua::prelude::LuaValue<'lua>,_lua:&'lua mlua::prelude::Lua)->mlua::prelude::LuaResult{ match value{ mlua::Value::UserData(ud)=>ud.take(), other=>Err(mlua::Error::runtime(format!("Expected Instance got {:?}",other))), } } } pub struct DataModel{ referent:Ref, } impl Referent for DataModel{ fn referent(&self)->Ref{ self.referent } } impl DataModel{ fn add_methods<'lua,T,M:mlua::UserDataMethods<'lua,T>>(methods:&mut M){ methods.add_method("GetService",|lua,this,service:String| dom(lua,|dom|{ match service.as_str(){ //"Lighting"=>Ok(Lighting::new()), other=>Err::<(),_>(mlua::Error::runtime("Service '{other}' not supported")), } }) ); } } impl mlua::UserData for DataModel{ fn add_fields<'lua,F:mlua::UserDataFields<'lua,Self>>(fields:&mut F){ Instance::add_fields(fields); //DataModel::add_fields(fields); } fn add_methods<'lua,M:mlua::UserDataMethods<'lua,Self>>(methods:&mut M){ Instance::add_methods(methods); DataModel::add_methods(methods); } } pub struct Lighting{ referent:Ref, } impl Referent for Lighting{ fn referent(&self)->Ref{ self.referent } } impl mlua::UserData for Lighting{ fn add_fields<'lua,F:mlua::UserDataFields<'lua,Self>>(fields:&mut F){ Instance::add_fields(fields); } fn add_methods<'lua,M:mlua::UserDataMethods<'lua,Self>>(methods:&mut M){ Instance::add_methods(methods); } }