2020-09-19 18:34:06 +00:00
|
|
|
use crate::{
|
2021-07-21 11:23:45 +00:00
|
|
|
game_input::GameInput,
|
2020-09-19 18:34:06 +00:00
|
|
|
settings::ControlSettings,
|
2020-05-25 18:11:39 +00:00
|
|
|
ui::{fonts::Fonts, Ingameable},
|
2020-09-19 18:34:06 +00:00
|
|
|
};
|
2020-07-29 13:21:12 +00:00
|
|
|
use conrod_core::{
|
2021-05-22 20:47:08 +00:00
|
|
|
color,
|
2020-09-19 18:34:06 +00:00
|
|
|
widget::{self, RoundedRectangle, Text},
|
2022-01-16 17:06:35 +00:00
|
|
|
widget_ids, Color, Colorable, Positionable, Sizeable, Widget, WidgetCommon,
|
2020-07-29 13:21:12 +00:00
|
|
|
};
|
2021-07-29 18:47:45 +00:00
|
|
|
use i18n::Localization;
|
2021-03-30 00:04:23 +00:00
|
|
|
use std::borrow::Cow;
|
|
|
|
|
2022-05-28 12:06:49 +00:00
|
|
|
use crate::hud::{CollectFailedData, HudCollectFailedReason, HudLootOwner};
|
2021-03-31 12:56:56 +00:00
|
|
|
use keyboard_keynames::key_layout::KeyLayout;
|
2020-07-29 13:21:12 +00:00
|
|
|
|
2021-05-22 20:47:08 +00:00
|
|
|
pub const TEXT_COLOR: Color = Color::Rgba(0.61, 0.61, 0.89, 1.0);
|
|
|
|
pub const PICKUP_FAILED_FADE_OUT_TIME: f32 = 1.5;
|
|
|
|
|
2020-07-29 13:21:12 +00:00
|
|
|
widget_ids! {
|
|
|
|
struct Ids {
|
|
|
|
// Name
|
|
|
|
name_bg,
|
|
|
|
name,
|
2022-01-16 13:57:38 +00:00
|
|
|
// Interaction hints
|
2020-09-19 18:34:06 +00:00
|
|
|
btn_bg,
|
|
|
|
btn,
|
2021-05-22 20:47:08 +00:00
|
|
|
// Inventory full
|
|
|
|
inv_full_bg,
|
|
|
|
inv_full,
|
2020-07-29 13:21:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-19 18:34:06 +00:00
|
|
|
/// UI widget containing everything that goes over a item
|
2020-07-29 13:21:12 +00:00
|
|
|
/// (Item, DistanceFromPlayer, Rarity, etc.)
|
|
|
|
#[derive(WidgetCommon)]
|
|
|
|
pub struct Overitem<'a> {
|
2021-03-30 00:04:23 +00:00
|
|
|
name: Cow<'a, str>,
|
|
|
|
quality: Color,
|
|
|
|
distance_from_player_sqr: f32,
|
2020-05-25 18:11:39 +00:00
|
|
|
fonts: &'a Fonts,
|
2021-05-22 20:47:08 +00:00
|
|
|
localized_strings: &'a Localization,
|
2020-09-19 18:34:06 +00:00
|
|
|
controls: &'a ControlSettings,
|
2020-07-29 13:21:12 +00:00
|
|
|
#[conrod(common_builder)]
|
|
|
|
common: widget::CommonBuilder,
|
2021-05-22 20:47:08 +00:00
|
|
|
properties: OveritemProperties,
|
|
|
|
pulse: f32,
|
2021-03-31 12:56:56 +00:00
|
|
|
key_layout: &'a Option<KeyLayout>,
|
2023-03-12 02:07:10 +00:00
|
|
|
// GameInput optional so we can just show stuff like "needs pickaxe"
|
|
|
|
interaction_options: Vec<(Option<GameInput>, String)>,
|
2020-07-29 13:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Overitem<'a> {
|
2020-09-19 18:34:06 +00:00
|
|
|
pub fn new(
|
2021-03-30 00:04:23 +00:00
|
|
|
name: Cow<'a, str>,
|
|
|
|
quality: Color,
|
|
|
|
distance_from_player_sqr: f32,
|
2020-05-25 18:11:39 +00:00
|
|
|
fonts: &'a Fonts,
|
2021-05-22 20:47:08 +00:00
|
|
|
localized_strings: &'a Localization,
|
2020-09-19 18:34:06 +00:00
|
|
|
controls: &'a ControlSettings,
|
2021-05-22 20:47:08 +00:00
|
|
|
properties: OveritemProperties,
|
|
|
|
pulse: f32,
|
2021-03-31 12:56:56 +00:00
|
|
|
key_layout: &'a Option<KeyLayout>,
|
2023-03-12 02:07:10 +00:00
|
|
|
interaction_options: Vec<(Option<GameInput>, String)>,
|
2020-09-19 18:34:06 +00:00
|
|
|
) -> Self {
|
2020-07-29 13:21:12 +00:00
|
|
|
Self {
|
|
|
|
name,
|
2021-03-02 00:45:02 +00:00
|
|
|
quality,
|
2020-09-19 18:34:06 +00:00
|
|
|
distance_from_player_sqr,
|
2020-07-29 13:21:12 +00:00
|
|
|
fonts,
|
2021-05-22 20:47:08 +00:00
|
|
|
localized_strings,
|
2020-09-19 18:34:06 +00:00
|
|
|
controls,
|
2020-07-29 13:21:12 +00:00
|
|
|
common: widget::CommonBuilder::default(),
|
2021-05-22 20:47:08 +00:00
|
|
|
properties,
|
|
|
|
pulse,
|
2021-03-31 12:56:56 +00:00
|
|
|
key_layout,
|
2022-01-16 13:57:38 +00:00
|
|
|
interaction_options,
|
2020-07-29 13:21:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-22 20:47:08 +00:00
|
|
|
pub struct OveritemProperties {
|
|
|
|
pub active: bool,
|
2022-05-28 12:06:49 +00:00
|
|
|
pub pickup_failed_pulse: Option<CollectFailedData>,
|
2021-05-22 20:47:08 +00:00
|
|
|
}
|
|
|
|
|
2020-07-29 13:21:12 +00:00
|
|
|
pub struct State {
|
|
|
|
ids: Ids,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Ingameable for Overitem<'a> {
|
|
|
|
fn prim_count(&self) -> usize {
|
2020-09-19 18:34:06 +00:00
|
|
|
// Number of conrod primitives contained in the overitem display.
|
|
|
|
// TODO maybe this could be done automatically?
|
|
|
|
// - 2 Text for name
|
|
|
|
// - 0 or 2 Rectangle and Text for button
|
2021-04-15 20:45:09 +00:00
|
|
|
2 + match self
|
|
|
|
.controls
|
|
|
|
.get_binding(GameInput::Interact)
|
2021-05-22 20:47:08 +00:00
|
|
|
.filter(|_| self.properties.active)
|
2021-04-15 20:45:09 +00:00
|
|
|
{
|
2020-09-19 18:34:06 +00:00
|
|
|
Some(_) => 2,
|
|
|
|
None => 0,
|
2021-05-22 20:47:08 +00:00
|
|
|
} + if self.properties.pickup_failed_pulse.is_some() {
|
|
|
|
2
|
|
|
|
} else {
|
|
|
|
0
|
2020-09-19 18:34:06 +00:00
|
|
|
}
|
2020-07-29 13:21:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Widget for Overitem<'a> {
|
|
|
|
type Event = ();
|
|
|
|
type State = State;
|
|
|
|
type Style = ();
|
|
|
|
|
|
|
|
fn init_state(&self, id_gen: widget::id::Generator) -> Self::State {
|
|
|
|
State {
|
|
|
|
ids: Ids::new(id_gen),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn style(&self) -> Self::Style {}
|
|
|
|
|
|
|
|
fn update(self, args: widget::UpdateArgs<Self>) -> Self::Event {
|
2020-09-19 18:34:06 +00:00
|
|
|
let widget::UpdateArgs { id, state, ui, .. } = args;
|
|
|
|
|
2022-01-16 13:57:38 +00:00
|
|
|
let btn_color = Color::Rgba(0.0, 0.0, 0.0, 0.8);
|
2020-07-29 13:21:12 +00:00
|
|
|
|
2020-09-19 18:34:06 +00:00
|
|
|
// Example:
|
|
|
|
// MUSHROOM
|
|
|
|
// ___
|
|
|
|
// | E |
|
|
|
|
// ———
|
2020-07-29 13:21:12 +00:00
|
|
|
|
2020-11-26 20:11:49 +00:00
|
|
|
// Scale at max distance is 10, and at min distance is 30. Disabled since the
|
|
|
|
// scaling ruins glyph caching, causing performance issues near lootbags
|
|
|
|
// let scale: f64 = ((1.5
|
|
|
|
// - (self.distance_from_player_sqr /
|
|
|
|
// common::consts::MAX_PICKUP_RANGE.powi(2)))
|
|
|
|
// * 20.0)
|
|
|
|
// .into();
|
|
|
|
let scale = 30.0;
|
2021-05-22 20:47:08 +00:00
|
|
|
|
2020-09-19 18:34:06 +00:00
|
|
|
let text_font_size = scale * 1.0;
|
|
|
|
let text_pos_y = scale * 1.2;
|
2021-05-22 20:47:08 +00:00
|
|
|
|
2020-09-19 18:34:06 +00:00
|
|
|
let btn_rect_size = scale * 0.8;
|
|
|
|
let btn_font_size = scale * 0.6;
|
|
|
|
let btn_rect_pos_y = 0.0;
|
|
|
|
let btn_text_pos_y = btn_rect_pos_y + ((btn_rect_size - btn_font_size) * 0.5);
|
|
|
|
let btn_radius = btn_rect_size / 5.0;
|
|
|
|
|
2021-05-22 20:47:08 +00:00
|
|
|
let inv_full_font_size = scale * 1.0;
|
|
|
|
let inv_full_pos_y = scale * 2.4;
|
|
|
|
|
2020-09-19 18:34:06 +00:00
|
|
|
// Item Name
|
2020-07-29 13:21:12 +00:00
|
|
|
Text::new(&self.name)
|
|
|
|
.font_id(self.fonts.cyri.conrod_id)
|
2020-09-19 18:34:06 +00:00
|
|
|
.font_size(text_font_size as u32)
|
2020-07-29 13:21:12 +00:00
|
|
|
.color(Color::Rgba(0.0, 0.0, 0.0, 1.0))
|
2020-09-19 18:34:06 +00:00
|
|
|
.x_y(-1.0, text_pos_y - 2.0)
|
|
|
|
.parent(id)
|
|
|
|
.depth(self.distance_from_player_sqr + 4.0)
|
2020-07-29 13:21:12 +00:00
|
|
|
.set(state.ids.name_bg, ui);
|
|
|
|
Text::new(&self.name)
|
|
|
|
.font_id(self.fonts.cyri.conrod_id)
|
2020-09-19 18:34:06 +00:00
|
|
|
.font_size(text_font_size as u32)
|
2021-03-30 00:04:23 +00:00
|
|
|
.color(self.quality)
|
2020-09-19 18:34:06 +00:00
|
|
|
.x_y(0.0, text_pos_y)
|
|
|
|
.depth(self.distance_from_player_sqr + 3.0)
|
|
|
|
.parent(id)
|
2020-07-29 13:21:12 +00:00
|
|
|
.set(state.ids.name, ui);
|
2020-09-19 18:34:06 +00:00
|
|
|
|
2022-01-16 13:57:38 +00:00
|
|
|
// Interaction hints
|
2022-02-27 21:50:07 +00:00
|
|
|
if !self.interaction_options.is_empty() && self.properties.active {
|
2022-01-16 17:06:35 +00:00
|
|
|
let text = self
|
|
|
|
.interaction_options
|
2022-01-16 13:57:38 +00:00
|
|
|
.iter()
|
2022-01-16 17:06:35 +00:00
|
|
|
.filter_map(|(input, action)| {
|
2023-03-12 02:07:10 +00:00
|
|
|
let binding = if let Some(input) = input {
|
|
|
|
Some(self.controls.get_binding(*input)?)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
};
|
|
|
|
Some((binding, action))
|
2022-01-16 17:06:35 +00:00
|
|
|
})
|
|
|
|
.map(|(input, action)| {
|
2023-03-12 02:07:10 +00:00
|
|
|
if let Some(input) = input {
|
|
|
|
let input = input.display_string(self.key_layout);
|
|
|
|
format!("{} {action}", input.as_str())
|
|
|
|
} else {
|
|
|
|
action.to_string()
|
|
|
|
}
|
2022-01-16 17:06:35 +00:00
|
|
|
})
|
2022-01-16 13:57:38 +00:00
|
|
|
.collect::<Vec<_>>()
|
|
|
|
.join("\n");
|
|
|
|
|
|
|
|
let hints_text = Text::new(&text)
|
2020-09-19 18:34:06 +00:00
|
|
|
.font_id(self.fonts.cyri.conrod_id)
|
|
|
|
.font_size(btn_font_size as u32)
|
2021-03-30 00:04:23 +00:00
|
|
|
.color(TEXT_COLOR)
|
2020-09-19 18:34:06 +00:00
|
|
|
.x_y(0.0, btn_text_pos_y)
|
2022-01-16 13:57:38 +00:00
|
|
|
.depth(self.distance_from_player_sqr + 1.0)
|
|
|
|
.parent(id);
|
|
|
|
|
|
|
|
let [w, h] = hints_text.get_wh(ui).unwrap_or([btn_rect_size; 2]);
|
|
|
|
|
|
|
|
hints_text.set(state.ids.btn, ui);
|
|
|
|
|
2022-01-16 17:06:35 +00:00
|
|
|
RoundedRectangle::fill_with(
|
|
|
|
[w + btn_radius * 2.0, h + btn_radius * 2.0],
|
|
|
|
btn_radius,
|
|
|
|
btn_color,
|
|
|
|
)
|
|
|
|
.x_y(0.0, btn_rect_pos_y)
|
|
|
|
.depth(self.distance_from_player_sqr + 2.0)
|
|
|
|
.parent(id)
|
|
|
|
.set(state.ids.btn_bg, ui);
|
2020-09-19 18:34:06 +00:00
|
|
|
}
|
2022-05-28 12:06:49 +00:00
|
|
|
if let Some(collect_failed_data) = self.properties.pickup_failed_pulse {
|
2021-05-22 20:47:08 +00:00
|
|
|
//should never exceed 1.0, but just in case
|
2022-05-28 12:06:49 +00:00
|
|
|
let age = ((self.pulse - collect_failed_data.pulse) / PICKUP_FAILED_FADE_OUT_TIME)
|
|
|
|
.clamp(0.0, 1.0);
|
2021-05-22 20:47:08 +00:00
|
|
|
|
|
|
|
let alpha = 1.0 - age.powi(4);
|
|
|
|
let brightness = 1.0 / (age / 0.07 - 1.0).abs().clamp(0.01, 1.0);
|
|
|
|
let shade_color = |color: Color| {
|
|
|
|
let color::Hsla(hue, sat, lum, alp) = color.to_hsl();
|
|
|
|
color::hsla(hue, sat / brightness, lum * brightness.sqrt(), alp * alpha)
|
|
|
|
};
|
|
|
|
|
2022-05-28 12:06:49 +00:00
|
|
|
let text = match collect_failed_data.reason {
|
|
|
|
HudCollectFailedReason::InventoryFull => {
|
2022-08-23 11:14:29 +00:00
|
|
|
self.localized_strings.get_msg("hud-inventory_full")
|
2022-05-28 12:06:49 +00:00
|
|
|
},
|
|
|
|
HudCollectFailedReason::LootOwned { owner, expiry_secs } => {
|
|
|
|
let owner_name = match owner {
|
2022-08-06 17:47:45 +00:00
|
|
|
HudLootOwner::Name(name) => Cow::Owned(name),
|
2022-08-23 11:14:29 +00:00
|
|
|
HudLootOwner::Group => self.localized_strings.get_msg("hud-another_group"),
|
|
|
|
HudLootOwner::Unknown => self.localized_strings.get_msg("hud-someone_else"),
|
2022-05-28 12:06:49 +00:00
|
|
|
};
|
2022-08-06 17:47:45 +00:00
|
|
|
self.localized_strings.get_msg_ctx(
|
|
|
|
"hud-owned_by_for_secs",
|
|
|
|
&i18n::fluent_args! {
|
|
|
|
"name" => owner_name,
|
|
|
|
"secs" => expiry_secs,
|
|
|
|
},
|
|
|
|
)
|
2022-05-28 12:06:49 +00:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
Text::new(&text)
|
2021-05-22 20:47:08 +00:00
|
|
|
.font_id(self.fonts.cyri.conrod_id)
|
|
|
|
.font_size(inv_full_font_size as u32)
|
|
|
|
.color(shade_color(Color::Rgba(0.0, 0.0, 0.0, 1.0)))
|
|
|
|
.x_y(-1.0, inv_full_pos_y - 2.0)
|
|
|
|
.parent(id)
|
|
|
|
.depth(self.distance_from_player_sqr + 6.0)
|
|
|
|
.set(state.ids.inv_full_bg, ui);
|
|
|
|
|
2022-05-28 12:06:49 +00:00
|
|
|
Text::new(&text)
|
2021-05-22 20:47:08 +00:00
|
|
|
.font_id(self.fonts.cyri.conrod_id)
|
|
|
|
.font_size(inv_full_font_size as u32)
|
|
|
|
.color(shade_color(Color::Rgba(1.0, 0.0, 0.0, 1.0)))
|
|
|
|
.x_y(0.0, inv_full_pos_y)
|
|
|
|
.parent(id)
|
|
|
|
.depth(self.distance_from_player_sqr + 5.0)
|
|
|
|
.set(state.ids.inv_full, ui);
|
|
|
|
}
|
2020-07-29 13:21:12 +00:00
|
|
|
}
|
|
|
|
}
|