use crate::{ component::Component, schedule::{ RunCriteriaDescriptor, RunCriteriaDescriptorCoercion, RunCriteriaLabel, ShouldRun, SystemSet, }, system::{In, IntoChainSystem, IntoSystem, Local, Res, ResMut}, }; use std::{any::TypeId, fmt::Debug, hash::Hash}; use thiserror::Error; /// ### Stack based state machine /// /// This state machine has four operations: Push, Pop, Set and Replace. /// * Push pushes a new state to the state stack, pausing the previous state /// * Pop removes the current state, and unpauses the last paused state /// * Set replaces the active state with a new one /// * Replace unwinds the state stack, and replaces the entire stack with a single new state #[derive(Debug)] pub struct State { transition: Option>, stack: Vec, scheduled: Option>, end_next_loop: bool, } #[derive(Debug)] enum StateTransition { PreStartup, Startup, // The parameter order is always (leaving, entering) ExitingToResume(T, T), ExitingFull(T, T), Entering(T, T), Resuming(T, T), Pausing(T, T), } #[derive(Debug)] enum ScheduledOperation { Set(T), Replace(T), Pop, Push(T), } #[derive(Debug, PartialEq, Eq, Clone, Hash)] enum StateCallback { Update, InactiveUpdate, InStackUpdate, Enter, Exit, Pause, Resume, } impl StateCallback { fn into_label(self, state: T) -> StateRunCriteriaLabel where T: Component + Debug + Clone + Eq + Hash, { StateRunCriteriaLabel(state, self) } } #[derive(Debug, PartialEq, Eq, Clone, Hash)] struct StateRunCriteriaLabel(T, StateCallback); impl RunCriteriaLabel for StateRunCriteriaLabel where T: Component + Debug + Clone + Eq + Hash, { fn dyn_clone(&self) -> Box { Box::new(self.clone()) } } #[derive(Debug, PartialEq, Eq, Clone, Hash)] struct DriverLabel(TypeId); impl RunCriteriaLabel for DriverLabel { fn dyn_clone(&self) -> Box { Box::new(self.clone()) } } impl DriverLabel { fn of() -> Self { Self(TypeId::of::()) } } impl State where T: Component + Debug + Clone + Eq + Hash, { pub fn on_update(s: T) -> RunCriteriaDescriptor { (|state: Res>, pred: Local>| { state.stack.last().unwrap() == pred.as_ref().unwrap() && state.transition.is_none() }) .system() .config(|(_, pred)| *pred = Some(Some(s.clone()))) .chain(should_run_adapter::.system()) .after(DriverLabel::of::()) .label_discard_if_duplicate(StateCallback::Update.into_label(s)) } pub fn on_inactive_update(s: T) -> RunCriteriaDescriptor { (|state: Res>, mut is_inactive: Local, pred: Local>| match &state .transition { Some(StateTransition::Pausing(ref relevant, _)) | Some(StateTransition::Resuming(_, ref relevant)) => { if relevant == pred.as_ref().unwrap() { *is_inactive = !*is_inactive; } false } Some(_) => false, None => *is_inactive, }) .system() .config(|(_, _, pred)| *pred = Some(Some(s.clone()))) .chain(should_run_adapter::.system()) .after(DriverLabel::of::()) .label_discard_if_duplicate(StateCallback::InactiveUpdate.into_label(s)) } pub fn on_in_stack_update(s: T) -> RunCriteriaDescriptor { (|state: Res>, mut is_in_stack: Local, pred: Local>| match &state .transition { Some(StateTransition::Entering(ref relevant, _)) | Some(StateTransition::ExitingToResume(_, ref relevant)) => { if relevant == pred.as_ref().unwrap() { *is_in_stack = !*is_in_stack; } false } Some(StateTransition::ExitingFull(_, ref relevant)) => { if relevant == pred.as_ref().unwrap() { *is_in_stack = !*is_in_stack; } false } Some(StateTransition::Startup) => { if state.stack.last().unwrap() == pred.as_ref().unwrap() { *is_in_stack = !*is_in_stack; } false } Some(_) => false, None => *is_in_stack, }) .system() .config(|(_, _, pred)| *pred = Some(Some(s.clone()))) .chain(should_run_adapter::.system()) .after(DriverLabel::of::()) .label_discard_if_duplicate(StateCallback::InStackUpdate.into_label(s)) } pub fn on_enter(s: T) -> RunCriteriaDescriptor { (|state: Res>, pred: Local>| { state .transition .as_ref() .map_or(false, |transition| match transition { StateTransition::Entering(_, entering) => entering == pred.as_ref().unwrap(), StateTransition::Startup => { state.stack.last().unwrap() == pred.as_ref().unwrap() } _ => false, }) }) .system() .config(|(_, pred)| *pred = Some(Some(s.clone()))) .chain(should_run_adapter::.system()) .after(DriverLabel::of::()) .label_discard_if_duplicate(StateCallback::Enter.into_label(s)) } pub fn on_exit(s: T) -> RunCriteriaDescriptor { (|state: Res>, pred: Local>| { state .transition .as_ref() .map_or(false, |transition| match transition { StateTransition::ExitingToResume(exiting, _) | StateTransition::ExitingFull(exiting, _) => exiting == pred.as_ref().unwrap(), _ => false, }) }) .system() .config(|(_, pred)| *pred = Some(Some(s.clone()))) .chain(should_run_adapter::.system()) .after(DriverLabel::of::()) .label_discard_if_duplicate(StateCallback::Exit.into_label(s)) } pub fn on_pause(s: T) -> RunCriteriaDescriptor { (|state: Res>, pred: Local>| { state .transition .as_ref() .map_or(false, |transition| match transition { StateTransition::Pausing(pausing, _) => pausing == pred.as_ref().unwrap(), _ => false, }) }) .system() .config(|(_, pred)| *pred = Some(Some(s.clone()))) .chain(should_run_adapter::.system()) .after(DriverLabel::of::()) .label_discard_if_duplicate(StateCallback::Pause.into_label(s)) } pub fn on_resume(s: T) -> RunCriteriaDescriptor { (|state: Res>, pred: Local>| { state .transition .as_ref() .map_or(false, |transition| match transition { StateTransition::Resuming(_, resuming) => resuming == pred.as_ref().unwrap(), _ => false, }) }) .system() .config(|(_, pred)| *pred = Some(Some(s.clone()))) .chain(should_run_adapter::.system()) .after(DriverLabel::of::()) .label_discard_if_duplicate(StateCallback::Resume.into_label(s)) } pub fn on_update_set(s: T) -> SystemSet { SystemSet::new().with_run_criteria(Self::on_update(s)) } pub fn on_inactive_update_set(s: T) -> SystemSet { SystemSet::new().with_run_criteria(Self::on_inactive_update(s)) } pub fn on_enter_set(s: T) -> SystemSet { SystemSet::new().with_run_criteria(Self::on_enter(s)) } pub fn on_exit_set(s: T) -> SystemSet { SystemSet::new().with_run_criteria(Self::on_exit(s)) } pub fn on_pause_set(s: T) -> SystemSet { SystemSet::new().with_run_criteria(Self::on_pause(s)) } pub fn on_resume_set(s: T) -> SystemSet { SystemSet::new().with_run_criteria(Self::on_resume(s)) } /// Creates a driver set for the State. /// /// Important note: this set must be inserted **before** all other state-dependant sets to work /// properly! pub fn get_driver() -> SystemSet { SystemSet::default() .with_run_criteria(state_cleaner::.system().label(DriverLabel::of::())) } pub fn new(initial: T) -> Self { Self { stack: vec![initial], transition: Some(StateTransition::PreStartup), scheduled: None, end_next_loop: false, } } /// Schedule a state change that replaces the active state with the given state. /// This will fail if there is a scheduled operation, or if the given `state` matches the /// current state pub fn set(&mut self, state: T) -> Result<(), StateError> { if self.stack.last().unwrap() == &state { return Err(StateError::AlreadyInState); } if self.scheduled.is_some() { return Err(StateError::StateAlreadyQueued); } self.scheduled = Some(ScheduledOperation::Set(state)); Ok(()) } /// Same as [Self::set], but if there is already a next state, it will be overwritten /// instead of failing pub fn overwrite_set(&mut self, state: T) -> Result<(), StateError> { if self.stack.last().unwrap() == &state { return Err(StateError::AlreadyInState); } self.scheduled = Some(ScheduledOperation::Set(state)); Ok(()) } /// Schedule a state change that replaces the full stack with the given state. /// This will fail if there is a scheduled operation, or if the given `state` matches the current state pub fn replace(&mut self, state: T) -> Result<(), StateError> { if self.stack.last().unwrap() == &state { return Err(StateError::AlreadyInState); } if self.scheduled.is_some() { return Err(StateError::StateAlreadyQueued); } self.scheduled = Some(ScheduledOperation::Replace(state)); Ok(()) } /// Same as [Self::replace], but if there is already a next state, it will be overwritten instead of failing pub fn overwrite_replace(&mut self, state: T) -> Result<(), StateError> { if self.stack.last().unwrap() == &state { return Err(StateError::AlreadyInState); } self.scheduled = Some(ScheduledOperation::Replace(state)); Ok(()) } /// Same as [Self::set], but does a push operation instead of a next operation pub fn push(&mut self, state: T) -> Result<(), StateError> { if self.stack.last().unwrap() == &state { return Err(StateError::AlreadyInState); } if self.scheduled.is_some() { return Err(StateError::StateAlreadyQueued); } self.scheduled = Some(ScheduledOperation::Push(state)); Ok(()) } /// Same as [Self::push], but if there is already a next state, it will be overwritten /// instead of failing pub fn overwrite_push(&mut self, state: T) -> Result<(), StateError> { if self.stack.last().unwrap() == &state { return Err(StateError::AlreadyInState); } self.scheduled = Some(ScheduledOperation::Push(state)); Ok(()) } /// Same as [Self::set], but does a pop operation instead of a set operation pub fn pop(&mut self) -> Result<(), StateError> { if self.scheduled.is_some() { return Err(StateError::StateAlreadyQueued); } if self.stack.len() == 1 { return Err(StateError::StackEmpty); } self.scheduled = Some(ScheduledOperation::Pop); Ok(()) } /// Same as [Self::pop], but if there is already a next state, it will be overwritten /// instead of failing pub fn overwrite_pop(&mut self) -> Result<(), StateError> { if self.stack.len() == 1 { return Err(StateError::StackEmpty); } self.scheduled = Some(ScheduledOperation::Pop); Ok(()) } pub fn current(&self) -> &T { self.stack.last().unwrap() } pub fn inactives(&self) -> &[T] { &self.stack[0..self.stack.len() - 2] } } #[derive(Debug, Error)] pub enum StateError { #[error("Attempted to change the state to the current state.")] AlreadyInState, #[error("Attempted to queue a state change, but there was already a state queued.")] StateAlreadyQueued, #[error("Attempted to queue a pop, but there is nothing to pop.")] StackEmpty, } fn should_run_adapter( In(cmp_result): In, state: Res>, ) -> ShouldRun { if state.end_next_loop { return ShouldRun::No; } if cmp_result { ShouldRun::YesAndCheckAgain } else { ShouldRun::NoAndCheckAgain } } fn state_cleaner( mut state: ResMut>, mut prep_exit: Local, ) -> ShouldRun { if *prep_exit { *prep_exit = false; if state.scheduled.is_none() { state.end_next_loop = true; return ShouldRun::YesAndCheckAgain; } } else if state.end_next_loop { state.end_next_loop = false; return ShouldRun::No; } match state.scheduled.take() { Some(ScheduledOperation::Set(next)) => { state.transition = Some(StateTransition::ExitingFull( state.stack.last().unwrap().clone(), next, )); } Some(ScheduledOperation::Replace(next)) => { if state.stack.len() <= 1 { state.transition = Some(StateTransition::ExitingFull( state.stack.last().unwrap().clone(), next, )); } else { state.scheduled = Some(ScheduledOperation::Replace(next)); match state.transition.take() { Some(StateTransition::ExitingToResume(p, n)) => { state.stack.pop(); state.transition = Some(StateTransition::Resuming(p, n)); } _ => { state.transition = Some(StateTransition::ExitingToResume( state.stack[state.stack.len() - 1].clone(), state.stack[state.stack.len() - 2].clone(), )); } } } } Some(ScheduledOperation::Push(next)) => { let last_type_id = state.stack.last().unwrap().clone(); state.transition = Some(StateTransition::Pausing(last_type_id, next)); } Some(ScheduledOperation::Pop) => { state.transition = Some(StateTransition::ExitingToResume( state.stack[state.stack.len() - 1].clone(), state.stack[state.stack.len() - 2].clone(), )); } None => match state.transition.take() { Some(StateTransition::ExitingFull(p, n)) => { state.transition = Some(StateTransition::Entering(p, n.clone())); *state.stack.last_mut().unwrap() = n; } Some(StateTransition::Pausing(p, n)) => { state.transition = Some(StateTransition::Entering(p, n.clone())); state.stack.push(n); } Some(StateTransition::ExitingToResume(p, n)) => { state.stack.pop(); state.transition = Some(StateTransition::Resuming(p, n)); } Some(StateTransition::PreStartup) => { state.transition = Some(StateTransition::Startup); } _ => {} }, }; if state.transition.is_none() { *prep_exit = true; } ShouldRun::YesAndCheckAgain } #[cfg(test)] mod test { use super::*; use crate::prelude::*; #[derive(Debug, Clone, Copy, Eq, PartialEq, Hash)] enum MyState { S1, S2, S3, S4, S5, S6, Final, } #[test] fn state_test() { let mut world = World::default(); world.insert_resource(Vec::<&'static str>::new()); world.insert_resource(State::new(MyState::S1)); let mut stage = SystemStage::parallel(); stage.add_system_set(State::::get_driver()); stage .add_system_set( State::on_enter_set(MyState::S1) .with_system((|mut r: ResMut>| r.push("startup")).system()), ) .add_system_set( State::on_update_set(MyState::S1).with_system( (|mut r: ResMut>, mut s: ResMut>| { r.push("update S1"); s.overwrite_replace(MyState::S2).unwrap(); }) .system(), ), ) .add_system_set( State::on_enter_set(MyState::S2) .with_system((|mut r: ResMut>| r.push("enter S2")).system()), ) .add_system_set( State::on_update_set(MyState::S2).with_system( (|mut r: ResMut>, mut s: ResMut>| { r.push("update S2"); s.overwrite_replace(MyState::S3).unwrap(); }) .system(), ), ) .add_system_set( State::on_exit_set(MyState::S2) .with_system((|mut r: ResMut>| r.push("exit S2")).system()), ) .add_system_set( State::on_enter_set(MyState::S3) .with_system((|mut r: ResMut>| r.push("enter S3")).system()), ) .add_system_set( State::on_update_set(MyState::S3).with_system( (|mut r: ResMut>, mut s: ResMut>| { r.push("update S3"); s.overwrite_push(MyState::S4).unwrap(); }) .system(), ), ) .add_system_set( State::on_pause_set(MyState::S3) .with_system((|mut r: ResMut>| r.push("pause S3")).system()), ) .add_system_set( State::on_update_set(MyState::S4).with_system( (|mut r: ResMut>, mut s: ResMut>| { r.push("update S4"); s.overwrite_push(MyState::S5).unwrap(); }) .system(), ), ) .add_system_set( State::on_inactive_update_set(MyState::S4).with_system( (|mut r: ResMut>| r.push("inactive S4")) .system() .label("inactive s4"), ), ) .add_system_set( State::on_update_set(MyState::S5).with_system( (|mut r: ResMut>, mut s: ResMut>| { r.push("update S5"); s.overwrite_push(MyState::S6).unwrap(); }) .system() .after("inactive s4"), ), ) .add_system_set( State::on_inactive_update_set(MyState::S5).with_system( (|mut r: ResMut>| r.push("inactive S5")) .system() .label("inactive s5") .after("inactive s4"), ), ) .add_system_set( State::on_update_set(MyState::S6).with_system( (|mut r: ResMut>, mut s: ResMut>| { r.push("update S6"); s.overwrite_push(MyState::Final).unwrap(); }) .system() .after("inactive s5"), ), ) .add_system_set( State::on_resume_set(MyState::S4) .with_system((|mut r: ResMut>| r.push("resume S4")).system()), ) .add_system_set( State::on_exit_set(MyState::S5) .with_system((|mut r: ResMut>| r.push("exit S4")).system()), ); const EXPECTED: &[&str] = &[ // "startup", "update S1", // "enter S2", "update S2", // "exit S2", "enter S3", "update S3", // "pause S3", "update S4", // "inactive S4", "update S5", // "inactive S4", "inactive S5", "update S6", // "inactive S4", "inactive S5", ]; stage.run(&mut world); let mut collected = world.get_resource_mut::>().unwrap(); let mut count = 0; for (found, expected) in collected.drain(..).zip(EXPECTED) { assert_eq!(found, *expected); count += 1; } // If not equal, some elements weren't executed assert_eq!(EXPECTED.len(), count); assert_eq!( world.get_resource::>().unwrap().current(), &MyState::Final ); } #[test] fn issue_1753() { #[derive(Clone, PartialEq, Eq, Debug, Hash)] enum AppState { Main, } fn should_run_once(mut flag: ResMut, test_name: Res<&'static str>) { assert!(!*flag, "{:?}", *test_name); *flag = true; } let mut world = World::new(); world.insert_resource(State::new(AppState::Main)); world.insert_resource(false); world.insert_resource("control"); let mut stage = SystemStage::parallel().with_system(should_run_once.system()); stage.run(&mut world); assert!(*world.get_resource::().unwrap(), "after control"); world.insert_resource(false); world.insert_resource("test"); let mut stage = SystemStage::parallel() .with_system_set(State::::get_driver()) .with_system(should_run_once.system()); stage.run(&mut world); assert!(*world.get_resource::().unwrap(), "after test"); } }