2020-04-11 06:33:06 +00:00
|
|
|
use super::{
|
|
|
|
hotbar::{self, Slot as HotbarSlot},
|
|
|
|
img_ids,
|
|
|
|
item_imgs::{ItemImgs, ItemKey},
|
2021-11-10 22:25:37 +00:00
|
|
|
util,
|
2020-04-11 06:33:06 +00:00
|
|
|
};
|
2020-04-06 15:25:45 +00:00
|
|
|
use crate::ui::slot::{self, SlotKey, SumSlot};
|
2020-05-29 18:23:00 +00:00
|
|
|
use common::comp::{
|
2021-11-12 03:37:37 +00:00
|
|
|
ability::AbilityInput, slot::InvSlotId, Ability, ActiveAbilities, Body, Energy, Inventory,
|
|
|
|
SkillSet,
|
2020-05-29 18:23:00 +00:00
|
|
|
};
|
2020-04-11 06:33:06 +00:00
|
|
|
use conrod_core::{image, Color};
|
2021-02-28 02:44:57 +00:00
|
|
|
use specs::Entity as EcsEntity;
|
2020-04-04 05:40:00 +00:00
|
|
|
|
2020-04-10 02:36:35 +00:00
|
|
|
pub use common::comp::slot::{ArmorSlot, EquipSlot};
|
|
|
|
|
2021-01-08 19:12:09 +00:00
|
|
|
#[derive(Clone, Copy, Debug, PartialEq)]
|
2020-04-06 15:25:45 +00:00
|
|
|
pub enum SlotKind {
|
2020-04-04 05:40:00 +00:00
|
|
|
Inventory(InventorySlot),
|
2020-04-10 02:36:35 +00:00
|
|
|
Equip(EquipSlot),
|
2020-04-11 06:33:06 +00:00
|
|
|
Hotbar(HotbarSlot),
|
2021-02-12 02:53:25 +00:00
|
|
|
Trade(TradeSlot),
|
2020-04-11 06:33:06 +00:00
|
|
|
/* Spellbook(SpellbookSlot), TODO */
|
2020-04-04 05:40:00 +00:00
|
|
|
}
|
|
|
|
|
2020-04-06 15:25:45 +00:00
|
|
|
pub type SlotManager = slot::SlotManager<SlotKind>;
|
2020-04-04 05:40:00 +00:00
|
|
|
|
2021-01-08 19:12:09 +00:00
|
|
|
#[derive(Clone, Copy, Debug, PartialEq)]
|
2021-02-28 02:44:57 +00:00
|
|
|
pub struct InventorySlot {
|
|
|
|
pub slot: InvSlotId,
|
2021-03-30 22:37:38 +00:00
|
|
|
pub entity: EcsEntity,
|
2021-02-28 02:44:57 +00:00
|
|
|
pub ours: bool,
|
|
|
|
}
|
2020-04-04 05:40:00 +00:00
|
|
|
|
2020-04-06 15:25:45 +00:00
|
|
|
impl SlotKey<Inventory, ItemImgs> for InventorySlot {
|
2020-04-04 05:40:00 +00:00
|
|
|
type ImageKey = ItemKey;
|
|
|
|
|
2020-04-11 06:33:06 +00:00
|
|
|
fn image_key(&self, source: &Inventory) -> Option<(Self::ImageKey, Option<Color>)> {
|
2021-02-28 02:44:57 +00:00
|
|
|
source.get(self.slot).map(|i| (i.into(), None))
|
2020-04-04 05:40:00 +00:00
|
|
|
}
|
|
|
|
|
2020-04-06 15:25:45 +00:00
|
|
|
fn amount(&self, source: &Inventory) -> Option<u32> {
|
2020-04-04 05:40:00 +00:00
|
|
|
source
|
2021-02-28 02:44:57 +00:00
|
|
|
.get(self.slot)
|
2020-09-17 23:02:14 +00:00
|
|
|
.map(|item| item.amount())
|
2020-04-04 05:40:00 +00:00
|
|
|
.filter(|amount| *amount > 1)
|
|
|
|
}
|
|
|
|
|
2021-02-16 01:05:54 +00:00
|
|
|
fn image_ids(key: &Self::ImageKey, source: &ItemImgs) -> Vec<image::Id> {
|
|
|
|
source.img_ids_or_not_found_img(key.clone())
|
2020-04-04 05:40:00 +00:00
|
|
|
}
|
2020-04-04 17:51:41 +00:00
|
|
|
}
|
|
|
|
|
2021-01-08 19:12:09 +00:00
|
|
|
impl SlotKey<Inventory, ItemImgs> for EquipSlot {
|
2020-04-04 17:51:41 +00:00
|
|
|
type ImageKey = ItemKey;
|
|
|
|
|
2021-01-08 19:12:09 +00:00
|
|
|
fn image_key(&self, source: &Inventory) -> Option<(Self::ImageKey, Option<Color>)> {
|
|
|
|
let item = source.equipped(*self);
|
2020-11-07 20:52:40 +00:00
|
|
|
item.map(|i| (i.into(), None))
|
2020-04-04 17:51:41 +00:00
|
|
|
}
|
|
|
|
|
2021-01-08 19:12:09 +00:00
|
|
|
fn amount(&self, _: &Inventory) -> Option<u32> { None }
|
2020-04-04 17:51:41 +00:00
|
|
|
|
2021-02-16 01:05:54 +00:00
|
|
|
fn image_ids(key: &Self::ImageKey, source: &ItemImgs) -> Vec<image::Id> {
|
|
|
|
source.img_ids_or_not_found_img(key.clone())
|
2020-04-04 17:51:41 +00:00
|
|
|
}
|
2020-04-04 05:40:00 +00:00
|
|
|
}
|
|
|
|
|
2021-02-12 02:53:25 +00:00
|
|
|
#[derive(Clone, Copy, Debug, PartialEq)]
|
|
|
|
pub struct TradeSlot {
|
|
|
|
pub index: usize,
|
|
|
|
pub quantity: u32,
|
|
|
|
pub invslot: Option<InvSlotId>,
|
2021-02-28 02:44:57 +00:00
|
|
|
pub entity: EcsEntity,
|
|
|
|
pub ours: bool,
|
2021-02-12 02:53:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl SlotKey<Inventory, ItemImgs> for TradeSlot {
|
|
|
|
type ImageKey = ItemKey;
|
|
|
|
|
|
|
|
fn image_key(&self, source: &Inventory) -> Option<(Self::ImageKey, Option<Color>)> {
|
2021-02-28 02:44:57 +00:00
|
|
|
self.invslot.and_then(|inv_id| {
|
|
|
|
InventorySlot {
|
|
|
|
slot: inv_id,
|
|
|
|
ours: self.ours,
|
2021-03-30 22:37:38 +00:00
|
|
|
entity: self.entity,
|
2021-02-28 02:44:57 +00:00
|
|
|
}
|
|
|
|
.image_key(source)
|
|
|
|
})
|
2021-02-12 02:53:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn amount(&self, source: &Inventory) -> Option<u32> {
|
|
|
|
self.invslot
|
2021-02-28 02:44:57 +00:00
|
|
|
.and_then(|inv_id| {
|
|
|
|
InventorySlot {
|
|
|
|
slot: inv_id,
|
|
|
|
ours: self.ours,
|
2021-03-30 22:37:38 +00:00
|
|
|
entity: self.entity,
|
2021-02-28 02:44:57 +00:00
|
|
|
}
|
|
|
|
.amount(source)
|
|
|
|
})
|
2021-02-12 02:53:25 +00:00
|
|
|
.map(|x| x.min(self.quantity))
|
|
|
|
}
|
|
|
|
|
2021-02-16 01:05:54 +00:00
|
|
|
fn image_ids(key: &Self::ImageKey, source: &ItemImgs) -> Vec<image::Id> {
|
|
|
|
source.img_ids_or_not_found_img(key.clone())
|
2021-02-12 02:53:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-11 06:33:06 +00:00
|
|
|
#[derive(Clone, PartialEq)]
|
|
|
|
pub enum HotbarImage {
|
|
|
|
Item(ItemKey),
|
2021-11-10 22:25:37 +00:00
|
|
|
Ability(String),
|
2020-04-11 06:33:06 +00:00
|
|
|
}
|
2020-04-06 15:25:45 +00:00
|
|
|
|
2021-11-10 02:20:41 +00:00
|
|
|
type HotbarSource<'a> = (
|
|
|
|
&'a hotbar::State,
|
|
|
|
&'a Inventory,
|
|
|
|
&'a Energy,
|
|
|
|
&'a SkillSet,
|
2021-11-12 03:37:37 +00:00
|
|
|
&'a ActiveAbilities,
|
2021-11-10 02:20:41 +00:00
|
|
|
&'a Body,
|
|
|
|
);
|
2020-04-11 06:33:06 +00:00
|
|
|
type HotbarImageSource<'a> = (&'a ItemImgs, &'a img_ids::Imgs);
|
|
|
|
|
|
|
|
impl<'a> SlotKey<HotbarSource<'a>, HotbarImageSource<'a>> for HotbarSlot {
|
|
|
|
type ImageKey = HotbarImage;
|
|
|
|
|
|
|
|
fn image_key(
|
|
|
|
&self,
|
2021-11-12 03:37:37 +00:00
|
|
|
(hotbar, inventory, energy, skillset, active_abilities, body): &HotbarSource<'a>,
|
2020-04-11 06:33:06 +00:00
|
|
|
) -> Option<(Self::ImageKey, Option<Color>)> {
|
2021-12-16 18:37:15 +00:00
|
|
|
const GREYED_OUT: Color = Color::Rgba(0.3, 0.3, 0.3, 0.8);
|
2020-04-11 06:33:06 +00:00
|
|
|
hotbar.get(*self).and_then(|contents| match contents {
|
2021-12-16 18:37:15 +00:00
|
|
|
hotbar::SlotContents::Inventory(item_hash, item_key) => {
|
|
|
|
let item = inventory.get_by_hash(item_hash);
|
|
|
|
match item {
|
|
|
|
Some(item) => Some((HotbarImage::Item(item.into()), None)),
|
|
|
|
None => Some((HotbarImage::Item(item_key), Some(GREYED_OUT))),
|
|
|
|
}
|
|
|
|
},
|
2021-11-10 02:20:41 +00:00
|
|
|
hotbar::SlotContents::Ability(i) => {
|
2021-11-12 03:37:37 +00:00
|
|
|
let ability_id = active_abilities
|
2021-11-10 02:20:41 +00:00
|
|
|
.abilities
|
|
|
|
.get(i)
|
2021-11-11 22:55:14 +00:00
|
|
|
.and_then(|a| Ability::from(*a).ability_id(Some(inventory)));
|
2021-11-10 22:25:37 +00:00
|
|
|
|
|
|
|
ability_id
|
|
|
|
.map(|id| HotbarImage::Ability(id.to_string()))
|
2021-11-10 02:20:41 +00:00
|
|
|
.and_then(|image| {
|
2021-11-12 03:37:37 +00:00
|
|
|
active_abilities
|
2021-11-10 02:20:41 +00:00
|
|
|
.activate_ability(
|
2021-11-11 22:55:14 +00:00
|
|
|
AbilityInput::Auxiliary(i),
|
2021-11-10 02:20:41 +00:00
|
|
|
Some(inventory),
|
|
|
|
skillset,
|
2021-11-12 03:37:37 +00:00
|
|
|
Some(body),
|
2021-11-10 02:20:41 +00:00
|
|
|
)
|
|
|
|
.map(|(ability, _)| {
|
|
|
|
(
|
|
|
|
image,
|
|
|
|
if energy.current() > ability.get_energy_cost() {
|
|
|
|
Some(Color::Rgba(1.0, 1.0, 1.0, 1.0))
|
|
|
|
} else {
|
2021-12-16 18:37:15 +00:00
|
|
|
Some(GREYED_OUT)
|
2021-11-10 02:20:41 +00:00
|
|
|
},
|
|
|
|
)
|
|
|
|
})
|
2021-02-13 00:43:33 +00:00
|
|
|
})
|
|
|
|
},
|
2020-04-11 06:33:06 +00:00
|
|
|
})
|
2020-04-06 15:25:45 +00:00
|
|
|
}
|
|
|
|
|
2021-11-10 02:20:41 +00:00
|
|
|
fn amount(&self, (hotbar, inventory, _, _, _, _): &HotbarSource<'a>) -> Option<u32> {
|
2020-04-11 06:33:06 +00:00
|
|
|
hotbar
|
|
|
|
.get(*self)
|
|
|
|
.and_then(|content| match content {
|
2021-12-16 18:37:15 +00:00
|
|
|
hotbar::SlotContents::Inventory(item_hash, _) => inventory.get_by_hash(item_hash),
|
2021-11-10 02:20:41 +00:00
|
|
|
hotbar::SlotContents::Ability(_) => None,
|
2020-04-11 06:33:06 +00:00
|
|
|
})
|
2020-09-17 23:02:14 +00:00
|
|
|
.map(|item| item.amount())
|
2020-04-06 15:25:45 +00:00
|
|
|
.filter(|amount| *amount > 1)
|
|
|
|
}
|
|
|
|
|
2021-02-16 01:05:54 +00:00
|
|
|
fn image_ids(
|
|
|
|
key: &Self::ImageKey,
|
|
|
|
(item_imgs, imgs): &HotbarImageSource<'a>,
|
|
|
|
) -> Vec<image::Id> {
|
2020-04-11 06:33:06 +00:00
|
|
|
match key {
|
2021-02-16 01:05:54 +00:00
|
|
|
HotbarImage::Item(key) => item_imgs.img_ids_or_not_found_img(key.clone()),
|
2021-11-10 22:25:37 +00:00
|
|
|
HotbarImage::Ability(ability_id) => vec![util::ability_image(imgs, ability_id)],
|
2020-04-11 06:33:06 +00:00
|
|
|
}
|
2020-04-06 15:25:45 +00:00
|
|
|
}
|
2020-04-11 06:33:06 +00:00
|
|
|
}
|
2020-04-06 15:25:45 +00:00
|
|
|
|
|
|
|
impl From<InventorySlot> for SlotKind {
|
2020-04-04 05:40:00 +00:00
|
|
|
fn from(inventory: InventorySlot) -> Self { Self::Inventory(inventory) }
|
|
|
|
}
|
|
|
|
|
2020-04-10 02:36:35 +00:00
|
|
|
impl From<EquipSlot> for SlotKind {
|
|
|
|
fn from(equip: EquipSlot) -> Self { Self::Equip(equip) }
|
2020-04-04 05:40:00 +00:00
|
|
|
}
|
|
|
|
|
2020-04-11 06:33:06 +00:00
|
|
|
impl From<HotbarSlot> for SlotKind {
|
|
|
|
fn from(hotbar: HotbarSlot) -> Self { Self::Hotbar(hotbar) }
|
|
|
|
}
|
2021-02-12 02:53:25 +00:00
|
|
|
impl From<TradeSlot> for SlotKind {
|
|
|
|
fn from(trade: TradeSlot) -> Self { Self::Trade(trade) }
|
|
|
|
}
|
2020-04-04 05:40:00 +00:00
|
|
|
|
2020-04-06 15:25:45 +00:00
|
|
|
impl SumSlot for SlotKind {}
|