2024-09-21 01:21:00 +00:00
use rbx_types ::Ref ;
2024-09-19 01:47:38 +00:00
use rbx_dom_weak ::WeakDom ;
use super ::vector3 ::Vector3 ;
// LMAO look at this function!
fn dom < T > ( lua :& mlua ::Lua , mut f :impl FnMut ( & mut WeakDom ) ->mlua ::Result < T > ) ->mlua ::Result < T > {
2024-09-20 20:51:17 +00:00
let mut dom = lua . app_data_mut ::< & 'static mut WeakDom > ( ) . ok_or ( mlua ::Error ::runtime ( " DataModel missing " ) ) ? ;
f ( & mut * dom )
2024-09-19 01:47:38 +00:00
}
2024-09-21 00:54:16 +00:00
fn coerce_float32 ( value :& mlua ::Value ) ->Option < f32 > {
2024-09-19 03:16:14 +00:00
match value {
& mlua ::Value ::Integer ( i ) = > Some ( i as f32 ) ,
& mlua ::Value ::Number ( f ) = > Some ( f as f32 ) ,
_ = > None ,
}
}
2024-09-21 20:26:20 +00:00
fn get_full_name ( dom :& rbx_dom_weak ::WeakDom , instance :& rbx_dom_weak ::Instance ) ->String {
let mut full_name = instance . name . clone ( ) ;
let mut pref = instance . parent ( ) ;
while let Some ( parent ) = dom . get_by_ref ( pref ) {
full_name . insert ( 0 , '.' ) ;
full_name . insert_str ( 0 , parent . name . as_str ( ) ) ;
pref = parent . parent ( ) ;
}
full_name
}
2024-09-21 01:21:00 +00:00
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 " ) )
}
}
2024-09-21 20:26:20 +00:00
macro_rules ! class {
( $class :ident ) = > {
pub struct $class {
referent :Ref ,
}
impl $class {
pub const fn new ( referent :Ref ) ->Self {
Self { referent }
}
}
impl Referent for $class {
fn referent ( & self ) ->Ref {
self . referent
}
}
impl < ' lua > mlua ::FromLua < ' lua > for $class {
fn from_lua ( value :mlua ::Value < ' lua > , _lua :& ' lua mlua ::Lua ) ->mlua ::Result < Self > {
match value {
mlua ::Value ::UserData ( ud ) = > ud . take ( ) ,
other = > Err ( mlua ::Error ::runtime ( format! ( " Expected {} got {:?} " , stringify! ( $class ) , other ) ) ) ,
}
}
}
} ;
}
macro_rules ! class_composition {
( $class :ident , ( $( $superclass :ident ) , * ) ) = > {
impl mlua ::UserData for $class {
fn add_fields < ' lua , F :mlua ::UserDataFields < ' lua , Self > > ( fields :& mut F ) {
$(
$superclass ::composition_add_fields ( fields ) ;
) *
}
fn add_methods < ' lua , M :mlua ::UserDataMethods < ' lua , Self > > ( methods :& mut M ) {
$(
$superclass ::composition_add_methods ( methods ) ;
) *
}
}
} ;
2024-09-21 01:21:00 +00:00
}
2024-09-21 20:26:20 +00:00
class! ( Instance ) ;
class_composition! ( Instance , ( Instance ) ) ;
2024-09-21 01:21:00 +00:00
impl Instance {
2024-09-21 20:26:20 +00:00
fn composition_add_fields < ' lua , T :Referent , F :mlua ::UserDataFields < ' lua , T > > ( fields :& mut F ) {
2024-09-19 01:47:38 +00:00
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 | {
2024-09-21 01:21:00 +00:00
dom . transfer_within ( this . referent ( ) , val . referent ) ;
2024-09-19 01:47:38 +00:00
Ok ( ( ) )
} )
} ) ;
2024-09-21 01:01:33 +00:00
fields . add_field_method_get ( " Name " , | lua , this | {
dom ( lua , | dom | {
let instance = this . get ( dom ) ? ;
Ok ( instance . name . clone ( ) )
} )
} ) ;
2024-09-21 21:21:23 +00:00
fields . add_field_method_set ( " Name " , | lua , this , val :mlua ::String | {
dom ( lua , | dom | {
2024-09-21 01:01:33 +00:00
let instance = this . get_mut ( dom ) ? ;
//Why does this need to be cloned?
2024-09-21 21:21:23 +00:00
instance . name = val . to_str ( ) ? . to_owned ( ) ;
2024-09-21 01:01:33 +00:00
Ok ( ( ) )
} )
} ) ;
2024-09-19 01:47:38 +00:00
}
2024-09-21 20:26:20 +00:00
fn composition_add_methods < ' lua , T :Referent , M :mlua ::UserDataMethods < ' lua , T > > ( methods :& mut M ) {
2024-09-19 03:16:14 +00:00
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 )
} )
) ;
2024-09-19 03:16:28 +00:00
methods . add_method ( " GetDescendants " , | lua , this , _ :( ) |
dom ( lua , | dom | {
let children :Vec < _ > = dom
2024-09-21 01:21:00 +00:00
. descendants_of ( this . referent ( ) )
2024-09-19 03:16:28 +00:00
. map ( | instance |
Instance ::new ( instance . referent ( ) )
)
. collect ( ) ;
Ok ( children )
} )
) ;
2024-09-19 03:16:14 +00:00
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 ( ) ? ) )
} )
) ;
2024-09-19 01:47:38 +00:00
methods . add_meta_function ( mlua ::MetaMethod ::NewIndex , | lua , ( this , index , value ) :( Self , mlua ::String , mlua ::Value ) |
dom ( lua , | dom | {
2024-09-21 01:21:00 +00:00
//println!("__newindex t={this:?} i={index:?} v={value:?}");
2024-09-19 01:47:38 +00:00
let instance = this . get_mut ( dom ) ? ;
let index_str = index . to_str ( ) ? ;
2024-09-21 00:54:43 +00:00
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 ) ) ) ? ;
2024-09-19 01:47:38 +00:00
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 ( ) ) ) ;
} ,
2024-09-19 03:16:14 +00:00
rbx_types ::Variant ::Float32 ( _ ) = > {
2024-09-21 00:54:16 +00:00
let typed_value :f32 = coerce_float32 ( & value ) . ok_or ( mlua ::Error ::runtime ( " Expected f32 " ) ) ? ;
2024-09-19 03:16:14 +00:00
instance . properties . insert ( index_str . to_owned ( ) , rbx_types ::Variant ::Float32 ( typed_value ) ) ;
2024-09-19 01:47:38 +00:00
} ,
2024-09-21 21:25:56 +00:00
other = > return Err ( mlua ::Error ::runtime ( format! ( " Unimplemented property type: {other:?} " ) ) ) ,
2024-09-19 01:47:38 +00:00
}
Ok ( ( ) )
} )
) ;
}
}
2024-09-21 01:21:00 +00:00
2024-09-21 20:26:20 +00:00
class! ( DataModel ) ;
class_composition! ( DataModel , ( Instance , DataModel ) ) ;
2024-09-21 01:21:00 +00:00
impl DataModel {
2024-09-21 20:26:20 +00:00
fn composition_add_fields < ' lua , T :Referent , F :mlua ::UserDataFields < ' lua , T > > ( fields :& mut F ) {
}
fn composition_add_methods < ' lua , T , M :mlua ::UserDataMethods < ' lua , T > > ( methods :& mut M ) {
2024-09-21 01:21:00 +00:00
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 " ) ) ,
}
} )
) ;
}
}
2024-09-21 20:26:20 +00:00
class! ( Lighting ) ;
class_composition! ( Lighting , ( Instance ) ) ;
#[ derive(Debug) ]
pub enum GetScriptError {
NoScript ,
NoSource ,
2024-09-21 01:21:00 +00:00
}
2024-09-21 20:26:20 +00:00
class! ( Script ) ;
class_composition! ( Script , ( Instance ) ) ;
impl Script {
pub fn get_name_source ( & self , context :& crate ::context ::Context ) ->Result < ( String , String ) , GetScriptError > {
let instance = context . dom . get_by_ref ( self . referent ) . ok_or ( GetScriptError ::NoScript ) ? ;
let source = match instance . properties . get ( " Source " ) . ok_or ( GetScriptError ::NoSource ) ? {
rbx_dom_weak ::types ::Variant ::String ( s ) = > s . clone ( ) ,
_ = > Err ( GetScriptError ::NoSource ) ? ,
} ;
Ok ( ( get_full_name ( & context . dom , instance ) , source ) )
2024-09-21 01:21:00 +00:00
}
}