mod computed_states; mod freely_mutable_state; mod resources; mod state_set; mod states; mod sub_states; mod transitions; pub use bevy_state_macros::*; pub use computed_states::*; pub use freely_mutable_state::*; pub use resources::*; pub use state_set::*; pub use states::*; pub use sub_states::*; pub use transitions::*; #[cfg(test)] mod tests { use bevy_ecs::event::EventRegistry; use bevy_ecs::prelude::*; use bevy_ecs::schedule::ScheduleLabel; use bevy_state_macros::States; use bevy_state_macros::SubStates; use super::*; use crate as bevy_state; #[derive(States, PartialEq, Eq, Debug, Default, Hash, Clone)] enum SimpleState { #[default] A, B(bool), } #[derive(PartialEq, Eq, Debug, Hash, Clone)] enum TestComputedState { BisTrue, BisFalse, } impl ComputedStates for TestComputedState { type SourceStates = Option; fn compute(sources: Option) -> Option { sources.and_then(|source| match source { SimpleState::A => None, SimpleState::B(value) => Some(if value { Self::BisTrue } else { Self::BisFalse }), }) } } #[test] fn computed_state_with_a_single_source_is_correctly_derived() { let mut world = World::new(); EventRegistry::register_event::>(&mut world); EventRegistry::register_event::>(&mut world); world.init_resource::>(); let mut schedules = Schedules::new(); let mut apply_changes = Schedule::new(StateTransition); TestComputedState::register_computed_state_systems(&mut apply_changes); SimpleState::register_state(&mut apply_changes); schedules.insert(apply_changes); world.insert_resource(schedules); setup_state_transitions_in_world(&mut world, None); world.run_schedule(StateTransition); assert_eq!(world.resource::>().0, SimpleState::A); assert!(!world.contains_resource::>()); world.insert_resource(NextState::Pending(SimpleState::B(true))); world.run_schedule(StateTransition); assert_eq!( world.resource::>().0, SimpleState::B(true) ); assert_eq!( world.resource::>().0, TestComputedState::BisTrue ); world.insert_resource(NextState::Pending(SimpleState::B(false))); world.run_schedule(StateTransition); assert_eq!( world.resource::>().0, SimpleState::B(false) ); assert_eq!( world.resource::>().0, TestComputedState::BisFalse ); world.insert_resource(NextState::Pending(SimpleState::A)); world.run_schedule(StateTransition); assert_eq!(world.resource::>().0, SimpleState::A); assert!(!world.contains_resource::>()); } #[derive(SubStates, PartialEq, Eq, Debug, Default, Hash, Clone)] #[source(SimpleState = SimpleState::B(true))] enum SubState { #[default] One, Two, } #[test] fn sub_state_exists_only_when_allowed_but_can_be_modified_freely() { let mut world = World::new(); EventRegistry::register_event::>(&mut world); EventRegistry::register_event::>(&mut world); world.init_resource::>(); let mut schedules = Schedules::new(); let mut apply_changes = Schedule::new(StateTransition); SubState::register_sub_state_systems(&mut apply_changes); SimpleState::register_state(&mut apply_changes); schedules.insert(apply_changes); world.insert_resource(schedules); setup_state_transitions_in_world(&mut world, None); world.run_schedule(StateTransition); assert_eq!(world.resource::>().0, SimpleState::A); assert!(!world.contains_resource::>()); world.insert_resource(NextState::Pending(SubState::Two)); world.run_schedule(StateTransition); assert_eq!(world.resource::>().0, SimpleState::A); assert!(!world.contains_resource::>()); world.insert_resource(NextState::Pending(SimpleState::B(true))); world.run_schedule(StateTransition); assert_eq!( world.resource::>().0, SimpleState::B(true) ); assert_eq!(world.resource::>().0, SubState::One); world.insert_resource(NextState::Pending(SubState::Two)); world.run_schedule(StateTransition); assert_eq!( world.resource::>().0, SimpleState::B(true) ); assert_eq!(world.resource::>().0, SubState::Two); world.insert_resource(NextState::Pending(SimpleState::B(false))); world.run_schedule(StateTransition); assert_eq!( world.resource::>().0, SimpleState::B(false) ); assert!(!world.contains_resource::>()); } #[derive(SubStates, PartialEq, Eq, Debug, Default, Hash, Clone)] #[source(TestComputedState = TestComputedState::BisTrue)] enum SubStateOfComputed { #[default] One, Two, } #[test] fn substate_of_computed_states_works_appropriately() { let mut world = World::new(); EventRegistry::register_event::>(&mut world); EventRegistry::register_event::>(&mut world); EventRegistry::register_event::>(&mut world); world.init_resource::>(); let mut schedules = Schedules::new(); let mut apply_changes = Schedule::new(StateTransition); TestComputedState::register_computed_state_systems(&mut apply_changes); SubStateOfComputed::register_sub_state_systems(&mut apply_changes); SimpleState::register_state(&mut apply_changes); schedules.insert(apply_changes); world.insert_resource(schedules); setup_state_transitions_in_world(&mut world, None); world.run_schedule(StateTransition); assert_eq!(world.resource::>().0, SimpleState::A); assert!(!world.contains_resource::>()); world.insert_resource(NextState::Pending(SubStateOfComputed::Two)); world.run_schedule(StateTransition); assert_eq!(world.resource::>().0, SimpleState::A); assert!(!world.contains_resource::>()); world.insert_resource(NextState::Pending(SimpleState::B(true))); world.run_schedule(StateTransition); assert_eq!( world.resource::>().0, SimpleState::B(true) ); assert_eq!( world.resource::>().0, SubStateOfComputed::One ); world.insert_resource(NextState::Pending(SubStateOfComputed::Two)); world.run_schedule(StateTransition); assert_eq!( world.resource::>().0, SimpleState::B(true) ); assert_eq!( world.resource::>().0, SubStateOfComputed::Two ); world.insert_resource(NextState::Pending(SimpleState::B(false))); world.run_schedule(StateTransition); assert_eq!( world.resource::>().0, SimpleState::B(false) ); assert!(!world.contains_resource::>()); } #[derive(States, PartialEq, Eq, Debug, Default, Hash, Clone)] struct OtherState { a_flexible_value: &'static str, another_value: u8, } #[derive(PartialEq, Eq, Debug, Hash, Clone)] enum ComplexComputedState { InAAndStrIsBobOrJane, InTrueBAndUsizeAbove8, } impl ComputedStates for ComplexComputedState { type SourceStates = (Option, Option); fn compute(sources: (Option, Option)) -> Option { match sources { (Some(simple), Some(complex)) => { if simple == SimpleState::A && (complex.a_flexible_value == "bob" || complex.a_flexible_value == "jane") { Some(ComplexComputedState::InAAndStrIsBobOrJane) } else if simple == SimpleState::B(true) && complex.another_value > 8 { Some(ComplexComputedState::InTrueBAndUsizeAbove8) } else { None } } _ => None, } } } #[test] fn complex_computed_state_gets_derived_correctly() { let mut world = World::new(); EventRegistry::register_event::>(&mut world); EventRegistry::register_event::>(&mut world); EventRegistry::register_event::>(&mut world); world.init_resource::>(); world.init_resource::>(); let mut schedules = Schedules::new(); let mut apply_changes = Schedule::new(StateTransition); ComplexComputedState::register_computed_state_systems(&mut apply_changes); SimpleState::register_state(&mut apply_changes); OtherState::register_state(&mut apply_changes); schedules.insert(apply_changes); world.insert_resource(schedules); setup_state_transitions_in_world(&mut world, None); world.run_schedule(StateTransition); assert_eq!(world.resource::>().0, SimpleState::A); assert_eq!( world.resource::>().0, OtherState::default() ); assert!(!world.contains_resource::>()); world.insert_resource(NextState::Pending(SimpleState::B(true))); world.run_schedule(StateTransition); assert!(!world.contains_resource::>()); world.insert_resource(NextState::Pending(OtherState { a_flexible_value: "felix", another_value: 13, })); world.run_schedule(StateTransition); assert_eq!( world.resource::>().0, ComplexComputedState::InTrueBAndUsizeAbove8 ); world.insert_resource(NextState::Pending(SimpleState::A)); world.insert_resource(NextState::Pending(OtherState { a_flexible_value: "jane", another_value: 13, })); world.run_schedule(StateTransition); assert_eq!( world.resource::>().0, ComplexComputedState::InAAndStrIsBobOrJane ); world.insert_resource(NextState::Pending(SimpleState::B(false))); world.insert_resource(NextState::Pending(OtherState { a_flexible_value: "jane", another_value: 13, })); world.run_schedule(StateTransition); assert!(!world.contains_resource::>()); } #[derive(Resource, Default)] struct ComputedStateTransitionCounter { enter: usize, exit: usize, } #[derive(States, PartialEq, Eq, Debug, Default, Hash, Clone)] enum SimpleState2 { #[default] A1, B2, } #[derive(PartialEq, Eq, Debug, Hash, Clone)] enum TestNewcomputedState { A1, B2, B1, } impl ComputedStates for TestNewcomputedState { type SourceStates = (Option, Option); fn compute((s1, s2): (Option, Option)) -> Option { match (s1, s2) { (Some(SimpleState::A), Some(SimpleState2::A1)) => Some(TestNewcomputedState::A1), (Some(SimpleState::B(true)), Some(SimpleState2::B2)) => { Some(TestNewcomputedState::B2) } (Some(SimpleState::B(true)), _) => Some(TestNewcomputedState::B1), _ => None, } } } #[derive(ScheduleLabel, Clone, Debug, PartialEq, Eq, Hash)] struct Startup; #[test] fn computed_state_transitions_are_produced_correctly() { let mut world = World::new(); EventRegistry::register_event::>(&mut world); EventRegistry::register_event::>(&mut world); EventRegistry::register_event::>(&mut world); world.init_resource::>(); world.init_resource::>(); world.init_resource::(); setup_state_transitions_in_world(&mut world, Some(Startup.intern())); let mut schedules = world .get_resource_mut::() .expect("Schedules don't exist in world"); let apply_changes = schedules .get_mut(StateTransition) .expect("State Transition Schedule Doesn't Exist"); TestNewcomputedState::register_computed_state_systems(apply_changes); SimpleState::register_state(apply_changes); SimpleState2::register_state(apply_changes); schedules.insert({ let mut schedule = Schedule::new(OnEnter(TestNewcomputedState::A1)); schedule.add_systems(|mut count: ResMut| { count.enter += 1; }); schedule }); schedules.insert({ let mut schedule = Schedule::new(OnExit(TestNewcomputedState::A1)); schedule.add_systems(|mut count: ResMut| { count.exit += 1; }); schedule }); schedules.insert({ let mut schedule = Schedule::new(OnEnter(TestNewcomputedState::B1)); schedule.add_systems(|mut count: ResMut| { count.enter += 1; }); schedule }); schedules.insert({ let mut schedule = Schedule::new(OnExit(TestNewcomputedState::B1)); schedule.add_systems(|mut count: ResMut| { count.exit += 1; }); schedule }); schedules.insert({ let mut schedule = Schedule::new(OnEnter(TestNewcomputedState::B2)); schedule.add_systems(|mut count: ResMut| { count.enter += 1; }); schedule }); schedules.insert({ let mut schedule = Schedule::new(OnExit(TestNewcomputedState::B2)); schedule.add_systems(|mut count: ResMut| { count.exit += 1; }); schedule }); world.init_resource::(); setup_state_transitions_in_world(&mut world, None); assert_eq!(world.resource::>().0, SimpleState::A); assert_eq!(world.resource::>().0, SimpleState2::A1); assert!(!world.contains_resource::>()); world.insert_resource(NextState::Pending(SimpleState::B(true))); world.insert_resource(NextState::Pending(SimpleState2::B2)); world.run_schedule(StateTransition); assert_eq!( world.resource::>().0, TestNewcomputedState::B2 ); assert_eq!(world.resource::().enter, 1); assert_eq!(world.resource::().exit, 0); world.insert_resource(NextState::Pending(SimpleState2::A1)); world.insert_resource(NextState::Pending(SimpleState::A)); world.run_schedule(StateTransition); assert_eq!( world.resource::>().0, TestNewcomputedState::A1 ); assert_eq!( world.resource::().enter, 2, "Should Only Enter Twice" ); assert_eq!( world.resource::().exit, 1, "Should Only Exit Once" ); world.insert_resource(NextState::Pending(SimpleState::B(true))); world.insert_resource(NextState::Pending(SimpleState2::B2)); world.run_schedule(StateTransition); assert_eq!( world.resource::>().0, TestNewcomputedState::B2 ); assert_eq!( world.resource::().enter, 3, "Should Only Enter Three Times" ); assert_eq!( world.resource::().exit, 2, "Should Only Exit Twice" ); world.insert_resource(NextState::Pending(SimpleState::A)); world.run_schedule(StateTransition); assert!(!world.contains_resource::>()); assert_eq!( world.resource::().enter, 3, "Should Only Enter Three Times" ); assert_eq!( world.resource::().exit, 3, "Should Only Exit Twice" ); } #[derive(Resource, Default, PartialEq, Debug)] struct TransitionCounter { exit: u8, transition: u8, enter: u8, } #[test] fn same_state_transition_should_emit_event_and_not_run_schedules() { let mut world = World::new(); EventRegistry::register_event::>(&mut world); world.init_resource::>(); let mut schedules = Schedules::new(); let mut apply_changes = Schedule::new(StateTransition); SimpleState::register_state(&mut apply_changes); schedules.insert(apply_changes); world.insert_resource(TransitionCounter::default()); let mut on_exit = Schedule::new(OnExit(SimpleState::A)); on_exit.add_systems(|mut c: ResMut| c.exit += 1); schedules.insert(on_exit); let mut on_transition = Schedule::new(OnTransition { exited: SimpleState::A, entered: SimpleState::A, }); on_transition.add_systems(|mut c: ResMut| c.transition += 1); schedules.insert(on_transition); let mut on_enter = Schedule::new(OnEnter(SimpleState::A)); on_enter.add_systems(|mut c: ResMut| c.enter += 1); schedules.insert(on_enter); world.insert_resource(schedules); setup_state_transitions_in_world(&mut world, None); world.run_schedule(StateTransition); assert_eq!(world.resource::>().0, SimpleState::A); assert!(world .resource::>>() .is_empty()); world.insert_resource(TransitionCounter::default()); world.insert_resource(NextState::Pending(SimpleState::A)); world.run_schedule(StateTransition); assert_eq!(world.resource::>().0, SimpleState::A); assert_eq!( *world.resource::(), TransitionCounter { exit: 0, transition: 0, enter: 0 } ); assert_eq!( world .resource::>>() .len(), 1 ); } #[test] fn same_state_transition_should_propagate_to_sub_state() { let mut world = World::new(); EventRegistry::register_event::>(&mut world); EventRegistry::register_event::>(&mut world); world.insert_resource(State(SimpleState::B(true))); world.init_resource::>(); let mut schedules = Schedules::new(); let mut apply_changes = Schedule::new(StateTransition); SimpleState::register_state(&mut apply_changes); SubState::register_sub_state_systems(&mut apply_changes); schedules.insert(apply_changes); world.insert_resource(schedules); setup_state_transitions_in_world(&mut world, None); world.insert_resource(NextState::Pending(SimpleState::B(true))); world.run_schedule(StateTransition); assert_eq!( world .resource::>>() .len(), 1 ); assert_eq!( world .resource::>>() .len(), 1 ); } #[test] fn same_state_transition_should_propagate_to_computed_state() { let mut world = World::new(); EventRegistry::register_event::>(&mut world); EventRegistry::register_event::>(&mut world); world.insert_resource(State(SimpleState::B(true))); world.insert_resource(State(TestComputedState::BisTrue)); let mut schedules = Schedules::new(); let mut apply_changes = Schedule::new(StateTransition); SimpleState::register_state(&mut apply_changes); TestComputedState::register_computed_state_systems(&mut apply_changes); schedules.insert(apply_changes); world.insert_resource(schedules); setup_state_transitions_in_world(&mut world, None); world.insert_resource(NextState::Pending(SimpleState::B(true))); world.run_schedule(StateTransition); assert_eq!( world .resource::>>() .len(), 1 ); assert_eq!( world .resource::>>() .len(), 1 ); } }