use std::fmt::Debug; use std::hash::Hash; use std::mem; use std::ops::Deref; use crate as bevy_ecs; use crate::change_detection::DetectChangesMut; use crate::schedule::ScheduleLabel; use crate::system::Resource; use crate::world::World; pub use bevy_ecs_macros::States; /// Types that can define world-wide states in a finite-state machine. /// /// The [`Default`] trait defines the starting state. /// Multiple states can be defined for the same world, /// allowing you to classify the state of the world across orthogonal dimensions. /// You can access the current state of type `T` with the [`State`] resource, /// and the queued state with the [`NextState`] resource. /// /// State transitions typically occur in the [`OnEnter`] and [`OnExit`] schedules, /// which can be run via the [`apply_state_transition::`] system. /// /// # Example /// /// ```rust /// use bevy_ecs::prelude::States; /// /// #[derive(Clone, Copy, PartialEq, Eq, Hash, Debug, Default, States)] /// enum GameState { /// #[default] /// MainMenu, /// SettingsMenu, /// InGame, /// } /// /// ``` pub trait States: 'static + Send + Sync + Clone + PartialEq + Eq + Hash + Debug + Default { /// The type returned when iterating over all [`variants`](States::variants) of this type. type Iter: Iterator; /// Returns an iterator over all the state variants. fn variants() -> Self::Iter; } /// The label of a [`Schedule`](super::Schedule) that runs whenever [`State`] /// enters this state. #[derive(ScheduleLabel, Clone, Debug, PartialEq, Eq, Hash)] pub struct OnEnter(pub S); /// The label of a [`Schedule`](super::Schedule) that runs whenever [`State`] /// exits this state. #[derive(ScheduleLabel, Clone, Debug, PartialEq, Eq, Hash)] pub struct OnExit(pub S); /// The label of a [`Schedule`](super::Schedule) that **only** runs whenever [`State`] /// exits the `from` state, AND enters the `to` state. /// /// Systems added to this schedule are always ran *after* [`OnExit`], and *before* [`OnEnter`]. #[derive(ScheduleLabel, Clone, Debug, PartialEq, Eq, Hash)] pub struct OnTransition { /// The state being exited. pub from: S, /// The state being entered. pub to: S, } /// A finite-state machine whose transitions have associated schedules /// ([`OnEnter(state)`] and [`OnExit(state)`]). /// /// The current state value can be accessed through this resource. To *change* the state, /// queue a transition in the [`NextState`] resource, and it will be applied by the next /// [`apply_state_transition::`] system. /// /// The starting state is defined via the [`Default`] implementation for `S`. #[derive(Resource, Default, Debug)] pub struct State(S); impl State { /// Creates a new state with a specific value. /// /// To change the state use [`NextState`] rather than using this to modify the `State`. pub fn new(state: S) -> Self { Self(state) } /// Get the current state. pub fn get(&self) -> &S { &self.0 } } impl PartialEq for State { fn eq(&self, other: &S) -> bool { self.get() == other } } impl Deref for State { type Target = S; fn deref(&self) -> &Self::Target { self.get() } } /// The next state of [`State`]. /// /// To queue a transition, just set the contained value to `Some(next_state)`. /// Note that these transitions can be overridden by other systems: /// only the actual value of this resource at the time of [`apply_state_transition`] matters. #[derive(Resource, Default, Debug)] pub struct NextState(pub Option); impl NextState { /// Tentatively set a planned state transition to `Some(state)`. pub fn set(&mut self, state: S) { self.0 = Some(state); } } /// Run the enter schedule (if it exists) for the current state. pub fn run_enter_schedule(world: &mut World) { world .try_run_schedule(OnEnter(world.resource::>().0.clone())) .ok(); } /// If a new state is queued in [`NextState`], this system: /// - Takes the new state value from [`NextState`] and updates [`State`]. /// - Runs the [`OnExit(exited_state)`] schedule, if it exists. /// - Runs the [`OnTransition { from: exited_state, to: entered_state }`](OnTransition), if it exists. /// - Runs the [`OnEnter(entered_state)`] schedule, if it exists. pub fn apply_state_transition(world: &mut World) { // We want to take the `NextState` resource, // but only mark it as changed if it wasn't empty. let mut next_state_resource = world.resource_mut::>(); if let Some(entered) = next_state_resource.bypass_change_detection().0.take() { next_state_resource.set_changed(); let mut state_resource = world.resource_mut::>(); if *state_resource != entered { let exited = mem::replace(&mut state_resource.0, entered.clone()); // Try to run the schedules if they exist. world.try_run_schedule(OnExit(exited.clone())).ok(); world .try_run_schedule(OnTransition { from: exited, to: entered.clone(), }) .ok(); world.try_run_schedule(OnEnter(entered)).ok(); } } }