Crates.io | leafwing_abilities |
lib.rs | leafwing_abilities |
version | 0.9.0 |
source | src |
created_at | 2022-10-17 23:46:48.091525 |
updated_at | 2024-11-15 21:22:44.510467 |
description | A convenient, well-tested ability management suite. Built for the Bevy game engine. |
homepage | https://leafwing-studios.com/ |
repository | https://github.com/leafwing-studios/leafwing_abilities |
max_upload_size | |
id | 690494 |
size | 238,317 |
A fully-featured set of tools for managing abilities in Bevy. This crate is meant to be used with Leafwing Input Manager, which converts inputs into actions.
Some of those actions will be abilities! Abilities are intended for gameplay use, and follow complex but relatively standardized logic about how they might be used.
use bevy::prelude::*;
use bevy::reflect::Reflect;
use leafwing_abilities::prelude::*;
use leafwing_abilities::premade_pools::mana::{ManaPool, Mana};
use leafwing_abilities::premade_pools::life::{LifePool, Life};
use leafwing_input_manager::prelude::*;
// We're modelling https://leagueoflegends.fandom.com/wiki/Zyra/LoL
// to show off this crate's features!
#[derive(Actionlike, Abilitylike, Clone, Copy, Hash, PartialEq, Eq, Reflect)]
pub enum ZyraAbility {
GardenOfThorns,
DeadlySpines,
RampantGrowth,
GraspingRoots,
Stranglethorns,
}
impl ZyraAbility {
/// You could use the `strum` crate to derive this automatically!
fn variants() -> Vec<ZyraAbility> {
use ZyraAbility::*;
vec![GardenOfThorns, DeadlySpines, RampantGrowth, GraspingRoots, Stranglethorns]
}
fn input_map() -> InputMap<ZyraAbility> {
use ZyraAbility::*;
// We can use this `new` idiom, which accepts an iterator of pairs
InputMap::new([
(DeadlySpines, KeyCode::KeyQ),
(RampantGrowth, KeyCode::KeyW),
(GraspingRoots, KeyCode::KeyE),
(Stranglethorns, KeyCode::KeyR),
])
}
// This match pattern is super useful to be sure you've defined an attribute for every variant
fn cooldown(&self) -> Cooldown {
use ZyraAbility::*;
let seconds: f32 = match *self {
GardenOfThorns => 13.0,
DeadlySpines => 7.0,
RampantGrowth => 18.0,
GraspingRoots => 12.0,
Stranglethorns => 110.0,
};
Cooldown::from_secs(seconds)
}
fn cooldowns() -> CooldownState<ZyraAbility> {
let mut cooldowns = CooldownState::default();
// Now, we can loop over all the variants to populate our struct
for ability in ZyraAbility::variants() {
cooldowns.set(ability, ability.cooldown());
}
cooldowns
}
fn charges() -> ChargeState<ZyraAbility> {
// The builder API can be very convenient when you only need to set a couple of values
ChargeState::default()
.set(ZyraAbility::RampantGrowth, Charges::replenish_one(2))
.build()
}
fn mana_costs() -> AbilityCosts<ZyraAbility, ManaPool> {
use ZyraAbility::*;
AbilityCosts::new([
(DeadlySpines, Mana(70.)),
(GraspingRoots, Mana(70.)),
(Stranglethorns, Mana(100.)),
])
}
}
/// Marker component for this champion
#[derive(Component)]
struct Zyra;
#[derive(Bundle)]
struct ZyraBundle {
champion: Zyra,
life_pool: LifePool,
input_manager_bundle: InputManagerBundle<ZyraAbility>,
abilities_bundle: AbilitiesBundle<ZyraAbility>,
mana_bundle: PoolBundle<ZyraAbility, ManaPool>,
}
impl Default for ZyraBundle {
fn default() -> Self {
ZyraBundle {
champion: Zyra,
// Max life, then regen
life_pool: LifePool::new(Life(574.), Life(574.), (Life(5.5))),
input_manager_bundle: InputManagerBundle::<ZyraAbility> {
input_map: ZyraAbility::input_map(),
..default()
},
abilities_bundle: AbilitiesBundle::<ZyraAbility> {
cooldowns: ZyraAbility::cooldowns(),
charges: ZyraAbility::charges(),
},
mana_bundle: PoolBundle::<ZyraAbility, ManaPool> {
pool: ManaPool::new(Mana(418.), Mana(418.), Mana(13.0)),
ability_costs: ZyraAbility::mana_costs(),
}
}
}
}
#[deny(missing_docs)]
Planned: