leafwing_input_manager/lib.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168
#![forbid(missing_docs)]
#![warn(clippy::doc_markdown)]
#![doc = include_str!("../README.md")]
use crate::action_state::ActionState;
use crate::input_map::InputMap;
use bevy::ecs::prelude::*;
use bevy::reflect::{FromReflect, Reflect, TypePath};
use serde::{Deserialize, Serialize};
use std::fmt::Debug;
use std::hash::Hash;
pub mod action_diff;
pub mod action_state;
pub mod axislike;
pub mod buttonlike;
pub mod clashing_inputs;
pub mod common_conditions;
pub mod input_map;
pub mod input_processing;
pub mod plugin;
pub mod systems;
#[cfg(feature = "timing")]
pub mod timing;
pub mod typetag;
pub mod user_input;
// Importing the derive macro
pub use leafwing_input_manager_macros::Actionlike;
/// Everything you need to get started
pub mod prelude {
pub use crate::InputControlKind;
pub use crate::action_state::ActionState;
pub use crate::clashing_inputs::ClashStrategy;
pub use crate::input_map::InputMap;
pub use crate::input_processing::*;
pub use crate::user_input::*;
pub use crate::plugin::InputManagerPlugin;
pub use crate::{Actionlike, InputManagerBundle};
pub use leafwing_input_manager_macros::serde_typetag;
}
/// Allows a type to be used as a gameplay action in an input-agnostic fashion
///
/// Actions are modelled as "virtual buttons" (or axes), cleanly abstracting over messy, customizable inputs
/// in a way that can be easily consumed by your game logic.
///
/// This trait should be implemented on the `A` type that you want to pass into [`InputManagerPlugin`](crate::plugin::InputManagerPlugin).
///
/// Generally, these types will be very small (often data-less) enums.
/// As a result, the APIs in this crate accept actions by value, rather than reference.
/// While `Copy` is not a required trait bound,
/// users are strongly encouraged to derive `Copy` on these enums whenever possible to improve ergonomics.
///
/// # Examples
///
/// ```rust
/// use bevy::prelude::Reflect;
/// use leafwing_input_manager::Actionlike;
///
/// #[derive(Actionlike, Debug, PartialEq, Eq, Clone, Copy, Hash, Reflect)]
/// enum PlayerAction {
/// // Movement
/// Up,
/// Down,
/// Left,
/// Right,
/// // Abilities
/// Ability1,
/// Ability2,
/// Ability3,
/// Ability4,
/// Ultimate,
/// }
/// ```
///
/// # Customizing variant behavior
///
/// By default, the derive macro for this trait assumes that all actions are buttonlike.
///
/// You can customize this behavior by using the `#[actionlike]` attribute,
/// either on the entire enum or on individual variants.
///
/// See the document of [`InputControlKind`] for available options.
///
/// ```rust
/// use bevy::prelude::*;
/// use leafwing_input_manager::prelude::*;
///
/// #[derive(Actionlike, Debug, PartialEq, Eq, Clone, Copy, Hash, Reflect)]
/// #[actionlike(Axis)] // This attribute applies to all variants in the enum
/// enum CameraAction {
/// Zoom, // This action is controlled by axes
/// #[actionlike(DualAxis)]
/// Move, // This action is controlled by dual axes since we have overridden the default option
/// #[actionlike(Button)]
/// TakePhoto, // This action is controlled by buttons since we have overridden the default option
/// }
/// ```
pub trait Actionlike:
Debug + Eq + Hash + Send + Sync + Clone + Reflect + TypePath + FromReflect + 'static
{
/// Returns the kind of input control this action represents: buttonlike, axislike, or dual-axislike.
fn input_control_kind(&self) -> InputControlKind;
}
/// This [`Bundle`] allows entities to collect and interpret inputs from across input sources
///
/// Use with [`InputManagerPlugin`](crate::plugin::InputManagerPlugin), providing the same enum type to both.
#[derive(Bundle)]
pub struct InputManagerBundle<A: Actionlike> {
/// An [`ActionState`] component
pub action_state: ActionState<A>,
/// An [`InputMap`] component
pub input_map: InputMap<A>,
}
// Cannot use derive(Default), as it forces an undesirable bound on our generics
impl<A: Actionlike> Default for InputManagerBundle<A> {
fn default() -> Self {
Self {
action_state: ActionState::default(),
input_map: InputMap::default(),
}
}
}
impl<A: Actionlike> InputManagerBundle<A> {
/// Creates a [`InputManagerBundle`] with the given [`InputMap`].
pub fn with_map(input_map: InputMap<A>) -> Self {
Self {
input_map,
action_state: ActionState::default(),
}
}
}
/// Classifies [`UserInput`](crate::user_input::UserInput)s and [`Actionlike`] actions based on their behavior (buttons, analog axes, etc.).
#[derive(Debug, Clone, Copy, PartialEq, Reflect, Serialize, Deserialize)]
#[must_use]
pub enum InputControlKind {
/// A single input with binary state (active or inactive), typically a button press (on or off).
///
/// Corresponds to [`Buttonlike`](crate::user_input::Buttonlike) inputs.
Button,
/// A single analog or digital input, often used for range controls like a thumb stick on a gamepad or mouse wheel,
/// providing a value within a min-max range.
///
/// Corresponds to [`Axislike`](crate::user_input::Axislike) inputs.
Axis,
/// A combination of two axis-like inputs, often used for directional controls like a D-pad on a gamepad,
/// providing separate values for the X and Y axes.
///
/// Corresponds to [`DualAxislike`](crate::user_input::DualAxislike) inputs.
DualAxis,
/// A combination of three axis-like inputs, providing separate values for the X, Y and Z axes.
///
/// Corresponds to [`TripleAxislike`](crate::user_input::TripleAxislike) inputs.
TripleAxis,
}