2019-05-04 10:43:37 +00:00
|
|
|
mod bag;
|
2019-05-07 05:40:03 +00:00
|
|
|
mod buttons;
|
|
|
|
mod character_window;
|
|
|
|
mod chat;
|
2019-05-03 16:56:18 +00:00
|
|
|
mod esc_menu;
|
2019-05-07 05:40:03 +00:00
|
|
|
mod img_ids;
|
2019-10-09 19:28:05 +00:00
|
|
|
mod item_imgs;
|
2019-05-07 05:40:03 +00:00
|
|
|
mod map;
|
2019-05-07 17:50:53 +00:00
|
|
|
mod minimap;
|
2019-08-07 13:14:26 +00:00
|
|
|
mod quest;
|
2019-05-07 05:40:03 +00:00
|
|
|
mod settings_window;
|
|
|
|
mod skillbar;
|
2019-08-07 13:14:26 +00:00
|
|
|
mod social;
|
|
|
|
mod spell;
|
2019-04-26 20:24:05 +00:00
|
|
|
|
2019-09-25 16:51:47 +00:00
|
|
|
use crate::hud::img_ids::ImgsRot;
|
2019-07-26 02:28:53 +00:00
|
|
|
pub use settings_window::ScaleChange;
|
|
|
|
|
2019-05-04 10:43:37 +00:00
|
|
|
use bag::Bag;
|
|
|
|
use buttons::Buttons;
|
2019-05-07 05:40:03 +00:00
|
|
|
use character_window::CharacterWindow;
|
|
|
|
use chat::Chat;
|
2019-09-01 19:04:03 +00:00
|
|
|
use chrono::NaiveTime;
|
2019-05-03 16:56:18 +00:00
|
|
|
use esc_menu::EscMenu;
|
2019-05-07 05:40:03 +00:00
|
|
|
use img_ids::Imgs;
|
2019-10-09 19:28:05 +00:00
|
|
|
use item_imgs::ItemImgs;
|
2019-05-07 05:40:03 +00:00
|
|
|
use map::Map;
|
2019-05-07 17:50:53 +00:00
|
|
|
use minimap::MiniMap;
|
2019-08-07 13:14:26 +00:00
|
|
|
use quest::Quest;
|
2019-07-23 01:02:57 +00:00
|
|
|
use serde::{Deserialize, Serialize};
|
2019-06-02 02:17:36 +00:00
|
|
|
use settings_window::{SettingsTab, SettingsWindow};
|
2019-05-07 05:40:03 +00:00
|
|
|
use skillbar::Skillbar;
|
2019-08-07 13:14:26 +00:00
|
|
|
use social::{Social, SocialTab};
|
|
|
|
use spell::Spell;
|
2019-03-15 04:55:52 +00:00
|
|
|
|
|
|
|
use crate::{
|
2019-09-26 07:28:40 +00:00
|
|
|
render::{AaMode, Consts, Globals, Renderer},
|
2019-05-20 06:09:20 +00:00
|
|
|
scene::camera::Camera,
|
2019-10-06 19:19:08 +00:00
|
|
|
//settings::ControlSettings,
|
2019-10-16 11:39:41 +00:00
|
|
|
ui::{Graphic, Ingameable, ScaleMode, Ui},
|
2019-07-26 02:28:53 +00:00
|
|
|
window::{Event as WinEvent, GameInput},
|
2019-04-20 18:08:39 +00:00
|
|
|
GlobalState,
|
2019-03-15 04:55:52 +00:00
|
|
|
};
|
2019-07-17 22:10:42 +00:00
|
|
|
use client::{Client, Event as ClientEvent};
|
common: Rework volume API
See the doc comments in `common/src/vol.rs` for more information on
the API itself.
The changes include:
* Consistent `Err`/`Error` naming.
* Types are named `...Error`.
* `enum` variants are named `...Err`.
* Rename `VolMap{2d, 3d}` -> `VolGrid{2d, 3d}`. This is in preparation
to an upcoming change where a “map” in the game related sense will
be added.
* Add volume iterators. There are two types of them:
* _Position_ iterators obtained from the trait `IntoPosIterator`
using the method
`fn pos_iter(self, lower_bound: Vec3<i32>, upper_bound: Vec3<i32>) -> ...`
which returns an iterator over `Vec3<i32>`.
* _Volume_ iterators obtained from the trait `IntoVolIterator`
using the method
`fn vol_iter(self, lower_bound: Vec3<i32>, upper_bound: Vec3<i32>) -> ...`
which returns an iterator over `(Vec3<i32>, &Self::Vox)`.
Those traits will usually be implemented by references to volume
types (i.e. `impl IntoVolIterator<'a> for &'a T` where `T` is some
type which usually implements several volume traits, such as `Chunk`).
* _Position_ iterators iterate over the positions valid for that
volume.
* _Volume_ iterators do the same but return not only the position
but also the voxel at that position, in each iteration.
* Introduce trait `RectSizedVol` for the use case which we have with
`Chonk`: A `Chonk` is sized only in x and y direction.
* Introduce traits `RasterableVol`, `RectRasterableVol`
* `RasterableVol` represents a volume that is compile-time sized and has
its lower bound at `(0, 0, 0)`. The name `RasterableVol` was chosen
because such a volume can be used with `VolGrid3d`.
* `RectRasterableVol` represents a volume that is compile-time sized at
least in x and y direction and has its lower bound at `(0, 0, z)`.
There's no requirement on he lower bound or size in z direction.
The name `RectRasterableVol` was chosen because such a volume can be
used with `VolGrid2d`.
2019-09-03 22:23:29 +00:00
|
|
|
use common::{comp, terrain::TerrainChunk, vol::RectRasterableVol};
|
2019-03-15 04:55:52 +00:00
|
|
|
use conrod_core::{
|
2019-10-16 11:39:41 +00:00
|
|
|
image::Id,
|
2019-07-05 16:21:11 +00:00
|
|
|
text::cursor::Index,
|
2019-05-07 05:40:03 +00:00
|
|
|
widget::{self, Button, Image, Rectangle, Text},
|
|
|
|
widget_ids, Color, Colorable, Labelable, Positionable, Sizeable, Widget,
|
2019-03-15 04:55:52 +00:00
|
|
|
};
|
2019-08-01 23:32:33 +00:00
|
|
|
use specs::Join;
|
2019-05-03 16:56:18 +00:00
|
|
|
use std::collections::VecDeque;
|
2019-05-15 00:04:58 +00:00
|
|
|
use vek::*;
|
2019-03-15 04:55:52 +00:00
|
|
|
|
2019-06-04 20:41:50 +00:00
|
|
|
#[cfg(feature = "discord")]
|
2019-06-20 06:16:08 +00:00
|
|
|
use crate::{discord, discord::DiscordUpdate};
|
2019-06-04 20:41:50 +00:00
|
|
|
|
2019-04-30 20:43:55 +00:00
|
|
|
const XP_COLOR: Color = Color::Rgba(0.59, 0.41, 0.67, 1.0);
|
2019-04-26 20:24:05 +00:00
|
|
|
const TEXT_COLOR: Color = Color::Rgba(1.0, 1.0, 1.0, 1.0);
|
2019-10-04 18:27:12 +00:00
|
|
|
//const TEXT_COLOR_2: Color = Color::Rgba(0.0, 0.0, 0.0, 1.0);
|
2019-08-07 13:14:26 +00:00
|
|
|
const TEXT_COLOR_3: Color = Color::Rgba(1.0, 1.0, 1.0, 0.1);
|
2019-08-18 18:07:21 +00:00
|
|
|
//const BG_COLOR: Color = Color::Rgba(1.0, 1.0, 1.0, 0.8);
|
2019-04-26 20:24:05 +00:00
|
|
|
const HP_COLOR: Color = Color::Rgba(0.33, 0.63, 0.0, 1.0);
|
2019-08-18 18:07:21 +00:00
|
|
|
const LOW_HP_COLOR: Color = Color::Rgba(0.93, 0.59, 0.03, 1.0);
|
|
|
|
const CRITICAL_HP_COLOR: Color = Color::Rgba(1.0, 0.0, 0.0, 1.0);
|
|
|
|
const MANA_COLOR: Color = Color::Rgba(0.47, 0.55, 1.0, 0.9);
|
|
|
|
//const FOCUS_COLOR: Color = Color::Rgba(1.0, 0.56, 0.04, 1.0);
|
|
|
|
//const RAGE_COLOR: Color = Color::Rgba(0.5, 0.04, 0.13, 1.0);
|
2019-07-29 14:40:46 +00:00
|
|
|
const META_COLOR: Color = Color::Rgba(1.0, 1.0, 0.0, 1.0);
|
|
|
|
const TELL_COLOR: Color = Color::Rgba(0.98, 0.71, 1.0, 1.0);
|
|
|
|
const PRIVATE_COLOR: Color = Color::Rgba(1.0, 1.0, 0.0, 1.0); // Difference between private and tell?
|
|
|
|
const BROADCAST_COLOR: Color = Color::Rgba(0.28, 0.83, 0.71, 1.0);
|
2019-07-21 18:34:52 +00:00
|
|
|
const GAME_UPDATE_COLOR: Color = Color::Rgba(1.0, 1.0, 0.0, 1.0);
|
2019-07-29 14:40:46 +00:00
|
|
|
const SAY_COLOR: Color = Color::Rgba(1.0, 1.0, 1.0, 1.0);
|
|
|
|
const GROUP_COLOR: Color = Color::Rgba(0.47, 0.84, 1.0, 1.0);
|
|
|
|
const FACTION_COLOR: Color = Color::Rgba(0.24, 1.0, 0.48, 1.0);
|
|
|
|
const KILL_COLOR: Color = Color::Rgba(1.0, 0.17, 0.17, 1.0);
|
2019-04-26 20:24:05 +00:00
|
|
|
|
2019-03-15 04:55:52 +00:00
|
|
|
widget_ids! {
|
|
|
|
struct Ids {
|
2019-07-02 19:54:38 +00:00
|
|
|
// Crosshair
|
2019-07-06 23:04:30 +00:00
|
|
|
crosshair_inner,
|
|
|
|
crosshair_outer,
|
2019-07-02 19:54:38 +00:00
|
|
|
|
2019-05-14 06:43:07 +00:00
|
|
|
// Character Names
|
2019-05-20 06:57:44 +00:00
|
|
|
name_tags[],
|
|
|
|
// Health Bars
|
|
|
|
health_bars[],
|
|
|
|
health_bar_backs[],
|
2019-05-14 06:43:07 +00:00
|
|
|
|
2019-04-02 04:54:27 +00:00
|
|
|
// Test
|
|
|
|
bag_space_add,
|
2019-06-23 19:49:15 +00:00
|
|
|
|
2019-04-26 16:13:30 +00:00
|
|
|
// Debug
|
2019-04-22 18:47:22 +00:00
|
|
|
debug_bg,
|
|
|
|
fps_counter,
|
2019-05-23 08:18:25 +00:00
|
|
|
ping,
|
2019-05-28 18:23:24 +00:00
|
|
|
coordinates,
|
2019-08-06 21:51:13 +00:00
|
|
|
velocity,
|
2019-06-23 19:49:15 +00:00
|
|
|
loaded_distance,
|
2019-09-01 19:04:03 +00:00
|
|
|
time,
|
2019-05-07 05:40:03 +00:00
|
|
|
|
2019-04-26 16:13:30 +00:00
|
|
|
// Game Version
|
|
|
|
version,
|
2019-04-02 14:25:14 +00:00
|
|
|
|
2019-04-25 12:20:35 +00:00
|
|
|
// Help
|
2019-03-28 02:25:08 +00:00
|
|
|
help,
|
2019-10-06 19:19:08 +00:00
|
|
|
help_info,
|
2019-04-25 12:20:35 +00:00
|
|
|
|
|
|
|
// Window Frames
|
2019-03-15 04:55:52 +00:00
|
|
|
window_frame_0,
|
|
|
|
window_frame_1,
|
|
|
|
window_frame_2,
|
|
|
|
window_frame_3,
|
|
|
|
window_frame_4,
|
|
|
|
window_frame_5,
|
2019-04-25 12:20:35 +00:00
|
|
|
|
|
|
|
// Contents
|
2019-03-16 02:03:21 +00:00
|
|
|
button_help2,
|
2019-04-30 14:39:19 +00:00
|
|
|
|
|
|
|
// External
|
2019-05-03 16:56:18 +00:00
|
|
|
chat,
|
2019-04-30 14:39:19 +00:00
|
|
|
map,
|
2019-10-16 11:39:41 +00:00
|
|
|
world_map,
|
2019-04-30 14:39:19 +00:00
|
|
|
character_window,
|
2019-05-07 17:21:53 +00:00
|
|
|
minimap,
|
2019-05-04 10:43:37 +00:00
|
|
|
bag,
|
2019-08-07 13:14:26 +00:00
|
|
|
social,
|
|
|
|
quest,
|
|
|
|
spell,
|
2019-05-04 06:07:23 +00:00
|
|
|
skillbar,
|
2019-05-04 10:43:37 +00:00
|
|
|
buttons,
|
2019-05-03 16:56:18 +00:00
|
|
|
esc_menu,
|
|
|
|
small_window,
|
2019-08-07 13:14:26 +00:00
|
|
|
social_window,
|
2019-05-03 16:56:18 +00:00
|
|
|
settings_window,
|
2019-03-15 04:55:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-08 20:25:15 +00:00
|
|
|
font_ids! {
|
|
|
|
pub struct Fonts {
|
2019-10-05 21:51:08 +00:00
|
|
|
opensans: "voxygen.font.OpenSans-Regular",
|
2019-08-06 06:31:48 +00:00
|
|
|
metamorph: "voxygen.font.Metamorphous-Regular",
|
2019-10-04 18:27:12 +00:00
|
|
|
alkhemi: "voxygen.font.Alkhemikal",
|
|
|
|
wizard: "voxygen.font.wizard",
|
2019-10-05 21:51:08 +00:00
|
|
|
cyri:"voxygen.font.haxrcorp_4089_cyrillic_altgr",
|
2019-05-08 20:25:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-23 09:30:00 +00:00
|
|
|
pub struct DebugInfo {
|
|
|
|
pub tps: f64,
|
|
|
|
pub ping_ms: f64,
|
2019-06-14 15:27:05 +00:00
|
|
|
pub coordinates: Option<comp::Pos>,
|
2019-08-06 21:51:13 +00:00
|
|
|
pub velocity: Option<comp::Vel>,
|
2019-05-23 09:30:00 +00:00
|
|
|
}
|
2019-07-27 19:26:31 +00:00
|
|
|
|
2019-03-16 02:03:21 +00:00
|
|
|
pub enum Event {
|
|
|
|
SendMessage(String),
|
2019-06-05 15:57:48 +00:00
|
|
|
AdjustMousePan(u32),
|
|
|
|
AdjustMouseZoom(u32),
|
2019-10-02 13:09:05 +00:00
|
|
|
ToggleZoomInvert(bool),
|
2019-05-19 00:45:02 +00:00
|
|
|
AdjustViewDistance(u32),
|
2019-09-24 16:18:09 +00:00
|
|
|
AdjustMusicVolume(f32),
|
2019-09-24 16:25:24 +00:00
|
|
|
AdjustSfxVolume(f32),
|
2019-05-20 14:53:12 +00:00
|
|
|
ChangeAudioDevice(String),
|
2019-06-06 19:11:39 +00:00
|
|
|
ChangeMaxFPS(u32),
|
2019-08-05 16:37:52 +00:00
|
|
|
ChangeFOV(u16),
|
2019-09-26 07:28:40 +00:00
|
|
|
ChangeAaMode(AaMode),
|
2019-07-07 00:15:22 +00:00
|
|
|
CrosshairTransp(f32),
|
2019-07-23 01:02:57 +00:00
|
|
|
CrosshairType(CrosshairType),
|
2019-08-18 18:07:21 +00:00
|
|
|
ToggleXpBar(XpBar),
|
|
|
|
ToggleBarNumbers(BarNumbers),
|
|
|
|
ToggleShortcutNumbers(ShortcutNumbers),
|
2019-07-26 02:28:53 +00:00
|
|
|
UiScale(ScaleChange),
|
2019-06-02 02:17:36 +00:00
|
|
|
CharacterSelection,
|
2019-08-30 20:46:45 +00:00
|
|
|
UseInventorySlot(usize),
|
2019-07-25 22:52:28 +00:00
|
|
|
SwapInventorySlots(usize, usize),
|
2019-07-26 17:08:40 +00:00
|
|
|
DropInventorySlot(usize),
|
2019-03-16 02:03:21 +00:00
|
|
|
Logout,
|
|
|
|
Quit,
|
|
|
|
}
|
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// TODO: Are these the possible layouts we want?
|
|
|
|
// TODO: Maybe replace this with bitflags.
|
|
|
|
// `map` is not here because it currently is displayed over the top of other open windows.
|
2019-05-04 10:43:37 +00:00
|
|
|
#[derive(PartialEq)]
|
|
|
|
pub enum Windows {
|
2019-08-07 13:14:26 +00:00
|
|
|
Settings, // Display settings window.
|
2019-03-28 04:00:51 +00:00
|
|
|
None,
|
|
|
|
}
|
|
|
|
|
2019-07-23 01:02:57 +00:00
|
|
|
#[derive(Clone, Copy, Debug, Serialize, Deserialize)]
|
|
|
|
pub enum CrosshairType {
|
|
|
|
Round,
|
|
|
|
RoundEdges,
|
|
|
|
Edges,
|
|
|
|
}
|
2019-08-18 18:07:21 +00:00
|
|
|
#[derive(Clone, Copy, Debug, Serialize, Deserialize)]
|
|
|
|
pub enum XpBar {
|
|
|
|
Always,
|
|
|
|
OnGain,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug, Serialize, Deserialize)]
|
|
|
|
pub enum BarNumbers {
|
|
|
|
Values,
|
|
|
|
Percent,
|
|
|
|
Off,
|
|
|
|
}
|
|
|
|
#[derive(Clone, Copy, Debug, Serialize, Deserialize)]
|
|
|
|
pub enum ShortcutNumbers {
|
|
|
|
On,
|
|
|
|
Off,
|
|
|
|
}
|
2019-07-23 01:02:57 +00:00
|
|
|
|
2019-05-04 17:24:26 +00:00
|
|
|
pub struct Show {
|
|
|
|
ui: bool,
|
|
|
|
help: bool,
|
|
|
|
debug: bool,
|
|
|
|
bag: bool,
|
2019-08-07 13:14:26 +00:00
|
|
|
social: bool,
|
|
|
|
spell: bool,
|
|
|
|
quest: bool,
|
|
|
|
character_window: bool,
|
2019-05-04 17:24:26 +00:00
|
|
|
esc_menu: bool,
|
|
|
|
open_windows: Windows,
|
|
|
|
map: bool,
|
|
|
|
inventory_test_button: bool,
|
2019-05-07 05:40:03 +00:00
|
|
|
mini_map: bool,
|
2019-05-21 04:55:20 +00:00
|
|
|
ingame: bool,
|
2019-06-02 02:17:36 +00:00
|
|
|
settings_tab: SettingsTab,
|
2019-08-07 13:14:26 +00:00
|
|
|
social_tab: SocialTab,
|
|
|
|
|
2019-05-09 16:55:13 +00:00
|
|
|
want_grab: bool,
|
2019-05-04 17:24:26 +00:00
|
|
|
}
|
|
|
|
impl Show {
|
2019-05-25 12:59:25 +00:00
|
|
|
fn bag(&mut self, open: bool) {
|
|
|
|
self.bag = open;
|
|
|
|
self.want_grab = !open;
|
|
|
|
}
|
2019-05-04 17:24:26 +00:00
|
|
|
fn toggle_bag(&mut self) {
|
2019-05-25 12:59:25 +00:00
|
|
|
self.bag(!self.bag);
|
2019-05-04 17:24:26 +00:00
|
|
|
}
|
2019-05-25 12:59:25 +00:00
|
|
|
fn map(&mut self, open: bool) {
|
|
|
|
self.map = open;
|
2019-05-04 17:24:26 +00:00
|
|
|
self.bag = false;
|
2019-05-25 12:59:25 +00:00
|
|
|
self.want_grab = !open;
|
|
|
|
}
|
2019-08-07 13:14:26 +00:00
|
|
|
fn character_window(&mut self, open: bool) {
|
|
|
|
self.character_window = open;
|
|
|
|
self.bag = false;
|
|
|
|
self.want_grab = !open;
|
|
|
|
}
|
|
|
|
fn social(&mut self, open: bool) {
|
|
|
|
self.social = open;
|
|
|
|
self.spell = false;
|
|
|
|
self.quest = false;
|
|
|
|
self.want_grab = !open;
|
|
|
|
}
|
|
|
|
fn spell(&mut self, open: bool) {
|
|
|
|
self.social = false;
|
|
|
|
self.spell = open;
|
|
|
|
self.quest = false;
|
|
|
|
self.want_grab = !open;
|
|
|
|
}
|
|
|
|
fn quest(&mut self, open: bool) {
|
|
|
|
self.social = false;
|
|
|
|
self.spell = false;
|
|
|
|
self.quest = open;
|
|
|
|
self.want_grab = !open;
|
|
|
|
}
|
2019-05-25 12:59:25 +00:00
|
|
|
fn toggle_map(&mut self) {
|
|
|
|
self.map(!self.map)
|
2019-05-04 17:24:26 +00:00
|
|
|
}
|
2019-05-07 17:21:53 +00:00
|
|
|
|
2019-05-04 17:24:26 +00:00
|
|
|
fn toggle_mini_map(&mut self) {
|
|
|
|
self.mini_map = !self.mini_map;
|
|
|
|
}
|
2019-05-07 17:21:53 +00:00
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
fn toggle_char_window(&mut self) {
|
2019-08-07 13:14:26 +00:00
|
|
|
self.character_window = !self.character_window
|
2019-05-04 17:24:26 +00:00
|
|
|
}
|
2019-05-07 17:21:53 +00:00
|
|
|
|
2019-05-25 12:59:25 +00:00
|
|
|
fn settings(&mut self, open: bool) {
|
|
|
|
self.open_windows = if open {
|
|
|
|
Windows::Settings
|
|
|
|
} else {
|
|
|
|
Windows::None
|
2019-05-04 17:24:26 +00:00
|
|
|
};
|
|
|
|
self.bag = false;
|
2019-08-07 13:14:26 +00:00
|
|
|
self.social = false;
|
|
|
|
self.spell = false;
|
|
|
|
self.quest = false;
|
2019-05-25 12:59:25 +00:00
|
|
|
self.want_grab = !open;
|
|
|
|
}
|
|
|
|
fn toggle_settings(&mut self) {
|
|
|
|
match self.open_windows {
|
|
|
|
Windows::Settings => self.settings(false),
|
|
|
|
_ => self.settings(true),
|
|
|
|
};
|
2019-05-04 17:24:26 +00:00
|
|
|
}
|
2019-05-07 17:21:53 +00:00
|
|
|
|
2019-05-04 17:24:26 +00:00
|
|
|
fn toggle_help(&mut self) {
|
|
|
|
self.help = !self.help
|
|
|
|
}
|
2019-05-07 17:21:53 +00:00
|
|
|
|
2019-05-04 17:24:26 +00:00
|
|
|
fn toggle_ui(&mut self) {
|
|
|
|
self.ui = !self.ui;
|
|
|
|
}
|
|
|
|
|
2019-05-07 20:00:03 +00:00
|
|
|
fn toggle_windows(&mut self) {
|
2019-05-04 17:24:26 +00:00
|
|
|
if self.bag
|
|
|
|
|| self.esc_menu
|
|
|
|
|| self.map
|
2019-08-07 13:14:26 +00:00
|
|
|
|| self.social
|
|
|
|
|| self.quest
|
|
|
|
|| self.spell
|
|
|
|
|| self.character_window
|
2019-05-04 17:24:26 +00:00
|
|
|
|| match self.open_windows {
|
|
|
|
Windows::None => false,
|
|
|
|
_ => true,
|
|
|
|
}
|
|
|
|
{
|
|
|
|
self.bag = false;
|
|
|
|
self.esc_menu = false;
|
|
|
|
self.map = false;
|
2019-08-07 13:14:26 +00:00
|
|
|
self.social = false;
|
|
|
|
self.quest = false;
|
|
|
|
self.spell = false;
|
|
|
|
self.character_window = false;
|
2019-05-04 17:24:26 +00:00
|
|
|
self.open_windows = Windows::None;
|
2019-05-09 16:55:13 +00:00
|
|
|
self.want_grab = true;
|
2019-05-04 17:24:26 +00:00
|
|
|
} else {
|
|
|
|
self.esc_menu = true;
|
2019-05-09 16:55:13 +00:00
|
|
|
self.want_grab = false;
|
2019-05-04 17:24:26 +00:00
|
|
|
}
|
|
|
|
}
|
2019-06-02 02:17:36 +00:00
|
|
|
|
|
|
|
fn open_setting_tab(&mut self, tab: SettingsTab) {
|
|
|
|
self.open_windows = Windows::Settings;
|
|
|
|
self.esc_menu = false;
|
|
|
|
self.settings_tab = tab;
|
|
|
|
self.bag = false;
|
|
|
|
self.want_grab = false;
|
|
|
|
}
|
2019-08-07 13:14:26 +00:00
|
|
|
|
|
|
|
fn toggle_social(&mut self) {
|
|
|
|
self.social = !self.social;
|
|
|
|
self.spell = false;
|
|
|
|
self.quest = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn open_social_tab(&mut self, social_tab: SocialTab) {
|
|
|
|
self.social_tab = social_tab;
|
|
|
|
self.spell = false;
|
|
|
|
self.quest = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn toggle_spell(&mut self) {
|
|
|
|
self.spell = !self.spell;
|
|
|
|
self.social = false;
|
|
|
|
self.quest = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn toggle_quest(&mut self) {
|
|
|
|
self.quest = !self.quest;
|
|
|
|
self.spell = false;
|
|
|
|
self.social = false;
|
|
|
|
}
|
2019-05-04 17:24:26 +00:00
|
|
|
}
|
|
|
|
|
2019-09-25 20:18:40 +00:00
|
|
|
pub struct Hud {
|
2019-03-15 04:55:52 +00:00
|
|
|
ui: Ui,
|
|
|
|
ids: Ids,
|
2019-10-16 11:39:41 +00:00
|
|
|
world_map: Id,
|
2019-03-15 04:55:52 +00:00
|
|
|
imgs: Imgs,
|
2019-10-09 19:28:05 +00:00
|
|
|
item_imgs: ItemImgs,
|
2019-04-30 20:43:55 +00:00
|
|
|
fonts: Fonts,
|
2019-09-25 16:51:47 +00:00
|
|
|
rot_imgs: ImgsRot,
|
2019-07-17 22:10:42 +00:00
|
|
|
new_messages: VecDeque<ClientEvent>,
|
2019-07-06 17:00:05 +00:00
|
|
|
inventory_space: usize,
|
2019-05-04 17:24:26 +00:00
|
|
|
show: Show,
|
2019-05-07 03:25:25 +00:00
|
|
|
to_focus: Option<Option<widget::Id>>,
|
2019-07-26 21:05:39 +00:00
|
|
|
force_ungrab: bool,
|
2019-07-05 16:21:11 +00:00
|
|
|
force_chat_input: Option<String>,
|
|
|
|
force_chat_cursor: Option<Index>,
|
2019-03-15 04:55:52 +00:00
|
|
|
}
|
|
|
|
|
2019-09-25 20:18:40 +00:00
|
|
|
impl Hud {
|
2019-10-16 11:39:41 +00:00
|
|
|
pub fn new(global_state: &mut GlobalState, client: &Client) -> Self {
|
2019-07-26 02:28:53 +00:00
|
|
|
let window = &mut global_state.window;
|
|
|
|
let settings = &global_state.settings;
|
|
|
|
|
2019-03-15 04:55:52 +00:00
|
|
|
let mut ui = Ui::new(window).unwrap();
|
2019-07-26 02:28:53 +00:00
|
|
|
ui.set_scaling_mode(settings.gameplay.ui_scale);
|
2019-05-17 09:22:32 +00:00
|
|
|
// Generate ids.
|
2019-03-16 02:03:21 +00:00
|
|
|
let ids = Ids::new(ui.id_generator());
|
2019-10-16 11:39:41 +00:00
|
|
|
// Load world map
|
|
|
|
let world_map = ui.add_graphic(Graphic::Image(client.world_map.clone()));
|
2019-05-17 09:22:32 +00:00
|
|
|
// Load images.
|
|
|
|
let imgs = Imgs::load(&mut ui).expect("Failed to load images!");
|
2019-09-25 20:18:40 +00:00
|
|
|
// Load rotation images.
|
|
|
|
let rot_imgs = ImgsRot::load(&mut ui).expect("Failed to load rot images!");
|
2019-10-09 19:28:05 +00:00
|
|
|
// Load item images.
|
|
|
|
let item_imgs = ItemImgs::new(&mut ui);
|
2019-05-17 09:22:32 +00:00
|
|
|
// Load fonts.
|
|
|
|
let fonts = Fonts::load(&mut ui).expect("Failed to load fonts!");
|
2019-04-20 22:02:48 +00:00
|
|
|
|
2019-03-15 04:55:52 +00:00
|
|
|
Self {
|
|
|
|
ui,
|
|
|
|
imgs,
|
2019-10-16 11:39:41 +00:00
|
|
|
world_map,
|
2019-09-25 16:51:47 +00:00
|
|
|
rot_imgs,
|
2019-10-09 19:28:05 +00:00
|
|
|
item_imgs,
|
2019-04-30 20:43:55 +00:00
|
|
|
fonts,
|
2019-07-26 21:05:39 +00:00
|
|
|
ids,
|
2019-05-03 16:56:18 +00:00
|
|
|
new_messages: VecDeque::new(),
|
2019-07-06 17:00:05 +00:00
|
|
|
inventory_space: 8,
|
2019-05-04 17:24:26 +00:00
|
|
|
show: Show {
|
2019-05-18 19:03:13 +00:00
|
|
|
help: false,
|
|
|
|
debug: true,
|
2019-05-04 17:24:26 +00:00
|
|
|
bag: false,
|
|
|
|
esc_menu: false,
|
|
|
|
open_windows: Windows::None,
|
|
|
|
map: false,
|
|
|
|
ui: true,
|
2019-08-07 13:14:26 +00:00
|
|
|
social: false,
|
|
|
|
quest: false,
|
|
|
|
spell: false,
|
|
|
|
character_window: false,
|
2019-05-04 17:24:26 +00:00
|
|
|
inventory_test_button: false,
|
|
|
|
mini_map: false,
|
2019-06-02 02:17:36 +00:00
|
|
|
settings_tab: SettingsTab::Interface,
|
2019-08-07 13:14:26 +00:00
|
|
|
social_tab: SocialTab::Online,
|
2019-05-09 16:55:13 +00:00
|
|
|
want_grab: true,
|
2019-05-21 04:55:20 +00:00
|
|
|
ingame: true,
|
2019-05-06 12:28:57 +00:00
|
|
|
},
|
|
|
|
to_focus: None,
|
2019-05-09 08:55:32 +00:00
|
|
|
force_ungrab: false,
|
2019-07-05 16:21:11 +00:00
|
|
|
force_chat_input: None,
|
|
|
|
force_chat_cursor: None,
|
2019-03-15 04:55:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-24 12:43:03 +00:00
|
|
|
fn update_layout(
|
|
|
|
&mut self,
|
|
|
|
client: &Client,
|
|
|
|
global_state: &GlobalState,
|
|
|
|
debug_info: DebugInfo,
|
|
|
|
) -> Vec<Event> {
|
2019-03-16 02:03:21 +00:00
|
|
|
let mut events = Vec::new();
|
2019-09-25 16:51:47 +00:00
|
|
|
let (ref mut ui_widgets, ref mut tooltip_manager) = self.ui.set_widgets();
|
2019-06-28 12:08:12 +00:00
|
|
|
|
2019-07-21 17:45:31 +00:00
|
|
|
let version = format!("{}-{}", env!("CARGO_PKG_VERSION"), common::util::GIT_HASH);
|
2019-04-02 04:54:27 +00:00
|
|
|
|
2019-05-21 04:55:20 +00:00
|
|
|
if self.show.ingame {
|
2019-07-02 19:54:38 +00:00
|
|
|
// Crosshair
|
2019-10-06 19:19:08 +00:00
|
|
|
if !self.show.help {
|
|
|
|
Image::new(
|
|
|
|
// TODO: Do we want to match on this every frame?
|
|
|
|
match global_state.settings.gameplay.crosshair_type {
|
|
|
|
CrosshairType::Round => self.imgs.crosshair_outer_round,
|
|
|
|
CrosshairType::RoundEdges => self.imgs.crosshair_outer_round_edges,
|
|
|
|
CrosshairType::Edges => self.imgs.crosshair_outer_edges,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
.w_h(21.0 * 1.5, 21.0 * 1.5)
|
|
|
|
.middle_of(ui_widgets.window)
|
|
|
|
.color(Some(Color::Rgba(
|
|
|
|
1.0,
|
|
|
|
1.0,
|
|
|
|
1.0,
|
|
|
|
global_state.settings.gameplay.crosshair_transp,
|
|
|
|
)))
|
|
|
|
.set(self.ids.crosshair_outer, ui_widgets);
|
|
|
|
Image::new(self.imgs.crosshair_inner)
|
|
|
|
.w_h(21.0 * 2.0, 21.0 * 2.0)
|
|
|
|
.middle_of(self.ids.crosshair_outer)
|
|
|
|
.color(Some(Color::Rgba(1.0, 1.0, 1.0, 0.6)))
|
|
|
|
.set(self.ids.crosshair_inner, ui_widgets);
|
|
|
|
}
|
2019-07-02 19:54:38 +00:00
|
|
|
|
2019-08-07 13:14:26 +00:00
|
|
|
// Nametags and healthbars
|
|
|
|
let ecs = client.state().ecs();
|
|
|
|
let pos = ecs.read_storage::<comp::Pos>();
|
|
|
|
let stats = ecs.read_storage::<comp::Stats>();
|
|
|
|
let players = ecs.read_storage::<comp::Player>();
|
|
|
|
let scales = ecs.read_storage::<comp::Scale>();
|
|
|
|
let entities = ecs.entities();
|
|
|
|
let me = client.entity();
|
|
|
|
let view_distance = client.view_distance().unwrap_or(1);
|
|
|
|
// Get player position.
|
|
|
|
let player_pos = client
|
|
|
|
.state()
|
|
|
|
.ecs()
|
|
|
|
.read_storage::<comp::Pos>()
|
|
|
|
.get(client.entity())
|
|
|
|
.map_or(Vec3::zero(), |pos| pos.0);
|
|
|
|
let mut name_id_walker = self.ids.name_tags.walk();
|
|
|
|
let mut health_id_walker = self.ids.health_bars.walk();
|
|
|
|
let mut health_back_id_walker = self.ids.health_bar_backs.walk();
|
|
|
|
|
2019-05-28 19:26:11 +00:00
|
|
|
// Render Name Tags
|
2019-10-05 13:58:58 +00:00
|
|
|
for (pos, name, level, scale) in
|
|
|
|
(&entities, &pos, &stats, players.maybe(), scales.maybe())
|
|
|
|
.join()
|
|
|
|
.filter(|(entity, _, stats, _, _)| *entity != me && !stats.is_dead)
|
|
|
|
// Don't process nametags outside the vd (visibility further limited by ui backend)
|
|
|
|
.filter(|(_, pos, _, _, _)| {
|
|
|
|
Vec2::from(pos.0 - player_pos)
|
|
|
|
.map2(TerrainChunk::RECT_SIZE, |d: f32, sz| {
|
|
|
|
d.abs() as f32 / sz as f32
|
|
|
|
})
|
|
|
|
.magnitude()
|
|
|
|
< view_distance as f32
|
|
|
|
})
|
|
|
|
.map(|(_, pos, stats, player, scale)| {
|
|
|
|
// TODO: This is temporary
|
|
|
|
// If the player used the default character name display their name instead
|
|
|
|
let name = if stats.name == "Character Name" {
|
|
|
|
player.map_or(&stats.name, |p| &p.alias)
|
|
|
|
} else {
|
|
|
|
&stats.name
|
|
|
|
};
|
|
|
|
(pos.0, name, stats.level, scale)
|
|
|
|
})
|
2019-05-14 06:43:07 +00:00
|
|
|
{
|
2019-10-04 15:19:15 +00:00
|
|
|
let info = format!("{} Level {}", name, level.level());
|
2019-08-03 18:40:03 +00:00
|
|
|
let scale = scale.map(|s| s.0).unwrap_or(1.0);
|
|
|
|
|
2019-05-20 06:57:44 +00:00
|
|
|
let id = name_id_walker.next(
|
|
|
|
&mut self.ids.name_tags,
|
2019-05-19 18:07:50 +00:00
|
|
|
&mut ui_widgets.widget_id_generator(),
|
|
|
|
);
|
2019-10-04 15:19:15 +00:00
|
|
|
Text::new(&info)
|
2019-05-19 14:15:50 +00:00
|
|
|
.font_size(20)
|
2019-07-06 17:29:44 +00:00
|
|
|
.color(Color::Rgba(0.61, 0.61, 0.89, 1.0))
|
2019-05-19 14:15:50 +00:00
|
|
|
.x_y(0.0, 0.0)
|
2019-08-03 18:40:03 +00:00
|
|
|
.position_ingame(pos + Vec3::new(0.0, 0.0, 1.5 * scale + 1.5))
|
2019-05-19 14:15:50 +00:00
|
|
|
.resolution(100.0)
|
2019-05-19 16:25:02 +00:00
|
|
|
.set(id, ui_widgets);
|
|
|
|
}
|
2019-05-26 14:04:44 +00:00
|
|
|
|
2019-05-28 19:26:11 +00:00
|
|
|
// Render Health Bars
|
2019-08-03 18:40:03 +00:00
|
|
|
for (_entity, pos, stats, scale) in (&entities, &pos, &stats, scales.maybe())
|
2019-05-27 17:01:00 +00:00
|
|
|
.join()
|
2019-08-03 18:40:03 +00:00
|
|
|
.filter(|(entity, _, stats, _)| {
|
2019-05-27 17:01:00 +00:00
|
|
|
*entity != me
|
|
|
|
&& !stats.is_dead
|
2019-08-01 12:48:09 +00:00
|
|
|
&& stats.health.current() != stats.health.maximum()
|
2019-05-27 17:01:00 +00:00
|
|
|
})
|
|
|
|
// Don't process health bars outside the vd (visibility further limited by ui backend)
|
2019-08-03 18:40:03 +00:00
|
|
|
.filter(|(_, pos, _, _)| {
|
common: Rework volume API
See the doc comments in `common/src/vol.rs` for more information on
the API itself.
The changes include:
* Consistent `Err`/`Error` naming.
* Types are named `...Error`.
* `enum` variants are named `...Err`.
* Rename `VolMap{2d, 3d}` -> `VolGrid{2d, 3d}`. This is in preparation
to an upcoming change where a “map” in the game related sense will
be added.
* Add volume iterators. There are two types of them:
* _Position_ iterators obtained from the trait `IntoPosIterator`
using the method
`fn pos_iter(self, lower_bound: Vec3<i32>, upper_bound: Vec3<i32>) -> ...`
which returns an iterator over `Vec3<i32>`.
* _Volume_ iterators obtained from the trait `IntoVolIterator`
using the method
`fn vol_iter(self, lower_bound: Vec3<i32>, upper_bound: Vec3<i32>) -> ...`
which returns an iterator over `(Vec3<i32>, &Self::Vox)`.
Those traits will usually be implemented by references to volume
types (i.e. `impl IntoVolIterator<'a> for &'a T` where `T` is some
type which usually implements several volume traits, such as `Chunk`).
* _Position_ iterators iterate over the positions valid for that
volume.
* _Volume_ iterators do the same but return not only the position
but also the voxel at that position, in each iteration.
* Introduce trait `RectSizedVol` for the use case which we have with
`Chonk`: A `Chonk` is sized only in x and y direction.
* Introduce traits `RasterableVol`, `RectRasterableVol`
* `RasterableVol` represents a volume that is compile-time sized and has
its lower bound at `(0, 0, 0)`. The name `RasterableVol` was chosen
because such a volume can be used with `VolGrid3d`.
* `RectRasterableVol` represents a volume that is compile-time sized at
least in x and y direction and has its lower bound at `(0, 0, z)`.
There's no requirement on he lower bound or size in z direction.
The name `RectRasterableVol` was chosen because such a volume can be
used with `VolGrid2d`.
2019-09-03 22:23:29 +00:00
|
|
|
Vec2::from(pos.0 - player_pos)
|
|
|
|
.map2(TerrainChunk::RECT_SIZE, |d: f32, sz| {
|
|
|
|
d.abs() as f32 / sz as f32
|
|
|
|
})
|
2019-06-23 19:49:15 +00:00
|
|
|
.magnitude()
|
|
|
|
< view_distance as f32
|
2019-05-27 17:01:00 +00:00
|
|
|
})
|
2019-05-19 18:07:50 +00:00
|
|
|
{
|
2019-08-03 18:40:03 +00:00
|
|
|
let scale = scale.map(|s| s.0).unwrap_or(1.0);
|
|
|
|
|
2019-05-20 06:57:44 +00:00
|
|
|
let back_id = health_back_id_walker.next(
|
|
|
|
&mut self.ids.health_bar_backs,
|
2019-05-19 18:07:50 +00:00
|
|
|
&mut ui_widgets.widget_id_generator(),
|
|
|
|
);
|
2019-05-20 06:57:44 +00:00
|
|
|
let bar_id = health_id_walker.next(
|
|
|
|
&mut self.ids.health_bars,
|
|
|
|
&mut ui_widgets.widget_id_generator(),
|
|
|
|
);
|
2019-05-28 19:26:11 +00:00
|
|
|
// Background
|
2019-05-20 06:57:44 +00:00
|
|
|
Rectangle::fill_with([120.0, 8.0], Color::Rgba(0.3, 0.3, 0.3, 0.5))
|
|
|
|
.x_y(0.0, -25.0)
|
2019-08-03 18:40:03 +00:00
|
|
|
.position_ingame(pos.0 + Vec3::new(0.0, 0.0, 1.5 * scale + 1.5))
|
2019-05-19 16:25:02 +00:00
|
|
|
.resolution(100.0)
|
2019-05-20 06:57:44 +00:00
|
|
|
.set(back_id, ui_widgets);
|
|
|
|
|
2019-05-28 19:26:11 +00:00
|
|
|
// % HP Filling
|
2019-05-20 06:57:44 +00:00
|
|
|
Rectangle::fill_with(
|
2019-05-26 14:04:44 +00:00
|
|
|
[
|
2019-08-01 12:48:09 +00:00
|
|
|
120.0 * (stats.health.current() as f64 / stats.health.maximum() as f64),
|
2019-05-26 14:04:44 +00:00
|
|
|
8.0,
|
|
|
|
],
|
2019-05-20 06:57:44 +00:00
|
|
|
HP_COLOR,
|
|
|
|
)
|
|
|
|
.x_y(0.0, -25.0)
|
2019-08-03 18:40:03 +00:00
|
|
|
.position_ingame(pos.0 + Vec3::new(0.0, 0.0, 1.5 * scale + 1.5))
|
2019-05-20 06:57:44 +00:00
|
|
|
.resolution(100.0)
|
|
|
|
.set(bar_id, ui_widgets);
|
2019-05-14 06:43:07 +00:00
|
|
|
}
|
2019-05-15 00:04:58 +00:00
|
|
|
}
|
2019-05-14 06:43:07 +00:00
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Display debug window.
|
2019-05-04 17:24:26 +00:00
|
|
|
if self.show.debug {
|
2019-04-26 16:13:30 +00:00
|
|
|
// Alpha Version
|
2019-06-28 12:08:12 +00:00
|
|
|
Text::new(&version)
|
2019-04-26 16:13:30 +00:00
|
|
|
.top_left_with_margins_on(ui_widgets.window, 5.0, 5.0)
|
|
|
|
.font_size(14)
|
2019-10-06 19:19:08 +00:00
|
|
|
.font_id(self.fonts.cyri)
|
2019-04-26 16:13:30 +00:00
|
|
|
.color(TEXT_COLOR)
|
|
|
|
.set(self.ids.version, ui_widgets);
|
2019-06-23 19:49:15 +00:00
|
|
|
// Ticks per second
|
2019-08-22 17:44:35 +00:00
|
|
|
Text::new(&format!("FPS: {:.0}", debug_info.tps))
|
2019-04-22 18:47:22 +00:00
|
|
|
.color(TEXT_COLOR)
|
2019-04-26 16:13:30 +00:00
|
|
|
.down_from(self.ids.version, 5.0)
|
2019-10-06 19:19:08 +00:00
|
|
|
.font_id(self.fonts.cyri)
|
2019-04-26 16:13:30 +00:00
|
|
|
.font_size(14)
|
2019-04-22 18:47:22 +00:00
|
|
|
.set(self.ids.fps_counter, ui_widgets);
|
2019-06-23 19:49:15 +00:00
|
|
|
// Ping
|
2019-08-22 17:44:35 +00:00
|
|
|
Text::new(&format!("Ping: {:.0}ms", debug_info.ping_ms))
|
2019-05-23 08:18:25 +00:00
|
|
|
.color(TEXT_COLOR)
|
|
|
|
.down_from(self.ids.fps_counter, 5.0)
|
2019-10-06 19:19:08 +00:00
|
|
|
.font_id(self.fonts.cyri)
|
2019-05-23 08:18:25 +00:00
|
|
|
.font_size(14)
|
|
|
|
.set(self.ids.ping, ui_widgets);
|
2019-08-06 21:51:13 +00:00
|
|
|
// Player's position
|
2019-05-28 19:40:50 +00:00
|
|
|
let coordinates_text = match debug_info.coordinates {
|
2019-08-22 17:44:35 +00:00
|
|
|
Some(coordinates) => format!(
|
|
|
|
"Coordinates: ({:.0}, {:.0}, {:.0})",
|
|
|
|
coordinates.0.x, coordinates.0.y, coordinates.0.z,
|
|
|
|
),
|
2019-05-28 19:40:50 +00:00
|
|
|
None => "Player has no Pos component".to_owned(),
|
|
|
|
};
|
|
|
|
Text::new(&coordinates_text)
|
2019-05-28 18:23:24 +00:00
|
|
|
.color(TEXT_COLOR)
|
|
|
|
.down_from(self.ids.ping, 5.0)
|
2019-10-06 19:19:08 +00:00
|
|
|
.font_id(self.fonts.cyri)
|
2019-05-28 18:23:24 +00:00
|
|
|
.font_size(14)
|
|
|
|
.set(self.ids.coordinates, ui_widgets);
|
2019-08-07 02:26:24 +00:00
|
|
|
// Player's velocity
|
2019-08-06 21:51:13 +00:00
|
|
|
let velocity_text = match debug_info.velocity {
|
2019-08-22 19:26:35 +00:00
|
|
|
Some(velocity) => format!(
|
|
|
|
"Velocity: ({:.1}, {:.1}, {:.1}) [{:.1} u/s]",
|
|
|
|
velocity.0.x,
|
|
|
|
velocity.0.y,
|
|
|
|
velocity.0.z,
|
|
|
|
velocity.0.magnitude()
|
|
|
|
),
|
2019-08-06 21:51:13 +00:00
|
|
|
None => "Player has no Vel component".to_owned(),
|
|
|
|
};
|
|
|
|
Text::new(&velocity_text)
|
|
|
|
.color(TEXT_COLOR)
|
|
|
|
.down_from(self.ids.coordinates, 5.0)
|
2019-10-06 19:19:08 +00:00
|
|
|
.font_id(self.fonts.cyri)
|
2019-08-06 21:51:13 +00:00
|
|
|
.font_size(14)
|
|
|
|
.set(self.ids.velocity, ui_widgets);
|
2019-06-23 19:49:15 +00:00
|
|
|
// Loaded distance
|
|
|
|
Text::new(&format!(
|
|
|
|
"View distance: {} chunks",
|
|
|
|
client.loaded_distance().unwrap_or(0)
|
|
|
|
))
|
|
|
|
.color(TEXT_COLOR)
|
2019-08-06 21:51:13 +00:00
|
|
|
.down_from(self.ids.velocity, 5.0)
|
2019-10-06 19:19:08 +00:00
|
|
|
.font_id(self.fonts.cyri)
|
2019-06-23 19:49:15 +00:00
|
|
|
.font_size(14)
|
|
|
|
.set(self.ids.loaded_distance, ui_widgets);
|
2019-09-01 19:04:03 +00:00
|
|
|
// Time
|
|
|
|
let time_in_seconds = client.state().get_time_of_day();
|
|
|
|
let current_time = NaiveTime::from_num_seconds_from_midnight(
|
|
|
|
// Wraps around back to 0s if it exceeds 24 hours (24 hours = 86400s)
|
|
|
|
(time_in_seconds as u64 % 86400) as u32,
|
|
|
|
0,
|
|
|
|
);
|
|
|
|
Text::new(&format!(
|
|
|
|
"Time: {}",
|
|
|
|
current_time.format("%H:%M").to_string()
|
|
|
|
))
|
|
|
|
.color(TEXT_COLOR)
|
|
|
|
.down_from(self.ids.loaded_distance, 5.0)
|
2019-10-06 19:19:08 +00:00
|
|
|
.font_id(self.fonts.cyri)
|
2019-09-01 19:04:03 +00:00
|
|
|
.font_size(14)
|
|
|
|
.set(self.ids.time, ui_widgets);
|
2019-10-06 19:19:08 +00:00
|
|
|
|
|
|
|
// Help Window
|
|
|
|
Text::new("Press 'F1' to show Keybindings")
|
|
|
|
.color(TEXT_COLOR)
|
|
|
|
.down_from(self.ids.time, 5.0)
|
|
|
|
.font_id(self.fonts.cyri)
|
|
|
|
.font_size(14)
|
|
|
|
.set(self.ids.help_info, ui_widgets);
|
2019-04-22 18:47:22 +00:00
|
|
|
}
|
2019-05-07 17:21:53 +00:00
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Add Bag-Space Button.
|
2019-05-04 17:24:26 +00:00
|
|
|
if self.show.inventory_test_button {
|
2019-06-29 01:47:38 +00:00
|
|
|
if Button::image(self.imgs.button)
|
2019-04-20 15:17:29 +00:00
|
|
|
.w_h(100.0, 100.0)
|
|
|
|
.middle_of(ui_widgets.window)
|
2019-06-30 14:04:10 +00:00
|
|
|
.label("Add 1 Space")
|
2019-04-20 15:17:29 +00:00
|
|
|
.label_font_size(20)
|
2019-06-29 13:02:27 +00:00
|
|
|
.label_color(TEXT_COLOR)
|
2019-06-29 01:47:38 +00:00
|
|
|
.hover_image(self.imgs.button_hover)
|
|
|
|
.press_image(self.imgs.button_press)
|
2019-04-20 15:17:29 +00:00
|
|
|
.set(self.ids.bag_space_add, ui_widgets)
|
|
|
|
.was_clicked()
|
2019-03-15 04:55:52 +00:00
|
|
|
{
|
2019-06-30 14:04:10 +00:00
|
|
|
if self.inventory_space < 100 {
|
|
|
|
self.inventory_space += 1;
|
2019-06-29 13:02:27 +00:00
|
|
|
} else {
|
|
|
|
}
|
2019-04-20 15:17:29 +00:00
|
|
|
};
|
2019-05-07 05:40:03 +00:00
|
|
|
}
|
2019-05-07 17:21:53 +00:00
|
|
|
|
2019-04-20 15:17:29 +00:00
|
|
|
// Help Text
|
2019-10-06 19:19:08 +00:00
|
|
|
if self.show.help && !self.show.map && !self.show.esc_menu {
|
|
|
|
Image::new(self.imgs.help)
|
|
|
|
.middle_of(ui_widgets.window)
|
|
|
|
.w_h(1260.0, 519.0)
|
2019-04-24 19:55:22 +00:00
|
|
|
.set(self.ids.help, ui_widgets);
|
2019-04-24 20:33:34 +00:00
|
|
|
// X-button
|
2019-04-28 16:44:44 +00:00
|
|
|
if Button::image(self.imgs.close_button)
|
2019-10-06 19:19:08 +00:00
|
|
|
.w_h(40.0, 40.0)
|
2019-04-28 16:44:44 +00:00
|
|
|
.hover_image(self.imgs.close_button_hover)
|
|
|
|
.press_image(self.imgs.close_button_press)
|
2019-10-06 19:19:08 +00:00
|
|
|
.top_right_with_margins_on(self.ids.help, 0.0, 0.0)
|
|
|
|
.color(Color::Rgba(1.0, 1.0, 1.0, 0.8))
|
2019-04-24 20:33:34 +00:00
|
|
|
.set(self.ids.button_help2, ui_widgets)
|
|
|
|
.was_clicked()
|
|
|
|
{
|
2019-05-04 17:24:26 +00:00
|
|
|
self.show.help = false;
|
2019-04-24 20:33:34 +00:00
|
|
|
};
|
2019-04-20 15:17:29 +00:00
|
|
|
}
|
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Bag button and nearby icons
|
2019-05-07 05:40:03 +00:00
|
|
|
match Buttons::new(
|
|
|
|
&self.show.open_windows,
|
|
|
|
self.show.map,
|
|
|
|
self.show.bag,
|
|
|
|
&self.imgs,
|
|
|
|
&self.fonts,
|
|
|
|
)
|
|
|
|
.set(self.ids.buttons, ui_widgets)
|
2019-05-03 13:49:05 +00:00
|
|
|
{
|
2019-05-04 17:24:26 +00:00
|
|
|
Some(buttons::Event::ToggleBag) => self.show.toggle_bag(),
|
|
|
|
Some(buttons::Event::ToggleSettings) => self.show.toggle_settings(),
|
2019-05-17 09:22:32 +00:00
|
|
|
Some(buttons::Event::ToggleCharacter) => self.show.toggle_char_window(),
|
2019-08-07 13:14:26 +00:00
|
|
|
Some(buttons::Event::ToggleSocial) => self.show.toggle_social(),
|
|
|
|
Some(buttons::Event::ToggleSpell) => self.show.toggle_spell(),
|
|
|
|
Some(buttons::Event::ToggleQuest) => self.show.toggle_quest(),
|
2019-05-04 17:24:26 +00:00
|
|
|
Some(buttons::Event::ToggleMap) => self.show.toggle_map(),
|
|
|
|
None => {}
|
2019-05-04 10:43:37 +00:00
|
|
|
}
|
2019-04-26 13:13:30 +00:00
|
|
|
|
2019-05-07 17:21:53 +00:00
|
|
|
// MiniMap
|
2019-10-16 11:39:41 +00:00
|
|
|
match MiniMap::new(&self.show, client, &self.imgs, self.world_map, &self.fonts)
|
2019-06-19 14:55:26 +00:00
|
|
|
.set(self.ids.minimap, ui_widgets)
|
|
|
|
{
|
2019-05-07 17:50:53 +00:00
|
|
|
Some(minimap::Event::Toggle) => self.show.toggle_mini_map(),
|
|
|
|
None => {}
|
|
|
|
}
|
2019-04-20 15:17:29 +00:00
|
|
|
|
|
|
|
// Bag contents
|
2019-05-04 17:24:26 +00:00
|
|
|
if self.show.bag {
|
2019-09-25 20:18:40 +00:00
|
|
|
match Bag::new(
|
|
|
|
client,
|
|
|
|
&self.imgs,
|
2019-10-09 19:28:05 +00:00
|
|
|
&self.item_imgs,
|
2019-09-25 20:18:40 +00:00
|
|
|
&self.fonts,
|
|
|
|
&self.rot_imgs,
|
|
|
|
tooltip_manager,
|
|
|
|
)
|
|
|
|
.set(self.ids.bag, ui_widgets)
|
|
|
|
{
|
2019-07-25 22:52:28 +00:00
|
|
|
Some(bag::Event::HudEvent(event)) => events.push(event),
|
2019-05-25 12:59:25 +00:00
|
|
|
Some(bag::Event::Close) => {
|
|
|
|
self.show.bag(false);
|
|
|
|
self.force_ungrab = true;
|
|
|
|
}
|
2019-05-04 10:43:37 +00:00
|
|
|
None => {}
|
2019-03-22 03:55:42 +00:00
|
|
|
}
|
2019-04-26 16:13:30 +00:00
|
|
|
}
|
2019-04-24 19:55:22 +00:00
|
|
|
|
2019-05-07 17:21:53 +00:00
|
|
|
// Skillbar
|
2019-05-19 18:07:50 +00:00
|
|
|
// Get player stats
|
2019-06-30 11:48:28 +00:00
|
|
|
if let Some(stats) = client
|
2019-05-19 18:07:50 +00:00
|
|
|
.state()
|
|
|
|
.ecs()
|
|
|
|
.read_storage::<comp::Stats>()
|
|
|
|
.get(client.entity())
|
2019-06-30 11:48:28 +00:00
|
|
|
{
|
2019-08-18 18:07:21 +00:00
|
|
|
Skillbar::new(global_state, &self.imgs, &self.fonts, stats)
|
|
|
|
.set(self.ids.skillbar, ui_widgets);
|
2019-06-30 11:48:28 +00:00
|
|
|
}
|
2019-05-04 06:07:23 +00:00
|
|
|
|
2019-05-03 16:56:18 +00:00
|
|
|
// Chat box
|
2019-09-02 04:54:44 +00:00
|
|
|
match Chat::new(&mut self.new_messages, &self.imgs, &self.fonts)
|
|
|
|
.and_then(self.force_chat_input.take(), |c, input| c.input(input))
|
|
|
|
.and_then(self.force_chat_cursor.take(), |c, pos| c.cursor_pos(pos))
|
|
|
|
.set(self.ids.chat, ui_widgets)
|
|
|
|
{
|
2019-05-03 16:56:18 +00:00
|
|
|
Some(chat::Event::SendMessage(message)) => {
|
|
|
|
events.push(Event::SendMessage(message));
|
|
|
|
}
|
2019-05-06 12:28:57 +00:00
|
|
|
Some(chat::Event::Focus(focus_id)) => {
|
|
|
|
self.to_focus = Some(Some(focus_id));
|
|
|
|
}
|
2019-05-03 16:56:18 +00:00
|
|
|
None => {}
|
|
|
|
}
|
2019-07-15 17:04:48 +00:00
|
|
|
|
2019-05-03 16:56:18 +00:00
|
|
|
self.new_messages = VecDeque::new();
|
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Windows
|
2019-03-15 04:55:52 +00:00
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Char Window will always appear at the left side. Other Windows default to the
|
|
|
|
// left side, but when the Char Window is opened they will appear to the right of it.
|
2019-03-15 04:55:52 +00:00
|
|
|
|
2019-05-03 16:56:18 +00:00
|
|
|
// Settings
|
2019-05-04 17:24:26 +00:00
|
|
|
if let Windows::Settings = self.show.open_windows {
|
2019-07-27 19:26:31 +00:00
|
|
|
for event in SettingsWindow::new(&global_state, &self.show, &self.imgs, &self.fonts)
|
|
|
|
.set(self.ids.settings_window, ui_widgets)
|
2019-03-15 04:55:52 +00:00
|
|
|
{
|
2019-05-18 23:55:06 +00:00
|
|
|
match event {
|
2019-08-18 18:07:21 +00:00
|
|
|
settings_window::Event::ToggleHelp => self.show.help = !self.show.help,
|
2019-05-18 23:55:06 +00:00
|
|
|
settings_window::Event::ToggleDebug => self.show.debug = !self.show.debug,
|
2019-06-02 02:17:36 +00:00
|
|
|
settings_window::Event::ChangeTab(tab) => self.show.open_setting_tab(tab),
|
2019-06-07 20:23:06 +00:00
|
|
|
settings_window::Event::Close => self.show.settings(false),
|
2019-06-05 15:57:48 +00:00
|
|
|
settings_window::Event::AdjustMousePan(sensitivity) => {
|
|
|
|
events.push(Event::AdjustMousePan(sensitivity));
|
|
|
|
}
|
|
|
|
settings_window::Event::AdjustMouseZoom(sensitivity) => {
|
|
|
|
events.push(Event::AdjustMouseZoom(sensitivity));
|
|
|
|
}
|
2019-10-02 13:09:05 +00:00
|
|
|
settings_window::Event::ToggleZoomInvert(zoom_inverted) => {
|
|
|
|
events.push(Event::ToggleZoomInvert(zoom_inverted));
|
|
|
|
}
|
2019-05-19 00:45:02 +00:00
|
|
|
settings_window::Event::AdjustViewDistance(view_distance) => {
|
|
|
|
events.push(Event::AdjustViewDistance(view_distance));
|
2019-05-18 23:55:06 +00:00
|
|
|
}
|
2019-07-07 00:15:22 +00:00
|
|
|
settings_window::Event::CrosshairTransp(crosshair_transp) => {
|
|
|
|
events.push(Event::CrosshairTransp(crosshair_transp));
|
|
|
|
}
|
2019-09-24 16:25:24 +00:00
|
|
|
settings_window::Event::AdjustMusicVolume(music_volume) => {
|
|
|
|
events.push(Event::AdjustMusicVolume(music_volume));
|
2019-09-24 16:18:09 +00:00
|
|
|
}
|
2019-09-24 16:25:24 +00:00
|
|
|
settings_window::Event::AdjustSfxVolume(sfx_volume) => {
|
|
|
|
events.push(Event::AdjustSfxVolume(sfx_volume));
|
2019-05-19 19:31:32 +00:00
|
|
|
}
|
2019-06-06 17:54:11 +00:00
|
|
|
settings_window::Event::MaximumFPS(max_fps) => {
|
2019-06-06 19:11:39 +00:00
|
|
|
events.push(Event::ChangeMaxFPS(max_fps));
|
2019-06-06 17:54:11 +00:00
|
|
|
}
|
2019-05-20 14:53:12 +00:00
|
|
|
settings_window::Event::ChangeAudioDevice(name) => {
|
|
|
|
events.push(Event::ChangeAudioDevice(name));
|
|
|
|
}
|
2019-07-23 01:02:57 +00:00
|
|
|
settings_window::Event::CrosshairType(crosshair_type) => {
|
|
|
|
events.push(Event::CrosshairType(crosshair_type));
|
|
|
|
}
|
2019-08-18 18:07:21 +00:00
|
|
|
settings_window::Event::ToggleXpBar(xp_bar) => {
|
|
|
|
events.push(Event::ToggleXpBar(xp_bar));
|
|
|
|
}
|
|
|
|
settings_window::Event::ToggleBarNumbers(bar_numbers) => {
|
|
|
|
events.push(Event::ToggleBarNumbers(bar_numbers));
|
|
|
|
}
|
|
|
|
settings_window::Event::ToggleShortcutNumbers(shortcut_numbers) => {
|
|
|
|
events.push(Event::ToggleShortcutNumbers(shortcut_numbers));
|
|
|
|
}
|
2019-07-26 02:28:53 +00:00
|
|
|
settings_window::Event::UiScale(scale_change) => {
|
|
|
|
events.push(Event::UiScale(scale_change));
|
|
|
|
}
|
2019-08-05 16:37:52 +00:00
|
|
|
settings_window::Event::AdjustFOV(new_fov) => {
|
|
|
|
events.push(Event::ChangeFOV(new_fov));
|
|
|
|
}
|
2019-09-26 07:28:40 +00:00
|
|
|
settings_window::Event::ChangeAaMode(new_aa_mode) => {
|
|
|
|
events.push(Event::ChangeAaMode(new_aa_mode));
|
|
|
|
}
|
2019-05-07 17:50:53 +00:00
|
|
|
}
|
2019-03-15 04:55:52 +00:00
|
|
|
}
|
|
|
|
}
|
2019-04-24 20:33:34 +00:00
|
|
|
|
2019-08-07 13:14:26 +00:00
|
|
|
// Social Window
|
|
|
|
if self.show.social {
|
|
|
|
for event in Social::new(
|
|
|
|
/*&global_state,*/ &self.show,
|
|
|
|
client,
|
|
|
|
&self.imgs,
|
|
|
|
&self.fonts,
|
|
|
|
)
|
|
|
|
.set(self.ids.social_window, ui_widgets)
|
2019-05-03 16:56:18 +00:00
|
|
|
{
|
2019-08-07 13:14:26 +00:00
|
|
|
match event {
|
|
|
|
social::Event::Close => self.show.social(false),
|
|
|
|
social::Event::ChangeSocialTab(social_tab) => {
|
|
|
|
self.show.open_social_tab(social_tab)
|
2019-05-07 05:40:03 +00:00
|
|
|
}
|
|
|
|
}
|
2019-03-15 04:55:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-30 20:43:55 +00:00
|
|
|
// Character Window
|
2019-08-07 13:14:26 +00:00
|
|
|
if self.show.character_window {
|
2019-07-27 13:04:34 +00:00
|
|
|
let ecs = client.state().ecs();
|
|
|
|
let stats = ecs.read_storage::<comp::Stats>();
|
|
|
|
let player_stats = stats.get(client.entity()).unwrap();
|
2019-08-07 13:14:26 +00:00
|
|
|
match CharacterWindow::new(&self.show, &player_stats, &self.imgs, &self.fonts)
|
2019-05-07 05:40:03 +00:00
|
|
|
.set(self.ids.character_window, ui_widgets)
|
2019-04-28 04:44:13 +00:00
|
|
|
{
|
2019-05-07 05:40:03 +00:00
|
|
|
Some(character_window::Event::Close) => {
|
2019-08-07 13:14:26 +00:00
|
|
|
self.show.character_window(false);
|
|
|
|
self.force_ungrab = true;
|
|
|
|
}
|
|
|
|
None => {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Spellbook
|
|
|
|
if self.show.spell {
|
|
|
|
match Spell::new(&self.show, client, &self.imgs, &self.fonts)
|
|
|
|
.set(self.ids.spell, ui_widgets)
|
|
|
|
{
|
|
|
|
Some(spell::Event::Close) => {
|
|
|
|
self.show.spell(false);
|
|
|
|
self.force_ungrab = true;
|
2019-05-07 05:40:03 +00:00
|
|
|
}
|
2019-05-03 16:56:18 +00:00
|
|
|
None => {}
|
2019-04-28 04:44:13 +00:00
|
|
|
}
|
2019-03-15 04:55:52 +00:00
|
|
|
}
|
|
|
|
|
2019-08-07 13:14:26 +00:00
|
|
|
// Quest Log
|
|
|
|
if self.show.quest {
|
|
|
|
match Quest::new(&self.show, client, &self.imgs, &self.fonts)
|
|
|
|
.set(self.ids.quest, ui_widgets)
|
|
|
|
{
|
|
|
|
Some(quest::Event::Close) => {
|
|
|
|
self.show.quest(false);
|
|
|
|
self.force_ungrab = true;
|
|
|
|
}
|
|
|
|
None => {}
|
|
|
|
}
|
|
|
|
}
|
2019-04-30 20:43:55 +00:00
|
|
|
// Map
|
2019-05-04 17:24:26 +00:00
|
|
|
if self.show.map {
|
2019-10-16 11:39:41 +00:00
|
|
|
match Map::new(&self.show, client, &self.imgs, self.world_map, &self.fonts)
|
2019-06-22 14:30:53 +00:00
|
|
|
.set(self.ids.map, ui_widgets)
|
|
|
|
{
|
2019-05-25 12:59:25 +00:00
|
|
|
Some(map::Event::Close) => {
|
|
|
|
self.show.map(false);
|
|
|
|
self.force_ungrab = true;
|
|
|
|
}
|
2019-05-03 16:56:18 +00:00
|
|
|
None => {}
|
2019-03-15 04:55:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-04 17:24:26 +00:00
|
|
|
if self.show.esc_menu {
|
2019-05-07 05:40:03 +00:00
|
|
|
match EscMenu::new(&self.imgs, &self.fonts).set(self.ids.esc_menu, ui_widgets) {
|
2019-06-02 02:17:36 +00:00
|
|
|
Some(esc_menu::Event::OpenSettings(tab)) => {
|
|
|
|
self.show.open_setting_tab(tab);
|
2019-05-03 16:56:18 +00:00
|
|
|
}
|
2019-05-25 12:59:25 +00:00
|
|
|
Some(esc_menu::Event::Close) => {
|
|
|
|
self.show.esc_menu = false;
|
|
|
|
self.show.want_grab = false;
|
|
|
|
self.force_ungrab = true;
|
|
|
|
}
|
2019-06-04 20:41:50 +00:00
|
|
|
Some(esc_menu::Event::Logout) => {
|
|
|
|
events.push(Event::Logout);
|
|
|
|
|
|
|
|
#[cfg(feature = "discord")]
|
|
|
|
{
|
2019-06-20 06:16:08 +00:00
|
|
|
discord::send_all(vec![
|
|
|
|
DiscordUpdate::Details("Menu".into()),
|
|
|
|
DiscordUpdate::State("Idling".into()),
|
|
|
|
DiscordUpdate::LargeImg("bg_main".into()),
|
|
|
|
]);
|
2019-06-04 20:41:50 +00:00
|
|
|
}
|
|
|
|
}
|
2019-05-03 16:56:18 +00:00
|
|
|
Some(esc_menu::Event::Quit) => events.push(Event::Quit),
|
2019-06-02 02:17:36 +00:00
|
|
|
Some(esc_menu::Event::CharacterSelection) => events.push(Event::CharacterSelection),
|
2019-05-07 05:40:03 +00:00
|
|
|
None => {}
|
2019-05-03 16:56:18 +00:00
|
|
|
}
|
|
|
|
}
|
2019-05-07 05:40:03 +00:00
|
|
|
|
2019-03-16 02:03:21 +00:00
|
|
|
events
|
2019-03-15 04:55:52 +00:00
|
|
|
}
|
|
|
|
|
2019-07-17 22:10:42 +00:00
|
|
|
pub fn new_message(&mut self, msg: ClientEvent) {
|
2019-05-03 16:56:18 +00:00
|
|
|
self.new_messages.push_back(msg);
|
2019-03-17 05:26:51 +00:00
|
|
|
}
|
|
|
|
|
2019-07-26 02:28:53 +00:00
|
|
|
pub fn scale_change(&mut self, scale_change: ScaleChange) -> ScaleMode {
|
|
|
|
let scale_mode = match scale_change {
|
|
|
|
ScaleChange::Adjust(scale) => ScaleMode::Absolute(scale),
|
|
|
|
ScaleChange::ToAbsolute => self.ui.scale().scaling_mode_as_absolute(),
|
|
|
|
ScaleChange::ToRelative => self.ui.scale().scaling_mode_as_relative(),
|
|
|
|
};
|
|
|
|
self.ui.set_scaling_mode(scale_mode);
|
|
|
|
scale_mode
|
|
|
|
}
|
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Checks if a TextEdit widget has the keyboard captured.
|
2019-05-07 03:25:25 +00:00
|
|
|
fn typing(&self) -> bool {
|
|
|
|
if let Some(id) = self.ui.widget_capturing_keyboard() {
|
|
|
|
self.ui
|
|
|
|
.widget_graph()
|
|
|
|
.widget(id)
|
2019-05-14 06:43:07 +00:00
|
|
|
.filter(|c| {
|
|
|
|
c.type_id == std::any::TypeId::of::<<widget::TextEdit as Widget>::State>()
|
|
|
|
})
|
2019-05-07 03:25:25 +00:00
|
|
|
.is_some()
|
|
|
|
} else {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-20 18:08:39 +00:00
|
|
|
pub fn handle_event(&mut self, event: WinEvent, global_state: &mut GlobalState) -> bool {
|
|
|
|
let cursor_grabbed = global_state.window.is_cursor_grabbed();
|
2019-05-07 20:00:03 +00:00
|
|
|
let handled = match event {
|
2019-03-22 03:55:42 +00:00
|
|
|
WinEvent::Ui(event) => {
|
2019-05-07 03:25:25 +00:00
|
|
|
if (self.typing() && event.is_keyboard() && self.show.ui)
|
2019-04-20 18:08:39 +00:00
|
|
|
|| !(cursor_grabbed && event.is_keyboard_or_mouse())
|
2019-04-01 20:31:29 +00:00
|
|
|
{
|
2019-03-22 03:55:42 +00:00
|
|
|
self.ui.handle_event(event);
|
|
|
|
}
|
|
|
|
true
|
2019-04-01 20:31:29 +00:00
|
|
|
}
|
2019-05-26 16:23:24 +00:00
|
|
|
WinEvent::InputUpdate(GameInput::ToggleInterface, true) if !self.typing() => {
|
2019-05-26 16:10:35 +00:00
|
|
|
self.show.toggle_ui();
|
|
|
|
true
|
|
|
|
}
|
2019-05-26 16:23:24 +00:00
|
|
|
WinEvent::InputUpdate(GameInput::ToggleCursor, true) if !self.typing() => {
|
2019-05-26 16:10:35 +00:00
|
|
|
self.force_ungrab = !self.force_ungrab;
|
|
|
|
true
|
|
|
|
}
|
2019-05-04 17:24:26 +00:00
|
|
|
_ if !self.show.ui => false,
|
2019-04-20 18:08:39 +00:00
|
|
|
WinEvent::Zoom(_) => !cursor_grabbed && !self.ui.no_widget_capturing_mouse(),
|
2019-05-25 14:39:27 +00:00
|
|
|
|
|
|
|
WinEvent::InputUpdate(GameInput::Enter, true) => {
|
2019-05-07 03:25:25 +00:00
|
|
|
self.ui.focus_widget(if self.typing() {
|
2019-05-06 12:28:57 +00:00
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some(self.ids.chat)
|
|
|
|
});
|
2019-03-30 02:15:27 +00:00
|
|
|
true
|
|
|
|
}
|
2019-05-25 14:39:27 +00:00
|
|
|
WinEvent::InputUpdate(GameInput::Escape, true) => {
|
2019-05-07 03:25:25 +00:00
|
|
|
if self.typing() {
|
2019-03-30 02:15:27 +00:00
|
|
|
self.ui.focus_widget(None);
|
|
|
|
} else {
|
2019-04-09 16:30:13 +00:00
|
|
|
// Close windows on esc
|
2019-05-07 20:00:03 +00:00
|
|
|
self.show.toggle_windows();
|
2019-03-30 02:15:27 +00:00
|
|
|
}
|
2019-04-09 16:30:13 +00:00
|
|
|
true
|
2019-04-02 04:54:27 +00:00
|
|
|
}
|
2019-05-25 14:39:27 +00:00
|
|
|
|
|
|
|
// Press key while not typing
|
|
|
|
WinEvent::InputUpdate(key, true) if !self.typing() => match key {
|
2019-07-05 16:21:11 +00:00
|
|
|
GameInput::Command => {
|
|
|
|
self.force_chat_input = Some("/".to_owned());
|
|
|
|
self.force_chat_cursor = Some(Index { line: 0, char: 1 });
|
|
|
|
self.ui.focus_widget(Some(self.ids.chat));
|
|
|
|
true
|
|
|
|
}
|
2019-05-25 14:39:27 +00:00
|
|
|
GameInput::Map => {
|
2019-05-04 17:24:26 +00:00
|
|
|
self.show.toggle_map();
|
2019-04-09 16:30:13 +00:00
|
|
|
true
|
|
|
|
}
|
2019-05-25 14:39:27 +00:00
|
|
|
GameInput::Bag => {
|
2019-05-04 17:24:26 +00:00
|
|
|
self.show.toggle_bag();
|
2019-04-09 16:30:13 +00:00
|
|
|
true
|
|
|
|
}
|
2019-05-25 14:39:27 +00:00
|
|
|
GameInput::QuestLog => {
|
2019-08-07 13:14:26 +00:00
|
|
|
self.show.toggle_quest();
|
2019-04-09 16:30:13 +00:00
|
|
|
true
|
|
|
|
}
|
2019-05-25 14:39:27 +00:00
|
|
|
GameInput::CharacterWindow => {
|
2019-05-17 09:22:32 +00:00
|
|
|
self.show.toggle_char_window();
|
2019-04-09 16:30:13 +00:00
|
|
|
true
|
|
|
|
}
|
2019-05-25 14:39:27 +00:00
|
|
|
GameInput::Social => {
|
2019-08-07 13:14:26 +00:00
|
|
|
self.show.toggle_social();
|
2019-04-09 16:30:13 +00:00
|
|
|
true
|
|
|
|
}
|
2019-05-25 14:39:27 +00:00
|
|
|
GameInput::Spellbook => {
|
2019-08-07 13:14:26 +00:00
|
|
|
self.show.toggle_spell();
|
2019-04-09 16:30:13 +00:00
|
|
|
true
|
|
|
|
}
|
2019-05-25 14:39:27 +00:00
|
|
|
GameInput::Settings => {
|
2019-05-04 17:24:26 +00:00
|
|
|
self.show.toggle_settings();
|
2019-04-09 16:30:13 +00:00
|
|
|
true
|
|
|
|
}
|
2019-05-25 14:39:27 +00:00
|
|
|
GameInput::Help => {
|
2019-05-04 17:24:26 +00:00
|
|
|
self.show.toggle_help();
|
2019-04-09 16:30:13 +00:00
|
|
|
true
|
|
|
|
}
|
2019-05-25 14:39:27 +00:00
|
|
|
GameInput::ToggleDebug => {
|
2019-05-18 21:16:35 +00:00
|
|
|
self.show.debug = !self.show.debug;
|
|
|
|
true
|
|
|
|
}
|
2019-05-25 22:00:38 +00:00
|
|
|
GameInput::ToggleIngameUi => {
|
2019-05-21 04:55:20 +00:00
|
|
|
self.show.ingame = !self.show.ingame;
|
|
|
|
true
|
|
|
|
}
|
2019-04-09 16:30:13 +00:00
|
|
|
_ => false,
|
|
|
|
},
|
2019-05-25 14:39:27 +00:00
|
|
|
// Else the player is typing in chat
|
2019-06-06 14:48:41 +00:00
|
|
|
WinEvent::InputUpdate(_key, _) => self.typing(),
|
2019-05-07 03:25:25 +00:00
|
|
|
WinEvent::Char(_) => self.typing(),
|
2019-06-29 01:39:47 +00:00
|
|
|
WinEvent::Focused(state) => {
|
|
|
|
self.force_ungrab = !state;
|
|
|
|
true
|
|
|
|
}
|
2019-05-25 14:39:27 +00:00
|
|
|
|
2019-03-22 03:55:42 +00:00
|
|
|
_ => false,
|
2019-05-07 20:00:03 +00:00
|
|
|
};
|
2019-05-17 09:22:32 +00:00
|
|
|
// Handle cursor grab.
|
2019-06-29 01:39:47 +00:00
|
|
|
global_state
|
|
|
|
.window
|
|
|
|
.grab_cursor(!self.force_ungrab && self.show.want_grab);
|
2019-05-08 20:15:58 +00:00
|
|
|
|
|
|
|
handled
|
2019-03-15 04:55:52 +00:00
|
|
|
}
|
|
|
|
|
2019-05-23 08:18:25 +00:00
|
|
|
pub fn maintain(
|
|
|
|
&mut self,
|
2019-05-14 06:43:07 +00:00
|
|
|
client: &Client,
|
2019-05-23 08:18:25 +00:00
|
|
|
global_state: &mut GlobalState,
|
2019-05-23 09:30:00 +00:00
|
|
|
debug_info: DebugInfo,
|
2019-05-20 06:09:20 +00:00
|
|
|
camera: &Camera,
|
2019-05-23 08:18:25 +00:00
|
|
|
) -> Vec<Event> {
|
2019-05-06 12:28:57 +00:00
|
|
|
if let Some(maybe_id) = self.to_focus.take() {
|
|
|
|
self.ui.focus_widget(maybe_id);
|
|
|
|
}
|
2019-05-14 06:43:07 +00:00
|
|
|
let events = self.update_layout(client, global_state, debug_info);
|
2019-05-20 06:09:20 +00:00
|
|
|
let (view_mat, _, _) = camera.compute_dependents(client);
|
|
|
|
let fov = camera.get_fov();
|
2019-05-24 12:43:03 +00:00
|
|
|
self.ui.maintain(
|
|
|
|
&mut global_state.window.renderer_mut(),
|
|
|
|
Some((view_mat, fov)),
|
|
|
|
);
|
2019-10-09 19:28:05 +00:00
|
|
|
|
|
|
|
// Check if item images need to be reloaded
|
|
|
|
self.item_imgs.reload_if_changed(&mut self.ui);
|
|
|
|
|
2019-03-16 02:03:21 +00:00
|
|
|
events
|
2019-03-15 04:55:52 +00:00
|
|
|
}
|
|
|
|
|
2019-05-14 06:43:07 +00:00
|
|
|
pub fn render(&self, renderer: &mut Renderer, globals: &Consts<Globals>) {
|
2019-06-29 14:30:46 +00:00
|
|
|
// Don't show anything if the UI is toggled off.
|
|
|
|
if self.show.ui {
|
|
|
|
self.ui.render(renderer, Some(globals));
|
|
|
|
}
|
2019-03-15 04:55:52 +00:00
|
|
|
}
|
|
|
|
}
|
2019-04-25 13:10:01 +00:00
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Get the text to show in the help window and use the
|
|
|
|
// length of the longest line to resize the window.
|
2019-10-06 19:19:08 +00:00
|
|
|
/*fn get_help_text(cs: &ControlSettings) -> String {
|
2019-05-01 22:15:43 +00:00
|
|
|
format!(
|
|
|
|
"{free_cursor:?} = Free cursor\n\
|
|
|
|
{escape:?} = Open/close menus\n\
|
|
|
|
\n\
|
|
|
|
{help:?} = Toggle this window\n\
|
|
|
|
{toggle_interface:?} = Toggle interface\n\
|
|
|
|
\n\
|
|
|
|
{chat:?} = Open chat\n\
|
|
|
|
Mouse Wheel = Scroll chat/zoom",
|
|
|
|
free_cursor = cs.toggle_cursor,
|
|
|
|
escape = cs.escape,
|
|
|
|
help = cs.help,
|
|
|
|
toggle_interface = cs.toggle_interface,
|
|
|
|
chat = cs.enter
|
|
|
|
)
|
2019-10-06 19:19:08 +00:00
|
|
|
}*/
|