mirror of
https://gitlab.com/veloren/veloren.git
synced 2024-08-30 18:12:32 +00:00
244 lines
8.2 KiB
Rust
244 lines
8.2 KiB
Rust
use super::Show;
|
|
use crate::ui::fonts::Fonts;
|
|
use client::{self, Client};
|
|
use common_net::msg::Notification;
|
|
use conrod_core::{
|
|
widget::{self, Text},
|
|
widget_ids, Color, Colorable, Positionable, Widget, WidgetCommon,
|
|
};
|
|
use i18n::Localization;
|
|
use std::{collections::VecDeque, time::Instant};
|
|
|
|
widget_ids! {
|
|
struct Ids {
|
|
error_bg,
|
|
error_text,
|
|
info_bg,
|
|
info_text,
|
|
message_bg,
|
|
message_text,
|
|
}
|
|
}
|
|
|
|
#[derive(WidgetCommon)]
|
|
pub struct Popup<'a> {
|
|
i18n: &'a Localization,
|
|
client: &'a Client,
|
|
new_notifications: &'a VecDeque<Notification>,
|
|
fonts: &'a Fonts,
|
|
#[conrod(common_builder)]
|
|
common: widget::CommonBuilder,
|
|
show: &'a Show,
|
|
}
|
|
|
|
/// Popup notifications for messages such as <Chunk Name>, Waypoint Saved,
|
|
/// Dungeon Cleared (TODO), and Quest Completed (TODO)
|
|
impl<'a> Popup<'a> {
|
|
pub fn new(
|
|
i18n: &'a Localization,
|
|
client: &'a Client,
|
|
new_notifications: &'a VecDeque<Notification>,
|
|
fonts: &'a Fonts,
|
|
show: &'a Show,
|
|
) -> Self {
|
|
Self {
|
|
i18n,
|
|
client,
|
|
new_notifications,
|
|
fonts,
|
|
common: widget::CommonBuilder::default(),
|
|
show,
|
|
}
|
|
}
|
|
}
|
|
|
|
pub struct State {
|
|
ids: Ids,
|
|
errors: VecDeque<String>,
|
|
infos: VecDeque<String>,
|
|
messages: VecDeque<String>,
|
|
last_error_update: Instant,
|
|
last_info_update: Instant,
|
|
last_message_update: Instant,
|
|
last_region_name: Option<String>,
|
|
}
|
|
|
|
impl<'a> Widget for Popup<'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),
|
|
errors: VecDeque::new(),
|
|
infos: VecDeque::new(),
|
|
messages: VecDeque::new(),
|
|
last_error_update: Instant::now(),
|
|
last_info_update: Instant::now(),
|
|
last_message_update: Instant::now(),
|
|
last_region_name: None,
|
|
}
|
|
}
|
|
|
|
fn style(&self) -> Self::Style {}
|
|
|
|
fn update(self, args: widget::UpdateArgs<Self>) -> Self::Event {
|
|
common_base::prof_span!("Popup::update");
|
|
let widget::UpdateArgs { state, ui, .. } = args;
|
|
|
|
const FADE_IN: f32 = 0.5;
|
|
const FADE_HOLD: f32 = 1.0;
|
|
const FADE_OUT: f32 = 3.0;
|
|
|
|
let bg_color = |fade| Color::Rgba(0.0, 0.0, 0.0, fade);
|
|
let error_color = |fade| Color::Rgba(1.0, 0.0, 0.0, fade);
|
|
let info_color = |fade| Color::Rgba(1.0, 1.0, 0.0, fade);
|
|
let message_color = |fade| Color::Rgba(1.0, 1.0, 1.0, fade);
|
|
|
|
// Push chunk name to message queue
|
|
if let Some(chunk) = self.client.current_chunk() {
|
|
let current = chunk.meta().name();
|
|
// Check if no other popup is displayed and a new one is needed
|
|
if state.messages.is_empty()
|
|
&& state
|
|
.last_region_name
|
|
.as_ref()
|
|
.map(|l| l != current)
|
|
.unwrap_or(true)
|
|
{
|
|
// Update last_region
|
|
state.update(|s| {
|
|
if s.messages.is_empty() {
|
|
s.last_message_update = Instant::now();
|
|
}
|
|
s.last_region_name = Some(current.to_owned());
|
|
s.messages.push_back(current.to_owned());
|
|
});
|
|
}
|
|
}
|
|
|
|
// Push waypoint to message queue
|
|
for notification in self.new_notifications {
|
|
match notification {
|
|
Notification::WaypointSaved => {
|
|
state.update(|s| {
|
|
if s.infos.is_empty() {
|
|
s.last_info_update = Instant::now();
|
|
}
|
|
let text = self.i18n.get("hud.waypoint_saved");
|
|
s.infos.push_back(text.to_string());
|
|
});
|
|
},
|
|
}
|
|
}
|
|
|
|
// Get next error from queue
|
|
if !state.errors.is_empty()
|
|
&& state.last_error_update.elapsed().as_secs_f32() > FADE_IN + FADE_HOLD + FADE_OUT
|
|
{
|
|
state.update(|s| {
|
|
s.errors.pop_front();
|
|
s.last_error_update = Instant::now();
|
|
});
|
|
}
|
|
|
|
// Display error as popup
|
|
if let Some(error) = state.errors.front() {
|
|
let seconds = state.last_error_update.elapsed().as_secs_f32();
|
|
let fade = if seconds < FADE_IN {
|
|
seconds / FADE_IN
|
|
} else if seconds < FADE_IN + FADE_HOLD {
|
|
1.0
|
|
} else {
|
|
(1.0 - (seconds - FADE_IN - FADE_HOLD) / FADE_OUT).max(0.0)
|
|
};
|
|
Text::new(error)
|
|
.mid_top_with_margin_on(ui.window, 50.0)
|
|
.font_size(self.fonts.cyri.scale(20))
|
|
.font_id(self.fonts.cyri.conrod_id)
|
|
.color(bg_color(fade))
|
|
.set(state.ids.error_bg, ui);
|
|
Text::new(error)
|
|
.top_left_with_margins_on(state.ids.error_bg, -1.0, -1.0)
|
|
.font_size(self.fonts.cyri.scale(20))
|
|
.font_id(self.fonts.cyri.conrod_id)
|
|
.color(error_color(fade))
|
|
.set(state.ids.error_text, ui);
|
|
}
|
|
|
|
// Get next info from queue
|
|
if !state.infos.is_empty()
|
|
&& state.last_info_update.elapsed().as_secs_f32() > FADE_IN + FADE_HOLD + FADE_OUT
|
|
{
|
|
state.update(|s| {
|
|
s.infos.pop_front();
|
|
s.last_info_update = Instant::now();
|
|
});
|
|
}
|
|
|
|
// Display info as popup
|
|
if !self.show.intro {
|
|
if let Some(info) = state.infos.front() {
|
|
let seconds = state.last_info_update.elapsed().as_secs_f32();
|
|
let fade = if seconds < FADE_IN {
|
|
seconds / FADE_IN
|
|
} else if seconds < FADE_IN + FADE_HOLD {
|
|
1.0
|
|
} else {
|
|
(1.0 - (seconds - FADE_IN - FADE_HOLD) / FADE_OUT).max(0.0)
|
|
};
|
|
|
|
Text::new(info)
|
|
.mid_top_with_margin_on(ui.window, 100.0)
|
|
.font_size(self.fonts.cyri.scale(20))
|
|
.font_id(self.fonts.cyri.conrod_id)
|
|
.color(bg_color(fade))
|
|
.set(state.ids.info_bg, ui);
|
|
Text::new(info)
|
|
.top_left_with_margins_on(state.ids.info_bg, -1.0, -1.0)
|
|
.font_size(self.fonts.cyri.scale(20))
|
|
.font_id(self.fonts.cyri.conrod_id)
|
|
.color(info_color(fade))
|
|
.set(state.ids.info_text, ui);
|
|
}
|
|
}
|
|
|
|
// Get next message from queue
|
|
if !state.messages.is_empty()
|
|
&& state.last_message_update.elapsed().as_secs_f32() > FADE_IN + FADE_HOLD + FADE_OUT
|
|
{
|
|
state.update(|s| {
|
|
s.messages.pop_front();
|
|
s.last_message_update = Instant::now();
|
|
});
|
|
}
|
|
|
|
// Display message as popup
|
|
if !self.show.intro {
|
|
if let Some(message) = state.messages.front() {
|
|
let seconds = state.last_message_update.elapsed().as_secs_f32();
|
|
let fade = if seconds < FADE_IN {
|
|
seconds / FADE_IN
|
|
} else if seconds < FADE_IN + FADE_HOLD {
|
|
1.0
|
|
} else {
|
|
(1.0 - (seconds - FADE_IN - FADE_HOLD) / FADE_OUT).max(0.0)
|
|
};
|
|
Text::new(message)
|
|
.mid_top_with_margin_on(ui.window, 200.0)
|
|
.font_size(self.fonts.alkhemi.scale(70))
|
|
.font_id(self.fonts.alkhemi.conrod_id)
|
|
.color(bg_color(fade))
|
|
.set(state.ids.message_bg, ui);
|
|
Text::new(message)
|
|
.top_left_with_margins_on(state.ids.message_bg, -2.5, -2.5)
|
|
.font_size(self.fonts.alkhemi.scale(70))
|
|
.font_id(self.fonts.alkhemi.conrod_id)
|
|
.color(message_color(fade))
|
|
.set(state.ids.message_text, ui);
|
|
}
|
|
}
|
|
}
|
|
}
|