Updated mod_scripting

This commit is contained in:
Silas Bartha 2025-02-23 01:40:01 -05:00
parent 9e5f782eb2
commit 114409322d
3 changed files with 138 additions and 112 deletions

View File

@ -5,14 +5,14 @@ edition = "2021"
[dependencies] [dependencies]
async-channel = "2.3" async-channel = "2.3"
notify = "7.0" notify = "8.0"
tar = "0.4" tar = "0.4"
xz2 = "0.1" xz2 = "0.1"
rust-crypto = "0.2" rust-crypto = "0.2"
multi_key_map = "0.3" multi_key_map = "0.3"
serde = "1.0" serde = "1.0"
rmp-serde = "1.3" rmp-serde = "1.3"
notify-debouncer-full = "0.4" notify-debouncer-full = "0.5"
md5 = "0.7" md5 = "0.7"
aes = "0.8" aes = "0.8"
hex = "0.4" hex = "0.4"
@ -30,7 +30,7 @@ version = "0.2"
features = ["serialize"] features = ["serialize"]
[dependencies.occule] [dependencies.occule]
git = "https://git.exvacuum.dev/occule" git = "https://git.soaos.dev/occule"
tag = "v0.3.1" tag = "v0.3.1"
[dependencies.yarnspinner] [dependencies.yarnspinner]
@ -39,15 +39,15 @@ optional = true
features = ["serde"] features = ["serde"]
[dependencies.bevy_mod_scripting] [dependencies.bevy_mod_scripting]
version = "0.8" version = "0.9"
features = ["lua54", "lua_script_api"] features = ["lua54", "bevy_bindings"]
[dependencies.bevy_basic_interaction] [dependencies.bevy_basic_interaction]
git = "https://git.exvacuum.dev/bevy_basic_interaction" git = "https://git.soaos.dev/bevy_basic_interaction"
tag = "v0.2.0" tag = "v0.2.0"
[dependencies.strum] [dependencies.strum]
version = "0.26" version = "0.27"
features = ["derive"] features = ["derive"]
[features] [features]

View File

@ -6,8 +6,8 @@ use std::{ffi::OsStr, path::PathBuf};
use actor::ActorPlugin; use actor::ActorPlugin;
use bevy::{ecs::system::IntoObserverSystem, prelude::*}; use bevy::{ecs::system::IntoObserverSystem, prelude::*};
use bevy_mod_scripting::core::{AddScriptApiProvider, AddScriptHost, AddScriptHostHandler, ScriptingPlugin}; use bevy_mod_scripting::core::ScriptingPlugin;
use bevy_mod_scripting::lua::LuaScriptHost; use bevy_mod_scripting::lua::LuaScriptingPlugin;
use cache::DirworldCache; use cache::DirworldCache;
use events::{DirworldChangeRoot, DirworldEnterRoom, DirworldLeaveRoom, DirworldNavigationComplete, DirworldSpawn}; use events::{DirworldChangeRoot, DirworldEnterRoom, DirworldLeaveRoom, DirworldNavigationComplete, DirworldSpawn};
use occule::Codec; use occule::Codec;
@ -69,7 +69,7 @@ impl Plugin for DirworldPlugin {
custom_function_registration: Some(yarnspinner_api::setup_yarnspinner_functions), custom_function_registration: Some(yarnspinner_api::setup_yarnspinner_functions),
}, },
DirworldPreloadPlugin, DirworldPreloadPlugin,
ScriptingPlugin, LuaScriptingPlugin::default(),
)) ))
.add_systems(Startup, watcher::setup) .add_systems(Startup, watcher::setup)
.add_systems( .add_systems(
@ -80,9 +80,6 @@ impl Plugin for DirworldPlugin {
yarnspinner_api::process_commands, yarnspinner_api::process_commands,
), ),
) )
.add_script_host::<LuaScriptHost<()>>(PostUpdate)
.add_script_handler::<LuaScriptHost<()>, 0, 0>(PostUpdate)
.add_api_provider::<LuaScriptHost<()>>(Box::new(lua_api::ConditionalAPI))
.add_systems(PostUpdate, watcher::update) .add_systems(PostUpdate, watcher::update)
.init_resource::<DirworldRootDir>() .init_resource::<DirworldRootDir>()
.init_resource::<DirworldCache>() .init_resource::<DirworldCache>()

