diff --git a/.gitignore b/.gitignore index 0371dc57a4..782816ee6f 100644 --- a/.gitignore +++ b/.gitignore @@ -20,3 +20,7 @@ **/server_conf.toml **/keybinds.toml assets/voxygen +UI3.rar +assets.rar +*.rar +assets/voxygen diff --git a/common/src/net/mod.rs b/common/src/net/mod.rs index 9439bf74e1..c1c7fd4107 100644 --- a/common/src/net/mod.rs +++ b/common/src/net/mod.rs @@ -1,8 +1,5 @@ pub mod data; -pub mod error; pub mod post; -pub mod postbox; -pub mod postoffice; // Reexports pub use self::{ diff --git a/common/src/net/post.rs b/common/src/net/post.rs index 4bd4b3bfd1..b9160ec35b 100644 --- a/common/src/net/post.rs +++ b/common/src/net/post.rs @@ -116,7 +116,7 @@ impl PostOffice { pub fn new_connections(&mut self) -> impl ExactSizeIterator> { let mut conns = VecDeque::new(); - if let Some(_) = self.err { + if self.err.is_some() { return conns.into_iter(); } @@ -132,7 +132,7 @@ impl PostOffice { for event in events { match event.token() { // Keep reading new postboxes from the channel - POSTBOX_TOKEN => loop { + POSTBOX_TOK => loop { match self.postbox_rx.try_recv() { Ok(Ok(conn)) => conns.push_back(conn), Err(TryRecvError::Empty) => break, @@ -141,7 +141,7 @@ impl PostOffice { return conns.into_iter(); }, Ok(Err(err)) => { - self.err = Some(err.into()); + self.err = Some(err); return conns.into_iter(); }, } @@ -209,7 +209,7 @@ pub struct PostBox { impl PostBox { pub fn to_server>(addr: A) -> Result { - Self::from_tcpstream(TcpStream::connect(&addr.into())?) + Self::from_tcpstream(TcpStream::from_stream(std::net::TcpStream::connect(&addr.into())?)?) } fn from_tcpstream(tcp_stream: TcpStream) -> Result { @@ -254,7 +254,7 @@ impl PostBox { pub fn new_messages(&mut self) -> impl ExactSizeIterator { let mut msgs = VecDeque::new(); - if let Some(_) = self.err { + if self.err.is_some() { return msgs.into_iter(); } @@ -270,7 +270,7 @@ impl PostBox { for event in events { match event.token() { // Keep reading new messages from the channel - RECV_TOKEN => loop { + RECV_TOK => loop { match self.recv_rx.try_recv() { Ok(Ok(msg)) => msgs.push_back(msg), Err(TryRecvError::Empty) => break, @@ -279,7 +279,7 @@ impl PostBox { return msgs.into_iter(); }, Ok(Err(err)) => { - self.err = Some(err.into()); + self.err = Some(err); return msgs.into_iter(); }, } @@ -322,17 +322,16 @@ fn postbox_worker( for event in &events { match event.token() { - CTRL_TOK => loop { + CTRL_TOK => match ctrl_rx.try_recv() { Ok(CtrlMsg::Shutdown) => { break 'work; }, - Err(TryRecvError::Empty) => break, + Err(TryRecvError::Empty) => (), Err(err) => { recv_tx.send(Err(err.into()))?; break 'work; }, - } }, SEND_TOK => loop { match send_rx.try_recv() { @@ -340,7 +339,7 @@ fn postbox_worker( let mut msg_bytes = match bincode::serialize(&outgoing_msg) { Ok(bytes) => bytes, Err(err) => { - recv_tx.send(Err((*err).into())); + recv_tx.send(Err((*err).into()))?; break 'work; }, }; @@ -355,7 +354,7 @@ fn postbox_worker( match tcp_stream.write_all(&packet) { Ok(()) => {}, Err(err) => { - recv_tx.send(Err(err.into())); + recv_tx.send(Err(err.into()))?; break 'work; }, } @@ -368,11 +367,11 @@ fn postbox_worker( match tcp_stream.take_error() { Ok(None) => {}, Ok(Some(err)) => { - recv_tx.send(Err(err.into())); + recv_tx.send(Err(err.into()))?; break 'work; }, Err(err) => { - recv_tx.send(Err(err.into())); + recv_tx.send(Err(err.into()))?; break 'work; }, } @@ -380,7 +379,7 @@ fn postbox_worker( RecvState::ReadHead(head) => if head.len() == 8 { let len = usize::from_le_bytes(<[u8; 8]>::try_from(head.as_slice()).unwrap()); if len > MAX_MSG_BYTES { - recv_tx.send(Err(Error::InvalidMsg)); + recv_tx.send(Err(Error::InvalidMsg))?; break 'work; } else if len == 0 { recv_state = RecvState::ReadHead(Vec::with_capacity(8)); @@ -427,7 +426,7 @@ fn postbox_worker( } } - tcp_stream.shutdown(Shutdown::Both); + tcp_stream.shutdown(Shutdown::Both)?; Ok(()) } @@ -453,6 +452,21 @@ fn connect() { assert_eq!(postoffice.error(), None); } +#[test] +fn connect_fail() { + let listen_addr = ([0; 4], 12345); + let connect_addr = ([127, 0, 0, 1], 12212); + + let mut postoffice = PostOffice::::bind(listen_addr).unwrap(); + + // We should start off with 0 incoming connections + thread::sleep(Duration::from_millis(250)); + assert_eq!(postoffice.new_connections().len(), 0); + assert_eq!(postoffice.error(), None); + + assert!(PostBox::::to_server(connect_addr).is_err()); +} + #[test] fn connection_count() { let srv_addr = ([127, 0, 0, 1], 12346); @@ -469,7 +483,7 @@ fn connection_count() { postboxes.push(PostBox::::to_server(srv_addr).unwrap()); } - // 10 postboxes created, we should have 10 + // 5 postboxes created, we should have 5 thread::sleep(Duration::from_millis(3500)); let incoming = postoffice.new_connections(); assert_eq!(incoming.len(), 5); diff --git a/voxygen/src/hud/chat.rs b/voxygen/src/hud/chat.rs index f007a09d65..56fbab83a1 100644 --- a/voxygen/src/hud/chat.rs +++ b/voxygen/src/hud/chat.rs @@ -50,15 +50,9 @@ impl Chat { // (i.e. the player is viewing new messages) // If so scroll down when new messages are added fn scroll_new_messages(&self, ui_widgets: &mut UiCell) { - if let Some(scroll) = ui_widgets - .widget_graph() - .widget(self.ids.message_box) - .and_then(|widget| widget.maybe_y_scroll_state) - { - // If previously scrolled to the bottom stay there - if self.scrolled_to_bottom(ui_widgets) { - self.scroll_to_bottom(ui_widgets); - } + // If previously scrolled to the bottom stay there + if self.scrolled_to_bottom(ui_widgets) { + self.scroll_to_bottom(ui_widgets); } } fn scrolled_to_bottom(&self, ui_widgets: &UiCell) -> bool { @@ -76,7 +70,7 @@ impl Chat { fn scroll_to_bottom(&self, ui_widgets: &mut UiCell) { ui_widgets.scroll_widget(self.ids.message_box, [0.0, std::f64::MAX]); } - pub fn update_layout(&mut self, ui_widgets: &mut UiCell, font: FontId, imgs: &super::Imgs) -> Option { + pub(super) fn update_layout(&mut self, ui_widgets: &mut UiCell, font: FontId, imgs: &super::Imgs) -> Option { // Maintain scrolling if self.new_messages { self.scroll_new_messages(ui_widgets); @@ -123,7 +117,7 @@ impl Chat { Text::new(&self.messages[item.i]) .font_size(14) .font_id(font) - .rgba(220.0, 220.0, 220.0, 1.0), + .rgba(0.86 , 0.86, 0.86, 1.0), ui_widgets, ) } diff --git a/voxygen/src/hud/mod.rs b/voxygen/src/hud/mod.rs index 4055fb967f..5110e6c9d7 100644 --- a/voxygen/src/hud/mod.rs +++ b/voxygen/src/hud/mod.rs @@ -2,25 +2,34 @@ mod chat; use crate::{ render::Renderer, - ui::{ScaleMode, Ui, ToggleButton}, - window::{Event as WinEvent, Window, Key}, + ui::{ScaleMode, ToggleButton, Ui}, + window::{Event as WinEvent, Key, Window}, }; use conrod_core::{ - color, - event::Input, + color, Color, image::Id as ImgId, text::font::Id as FontId, - widget::{Button, Image, Text, Rectangle}, + widget::{Button, Image, Rectangle, Scrollbar, Text}, widget_ids, Colorable, Labelable, Positionable, Sizeable, Widget, }; widget_ids! { struct Ids { - //Bag and Inventory + // Test + bag_space_add, + inventorytest_button, + inventorytest_button_label, + + // Bag and Inventory bag, bag_contents, bag_close, bag_map_open, + inv_alignment, + inv_grid, + inv_scrollbar, + inv_slot_0, + inv_slot[], //help help, help_bg, @@ -53,6 +62,12 @@ widget_ids! { sb_grid_r, sb_grid_bg_l, sb_grid_bg_r, + xp_bar_progress, + health_bar_color, + mana_bar_color, + // Level Display + level_text, + next_level_text, //Window Frames window_frame_0, window_frame_1, @@ -96,10 +111,23 @@ widget_ids! { spellbook_title, //4 Charwindow charwindow_frame, + charwindow, charwindow_bg, charwindow_icon, charwindow_close, charwindow_title, + charwindow_tab_bg, + charwindow_tab1, + charwindow_tab1_title, + charwindow_tab1_level, + charwindow_tab1_exp, + charwindow_tab1_stats, + charwindow_tab1_statnames, + charwindow_tab1_stats_numbers, + charwindow_tab1_expbar, + charwindow_rectangle, + charwindow_exp_rectangle, + charwindow_exp_progress_rectangle, //5 Quest-Log questlog_frame, questlog_bg, @@ -110,7 +138,7 @@ widget_ids! { } // TODO: make macro to mimic widget_ids! for images ids or find another solution to simplify addition of new images. -struct Imgs { +pub(self) struct Imgs { //Missing: ActionBar, Health/Mana/Energy Bar & Char Window BG/Frame // Bag bag: ImgId, @@ -120,6 +148,8 @@ struct Imgs { bag_open_hover: ImgId, bag_open_press: ImgId, bag_contents: ImgId, + inv_grid: ImgId, + inv_slot: ImgId, // Close button close_button: ImgId, @@ -176,6 +206,7 @@ struct Imgs { button_blank: ImgId, button_blue_mo: ImgId, button_blue_press: ImgId, + window_bg: ImgId, // Social-Window social_bg: ImgId, social_icon: ImgId, @@ -187,8 +218,14 @@ struct Imgs { spellbook_bg: ImgId, spellbook_icon: ImgId, // Char Window - charwindow_bg: ImgId, + charwindow: ImgId, charwindow_icon: ImgId, + charwindow_tab_bg: ImgId, + charwindow_tab: ImgId, + charwindow_expbar: ImgId, + progress_frame: ImgId, + progress: ImgId, + // Quest-Log Window questlog_bg: ImgId, questlog_icon: ImgId, @@ -198,6 +235,12 @@ struct Imgs { chat_arrow: ImgId, chat_arrow_mo: ImgId, chat_arrow_press: ImgId, + chat_arrow_up: ImgId, + chat_arrow_up_mo: ImgId, + chat_arrow_up_press: ImgId, + chat_arrow_down: ImgId, + chat_arrow_down_mo: ImgId, + chat_arrow_down_press: ImgId, } impl Imgs { fn new(ui: &mut Ui, renderer: &mut Renderer) -> Imgs { @@ -217,6 +260,8 @@ impl Imgs { bag_open_hover: load("element/buttons/bag/open_hover.png"), bag_open_press: load("element/buttons/bag/open_press.png"), bag_contents: load("element/frames/bag.png"), + inv_grid: load("element/frames/inv_grid.png"), + inv_slot: load("element/buttons/inv_slot.png"), // Close button close_button: load("element/buttons/x.png"), @@ -275,6 +320,9 @@ impl Imgs { button_blue_mo: load("element/buttons/blue_mo.png"), button_blue_press: load("element/buttons/blue_press.png"), + // Window BG + window_bg: load("element/misc_backgrounds/window_bg.png"), + //Social Window social_bg: load("element/misc_backgrounds/small_bg.png"), social_icon: load("element/icons/social.png"), @@ -289,8 +337,13 @@ impl Imgs { spellbook_icon: load("element/icons/spellbook.png"), //Char Window - charwindow_bg: load("element/misc_backgrounds/small_bg.png"), + charwindow: load("element/misc_backgrounds/charwindow.png"), charwindow_icon: load("element/icons/charwindow.png"), + charwindow_tab_bg: load("element/frames/tab.png"), + charwindow_tab: load("element/buttons/tab.png"), + charwindow_expbar: load("element/misc_backgrounds/small_bg.png"), + progress_frame: load("element/frames/progress_bar.png"), + progress: load("element/misc_backgrounds/progress.png"), //Quest-Log Window questlog_bg: load("element/misc_backgrounds/small_bg.png"), @@ -300,6 +353,12 @@ impl Imgs { chat_arrow: load("element/buttons/arrow/chat_arrow.png"), chat_arrow_mo: load("element/buttons/arrow/chat_arrow_mo.png"), chat_arrow_press: load("element/buttons/arrow/chat_arrow_press.png"), + chat_arrow_up: load("element/buttons/arrow/chat_arrow_up.png"), + chat_arrow_up_mo: load("element/buttons/arrow/chat_arrow_up_mo.png"), + chat_arrow_up_press: load("element/buttons/arrow/chat_arrow_up_press.png"), + chat_arrow_down: load("element/buttons/arrow/chat_arrow_down.png"), + chat_arrow_down_mo: load("element/buttons/arrow/chat_arrow_down_mo.png"), + chat_arrow_down_press: load("element/buttons/arrow/chat_arrow_down_press.png"), } } } @@ -322,7 +381,7 @@ pub enum Event { // TODO: maybe replace this with bitflags // map not here because it currently is displayed over the top of other open windows enum Windows { - Settings, // display settings window + Settings, // display settings window CharacterAnd(Option), // show character window + optionally another Small(Small), None, @@ -334,7 +393,6 @@ enum Small { Questlog, } - pub struct Hud { ui: Ui, ids: Ids, @@ -349,9 +407,20 @@ pub struct Hud { menu_open: bool, open_windows: Windows, map_open: bool, - settings_tab: SettingsTab + show_ui: bool, + inventory_space: u32, + xp_percentage: f64, + hp_percentage: f64, + mana_percentage: f64, + inventorytest_button: bool, + settings_tab: SettingsTab, } +//#[inline] +//pub fn rgba_bytes(r: u8, g: u8, b: u8, a: f32) -> Color { + //Color::Rgba(r as f32 / 255.0, g as f32 / 255.0, b as f32 / 255.0, a) +//} + impl Hud { pub fn new(window: &mut Window) -> Self { let mut ui = Ui::new(window).unwrap(); @@ -386,13 +455,19 @@ impl Hud { typing: false, cursor_grabbed: true, settings_tab: SettingsTab::Interface, - show_help: true, + show_help: false, bag_open: false, menu_open: false, map_open: false, + show_ui: true, + inventorytest_button: false, + inventory_space: 0, open_windows: Windows::None, font_metamorph, font_opensans, + xp_percentage: 0.4, + hp_percentage: 1.0, + mana_percentage: 1.0, } } @@ -400,291 +475,392 @@ impl Hud { let mut events = Vec::new(); let ref mut ui_widgets = self.ui.set_widgets(); - // Chat box - if let Some(msg) = self.chat.update_layout(ui_widgets, self.font_opensans, &self.imgs) { - events.push(Event::SendMessage(msg)); - } - // Help Text - if self.show_help { - Image::new(self.imgs.window_frame_2) - .top_left_with_margins_on(ui_widgets.window, 5.0, 5.0) - .w_h(300.0, 300.0) - .set(self.ids.help_bg, ui_widgets); + const TEXT_COLOR: Color = Color::Rgba(0.86, 0.86, 0.86, 0.8); + const HP_COLOR: Color = Color::Rgba(0.33, 0.63, 0.0, 1.0); + const MANA_COLOR: Color = Color::Rgba(0.42, 0.41, 0.66, 1.0); + const XP_COLOR: Color = Color::Rgba(0.59, 0.41, 0.67, 1.0); - Text::new( - "Tab = Free Cursor \n\ - Esc = Open/Close Menus \n\ - Q = Back to Login \n\ - \n\ - F1 = Toggle this Window \n\ - F2 = Toggle Interface \n\ - \n\ - M = Map \n\ - I = Inventory \n\ - L = Quest-Log \n\ - C = Character Window \n\ - O = Social \n\ - P = Spellbook \n\ - N = Settings") - .rgba(220.0, 220.0, 220.0, 0.8) - .top_left_with_margins_on(self.ids.help_bg, 20.0,20.0) - .font_id(self.font_opensans) - .font_size(18) - .set(self.ids.help, ui_widgets); - if Button::image(self.imgs.button_dark) - .w_h(50.0, 30.0) - .bottom_right_with_margins_on(self.ids.help_bg, 10.0, 10.0) - .hover_image(self.imgs.button_dark_hover) - .press_image(self.imgs.button_dark_press) - .label("Close") - .label_font_size(10) - .label_rgba(220.0, 220.0, 220.0, 0.8) - .set(self.ids.button_help2, ui_widgets) - .was_clicked() + + if self.show_ui { + // Add Bag-Space Button + if self.inventorytest_button { + if Button::image(self.imgs.mmap_button) + .w_h(100.0, 100.0) + .middle_of(ui_widgets.window) + .label("1 Up!") + .label_font_size(20) + .hover_image(self.imgs.mmap_button_hover) + .press_image(self.imgs.mmap_button_press) + .set(self.ids.bag_space_add, ui_widgets) + .was_clicked() + { + self.inventory_space = self.inventory_space + 1; + }; + } + // Chat box + if let Some(msg) = self + .chat + .update_layout(ui_widgets, self.font_opensans, &self.imgs) { - self.show_help = false; - }; - } + events.push(Event::SendMessage(msg)); + } + // Help Text + if self.show_help { + Image::new(self.imgs.window_frame_2) + .top_left_with_margins_on(ui_widgets.window, 5.0, 5.0) + .w_h(300.0, 300.0) + .set(self.ids.help_bg, ui_widgets); - // Minimap frame and bg - Image::new(self.imgs.mmap_frame_bg) - .w_h(1750.0 / 8.0, 1650.0 / 8.0) - .top_right_with_margins_on(ui_widgets.window, 20.0, 30.0) - .set(self.ids.mmap_frame_bg, ui_widgets); - - Image::new(self.imgs.mmap_frame) - .w_h(1750.0 / 8.0, 1650.0 / 8.0) - .top_right_with_margins_on(ui_widgets.window, 20.0, 30.0) - .set(self.ids.mmap_frame, ui_widgets); - - Image::new(self.imgs.mmap_icons) - .w_h(448.0 / 14.93, 2688.0 / 14.93) - .right_from(self.ids.mmap_frame, 0.0) - .align_bottom_of(self.ids.mmap_frame) - .set(self.ids.mmap_icons, ui_widgets); - // Title - // TODO Make it display the actual Location - Text::new("Unknown Location") - .mid_top_with_margin_on(self.ids.mmap_frame, 5.0) - .font_size(14) - .rgba(220.0, 220.0, 220.0, 0.8) - .set(self.ids.mmap_location, ui_widgets); - - // Minimap Buttons - - //0 Settings - if Button::image(self.imgs.mmap_button) - .w_h(448.0 / 15.0, 448.0 / 15.0) - .top_right_with_margins_on(self.ids.mmap_icons, 0.0, 0.0) - .hover_image(self.imgs.mmap_button_hover) - .press_image(self.imgs.mmap_button_press) - .set(self.ids.mmap_button_0, ui_widgets) - .was_clicked() - { - self.open_windows = match self.open_windows { - Windows::Settings => Windows::None, - _ => Windows::Settings, - }; - self.bag_open = false; - }; - // 2 Map - if Button::image(self.imgs.mmap_button) - .w_h(448.0 / 15.0, 448.0 / 15.0) - .down_from(self.ids.mmap_button_1, 0.0) - .hover_image(self.imgs.mmap_button_hover) - .press_image(self.imgs.mmap_button_press) - .set(self.ids.mmap_button_2, ui_widgets) - .was_clicked() - { - self.map_open = !self.map_open; - self.bag_open = false; - }; - - // Other Windows can only be accessed, when Settings are closed. Opening Settings will close all other Windows including the Bag. - // Opening the Map won't close the windows displayed before. - - - if match self.open_windows { - Windows::Settings => false, - _ => true, - } && self.map_open == false { - //1 Social - if Button::image(self.imgs.mmap_button) - .w_h(448.0 / 15.0, 448.0 / 15.0) - .down_from(self.ids.mmap_button_0, 0.0) - .hover_image(self.imgs.mmap_button_hover) - .press_image(self.imgs.mmap_button_press) - .set(self.ids.mmap_button_1, ui_widgets) - .was_clicked() - { - self.open_windows = match self.open_windows { - Windows::Small(Small::Social) => Windows::None, - Windows::None | Windows::Small(_) => Windows::Small(Small::Social), - Windows::CharacterAnd(small) => match small { - Some(Small::Social) => Windows::CharacterAnd(None), - _ => Windows::CharacterAnd(Some(Small::Social)), - } - Windows::Settings => unreachable!(), + Text::new( + "Tab = Free Cursor \n\ + Esc = Open/Close Menus \n\ + Q = Back to Login \n\ + \n\ + F1 = Toggle this Window \n\ + F2 = Toggle Interface \n\ + \n\ + M = Map \n\ + B = Bag \n\ + L = Quest-Log \n\ + C = Character Window \n\ + O = Social \n\ + P = Spellbook \n\ + N = Settings", + ) + .color(TEXT_COLOR) + .top_left_with_margins_on(self.ids.help_bg, 20.0, 20.0) + .font_id(self.font_opensans) + .font_size(18) + .set(self.ids.help, ui_widgets); + if Button::image(self.imgs.button_dark) + .w_h(50.0, 30.0) + .bottom_right_with_margins_on(self.ids.help_bg, 10.0, 10.0) + .hover_image(self.imgs.button_dark_hover) + .press_image(self.imgs.button_dark_press) + .label("Close") + .label_font_size(10) + .label_color(TEXT_COLOR) + .set(self.ids.button_help2, ui_widgets) + .was_clicked() + { + self.show_help = false; }; } - //3 Spellbook + // Minimap frame and bg + Image::new(self.imgs.mmap_frame_bg) + .w_h(1750.0 / 8.0, 1650.0 / 8.0) + .top_right_with_margins_on(ui_widgets.window, 5.0, 30.0) + .set(self.ids.mmap_frame_bg, ui_widgets); + + Image::new(self.imgs.mmap_frame) + .w_h(1750.0 / 8.0, 1650.0 / 8.0) + .top_right_with_margins_on(ui_widgets.window, 5.0, 30.0) + .set(self.ids.mmap_frame, ui_widgets); + + Image::new(self.imgs.mmap_icons) + .w_h(448.0 / 14.93, 2688.0 / 14.93) + .right_from(self.ids.mmap_frame, 0.0) + .align_bottom_of(self.ids.mmap_frame) + .set(self.ids.mmap_icons, ui_widgets); + // Title + // Make it display the actual location + Text::new("Uncanny Valley") + .mid_top_with_margin_on(self.ids.mmap_frame, 5.0) + .font_size(14) + .color(TEXT_COLOR) + .set(self.ids.mmap_location, ui_widgets); + + // Minimap Buttons + + //0 Settings if Button::image(self.imgs.mmap_button) .w_h(448.0 / 15.0, 448.0 / 15.0) - .down_from(self.ids.mmap_button_2, 0.0) + .top_right_with_margins_on(self.ids.mmap_icons, 0.0, 0.0) .hover_image(self.imgs.mmap_button_hover) .press_image(self.imgs.mmap_button_press) - .set(self.ids.mmap_button_3, ui_widgets) + .set(self.ids.mmap_button_0, ui_widgets) .was_clicked() { self.open_windows = match self.open_windows { - Windows::Small(Small::Spellbook) => Windows::None, - Windows::None | Windows::Small(_) => Windows::Small(Small::Spellbook), - Windows::CharacterAnd(small) => match small { - Some(Small::Spellbook) => Windows::CharacterAnd(None), - _ => Windows::CharacterAnd(Some(Small::Spellbook)), - } - Windows::Settings => unreachable!(), + Windows::Settings => Windows::None, + _ => Windows::Settings, }; - } - //4 Char-Window + self.bag_open = false; + }; + // 2 Map if Button::image(self.imgs.mmap_button) .w_h(448.0 / 15.0, 448.0 / 15.0) - .down_from(self.ids.mmap_button_3, 0.0) + .down_from(self.ids.mmap_button_1, 0.0) .hover_image(self.imgs.mmap_button_hover) .press_image(self.imgs.mmap_button_press) - .set(self.ids.mmap_button_4, ui_widgets) + .set(self.ids.mmap_button_2, ui_widgets) .was_clicked() { - self.open_windows = match self.open_windows { - Windows::CharacterAnd(small) => match small { - Some(small) => Windows::Small(small), - None => Windows::None, - }, - Windows::Small(small) => Windows::CharacterAnd(Some(small)), - Windows::None => Windows::CharacterAnd(None), - Windows::Settings => unreachable!() + self.map_open = !self.map_open; + self.bag_open = false; + }; + + // Other Windows can only be accessed, when Settings are closed. Opening Settings will close all other Windows including the Bag. + // Opening the Map won't close the windows displayed before. + + if match self.open_windows { + Windows::Settings => false, + _ => true, + } && self.map_open == false + { + //1 Social + if Button::image(self.imgs.mmap_button) + .w_h(448.0 / 15.0, 448.0 / 15.0) + .down_from(self.ids.mmap_button_0, 0.0) + .hover_image(self.imgs.mmap_button_hover) + .press_image(self.imgs.mmap_button_press) + .set(self.ids.mmap_button_1, ui_widgets) + .was_clicked() + { + self.open_windows = match self.open_windows { + Windows::Small(Small::Social) => Windows::None, + Windows::None | Windows::Small(_) => Windows::Small(Small::Social), + Windows::CharacterAnd(small) => match small { + Some(Small::Social) => Windows::CharacterAnd(None), + _ => Windows::CharacterAnd(Some(Small::Social)), + }, + Windows::Settings => Windows::Settings, + }; + } + + //3 Spellbook + if Button::image(self.imgs.mmap_button) + .w_h(448.0 / 15.0, 448.0 / 15.0) + .down_from(self.ids.mmap_button_2, 0.0) + .hover_image(self.imgs.mmap_button_hover) + .press_image(self.imgs.mmap_button_press) + .set(self.ids.mmap_button_3, ui_widgets) + .was_clicked() + { + self.open_windows = match self.open_windows { + Windows::Small(Small::Spellbook) => Windows::None, + Windows::None | Windows::Small(_) => Windows::Small(Small::Spellbook), + Windows::CharacterAnd(small) => match small { + Some(Small::Spellbook) => Windows::CharacterAnd(None), + _ => Windows::CharacterAnd(Some(Small::Spellbook)), + }, + Windows::Settings => Windows::Settings, + }; + } + //4 Char-Window + if Button::image(self.imgs.mmap_button) + .w_h(448.0 / 15.0, 448.0 / 15.0) + .down_from(self.ids.mmap_button_3, 0.0) + .hover_image(self.imgs.mmap_button_hover) + .press_image(self.imgs.mmap_button_press) + .set(self.ids.mmap_button_4, ui_widgets) + .was_clicked() + { + self.open_windows = match self.open_windows { + Windows::CharacterAnd(small) => match small { + Some(small) => Windows::Small(small), + None => Windows::None, + }, + Windows::Small(small) => Windows::CharacterAnd(Some(small)), + Windows::None => Windows::CharacterAnd(None), + Windows::Settings => Windows::Settings, + } + } + //5 Quest-Log + if Button::image(self.imgs.mmap_button) + .w_h(448.0 / 15.0, 448.0 / 15.0) + .down_from(self.ids.mmap_button_4, 0.0) + .hover_image(self.imgs.mmap_button_hover) + .press_image(self.imgs.mmap_button_press) + .set(self.ids.mmap_button_5, ui_widgets) + .was_clicked() + { + self.open_windows = match self.open_windows { + Windows::Small(Small::Questlog) => Windows::None, + Windows::None | Windows::Small(_) => Windows::Small(Small::Questlog), + Windows::CharacterAnd(small) => match small { + Some(Small::Questlog) => Windows::CharacterAnd(None), + _ => Windows::CharacterAnd(Some(Small::Questlog)), + }, + Windows::Settings => Windows::Settings, + }; } } - //5 Quest-Log - if Button::image(self.imgs.mmap_button) - .w_h(448.0 / 15.0, 448.0 / 15.0) - .down_from(self.ids.mmap_button_4, 0.0) - .hover_image(self.imgs.mmap_button_hover) - .press_image(self.imgs.mmap_button_press) - .set(self.ids.mmap_button_5, ui_widgets) - .was_clicked() - { - self.open_windows = match self.open_windows { - Windows::Small(Small::Questlog) => Windows::None, - Windows::None | Windows::Small(_) => Windows::Small(Small::Questlog), - Windows::CharacterAnd(small) => match small { - Some(Small::Questlog) => Windows::CharacterAnd(None), - _ => Windows::CharacterAnd(Some(Small::Questlog)), - } - Windows::Settings => unreachable!(), - }; - } - } - // Skillbar Module + // Skillbar Module - // Experience-Bar - Image::new(self.imgs.xp_bar) - .w_h(2688.0 / 4.0, 116.0 / 4.0) - .mid_bottom_of(ui_widgets.window) - .set(self.ids.xp_bar, ui_widgets); + // Experience-Bar + Image::new(self.imgs.xp_bar) + .w_h(2688.0 / 6.0, 116.0 / 6.0) + .mid_bottom_of(ui_widgets.window) + .set(self.ids.xp_bar, ui_widgets); - // Left Grid - Image::new(self.imgs.sb_grid) - .w_h(2240.0 / 8.0, 448.0 / 8.0) - .up_from(self.ids.xp_bar, 0.0) - .align_left_of(self.ids.xp_bar) - .set(self.ids.sb_grid_l, ui_widgets); + Rectangle::fill_with([406.0 * (self.xp_percentage), 5.0], XP_COLOR) // "W=406*[Exp. %]" + .top_left_with_margins_on(self.ids.xp_bar, 5.0, 21.0) + .set(self.ids.xp_bar_progress, ui_widgets); - Image::new(self.imgs.sb_grid_bg) - .w_h(2240.0 / 8.0, 448.0 / 8.0) - .middle_of(self.ids.sb_grid_l) - .set(self.ids.sb_grid_bg_l, ui_widgets); + // Left Grid + Image::new(self.imgs.sb_grid) + .w_h(2240.0 / 12.0, 448.0 / 12.0) + .up_from(self.ids.xp_bar, 0.0) + .align_left_of(self.ids.xp_bar) + .set(self.ids.sb_grid_l, ui_widgets); - // Right Grid - Image::new(self.imgs.sb_grid) - .w_h(2240.0 / 8.0, 448.0 / 8.0) - .up_from(self.ids.xp_bar, 0.0) - .align_right_of(self.ids.xp_bar) - .set(self.ids.sb_grid_r, ui_widgets); + Image::new(self.imgs.sb_grid_bg) + .w_h(2240.0 / 12.0, 448.0 / 12.0) + .middle_of(self.ids.sb_grid_l) + .set(self.ids.sb_grid_bg_l, ui_widgets); - Image::new(self.imgs.sb_grid_bg) - .w_h(2240.0 / 8.0, 448.0 / 8.0) - .middle_of(self.ids.sb_grid_r) - .set(self.ids.sb_grid_bg_r, ui_widgets); + // Right Grid + Image::new(self.imgs.sb_grid) + .w_h(2240.0 / 12.0, 448.0 / 12.0) + .up_from(self.ids.xp_bar, 0.0) + .align_right_of(self.ids.xp_bar) + .set(self.ids.sb_grid_r, ui_widgets); - // Right and Left Click - Image::new(self.imgs.l_click) - .w_h(224.0 / 4.0, 320.0 / 4.0) - .right_from(self.ids.sb_grid_bg_l, 0.0) - .align_bottom_of(self.ids.sb_grid_bg_l) - .set(self.ids.l_click, ui_widgets); + Image::new(self.imgs.sb_grid_bg) + .w_h(2240.0 / 12.0, 448.0 / 12.0) + .middle_of(self.ids.sb_grid_r) + .set(self.ids.sb_grid_bg_r, ui_widgets); - Image::new(self.imgs.r_click) - .w_h(224.0 / 4.0, 320.0 / 4.0) - .left_from(self.ids.sb_grid_bg_r, 0.0) - .align_bottom_of(self.ids.sb_grid_bg_r) - .set(self.ids.r_click, ui_widgets); + // Right and Left Click + Image::new(self.imgs.l_click) + .w_h(224.0 / 6.0, 320.0 / 6.0) + .right_from(self.ids.sb_grid_bg_l, 0.0) + .align_bottom_of(self.ids.sb_grid_bg_l) + .set(self.ids.l_click, ui_widgets); - // Health and mana bars - Image::new(self.imgs.health_bar) - .w_h(1120.0 / 4.0, 96.0 / 4.0) - .left_from(self.ids.l_click, 0.0) - .align_top_of(self.ids.l_click) - .set(self.ids.health_bar, ui_widgets); + Image::new(self.imgs.r_click) + .w_h(224.0 / 6.0, 320.0 / 6.0) + .left_from(self.ids.sb_grid_bg_r, 0.0) + .align_bottom_of(self.ids.sb_grid_bg_r) + .set(self.ids.r_click, ui_widgets); - Image::new(self.imgs.mana_bar) - .w_h(1120.0 / 4.0, 96.0 / 4.0) - .right_from(self.ids.r_click, 0.0) - .align_top_of(self.ids.r_click) - .set(self.ids.mana_bar, ui_widgets); + // Health Bar + Image::new(self.imgs.health_bar) + .w_h(1120.0 / 6.0, 96.0 / 6.0) + .left_from(self.ids.l_click, 0.0) + .align_top_of(self.ids.l_click) + .set(self.ids.health_bar, ui_widgets); - // Buffs/Debuffs + // Filling + Rectangle::fill_with([182.0 * (self.hp_percentage), 6.0], HP_COLOR) // "W=182.0 * [Health. %]" + .top_right_with_margins_on(self.ids.health_bar, 5.0, 0.0) + .set(self.ids.health_bar_color, ui_widgets); - // Buffs - // Debuffs + // Mana Bar + Image::new(self.imgs.mana_bar) + .w_h(1120.0 / 6.0, 96.0 / 6.0) + .right_from(self.ids.r_click, 0.0) + .align_top_of(self.ids.r_click) + .set(self.ids.mana_bar, ui_widgets); - // Bag contents - if self.bag_open { - // Contents - Image::new(self.imgs.bag_contents) - .w_h(1504.0 / 4.0, 1760.0 / 4.0) - .bottom_right_with_margins_on(ui_widgets.window, 88.0, 68.0) - .set(self.ids.bag_contents, ui_widgets); + // Filling + Rectangle::fill_with([182.0 * (self.mana_percentage), 6.0], MANA_COLOR) // "W=182.0 * [Mana. %]" + .top_left_with_margins_on(self.ids.mana_bar, 5.0, 0.0) + .set(self.ids.mana_bar_color, ui_widgets); - // X-button - if Button::image(self.imgs.close_button) - .w_h(244.0 * 0.22 / 3.0, 244.0 * 0.22 / 3.0) - .hover_image(self.imgs.close_button_hover) - .press_image(self.imgs.close_button_press) - .top_right_with_margins_on(self.ids.bag_contents, 5.0, 17.0) - .set(self.ids.bag_close, ui_widgets) - .was_clicked() - { - self.bag_open = false; + + // Buffs/Debuffs + + // Buffs + + // Debuffs + + // Level Display + + // Insert actual Level here + Text::new("1") + .left_from(self.ids.xp_bar, -15.0) + .font_size(14) + .color(TEXT_COLOR) + .set(self.ids.level_text, ui_widgets); + + // Insert next Level here + Text::new("2") + .right_from(self.ids.xp_bar, -15.0) + .font_size(14) + .color(TEXT_COLOR) + .set(self.ids.next_level_text, ui_widgets); + + // Bag contents + if self.bag_open { + // Contents + Image::new(self.imgs.bag_contents) + .w_h(307.0, 545.0) + .bottom_right_with_margins_on(ui_widgets.window, 90.0, 5.0) + .set(self.ids.bag_contents, ui_widgets); + + // Alignment for Grid + Rectangle::fill_with([246.0, 465.0], color::TRANSPARENT) + .top_left_with_margins_on(self.ids.bag_contents, 27.0, 23.0) + .scroll_kids() + .scroll_kids_vertically() + .set(self.ids.inv_alignment, ui_widgets); + // Grid + Image::new(self.imgs.inv_grid) + .w_h(232.0, 1104.0) + .mid_top_with_margin_on(self.ids.inv_alignment, 0.0) + .set(self.ids.inv_grid, ui_widgets); + Scrollbar::y_axis(self.ids.inv_alignment) + .thickness(5.0) + .rgba(0.86, 0.86, 0.86, 0.1) + .set(self.ids.inv_scrollbar, ui_widgets); + + // X-button + if Button::image(self.imgs.close_button) + .w_h(244.0 * 0.22 / 3.0, 244.0 * 0.22 / 3.0) + .hover_image(self.imgs.close_button_hover) + .press_image(self.imgs.close_button_press) + .top_right_with_margins_on(self.ids.bag_contents, 5.0, 17.0) + .set(self.ids.bag_close, ui_widgets) + .was_clicked() + { + self.bag_open = false; + } + + if self.inventory_space > 0 { + // First Slot + Button::image(self.imgs.inv_slot) + .top_left_with_margins_on(self.ids.inv_grid, 5.0, 5.0) + .w_h(40.0, 40.0) + .set(self.ids.inv_slot_0, ui_widgets); + } + // if self.ids.inv_slot.len() < self.inventory_space { + // self.ids.inv_slot.resize(self.inventory_space, &mut ui_widgets.widget_id_generator()); + //} + + //let num = self.ids.inv_slot.len(); + //println!("self.ids.inv_slot.len(): {:?}", num); + //if num > 0 { + //Button::image(self.imgs.inv_slot) + //.top_left_with_margins_on(self.ids.inv_grid, 5.0, 5.0) + //.w_h(40.0, 40.0) + //.set(self.ids.inv_slot[0], ui_widgets); + //} + //for i in 1..5 { + //Button::image(self.imgs.inv_slot) + //.right(10.0) + //.label(&format!("{}", i + 1)) + //.label_color(TEXT_COLOR) + //.label_font_size(5) + //.set(self.ids.inv_slot[i], ui_widgets);} } } // Bag if !self.map_open { - self.bag_open = - ToggleButton::new(self.bag_open, self.imgs.bag, self.imgs.bag_open) - .bottom_right_with_margin_on(ui_widgets.window, 20.0) - .hover_images(self.imgs.bag_hover, self.imgs.bag_open_hover) - .press_images(self.imgs.bag_press, self.imgs.bag_open_press) - .w_h(420.0 / 4.0, 480.0 / 4.0) - .set(self.ids.bag, ui_widgets); - + self.bag_open = ToggleButton::new(self.bag_open, self.imgs.bag, self.imgs.bag_open) + .bottom_right_with_margins_on(ui_widgets.window, 5.0, 5.0) + .hover_images(self.imgs.bag_hover, self.imgs.bag_open_hover) + .press_images(self.imgs.bag_press, self.imgs.bag_open_press) + .w_h(420.0 / 6.0, 480.0 / 6.0) + .set(self.ids.bag, ui_widgets); } else { Image::new(self.imgs.bag) - .bottom_right_with_margin_on(ui_widgets.window, 20.0) - .w_h(420.0 / 4.0, 480.0 / 4.0) + .bottom_right_with_margins_on(ui_widgets.window, 5.0, 5.0) + .w_h(420.0 / 6.0, 480.0 / 6.0) .set(self.ids.bag_map_open, ui_widgets); } @@ -718,7 +894,7 @@ impl Hud { Text::new("Settings") .mid_top_with_margin_on(self.ids.settings_bg, 10.0) .font_size(30) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.settings_title, ui_widgets); // Icon Image::new(self.imgs.settings_icon) @@ -727,95 +903,135 @@ impl Hud { .set(self.ids.settings_icon, ui_widgets); // TODO: Find out if we can remove this // Alignment Rectangle - Rectangle::fill_with([1008.0/ 2.5, 1616.0 / 2.5], color::TRANSPARENT) - .top_left_with_margins_on(self.ids.settings_bg, 77.0, 205.0) - .set(self.ids.rectangle, ui_widgets); + Rectangle::fill_with([1008.0 / 2.5, 1616.0 / 2.5], color::TRANSPARENT) + .top_left_with_margins_on(self.ids.settings_bg, 77.0, 205.0) + .set(self.ids.rectangle, ui_widgets); //1 Interface//////////////////////////// - if Button::image(if let SettingsTab::Interface = self.settings_tab {self.imgs.button_blue_mo} else {self.imgs.button_blank}) - .w_h(304.0 / 2.5, 80.0 / 2.5) - .hover_image(self.imgs.button_blue_mo) - .press_image(self.imgs.button_blue_press) - .top_left_with_margins_on(self.ids.settings_bg, 78.0, 50.0) - .label("Interface") - .label_font_size(14) - .label_rgba(220.0, 220.0, 220.0, 0.8) - .set(self.ids.interface, ui_widgets) - .was_clicked() + if Button::image(if let SettingsTab::Interface = self.settings_tab { + self.imgs.button_blue_mo + } else { + self.imgs.button_blank + }) + .w_h(304.0 / 2.5, 80.0 / 2.5) + .hover_image(self.imgs.button_blue_mo) + .press_image(self.imgs.button_blue_press) + .top_left_with_margins_on(self.ids.settings_bg, 78.0, 50.0) + .label("Interface") + .label_font_size(14) + .label_color(TEXT_COLOR) + .set(self.ids.interface, ui_widgets) + .was_clicked() { self.settings_tab = SettingsTab::Interface; } //Toggle Help if let SettingsTab::Interface = self.settings_tab { - self.show_help = ToggleButton::new(self.show_help, self.imgs.check, self.imgs.check_checked) - .w_h(288.0 / 24.0, 288.0 / 24.0) - .top_left_with_margins_on(self.ids.rectangle, 15.0, 15.0) - .hover_images(self.imgs.check_checked_mo, self.imgs.check_mo) - .press_images(self.imgs.check_press, self.imgs.check_press) - .set(self.ids.button_help, ui_widgets); + self.show_help = + ToggleButton::new(self.show_help, self.imgs.check, self.imgs.check_checked) + .w_h(288.0 / 24.0, 288.0 / 24.0) + .top_left_with_margins_on(self.ids.rectangle, 15.0, 15.0) + .hover_images(self.imgs.check_checked_mo, self.imgs.check_mo) + .press_images(self.imgs.check_press, self.imgs.check_press) + .set(self.ids.button_help, ui_widgets); Text::new("Show Help") - .x_relative_to(self.ids.button_help, 55.0) + .right_from(self.ids.button_help, 10.0) .font_size(12) + .font_id(self.font_opensans) .graphics_for(self.ids.button_help) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.show_help_label, ui_widgets); + + self.inventorytest_button = ToggleButton::new( + self.inventorytest_button, + self.imgs.check, + self.imgs.check_checked, + ) + .w_h(288.0 / 24.0, 288.0 / 24.0) + .top_left_with_margins_on(self.ids.rectangle, 40.0, 15.0) + .hover_images(self.imgs.check_checked_mo, self.imgs.check_mo) + .press_images(self.imgs.check_press, self.imgs.check_press) + .set(self.ids.inventorytest_button, ui_widgets); + Text::new("Show Inventory Test Button") + .right_from(self.ids.inventorytest_button, 10.0) + .font_size(12) + .font_id(self.font_opensans) + .graphics_for(self.ids.inventorytest_button) + .color(TEXT_COLOR) + .set(self.ids.inventorytest_button_label, ui_widgets); } //2 Gameplay//////////////// - if Button::image(if let SettingsTab::Gameplay = self.settings_tab {self.imgs.button_blue_mo} else {self.imgs.button_blank}) - .w_h(304.0 / 2.5, 80.0 / 2.5) - .hover_image(self.imgs.button_blue_mo) - .press_image(self.imgs.button_blue_press) - .down_from(self.ids.interface, 1.0) - .label("Gameplay") - .label_font_size(14) - .label_rgba(220.0, 220.0, 220.0, 0.8) - .set(self.ids.gameplay, ui_widgets) - .was_clicked() + if Button::image(if let SettingsTab::Gameplay = self.settings_tab { + self.imgs.button_blue_mo + } else { + self.imgs.button_blank + }) + .w_h(304.0 / 2.5, 80.0 / 2.5) + .hover_image(self.imgs.button_blue_mo) + .press_image(self.imgs.button_blue_press) + .down_from(self.ids.interface, 1.0) + .label("Gameplay") + .label_font_size(14) + .label_color(TEXT_COLOR) + .set(self.ids.gameplay, ui_widgets) + .was_clicked() { self.settings_tab = SettingsTab::Gameplay; } //3 Controls///////////////////// - if Button::image(if let SettingsTab::Controls = self.settings_tab {self.imgs.button_blue_mo} else {self.imgs.button_blank}) - .w_h(304.0 / 2.5, 80.0 / 2.5) - .hover_image(self.imgs.button_blue_mo) - .press_image(self.imgs.button_blue_press) - .down_from(self.ids.gameplay, 1.0) - .label("Controls") - .label_font_size(14) - .label_rgba(220.0, 220.0, 220.0, 0.8) - .set(self.ids.controls, ui_widgets) - .was_clicked() + if Button::image(if let SettingsTab::Controls = self.settings_tab { + self.imgs.button_blue_mo + } else { + self.imgs.button_blank + }) + .w_h(304.0 / 2.5, 80.0 / 2.5) + .hover_image(self.imgs.button_blue_mo) + .press_image(self.imgs.button_blue_press) + .down_from(self.ids.gameplay, 1.0) + .label("Controls") + .label_font_size(14) + .label_color(TEXT_COLOR) + .set(self.ids.controls, ui_widgets) + .was_clicked() { self.settings_tab = SettingsTab::Controls; } //4 Video//////////////////////////////// - if Button::image(if let SettingsTab::Video = self.settings_tab {self.imgs.button_blue_mo} else {self.imgs.button_blank}) - .w_h(304.0 / 2.5, 80.0 / 2.5) - .hover_image(self.imgs.button_blue_mo) - .press_image(self.imgs.button_blue_press) - .down_from(self.ids.controls, 1.0) - .label("Video") - .label_font_size(14) - .label_rgba(220.0, 220.0, 220.0, 0.8) - .set(self.ids.video, ui_widgets) - .was_clicked() + if Button::image(if let SettingsTab::Video = self.settings_tab { + self.imgs.button_blue_mo + } else { + self.imgs.button_blank + }) + .w_h(304.0 / 2.5, 80.0 / 2.5) + .hover_image(self.imgs.button_blue_mo) + .press_image(self.imgs.button_blue_press) + .down_from(self.ids.controls, 1.0) + .label("Video") + .label_font_size(14) + .label_color(TEXT_COLOR) + .set(self.ids.video, ui_widgets) + .was_clicked() { self.settings_tab = SettingsTab::Video; } //5 Sound/////////////////////////////// - if Button::image(if let SettingsTab::Sound = self.settings_tab {self.imgs.button_blue_mo} else {self.imgs.button_blank}) - .w_h(304.0 / 2.5, 80.0 / 2.5) - .hover_image(self.imgs.button_blue_mo) - .press_image(self.imgs.button_blue_press) - .down_from(self.ids.video, 1.0) - .label("Sound") - .label_font_size(14) - .label_rgba(220.0, 220.0, 220.0, 0.8) - .set(self.ids.sound, ui_widgets) - .was_clicked() + if Button::image(if let SettingsTab::Sound = self.settings_tab { + self.imgs.button_blue_mo + } else { + self.imgs.button_blank + }) + .w_h(304.0 / 2.5, 80.0 / 2.5) + .hover_image(self.imgs.button_blue_mo) + .press_image(self.imgs.button_blue_press) + .down_from(self.ids.video, 1.0) + .label("Sound") + .label_font_size(14) + .label_color(TEXT_COLOR) + .set(self.ids.sound, ui_widgets) + .was_clicked() { self.settings_tab = SettingsTab::Sound; } @@ -865,13 +1081,13 @@ impl Hud { self.open_windows = match self.open_windows { Windows::Small(_) => Windows::None, Windows::CharacterAnd(_) => Windows::CharacterAnd(None), - _ => unreachable!(), + _ => Windows::Settings, } } // Title Text::new("Social") .mid_top_with_margin_on(self.ids.social_frame, 7.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.social_title, ui_widgets); } Small::Spellbook => { @@ -912,13 +1128,13 @@ impl Hud { self.open_windows = match self.open_windows { Windows::Small(_) => Windows::None, Windows::CharacterAnd(_) => Windows::CharacterAnd(None), - _ => unreachable!(), + _ => Windows::Settings, } } // Title Text::new("Spellbook") .mid_top_with_margin_on(self.ids.spellbook_frame, 7.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.spellbook_title, ui_widgets); } Small::Questlog => { @@ -959,41 +1175,44 @@ impl Hud { self.open_windows = match self.open_windows { Windows::Small(_) => Windows::None, Windows::CharacterAnd(_) => Windows::CharacterAnd(None), - _ => unreachable!(), + _ => Windows::Settings, } } // Title Text::new("Quest-Log") .mid_top_with_margin_on(self.ids.questlog_frame, 7.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.questlog_title, ui_widgets); } } - - } - //4 Char-Window + // 4 Char-Window if let Windows::CharacterAnd(small) = self.open_windows { - //Frame + // Frame Image::new(self.imgs.window_frame) - .top_left_with_margins_on(ui_widgets.window, 200.0, 90.0) + .top_left_with_margins_on(ui_widgets.window, 200.0, 215.0) .w_h(1648.0 / 4.0, 1952.0 / 4.0) .set(self.ids.charwindow_frame, ui_widgets); - //BG - Image::new(self.imgs.charwindow_bg) - .w_h(1648.0 / 4.0, 1952.0 / 4.0) - .middle_of(self.ids.charwindow_frame) + // BG + Image::new(self.imgs.window_bg) + .w_h(348.0, 404.0) + .mid_top_with_margin_on(self.ids.charwindow_frame, 48.0) .set(self.ids.charwindow_bg, ui_widgets); - //Icon - Image::new(self.imgs.charwindow_icon) - .w_h(224.0 / 3.0, 224.0 / 3.0) - .top_left_with_margins_on(self.ids.charwindow_frame, -10.0, -10.0) - .set(self.ids.charwindow_icon, ui_widgets); + // Overlay + Image::new(self.imgs.charwindow) + .middle_of(self.ids.charwindow_bg) + .set(self.ids.charwindow, ui_widgets); - //X-Button + //Icon + //Image::new(self.imgs.charwindow_icon) + //.w_h(224.0 / 3.0, 224.0 / 3.0) + //.top_left_with_margins_on(self.ids.charwindow_frame, -10.0, -10.0) + //.set(self.ids.charwindow_icon, ui_widgets); + + // X-Button if Button::image(self.imgs.close_button) .w_h(244.0 * 0.22 / 4.0, 244.0 * 0.22 / 4.0) .hover_image(self.imgs.close_button_hover) @@ -1007,11 +1226,87 @@ impl Hud { None => Windows::None, } } + // Title Text::new("Character Name") //Add in actual Character Name .mid_top_with_margin_on(self.ids.charwindow_frame, 7.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.charwindow_title, ui_widgets); + // Tab BG + Image::new(self.imgs.charwindow_tab_bg) + .w_h(205.0, 412.0) + .mid_left_with_margin_on(self.ids.charwindow_frame, -205.0) + .set(self.ids.charwindow_tab_bg, ui_widgets); + // Tab Rectangle + Rectangle::fill_with([192.0, 371.0], color::rgba(0.0, 0.0, 0.0, 0.8)) + .top_right_with_margins_on(self.ids.charwindow_tab_bg, 20.0, 0.0) + .set(self.ids.charwindow_rectangle, ui_widgets); + // Tab Button + Button::image(self.imgs.charwindow_tab) + .w_h(65.0, 23.0) + .top_left_with_margins_on(self.ids.charwindow_tab_bg, -18.0, 2.0) + .label("Stats") + .label_color(TEXT_COLOR) + .label_font_id(self.font_opensans) + .label_font_size(14) + .set(self.ids.charwindow_tab1, ui_widgets); + Text::new("1") //Add in actual Character Level + .mid_top_with_margin_on(self.ids.charwindow_rectangle, 10.0) + .font_id(self.font_opensans) + .font_size(30) + .color(TEXT_COLOR) + .set(self.ids.charwindow_tab1_level, ui_widgets); + // Exp-Bar Background + Rectangle::fill_with([170.0, 10.0], color::BLACK) + .mid_top_with_margin_on(self.ids.charwindow_rectangle, 50.0) + .set(self.ids.charwindow_exp_rectangle, ui_widgets); + // Exp-Bar Progress + Rectangle::fill_with([170.0 * (self.xp_percentage), 6.0], XP_COLOR) // 0.8 = Experience percantage + .mid_left_with_margin_on(self.ids.charwindow_tab1_expbar, 1.0) + .set(self.ids.charwindow_exp_progress_rectangle, ui_widgets); + // Exp-Bar Foreground Frame + Image::new(self.imgs.progress_frame) + .w_h(170.0, 10.0) + .middle_of(self.ids.charwindow_exp_rectangle) + .set(self.ids.charwindow_tab1_expbar, ui_widgets); + // Exp-Text + Text::new("120/170") // Shows the Exp / Exp to reach the next level + .mid_top_with_margin_on(self.ids.charwindow_tab1_expbar, 10.0) + .font_id(self.font_opensans) + .font_size(15) + .color(TEXT_COLOR) + .set(self.ids.charwindow_tab1_exp, ui_widgets); + + // Stats + Text::new( + "Stamina\n\ + \n\ + Strength\n\ + \n\ + Dexterity\n\ + \n\ + Intelligence", + ) + .top_left_with_margins_on(self.ids.charwindow_rectangle, 100.0, 20.0) + .font_id(self.font_opensans) + .font_size(16) + .color(TEXT_COLOR) + .set(self.ids.charwindow_tab1_statnames, ui_widgets); + + Text::new( + "1234\n\ + \n\ + 12312\n\ + \n\ + 12414\n\ + \n\ + 124124", + ) + .right_from(self.ids.charwindow_tab1_statnames, 10.0) + .font_id(self.font_opensans) + .font_size(16) + .color(TEXT_COLOR) + .set(self.ids.charwindow_tab1_stats, ui_widgets); } //2 Map @@ -1049,7 +1344,7 @@ impl Hud { Text::new("Map") .mid_top_with_margin_on(self.ids.map_frame, -7.0) .font_size(50) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.map_title, ui_widgets); } @@ -1073,8 +1368,9 @@ impl Hud { .hover_image(self.imgs.button_dark_hover) .press_image(self.imgs.button_dark_press) .label("Settings") - .label_rgba(220.0, 220.0, 220.0, 0.8) - .label_font_size(20) + .label_y(conrod_core::position::Relative::Scalar(2.0)) + .label_color(TEXT_COLOR) + .label_font_size(17) .set(self.ids.menu_button_1, ui_widgets) .was_clicked() { @@ -1088,8 +1384,9 @@ impl Hud { .hover_image(self.imgs.button_dark_hover) .press_image(self.imgs.button_dark_press) .label("Controls") - .label_rgba(220.0, 220.0, 220.0, 0.8) - .label_font_size(20) + .label_y(conrod_core::position::Relative::Scalar(2.0)) + .label_color(TEXT_COLOR) + .label_font_size(17) .set(self.ids.menu_button_2, ui_widgets) .was_clicked() { @@ -1102,8 +1399,9 @@ impl Hud { .hover_image(self.imgs.button_dark_hover) .press_image(self.imgs.button_dark_press) .label("Servers") - .label_rgba(220.0, 220.0, 220.0, 0.8) - .label_font_size(20) + .label_y(conrod_core::position::Relative::Scalar(2.0)) + .label_color(TEXT_COLOR) + .label_font_size(17) .set(self.ids.menu_button_3, ui_widgets) .was_clicked() { @@ -1116,8 +1414,9 @@ impl Hud { .hover_image(self.imgs.button_dark_hover) .press_image(self.imgs.button_dark_press) .label("Logout") - .label_rgba(220.0, 220.0, 220.0, 0.8) - .label_font_size(20) + .label_y(conrod_core::position::Relative::Scalar(2.0)) + .label_color(TEXT_COLOR) + .label_font_size(17) .set(self.ids.menu_button_4, ui_widgets) .was_clicked() { @@ -1130,20 +1429,23 @@ impl Hud { .hover_image(self.imgs.button_dark_hover) .press_image(self.imgs.button_dark_press) .label("Quit") - .label_rgba(220.0, 220.0, 220.0, 0.8) - .label_font_size(20) + .label_y(conrod_core::position::Relative::Scalar(2.0)) + .label_color(TEXT_COLOR) + .label_font_size(17) .set(self.ids.menu_button_5, ui_widgets) .was_clicked() { events.push(Event::Quit); }; } + // update whether keyboard is captured - self.typing = if let Some(widget_id) = ui_widgets.global_input().current.widget_capturing_keyboard { - widget_id == self.chat.input_box_id() - } else { - false - }; + self.typing = + if let Some(widget_id) = ui_widgets.global_input().current.widget_capturing_keyboard { + widget_id == self.chat.input_box_id() + } else { + false + }; events } @@ -1155,6 +1457,70 @@ impl Hud { pub fn toggle_menu(&mut self) { self.menu_open = !self.menu_open; } + pub fn toggle_bag(&mut self) { + self.bag_open = !self.bag_open + } + pub fn toggle_questlog(&mut self) { + self.open_windows = match self.open_windows { + Windows::Small(Small::Questlog) => Windows::None, + Windows::None | Windows::Small(_) => Windows::Small(Small::Questlog), + Windows::CharacterAnd(small) => match small { + Some(Small::Questlog) => Windows::CharacterAnd(None), + _ => Windows::CharacterAnd(Some(Small::Questlog)), + }, + Windows::Settings => Windows::Settings, + }; + } + pub fn toggle_map(&mut self) { + self.map_open = !self.map_open; + self.bag_open = false; + } + pub fn toggle_charwindow(&mut self) { + self.open_windows = match self.open_windows { + Windows::CharacterAnd(small) => match small { + Some(small) => Windows::Small(small), + None => Windows::None, + }, + Windows::Small(small) => Windows::CharacterAnd(Some(small)), + Windows::None => Windows::CharacterAnd(None), + Windows::Settings => Windows::Settings, + } + } + pub fn toggle_social(&mut self) { + self.open_windows = match self.open_windows { + Windows::Small(Small::Social) => Windows::None, + Windows::None | Windows::Small(_) => Windows::Small(Small::Social), + Windows::CharacterAnd(small) => match small { + Some(Small::Social) => Windows::CharacterAnd(None), + _ => Windows::CharacterAnd(Some(Small::Social)), + }, + Windows::Settings => Windows::Settings, + }; + } + pub fn toggle_spellbook(&mut self) { + self.open_windows = match self.open_windows { + Windows::Small(Small::Spellbook) => Windows::None, + Windows::None | Windows::Small(_) => Windows::Small(Small::Spellbook), + Windows::CharacterAnd(small) => match small { + Some(Small::Spellbook) => Windows::CharacterAnd(None), + _ => Windows::CharacterAnd(Some(Small::Spellbook)), + }, + Windows::Settings => Windows::Settings, + }; + } + pub fn toggle_settings(&mut self) { + self.open_windows = match self.open_windows { + Windows::Settings => Windows::None, + _ => Windows::Settings, + }; + self.bag_open = false; + } + pub fn toggle_help(&mut self) { + self.show_help = !self.show_help + } + pub fn toggle_ui(&mut self) { + self.show_ui = !self.show_ui; + } pub fn update_grab(&mut self, cursor_grabbed: bool) { self.cursor_grabbed = cursor_grabbed; @@ -1163,30 +1529,72 @@ impl Hud { pub fn handle_event(&mut self, event: WinEvent) -> bool { match event { WinEvent::Ui(event) => { - if (self.typing && event.is_keyboard()) || !(self.cursor_grabbed && event.is_keyboard_or_mouse()) { + if (self.typing && event.is_keyboard()) + || !(self.cursor_grabbed && event.is_keyboard_or_mouse()) + { self.ui.handle_event(event); } true - }, + } WinEvent::KeyDown(Key::Enter) => { - if self.cursor_grabbed && self.typing { + if self.typing { self.ui.focus_widget(None); self.typing = false; - } else if !self.typing { + } else { self.ui.focus_widget(Some(self.chat.input_box_id())); self.typing = true; }; true } - WinEvent::KeyDown(Key::Escape) if self.typing => { + WinEvent::KeyDown(Key::Escape) => { if self.typing { self.typing = false; self.ui.focus_widget(None); - true } else { - false + // Close windows on esc + self.toggle_windows(); } + true } + WinEvent::KeyDown(key) if !self.typing => match key { + Key::Map => { + self.toggle_map(); + true + } + Key::Bag => { + self.toggle_bag(); + true + } + Key::QuestLog => { + self.toggle_questlog(); + true + } + Key::CharacterWindow => { + self.toggle_charwindow(); + true + } + Key::Social => { + self.toggle_social(); + true + } + Key::Spellbook => { + self.toggle_spellbook(); + true + } + Key::Settings => { + self.toggle_settings(); + true + } + Key::Help => { + self.toggle_help(); + true + } + Key::Interface => { + self.toggle_ui(); + true + } + _ => false, + }, WinEvent::KeyDown(key) | WinEvent::KeyUp(key) => match key { Key::ToggleCursor => false, _ => self.typing, diff --git a/voxygen/src/menu/char_selection/mod.rs b/voxygen/src/menu/char_selection/mod.rs index 2beb15bc3b..3ba0130622 100644 --- a/voxygen/src/menu/char_selection/mod.rs +++ b/voxygen/src/menu/char_selection/mod.rs @@ -5,8 +5,9 @@ use crate::{ session::SessionState, GlobalState, PlayState, PlayStateResult, }; +use client::{self, Client}; use common::clock::Clock; -use std::time::Duration; +use std::{cell::RefCell, rc::Rc, time::Duration}; use ui::CharSelectionUi; use vek::*; @@ -14,13 +15,15 @@ const FPS: u64 = 60; pub struct CharSelectionState { char_selection_ui: CharSelectionUi, + client: Rc>, } impl CharSelectionState { /// Create a new `CharSelectionState` - pub fn new(window: &mut Window) -> Self { + pub fn new(window: &mut Window, client: Rc>) -> Self { Self { char_selection_ui: CharSelectionUi::new(window), + client, } } } @@ -59,7 +62,7 @@ impl PlayState for CharSelectionState { match event { ui::Event::Logout => return PlayStateResult::Pop, ui::Event::Play => return PlayStateResult::Push( - Box::new(SessionState::new(&mut global_state.window).unwrap()) // TODO: Handle this error + Box::new(SessionState::new(&mut global_state.window, self.client.clone())) ), } } @@ -67,6 +70,11 @@ impl PlayState for CharSelectionState { // Draw the UI to the screen self.char_selection_ui.render(global_state.window.renderer_mut()); + // Tick the client (currently only to keep the connection alive) + self.client.borrow_mut().tick(client::Input::default(), clock.get_last_delta()) + .expect("Failed to tick the client"); + self.client.borrow_mut().cleanup(); + // Finish the frame global_state.window.renderer_mut().flush(); global_state diff --git a/voxygen/src/menu/char_selection/ui.rs b/voxygen/src/menu/char_selection/ui.rs index dfb9f5aaf3..7341da88c7 100644 --- a/voxygen/src/menu/char_selection/ui.rs +++ b/voxygen/src/menu/char_selection/ui.rs @@ -6,10 +6,9 @@ use crate::{ use conrod_core::{ color, color::TRANSPARENT, - event::Input, image::Id as ImgId, text::font::Id as FontId, - widget::{text_box::Event as TextBoxEvent, Button, Image, Rectangle, Text, TextBox, TitleBar}, + widget::{text_box::Event as TextBoxEvent, Button, Image, Rectangle, Text, TextBox}, widget_ids, Borderable, Color, Colorable, Labelable, Positionable, Sizeable, Widget, }; @@ -23,6 +22,8 @@ widget_ids! { // Windows selection_window, + char_name, + char_level, creation_window, select_window_title, race_heading, @@ -345,6 +346,8 @@ pub enum Event { Play, } +const TEXT_COLOR: Color = Color::Rgba(0.86, 0.86, 0.86, 0.8); + pub struct CharSelectionUi { ui: Ui, ids: Ids, @@ -427,7 +430,7 @@ impl CharSelectionUi { .hover_image(self.imgs.button_dark_hover) .press_image(self.imgs.button_dark_press) .label("Logout") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(18) .label_y(conrod_core::position::Relative::Scalar(3.0)) .set(self.ids.logout_button, ui_widgets) @@ -443,7 +446,7 @@ impl CharSelectionUi { .hover_image(self.imgs.button_dark_hover) .press_image(self.imgs.button_dark_press) .label("Create Character") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(20) .label_y(conrod_core::position::Relative::Scalar(3.0)) .set(self.ids.create_character_button, ui_widgets) @@ -461,8 +464,7 @@ impl CharSelectionUi { .set(self.ids.test_char_l_button, ui_widgets) .was_clicked() { - self.selected_char_no = Some(1); - self.creation_state = CreationState::Race; + self.selected_char_no = Some(1); } // Veloren Logo and Alpha Version @@ -470,22 +472,35 @@ impl CharSelectionUi { .w_h(346.0, 111.0) .top_left_with_margins_on(self.ids.bg_selection, 30.0, 40.0) .label("Alpha 0.1") - .label_rgba(255.0, 255.0, 255.0, 1.0) + .label_rgba(1.0, 1.0, 1.0, 1.0) .label_font_size(10) .label_y(conrod_core::position::Relative::Scalar(-40.0)) .label_x(conrod_core::position::Relative::Scalar(-100.0)) .set(self.ids.v_logo, ui_widgets); - if let Some(no) = self.selected_char_no { + if let Some(no) = self.selected_char_no { // Selection_Window Image::new(self.imgs.selection_window) .w_h(522.0, 722.0) .mid_right_with_margin_on(ui_widgets.window, 10.0) .set(self.ids.selection_window, ui_widgets); + // Character Name & Level + Text::new("Character Name") + .mid_top_with_margin_on(self.ids.selection_window, 80.0) + .font_size(30) + .color(TEXT_COLOR) + .set(self.ids.char_name, ui_widgets); + Text::new("1") + .mid_top_with_margin_on(self.ids.char_name, 40.0) + .font_size(30) + .color(TEXT_COLOR) + .set(self.ids.char_level, ui_widgets); + // Selected Character if no == 1 { Image::new(self.imgs.test_char_l_big) + .w_h(522.0, 722.0) .middle_of(self.ids.selection_window) .set(self.ids.test_char_l_big, ui_widgets); } @@ -497,7 +512,7 @@ impl CharSelectionUi { .hover_image(self.imgs.button_dark_hover) .press_image(self.imgs.button_dark_press) .label("Enter World") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(22) .label_y(conrod_core::position::Relative::Scalar(3.0)) .set(self.ids.enter_world_button, ui_widgets) @@ -514,7 +529,7 @@ impl CharSelectionUi { .hover_image(self.imgs.button_dark_red_hover) .press_image(self.imgs.button_dark_red_press) .label("Delete") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(12) .label_y(conrod_core::position::Relative::Scalar(3.0)) .set(self.ids.delete_button, ui_widgets) @@ -535,7 +550,7 @@ impl CharSelectionUi { .hover_image(self.imgs.button_dark_hover) .press_image(self.imgs.button_dark_press) .label("Back") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(18) .label_y(conrod_core::position::Relative::Scalar(3.0)) .set(self.ids.back_button, ui_widgets) @@ -550,7 +565,7 @@ impl CharSelectionUi { .hover_image(self.imgs.button_dark_hover) .press_image(self.imgs.button_dark_press) .label("Create") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(18) .label_y(conrod_core::position::Relative::Scalar(3.0)) .set(self.ids.create_button, ui_widgets) @@ -569,7 +584,7 @@ impl CharSelectionUi { .font_size(26) .font_id(self.font_metamorph) .center_justify() - .text_color(Color::Rgba(220.0, 220.0, 220.0, 0.8)) + .text_color(TEXT_COLOR) .color(TRANSPARENT) .border_color(TRANSPARENT) .set(self.ids.name_field, ui_widgets) @@ -665,7 +680,7 @@ impl CharSelectionUi { Text::new("Choose your Race") .mid_top_with_margin_on(self.ids.creation_window, 74.0) .font_size(28) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.select_window_title, ui_widgets); // Male/Female/Race Icons @@ -907,14 +922,14 @@ impl CharSelectionUi { Text::new(race_str) .mid_top_with_margin_on(self.ids.creation_window, 370.0) .font_size(30) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.race_heading, ui_widgets); Text::new(race_desc) .mid_top_with_margin_on(self.ids.creation_window, 410.0) .w(500.0) .font_size(20) .font_id(self.font_opensans) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .wrap_by_word() .set(self.ids.race_description, ui_widgets); // Races Descriptions @@ -924,7 +939,7 @@ impl CharSelectionUi { Text::new("Choose your Weapon") .mid_top_with_margin_on(self.ids.creation_window, 74.0) .font_size(28) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.select_window_title, ui_widgets); // BG for Alignment Rectangle::fill_with([470.0, 60.0], color::TRANSPARENT) @@ -1081,14 +1096,14 @@ impl CharSelectionUi { Text::new(weapon_str) .mid_top_with_margin_on(self.ids.creation_window, 370.0) .font_size(30) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.race_heading, ui_widgets); Text::new(weapon_desc) .mid_top_with_margin_on(self.ids.creation_window, 410.0) .w(500.0) .font_size(20) .font_id(self.font_opensans) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .wrap_by_word() .set(self.ids.race_description, ui_widgets); // Races Descriptions @@ -1103,7 +1118,7 @@ impl CharSelectionUi { Text::new("Body Customization") .mid_top_with_margin_on(self.ids.creation_window, 74.0) .font_size(28) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.select_window_title, ui_widgets); match state { @@ -1120,7 +1135,7 @@ impl CharSelectionUi { //.hover_image(self.imgs.frame_open_mo) //.press_image(self.imgs.frame_open_press) .label("Skin & Eyes") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_y(conrod_core::position::Relative::Scalar(4.0)) .label_font_size(16) .set(self.ids.skin_eyes_button, ui_widgets) @@ -1135,7 +1150,7 @@ impl CharSelectionUi { .hover_image(self.imgs.frame_closed_mo) .press_image(self.imgs.frame_closed_press) .label("Hair") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(16) .set(self.ids.hair_button, ui_widgets) .was_clicked() @@ -1149,7 +1164,7 @@ impl CharSelectionUi { .hover_image(self.imgs.frame_closed_mo) .press_image(self.imgs.frame_closed_press) .label("Accessories") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(16) .set(self.ids.accessories_button, ui_widgets) .was_clicked() @@ -1171,7 +1186,7 @@ impl CharSelectionUi { .hover_image(self.imgs.frame_closed_mo) .press_image(self.imgs.frame_closed_press) .label("Skin & Eyes") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(16) .set(self.ids.skin_eyes_button, ui_widgets) .was_clicked() @@ -1185,7 +1200,7 @@ impl CharSelectionUi { //.hover_image(self.imgs.frame_closed_mo) //.press_image(self.imgs.frame_closed_press) .label("Hair") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_y(conrod_core::position::Relative::Scalar(4.0)) .label_font_size(16) .set(self.ids.hair_button, ui_widgets) @@ -1200,7 +1215,7 @@ impl CharSelectionUi { .hover_image(self.imgs.frame_closed_mo) .press_image(self.imgs.frame_closed_press) .label("Accessories") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(16) .set(self.ids.accessories_button, ui_widgets) .was_clicked() @@ -1222,7 +1237,7 @@ impl CharSelectionUi { .hover_image(self.imgs.frame_closed_mo) .press_image(self.imgs.frame_closed_press) .label("Skin & Eyes") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(16) .set(self.ids.skin_eyes_button, ui_widgets) .was_clicked() @@ -1236,7 +1251,7 @@ impl CharSelectionUi { .hover_image(self.imgs.frame_closed_mo) .press_image(self.imgs.frame_closed_press) .label("Hair") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(16) .set(self.ids.hair_button, ui_widgets) .was_clicked() @@ -1251,7 +1266,7 @@ impl CharSelectionUi { //.press_image(self.imgs.frame_closed_press) .label("Accessories") .label_y(conrod_core::position::Relative::Scalar(4.0)) - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(16) .set(self.ids.accessories_button, ui_widgets) .was_clicked() @@ -1268,7 +1283,7 @@ impl CharSelectionUi { Text::new("Skin Color") .top_left_with_margins_on(self.ids.skin_rect, 0.0, -250.0) .font_size(25) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.skin_color_text, ui_widgets); // TODO: Align Buttons here // They set an i32 to a value from 0-14 @@ -1276,7 +1291,7 @@ impl CharSelectionUi { // Here only the BG image changes depending on the race. Rectangle::fill_with([192.0, 116.0], color::WHITE) .top_right_with_margins_on(self.ids.skin_eyes_window, 60.0, 30.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.skin_rect, ui_widgets); // TODO:Slider @@ -1294,7 +1309,7 @@ impl CharSelectionUi { Text::new("Brightness") .top_left_with_margins_on(self.ids.skin_color_slider_range, -27.0, 0.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(14) .set(self.ids.skin_color_slider_text, ui_widgets); @@ -1302,7 +1317,7 @@ impl CharSelectionUi { Text::new("Eye Color") .top_left_with_margins_on(self.ids.eyes_rect, 0.0, -250.0) .font_size(25) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.eye_color_text, ui_widgets); // TODO: Align 16 Buttons here // @@ -1311,7 +1326,7 @@ impl CharSelectionUi { // Only the BG image (190x114 -> 2px border!) changes depending on the race. Rectangle::fill_with([192.0, 116.0], color::WHITE) .top_right_with_margins_on(self.ids.skin_eyes_window, 186.0, 30.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.eyes_rect, ui_widgets); // TODO:Slider @@ -1328,7 +1343,7 @@ impl CharSelectionUi { Text::new("Brightness") .top_left_with_margins_on(self.ids.eye_color_slider_range, -27.0, 0.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(14) .set(self.ids.eye_color_slider_text, ui_widgets); } @@ -1343,7 +1358,7 @@ impl CharSelectionUi { // Hair Text::new("Hair Style") .mid_top_with_margin_on(self.ids.hair_window, 60.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(24) .set(self.ids.hair_style_text, ui_widgets); if Button::image(self.imgs.arrow_right) @@ -1366,12 +1381,12 @@ impl CharSelectionUi { Text::new("Hair Color") .top_left_with_margins_on(self.ids.hair_color_picker_bg, 0.0, -250.0) .font_size(25) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.hair_color_text, ui_widgets); Rectangle::fill_with([192.0, 116.0], color::WHITE) .top_right_with_margins_on(self.ids.hair_window, 114.0, 30.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.hair_color_picker_bg, ui_widgets); Image::new(self.imgs.slider_range) @@ -1390,13 +1405,13 @@ impl CharSelectionUi { Text::new("Brightness") .top_left_with_margins_on(self.ids.hair_color_slider_range, -27.0, 0.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(14) .set(self.ids.hair_color_slider_text, ui_widgets); // Eyebrows Text::new("Eyebrow Style") .mid_top_with_margin_on(self.ids.hair_window, 280.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(24) .set(self.ids.eyebrow_style_text, ui_widgets); if Button::image(self.imgs.arrow_right) @@ -1419,7 +1434,7 @@ impl CharSelectionUi { if let Sex::Male = self.sex { Text::new("Beard Style") .mid_top_with_margin_on(self.ids.hair_window, 340.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(24) .set(self.ids.beard_style_text, ui_widgets); if Button::image(self.imgs.arrow_right) @@ -1451,7 +1466,7 @@ impl CharSelectionUi { Races::Human => { Text::new("Head Band") .mid_top_with_margin_on(self.ids.accessories_window, 60.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(24) .set(self.ids.warpaint_text, ui_widgets); if Button::image(self.imgs.arrow_right) @@ -1478,7 +1493,7 @@ impl CharSelectionUi { -250.0, ) .font_size(25) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.warpaint_color_text, ui_widgets); Rectangle::fill_with([192.0, 116.0], color::WHITE) @@ -1487,7 +1502,7 @@ impl CharSelectionUi { 114.0, 30.0, ) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.warpaint_color_picker_bg, ui_widgets); Image::new(self.imgs.slider_range) @@ -1510,14 +1525,14 @@ impl CharSelectionUi { -27.0, 0.0, ) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(14) .set(self.ids.warpaint_slider_text, ui_widgets); } // Human Races::Orc => { Text::new("Head Band") .mid_top_with_margin_on(self.ids.accessories_window, 60.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(24) .set(self.ids.warpaint_text, ui_widgets); if Button::image(self.imgs.arrow_right) @@ -1544,7 +1559,7 @@ impl CharSelectionUi { -250.0, ) .font_size(25) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.warpaint_color_text, ui_widgets); Rectangle::fill_with([192.0, 116.0], color::WHITE) @@ -1553,7 +1568,7 @@ impl CharSelectionUi { 114.0, 30.0, ) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.warpaint_color_picker_bg, ui_widgets); Image::new(self.imgs.slider_range) @@ -1576,14 +1591,14 @@ impl CharSelectionUi { -27.0, 0.0, ) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(14) .set(self.ids.warpaint_slider_text, ui_widgets); } // Orc Races::Elf => { Text::new("Tribe Markings") .mid_top_with_margin_on(self.ids.accessories_window, 60.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(24) .set(self.ids.warpaint_text, ui_widgets); if Button::image(self.imgs.arrow_right) @@ -1610,7 +1625,7 @@ impl CharSelectionUi { -250.0, ) .font_size(25) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.warpaint_color_text, ui_widgets); Rectangle::fill_with([192.0, 116.0], color::WHITE) @@ -1619,7 +1634,7 @@ impl CharSelectionUi { 114.0, 30.0, ) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.warpaint_color_picker_bg, ui_widgets); Image::new(self.imgs.slider_range) @@ -1642,14 +1657,14 @@ impl CharSelectionUi { -27.0, 0.0, ) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(14) .set(self.ids.warpaint_slider_text, ui_widgets); } // Elf Races::Dwarf => { Text::new("War Paint") .mid_top_with_margin_on(self.ids.accessories_window, 60.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(24) .set(self.ids.warpaint_text, ui_widgets); if Button::image(self.imgs.arrow_right) @@ -1676,7 +1691,7 @@ impl CharSelectionUi { -250.0, ) .font_size(25) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.warpaint_color_text, ui_widgets); Rectangle::fill_with([192.0, 116.0], color::WHITE) @@ -1685,7 +1700,7 @@ impl CharSelectionUi { 114.0, 30.0, ) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.warpaint_color_picker_bg, ui_widgets); Image::new(self.imgs.slider_range) @@ -1708,14 +1723,14 @@ impl CharSelectionUi { -27.0, 0.0, ) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(14) .set(self.ids.warpaint_slider_text, ui_widgets); } // Dwarf Races::Undead => { Text::new("Teeth") .mid_top_with_margin_on(self.ids.accessories_window, 60.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(24) .set(self.ids.warpaint_text, ui_widgets); if Button::image(self.imgs.arrow_right) @@ -1742,7 +1757,7 @@ impl CharSelectionUi { -250.0, ) .font_size(25) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.warpaint_color_text, ui_widgets); Rectangle::fill_with([192.0, 116.0], color::WHITE) @@ -1751,7 +1766,7 @@ impl CharSelectionUi { 114.0, 30.0, ) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.warpaint_color_picker_bg, ui_widgets); Image::new(self.imgs.slider_range) @@ -1774,14 +1789,14 @@ impl CharSelectionUi { -27.0, 0.0, ) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(14) .set(self.ids.warpaint_slider_text, ui_widgets); } // Undead Races::Danari => { Text::new("Horns") .mid_top_with_margin_on(self.ids.accessories_window, 60.0) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(24) .set(self.ids.warpaint_text, ui_widgets); if Button::image(self.imgs.arrow_right) @@ -1808,7 +1823,7 @@ impl CharSelectionUi { -250.0, ) .font_size(25) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.warpaint_color_text, ui_widgets); Rectangle::fill_with([192.0, 116.0], color::WHITE) @@ -1817,7 +1832,7 @@ impl CharSelectionUi { 114.0, 30.0, ) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .set(self.ids.warpaint_color_picker_bg, ui_widgets); Image::new(self.imgs.slider_range) @@ -1840,7 +1855,7 @@ impl CharSelectionUi { -27.0, 0.0, ) - .rgba(220.0, 220.0, 220.0, 0.8) + .color(TEXT_COLOR) .font_size(14) .set(self.ids.warpaint_slider_text, ui_widgets); } // Danari diff --git a/voxygen/src/menu/main/mod.rs b/voxygen/src/menu/main/mod.rs index c32b2a3d5d..0dfe93880d 100644 --- a/voxygen/src/menu/main/mod.rs +++ b/voxygen/src/menu/main/mod.rs @@ -5,6 +5,7 @@ use crate::{ window::{Event, Window}, GlobalState, PlayState, PlayStateResult, }; +use client::{self, Client}; use common::clock::Clock; use std::time::Duration; use ui::{Event as MainMenuEvent, MainMenuUi}; @@ -56,14 +57,42 @@ impl PlayState for MainMenuState { // Maintain the UI for event in self.main_menu_ui.maintain(global_state.window.renderer_mut()) { - match event { - MainMenuEvent::LoginAttempt{ username, server_address } => - // For now just start a new session - return PlayStateResult::Push( - Box::new(CharSelectionState::new(&mut global_state.window)) - ), - MainMenuEvent::Quit => return PlayStateResult::Shutdown, - } + match event { + MainMenuEvent::LoginAttempt{ username, server_address } => { + use std::net::ToSocketAddrs; + const DEFAULT_PORT: u16 = 59003; + // Parses ip address or resolves hostname + // Note: if you use an ipv6 address the number after the last colon will be used as the port unless you use [] around the address + match server_address.to_socket_addrs().or((server_address.as_str(), DEFAULT_PORT).to_socket_addrs()) { + Ok(mut socket_adders) => { + while let Some(socket_addr) = socket_adders.next() { + // TODO: handle error + match Client::new(socket_addr) { + Ok(client) => { + return PlayStateResult::Push( + Box::new(CharSelectionState::new( + &mut global_state.window, + std::rc::Rc::new(std::cell::RefCell::new(client.with_test_state())) // <--- TODO: Remove this + )) + ); + } + Err(client::Error::Network(_)) => {} // assume connection failed and try next address + Err(err) => { + panic!("Unexpected non Network error when creating client: {:?}", err); + } + } + } + // Parsing/host name resolution successful but no connection succeeded + self.main_menu_ui.login_error("Could not connect to address".to_string()); + } + Err(err) => { + // Error parsing input string or error resolving host name + self.main_menu_ui.login_error("No such host is known".to_string()); + } + } + } + MainMenuEvent::Quit => return PlayStateResult::Shutdown, + } } // Draw the UI to the screen diff --git a/voxygen/src/menu/main/ui.rs b/voxygen/src/menu/main/ui.rs index e50a405ade..d1d6311b12 100644 --- a/voxygen/src/menu/main/ui.rs +++ b/voxygen/src/menu/main/ui.rs @@ -6,8 +6,9 @@ use crate::{ use conrod_core::{ color::TRANSPARENT, image::Id as ImgId, + position::Dimension, text::font::Id as FontId, - widget::{text_box::Event as TextBoxEvent, Button, Image, TextBox}, + widget::{text_box::Event as TextBoxEvent, Button, Image, Rectangle, Text, TextBox}, widget_ids, Borderable, Color, Colorable, Labelable, Positionable, Sizeable, Widget, }; @@ -21,6 +22,8 @@ widget_ids! { // Login, Singleplayer login_button, login_text, + login_error, + login_error_bg, address_text, address_bg, address_field, @@ -101,6 +104,7 @@ pub struct MainMenuUi { font_opensans: FontId, username: String, server_address: String, + login_error: Option, } impl MainMenuUi { @@ -134,7 +138,8 @@ impl MainMenuUi { font_metamorph, font_opensans, username: "Username".to_string(), - server_address: "Server Address".to_string(), + server_address: "veloren.mac94.de".to_string(), + login_error: None, } } @@ -149,7 +154,7 @@ impl MainMenuUi { .w_h(346.0, 111.0) .top_left_with_margins(30.0, 40.0) .label("Alpha 0.1") - .label_rgba(255.0, 255.0, 255.0, 1.0) + .label_rgba(1.0, 1.0, 1.0, 1.0) .label_font_size(10) .label_y(conrod_core::position::Relative::Scalar(-40.0)) .label_x(conrod_core::position::Relative::Scalar(-100.0)) @@ -159,12 +164,14 @@ impl MainMenuUi { // Used when the login button is pressed, or enter is pressed within input field macro_rules! login { () => { + self.login_error = None; events.push(Event::LoginAttempt { username: self.username.clone(), server_address: self.server_address.clone(), }); }; } + const TEXT_COLOR: Color = Color::Rgba(0.94, 0.94, 0.94, 0.8); // Username // TODO: get a lower resolution and cleaner input_bg.png Image::new(self.imgs.input_bg) @@ -176,7 +183,7 @@ impl MainMenuUi { .mid_bottom_with_margin_on(self.ids.username_bg, 44.0 / 2.0) .font_size(20) .font_id(self.font_opensans) - .text_color(Color::Rgba(220.0, 220.0, 220.0, 0.8)) + .text_color(TEXT_COLOR) // transparent background .color(TRANSPARENT) .border_color(TRANSPARENT) @@ -187,9 +194,28 @@ impl MainMenuUi { // Note: TextBox limits the input string length to what fits in it self.username = username.to_string(); } - TextBoxEvent::Enter => login!(), + TextBoxEvent::Enter => { login!(); } } } + // Login error + if let Some(msg) = &self.login_error { + let text = Text::new(&msg) + .rgba(0.5, 0.0, 0.0, 1.0) + .font_size(30) + .font_id(self.font_opensans); + let x = match text.get_x_dimension(ui_widgets) { + Dimension::Absolute(x) => x + 10.0, + _ => 0.0, + }; + Rectangle::fill([x, 40.0]) + .rgba(0.2, 0.3, 0.3, 0.7) + .parent(ui_widgets.window) + .up_from(self.ids.username_bg, 35.0) + .set(self.ids.login_error_bg, ui_widgets); + text + .middle_of(self.ids.login_error_bg) + .set(self.ids.login_error, ui_widgets); + } // Server address Image::new(self.imgs.input_bg) .w_h(337.0, 67.0) @@ -200,7 +226,7 @@ impl MainMenuUi { .mid_bottom_with_margin_on(self.ids.address_bg, 44.0 / 2.0) .font_size(20) .font_id(self.font_opensans) - .text_color(Color::Rgba(220.0, 220.0, 220.0, 0.8)) + .text_color(TEXT_COLOR) // transparent background .color(TRANSPARENT) .border_color(TRANSPARENT) @@ -210,7 +236,7 @@ impl MainMenuUi { TextBoxEvent::Update(server_address) => { self.server_address = server_address.to_string(); } - TextBoxEvent::Enter => login!(), + TextBoxEvent::Enter => { login!(); } } } // Login button @@ -221,7 +247,7 @@ impl MainMenuUi { .down_from(self.ids.address_bg, 20.0) .align_middle_x_of(self.ids.address_bg) .label("Login") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(28) .label_y(conrod_core::position::Relative::Scalar(5.0)) .set(self.ids.login_button, ui_widgets) @@ -229,7 +255,7 @@ impl MainMenuUi { { login!(); } - //Singleplayer button + // Singleplayer button if Button::image(self.imgs.login_button) .hover_image(self.imgs.login_button_hover) .press_image(self.imgs.login_button_press) @@ -237,7 +263,7 @@ impl MainMenuUi { .down_from(self.ids.login_button, 20.0) .align_middle_x_of(self.ids.address_bg) .label("Singleplayer") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(26) .label_y(conrod_core::position::Relative::Scalar(5.0)) .label_x(conrod_core::position::Relative::Scalar(2.0)) @@ -253,7 +279,7 @@ impl MainMenuUi { .hover_image(self.imgs.button_hover) .press_image(self.imgs.button_press) .label("Quit") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(20) .label_y(conrod_core::position::Relative::Scalar(3.0)) .set(self.ids.quit_button, ui_widgets) @@ -268,7 +294,7 @@ impl MainMenuUi { .hover_image(self.imgs.button_hover) .press_image(self.imgs.button_press) .label("Settings") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(20) .label_y(conrod_core::position::Relative::Scalar(3.0)) .set(self.ids.settings_button, ui_widgets) @@ -281,7 +307,7 @@ impl MainMenuUi { .hover_image(self.imgs.button_hover) .press_image(self.imgs.button_press) .label("Servers") - .label_rgba(220.0, 220.0, 220.0, 0.8) + .label_color(TEXT_COLOR) .label_font_size(20) .label_y(conrod_core::position::Relative::Scalar(3.0)) .set(self.ids.servers_button, ui_widgets) @@ -291,6 +317,10 @@ impl MainMenuUi { events } + pub fn login_error(&mut self, msg: String) { + self.login_error = Some(msg); + } + pub fn handle_event(&mut self, event: ui::Event) { self.ui.handle_event(event); } diff --git a/voxygen/src/render/pipelines/ui.rs b/voxygen/src/render/pipelines/ui.rs index bd88dbe69b..0f3a3b4800 100644 --- a/voxygen/src/render/pipelines/ui.rs +++ b/voxygen/src/render/pipelines/ui.rs @@ -12,7 +12,6 @@ use super::super::{ Pipeline, TgtColorFmt, TgtDepthFmt, - Mesh, Quad, Tri, }; diff --git a/voxygen/src/scene/mod.rs b/voxygen/src/scene/mod.rs index 5c23dbc709..364c80f0b0 100644 --- a/voxygen/src/scene/mod.rs +++ b/voxygen/src/scene/mod.rs @@ -78,7 +78,7 @@ impl Scene { test_figure: Figure::new( renderer, [ - Some(load_segment("head.vox").generate_mesh(Vec3::new(-7.0, -5.5, -1.0))), + Some(load_segment("head.vox").generate_mesh(Vec3::new(-7.0, -6.5, -6.0))), Some(load_segment("chest.vox").generate_mesh(Vec3::new(-6.0, -3.0, 0.0))), Some(load_segment("belt.vox").generate_mesh(Vec3::new(-5.0, -3.0, 0.0))), Some(load_segment("pants.vox").generate_mesh(Vec3::new(-5.0, -3.0, 0.0))), diff --git a/voxygen/src/session.rs b/voxygen/src/session.rs index f24fa2cd4e..2d4f4911c5 100644 --- a/voxygen/src/session.rs +++ b/voxygen/src/session.rs @@ -1,17 +1,10 @@ -// Standard -use std::time::Duration; - -// Library +use std::{cell::RefCell, rc::Rc, time::Duration}; use vek::*; - -// Project use common::clock::Clock; use client::{ self, Client, }; - -// Crate use crate::{ Error, PlayState, @@ -28,28 +21,35 @@ const FPS: u64 = 60; pub struct SessionState { scene: Scene, - client: Client, + client: Rc>, key_state: KeyState, hud: Hud, } + /// Represents an active game session (i.e: one that is being played) impl SessionState { /// Create a new `SessionState` - pub fn new(window: &mut Window) -> Result { - let client = Client::new(([127, 0, 0, 1], 59003))?.with_test_state(); // <--- TODO: Remove this - Ok(Self { - // Create a scene for this session. The scene handles visible elements of the game world - scene: Scene::new(window.renderer_mut(), &client), + pub fn new(window: &mut Window, client: Rc>) -> Self { + // Create a scene for this session. The scene handles visible elements of the game world + let scene = Scene::new(window.renderer_mut(), &client.borrow()); + Self { + scene, client, key_state: KeyState::new(), hud: Hud::new(window), - }) + } } } + // The background colour -const BG_COLOR: Rgba = Rgba { r: 0.0, g: 0.3, b: 1.0, a: 1.0 }; +const BG_COLOR: Rgba = Rgba { + r: 0.0, + g: 0.3, + b: 1.0, + a: 1.0, +}; impl SessionState { /// Tick the session (and the client attached to it) @@ -63,20 +63,20 @@ impl SessionState { let dir_vec = self.key_state.dir_vec(); let move_dir = unit_vecs.0 * dir_vec[0] + unit_vecs.1 * dir_vec[1]; - for event in self.client.tick(client::Input { move_dir }, dt)? { + for event in self.client.borrow_mut().tick(client::Input { move_dir }, dt)? { match event { client::Event::Chat(msg) => { self.hud.new_message(msg); } } } - + Ok(()) } /// Clean up the session (and the client attached to it) after a tick pub fn cleanup(&mut self) { - self.client.cleanup(); + self.client.borrow_mut().cleanup(); } /// Render the session to the screen. @@ -108,7 +108,7 @@ impl PlayState for SessionState { for x in -6..7 { for y in -6..7 { for z in -1..2 { - self.client.load_chunk(Vec3::new(x, y, z)); + self.client.borrow_mut().load_chunk(Vec3::new(x, y, z)); } } } @@ -124,17 +124,14 @@ impl PlayState for SessionState { } let _handled = match event { Event::Close => return PlayStateResult::Shutdown, - // When 'q' is pressed, exit the session - Event::Char('q') => return PlayStateResult::Pop, - // When 'm' is pressed, open/close the in-game test menu - Event::Char('m') => self.hud.toggle_menu(), - // Close windows on esc - Event::KeyDown(Key::Escape) => self.hud.toggle_windows(), // Toggle cursor grabbing Event::KeyDown(Key::ToggleCursor) => { - global_state.window.grab_cursor(!global_state.window.is_cursor_grabbed()); - self.hud.update_grab(global_state.window.is_cursor_grabbed()); - }, + global_state + .window + .grab_cursor(!global_state.window.is_cursor_grabbed()); + self.hud + .update_grab(global_state.window.is_cursor_grabbed()); + } // Movement Key Pressed Event::KeyDown(Key::MoveForward) => self.key_state.up = true, Event::KeyDown(Key::MoveBack) => self.key_state.down = true, @@ -146,7 +143,9 @@ impl PlayState for SessionState { Event::KeyUp(Key::MoveLeft) => self.key_state.left = false, Event::KeyUp(Key::MoveRight) => self.key_state.right = false, // Pass all other events to the scene - event => { self.scene.handle_input_event(event); }, + event => { + self.scene.handle_input_event(event); + } }; // TODO: Do something if the event wasn't handled? } @@ -156,13 +155,13 @@ impl PlayState for SessionState { .expect("Failed to tick the scene"); // Maintain the scene - self.scene.maintain(global_state.window.renderer_mut(), &self.client); + self.scene.maintain(global_state.window.renderer_mut(), &self.client.borrow()); // Maintain the UI for event in self.hud.maintain(global_state.window.renderer_mut()) { match event { HudEvent::SendMessage(msg) => { // TODO: Handle result - self.client.send_chat(msg); + self.client.borrow_mut().send_chat(msg); }, HudEvent::Logout => return PlayStateResult::Pop, HudEvent::Quit => return PlayStateResult::Shutdown, @@ -173,7 +172,8 @@ impl PlayState for SessionState { self.render(global_state.window.renderer_mut()); // Display the frame on the window - global_state.window + global_state + .window .swap_buffers() .expect("Failed to swap window buffers"); @@ -185,5 +185,7 @@ impl PlayState for SessionState { } } - fn name(&self) -> &'static str { "Session" } + fn name(&self) -> &'static str { + "Session" + } } diff --git a/voxygen/src/ui/mod.rs b/voxygen/src/ui/mod.rs index 8cd9cc92b0..2ab107e12b 100644 --- a/voxygen/src/ui/mod.rs +++ b/voxygen/src/ui/mod.rs @@ -367,7 +367,6 @@ impl Ui { use conrod_core::render::PrimitiveKind; match kind { - // TODO: use source_rect PrimitiveKind::Image { image_id, color, source_rect } => { // Switch to the `Image` state for this image if we're not in it already. @@ -389,7 +388,7 @@ impl Ui { } } - let color = color.unwrap_or(conrod_core::color::WHITE).to_fsa(); + let color = srgb_to_linear(color.unwrap_or(conrod_core::color::WHITE).to_fsa()); // Transform the source rectangle into uv coordinates let (image_w, image_h) = self.image_map @@ -443,8 +442,7 @@ impl Ui { renderer.update_texture(cache_tex, offset, size, &new_data); }).unwrap(); - // TODO: consider gamma.... - let color = color.to_fsa(); + let color = srgb_to_linear(color.to_fsa()); for g in positioned_glyphs { if let Ok(Some((uv_rect, screen_rect))) = glyph_cache.rect_for(font_id.index(), g) { @@ -472,8 +470,7 @@ impl Ui { } } PrimitiveKind::Rectangle { color } => { - // TODO: consider gamma/linear conversion.... - let color = color.to_fsa(); + let color = srgb_to_linear(color.to_fsa()); // Don't draw a transparent rectangle if color[3] == 0.0 { continue; @@ -493,7 +490,7 @@ impl Ui { } PrimitiveKind::TrianglesSingleColor { color, triangles } => { // Don't draw transparent triangle or switch state if there are actually no triangles - let color: [f32; 4] = color.into(); + let color: [f32; 4] = srgb_to_linear(color.into()); if triangles.is_empty() || color[3] == 0.0 { continue; } @@ -577,3 +574,15 @@ fn default_scissor(renderer: &mut Renderer) -> Aabr { max: Vec2 { x: screen_w, y: screen_h } } } + +fn srgb_to_linear(color: [f32; 4]) -> [f32; 4] { + fn linearize(comp: f32) -> f32 { + if comp <= 0.04045 { + comp / 12.92 + } else { + ((comp + 0.055) / 1.055).powf(2.4) + } + } + + [linearize(color[0]), linearize(color[1]), linearize(color[2]), color[3]] +} diff --git a/voxygen/src/window.rs b/voxygen/src/window.rs index 959fe36c9e..81395e23d4 100644 --- a/voxygen/src/window.rs +++ b/voxygen/src/window.rs @@ -1,14 +1,9 @@ -use vek::*; -use std::collections::HashMap; use crate::{ - Error, - render::{ - Renderer, - TgtColorFmt, - TgtDepthFmt, - }, - ui, + render::{Renderer, TgtColorFmt, TgtDepthFmt}, + ui, Error, }; +use std::collections::HashMap; +use vek::*; pub struct Window { events_loop: glutin::EventsLoop, @@ -19,7 +14,6 @@ pub struct Window { key_map: HashMap, } - impl Window { pub fn new() -> Result { let events_loop = glutin::EventsLoop::new(); @@ -33,17 +27,13 @@ impl Window { .with_gl(glutin::GlRequest::Specific(glutin::Api::OpenGl, (3, 2))) .with_vsync(false); - let ( - window, - device, - factory, - tgt_color_view, - tgt_depth_view, - ) = gfx_window_glutin::init::( - win_builder, - ctx_builder, - &events_loop, - ).map_err(|err| Error::BackendError(Box::new(err)))?; + let (window, device, factory, tgt_color_view, tgt_depth_view) = + gfx_window_glutin::init::( + win_builder, + ctx_builder, + &events_loop, + ) + .map_err(|err| Error::BackendError(Box::new(err)))?; let mut key_map = HashMap::new(); key_map.insert(glutin::VirtualKeyCode::Tab, Key::ToggleCursor); @@ -53,15 +43,19 @@ impl Window { key_map.insert(glutin::VirtualKeyCode::A, Key::MoveLeft); key_map.insert(glutin::VirtualKeyCode::S, Key::MoveBack); key_map.insert(glutin::VirtualKeyCode::D, Key::MoveRight); + key_map.insert(glutin::VirtualKeyCode::M, Key::Map); + key_map.insert(glutin::VirtualKeyCode::B, Key::Bag); + key_map.insert(glutin::VirtualKeyCode::L, Key::QuestLog); + key_map.insert(glutin::VirtualKeyCode::C, Key::CharacterWindow); + key_map.insert(glutin::VirtualKeyCode::O, Key::Social); + key_map.insert(glutin::VirtualKeyCode::P, Key::Spellbook); + key_map.insert(glutin::VirtualKeyCode::N, Key::Settings); + key_map.insert(glutin::VirtualKeyCode::F1, Key::Help); + key_map.insert(glutin::VirtualKeyCode::F2, Key::Interface); let tmp = Ok(Self { events_loop, - renderer: Renderer::new( - device, - factory, - tgt_color_view, - tgt_depth_view, - )?, + renderer: Renderer::new(device, factory, tgt_color_view, tgt_depth_view)?, window, cursor_grabbed: false, needs_refresh_resize: false, @@ -70,8 +64,12 @@ impl Window { tmp } - pub fn renderer(&self) -> &Renderer { &self.renderer } - pub fn renderer_mut(&mut self) -> &mut Renderer { &mut self.renderer } + pub fn renderer(&self) -> &Renderer { + &self.renderer + } + pub fn renderer_mut(&mut self) -> &mut Renderer { + &mut self.renderer + } pub fn fetch_events(&mut self) -> Vec { let mut events = vec![]; @@ -99,44 +97,45 @@ impl Window { glutin::WindowEvent::CloseRequested => events.push(Event::Close), glutin::WindowEvent::Resized(glutin::dpi::LogicalSize { width, height }) => { let (mut color_view, mut depth_view) = renderer.target_views_mut(); - gfx_window_glutin::update_views( - &window, - &mut color_view, - &mut depth_view, - ); + gfx_window_glutin::update_views(&window, &mut color_view, &mut depth_view); events.push(Event::Resize(Vec2::new(width as u32, height as u32))); - }, + } glutin::WindowEvent::ReceivedCharacter(c) => events.push(Event::Char(c)), - glutin::WindowEvent::KeyboardInput { input, .. } => match input.virtual_keycode { + glutin::WindowEvent::KeyboardInput { input, .. } => match input.virtual_keycode + { Some(keycode) => match key_map.get(&keycode) { Some(&key) => events.push(match input.state { glutin::ElementState::Pressed => Event::KeyDown(key), _ => Event::KeyUp(key), }), - _ => {}, + _ => {} }, - _ => {}, + _ => {} }, - _ => {}, + _ => {} }, glutin::Event::DeviceEvent { event, .. } => match event { - glutin::DeviceEvent::MouseMotion { delta: (dx, dy), .. } if cursor_grabbed => - events.push(Event::CursorPan(Vec2::new(dx as f32, dy as f32))), + glutin::DeviceEvent::MouseMotion { + delta: (dx, dy), .. + } if cursor_grabbed => { + events.push(Event::CursorPan(Vec2::new(dx as f32, dy as f32))) + } glutin::DeviceEvent::MouseWheel { delta: glutin::MouseScrollDelta::LineDelta(_x, y), .. } if cursor_grabbed => events.push(Event::Zoom(y as f32)), - _ => {}, + _ => {} }, - _ => {}, + _ => {} } }); events } pub fn swap_buffers(&self) -> Result<(), Error> { - self.window.swap_buffers() + self.window + .swap_buffers() .map_err(|err| Error::BackendError(Box::new(err))) } @@ -147,7 +146,8 @@ impl Window { pub fn grab_cursor(&mut self, grab: bool) { self.cursor_grabbed = grab; self.window.hide_cursor(grab); - self.window.grab_cursor(grab) + self.window + .grab_cursor(grab) .expect("Failed to grab/ungrab cursor"); } @@ -156,7 +156,11 @@ impl Window { } pub fn logical_size(&self) -> Vec2 { - let (w, h) = self.window.get_inner_size().unwrap_or(glutin::dpi::LogicalSize::new(0.0, 0.0)).into(); + let (w, h) = self + .window + .get_inner_size() + .unwrap_or(glutin::dpi::LogicalSize::new(0.0, 0.0)) + .into(); Vec2::new(w, h) } } @@ -170,7 +174,16 @@ pub enum Key { MoveLeft, MoveRight, Enter, - Escape, + Escape, + Map, + Bag, + QuestLog, + CharacterWindow, + Social, + Spellbook, + Settings, + Interface, + Help, } /// Represents an incoming event from the window