2020-07-24 00:32:53 +00:00
use crate ::prelude ::{ Children , LocalTransform , Parent , PreviousParent } ;
use bevy_ecs ::{ Commands , CommandsInternal , Component , DynamicBundle , Entity , WorldWriter } ;
use smallvec ::SmallVec ;
pub struct InsertChildren {
parent : Entity ,
children : SmallVec < [ Entity ; 8 ] > ,
index : usize ,
}
impl WorldWriter for InsertChildren {
fn write ( self : Box < Self > , world : & mut bevy_ecs ::World ) {
for child in self . children . iter ( ) {
world
. insert (
* child ,
(
Parent ( self . parent ) ,
2020-07-24 01:26:08 +00:00
PreviousParent ( Some ( self . parent ) ) ,
2020-07-24 00:32:53 +00:00
LocalTransform ::default ( ) ,
) ,
)
. unwrap ( ) ;
}
{
let mut added = false ;
if let Ok ( mut children ) = world . get_mut ::< Children > ( self . parent ) {
children . insert_from_slice ( self . index , & self . children ) ;
added = true ;
}
// NOTE: ideally this is just an else statement, but currently that _incorrectly_ fails borrow-checking
if ! added {
world
. insert_one ( self . parent , Children ( SmallVec ::from ( self . children ) ) )
. unwrap ( ) ;
}
}
}
}
pub struct PushChildren {
parent : Entity ,
children : SmallVec < [ Entity ; 8 ] > ,
}
2020-07-10 04:18:35 +00:00
pub struct ChildBuilder < ' a > {
commands : & ' a mut CommandsInternal ,
2020-07-24 00:32:53 +00:00
push_children : PushChildren ,
}
impl WorldWriter for PushChildren {
fn write ( self : Box < Self > , world : & mut bevy_ecs ::World ) {
for child in self . children . iter ( ) {
world
. insert (
* child ,
(
Parent ( self . parent ) ,
2020-07-24 01:26:08 +00:00
PreviousParent ( Some ( self . parent ) ) ,
2020-07-24 00:32:53 +00:00
LocalTransform ::default ( ) ,
) ,
)
. unwrap ( ) ;
}
{
let mut added = false ;
if let Ok ( mut children ) = world . get_mut ::< Children > ( self . parent ) {
children . extend ( self . children . iter ( ) . cloned ( ) ) ;
added = true ;
}
// NOTE: ideally this is just an else statement, but currently that _incorrectly_ fails borrow-checking
if ! added {
world
. insert_one ( self . parent , Children ( SmallVec ::from ( self . children ) ) )
. unwrap ( ) ;
}
}
}
2020-07-10 04:18:35 +00:00
}
impl < ' a > ChildBuilder < ' a > {
pub fn spawn ( & mut self , components : impl DynamicBundle + Send + Sync + 'static ) -> & mut Self {
self . spawn_as_entity ( Entity ::new ( ) , components )
}
pub fn spawn_as_entity (
& mut self ,
entity : Entity ,
components : impl DynamicBundle + Send + Sync + 'static ,
) -> & mut Self {
2020-07-24 00:32:53 +00:00
self . commands . spawn_as_entity ( entity , components ) ;
self . push_children . children . push ( entity ) ;
2020-07-10 04:18:35 +00:00
self
}
pub fn with_bundle (
& mut self ,
components : impl DynamicBundle + Send + Sync + 'static ,
) -> & mut Self {
self . commands . with_bundle ( components ) ;
self
}
pub fn with ( & mut self , component : impl Component ) -> & mut Self {
self . commands . with ( component ) ;
self
}
2020-07-24 00:32:53 +00:00
pub fn for_current_entity ( & mut self , mut func : impl FnMut ( Entity ) ) -> & mut Self {
let current_entity = self
. commands
. current_entity
. expect ( " The 'current entity' is not set. You should spawn an entity first. " ) ;
func ( current_entity ) ;
self
}
2020-07-10 04:18:35 +00:00
}
pub trait BuildChildren {
2020-07-24 01:26:08 +00:00
fn with_children ( & mut self , f : impl FnMut ( & mut ChildBuilder ) ) -> & mut Self ;
2020-07-24 00:32:53 +00:00
fn push_children ( & mut self , parent : Entity , children : & [ Entity ] ) -> & mut Self ;
fn insert_children ( & mut self , parent : Entity , index : usize , children : & [ Entity ] ) -> & mut Self ;
2020-07-10 04:18:35 +00:00
}
impl BuildChildren for Commands {
2020-07-19 00:03:37 +00:00
fn with_children ( & mut self , mut parent : impl FnMut ( & mut ChildBuilder ) ) -> & mut Self {
2020-07-10 04:18:35 +00:00
{
let mut commands = self . commands . lock ( ) . unwrap ( ) ;
let current_entity = commands . current_entity . expect ( " Cannot add children because the 'current entity' is not set. You should spawn an entity first. " ) ;
2020-07-24 00:32:53 +00:00
commands . current_entity = None ;
let push_children = {
let mut builder = ChildBuilder {
commands : & mut commands ,
push_children : PushChildren {
children : SmallVec ::default ( ) ,
parent : current_entity ,
} ,
} ;
parent ( & mut builder ) ;
builder . push_children
2020-07-10 04:18:35 +00:00
} ;
2020-07-24 00:32:53 +00:00
commands . current_entity = Some ( current_entity ) ;
commands . write_world ( push_children ) ;
}
self
}
fn push_children ( & mut self , parent : Entity , children : & [ Entity ] ) -> & mut Self {
{
let mut commands = self . commands . lock ( ) . unwrap ( ) ;
commands . write_world ( PushChildren {
children : SmallVec ::from ( children ) ,
parent ,
} ) ;
}
self
}
fn insert_children ( & mut self , parent : Entity , index : usize , children : & [ Entity ] ) -> & mut Self {
{
let mut commands = self . commands . lock ( ) . unwrap ( ) ;
commands . write_world ( InsertChildren {
children : SmallVec ::from ( children ) ,
index ,
parent ,
} ) ;
2020-07-10 04:18:35 +00:00
}
self
}
}
impl < ' a > BuildChildren for ChildBuilder < ' a > {
fn with_children ( & mut self , mut spawn_children : impl FnMut ( & mut ChildBuilder ) ) -> & mut Self {
2020-07-24 00:32:53 +00:00
let current_entity = self . commands . current_entity . expect ( " Cannot add children because the 'current entity' is not set. You should spawn an entity first. " ) ;
2020-07-10 04:18:35 +00:00
self . commands . current_entity = None ;
2020-07-24 00:32:53 +00:00
let push_children = {
let mut builder = ChildBuilder {
commands : self . commands ,
push_children : PushChildren {
children : SmallVec ::default ( ) ,
parent : current_entity ,
} ,
} ;
spawn_children ( & mut builder ) ;
builder . push_children
} ;
2020-07-10 04:18:35 +00:00
2020-07-24 00:32:53 +00:00
self . commands . current_entity = Some ( current_entity ) ;
self . commands . write_world ( push_children ) ;
self
}
2020-07-10 04:18:35 +00:00
2020-07-24 00:32:53 +00:00
fn push_children ( & mut self , parent : Entity , children : & [ Entity ] ) -> & mut Self {
self . commands . write_world ( PushChildren {
children : SmallVec ::from ( children ) ,
parent ,
} ) ;
self
}
fn insert_children ( & mut self , parent : Entity , index : usize , children : & [ Entity ] ) -> & mut Self {
self . commands . write_world ( InsertChildren {
children : SmallVec ::from ( children ) ,
index ,
parent ,
} ) ;
2020-07-10 04:18:35 +00:00
self
}
}
2020-07-24 00:32:53 +00:00
#[ cfg(test) ]
mod tests {
use super ::BuildChildren ;
use crate ::prelude ::{ Children , LocalTransform , Parent , PreviousParent } ;
use bevy_ecs ::{ Commands , Entity , Resources , World } ;
use smallvec ::{ smallvec , SmallVec } ;
#[ test ]
fn build_children ( ) {
let mut world = World ::default ( ) ;
let mut resources = Resources ::default ( ) ;
let mut commands = Commands ::default ( ) ;
let mut parent = None ;
let mut child1 = None ;
let mut child2 = None ;
commands
. spawn ( ( 1 , ) )
. for_current_entity ( | e | parent = Some ( e ) )
. with_children ( | parent | {
parent
. spawn ( ( 2 , ) )
. for_current_entity ( | e | child1 = Some ( e ) )
. spawn ( ( 3 , ) )
. for_current_entity ( | e | child2 = Some ( e ) ) ;
} ) ;
commands . apply ( & mut world , & mut resources ) ;
let parent = parent . expect ( " parent should exist " ) ;
let child1 = child1 . expect ( " child1 should exist " ) ;
let child2 = child2 . expect ( " child2 should exist " ) ;
let expected_children : SmallVec < [ Entity ; 8 ] > = smallvec! [ child1 , child2 ] ;
assert_eq! (
world . get ::< Children > ( parent ) . unwrap ( ) . 0. clone ( ) ,
expected_children
) ;
assert_eq! ( * world . get ::< Parent > ( child1 ) . unwrap ( ) , Parent ( parent ) ) ;
assert_eq! ( * world . get ::< Parent > ( child2 ) . unwrap ( ) , Parent ( parent ) ) ;
assert_eq! (
* world . get ::< PreviousParent > ( child1 ) . unwrap ( ) ,
2020-07-24 01:26:08 +00:00
PreviousParent ( Some ( parent ) )
2020-07-24 00:32:53 +00:00
) ;
assert_eq! (
* world . get ::< PreviousParent > ( child2 ) . unwrap ( ) ,
2020-07-24 01:26:08 +00:00
PreviousParent ( Some ( parent ) )
2020-07-24 00:32:53 +00:00
) ;
assert! ( world . get ::< LocalTransform > ( child1 ) . is_ok ( ) ) ;
assert! ( world . get ::< LocalTransform > ( child2 ) . is_ok ( ) ) ;
}
#[ test ]
fn push_and_insert_children ( ) {
let mut world = World ::default ( ) ;
let mut resources = Resources ::default ( ) ;
let mut commands = Commands ::default ( ) ;
let entities = world
. spawn_batch ( vec! [ ( 1 , ) , ( 2 , ) , ( 3 , ) , ( 4 , ) , ( 5 , ) ] )
. collect ::< Vec < Entity > > ( ) ;
commands . push_children ( entities [ 0 ] , & entities [ 1 .. 3 ] ) ;
commands . apply ( & mut world , & mut resources ) ;
let parent = entities [ 0 ] ;
let child1 = entities [ 1 ] ;
let child2 = entities [ 2 ] ;
let child3 = entities [ 3 ] ;
let child4 = entities [ 4 ] ;
let expected_children : SmallVec < [ Entity ; 8 ] > = smallvec! [ child1 , child2 ] ;
assert_eq! (
world . get ::< Children > ( parent ) . unwrap ( ) . 0. clone ( ) ,
expected_children
) ;
assert_eq! ( * world . get ::< Parent > ( child1 ) . unwrap ( ) , Parent ( parent ) ) ;
assert_eq! ( * world . get ::< Parent > ( child2 ) . unwrap ( ) , Parent ( parent ) ) ;
assert_eq! (
* world . get ::< PreviousParent > ( child1 ) . unwrap ( ) ,
2020-07-24 01:26:08 +00:00
PreviousParent ( Some ( parent ) )
2020-07-24 00:32:53 +00:00
) ;
assert_eq! (
* world . get ::< PreviousParent > ( child2 ) . unwrap ( ) ,
2020-07-24 01:26:08 +00:00
PreviousParent ( Some ( parent ) )
2020-07-24 00:32:53 +00:00
) ;
assert! ( world . get ::< LocalTransform > ( child1 ) . is_ok ( ) ) ;
assert! ( world . get ::< LocalTransform > ( child2 ) . is_ok ( ) ) ;
commands . insert_children ( parent , 1 , & entities [ 3 .. ] ) ;
commands . apply ( & mut world , & mut resources ) ;
let expected_children : SmallVec < [ Entity ; 8 ] > = smallvec! [ child1 , child3 , child4 , child2 ] ;
assert_eq! (
world . get ::< Children > ( parent ) . unwrap ( ) . 0. clone ( ) ,
expected_children
) ;
assert_eq! ( * world . get ::< Parent > ( child3 ) . unwrap ( ) , Parent ( parent ) ) ;
assert_eq! ( * world . get ::< Parent > ( child4 ) . unwrap ( ) , Parent ( parent ) ) ;
assert_eq! (
* world . get ::< PreviousParent > ( child3 ) . unwrap ( ) ,
2020-07-24 01:26:08 +00:00
PreviousParent ( Some ( parent ) )
2020-07-24 00:32:53 +00:00
) ;
assert_eq! (
* world . get ::< PreviousParent > ( child4 ) . unwrap ( ) ,
2020-07-24 01:26:08 +00:00
PreviousParent ( Some ( parent ) )
2020-07-24 00:32:53 +00:00
) ;
assert! ( world . get ::< LocalTransform > ( child3 ) . is_ok ( ) ) ;
assert! ( world . get ::< LocalTransform > ( child4 ) . is_ok ( ) ) ;
}
}