View File

@ -1,37 +1,40 @@
use std::{str::FromStr, sync::Mutex}; use std::str::FromStr;
use bevy::prelude::*; use bevy::prelude::*;
use bevy_mod_scripting::api::providers::bevy_reflect::LuaVec3; use bevy_mod_scripting::core::{
use bevy_mod_scripting::{api::providers::bevy_ecs::LuaEntity, lua::tealr::mlu::mlua::Error as LuaError}; bindings::function::{
use bevy_mod_scripting::lua::LuaEvent; from::Val,
use bevy_mod_scripting::prelude::*; namespace::{GlobalNamespace, NamespaceBuilder},
script_function::FunctionCallContext,
},
callback_labels,
error::InteropError,
event::ScriptCallbackEvent,
};
use uuid::Uuid; use uuid::Uuid;
use crate::{components::DirworldEntity, conditionals::Condition}; use crate::{components::DirworldEntity, conditionals::Condition};
pub fn trigger_update(mut w: PriorityEventWriter<LuaEvent<()>>) { callback_labels!(OnUpdate => "on_update");
let event = LuaEvent::<()> {
args: (), pub fn trigger_update(mut w: EventWriter<ScriptCallbackEvent>) {
hook_name: "on_update".into(), w.send(ScriptCallbackEvent::new_for_all(OnUpdate, vec![]));
recipients: Recipients::All,
};
w.send(event, 0);
} }
// ACTUAL API STUFF BELOW THIS POINT {{{ // ACTUAL API STUFF BELOW THIS POINT {{{
macro_rules! register_fns { macro_rules! register_fns {
($runtime:expr, $($function:expr),+) => { ($world:expr, $($function:expr),+) => {
{ {
let ctx = $runtime.get_mut().unwrap(); NamespaceBuilder::<GlobalNamespace>::new_unregistered($world)
$(ctx.globals().set(stringify!($function).to_string(), ctx.create_function($function).unwrap()).unwrap();)+ $(.register(stringify!($function), $function))+;
} }
}; };
} }
pub fn register(api: &mut Mutex<Lua>) { pub fn register(world: &mut World) {
register_fns!( register_fns!(
api, world,
translate, translate,
rotate, rotate,
get_dirworld_id, get_dirworld_id,
@ -45,65 +48,82 @@ pub fn register(api: &mut Mutex<Lua>) {
) )
} }
fn translate(ctx: &Lua, (entity, translation): (LuaEntity, LuaVec3)) -> Result<(), LuaError> { fn translate(ctx: FunctionCallContext, entity: Val<Entity>, translation: Val<Vec3>) {
let world = ctx.get_world()?; let world = ctx.world().unwrap();
let mut world = world.write(); world.with_global_access(|world| {
if let Some(mut transform) = world.entity_mut(entity.inner().unwrap()).get_mut::<Transform>() { if let Some(mut transform) = world.entity_mut(*entity).get_mut::<Transform>() {
transform.translation += translation.inner().unwrap(); transform.translation += *translation;
} }
Ok(()) });
} }
fn rotate(ctx: &Lua, (entity, axis, angle): (LuaEntity, LuaVec3, f32)) -> Result<(), LuaError> { fn rotate(ctx: FunctionCallContext, entity: Val<Entity>, axis: Val<Vec3>, angle: f32) {
let world = ctx.get_world()?; let world = ctx.world().unwrap();
let mut world = world.write(); world.with_global_access(|world| {
if let Some(mut transform) = world.entity_mut(entity.inner().unwrap()).get_mut::<Transform>() { if let Some(mut transform) = world.entity_mut(*entity).get_mut::<Transform>() {
transform.rotation *= Quat::from_axis_angle(axis.inner().unwrap(), angle); transform.rotation *= Quat::from_axis_angle(*axis, angle);
} }
Ok(()) });
} }
fn get_dirworld_id(ctx: &Lua, (entity,): (LuaEntity,)) -> Result<String, LuaError> { fn get_dirworld_id(ctx: FunctionCallContext, entity: Val<Entity>) -> Result<String, InteropError> {
let world = ctx.get_world()?; let world = ctx.world()?;
let world = world.read(); let mut result = Ok(Default::default());
if let Some(dirworld_entity) = world.entity(entity.inner().unwrap()).get::<DirworldEntity>() { world.with_global_access(|world| {
dirworld_entity.payload.as_ref().map(|p| p.id.to_string()).ok_or(LuaError::runtime("Failed to get entity id from payload")) result = if let Some(dirworld_entity) = world.entity(*entity).get::<DirworldEntity>() {
Ok(dirworld_entity
.payload
.as_ref()
.map(|p| p.id.to_string())
.unwrap())
} else { } else {
Err(LuaError::runtime("Entity missing DirworldEntity component")) Err(InteropError::missing_entity(*entity))
} };
});
result
} }
// Conditionals // Conditionals
pub struct ConditionalAPI; fn condition_true(ctx: FunctionCallContext) -> Result<bool, InteropError> {
let world = ctx.world()?;
let mut result = Ok(Default::default());
world.with_global_access(|world| {
result = Ok(Condition::True.evaluate(world));
});
result
}
impl APIProvider for ConditionalAPI { fn condition_ancestor_of(
type APITarget = Mutex<Lua>; ctx: FunctionCallContext,
ancestor: String,
type ScriptContext = Mutex<Lua>; descendant: String,
) -> Result<bool, InteropError> {
type DocTarget = LuaDocFragment; let Ok(ancestor) = Uuid::from_str(&ancestor) else {
warn!("Provided ancestor is not a valid UUID");
fn attach_api( return Ok(false);
&mut self, };
api: &mut Self::APITarget, let Ok(descendant) = Uuid::from_str(&descendant) else {
) -> Result<(), bevy_mod_scripting::prelude::ScriptError> { warn!("Provided descendant is not a valid UUID");
register(api); return Ok(false);
Ok(()) };
let world = ctx.world()?;
let mut result = Ok(Default::default());
world.with_global_access(|world| {
result = Ok(Condition::AncestorOf {
ancestor,
descendant,
} }
.evaluate(world));
});
result
} }
fn condition_descendant_of(
ctx: FunctionCallContext,
fn condition_true(ctx: &Lua, _: ()) -> Result<bool, LuaError> { descendant: String,
let world = ctx.get_world()?; ancestor: String,
let mut world = world.write(); ) -> Result<bool, InteropError> {
Ok(Condition::True.evaluate(&mut world))
}
fn condition_ancestor_of(ctx: &Lua, (ancestor, descendant): (String, String)) -> Result<bool, LuaError> {
let world = ctx.get_world()?;
let mut world = world.write();
let Ok(ancestor) = Uuid::from_str(&ancestor) else { let Ok(ancestor) = Uuid::from_str(&ancestor) else {
warn!("Provided ancestor is not a valid UUID"); warn!("Provided ancestor is not a valid UUID");
return Ok(false); return Ok(false);
@ -112,32 +132,23 @@ fn condition_ancestor_of(ctx: &Lua, (ancestor, descendant): (String, String)) ->
warn!("Provided descendant is not a valid UUID"); warn!("Provided descendant is not a valid UUID");
return Ok(false); return Ok(false);
}; };
Ok(Condition::AncestorOf { let world = ctx.world()?;
let mut result = Ok(Default::default());
world.with_global_access(|world| {
result = Ok(Condition::DescendantOf {
ancestor, ancestor,
descendant, descendant,
}.evaluate(&mut world)) }
.evaluate(world));
});
result
} }
fn condition_descendant_of(ctx: &Lua, (descendant, ancestor): (String, String)) -> Result<bool, LuaError> { fn condition_parent_of(
let world = ctx.get_world()?; ctx: FunctionCallContext,
let mut world = world.write(); parent: String,
let Ok(ancestor) = Uuid::from_str(&ancestor) else { child: String,
warn!("Provided ancestor is not a valid UUID"); ) -> Result<bool, InteropError> {
return Ok(false);
};
let Ok(descendant) = Uuid::from_str(&descendant) else {
warn!("Provided descendant is not a valid UUID");
return Ok(false);
};
Ok(Condition::DescendantOf {
ancestor,
descendant,
}.evaluate(&mut world))
}
fn condition_parent_of(ctx: &Lua, (parent, child): (String, String)) -> Result<bool, LuaError> {
let world = ctx.get_world()?;
let mut world = world.write();
let Ok(parent) = Uuid::from_str(&parent) else { let Ok(parent) = Uuid::from_str(&parent) else {
warn!("Provided ancestor is not a valid UUID"); warn!("Provided ancestor is not a valid UUID");
return Ok(false); return Ok(false);
@ -146,15 +157,19 @@ fn condition_parent_of(ctx: &Lua, (parent, child): (String, String)) -> Result<b
warn!("Provided descendant is not a valid UUID"); warn!("Provided descendant is not a valid UUID");
return Ok(false); return Ok(false);
}; };
Ok(Condition::ParentOf { let world = ctx.world()?;
parent, let mut result = Ok(Default::default());
child, world.with_global_access(|world| {
}.evaluate(&mut world)) result = Ok(Condition::ParentOf { parent, child }.evaluate(world));
});
result
} }
fn condition_child_of(ctx: &Lua, (child, parent): (String, String)) -> Result<bool, LuaError> { fn condition_child_of(
let world = ctx.get_world()?; ctx: FunctionCallContext,
let mut world = world.write(); child: String,
parent: String,
) -> Result<bool, InteropError> {
let Ok(parent) = Uuid::from_str(&parent) else { let Ok(parent) = Uuid::from_str(&parent) else {
warn!("Provided ancestor is not a valid UUID"); warn!("Provided ancestor is not a valid UUID");
return Ok(false); return Ok(false);
@ -163,30 +178,44 @@ fn condition_child_of(ctx: &Lua, (child, parent): (String, String)) -> Result<bo
warn!("Provided descendant is not a valid UUID"); warn!("Provided descendant is not a valid UUID");
return Ok(false); return Ok(false);
}; };
Ok(Condition::ChildOf { let world = ctx.world()?;
parent, let mut result = Ok(Default::default());
child, world.with_global_access(|world| {
}.evaluate(&mut world)) result = Ok(Condition::ChildOf { parent, child }.evaluate(world));
});
result
} }
fn condition_in_room(ctx: &Lua, (room,): (String,)) -> Result<bool, LuaError> { fn condition_in_room(
let world = ctx.get_world()?; ctx: FunctionCallContext,
let mut world = world.write(); room: String,
) -> Result<bool, InteropError> {
let Ok(room) = Uuid::from_str(&room) else { let Ok(room) = Uuid::from_str(&room) else {
warn!("Provided room is not a valid UUID"); warn!("Provided room is not a valid UUID");
return Ok(false); return Ok(false);
}; };
Ok(Condition::InRoom(room).evaluate(&mut world)) let world = ctx.world()?;
let mut result = Ok(Default::default());
world.with_global_access(|world| {
result = Ok(Condition::InRoom(room).evaluate(world));
});
result
} }
fn condition_object_in_room(ctx: &Lua, (object,): (String,)) -> Result<bool, LuaError> { fn condition_object_in_room(
let world = ctx.get_world()?; ctx: FunctionCallContext,
let mut world = world.write(); object: String
) -> Result<bool, InteropError> {
let Ok(object) = Uuid::from_str(&object) else { let Ok(object) = Uuid::from_str(&object) else {
warn!("Provided object is not a valid UUID"); warn!("Provided object is not a valid UUID");
return Ok(false); return Ok(false);
}; };
Ok(Condition::ObjectInRoom(object).evaluate(&mut world)) let world = ctx.world()?;
let mut result = Ok(Default::default());
world.with_global_access(|world| {
result = Ok(Condition::ObjectInRoom(object).evaluate(world));
});
result
} }
// }}} // }}}