From a4dc47bc2cd29760db69eacbcd7556da72ece6b1 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sun, 15 Dec 2019 12:13:52 -0500 Subject: [PATCH 01/61] Initial setup to use iced --- Cargo.lock | 127 +++++++- voxygen/Cargo.toml | 10 +- voxygen/src/hud/settings_window.rs | 1 - voxygen/src/menu/main/ui.rs | 74 ++++- voxygen/src/run.rs | 7 + voxygen/src/ui/ice/clipboard.rs | 12 + voxygen/src/ui/ice/mod.rs | 72 +++++ voxygen/src/ui/ice/renderer.rs | 400 +++++++++++++++++++++++++ voxygen/src/ui/ice/renderer/column.rs | 34 +++ voxygen/src/ui/ice/renderer/image.rs | 14 + voxygen/src/ui/ice/widget.rs | 1 + voxygen/src/ui/ice/widget/image.rs | 63 ++++ voxygen/src/ui/ice/winit_conversion.rs | 271 +++++++++++++++++ voxygen/src/ui/img_ids.rs | 19 ++ voxygen/src/ui/mod.rs | 9 +- voxygen/src/window.rs | 10 +- 16 files changed, 1104 insertions(+), 20 deletions(-) create mode 100644 voxygen/src/ui/ice/clipboard.rs create mode 100644 voxygen/src/ui/ice/mod.rs create mode 100644 voxygen/src/ui/ice/renderer.rs create mode 100644 voxygen/src/ui/ice/renderer/column.rs create mode 100644 voxygen/src/ui/ice/renderer/image.rs create mode 100644 voxygen/src/ui/ice/widget.rs create mode 100644 voxygen/src/ui/ice/widget/image.rs create mode 100644 voxygen/src/ui/ice/winit_conversion.rs diff --git a/Cargo.lock b/Cargo.lock index 42287b4056..e7f7f63ce1 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -535,6 +535,46 @@ dependencies = [ "winapi 0.3.9", ] +[[package]] +name = "clipboard-win" +version = "4.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5123c6b97286809fea9e38d2c9bf530edbcb9fc0d8f8272c28b0c95f067fa92d" +dependencies = [ + "error-code", + "str-buf", + "winapi 0.3.9", +] + +[[package]] +name = "clipboard_macos" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "145a7f9e9b89453bc0a5e32d166456405d389cea5b578f57f1274b1397588a95" +dependencies = [ + "objc", + "objc-foundation", + "objc_id", +] + +[[package]] +name = "clipboard_wayland" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "926d872adca0fc88173f8b7532c651e29ce67dc97323f4546c1c8af6610937fb" +dependencies = [ + "smithay-clipboard 0.5.2", +] + +[[package]] +name = "clipboard_x11" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "137cbd60c42327a8d63e710cee5a4d6a1ac41cdc90449ea2c2c63bd5e186290a" +dependencies = [ + "xcb", +] + [[package]] name = "cloudabi" version = "0.0.3" @@ -708,11 +748,11 @@ version = "0.7.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "4423d79fed83ebd9ab81ec21fa97144300a961782158287dc9bf7eddac37ff0b" dependencies = [ - "clipboard-win", + "clipboard-win 3.1.1", "objc", "objc-foundation", "objc_id", - "smithay-clipboard", + "smithay-clipboard 0.6.1", "x11-clipboard", ] @@ -1324,6 +1364,16 @@ dependencies = [ "version_check 0.9.2", ] +[[package]] +name = "error-code" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b49c94f66f2d2c5ee8685039e458b4e6c9f13af7c28736baf10ce42966a5ab52" +dependencies = [ + "libc", + "str-buf", +] + [[package]] name = "euc" version = "0.5.1" @@ -1988,6 +2038,33 @@ dependencies = [ "want", ] +[[package]] +name = "iced_core" +version = "0.2.1" +source = "git+https://github.com/hecrj/iced#f46431600cb61d4e83e0ded1ca79525478436be3" + +[[package]] +name = "iced_futures" +version = "0.1.2" +source = "git+https://github.com/hecrj/iced#f46431600cb61d4e83e0ded1ca79525478436be3" +dependencies = [ + "futures 0.3.5", + "log", + "wasm-bindgen-futures", +] + +[[package]] +name = "iced_native" +version = "0.2.2" +source = "git+https://github.com/hecrj/iced#f46431600cb61d4e83e0ded1ca79525478436be3" +dependencies = [ + "iced_core", + "iced_futures", + "num-traits 0.2.12", + "twox-hash", + "unicode-segmentation", +] + [[package]] name = "ident_case" version = "1.0.1" @@ -4039,6 +4116,16 @@ dependencies = [ "wayland-protocols 0.28.1", ] +[[package]] +name = "smithay-clipboard" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b9e9db50a9b272938b767b731a1291f22f407315def4049db93871e8828034d5" +dependencies = [ + "smithay-client-toolkit 0.11.0", + "wayland-client 0.27.0", +] + [[package]] name = "smithay-clipboard" version = "0.6.1" @@ -4173,6 +4260,12 @@ version = "0.1.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "213701ba3370744dcd1a12960caa4843b3d68b4d1c0a5d575e0d65b2ee9d16c0" +[[package]] +name = "str-buf" +version = "1.0.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d44a3643b4ff9caf57abcee9c2c621d6c03d9135e0d8b589bd9afb5992cb176a" + [[package]] name = "string" version = "0.2.1" @@ -4669,6 +4762,9 @@ name = "twox-hash" version = "1.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3bfd5b7557925ce778ff9b9ef90e3ade34c524b5ff10e239c69a42d546d2af56" +dependencies = [ + "rand 0.4.6", +] [[package]] name = "tynm" @@ -4989,6 +5085,7 @@ dependencies = [ "glutin", "guillotiere", "hashbrown 0.7.2", + "iced_native", "image", "inline_tweak", "itertools", @@ -5014,6 +5111,7 @@ dependencies = [ "veloren-server", "veloren-voxygen-anim", "veloren-world", + "window_clipboard", "winit", "winres", ] @@ -5171,6 +5269,18 @@ dependencies = [ "wasm-bindgen-shared", ] +[[package]] +name = "wasm-bindgen-futures" +version = "0.4.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b7866cab0aa01de1edf8b5d7936938a7e397ee50ce24119aef3e1eaa3b6171da" +dependencies = [ + "cfg-if", + "js-sys", + "wasm-bindgen", + "web-sys", +] + [[package]] name = "wasm-bindgen-macro" version = "0.2.68" @@ -5448,6 +5558,19 @@ version = "0.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" +[[package]] +name = "window_clipboard" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b849e24b344ea3535bcda7320b8b7f3560bd2c3692de73153d3c64acc84203e5" +dependencies = [ + "clipboard-win 4.0.3", + "clipboard_macos", + "clipboard_wayland", + "clipboard_x11", + "raw-window-handle", +] + [[package]] name = "winit" version = "0.22.2" diff --git a/voxygen/Cargo.toml b/voxygen/Cargo.toml index e7a4d8a30d..7dfcd90ea8 100644 --- a/voxygen/Cargo.toml +++ b/voxygen/Cargo.toml @@ -24,9 +24,6 @@ common = {package = "veloren-common", path = "../common"} anim = {package = "veloren-voxygen-anim", path = "src/anim", default-features = false} # Graphics -conrod_core = {git = "https://gitlab.com/veloren/conrod.git", branch="copypasta_0.7"} -conrod_winit = {git = "https://gitlab.com/veloren/conrod.git", branch="copypasta_0.7"} -euc = {git = "https://github.com/zesterer/euc.git"} gfx = "0.18.2" gfx_device_gl = {version = "0.16.2", optional = true} gfx_gl = {version = "0.6.1", optional = true} @@ -34,6 +31,13 @@ glutin = {git = "https://github.com/rust-windowing/glutin.git", rev="63a1ea7d6e6 old_school_gfx_glutin_ext = "0.24" winit = {version = "0.22.2", features = ["serde"]} +# Ui +conrod_core = {git = "https://gitlab.com/veloren/conrod.git", branch="copypasta_0.7"} +conrod_winit = {git = "https://gitlab.com/veloren/conrod.git", branch="copypasta_0.7"} +euc = {git = "https://github.com/zesterer/euc.git"} +iced = {package = "iced_native", git = "https://github.com/hecrj/iced"} +window_clipboard = "0.1.1" + # ECS specs = {git = "https://github.com/amethyst/specs.git", rev = "7a2e348ab2223818bad487695c66c43db88050a5"} specs-idvs = {git = "https://gitlab.com/veloren/specs-idvs.git", branch = "specs-git"} diff --git a/voxygen/src/hud/settings_window.rs b/voxygen/src/hud/settings_window.rs index 59f5a6f14b..daa5148219 100644 --- a/voxygen/src/hud/settings_window.rs +++ b/voxygen/src/hud/settings_window.rs @@ -2317,7 +2317,6 @@ impl<'a> Widget for SettingsWindow<'a> { .global_state .window .window() - .window() .current_monitor() .unwrap() .video_modes() diff --git a/voxygen/src/menu/main/ui.rs b/voxygen/src/menu/main/ui.rs index bcb5a4a74b..72acdb123f 100644 --- a/voxygen/src/menu/main/ui.rs +++ b/voxygen/src/menu/main/ui.rs @@ -4,11 +4,13 @@ use crate::{ ui::{ self, fonts::ConrodVoxygenFonts, + ice::IcedUi, img_ids::{BlankGraphic, ImageGraphic, VoxelGraphic}, - Graphic, ImageFrame, Tooltip, Ui, + Graphic, Ui, }, GlobalState, }; +//ImageFrame, Tooltip, use common::assets::Asset; use conrod_core::{ color, @@ -17,6 +19,7 @@ use conrod_core::{ widget::{text_box::Event as TextBoxEvent, Button, Image, List, Rectangle, Text, TextBox}, widget_ids, Borderable, Color, Colorable, Labelable, Positionable, Sizeable, Widget, }; +use iced::Column; use image::DynamicImage; use rand::{seq::SliceRandom, thread_rng, Rng}; use std::time::Duration; @@ -122,6 +125,32 @@ image_ids! { } } +image_ids_ice! { + struct IcedImgs { + + v_logo: "voxygen.element.v_logo", + + info_frame: "voxygen.element.frames.info_frame_2", + banner: "voxygen.element.frames.banner", + + banner_bottom: "voxygen.element.frames.banner_bottom", + + + bg: "voxygen.background.bg_main", + banner_top: "voxygen.element.frames.banner_top", + button: "voxygen.element.buttons.button", + button_hover: "voxygen.element.buttons.button_hover", + button_press: "voxygen.element.buttons.button_press", + input_bg_top: "voxygen.element.misc_bg.textbox_top", + input_bg_mid: "voxygen.element.misc_bg.textbox_mid", + input_bg_bot: "voxygen.element.misc_bg.textbox_bot", + disclaimer: "voxygen.element.frames.disclaimer", + + + nothing: (), + } +} + rotation_image_ids! { pub struct ImgsRot { @@ -158,8 +187,31 @@ pub struct PopupData { popup_type: PopupType, } +// No state currently +struct IcedState { + imgs: IcedImgs, +} +pub type Message = Event; +impl IcedState { + pub fn view(&mut self) -> iced::Column { + Column::new().push(ui::ice::Image::new( + (self.imgs.bg, ui::Rotation::None), + 500.0, + 500.0, + )) + } + + pub fn update(message: Message) { + match message { + _ => unimplemented!(), + } + } +} + pub struct MainMenuUi { ui: Ui, + ice_ui: IcedUi, + ice_state: IcedState, ids: Ids, imgs: Imgs, rot_imgs: ImgsRot, @@ -225,8 +277,15 @@ impl<'a> MainMenuUi { let fonts = ConrodVoxygenFonts::load(&voxygen_i18n.fonts, &mut ui) .expect("Impossible to load fonts!"); + let mut ice_ui = IcedUi::new(window).unwrap(); + let ice_state = IcedState { + imgs: IcedImgs::load(&mut ice_ui).expect("Failed to load images"), + }; + Self { ui, + ice_ui, + ice_state, ids, imgs, rot_imgs, @@ -276,7 +335,7 @@ impl<'a> MainMenuUi { let intro_text = &self.voxygen_i18n.get("main.login_process"); // Tooltip - let _tooltip = Tooltip::new({ + /*let _tooltip = Tooltip::new({ // Edge images [t, b, r, l] // Corner images [tr, tl, br, bl] let edge = &self.rot_imgs.tt_side; @@ -291,7 +350,7 @@ impl<'a> MainMenuUi { .title_font_size(self.fonts.cyri.scale(15)) .desc_font_size(self.fonts.cyri.scale(10)) .font_id(self.fonts.cyri.conrod_id) - .desc_text_color(TEXT_COLOR_2); + .desc_text_color(TEXT_COLOR_2);*/ // Background image, Veloren logo, Alpha-Version Label Image::new(if self.connect { @@ -897,11 +956,18 @@ impl<'a> MainMenuUi { pub fn handle_event(&mut self, event: ui::Event) { self.ui.handle_event(event); } + pub fn handle_iced_event(&mut self, event: ui::ice::Event) { self.ice_ui.handle_event(event); } + pub fn maintain(&mut self, global_state: &mut GlobalState, dt: Duration) -> Vec { let events = self.update_layout(global_state, dt); self.ui.maintain(global_state.window.renderer_mut(), None); + self.ice_ui + .maintain(self.ice_state.view(), global_state.window.renderer_mut()); events } - pub fn render(&self, renderer: &mut Renderer) { self.ui.render(renderer, None); } + pub fn render(&self, renderer: &mut Renderer) { + self.ui.render(renderer, None); + self.ice_ui.render(renderer); + } } diff --git a/voxygen/src/run.rs b/voxygen/src/run.rs index a8eac50d09..8253440906 100644 --- a/voxygen/src/run.rs +++ b/voxygen/src/run.rs @@ -34,6 +34,13 @@ pub fn run(mut global_state: GlobalState, event_loop: EventLoop) { if let Some(event) = ui::Event::try_from(&event, global_state.window.window()) { global_state.window.send_event(Event::Ui(event)); } + // iced ui events + // TODO: no clone + if let winit::Event::WindowEvent { event, .. } = event.clone() { + if let Some(event) = ui::ice::window_event(event) { + global_state.window.send_event(Event::IcedUi(event)); + } + } match event { winit::event::Event::NewEvents(_) => { diff --git a/voxygen/src/ui/ice/clipboard.rs b/voxygen/src/ui/ice/clipboard.rs new file mode 100644 index 0000000000..feb78983ab --- /dev/null +++ b/voxygen/src/ui/ice/clipboard.rs @@ -0,0 +1,12 @@ +// Taken from https://github.com/hecrj/iced/blob/e1438774af809c2951c4c7446638500446c81111/winit/src/clipboard.rs +pub struct Clipboard(window_clipboard::Clipboard); + +impl Clipboard { + pub fn new(window: &winit::Window) -> Option { + window_clipboard::Clipboard::new(window).map(Clipboard).ok() + } +} + +impl iced::Clipboard for Clipboard { + fn content(&self) -> Option { self.0.read().ok() } +} diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs new file mode 100644 index 0000000000..3b7c2b5902 --- /dev/null +++ b/voxygen/src/ui/ice/mod.rs @@ -0,0 +1,72 @@ +// tooltip_manager: TooltipManager, +mod clipboard; +mod renderer; +mod widget; +mod winit_conversion; + +pub use graphic::{Id, Rotation}; +pub use iced::Event; +pub use renderer::IcedRenderer; +pub use widget::image::Image; +pub use winit_conversion::window_event; + +use super::graphic::{self, Graphic}; +use crate::{render::Renderer, window::Window, Error}; +use clipboard::Clipboard; +use iced::{Cache, Element, MouseCursor, Size, UserInterface}; + +pub struct IcedUi { + renderer: IcedRenderer, + cache: Option, + events: Vec, + clipboard: Clipboard, +} +impl IcedUi { + pub fn new(window: &mut Window) -> Result { + Ok(Self { + renderer: IcedRenderer::new(window)?, + cache: Some(Cache::new()), + events: Vec::new(), + // TODO: handle None + clipboard: Clipboard::new(window.window()).unwrap(), + }) + } + + // Add an new graphic that is referencable via the returned Id + pub fn add_graphic(&mut self, graphic: Graphic) -> graphic::Id { + self.renderer.add_graphic(graphic) + } + + // TODO: handle scaling here + pub fn handle_event(&mut self, event: Event) { self.events.push(event); } + + // TODO: produce root internally??? + pub fn maintain<'a, M, E: Into>>( + &mut self, + root: E, + renderer: &mut Renderer, + ) -> (Vec, MouseCursor) { + // TODO: convert to f32 at source + let window_size = self.renderer.scaled_window_size().map(|e| e as f32); + + let mut user_interface = UserInterface::build( + root, + Size::new(window_size.x, window_size.y), + self.cache.take().unwrap(), + &mut self.renderer, + ); + + let messages = + user_interface.update(self.events.drain(..), Some(&self.clipboard), &self.renderer); + + let (primitive, mouse_cursor) = user_interface.draw(&mut self.renderer); + + self.renderer.draw(primitive, renderer); + + self.cache = Some(user_interface.into_cache()); + + (messages, mouse_cursor) + } + + pub fn render(&self, renderer: &mut Renderer) { self.renderer.render(renderer, None); } +} diff --git a/voxygen/src/ui/ice/renderer.rs b/voxygen/src/ui/ice/renderer.rs new file mode 100644 index 0000000000..0f3a1b1403 --- /dev/null +++ b/voxygen/src/ui/ice/renderer.rs @@ -0,0 +1,400 @@ +mod column; +mod image; + +use super::{ + super::{ + cache::Cache, + graphic::{self, Graphic, TexId}, + scale::{Scale, ScaleMode}, + }, + widget, +}; +use crate::{ + render::{ + create_ui_quad, Consts, DynamicModel, Globals, Mesh, Renderer, UiLocals, UiMode, UiPipeline, + }, + window::Window, + Error, +}; +//use log::warn; +use std::ops::Range; +use vek::*; + +enum DrawKind { + Image(TexId), + // Text and non-textured geometry + Plain, +} +enum DrawCommand { + Draw { kind: DrawKind, verts: Range }, + Scissor(Aabr), + WorldPos(Option), +} +impl DrawCommand { + fn image(verts: Range, id: TexId) -> DrawCommand { + DrawCommand::Draw { + kind: DrawKind::Image(id), + verts, + } + } + + fn plain(verts: Range) -> DrawCommand { + DrawCommand::Draw { + kind: DrawKind::Plain, + verts, + } + } +} + +enum State { + Image(TexId), + Plain, +} + +pub enum Primitive { + // Allocation :( + Group { + primitives: Vec, + }, + Image { + handle: widget::image::Handle, + bounds: iced::Rectangle, + }, +} + +pub struct IcedRenderer { + //image_map: Map<(Image, Rotation)>, + cache: Cache, + // Model for drawing the ui + model: DynamicModel, + // Consts to specify positions of ingame elements (e.g. Nametags) + ingame_locals: Vec>, + // Consts for default ui drawing position (ie the interface) + interface_locals: Consts, + default_globals: Consts, + + // Window size for updating scaling + //window_resized: Option>, + // Used to delay cache resizing until after current frame is drawn + //need_cache_resize: bool, + // Scaling of the ui + scale: Scale, + + half_res: Vec2, + // Pixel perfection alignment + align: Vec2, + // Pretend dims :) (i.e. scaled) + win_dims: Vec2, + + // Per-frame/update + current_state: State, + mesh: Mesh, + start: usize, + // Draw commands for the next render + draw_commands: Vec, + //current_scissor: Aabr, +} +impl IcedRenderer { + pub fn new(window: &mut Window) -> Result { + let scale = Scale::new(window, ScaleMode::Absolute(1.0)); + // TODO: looks like we can just get this from scale + let win_dims = scale.scaled_window_size().map(|e| e as f32); + + let renderer = window.renderer_mut(); + let res = renderer.get_resolution(); + + let half_res = res.map(|e| e as f32 / 2.0); + let align = align(res); + + Ok(Self { + cache: Cache::new(renderer)?, + draw_commands: Vec::new(), + model: renderer.create_dynamic_model(100)?, + interface_locals: renderer.create_consts(&[UiLocals::default()])?, + default_globals: renderer.create_consts(&[Globals::default()])?, + ingame_locals: Vec::new(), + //window_resized: None, + //need_cache_resize: false, + mesh: Mesh::new(), + current_state: State::Plain, + scale, + half_res, + align, + win_dims, + start: 0, + //current_scissor: default_scissor(renderer), + }) + } + + pub fn scaled_window_size(&self) -> Vec2 { self.scale.scaled_window_size() } + + pub fn add_graphic(&mut self, graphic: Graphic) -> graphic::Id { + self.cache.add_graphic(graphic) + } + + pub fn draw(&mut self, primitive: Primitive, renderer: &mut Renderer) { + /*if self.need_cache_resize { + // Resize graphic cache + self.cache.resize_graphic_cache(renderer).unwrap(); + // Resize glyph cache + self.cache.resize_glyph_cache(renderer).unwrap(); + + self.need_cache_resize = false; + }*/ + + // Re-use memory + self.draw_commands.clear(); + self.mesh.clear(); + + self.current_state = State::Plain; + self.start = 0; + + //self.current_scissor = default_scissor(renderer); + + self.draw_primitive(primitive, renderer); + + // Enter the final command. + self.draw_commands.push(match self.current_state { + State::Plain => DrawCommand::plain(self.start..self.mesh.vertices().len()), + State::Image(id) => DrawCommand::image(self.start..self.mesh.vertices().len(), id), + }); + + // Draw glyph cache (use for debugging). + /*self.draw_commands + .push(DrawCommand::Scissor(default_scissor(renderer))); + start = mesh.vertices().len(); + mesh.push_quad(create_ui_quad( + Aabr { + min: (-1.0, -1.0).into(), + max: (1.0, 1.0).into(), + }, + Aabr { + min: (0.0, 1.0).into(), + max: (1.0, 0.0).into(), + }, + Rgba::new(1.0, 1.0, 1.0, 0.8), + UiMode::Text, + )); + self.draw_commands + .push(DrawCommand::plain(start..mesh.vertices().len()));*/ + + // Create a larger dynamic model if the mesh is larger than the current model + // size. + if self.model.vbuf.len() < self.mesh.vertices().len() { + self.model = renderer + .create_dynamic_model(self.mesh.vertices().len() * 4 / 3) + .unwrap(); + } + // Update model with new mesh. + renderer.update_model(&self.model, &self.mesh, 0).unwrap(); + + // Handle window resizing. + /*if let Some(new_dims) = self.window_resized.take() { + let (old_w, old_h) = self.scale.scaled_window_size().into_tuple(); + self.scale.window_resized(new_dims, renderer); + let (w, h) = self.scale.scaled_window_size().into_tuple(); + self.ui.handle_event(Input::Resize(w, h)); + + // Avoid panic in graphic cache when minimizing. + // Avoid resetting cache if window size didn't change + // Somewhat inefficient for elements that won't change size after a window resize + let res = renderer.get_resolution(); + self.need_cache_resize = res.x > 0 && res.y > 0 && !(old_w == w && old_h == h); + }*/ + } + + fn gl_aabr(&self, bounds: iced::Rectangle) -> Aabr { + /*let (ui_win_w, ui_win_h) = self.win_dims.into_tuple(); + let (l, b) = aabr.min.into_tuple(); + let (r, t) = aabr.max.into_tuple(); + let vx = |x: f64| (x / ui_win_w * 2.0) as f32; + let vy = |y: f64| (y / ui_win_h * 2.0) as f32; + let min = Vec2::new( + ((vx(l) * half_res.x + x_align).round() - x_align) / half_res.x, + ((vy(b) * half_res.y + y_align).round() - y_align) / half_res.y, + ); + let max = Vec2::new( + ((vx(r) * half_res.x + x_align).round() - x_align) / half_res.x, + ((vy(t) * half_res.y + y_align).round() - y_align) / half_res.y, + );*/ + let flipped_y = self.win_dims.y - bounds.y; + let half_win_dims = self.win_dims.map(|e| e / 2.0); + let half_res = self.half_res; + let min = (((Vec2::new(bounds.x, flipped_y - bounds.height) - half_win_dims) + / half_win_dims + * half_res + + self.align) + .map(|e| e.round()) + - self.align) + / half_res; + let max = (((Vec2::new(bounds.x + bounds.width, flipped_y) - half_win_dims) + / half_win_dims + * half_res + + self.align) + .map(|e| e.round()) + - self.align) + / half_res; + Aabr { min, max } + } + + fn draw_primitive(&mut self, primitive: Primitive, renderer: &mut Renderer) { + match primitive { + Primitive::Group { primitives } => { + primitives + .into_iter() + .for_each(|p| self.draw_primitive(p, renderer)); + }, + Primitive::Image { handle, bounds } => { + let (graphic_id, rotation) = handle; + let gl_aabr = self.gl_aabr(bounds); + + let graphic_cache = self.cache.graphic_cache_mut(); + + match graphic_cache.get_graphic(graphic_id) { + Some(Graphic::Blank) | None => return, + _ => {}, + } + + // TODO provide color with the image + // let color = + // srgba_to_linear(color.unwrap_or(conrod_core::color::WHITE).to_fsa(). + // into()); + let color = Rgba::from([1.0, 1.0, 1.0, 1.0]); + + let resolution = Vec2::new( + (gl_aabr.size().w * self.half_res.x).round() as u16, + (gl_aabr.size().h * self.half_res.y).round() as u16, + ); + // Transform the source rectangle into uv coordinate. + // TODO: Make sure this is right. + let source_aabr = { + let (uv_l, uv_r, uv_b, uv_t) = (0.0, 1.0, 0.0, 1.0); + /*match source_rect { + Some(src_rect) => { + let (l, r, b, t) = src_rect.l_r_b_t(); + ((l / image_w) as f32, + (r / image_w) as f32, + (b / image_h) as f32, + (t / image_h) as f32) + } + None => (0.0, 1.0, 0.0, 1.0), + };*/ + Aabr { + min: Vec2::new(uv_l, uv_b), + max: Vec2::new(uv_r, uv_t), + } + }; + + // Cache graphic at particular resolution. + let (uv_aabr, tex_id) = match graphic_cache.cache_res( + renderer, + graphic_id, + resolution, + source_aabr, + rotation, + ) { + // TODO: get dims from graphic_cache (or have it return floats directly) + Some((aabr, tex_id)) => { + let cache_dims = graphic_cache + .get_tex(tex_id) + .get_dimensions() + .map(|e| e as f32); + let min = Vec2::new(aabr.min.x as f32, aabr.max.y as f32) / cache_dims; + let max = Vec2::new(aabr.max.x as f32, aabr.min.y as f32) / cache_dims; + (Aabr { min, max }, tex_id) + }, + None => return, + }; + + match self.current_state { + // Switch to the image state if we are not in it already. + State::Plain => { + self.draw_commands + .push(DrawCommand::plain(self.start..self.mesh.vertices().len())); + self.start = self.mesh.vertices().len(); + self.current_state = State::Image(tex_id); + }, + // If the image is cached in a different texture switch to the new one + State::Image(id) => { + if id != tex_id { + self.draw_commands.push(DrawCommand::image( + self.start..self.mesh.vertices().len(), + id, + )); + self.start = self.mesh.vertices().len(); + self.current_state = State::Image(tex_id); + } + }, + } + + self.mesh + .push_quad(create_ui_quad(gl_aabr, uv_aabr, color, UiMode::Image)); + }, + } + } + + pub fn render(&self, renderer: &mut Renderer, maybe_globals: Option<&Consts>) { + let mut scissor = default_scissor(renderer); + let globals = maybe_globals.unwrap_or(&self.default_globals); + let mut locals = &self.interface_locals; + for draw_command in self.draw_commands.iter() { + match draw_command { + DrawCommand::Scissor(new_scissor) => { + scissor = *new_scissor; + }, + DrawCommand::WorldPos(index) => { + locals = index.map_or(&self.interface_locals, |i| &self.ingame_locals[i]); + }, + DrawCommand::Draw { kind, verts } => { + let tex = match kind { + DrawKind::Image(tex_id) => self.cache.graphic_cache().get_tex(*tex_id), + DrawKind::Plain => self.cache.glyph_cache_tex(), + }; + let model = self.model.submodel(verts.clone()); + renderer.render_ui_element(&model, tex, scissor, globals, locals); + }, + } + } + } +} + +// Given the the resolution determines the offset needed to align integer +// offsets from the center of the sceen to pixels +fn align(res: Vec2) -> Vec2 { + // If the resolution is odd then the center of the screen will be within the + // middle of a pixel so we need to offset by 0.5 pixels to be on the edge of + // a pixel + res.map(|e| (e & 1) as f32 * 0.5) +} + +fn default_scissor(renderer: &Renderer) -> Aabr { + let (screen_w, screen_h) = renderer.get_resolution().map(|e| e as u16).into_tuple(); + Aabr { + min: Vec2 { x: 0, y: 0 }, + max: Vec2 { + x: screen_w, + y: screen_h, + }, + } +} + +impl iced::Renderer for IcedRenderer { + // Default styling + type Defaults = (); + // TODO: use graph of primitives to enable diffing??? + type Output = (Primitive, iced::MouseCursor); + + fn layout<'a, M>( + &mut self, + element: &iced::Element<'a, M, Self>, + limits: &iced::layout::Limits, + ) -> iced::layout::Node { + let node = element.layout(self, limits); + + // Trim text measurements cache? + + node + } +} + +// TODO: impl Debugger diff --git a/voxygen/src/ui/ice/renderer/column.rs b/voxygen/src/ui/ice/renderer/column.rs new file mode 100644 index 0000000000..80a9f7ace3 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/column.rs @@ -0,0 +1,34 @@ +use super::{IcedRenderer, Primitive}; +use iced::{column, Element, Layout, MouseCursor, Point}; + +impl column::Renderer for IcedRenderer { + fn draw( + &mut self, + defaults: &Self::Defaults, + content: &[Element<'_, M, Self>], + layout: Layout<'_>, + cursor_position: Point, + ) -> Self::Output { + let mut mouse_cursor = MouseCursor::OutOfBounds; + + ( + Primitive::Group { + primitives: content + .iter() + .zip(layout.children()) + .map(|(child, layout)| { + let (primitive, new_mouse_cursor) = + child.draw(self, defaults, layout, cursor_position); + + if new_mouse_cursor > mouse_cursor { + mouse_cursor = new_mouse_cursor; + } + + primitive + }) + .collect(), + }, + mouse_cursor, + ) + } +} diff --git a/voxygen/src/ui/ice/renderer/image.rs b/voxygen/src/ui/ice/renderer/image.rs new file mode 100644 index 0000000000..7f25595774 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/image.rs @@ -0,0 +1,14 @@ +use super::{super::widget::image, IcedRenderer, Primitive}; +use iced::MouseCursor; + +impl image::Renderer for IcedRenderer { + fn draw(&mut self, handle: image::Handle, layout: iced::Layout<'_>) -> Self::Output { + ( + Primitive::Image { + handle, + bounds: layout.bounds(), + }, + MouseCursor::OutOfBounds, + ) + } +} diff --git a/voxygen/src/ui/ice/widget.rs b/voxygen/src/ui/ice/widget.rs new file mode 100644 index 0000000000..14995d4228 --- /dev/null +++ b/voxygen/src/ui/ice/widget.rs @@ -0,0 +1 @@ +pub mod image; diff --git a/voxygen/src/ui/ice/widget/image.rs b/voxygen/src/ui/ice/widget/image.rs new file mode 100644 index 0000000000..bae4a1ea18 --- /dev/null +++ b/voxygen/src/ui/ice/widget/image.rs @@ -0,0 +1,63 @@ +use super::super::super::graphic::{self, Rotation}; +use iced::{layout, Element, Hasher, Layout, Length, Point, Size, Widget}; +use std::hash::Hash; + +// TODO: consider iced's approach to images and caching image data +// Also `Graphic` might be a better name for this is it wasn't already in use +// elsewhere + +pub type Handle = (graphic::Id, Rotation); + +pub struct Image { + handle: Handle, + size: Size, +} + +impl Image { + pub fn new(handle: Handle, w: f32, h: f32) -> Self { + let size = Size::new(w, h); + Self { handle, size } + } +} + +impl Widget for Image +where + R: self::Renderer, +{ + fn width(&self) -> Length { Length::Fill } + + fn height(&self) -> Length { Length::Fill } + + fn layout(&self, _renderer: &R, _limits: &layout::Limits) -> layout::Node { + // We don't care about aspect ratios here :p + layout::Node::new(self.size) + // Infinite sizes confusing + //layout::Node::new(limits.resolve(self.size)) + } + + fn draw( + &self, + renderer: &mut R, + _defaults: &R::Defaults, + layout: Layout<'_>, + _cursor_position: Point, + ) -> R::Output { + renderer.draw(self.handle, layout) + } + + fn hash_layout(&self, state: &mut Hasher) { + self.size.width.to_bits().hash(state); + self.size.height.to_bits().hash(state); + } +} + +pub trait Renderer: iced::Renderer { + fn draw(&mut self, handle: Handle, layout: Layout<'_>) -> Self::Output; +} + +impl<'a, M, R> From for Element<'a, M, R> +where + R: self::Renderer, +{ + fn from(image: Image) -> Element<'a, M, R> { Element::new(image) } +} diff --git a/voxygen/src/ui/ice/winit_conversion.rs b/voxygen/src/ui/ice/winit_conversion.rs new file mode 100644 index 0000000000..79538094ff --- /dev/null +++ b/voxygen/src/ui/ice/winit_conversion.rs @@ -0,0 +1,271 @@ +// Using reference impl: https://github.com/hecrj/iced/blob/e1438774af809c2951c4c7446638500446c81111/winit/src/conversion.rs +use iced::{ + input::{ + keyboard::{self, KeyCode, ModifiersState}, + mouse, ButtonState, + }, + window, Event, +}; + +/// Converts a winit event into an iced event. +pub fn window_event(event: winit::WindowEvent) -> Option { + use winit::WindowEvent; + + match event { + WindowEvent::Resized(new_size) => { + let logical_size: winit::dpi::LogicalSize = new_size; + + Some(Event::Window(window::Event::Resized { + width: logical_size.width as u32, + height: logical_size.height as u32, + })) + }, + WindowEvent::CursorMoved { position, .. } => { + let position: winit::dpi::LogicalPosition = position; + + Some(Event::Mouse(mouse::Event::CursorMoved { + x: position.x as f32, + y: position.y as f32, + })) + }, + WindowEvent::MouseInput { button, state, .. } => Some(Event::Mouse(mouse::Event::Input { + button: mouse_button(button), + state: button_state(state), + })), + WindowEvent::MouseWheel { delta, .. } => match delta { + winit::MouseScrollDelta::LineDelta(delta_x, delta_y) => { + Some(Event::Mouse(mouse::Event::WheelScrolled { + delta: mouse::ScrollDelta::Lines { + x: delta_x, + y: delta_y, + }, + })) + }, + winit::MouseScrollDelta::PixelDelta(position) => { + let position: winit::dpi::LogicalPosition = position; + Some(Event::Mouse(mouse::Event::WheelScrolled { + delta: mouse::ScrollDelta::Pixels { + x: position.x as f32, + y: position.y as f32, + }, + })) + }, + }, + WindowEvent::ReceivedCharacter(c) => { + Some(Event::Keyboard(keyboard::Event::CharacterReceived(c))) + }, + WindowEvent::KeyboardInput { + input: + winit::KeyboardInput { + virtual_keycode: Some(virtual_keycode), + state, + modifiers, + .. + }, + .. + } => Some(Event::Keyboard(keyboard::Event::Input { + key_code: key_code(virtual_keycode), + state: button_state(state), + modifiers: modifiers_state(modifiers), + })), + // iced also can use file hovering events but we don't need them right now + _ => None, + } +} + +// iced has a function for converting mouse cursors here + +/// Converts winit mouse button to iced mouse button +fn mouse_button(mouse_button: winit::MouseButton) -> mouse::Button { + match mouse_button { + winit::MouseButton::Left => mouse::Button::Left, + winit::MouseButton::Right => mouse::Button::Right, + winit::MouseButton::Middle => mouse::Button::Middle, + winit::MouseButton::Other(other) => mouse::Button::Other(other), + } +} + +/// Converts winit `ElementState` to an iced button state +fn button_state(element_state: winit::ElementState) -> ButtonState { + match element_state { + winit::ElementState::Pressed => ButtonState::Pressed, + winit::ElementState::Released => ButtonState::Released, + } +} + +/// Converts winit `ModifiersState` to iced `ModifiersState` +fn modifiers_state(modifiers: winit::ModifiersState) -> ModifiersState { + ModifiersState { + shift: modifiers.shift, + control: modifiers.ctrl, + alt: modifiers.alt, + logo: modifiers.logo, + } +} + +/// Converts winit `VirtualKeyCode` to iced `KeyCode` +fn key_code(virtual_keycode: winit::VirtualKeyCode) -> KeyCode { + match virtual_keycode { + winit::VirtualKeyCode::Key1 => KeyCode::Key1, + winit::VirtualKeyCode::Key2 => KeyCode::Key2, + winit::VirtualKeyCode::Key3 => KeyCode::Key3, + winit::VirtualKeyCode::Key4 => KeyCode::Key4, + winit::VirtualKeyCode::Key5 => KeyCode::Key5, + winit::VirtualKeyCode::Key6 => KeyCode::Key6, + winit::VirtualKeyCode::Key7 => KeyCode::Key7, + winit::VirtualKeyCode::Key8 => KeyCode::Key8, + winit::VirtualKeyCode::Key9 => KeyCode::Key9, + winit::VirtualKeyCode::Key0 => KeyCode::Key0, + winit::VirtualKeyCode::A => KeyCode::A, + winit::VirtualKeyCode::B => KeyCode::B, + winit::VirtualKeyCode::C => KeyCode::C, + winit::VirtualKeyCode::D => KeyCode::D, + winit::VirtualKeyCode::E => KeyCode::E, + winit::VirtualKeyCode::F => KeyCode::F, + winit::VirtualKeyCode::G => KeyCode::G, + winit::VirtualKeyCode::H => KeyCode::H, + winit::VirtualKeyCode::I => KeyCode::I, + winit::VirtualKeyCode::J => KeyCode::J, + winit::VirtualKeyCode::K => KeyCode::K, + winit::VirtualKeyCode::L => KeyCode::L, + winit::VirtualKeyCode::M => KeyCode::M, + winit::VirtualKeyCode::N => KeyCode::N, + winit::VirtualKeyCode::O => KeyCode::O, + winit::VirtualKeyCode::P => KeyCode::P, + winit::VirtualKeyCode::Q => KeyCode::Q, + winit::VirtualKeyCode::R => KeyCode::R, + winit::VirtualKeyCode::S => KeyCode::S, + winit::VirtualKeyCode::T => KeyCode::T, + winit::VirtualKeyCode::U => KeyCode::U, + winit::VirtualKeyCode::V => KeyCode::V, + winit::VirtualKeyCode::W => KeyCode::W, + winit::VirtualKeyCode::X => KeyCode::X, + winit::VirtualKeyCode::Y => KeyCode::Y, + winit::VirtualKeyCode::Z => KeyCode::Z, + winit::VirtualKeyCode::Escape => KeyCode::Escape, + winit::VirtualKeyCode::F1 => KeyCode::F1, + winit::VirtualKeyCode::F2 => KeyCode::F2, + winit::VirtualKeyCode::F3 => KeyCode::F3, + winit::VirtualKeyCode::F4 => KeyCode::F4, + winit::VirtualKeyCode::F5 => KeyCode::F5, + winit::VirtualKeyCode::F6 => KeyCode::F6, + winit::VirtualKeyCode::F7 => KeyCode::F7, + winit::VirtualKeyCode::F8 => KeyCode::F8, + winit::VirtualKeyCode::F9 => KeyCode::F9, + winit::VirtualKeyCode::F10 => KeyCode::F10, + winit::VirtualKeyCode::F11 => KeyCode::F11, + winit::VirtualKeyCode::F12 => KeyCode::F12, + winit::VirtualKeyCode::F13 => KeyCode::F13, + winit::VirtualKeyCode::F14 => KeyCode::F14, + winit::VirtualKeyCode::F15 => KeyCode::F15, + winit::VirtualKeyCode::F16 => KeyCode::F16, + winit::VirtualKeyCode::F17 => KeyCode::F17, + winit::VirtualKeyCode::F18 => KeyCode::F18, + winit::VirtualKeyCode::F19 => KeyCode::F19, + winit::VirtualKeyCode::F20 => KeyCode::F20, + winit::VirtualKeyCode::F21 => KeyCode::F21, + winit::VirtualKeyCode::F22 => KeyCode::F22, + winit::VirtualKeyCode::F23 => KeyCode::F23, + winit::VirtualKeyCode::F24 => KeyCode::F24, + winit::VirtualKeyCode::Snapshot => KeyCode::Snapshot, + winit::VirtualKeyCode::Scroll => KeyCode::Scroll, + winit::VirtualKeyCode::Pause => KeyCode::Pause, + winit::VirtualKeyCode::Insert => KeyCode::Insert, + winit::VirtualKeyCode::Home => KeyCode::Home, + winit::VirtualKeyCode::Delete => KeyCode::Delete, + winit::VirtualKeyCode::End => KeyCode::End, + winit::VirtualKeyCode::PageDown => KeyCode::PageDown, + winit::VirtualKeyCode::PageUp => KeyCode::PageUp, + winit::VirtualKeyCode::Left => KeyCode::Left, + winit::VirtualKeyCode::Up => KeyCode::Up, + winit::VirtualKeyCode::Right => KeyCode::Right, + winit::VirtualKeyCode::Down => KeyCode::Down, + winit::VirtualKeyCode::Back => KeyCode::Backspace, + winit::VirtualKeyCode::Return => KeyCode::Enter, + winit::VirtualKeyCode::Space => KeyCode::Space, + winit::VirtualKeyCode::Compose => KeyCode::Compose, + winit::VirtualKeyCode::Caret => KeyCode::Caret, + winit::VirtualKeyCode::Numlock => KeyCode::Numlock, + winit::VirtualKeyCode::Numpad0 => KeyCode::Numpad0, + winit::VirtualKeyCode::Numpad1 => KeyCode::Numpad1, + winit::VirtualKeyCode::Numpad2 => KeyCode::Numpad2, + winit::VirtualKeyCode::Numpad3 => KeyCode::Numpad3, + winit::VirtualKeyCode::Numpad4 => KeyCode::Numpad4, + winit::VirtualKeyCode::Numpad5 => KeyCode::Numpad5, + winit::VirtualKeyCode::Numpad6 => KeyCode::Numpad6, + winit::VirtualKeyCode::Numpad7 => KeyCode::Numpad7, + winit::VirtualKeyCode::Numpad8 => KeyCode::Numpad8, + winit::VirtualKeyCode::Numpad9 => KeyCode::Numpad9, + winit::VirtualKeyCode::AbntC1 => KeyCode::AbntC1, + winit::VirtualKeyCode::AbntC2 => KeyCode::AbntC2, + winit::VirtualKeyCode::Add => KeyCode::Add, + winit::VirtualKeyCode::Apostrophe => KeyCode::Apostrophe, + winit::VirtualKeyCode::Apps => KeyCode::Apps, + winit::VirtualKeyCode::At => KeyCode::At, + winit::VirtualKeyCode::Ax => KeyCode::Ax, + winit::VirtualKeyCode::Backslash => KeyCode::Backslash, + winit::VirtualKeyCode::Calculator => KeyCode::Calculator, + winit::VirtualKeyCode::Capital => KeyCode::Capital, + winit::VirtualKeyCode::Colon => KeyCode::Colon, + winit::VirtualKeyCode::Comma => KeyCode::Comma, + winit::VirtualKeyCode::Convert => KeyCode::Convert, + winit::VirtualKeyCode::Decimal => KeyCode::Decimal, + winit::VirtualKeyCode::Divide => KeyCode::Divide, + winit::VirtualKeyCode::Equals => KeyCode::Equals, + winit::VirtualKeyCode::Grave => KeyCode::Grave, + winit::VirtualKeyCode::Kana => KeyCode::Kana, + winit::VirtualKeyCode::Kanji => KeyCode::Kanji, + winit::VirtualKeyCode::LAlt => KeyCode::LAlt, + winit::VirtualKeyCode::LBracket => KeyCode::LBracket, + winit::VirtualKeyCode::LControl => KeyCode::LControl, + winit::VirtualKeyCode::LShift => KeyCode::LShift, + winit::VirtualKeyCode::LWin => KeyCode::LWin, + winit::VirtualKeyCode::Mail => KeyCode::Mail, + winit::VirtualKeyCode::MediaSelect => KeyCode::MediaSelect, + winit::VirtualKeyCode::MediaStop => KeyCode::MediaStop, + winit::VirtualKeyCode::Minus => KeyCode::Minus, + winit::VirtualKeyCode::Multiply => KeyCode::Multiply, + winit::VirtualKeyCode::Mute => KeyCode::Mute, + winit::VirtualKeyCode::MyComputer => KeyCode::MyComputer, + winit::VirtualKeyCode::NavigateForward => KeyCode::NavigateForward, + winit::VirtualKeyCode::NavigateBackward => KeyCode::NavigateBackward, + winit::VirtualKeyCode::NextTrack => KeyCode::NextTrack, + winit::VirtualKeyCode::NoConvert => KeyCode::NoConvert, + winit::VirtualKeyCode::NumpadComma => KeyCode::NumpadComma, + winit::VirtualKeyCode::NumpadEnter => KeyCode::NumpadEnter, + winit::VirtualKeyCode::NumpadEquals => KeyCode::NumpadEquals, + winit::VirtualKeyCode::OEM102 => KeyCode::OEM102, + winit::VirtualKeyCode::Period => KeyCode::Period, + winit::VirtualKeyCode::PlayPause => KeyCode::PlayPause, + winit::VirtualKeyCode::Power => KeyCode::Power, + winit::VirtualKeyCode::PrevTrack => KeyCode::PrevTrack, + winit::VirtualKeyCode::RAlt => KeyCode::RAlt, + winit::VirtualKeyCode::RBracket => KeyCode::RBracket, + winit::VirtualKeyCode::RControl => KeyCode::RControl, + winit::VirtualKeyCode::RShift => KeyCode::RShift, + winit::VirtualKeyCode::RWin => KeyCode::RWin, + winit::VirtualKeyCode::Semicolon => KeyCode::Semicolon, + winit::VirtualKeyCode::Slash => KeyCode::Slash, + winit::VirtualKeyCode::Sleep => KeyCode::Sleep, + winit::VirtualKeyCode::Stop => KeyCode::Stop, + winit::VirtualKeyCode::Subtract => KeyCode::Subtract, + winit::VirtualKeyCode::Sysrq => KeyCode::Sysrq, + winit::VirtualKeyCode::Tab => KeyCode::Tab, + winit::VirtualKeyCode::Underline => KeyCode::Underline, + winit::VirtualKeyCode::Unlabeled => KeyCode::Unlabeled, + winit::VirtualKeyCode::VolumeDown => KeyCode::VolumeDown, + winit::VirtualKeyCode::VolumeUp => KeyCode::VolumeUp, + winit::VirtualKeyCode::Wake => KeyCode::Wake, + winit::VirtualKeyCode::WebBack => KeyCode::WebBack, + winit::VirtualKeyCode::WebFavorites => KeyCode::WebFavorites, + winit::VirtualKeyCode::WebForward => KeyCode::WebForward, + winit::VirtualKeyCode::WebHome => KeyCode::WebHome, + winit::VirtualKeyCode::WebRefresh => KeyCode::WebRefresh, + winit::VirtualKeyCode::WebSearch => KeyCode::WebSearch, + winit::VirtualKeyCode::WebStop => KeyCode::WebStop, + winit::VirtualKeyCode::Yen => KeyCode::Yen, + winit::VirtualKeyCode::Copy => KeyCode::Copy, + winit::VirtualKeyCode::Paste => KeyCode::Paste, + winit::VirtualKeyCode::Cut => KeyCode::Cut, + } +} diff --git a/voxygen/src/ui/img_ids.rs b/voxygen/src/ui/img_ids.rs index cf828a0502..e6e740eb24 100644 --- a/voxygen/src/ui/img_ids.rs +++ b/voxygen/src/ui/img_ids.rs @@ -165,6 +165,25 @@ macro_rules! image_ids { )* }; } +#[macro_export] +macro_rules! image_ids_ice { + ($($v:vis struct $Ids:ident { $( <$T:ty> $( $name:ident: $specifier:expr ),* $(,)? )* })*) => { + $( + $v struct $Ids { + $($( $v $name: crate::ui::GraphicId, )*)* + } + + impl $Ids { + pub fn load(ui: &mut crate::ui::ice::IcedUi) -> Result { + use crate::ui::img_ids::GraphicCreator; + Ok(Self { + $($( $name: ui.add_graphic(<$T as GraphicCreator>::new_graphic($specifier)?), )*)* + }) + } + } + )* + }; +} // TODO: combine with the img_ids macro above using a marker for specific fields // that should be `Rotations` instead of `widget::Id` diff --git a/voxygen/src/ui/mod.rs b/voxygen/src/ui/mod.rs index 0e5ad64e6b..e5140145f9 100644 --- a/voxygen/src/ui/mod.rs +++ b/voxygen/src/ui/mod.rs @@ -7,9 +7,10 @@ mod widgets; pub mod img_ids; #[macro_use] pub mod fonts; +pub mod ice; pub use event::Event; -pub use graphic::{Graphic, SampleStrat, Transform}; +pub use graphic::{Graphic, Id as GraphicId, Rotation, SampleStrat, Transform}; pub use scale::{Scale, ScaleMode}; pub use widgets::{ image_frame::ImageFrame, @@ -36,7 +37,7 @@ use common::{assets, span, util::srgba_to_linear}; use conrod_core::{ event::Input, graph::{self, Graph}, - image::{self, Map}, + image::{Id as ImageId, Map}, input::{touch::Touch, Motion, Widget}, render::{Primitive, PrimitiveKind}, text::{self, font}, @@ -187,7 +188,7 @@ impl Ui { // Get a copy of Scale pub fn scale(&self) -> Scale { self.scale } - pub fn add_graphic(&mut self, graphic: Graphic) -> image::Id { + pub fn add_graphic(&mut self, graphic: Graphic) -> ImageId { self.image_map .insert((self.cache.add_graphic(graphic), Rotation::None)) } @@ -212,7 +213,7 @@ impl Ui { } } - pub fn replace_graphic(&mut self, id: image::Id, graphic: Graphic) { + pub fn replace_graphic(&mut self, id: ImageId, graphic: Graphic) { let graphic_id = if let Some((graphic_id, _)) = self.image_map.get(&id) { *graphic_id } else { diff --git a/voxygen/src/window.rs b/voxygen/src/window.rs index 79eb08a429..499b40777a 100644 --- a/voxygen/src/window.rs +++ b/voxygen/src/window.rs @@ -282,6 +282,8 @@ pub enum Event { InputUpdate(GameInput, bool), /// Event that the ui uses. Ui(ui::Event), + /// Event that the iced ui uses. + IcedUi(ui::ice::Event), /// The view distance has changed. ViewDistanceChanged(u32), /// Game settings have changed. @@ -622,12 +624,6 @@ impl Window { Ok((this, event_loop)) } - pub fn window( - &self, - ) -> &glutin::ContextWrapper { - &self.window - } - pub fn renderer(&self) -> &Renderer { &self.renderer } pub fn renderer_mut(&mut self) -> &mut Renderer { &mut self.renderer } @@ -1377,6 +1373,8 @@ impl Window { pub fn set_keybinding_mode(&mut self, game_input: GameInput) { self.remapping_keybindings = Some(game_input); } + + pub fn window(&self) -> &winit::Window { self.window.window() } } #[derive(Copy, Clone, Hash, Eq, PartialEq, Debug, Serialize, Deserialize)] From e64f8aec6087b89b7c5ecd4a463cb51fa7d88011 Mon Sep 17 00:00:00 2001 From: Imbris Date: Mon, 16 Mar 2020 17:56:50 -0400 Subject: [PATCH 02/61] Get resizing working --- voxygen/src/ui/ice/mod.rs | 81 +++++++++++++++++++++++++++++++--- voxygen/src/ui/ice/renderer.rs | 61 ++++++++++++------------- 2 files changed, 105 insertions(+), 37 deletions(-) diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index 3b7c2b5902..23311491b5 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -10,25 +10,39 @@ pub use renderer::IcedRenderer; pub use widget::image::Image; pub use winit_conversion::window_event; -use super::graphic::{self, Graphic}; +use super::{ + graphic::{self, Graphic}, + scale::{Scale, ScaleMode}, +}; use crate::{render::Renderer, window::Window, Error}; use clipboard::Clipboard; use iced::{Cache, Element, MouseCursor, Size, UserInterface}; +use vek::*; pub struct IcedUi { renderer: IcedRenderer, cache: Option, events: Vec, clipboard: Clipboard, + // Scaling of the ui + scale: Scale, + window_resized: Option>, } impl IcedUi { pub fn new(window: &mut Window) -> Result { + let scale = Scale::new(window, ScaleMode::Absolute(1.0)); + let renderer = window.renderer_mut(); + + let scaled_dims = scale.scaled_window_size().map(|e| e as f32); + Ok(Self { - renderer: IcedRenderer::new(window)?, + renderer: IcedRenderer::new(renderer, scaled_dims)?, cache: Some(Cache::new()), events: Vec::new(), // TODO: handle None clipboard: Clipboard::new(window.window()).unwrap(), + scale, + window_resized: None, }) } @@ -37,8 +51,40 @@ impl IcedUi { self.renderer.add_graphic(graphic) } - // TODO: handle scaling here - pub fn handle_event(&mut self, event: Event) { self.events.push(event); } + pub fn handle_event(&mut self, event: Event) { + use iced::{input::mouse, window}; + match event { + // Intercept resizing events + Event::Window(window::Event::Resized { width, height }) => { + self.window_resized = Some(Vec2::new(width, height)); + }, + // Scale cursor movement events + // Note: in some cases the scaling could be off if a resized event occured in the same + // frame, in practice this shouldn't be an issue + Event::Mouse(mouse::Event::CursorMoved { x, y }) => { + // TODO: return f32 here + let scale = self.scale.scale_factor_logical() as f32; + self.events.push(Event::Mouse(mouse::Event::CursorMoved { + x: x * scale, + y: y * scale, + })); + }, + // Scale pixel scrolling events + Event::Mouse(mouse::Event::WheelScrolled { + delta: mouse::ScrollDelta::Pixels { x, y }, + }) => { + // TODO: return f32 here + let scale = self.scale.scale_factor_logical() as f32; + self.events.push(Event::Mouse(mouse::Event::WheelScrolled { + delta: mouse::ScrollDelta::Pixels { + x: x * scale, + y: y * scale, + }, + })); + }, + event => self.events.push(event), + } + } // TODO: produce root internally??? pub fn maintain<'a, M, E: Into>>( @@ -46,8 +92,33 @@ impl IcedUi { root: E, renderer: &mut Renderer, ) -> (Vec, MouseCursor) { + // Handle window resizing + if let Some(new_dims) = self.window_resized.take() { + let old_scaled_dims = self.scale.scaled_window_size(); + // TODO maybe use u32 in Scale to be consistent with iced + self.scale + .window_resized(new_dims.map(|e| e as f64), renderer); + let scaled_dims = self.scale.scaled_window_size(); + + self.events + .push(Event::Window(iced::window::Event::Resized { + width: scaled_dims.x as u32, + height: scaled_dims.y as u32, + })); + + // Avoid panic in graphic cache when minimizing. + // Avoid resetting cache if window size didn't change + // Somewhat inefficient for elements that won't change size after a window + // resize + let res = renderer.get_resolution(); + if res.x > 0 && res.y > 0 && scaled_dims != old_scaled_dims { + self.renderer + .resize(scaled_dims.map(|e| e as f32), renderer); + } + } + // TODO: convert to f32 at source - let window_size = self.renderer.scaled_window_size().map(|e| e as f32); + let window_size = self.scale.scaled_window_size().map(|e| e as f32); let mut user_interface = UserInterface::build( root, diff --git a/voxygen/src/ui/ice/renderer.rs b/voxygen/src/ui/ice/renderer.rs index 0f3a1b1403..4a04cb5c82 100644 --- a/voxygen/src/ui/ice/renderer.rs +++ b/voxygen/src/ui/ice/renderer.rs @@ -5,7 +5,6 @@ use super::{ super::{ cache::Cache, graphic::{self, Graphic, TexId}, - scale::{Scale, ScaleMode}, }, widget, }; @@ -13,7 +12,6 @@ use crate::{ render::{ create_ui_quad, Consts, DynamicModel, Globals, Mesh, Renderer, UiLocals, UiMode, UiPipeline, }, - window::Window, Error, }; //use log::warn; @@ -73,13 +71,8 @@ pub struct IcedRenderer { interface_locals: Consts, default_globals: Consts, - // Window size for updating scaling - //window_resized: Option>, // Used to delay cache resizing until after current frame is drawn //need_cache_resize: bool, - // Scaling of the ui - scale: Scale, - half_res: Vec2, // Pixel perfection alignment align: Vec2, @@ -95,16 +88,8 @@ pub struct IcedRenderer { //current_scissor: Aabr, } impl IcedRenderer { - pub fn new(window: &mut Window) -> Result { - let scale = Scale::new(window, ScaleMode::Absolute(1.0)); - // TODO: looks like we can just get this from scale - let win_dims = scale.scaled_window_size().map(|e| e as f32); - - let renderer = window.renderer_mut(); - let res = renderer.get_resolution(); - - let half_res = res.map(|e| e as f32 / 2.0); - let align = align(res); + pub fn new(renderer: &mut Renderer, scaled_dims: Vec2) -> Result { + let (half_res, align) = Self::calculate_resolution_dependents(renderer.get_resolution()); Ok(Self { cache: Cache::new(renderer)?, @@ -113,35 +98,32 @@ impl IcedRenderer { interface_locals: renderer.create_consts(&[UiLocals::default()])?, default_globals: renderer.create_consts(&[Globals::default()])?, ingame_locals: Vec::new(), - //window_resized: None, - //need_cache_resize: false, mesh: Mesh::new(), current_state: State::Plain, - scale, half_res, align, - win_dims, + win_dims: scaled_dims, start: 0, //current_scissor: default_scissor(renderer), }) } - pub fn scaled_window_size(&self) -> Vec2 { self.scale.scaled_window_size() } - pub fn add_graphic(&mut self, graphic: Graphic) -> graphic::Id { self.cache.add_graphic(graphic) } + pub fn resize(&mut self, scaled_dims: Vec2, renderer: &mut Renderer) { + self.win_dims = scaled_dims; + + self.update_resolution_dependents(renderer.get_resolution()); + + // Resize graphic cache + self.cache.resize_graphic_cache(renderer); + // Resize glyph cache + self.cache.resize_glyph_cache(renderer).unwrap(); + } + pub fn draw(&mut self, primitive: Primitive, renderer: &mut Renderer) { - /*if self.need_cache_resize { - // Resize graphic cache - self.cache.resize_graphic_cache(renderer).unwrap(); - // Resize glyph cache - self.cache.resize_glyph_cache(renderer).unwrap(); - - self.need_cache_resize = false; - }*/ - // Re-use memory self.draw_commands.clear(); self.mesh.clear(); @@ -203,6 +185,20 @@ impl IcedRenderer { }*/ } + // Returns (half_res, align) + fn calculate_resolution_dependents(res: Vec2) -> (Vec2, Vec2) { + let half_res = res.map(|e| e as f32 / 2.0); + let align = align(res); + + (half_res, align) + } + + fn update_resolution_dependents(&mut self, res: Vec2) { + let (half_res, align) = Self::calculate_resolution_dependents(res); + self.half_res = half_res; + self.align = align; + } + fn gl_aabr(&self, bounds: iced::Rectangle) -> Aabr { /*let (ui_win_w, ui_win_h) = self.win_dims.into_tuple(); let (l, b) = aabr.min.into_tuple(); @@ -360,6 +356,7 @@ impl IcedRenderer { // Given the the resolution determines the offset needed to align integer // offsets from the center of the sceen to pixels +#[inline(always)] fn align(res: Vec2) -> Vec2 { // If the resolution is odd then the center of the screen will be within the // middle of a pixel so we need to offset by 0.5 pixels to be on the edge of From 3b644a40148f924c1981e32456a4e9588afbc407 Mon Sep 17 00:00:00 2001 From: Imbris Date: Tue, 17 Mar 2020 02:31:55 -0400 Subject: [PATCH 03/61] Start experimenting with replicating main menu --- voxygen/src/menu/main/ui.rs | 23 +++++++++------ voxygen/src/ui/ice/mod.rs | 6 ++-- voxygen/src/ui/ice/renderer.rs | 3 +- voxygen/src/ui/ice/renderer/column.rs | 4 +-- voxygen/src/ui/ice/widget/image.rs | 40 +++++++++++++++++++-------- 5 files changed, 51 insertions(+), 25 deletions(-) diff --git a/voxygen/src/menu/main/ui.rs b/voxygen/src/menu/main/ui.rs index 72acdb123f..9c498f7d52 100644 --- a/voxygen/src/menu/main/ui.rs +++ b/voxygen/src/menu/main/ui.rs @@ -4,7 +4,7 @@ use crate::{ ui::{ self, fonts::ConrodVoxygenFonts, - ice::IcedUi, + ice::{Element, IcedUi}, img_ids::{BlankGraphic, ImageGraphic, VoxelGraphic}, Graphic, Ui, }, @@ -19,7 +19,7 @@ use conrod_core::{ widget::{text_box::Event as TextBoxEvent, Button, Image, List, Rectangle, Text, TextBox}, widget_ids, Borderable, Color, Colorable, Labelable, Positionable, Sizeable, Widget, }; -use iced::Column; +use iced::{Column, Row}; use image::DynamicImage; use rand::{seq::SliceRandom, thread_rng, Rng}; use std::time::Duration; @@ -193,12 +193,19 @@ struct IcedState { } pub type Message = Event; impl IcedState { - pub fn view(&mut self) -> iced::Column { - Column::new().push(ui::ice::Image::new( - (self.imgs.bg, ui::Rotation::None), - 500.0, - 500.0, - )) + pub fn view(&mut self) -> Element { + use iced::Length; + + let image1 = ui::ice::Image::new((self.imgs.bg, ui::Rotation::None)); + let image2 = ui::ice::Image::new((self.imgs.bg, ui::Rotation::None)); + let image3 = ui::ice::Image::new((self.imgs.bg, ui::Rotation::None)); + + Row::with_children(vec![image1.into(), image2.into(), image3.into()]) + .width(Length::Fill) + .height(Length::Fill) + .spacing(20) + .padding(20) + .into() } pub fn update(message: Message) { diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index 23311491b5..231d112c30 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -16,9 +16,11 @@ use super::{ }; use crate::{render::Renderer, window::Window, Error}; use clipboard::Clipboard; -use iced::{Cache, Element, MouseCursor, Size, UserInterface}; +use iced::{Cache, MouseCursor, Size, UserInterface}; use vek::*; +pub type Element<'a, M> = iced::Element<'a, M, IcedRenderer>; + pub struct IcedUi { renderer: IcedRenderer, cache: Option, @@ -87,7 +89,7 @@ impl IcedUi { } // TODO: produce root internally??? - pub fn maintain<'a, M, E: Into>>( + pub fn maintain<'a, M, E: Into>>( &mut self, root: E, renderer: &mut Renderer, diff --git a/voxygen/src/ui/ice/renderer.rs b/voxygen/src/ui/ice/renderer.rs index 4a04cb5c82..94a35bb74a 100644 --- a/voxygen/src/ui/ice/renderer.rs +++ b/voxygen/src/ui/ice/renderer.rs @@ -1,5 +1,6 @@ mod column; mod image; +mod row; use super::{ super::{ @@ -255,7 +256,7 @@ impl IcedRenderer { // let color = // srgba_to_linear(color.unwrap_or(conrod_core::color::WHITE).to_fsa(). // into()); - let color = Rgba::from([1.0, 1.0, 1.0, 1.0]); + let color = Rgba::from([1.0, 0.0, 1.0, 0.5]); let resolution = Vec2::new( (gl_aabr.size().w * self.half_res.x).round() as u16, diff --git a/voxygen/src/ui/ice/renderer/column.rs b/voxygen/src/ui/ice/renderer/column.rs index 80a9f7ace3..ab1a7abbd2 100644 --- a/voxygen/src/ui/ice/renderer/column.rs +++ b/voxygen/src/ui/ice/renderer/column.rs @@ -5,7 +5,7 @@ impl column::Renderer for IcedRenderer { fn draw( &mut self, defaults: &Self::Defaults, - content: &[Element<'_, M, Self>], + children: &[Element<'_, M, Self>], layout: Layout<'_>, cursor_position: Point, ) -> Self::Output { @@ -13,7 +13,7 @@ impl column::Renderer for IcedRenderer { ( Primitive::Group { - primitives: content + primitives: children .iter() .zip(layout.children()) .map(|(child, layout)| { diff --git a/voxygen/src/ui/ice/widget/image.rs b/voxygen/src/ui/ice/widget/image.rs index bae4a1ea18..15830839a8 100644 --- a/voxygen/src/ui/ice/widget/image.rs +++ b/voxygen/src/ui/ice/widget/image.rs @@ -10,13 +10,29 @@ pub type Handle = (graphic::Id, Rotation); pub struct Image { handle: Handle, - size: Size, + width: Length, + height: Length, } impl Image { - pub fn new(handle: Handle, w: f32, h: f32) -> Self { - let size = Size::new(w, h); - Self { handle, size } + pub fn new(handle: Handle) -> Self { + let width = Length::Fill; + let height = Length::Fill; + Self { + handle, + width, + height, + } + } + + pub fn width(mut self, width: Length) -> Self { + self.width = width; + self + } + + pub fn height(mut self, height: Length) -> Self { + self.height = height; + self } } @@ -24,15 +40,15 @@ impl Widget for Image where R: self::Renderer, { - fn width(&self) -> Length { Length::Fill } + fn width(&self) -> Length { self.width } - fn height(&self) -> Length { Length::Fill } + fn height(&self) -> Length { self.height } - fn layout(&self, _renderer: &R, _limits: &layout::Limits) -> layout::Node { + fn layout(&self, _renderer: &R, limits: &layout::Limits) -> layout::Node { // We don't care about aspect ratios here :p - layout::Node::new(self.size) - // Infinite sizes confusing - //layout::Node::new(limits.resolve(self.size)) + let size = limits.width(self.width).height(self.height).max(); + + layout::Node::new(size) } fn draw( @@ -46,8 +62,8 @@ where } fn hash_layout(&self, state: &mut Hasher) { - self.size.width.to_bits().hash(state); - self.size.height.to_bits().hash(state); + self.width.hash(state); + self.height.hash(state); } } From 568795fe6fa4e46b9eef782b869af89520487bb1 Mon Sep 17 00:00:00 2001 From: Imbris Date: Wed, 18 Mar 2020 12:36:06 -0400 Subject: [PATCH 04/61] Begin implementing container widget with an image background --- voxygen/src/menu/main/mod.rs | 7 + voxygen/src/menu/main/ui.rs | 48 ++++- voxygen/src/ui/graphic/mod.rs | 21 +++ voxygen/src/ui/ice/mod.rs | 2 +- voxygen/src/ui/ice/renderer.rs | 25 ++- .../ui/ice/renderer/background_container.rs | 28 +++ voxygen/src/ui/ice/renderer/container.rs | 22 +++ voxygen/src/ui/ice/renderer/image.rs | 24 ++- voxygen/src/ui/ice/renderer/row.rs | 34 ++++ voxygen/src/ui/ice/widget.rs | 1 + .../src/ui/ice/widget/background_container.rs | 169 ++++++++++++++++++ voxygen/src/ui/ice/widget/image.rs | 48 ++++- 12 files changed, 399 insertions(+), 30 deletions(-) create mode 100644 voxygen/src/ui/ice/renderer/background_container.rs create mode 100644 voxygen/src/ui/ice/renderer/container.rs create mode 100644 voxygen/src/ui/ice/renderer/row.rs create mode 100644 voxygen/src/ui/ice/widget/background_container.rs diff --git a/voxygen/src/menu/main/mod.rs b/voxygen/src/menu/main/mod.rs index 7f4bb43bd0..b2d685b92d 100644 --- a/voxygen/src/menu/main/mod.rs +++ b/voxygen/src/menu/main/mod.rs @@ -87,6 +87,13 @@ impl PlayState for MainMenuState { Event::Ui(event) => { self.main_menu_ui.handle_event(event); }, + // Pass events to iced ui. + Event::IcedUi(event) => { + self.main_menu_ui.handle_iced_event(event); + }, + Event::InputUpdate(crate::window::GameInput::Jump, true) => { + self.main_menu_ui.show_iced ^= true; + }, // Ignore all other events. _ => {}, } diff --git a/voxygen/src/menu/main/ui.rs b/voxygen/src/menu/main/ui.rs index 9c498f7d52..cb633e5a09 100644 --- a/voxygen/src/menu/main/ui.rs +++ b/voxygen/src/menu/main/ui.rs @@ -19,7 +19,6 @@ use conrod_core::{ widget::{text_box::Event as TextBoxEvent, Button, Image, List, Rectangle, Text, TextBox}, widget_ids, Borderable, Color, Colorable, Labelable, Positionable, Sizeable, Widget, }; -use iced::{Column, Row}; use image::DynamicImage; use rand::{seq::SliceRandom, thread_rng, Rng}; use std::time::Duration; @@ -194,17 +193,44 @@ struct IcedState { pub type Message = Event; impl IcedState { pub fn view(&mut self) -> Element { - use iced::Length; + use iced::{Align, Column, Container, Length, Row}; + use ui::ice::Image; + use vek::*; - let image1 = ui::ice::Image::new((self.imgs.bg, ui::Rotation::None)); - let image2 = ui::ice::Image::new((self.imgs.bg, ui::Rotation::None)); - let image3 = ui::ice::Image::new((self.imgs.bg, ui::Rotation::None)); + let buttons = Column::with_children(vec![ + Image::new(self.imgs.button).fix_aspect_ratio().into(), + Image::new(self.imgs.button).fix_aspect_ratio().into(), + Image::new(self.imgs.button).fix_aspect_ratio().into(), + ]) + .width(Length::Fill) + .max_width(200) + .spacing(5) + .padding(10); - Row::with_children(vec![image1.into(), image2.into(), image3.into()]) + let buttons = Container::new(buttons) + .width(Length::Fill) + .height(Length::Fill) + .align_y(Align::End) + .padding(20); + + let banner_content = + Column::with_children(vec![Image::new(self.imgs.v_logo).fix_aspect_ratio().into()]); + let banner = ui::ice::BackgroundContainer::new(self.imgs.banner, banner_content) + .color(Rgba::new(255, 255, 255, 230)) + .fix_aspect_ratio() + .width(Length::Fill) + .height(Length::Fill); + + let image3 = Image::new(self.imgs.banner_bottom).fix_aspect_ratio(); + + let content = Row::with_children(vec![buttons.into(), banner.into(), image3.into()]) + .width(Length::Fill) + .height(Length::Fill) + .spacing(10); + + ui::ice::BackgroundContainer::new(self.imgs.bg, content) .width(Length::Fill) .height(Length::Fill) - .spacing(20) - .padding(20) .into() } @@ -236,6 +262,7 @@ pub struct MainMenuUi { voxygen_i18n: std::sync::Arc, fonts: ConrodVoxygenFonts, tip_no: u16, + pub show_iced: bool, } impl<'a> MainMenuUi { @@ -314,6 +341,7 @@ impl<'a> MainMenuUi { voxygen_i18n, fonts, tip_no: 0, + show_iced: false, } } @@ -975,6 +1003,8 @@ impl<'a> MainMenuUi { pub fn render(&self, renderer: &mut Renderer) { self.ui.render(renderer, None); - self.ice_ui.render(renderer); + if self.show_iced { + self.ice_ui.render(renderer); + } } } diff --git a/voxygen/src/ui/graphic/mod.rs b/voxygen/src/ui/graphic/mod.rs index b89a9ac9ae..b4e52c4b5c 100644 --- a/voxygen/src/ui/graphic/mod.rs +++ b/voxygen/src/ui/graphic/mod.rs @@ -187,6 +187,27 @@ impl GraphicCache { self.textures.get(id.0).expect("Invalid TexId used") } + pub fn get_graphic_dims(&self, (id, rot): (Id, Rotation)) -> Option<(u32, u32)> { + use image::GenericImageView; + self.get_graphic(id) + .and_then(|graphic| match graphic { + Graphic::Image(image) => Some(image.dimensions()), + Graphic::Voxel(segment, _, _) => { + use common::vol::SizedVol; + let size = segment.size(); + // TODO: HACK because they can be rotated arbitrarily, remove + Some((size.x, size.z)) + }, + Graphic::Blank => None, + }) + .and_then(|(w, h)| match rot { + Rotation::None | Rotation::Cw180 => Some((w, h)), + Rotation::Cw90 | Rotation::Cw270 => Some((h, w)), + // TODO: need dims for these? + Rotation::SourceNorth | Rotation::TargetNorth => None, + }) + } + pub fn clear_cache(&mut self, renderer: &mut Renderer) { self.cache_map.clear(); diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index 231d112c30..1cbeb9f4be 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -7,7 +7,7 @@ mod winit_conversion; pub use graphic::{Id, Rotation}; pub use iced::Event; pub use renderer::IcedRenderer; -pub use widget::image::Image; +pub use widget::{background_container::BackgroundContainer, image::Image}; pub use winit_conversion::window_event; use super::{ diff --git a/voxygen/src/ui/ice/renderer.rs b/voxygen/src/ui/ice/renderer.rs index 94a35bb74a..138f9526ec 100644 --- a/voxygen/src/ui/ice/renderer.rs +++ b/voxygen/src/ui/ice/renderer.rs @@ -1,4 +1,6 @@ +mod background_container; mod column; +mod container; mod image; mod row; @@ -7,7 +9,7 @@ use super::{ cache::Cache, graphic::{self, Graphic, TexId}, }, - widget, + widget, Rotation, }; use crate::{ render::{ @@ -15,6 +17,7 @@ use crate::{ }, Error, }; +use common::util::srgba_to_linear; //use log::warn; use std::ops::Range; use vek::*; @@ -56,11 +59,16 @@ pub enum Primitive { primitives: Vec, }, Image { - handle: widget::image::Handle, + handle: (widget::image::Handle, Rotation), bounds: iced::Rectangle, + color: Rgba, }, } +// Optimization idea inspired by what I think iced wgpu renderer may be doing +// Could have layers of things which don't intersect and thus can be reordered +// arbitrarily + pub struct IcedRenderer { //image_map: Map<(Image, Rotation)>, cache: Cache, @@ -241,7 +249,11 @@ impl IcedRenderer { .into_iter() .for_each(|p| self.draw_primitive(p, renderer)); }, - Primitive::Image { handle, bounds } => { + Primitive::Image { + handle, + bounds, + color, + } => { let (graphic_id, rotation) = handle; let gl_aabr = self.gl_aabr(bounds); @@ -252,11 +264,7 @@ impl IcedRenderer { _ => {}, } - // TODO provide color with the image - // let color = - // srgba_to_linear(color.unwrap_or(conrod_core::color::WHITE).to_fsa(). - // into()); - let color = Rgba::from([1.0, 0.0, 1.0, 0.5]); + let color = srgba_to_linear(color.map(|e| e as f32 / 255.0)); let resolution = Vec2::new( (gl_aabr.size().w * self.half_res.x).round() as u16, @@ -359,6 +367,7 @@ impl IcedRenderer { // offsets from the center of the sceen to pixels #[inline(always)] fn align(res: Vec2) -> Vec2 { + // TODO: does this logic still apply in iced's coordinate system? // If the resolution is odd then the center of the screen will be within the // middle of a pixel so we need to offset by 0.5 pixels to be on the edge of // a pixel diff --git a/voxygen/src/ui/ice/renderer/background_container.rs b/voxygen/src/ui/ice/renderer/background_container.rs new file mode 100644 index 0000000000..2a31ab763a --- /dev/null +++ b/voxygen/src/ui/ice/renderer/background_container.rs @@ -0,0 +1,28 @@ +use super::{ + super::widget::{background_container, image}, + IcedRenderer, Primitive, +}; +use iced::{Element, Layout, Point}; + +impl background_container::Renderer for IcedRenderer { + fn draw( + &mut self, + defaults: &Self::Defaults, + layout: Layout<'_>, + cursor_position: Point, + background: image::Handle, + color: vek::Rgba, + content: &Element<'_, M, Self>, + content_layout: Layout<'_>, + ) -> Self::Output { + let image_primitive = image::Renderer::draw(self, background, color, layout).0; + let (content_primitive, mouse_cursor) = + content.draw(self, defaults, content_layout, cursor_position); + ( + Primitive::Group { + primitives: vec![image_primitive, content_primitive], + }, + mouse_cursor, + ) + } +} diff --git a/voxygen/src/ui/ice/renderer/container.rs b/voxygen/src/ui/ice/renderer/container.rs new file mode 100644 index 0000000000..4a4eaad9ab --- /dev/null +++ b/voxygen/src/ui/ice/renderer/container.rs @@ -0,0 +1,22 @@ +use super::{IcedRenderer, Primitive}; +use iced::{container, Element, Layout, MouseCursor, Point, Rectangle}; + +impl container::Renderer for IcedRenderer { + type Style = (); + + fn draw( + &mut self, + defaults: &Self::Defaults, + bounds: Rectangle, + cursor_position: Point, + _style_sheet: &Self::Style, + content: &Element<'_, M, Self>, + content_layout: Layout<'_>, + ) -> Self::Output { + let (content, mouse_cursor) = content.draw(self, defaults, content_layout, cursor_position); + + // We may have more stuff here if styles are used + + (content, mouse_cursor) + } +} diff --git a/voxygen/src/ui/ice/renderer/image.rs b/voxygen/src/ui/ice/renderer/image.rs index 7f25595774..22b3f8a953 100644 --- a/voxygen/src/ui/ice/renderer/image.rs +++ b/voxygen/src/ui/ice/renderer/image.rs @@ -1,12 +1,30 @@ -use super::{super::widget::image, IcedRenderer, Primitive}; +use super::{ + super::{widget::image, Rotation}, + IcedRenderer, Primitive, +}; use iced::MouseCursor; +use vek::Rgba; impl image::Renderer for IcedRenderer { - fn draw(&mut self, handle: image::Handle, layout: iced::Layout<'_>) -> Self::Output { + fn dimensions(&self, handle: image::Handle) -> (u32, u32) { + self.cache + .graphic_cache() + .get_graphic_dims((handle, Rotation::None)) + // TODO: don't unwrap + .unwrap() + } + + fn draw( + &mut self, + handle: image::Handle, + color: Rgba, + layout: iced::Layout<'_>, + ) -> Self::Output { ( Primitive::Image { - handle, + handle: (handle, Rotation::None), bounds: layout.bounds(), + color, }, MouseCursor::OutOfBounds, ) diff --git a/voxygen/src/ui/ice/renderer/row.rs b/voxygen/src/ui/ice/renderer/row.rs new file mode 100644 index 0000000000..47d22004f4 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/row.rs @@ -0,0 +1,34 @@ +use super::{IcedRenderer, Primitive}; +use iced::{row, Element, Layout, MouseCursor, Point}; + +impl row::Renderer for IcedRenderer { + fn draw( + &mut self, + defaults: &Self::Defaults, + children: &[Element<'_, M, Self>], + layout: Layout<'_>, + cursor_position: Point, + ) -> Self::Output { + let mut mouse_cursor = MouseCursor::OutOfBounds; + + ( + Primitive::Group { + primitives: children + .iter() + .zip(layout.children()) + .map(|(child, layout)| { + let (primitive, new_mouse_cursor) = + child.draw(self, defaults, layout, cursor_position); + + if new_mouse_cursor > mouse_cursor { + mouse_cursor = new_mouse_cursor; + } + + primitive + }) + .collect(), + }, + mouse_cursor, + ) + } +} diff --git a/voxygen/src/ui/ice/widget.rs b/voxygen/src/ui/ice/widget.rs index 14995d4228..7d9718b424 100644 --- a/voxygen/src/ui/ice/widget.rs +++ b/voxygen/src/ui/ice/widget.rs @@ -1 +1,2 @@ +pub mod background_container; pub mod image; diff --git a/voxygen/src/ui/ice/widget/background_container.rs b/voxygen/src/ui/ice/widget/background_container.rs new file mode 100644 index 0000000000..6bacbcdcc1 --- /dev/null +++ b/voxygen/src/ui/ice/widget/background_container.rs @@ -0,0 +1,169 @@ +use iced::{layout, Element, Hasher, Layout, Length, Point, Size, Widget}; +use std::u32; +use vek::Rgba; + +// Note: it might be more efficient to make this generic over the content type + +// Note: maybe we could just use the container styling for this + +/// This widget is displays a background image behind it's content +pub struct BackgroundContainer<'a, M, R: self::Renderer> { + width: Length, + height: Length, + max_width: u32, + max_height: u32, + background: super::image::Handle, + fix_aspect_ratio: bool, + content: Element<'a, M, R>, + color: Rgba, +} + +impl<'a, M, R> BackgroundContainer<'a, M, R> +where + R: self::Renderer, +{ + pub fn new(background: super::image::Handle, content: impl Into>) -> Self { + Self { + width: Length::Shrink, + height: Length::Shrink, + max_width: u32::MAX, + max_height: u32::MAX, + background, + fix_aspect_ratio: false, + content: content.into(), + color: Rgba::broadcast(255), + } + } + + pub fn width(mut self, width: Length) -> Self { + self.width = width; + self + } + + pub fn height(mut self, height: Length) -> Self { + self.height = height; + self + } + + pub fn max_width(mut self, max_width: u32) -> Self { + self.max_width = max_width; + self + } + + pub fn max_height(mut self, max_height: u32) -> Self { + self.max_height = max_height; + self + } + + pub fn fix_aspect_ratio(mut self) -> Self { + self.fix_aspect_ratio = true; + self + } + + pub fn color(mut self, color: Rgba) -> Self { + self.color = color; + self + } +} + +impl<'a, M, R> Widget for BackgroundContainer<'a, M, R> +where + R: self::Renderer, +{ + fn width(&self) -> Length { self.width } + + fn height(&self) -> Length { self.height } + + fn layout(&self, renderer: &R, limits: &layout::Limits) -> layout::Node { + let limits = limits + .loose() // why does iced's container do this? + .max_width(self.max_width) + .max_height(self.max_height) + .width(self.width) + .height(self.height); + + let (size, content) = if self.fix_aspect_ratio { + let (w, h) = renderer.dimensions(self.background); + // To fix the aspect ratio we have to have a separate layout from the content + // because we can't force the content to have a specific aspect ratio + let aspect_ratio = w as f32 / h as f32; + // To do this we need to figure out the max width/height of the limits + // and then adjust one down to meet the aspect ratio + let max_size = limits.max(); + let (max_width, max_height) = (max_size.width as f32, max_size.height as f32); + let max_aspect_ratio = max_width / max_height; + let limits = if max_aspect_ratio > aspect_ratio { + limits.max_width((max_height * aspect_ratio) as u32) + } else { + limits.max_height((max_width / aspect_ratio) as u32) + }; + // Get content size + // again, why is loose() used here? + let content = self.content.layout(renderer, &limits.loose()); + // This time we need to adjust up to meet the aspect ratio + // so that the container is larger than the contents + let content_size = content.size(); + let content_aspect_ratio = content_size.width as f32 / content_size.height as f32; + let size = if content_aspect_ratio > aspect_ratio { + Size::new(content_size.width, content_size.width / aspect_ratio) + } else { + Size::new(content_size.height * aspect_ratio, content_size.width) + }; + + (size, content) + } else { + // again, why is loose() used here? + let content = self.content.layout(renderer, &limits.loose()); + let size = limits.resolve(content.size()); + //self.content.layout(renderer, limits) + + (size, content) + }; + + layout::Node::with_children(size, vec![content]) + } + + fn draw( + &self, + renderer: &mut R, + defaults: &R::Defaults, + layout: Layout<'_>, + cursor_position: Point, + ) -> R::Output { + self::Renderer::draw( + renderer, + defaults, + layout, + cursor_position, + self.background, + self.color, + &self.content, + layout.children().next().unwrap(), + ) + } + + fn hash_layout(&self, state: &mut Hasher) { self.content.hash_layout(state); } +} + +pub trait Renderer: iced::Renderer + super::image::Renderer { + fn draw( + &mut self, + defaults: &Self::Defaults, + layout: Layout<'_>, + cursor_position: Point, + background: super::image::Handle, + color: Rgba, + content: &Element<'_, M, Self>, + content_layout: Layout<'_>, + ) -> Self::Output; +} + +// They got to live ¯\_(ツ)_/¯ +impl<'a, M: 'a, R: 'a> From> for Element<'a, M, R> +where + R: self::Renderer, +{ + fn from(background_container: BackgroundContainer<'a, M, R>) -> Element<'a, M, R> { + Element::new(background_container) + } +} diff --git a/voxygen/src/ui/ice/widget/image.rs b/voxygen/src/ui/ice/widget/image.rs index 15830839a8..b793399cd5 100644 --- a/voxygen/src/ui/ice/widget/image.rs +++ b/voxygen/src/ui/ice/widget/image.rs @@ -1,17 +1,20 @@ -use super::super::super::graphic::{self, Rotation}; -use iced::{layout, Element, Hasher, Layout, Length, Point, Size, Widget}; +use super::super::graphic; +use iced::{layout, Element, Hasher, Layout, Length, Point, Widget}; use std::hash::Hash; +use vek::Rgba; // TODO: consider iced's approach to images and caching image data // Also `Graphic` might be a better name for this is it wasn't already in use // elsewhere -pub type Handle = (graphic::Id, Rotation); +pub type Handle = graphic::Id; pub struct Image { handle: Handle, width: Length, height: Length, + fix_aspect_ratio: bool, + color: Rgba, } impl Image { @@ -22,6 +25,8 @@ impl Image { handle, width, height, + fix_aspect_ratio: false, + color: Rgba::broadcast(255), } } @@ -34,6 +39,16 @@ impl Image { self.height = height; self } + + pub fn fix_aspect_ratio(mut self) -> Self { + self.fix_aspect_ratio = true; + self + } + + pub fn color(mut self, color: Rgba) -> Self { + self.color = color; + self + } } impl Widget for Image @@ -44,9 +59,23 @@ where fn height(&self) -> Length { self.height } - fn layout(&self, _renderer: &R, limits: &layout::Limits) -> layout::Node { - // We don't care about aspect ratios here :p - let size = limits.width(self.width).height(self.height).max(); + fn layout(&self, renderer: &R, limits: &layout::Limits) -> layout::Node { + let mut size = limits.width(self.width).height(self.height).max(); + + if self.fix_aspect_ratio { + let aspect_ratio = { + let (w, h) = renderer.dimensions(self.handle); + w as f32 / h as f32 + }; + + let max_aspect_ratio = size.width / size.height; + + if max_aspect_ratio > aspect_ratio { + size.width = size.height * aspect_ratio; + } else { + size.height = size.width / aspect_ratio; + } + } layout::Node::new(size) } @@ -54,11 +83,11 @@ where fn draw( &self, renderer: &mut R, - _defaults: &R::Defaults, + defaults: &R::Defaults, layout: Layout<'_>, _cursor_position: Point, ) -> R::Output { - renderer.draw(self.handle, layout) + renderer.draw(self.handle, self.color, layout) } fn hash_layout(&self, state: &mut Hasher) { @@ -68,7 +97,8 @@ where } pub trait Renderer: iced::Renderer { - fn draw(&mut self, handle: Handle, layout: Layout<'_>) -> Self::Output; + fn dimensions(&self, handle: Handle) -> (u32, u32); + fn draw(&mut self, handle: Handle, color: Rgba, layout: Layout<'_>) -> Self::Output; } impl<'a, M, R> From for Element<'a, M, R> From 43139873d54720f3573a72ac9ff63b7279ddf338 Mon Sep 17 00:00:00 2001 From: Imbris Date: Thu, 16 Apr 2020 00:27:25 -0400 Subject: [PATCH 05/61] Add Stack and CompoundGraphic widgets --- assets/voxygen/element/frames/banner_top.png | Bin 4881 -> 5125 bytes voxygen/src/menu/main/ui.rs | 21 +- voxygen/src/ui/ice/renderer.rs | 67 ++++-- .../src/ui/ice/renderer/compound_graphic.rs | 35 ++++ voxygen/src/ui/ice/renderer/container.rs | 6 +- voxygen/src/ui/ice/renderer/stack.rs | 34 ++++ voxygen/src/ui/ice/widget.rs | 2 + .../src/ui/ice/widget/background_container.rs | 15 +- voxygen/src/ui/ice/widget/compound_graphic.rs | 190 ++++++++++++++++++ voxygen/src/ui/ice/widget/image.rs | 7 +- voxygen/src/ui/ice/widget/stack.rs | 93 +++++++++ voxygen/src/ui/mod.rs | 2 +- 12 files changed, 439 insertions(+), 33 deletions(-) create mode 100644 voxygen/src/ui/ice/renderer/compound_graphic.rs create mode 100644 voxygen/src/ui/ice/renderer/stack.rs create mode 100644 voxygen/src/ui/ice/widget/compound_graphic.rs create mode 100644 voxygen/src/ui/ice/widget/stack.rs diff --git a/assets/voxygen/element/frames/banner_top.png b/assets/voxygen/element/frames/banner_top.png index e75a275660511ca78201e543a94edcba06e9586c..742e767b0fe042fce58f8a0666ad959012c844cc 100644 GIT binary patch delta 5097 zcmVdU&h#$6g7@Z`=Fvp3<)Y z`Yp&mK0W^&{e1sC_+@?XsQ(np{*4hI{`3!7{yg|^h<};5{#;S{{SS91`@?p=f8Y1| zxc7W^w^}En{fnsd{D{sw$n{Og>-$*erF;vY^?glVBdug4$C+;PvOIGkEBDDQdtUkG z^S)a)me|TE%6lnBTH@NMT4piwlgSyce2dmuy$&*GQsFFc> z!#vpFO8Ya!mi!NF)i=+T`-JyTRpdwgI{+bKZecOGl7ZBzf!uS_QZ#4cI7rZ9<)yZ= z$v_q<<^zwUtczr|Cg4Zg07FZve>4&4qB0(t zvU=^UOYdEKklfO$O+~w^4xPG29c{?yL&q3%tXXD-17-H9bIdu{D$in~#aN5`7gu)K zb+;|MZ{1_hy$twRGqG-J!=^1SULBh7?&`y*ualnxQ=E3>^rL5-dDaDMx88Q;_N#Z? zdDqXZeY5(9*Z+{U{LNZCdcj^-=vXL=HNU$QcO7-0oGd_(8ZK^F6ptI|&d3umluj7iPwbwq`oX`=8f7ZRVar+j%H?7W>THnis1x2EaxJ@Y0PC9NZTCy<@Tdy|@pwKaqggdlY6fFa%X1*tY z8HAlo9b!#b&OQ7qIz{Y^$&#lf0^@{NccOBKduh9$vsQhIqH=AR%4%abap_Feh0n&G zOVvEt(#V6d_m+)Zf8JAz>oszr$(j9ZeO5QNkeZikFl=WZ8$MXT8oHe}K_lj}}7BZk&Cdao5Nfpb&_|tr3J4s!Y@>GVhFCoB^9}V|H7oR%cw5vuF2H(j8zhpH5+ze@~cF&xP}duRlb9><_(6 zkfm(}I%~Xu?eXE%MpFA8)2Wjc_}Q)YlfVNoDkvDPV@-mszR#^BG|$L%Ei9ppQ}+hp zd$Izf_9$QN2W=WG$Zsv{v_u38v!D$yq~_8W+nYJ}c~kY1c(!)#t%_VsbnvvBse^{{ z5}phNz~NZNe`2z$@jmpa_pVw<^p-T4h0LfBnli>Z=2o|jwWG$|+D!EfriGql=NUQf z?tb0uInLZxGid_y$c5Ik;0=mPS8gY@UPSHm@C`?N_hJ3yTOmJPz8%4iOg@ay45URf zPs$|6`3O03A2Z*MZf@vGDOj8ibcY!r^@Y;tc7;7Be;(=n=stzm^fNJnJc@PD8?Xx} zFiQj6gb9d_1G%V4rD}LchGxhCB7~*d@P}cGZ^W)cc%$ddot=XU`f1e;om>EUCXtYBRkOvgrf#cCplkdG6DU!J?{HsmEIm8VuAb@M&pn=CD%UT3( zCfMWTs8tN_E*=RNMG6NH*vYdFw{aPId4q` z<;BV~Bukd9!{40URNK5=o}H+OH^SXW!pJo$Xem#^ddO@E z#Q^yGEYJ^wu){=d@809&?L+`xp{m;g*-blzq~I#+4PfVRF?TM#4kbc7cUW<}gWX^r zIetzdVbl%_SZ7d%>4D+LNu$bgnyizFfAckj_%0v{zHsrU_ZE%>Y8(cj6S#U2Zbg|` z_k`Ubwtf9{h3jGXbgqO0v)v%_iru%UNI=gsG=>)jUqW9o?m$De!8~L6Jo}N-7^0~$ zA2VNf1MBn7OeqhkNDs<(+nNYOOFP5M192KK=31!V=!kad&!AwzPmdk}X$p&Qe{e#O zVX-PzuVCU1MQbtuyrb&`B2!>k4DFuSKFSa+(--MM&_#B-Ok3_*0 zxLHUo^7g>va0Ge85~Si?k*bEJe<|H}IkaZjs&GUIC1OUbHk69Vg4JREMeQ#}!uelT z4}D=9d))%K(gcSBy9!l=!zrmpwy_sgc7m0{G~irrWlYZ)mY{BZ9NULsY3>PvgoZGg zq!1rV4N(J>5it%*5A{s@4FOIINVH{0!Oj_+<7zi+womVg>#H@0m6=xve-C7eTu_XS z4I6S^dRrwNy-E+cPRu``yI!AQX2W4Z5Y`vaI4JePI)M*l1YCicgMpT^HmgSIh*_@e zCw#MH94TOt2mXYo#tl!=2;exo2II2dO!N&qN8Pq?m00(%@{kt2ZMqOZVG({QzNF9y z@kNotL<@2@&HJ4mOf}MiWC31!*{!Av;VGnSa<2 zzia|F0=rHlq>~02pNR9P!pw1*A=gZVmrB?&v}53bSUGy+g#d(#;HTaU*3jq*pH)by zWg4JHY#<5zCf4Kpwoni%|6a!h5IOCf6IS0}VW9?cd4YbC|6mp$n%ZN6y>>C0bcx8IC zNmuiBc=*5Y>_Be@-X>X#EyJP%6~XD!_7fctK+{i_3Db&Jm)anv6jO=lkc%+o!4Oav zUf2zvjiq&5IpWpB;o;)t0J0!xTEa`E2ayQfA4wvtf2;$o@D8FSyx*Xe)A=wcCBY-; z3p|h6#i)lRp*PkL1QAk+hLcp4cqbC_zCJIO5X!pPg5akFN z;Sbm?e~PH$ndH#-%$|bu zx|F@5g*$|u$LYdp?yTjYZ)fhc$3x#9r!Ocm2*KJ;8#q-wMz~{K7(_yI43~SHHz*Za zf8$Jm=QYu9qWsi2RQ_m=R)LNcQxT4kP|q+L{N5Jbt^@#R3*Nm`Jy^dgoy}-D434iP zqK9Aua`H^ajHwNW5vNfoK28+`IPfkkEN@nNW3GBw!Z%8dm$fZBY<3hMd+~f1(v{vasWnB>wZUIJKti6Ar#m+Yp-E?a{%8 zk;Kz$r+%H921$zMt!TX_u?koI@+wVCdeDhD1$x~-K5x->gz`|1AU81}2t%(!CIMqZ z5J1zwII}tc8RuXfHO&tk=X4`rgcNz0Kz)%-aN}w0u?DQEQm6`SuITl5UvV0xf8nr0 zW@(6@=-)~lLepUq4nZsVr(rT~rhX(2B7qCaLnVMKM0x6@QI+_wo87g+F+qO57`V)AE(#{f3JXf&oR(~)PqJXj_FdEbHaO5+0q4H;Ng{Ki&Gdh zAjl&~yp*?q0C&*Y)u!t~EolQ`96B5_O9PC+U|YcJ)YanT$nh~O^U{Psz|4Wq#l1w2 z$+pkFXp=r@GWtX>#@Z)_35VIq0HkO9aa zQ$wuJ3SI$IaZiZTJNBRh2Q-A(aYP+^hV~?MqdQgIh;A_0^LO>>epG9(0kCBZxhH>NsGzSbbDd@sDJ)_M5=1Ddqk<}I#A($@v5=wtq=$c` z?U%@R0(7J=IfsjPqGuhKSgxM zT@uT-BtJjNJ~otIhCq86p|fNuBmLC+bK;atHva*TbJG_5obReXh10$s`T2Ug7mvOp zBBIOjFzRpIkAM5fHqKNdS|jL?s8G7l#+< z(RcLdyQlWkz8>>k1wn!qF!YY*4e5qJ!ejAxa$%(}5&DA#64^)EjmH0`Y$s6N+aei5 zzn`17V6gCa#1;%3*0**W+Ku_Hda*&yuK|Qu!rKIozJH^0(-xQGVLa{YQTh_nM~tiO z)f*|%x|I?rgO?J};##zbXgoy0ydVfH2qY5B2ACsr?Rpp8^sTKklddAb%`fw1zS@ zZir}*#($^WwJ}tV(g_MAw`pBn7o$Y)*n zqJr7-g&AS4ZI|O=^v3pftHmr!0||=XR@Vp~9le`+G-aZoSm}m!ty%_s*UsjTX-MBog>u@I3ynf8OUG{Hvps zr*)ZQjgk1*R$E1RGSk<;ufI99_xIO}_IZ|nJ^qyPIjL}{^0$7y9R4{`Io^MqAnkK} z|9bq9^SRFaTeJu?4XyXX7U--?&-pKF)&^NIRvDV{Vw9~i&Y z=yN>xKiBT9(9bK%-=3CV-$*xE`uX2awY%5mz5ca3ZG%dU=d-Ewxy1X74{l0Xf8Wyv z58ceczYOXgykGyd{;{&etvd3?Xny#d{vqnNZl ze>M0e5X*?p#cw{0p1$JlUY8{dI}m^ABv*x6hT&_0*p9 z1@|oWxd0{N-krtdN(a)&wdawOlp;A($4P&bl^0;Y=|JwiSbi?OsR)XcS>oQDBl-Po z>E-J%(n2G89gt+wA*ETZb?_r?f+6j-x2N|$y7bvs*HK3seT*R^G)kpPf0HglM5Zj! zS!bJljwy4_HFcF$k+8nTl4s4emMS+@F0Gucd_r}XU3c4kk1c!dwe(@sC*$eCvy zwPMwvzYUYerY%#?4wSrk_u=Bxm+J*+1KeeY*N<*Ya1!e;q3=({YIo0O%4uOEb4tqI1!+v_h_Wp=-HEb1X;4=wS0) z`pw^a_oH)v+qb8;|Eh20KXvYj*8M*^_eASHo%?Iw{?Ikk*HP-*pkQItG}Zfr6vizZ zH#|#@vH)2_R?W*ot45u-pQKy2=VOu&g=pPcn|JIk_a?nfDYlq`f6ci(FLvNusE@7I zSx^~^3vu(aUU_7#{9G8I>MdWc7Z0 zif&scQ5U1QvJ=FeW~DuQxU1G8_*a`py0*Dlc^@*@=u_I^)KVocb~N*Z^@b>vy#dMc zeZ985=i{fC^lepXf6*LcQE)!Xo%&g+gQ}AeWtM5Tv2WeU*-!N?6Vy&N)CYS0v{?t8 z2U}>6BaZ+>itb5&*q~VTTaQNtwtBs0mUpL3z(ROQdb!S`pH|(JI^2=>y1SQjcFW%O zxO8@L1%AyYE}caT^FoP1VD?dNIyN`-@|(HP0vr^8-cOOvf5v-T!cug;LX(D8neKTE zK*@HDW+o^wX-#q@RlK+QOif5-^Si}-=*75BHWgZgkem*tx=1dydK)iYpJjn)qcfI@Hc#PaL7oMSBwE-VYxh6PQ=RYz7<&zrKP(-$kMmUiJ3 zr$Pr1a%DELe};|X&TdekPJP+-%EUxmf6rBbaWdq9UK1cG=F@ENefL_nyAvk?1O29? z)5V$|xw~ED=cIY);@%bi=u+HvLsimUJJoFi(z1h`tWalb3tBBdxt!ae0=Ki_2UrGI zi_Mj>eb9wpf!^%a%LVP=_<=FWgMQSyu#>X1_pNh*e>V7X1bJY2^?+WYz(y0BgwJkj)Y#Q7ShE9M&uW*r(=c2+~+b+~Bz=ld(-DB#Oo*ROV zIS|AtDq=EFdFlaVM}vuq`roY&Acild>d`%9fG#YZz*?KHP3cNlRI@oySihmD?b7So zhOu@UyFHElR0vBC^1Nr4Npe`!pjMnoB2y0}e*+QE?3B3SijJb9kZpqn7CJ7X*a?OY zsWdzWvpVVA+Be&qxeid82q9-@Bz!h3ff}82Za7>r6anhlj0#$JLIgpLis`zivKka- zR7@_(<~5Os>&w-Ijj6bTlZpEHPJrPhD8&llyKpU1CQV+Kmy7OV@dmOhCXo<-yt!j1 ze+PZWLjX3QPXto1c3MM(v8##{Kd&9!uzfadjetASs%);3AXGTFr;Gk)E=7~zA6kMS zpk9ep-*^X5>O(U#9v@@QoE}NI`WaLPvSMVfv{0D$uiZt33<)Uk30P*ov$IKj?pf4f zj7}P${InSDhTaKGV=|~g{aP>nfS_N}e>|i|E1=z+hf3G7=Be2q+3~@&V3N5vm4KmL z6d&71!+}+YFm9vj;EaV-;V^OzO{)gWDH0TVI#b5HlOaJGYn^HV@&Z~%h%Qz!Aay?w za^00<~&zBR03I-N<6Wi3GtCWKz@|~oLXA}C|kk$QmGw-CmaOX4e?W-e>L=Q z1}78L1r+uOV8$J6=ev>}#g2$&oz!(pETRIhz0N44Cxny>u;`3XwwA~mPF)FKmL3Su zP%#V909Md=hp-=}ms1;lUvdI37(T}icv$7KRqlbLMPjpI6cxq=PCa3MuLF9c;kit zK&&C83~#baC)pfOdkhO>e~afnJrgHbF$RxmLQWi0;`U)6p=c4ZjR^0^k~i22%^Q15 zgaEI5)J0@~;T(Rwj_T+p`>FV=SB&1SY>5WUZZ{6p|kR{Y*=S9xku z2!km|Py$lHTqE$84IG(nJWMZQ9}ox7^J*v-IK5!F8-a>xA4zBKe*<@07$~bu%bH+s_ZxYL4H_(q+>%kczq1R5Ml*JUNk10i7BEY7HY(Y54GD- z;DcDG&2F_bNAFE()6Q~)2zSPK^o{6<1>_cL!AxD=BMB`D|5GMSW`T&72rqXhUuGS= zz*f>Zok+>OpU~iZe~=^mSeWy0LV(5G473&;Cc?`5)j{NG=I}QjqUaYxO3EY`36xm; zL@tsr6~#wEJmd#!i`gr;fky!+(k37z&#)+9wIf$CK>#|g#m!)s+e6AGOA;Fy2vb2o zl;$JM=aO_P2=3k)TvjCWxC?r+$h-&}YKGv{0kmeZ=Rnire@qDUxxskiq2g~+NIQaR zOcVNBF>sQNHB7)DCV^X+jmfsbsr7!zQpuUw7>!5SES{QNFAa0zc6%I#HI$%5cp!EY z=oW=W_VMo>*27G>iN*5)u~ayLmj(f1utV6HBB<=ZHqobAss{W$+C#vmad55sS;HYf zs3#Nh1A+;ve}pYO3N2*B0&G+;$2g1ZD2b0~MogzeSgVhU*_-1748an^un-peL+4l{ zJEE#(HS7bWR8L@v(XlfCdj-6s8nd>MSqXNmgi>&u%vgZQVLI}R+oIMQ12K1B1W{qg zWxCt-(AAHaTOu zWeEgmR77Ytl_bgc7$ca(F7h3zg?x17b!u22nX;0!nB>t{2kC5!UI9GDMg}^i;zT&? zrH7G3`U%O-9Wp!|cGT%)y0SMh4mHAqDIHA&y$;txJ&@!E%XyP2=2&4~WAY?Zfnaxh z5m_3G(i2-8|6uyW@Z>BK;dp6_KS~95%mNGBfl&0bbTpfExWn0iL&qQyOZ)(`YfMIK zc4fQ|h_wF7k`$#G^tGRP5b ze~<5;0V5oH@kbbo?}MnCdy2f8EAZuJAd7%b6N!i`fZuv}gFG=)#CKd85MZLoOllh6 z-b$EkZ%%%stIXWEpV7({dcW}%9f{MK;15aS?t%ICulD1*f6VI< z@|Z9fkqOQuGaMd!Vvbkx5k)H%1hEV6y_c%#b1Ml6FSp2aQ7&2(GPXnL zgI%}}o*n;%vxk&uFqs`^^UIFmT3r_u&c{%AavlP!%oc13M3X3>K0w3)Nmql45{V`S zYLN}eyanDD&;nX7VS|;wBZNdaf6GyD#25>?qbGEY{-XyLUz;LOhg8af2^x&P+=Jio z{{z^RGQ5_De1P)P_Q<3Segfzj&VleTYA~!1SR4ZRc)Vv$Rs2zii^G^U9cu*`BZA!!!2U9b*%*8$Eu zb3OkC-(zvsF#C$Plj{&Me**;sISSY(wXOgF15!yuK~#9!?VZ7q6fq1%J=6xl&ItnG zNE+Nog8&F%10W4KXynuTch5)1f4_YH@z`2xzkmJw z=lIjd56{YVV|`;iRyno2b)Oz;KlJu>tai>a_NO|Y6ZX%Mc@?ey%KdDmSWqYcP{zZq zH`YDr=wU5SJbGHIeE-`z?}?TWTDtBAVYt^ic!S785QD2A)b#Op70 zUZGF`00f1yMK;6pe*o6`?t!{z@svt5wldm}zI&_h?%i`nJz=-6Ditr*kyVzKlJFGb ztobX?6o%trl3kE<#bRhjIwpS+6f3_bfm8?e`mCa~(l~SeMv{@zM-IuqHCaWZ7ulMFvR+${ly`#SMVnx~UiQ2W=Kh|f(o=_+NP{yf0SuD@8 zYb+p3G@BFkaAP^ouD4FZDsQ!?x_-~|S@o-St@)XaZB;3gUGbh1C=>uFjiEG@Nu4sW zVlZB|$g4LSe}k#Yer3t7qp{xo)$+H>=Io9)|4=9Zuuh?5vk=u^!B%8>u4Q+k3L%=! z$to4~dQU-G(fQ8m^(4`6T&myk)s?lXGO?76H*cc%tgai0_Mc}rYj5m76bb;8@i)s; ztt7HaMO`zJRY2;y@v3aj>g%gKyyw}hEK*eQWwTW3e|T8!Q~NcWEmE^xw^kSVJ;)iIEJC93$M*`lg>ziM1(?T-C7_5%t90P15X z>bvK?FnVxxpO4wEhSeRtW>=EV+E*3I?4GjTj&a$QwzK8E`-|==?2SWj=Po<8S$TG=?u#A4AzC zmI45E@U<}jyciGGyEy=W2P, }, + Rectangle { + bounds: iced::Rectangle, + color: Rgba, + }, } // Optimization idea inspired by what I think iced wgpu renderer may be doing @@ -265,6 +271,10 @@ impl IcedRenderer { } let color = srgba_to_linear(color.map(|e| e as f32 / 255.0)); + // Don't draw a transparent image. + if color[3] == 0.0 { + return; + } let resolution = Vec2::new( (gl_aabr.size().w * self.half_res.x).round() as u16, @@ -311,30 +321,47 @@ impl IcedRenderer { None => return, }; - match self.current_state { - // Switch to the image state if we are not in it already. - State::Plain => { - self.draw_commands - .push(DrawCommand::plain(self.start..self.mesh.vertices().len())); - self.start = self.mesh.vertices().len(); - self.current_state = State::Image(tex_id); - }, - // If the image is cached in a different texture switch to the new one - State::Image(id) => { - if id != tex_id { - self.draw_commands.push(DrawCommand::image( - self.start..self.mesh.vertices().len(), - id, - )); - self.start = self.mesh.vertices().len(); - self.current_state = State::Image(tex_id); - } - }, - } + // Switch to the image state if we are not in it already or if a different + // texture id was being used. + self.switch_state(State::Image(tex_id)); self.mesh .push_quad(create_ui_quad(gl_aabr, uv_aabr, color, UiMode::Image)); }, + Primitive::Rectangle { bounds, color } => { + let color = srgba_to_linear(color.map(|e| e as f32 / 255.0)); + // Don't draw a transparent rectangle. + if color[3] == 0.0 { + return; + } + + self.switch_state(State::Plain); + + self.mesh.push_quad(create_ui_quad( + self.gl_aabr(bounds), + Aabr { + min: Vec2::zero(), + max: Vec2::zero(), + }, + color, + UiMode::Geometry, + )); + }, + } + } + + // Switches to the specified state if not already in it + // If switch occurs current state is converted into a draw command + fn switch_state(&mut self, state: State) { + if self.current_state != state { + let vert_range = self.start..self.mesh.vertices().len(); + let draw_command = match self.current_state { + State::Plain => DrawCommand::plain(vert_range), + State::Image(id) => DrawCommand::image(vert_range, id), + }; + self.draw_commands.push(draw_command); + self.start = self.mesh.vertices().len(); + self.current_state = state; } } diff --git a/voxygen/src/ui/ice/renderer/compound_graphic.rs b/voxygen/src/ui/ice/renderer/compound_graphic.rs new file mode 100644 index 0000000000..d6bb89e6c0 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/compound_graphic.rs @@ -0,0 +1,35 @@ +use super::{ + super::{widget::compound_graphic, Rotation}, + IcedRenderer, Primitive, +}; +use compound_graphic::GraphicKind; +use iced::{MouseCursor, Rectangle}; +use vek::Rgba; + +impl compound_graphic::Renderer for IcedRenderer { + fn draw( + &mut self, + graphics: I, + //color: Rgba, + _layout: iced::Layout<'_>, + ) -> Self::Output + where + I: Iterator, + { + ( + Primitive::Group { + primitives: graphics + .map(|(bounds, kind)| match kind { + GraphicKind::Image(handle) => Primitive::Image { + handle: (handle, Rotation::None), + bounds, + color: Rgba::broadcast(255), + }, + GraphicKind::Color(color) => Primitive::Rectangle { bounds, color }, + }) + .collect(), + }, + MouseCursor::OutOfBounds, + ) + } +} diff --git a/voxygen/src/ui/ice/renderer/container.rs b/voxygen/src/ui/ice/renderer/container.rs index 4a4eaad9ab..fd144f66ce 100644 --- a/voxygen/src/ui/ice/renderer/container.rs +++ b/voxygen/src/ui/ice/renderer/container.rs @@ -1,5 +1,5 @@ -use super::{IcedRenderer, Primitive}; -use iced::{container, Element, Layout, MouseCursor, Point, Rectangle}; +use super::IcedRenderer; +use iced::{container, Element, Layout, Point, Rectangle}; impl container::Renderer for IcedRenderer { type Style = (); @@ -7,7 +7,7 @@ impl container::Renderer for IcedRenderer { fn draw( &mut self, defaults: &Self::Defaults, - bounds: Rectangle, + _bounds: Rectangle, cursor_position: Point, _style_sheet: &Self::Style, content: &Element<'_, M, Self>, diff --git a/voxygen/src/ui/ice/renderer/stack.rs b/voxygen/src/ui/ice/renderer/stack.rs new file mode 100644 index 0000000000..3eb77d5ad7 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/stack.rs @@ -0,0 +1,34 @@ +use super::{super::widget::stack, IcedRenderer, Primitive}; +use iced::{Element, Layout, MouseCursor, Point}; + +impl stack::Renderer for IcedRenderer { + fn draw( + &mut self, + defaults: &Self::Defaults, + children: &[Element<'_, M, Self>], + layout: Layout<'_>, + cursor_position: Point, + ) -> Self::Output { + let mut mouse_cursor = MouseCursor::OutOfBounds; + + ( + Primitive::Group { + primitives: children + .iter() + .zip(layout.children()) + .map(|(child, layout)| { + let (primitive, new_mouse_cursor) = + child.draw(self, defaults, layout, cursor_position); + + if new_mouse_cursor > mouse_cursor { + mouse_cursor = new_mouse_cursor; + } + + primitive + }) + .collect(), + }, + mouse_cursor, + ) + } +} diff --git a/voxygen/src/ui/ice/widget.rs b/voxygen/src/ui/ice/widget.rs index 7d9718b424..ebf8ee659b 100644 --- a/voxygen/src/ui/ice/widget.rs +++ b/voxygen/src/ui/ice/widget.rs @@ -1,2 +1,4 @@ pub mod background_container; +pub mod compound_graphic; pub mod image; +pub mod stack; diff --git a/voxygen/src/ui/ice/widget/background_container.rs b/voxygen/src/ui/ice/widget/background_container.rs index 6bacbcdcc1..34db33c629 100644 --- a/voxygen/src/ui/ice/widget/background_container.rs +++ b/voxygen/src/ui/ice/widget/background_container.rs @@ -1,5 +1,5 @@ use iced::{layout, Element, Hasher, Layout, Length, Point, Size, Widget}; -use std::u32; +use std::{hash::Hash, u32}; use vek::Rgba; // Note: it might be more efficient to make this generic over the content type @@ -142,7 +142,18 @@ where ) } - fn hash_layout(&self, state: &mut Hasher) { self.content.hash_layout(state); } + fn hash_layout(&self, state: &mut Hasher) { + struct Marker; + std::any::TypeId::of::().hash(state); + + self.width.hash(state); + self.height.hash(state); + self.max_width.hash(state); + self.max_height.hash(state); + self.fix_aspect_ratio.hash(state); + + self.content.hash_layout(state); + } } pub trait Renderer: iced::Renderer + super::image::Renderer { diff --git a/voxygen/src/ui/ice/widget/compound_graphic.rs b/voxygen/src/ui/ice/widget/compound_graphic.rs new file mode 100644 index 0000000000..48b6f759d4 --- /dev/null +++ b/voxygen/src/ui/ice/widget/compound_graphic.rs @@ -0,0 +1,190 @@ +use super::image::Handle; +use iced::{layout, Element, Hasher, Layout, Length, Point, Rectangle, Widget}; +use std::hash::Hash; +use vek::{Aabr, Rgba, Vec2}; + +// TODO: this widget combines multiple images in precise ways, they may or may +// nor overlap and it would be helpful for optimising the renderer by telling it +// if there is no overlap (i.e. draw calls can be reordered freely), we don't +// need to do this yet since the renderer isn't that advanced + +// TODO: design trait to interface with background container +#[derive(Copy, Clone)] +pub enum GraphicKind { + // TODO: if there is a use case, allow coloring individual images + Image(Handle), + Color(Rgba), +} + +// TODO: consider faculties for composing compound graphics (if a use case pops +// up) +pub struct Graphic { + aabr: Aabr, + kind: GraphicKind, +} + +impl Graphic { + fn new(kind: GraphicKind, size: [u16; 2], offset: [u16; 2]) -> Self { + let size = Vec2::from(size); + let offset = Vec2::from(offset); + Self { + aabr: Aabr { + min: offset, + max: offset + size, + }, + kind, + } + } + + pub fn image(handle: Handle, size: [u16; 2], offset: [u16; 2]) -> Self { + Self::new(GraphicKind::Image(handle), size, offset) + } + + pub fn rect(color: Rgba, size: [u16; 2], offset: [u16; 2]) -> Self { + Self::new(GraphicKind::Color(color), size, offset) + } +} + +pub struct CompoundGraphic { + graphics: Vec, + // move into option inside fix_aspect_ratio? + graphics_size: [u16; 2], + width: Length, + height: Length, + fix_aspect_ratio: bool, + /* TODO: allow coloring the widget as a whole (if there is a use case) + *color: Rgba, */ +} + +impl CompoundGraphic { + pub fn with_graphics(graphics: Vec) -> Self { + let width = Length::Fill; + let height = Length::Fill; + let graphics_size = graphics + .iter() + .fold(Vec2::zero(), |size, graphic| { + Vec2::max(size, graphic.aabr.max) + }) + .into_array(); + Self { + graphics, + graphics_size, + width, + height, + fix_aspect_ratio: false, + //color: Rgba::broadcast(255), + } + } + + pub fn width(mut self, width: Length) -> Self { + self.width = width; + self + } + + pub fn height(mut self, height: Length) -> Self { + self.height = height; + self + } + + pub fn fix_aspect_ratio(mut self) -> Self { + self.fix_aspect_ratio = true; + self + } + + //pub fn color(mut self, color: Rgba) -> Self { + // self.color = color; + // self + //} +} + +impl Widget for CompoundGraphic +where + R: self::Renderer, +{ + fn width(&self) -> Length { self.width } + + fn height(&self) -> Length { self.height } + + fn layout(&self, _renderer: &R, limits: &layout::Limits) -> layout::Node { + let mut size = limits.width(self.width).height(self.height).max(); + + if self.fix_aspect_ratio { + let aspect_ratio = { + let [w, h] = self.graphics_size; + w as f32 / h as f32 + }; + + let max_aspect_ratio = size.width / size.height; + + if max_aspect_ratio > aspect_ratio { + size.width = size.height * aspect_ratio; + } else { + size.height = size.width / aspect_ratio; + } + } + + layout::Node::new(size) + } + + fn draw( + &self, + renderer: &mut R, + _defaults: &R::Defaults, + layout: Layout<'_>, + _cursor_position: Point, + ) -> R::Output { + let [pixel_w, pixel_h] = self.graphics_size; + let bounds = layout.bounds(); + let scale = Vec2::new( + pixel_w as f32 / bounds.width, + pixel_h as f32 / bounds.height, + ); + let graphics = self.graphics.iter().map(|graphic| { + let bounds = { + let Aabr { min, max } = graphic.aabr.map(|e| e as f32); + let min = min * scale; + let size = max * scale - min; + Rectangle { + x: min.x + bounds.x, + y: min.y + bounds.y, + width: size.x, + height: size.y, + } + }; + (bounds, graphic.kind) + }); + + renderer.draw(graphics, /* self.color, */ layout) + } + + fn hash_layout(&self, state: &mut Hasher) { + struct Marker; + std::any::TypeId::of::().hash(state); + + self.width.hash(state); + self.height.hash(state); + if self.fix_aspect_ratio { + self.graphics_size.hash(state); + } + } +} + +pub trait Renderer: iced::Renderer { + fn draw( + &mut self, + graphics: I, + //color: Rgba, + layout: Layout<'_>, + ) -> Self::Output + where + I: Iterator; +} + +impl<'a, M, R> From for Element<'a, M, R> +where + R: self::Renderer, +{ + fn from(compound_graphic: CompoundGraphic) -> Element<'a, M, R> { + Element::new(compound_graphic) + } +} diff --git a/voxygen/src/ui/ice/widget/image.rs b/voxygen/src/ui/ice/widget/image.rs index b793399cd5..a5f5df9f11 100644 --- a/voxygen/src/ui/ice/widget/image.rs +++ b/voxygen/src/ui/ice/widget/image.rs @@ -83,7 +83,7 @@ where fn draw( &self, renderer: &mut R, - defaults: &R::Defaults, + _defaults: &R::Defaults, layout: Layout<'_>, _cursor_position: Point, ) -> R::Output { @@ -91,8 +91,13 @@ where } fn hash_layout(&self, state: &mut Hasher) { + struct Marker; + std::any::TypeId::of::().hash(state); + self.width.hash(state); self.height.hash(state); + self.fix_aspect_ratio.hash(state); + // TODO: also depends on dims but we have no way to access } } diff --git a/voxygen/src/ui/ice/widget/stack.rs b/voxygen/src/ui/ice/widget/stack.rs new file mode 100644 index 0000000000..dd215dfae1 --- /dev/null +++ b/voxygen/src/ui/ice/widget/stack.rs @@ -0,0 +1,93 @@ +use iced::{layout, Element, Hasher, Layout, Length, Point, Size, Widget}; +use std::hash::Hash; + +/// Stack up some widgets +pub struct Stack<'a, M, R> { + // Add these if it is useful + /* + padding: u16, + width: Length, + height: Length, + max_width: u32, + max_height: u32, + horizontal_alignment: Align, + vertical_alignment: Align + align_items: Align, + */ + children: Vec>, +} + +impl<'a, M, R> Stack<'a, M, R> +where + R: self::Renderer, +{ + pub fn with_children(children: Vec>) -> Self { Self { children } } +} + +impl<'a, M, R> Widget for Stack<'a, M, R> +where + R: self::Renderer, +{ + fn width(&self) -> Length { Length::Fill } + + fn height(&self) -> Length { Length::Fill } + + fn layout(&self, renderer: &R, limits: &layout::Limits) -> layout::Node { + let limits = limits.width(Length::Fill).height(Length::Fill); + + let loosed_limits = limits.loose(); + + let (max_size, nodes) = self.children.iter().fold( + (Size::ZERO, Vec::with_capacity(self.children.len())), + |(mut max_size, mut nodes), child| { + let node = child.layout(renderer, &loosed_limits); + let size = node.size(); + nodes.push(node); + max_size.width = max_size.width.max(size.width); + max_size.height = max_size.height.max(size.height); + (max_size, nodes) + }, + ); + + let size = limits.resolve(max_size); + + layout::Node::with_children(size, nodes) + } + + fn draw( + &self, + renderer: &mut R, + defaults: &R::Defaults, + layout: Layout<'_>, + cursor_position: Point, + ) -> R::Output { + renderer.draw(defaults, &self.children, layout, cursor_position) + } + + fn hash_layout(&self, state: &mut Hasher) { + struct Marker; + std::any::TypeId::of::().hash(state); + + self.children + .iter() + .for_each(|child| child.hash_layout(state)); + } +} + +pub trait Renderer: iced::Renderer { + fn draw( + &mut self, + defaults: &Self::Defaults, + children: &[Element<'_, M, Self>], + layout: Layout<'_>, + cursor_position: Point, + ) -> Self::Output; +} + +impl<'a, M, R> From> for Element<'a, M, R> +where + R: 'a + self::Renderer, + M: 'a, +{ + fn from(stack: Stack<'a, M, R>) -> Element<'a, M, R> { Element::new(stack) } +} diff --git a/voxygen/src/ui/mod.rs b/voxygen/src/ui/mod.rs index e5140145f9..158bcd31d4 100644 --- a/voxygen/src/ui/mod.rs +++ b/voxygen/src/ui/mod.rs @@ -45,7 +45,7 @@ use conrod_core::{ Rect, Scalar, UiBuilder, UiCell, }; use core::{convert::TryInto, f32, f64, ops::Range}; -use graphic::{Rotation, TexId}; +use graphic::TexId; use hashbrown::hash_map::Entry; use std::{ fs::File, From aeff9101209617e139b866487e29b40528ea193f Mon Sep 17 00:00:00 2001 From: Imbris Date: Thu, 16 Apr 2020 03:26:06 -0400 Subject: [PATCH 06/61] Revamp BackgroundContainer with scale specific padding, and use in main menu attempt --- ...anner_bottom.png => banner_bottom_png.png} | Bin .../frames/{banner.png => banner_png.png} | Bin voxygen/src/menu/main/ui.rs | 43 ++- voxygen/src/ui/ice/mod.rs | 6 +- .../ui/ice/renderer/background_container.rs | 20 +- .../src/ui/ice/renderer/compound_graphic.rs | 4 +- .../src/ui/ice/widget/background_container.rs | 265 ++++++++++++++---- voxygen/src/ui/ice/widget/compound_graphic.rs | 98 +++++-- voxygen/src/ui/ice/widget/image.rs | 26 ++ 9 files changed, 350 insertions(+), 112 deletions(-) rename assets/voxygen/element/frames/{banner_bottom.png => banner_bottom_png.png} (100%) rename assets/voxygen/element/frames/{banner.png => banner_png.png} (100%) diff --git a/assets/voxygen/element/frames/banner_bottom.png b/assets/voxygen/element/frames/banner_bottom_png.png similarity index 100% rename from assets/voxygen/element/frames/banner_bottom.png rename to assets/voxygen/element/frames/banner_bottom_png.png diff --git a/assets/voxygen/element/frames/banner.png b/assets/voxygen/element/frames/banner_png.png similarity index 100% rename from assets/voxygen/element/frames/banner.png rename to assets/voxygen/element/frames/banner_png.png diff --git a/voxygen/src/menu/main/ui.rs b/voxygen/src/menu/main/ui.rs index 185d4006dc..2c6d02b128 100644 --- a/voxygen/src/menu/main/ui.rs +++ b/voxygen/src/menu/main/ui.rs @@ -130,12 +130,12 @@ image_ids_ice! { v_logo: "voxygen.element.v_logo", info_frame: "voxygen.element.frames.info_frame_2", - banner: "voxygen.element.frames.banner", - - banner_bottom: "voxygen.element.frames.banner_bottom", + //banner: "voxygen.element.frames.banner", bg: "voxygen.background.bg_main", + banner: "voxygen.element.frames.banner_png", + banner_bottom: "voxygen.element.frames.banner_bottom_png", banner_top: "voxygen.element.frames.banner_top", button: "voxygen.element.buttons.button", button_hover: "voxygen.element.buttons.button_hover", @@ -194,7 +194,10 @@ pub type Message = Event; impl IcedState { pub fn view(&mut self) -> Element { use iced::{Align, Column, Container, Length, Row}; - use ui::ice::Image; + use ui::ice::{ + compound_graphic::{CompoundGraphic, Graphic}, + BackgroundContainer, Image, Padding, + }; use vek::*; let buttons = Column::with_children(vec![ @@ -214,14 +217,23 @@ impl IcedState { .padding(20); let banner_content = - Column::with_children(vec![Image::new(self.imgs.v_logo).fix_aspect_ratio().into()]) - .padding(15); + Column::with_children(vec![Image::new(self.imgs.v_logo).fix_aspect_ratio().into()]); + //.padding(15); - let banner = ui::ice::BackgroundContainer::new(self.imgs.banner, banner_content) - .color(Rgba::new(255, 255, 255, 230)) + let banner = BackgroundContainer::new( + CompoundGraphic::with_graphics(vec![ + Graphic::image(self.imgs.banner_top, [138, 17], [0, 0]), + Graphic::rect(Rgba::new(0, 0, 0, 230), [130, 175], [4, 17]), + // Might need floats here + Graphic::image(self.imgs.banner, [130, 15], [4, 192]) + .color(Rgba::new(255, 255, 255, 230)), + ]) .fix_aspect_ratio() - .max_width(300) - .height(Length::Fill); + .height(Length::Fill), + banner_content, + ) + .padding(Padding::new().horizontal(16).top(21).bottom(4)) + .max_width(330); let central_column = Container::new(banner) .width(Length::Fill) @@ -237,10 +249,13 @@ impl IcedState { .height(Length::Fill) .spacing(10); - ui::ice::BackgroundContainer::new(self.imgs.bg, content) - .width(Length::Fill) - .height(Length::Fill) - .into() + BackgroundContainer::new( + Image::new(self.imgs.bg) + .width(Length::Fill) + .height(Length::Fill), + content, + ) + .into() } pub fn update(message: Message) { diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index 1cbeb9f4be..ed3ac1418e 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -7,7 +7,11 @@ mod winit_conversion; pub use graphic::{Id, Rotation}; pub use iced::Event; pub use renderer::IcedRenderer; -pub use widget::{background_container::BackgroundContainer, image::Image}; +pub use widget::{ + background_container::{BackgroundContainer, Padding}, + compound_graphic, + image::Image, +}; pub use winit_conversion::window_event; use super::{ diff --git a/voxygen/src/ui/ice/renderer/background_container.rs b/voxygen/src/ui/ice/renderer/background_container.rs index 2a31ab763a..599afb7806 100644 --- a/voxygen/src/ui/ice/renderer/background_container.rs +++ b/voxygen/src/ui/ice/renderer/background_container.rs @@ -5,22 +5,26 @@ use super::{ use iced::{Element, Layout, Point}; impl background_container::Renderer for IcedRenderer { - fn draw( + fn draw( &mut self, defaults: &Self::Defaults, - layout: Layout<'_>, - cursor_position: Point, - background: image::Handle, - color: vek::Rgba, + background: &B, + background_layout: Layout<'_>, content: &Element<'_, M, Self>, content_layout: Layout<'_>, - ) -> Self::Output { - let image_primitive = image::Renderer::draw(self, background, color, layout).0; + cursor_position: Point, + ) -> Self::Output + where + B: background_container::Background, + { + let back_primitive = background + .draw(self, defaults, background_layout, cursor_position) + .0; let (content_primitive, mouse_cursor) = content.draw(self, defaults, content_layout, cursor_position); ( Primitive::Group { - primitives: vec![image_primitive, content_primitive], + primitives: vec![back_primitive, content_primitive], }, mouse_cursor, ) diff --git a/voxygen/src/ui/ice/renderer/compound_graphic.rs b/voxygen/src/ui/ice/renderer/compound_graphic.rs index d6bb89e6c0..b3b43ce8da 100644 --- a/voxygen/src/ui/ice/renderer/compound_graphic.rs +++ b/voxygen/src/ui/ice/renderer/compound_graphic.rs @@ -20,10 +20,10 @@ impl compound_graphic::Renderer for IcedRenderer { Primitive::Group { primitives: graphics .map(|(bounds, kind)| match kind { - GraphicKind::Image(handle) => Primitive::Image { + GraphicKind::Image(handle, color) => Primitive::Image { handle: (handle, Rotation::None), bounds, - color: Rgba::broadcast(255), + color, }, GraphicKind::Color(color) => Primitive::Rectangle { bounds, color }, }) diff --git a/voxygen/src/ui/ice/widget/background_container.rs b/voxygen/src/ui/ice/widget/background_container.rs index 34db33c629..7072e33d38 100644 --- a/voxygen/src/ui/ice/widget/background_container.rs +++ b/voxygen/src/ui/ice/widget/background_container.rs @@ -1,41 +1,114 @@ use iced::{layout, Element, Hasher, Layout, Length, Point, Size, Widget}; use std::{hash::Hash, u32}; -use vek::Rgba; // Note: it might be more efficient to make this generic over the content type -// Note: maybe we could just use the container styling for this +// Note: maybe we could just use the container styling for this (not really with +// the aspect ratio stuff) -/// This widget is displays a background image behind it's content -pub struct BackgroundContainer<'a, M, R: self::Renderer> { - width: Length, - height: Length, - max_width: u32, - max_height: u32, - background: super::image::Handle, - fix_aspect_ratio: bool, - content: Element<'a, M, R>, - color: Rgba, +#[derive(Copy, Clone, Hash)] +pub struct Padding { + pub top: u16, + pub bottom: u16, + pub right: u16, + pub left: u16, } -impl<'a, M, R> BackgroundContainer<'a, M, R> -where - R: self::Renderer, -{ - pub fn new(background: super::image::Handle, content: impl Into>) -> Self { - Self { - width: Length::Shrink, - height: Length::Shrink, - max_width: u32::MAX, - max_height: u32::MAX, - background, - fix_aspect_ratio: false, - content: content.into(), - color: Rgba::broadcast(255), +impl Padding { + pub fn new() -> Self { + Padding { + top: 0, + bottom: 0, + right: 0, + left: 0, } } - pub fn width(mut self, width: Length) -> Self { + pub fn top(mut self, pad: u16) -> Self { + self.top = pad; + self + } + + pub fn bottom(mut self, pad: u16) -> Self { + self.bottom = pad; + self + } + + pub fn right(mut self, pad: u16) -> Self { + self.right = pad; + self + } + + pub fn left(mut self, pad: u16) -> Self { + self.left = pad; + self + } + + pub fn vertical(mut self, pad: u16) -> Self { + self.top = pad; + self.bottom = pad; + self + } + + pub fn horizontal(mut self, pad: u16) -> Self { + self.left = pad; + self.right = pad; + self + } +} + +pub trait Background: Sized { + // The intended implementors already store the state accessed in the three + // functions below + fn width(&self) -> Length; + fn height(&self) -> Length; + fn aspect_ratio_fixed(&self) -> bool; + fn pixel_dims(&self, renderer: &R) -> [u16; 2]; + fn draw( + &self, + renderer: &mut R, + defaults: &R::Defaults, + layout: Layout<'_>, + cursor_position: Point, + ) -> R::Output; +} + +/// This widget is displays a background image behind it's content +pub struct BackgroundContainer<'a, M, R: self::Renderer, B: Background> { + //width: Length, + //height: Length, + max_width: u32, + max_height: u32, + background: B, + // Padding in same pixel units as background image + // Scaled relative to the background's scaling + padding: Padding, + content: Element<'a, M, R>, +} + +impl<'a, M, R, B> BackgroundContainer<'a, M, R, B> +where + R: self::Renderer, + B: Background, +{ + pub fn new(background: B, content: impl Into>) -> Self { + Self { + //width: Length::Shrink, + //height: Length::Shrink, + max_width: u32::MAX, + max_height: u32::MAX, + background, + padding: Padding::new(), + content: content.into(), + } + } + + pub fn padding(mut self, padding: Padding) -> Self { + self.padding = padding; + self + } + + /*pub fn width(mut self, width: Length) -> Self { self.width = width; self } @@ -43,7 +116,7 @@ where pub fn height(mut self, height: Length) -> Self { self.height = height; self - } + }*/ pub fn max_width(mut self, max_width: u32) -> Self { self.max_width = max_width; @@ -55,7 +128,8 @@ where self } - pub fn fix_aspect_ratio(mut self) -> Self { + // Consider having these wire into underlying background + /*pub fn fix_aspect_ratio(mut self) -> Self { self.fix_aspect_ratio = true; self } @@ -63,30 +137,54 @@ where pub fn color(mut self, color: Rgba) -> Self { self.color = color; self - } + }*/ } -impl<'a, M, R> Widget for BackgroundContainer<'a, M, R> +impl<'a, M, R, B> Widget for BackgroundContainer<'a, M, R, B> where R: self::Renderer, + B: Background, { - fn width(&self) -> Length { self.width } + // Uses the width and height from the background + fn width(&self) -> Length { self.background.width() } - fn height(&self) -> Length { self.height } + fn height(&self) -> Length { self.background.height() } fn layout(&self, renderer: &R, limits: &layout::Limits) -> layout::Node { let limits = limits .loose() // why does iced's container do this? .max_width(self.max_width) .max_height(self.max_height) - .width(self.width) - .height(self.height); + .width(self.width()) + .height(self.height()); - let (size, content) = if self.fix_aspect_ratio { - let (w, h) = renderer.dimensions(self.background); + let [pixel_w, pixel_h] = self.background.pixel_dims(renderer); + let (horizontal_pad_frac, vertical_pad_frac, top_pad_frac, left_pad_frac) = { + let Padding { + top, + bottom, + right, + left, + } = self.padding; + // Just in case + // could convert to gracefully handling + debug_assert!(pixel_w != 0); + debug_assert!(pixel_h != 0); + debug_assert!(top + bottom < pixel_h); + debug_assert!(right + left < pixel_w); + ( + (right + left) as f32 / pixel_w as f32, + (top + bottom) as f32 / pixel_h as f32, + top as f32 / pixel_h as f32, + left as f32 / pixel_w as f32, + ) + }; + + let (size, content) = if self.background.aspect_ratio_fixed() { // To fix the aspect ratio we have to have a separate layout from the content // because we can't force the content to have a specific aspect ratio - let aspect_ratio = w as f32 / h as f32; + let aspect_ratio = pixel_w as f32 / pixel_h as f32; + // To do this we need to figure out the max width/height of the limits // and then adjust one down to meet the aspect ratio let max_size = limits.max(); @@ -97,12 +195,28 @@ where } else { limits.max_height((max_width / aspect_ratio) as u32) }; + limits; + // Account for padding at max size in the limits for the children + let limits = limits.shrink({ + let max = limits.max(); + Size::new( + max.width * horizontal_pad_frac, + max.height * vertical_pad_frac, + ) + }); + limits; + // Get content size // again, why is loose() used here? - let content = self.content.layout(renderer, &limits.loose()); + let mut content = self.content.layout(renderer, &limits.loose()); + // This time we need to adjust up to meet the aspect ratio // so that the container is larger than the contents - let content_size = content.size(); + let mut content_size = content.size(); + // Add minimum padding to content size (this works to ensure we have enough + // space for padding because the available space can only increase) + content_size.width /= 1.0 - horizontal_pad_frac; + content_size.height /= 1.0 - vertical_pad_frac; let content_aspect_ratio = content_size.width as f32 / content_size.height as f32; let size = if content_aspect_ratio > aspect_ratio { Size::new(content_size.width, content_size.width / aspect_ratio) @@ -110,12 +224,39 @@ where Size::new(content_size.height * aspect_ratio, content_size.width) }; + // Move content to account for padding + content.move_to(Point::new( + left_pad_frac * size.width, + top_pad_frac * size.height, + )); + size; + (size, content) } else { - // again, why is loose() used here? - let content = self.content.layout(renderer, &limits.loose()); - let size = limits.resolve(content.size()); - //self.content.layout(renderer, limits) + // Account for padding at max size in the limits for the children + let limits = limits + .shrink({ + let max = limits.max(); + Size::new( + max.width * horizontal_pad_frac, + max.height * vertical_pad_frac, + ) + }) + .loose(); // again, why is loose() used here? + + let mut content = self.content.layout(renderer, &limits); + + let mut size = limits.resolve(content.size()); + // Add padding back + size.width /= 1.0 - horizontal_pad_frac; + size.height /= 1.0 - vertical_pad_frac; + + // Move to account for padding + content.move_to(Point::new( + left_pad_frac * size.width, + top_pad_frac * size.height, + )); + // No aligning since child is currently assumed to be fill (size, content) }; @@ -130,15 +271,13 @@ where layout: Layout<'_>, cursor_position: Point, ) -> R::Output { - self::Renderer::draw( - renderer, + renderer.draw( defaults, + &self.background, layout, - cursor_position, - self.background, - self.color, &self.content, layout.children().next().unwrap(), + cursor_position, ) } @@ -146,35 +285,39 @@ where struct Marker; std::any::TypeId::of::().hash(state); - self.width.hash(state); - self.height.hash(state); + self.width().hash(state); + self.height().hash(state); self.max_width.hash(state); self.max_height.hash(state); - self.fix_aspect_ratio.hash(state); + self.background.aspect_ratio_fixed().hash(state); + self.padding.hash(state); + // TODO: add pixel dims (need renderer) self.content.hash_layout(state); } } -pub trait Renderer: iced::Renderer + super::image::Renderer { - fn draw( +pub trait Renderer: iced::Renderer { + fn draw( &mut self, defaults: &Self::Defaults, - layout: Layout<'_>, - cursor_position: Point, - background: super::image::Handle, - color: Rgba, + background: &B, + background_layout: Layout<'_>, content: &Element<'_, M, Self>, content_layout: Layout<'_>, - ) -> Self::Output; + cursor_position: Point, + ) -> Self::Output + where + B: Background; } // They got to live ¯\_(ツ)_/¯ -impl<'a, M: 'a, R: 'a> From> for Element<'a, M, R> +impl<'a, M: 'a, R: 'a, B> From> for Element<'a, M, R> where R: self::Renderer, + B: 'a + Background, { - fn from(background_container: BackgroundContainer<'a, M, R>) -> Element<'a, M, R> { + fn from(background_container: BackgroundContainer<'a, M, R, B>) -> Element<'a, M, R> { Element::new(background_container) } } diff --git a/voxygen/src/ui/ice/widget/compound_graphic.rs b/voxygen/src/ui/ice/widget/compound_graphic.rs index 48b6f759d4..40798b2dab 100644 --- a/voxygen/src/ui/ice/widget/compound_graphic.rs +++ b/voxygen/src/ui/ice/widget/compound_graphic.rs @@ -12,7 +12,7 @@ use vek::{Aabr, Rgba, Vec2}; #[derive(Copy, Clone)] pub enum GraphicKind { // TODO: if there is a use case, allow coloring individual images - Image(Handle), + Image(Handle, Rgba), Color(Rgba), } @@ -37,9 +37,22 @@ impl Graphic { } pub fn image(handle: Handle, size: [u16; 2], offset: [u16; 2]) -> Self { - Self::new(GraphicKind::Image(handle), size, offset) + Self::new( + GraphicKind::Image(handle, Rgba::broadcast(255)), + size, + offset, + ) } + pub fn color(mut self, color: Rgba) -> Self { + match &mut self.kind { + GraphicKind::Image(_, c) => *c = color, + GraphicKind::Color(c) => *c = color, + } + self + } + + // TODO: consider removing color here pub fn rect(color: Rgba, size: [u16; 2], offset: [u16; 2]) -> Self { Self::new(GraphicKind::Color(color), size, offset) } @@ -95,6 +108,37 @@ impl CompoundGraphic { // self.color = color; // self //} + + fn draw( + &self, + renderer: &mut R, + _defaults: &R::Defaults, + layout: Layout<'_>, + _cursor_position: Point, + ) -> R::Output { + let [pixel_w, pixel_h] = self.graphics_size; + let bounds = layout.bounds(); + let scale = Vec2::new( + bounds.width / pixel_w as f32, + bounds.height / pixel_h as f32, + ); + let graphics = self.graphics.iter().map(|graphic| { + let bounds = { + let Aabr { min, max } = graphic.aabr.map(|e| e as f32); + let min = min * scale; + let size = max * scale - min; + Rectangle { + x: min.x + bounds.x, + y: min.y + bounds.y, + width: size.x, + height: size.y, + } + }; + (bounds, graphic.kind) + }); + + renderer.draw(graphics, /* self.color, */ layout) + } } impl Widget for CompoundGraphic @@ -129,32 +173,11 @@ where fn draw( &self, renderer: &mut R, - _defaults: &R::Defaults, + defaults: &R::Defaults, layout: Layout<'_>, - _cursor_position: Point, + cursor_position: Point, ) -> R::Output { - let [pixel_w, pixel_h] = self.graphics_size; - let bounds = layout.bounds(); - let scale = Vec2::new( - pixel_w as f32 / bounds.width, - pixel_h as f32 / bounds.height, - ); - let graphics = self.graphics.iter().map(|graphic| { - let bounds = { - let Aabr { min, max } = graphic.aabr.map(|e| e as f32); - let min = min * scale; - let size = max * scale - min; - Rectangle { - x: min.x + bounds.x, - y: min.y + bounds.y, - width: size.x, - height: size.y, - } - }; - (bounds, graphic.kind) - }); - - renderer.draw(graphics, /* self.color, */ layout) + Self::draw(self, renderer, defaults, layout, cursor_position) } fn hash_layout(&self, state: &mut Hasher) { @@ -188,3 +211,26 @@ where Element::new(compound_graphic) } } + +impl super::background_container::Background for CompoundGraphic +where + R: self::Renderer, +{ + fn width(&self) -> Length { self.width } + + fn height(&self) -> Length { self.height } + + fn aspect_ratio_fixed(&self) -> bool { self.fix_aspect_ratio } + + fn pixel_dims(&self, _renderer: &R) -> [u16; 2] { self.graphics_size } + + fn draw( + &self, + renderer: &mut R, + defaults: &R::Defaults, + layout: Layout<'_>, + cursor_position: Point, + ) -> R::Output { + Self::draw(self, renderer, defaults, layout, cursor_position) + } +} diff --git a/voxygen/src/ui/ice/widget/image.rs b/voxygen/src/ui/ice/widget/image.rs index a5f5df9f11..22eb73e15d 100644 --- a/voxygen/src/ui/ice/widget/image.rs +++ b/voxygen/src/ui/ice/widget/image.rs @@ -112,3 +112,29 @@ where { fn from(image: Image) -> Element<'a, M, R> { Element::new(image) } } + +impl super::background_container::Background for Image +where + R: self::Renderer, +{ + fn width(&self) -> Length { self.width } + + fn height(&self) -> Length { self.height } + + fn aspect_ratio_fixed(&self) -> bool { self.fix_aspect_ratio } + + fn pixel_dims(&self, renderer: &R) -> [u16; 2] { + let (w, h) = renderer.dimensions(self.handle); + [w as u16, h as u16] + } + + fn draw( + &self, + renderer: &mut R, + _defaults: &R::Defaults, + layout: Layout<'_>, + _cursor_position: Point, + ) -> R::Output { + renderer.draw(self.handle, self.color, layout) + } +} From 3f6faecbb59582092d745a13b06a464b8cab8624 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sun, 19 Apr 2020 23:39:58 -0400 Subject: [PATCH 07/61] Add support for Space widget, refine main menu ui, misc fixes and improvements. --- assets/voxygen/element/buttons/button.png | Bin 1717 -> 4971 bytes assets/voxygen/element/misc_bg/textbox.png | Bin 0 -> 7799 bytes .../voxygen/element/misc_bg/textbox_bot.png | Bin 6377 -> 0 bytes .../voxygen/element/misc_bg/textbox_mid.png | Bin 2204 -> 0 bytes .../voxygen/element/misc_bg/textbox_top.png | Bin 6358 -> 0 bytes voxygen/src/menu/char_selection/ui.rs | 2 +- voxygen/src/menu/main/ui.rs | 69 ++++++++++++++---- voxygen/src/ui/ice/renderer.rs | 3 + .../ui/ice/renderer/background_container.rs | 5 +- .../src/ui/ice/renderer/compound_graphic.rs | 1 - voxygen/src/ui/ice/renderer/space.rs | 8 ++ .../src/ui/ice/widget/background_container.rs | 5 +- voxygen/src/ui/ice/widget/compound_graphic.rs | 10 ++- 13 files changed, 79 insertions(+), 24 deletions(-) create mode 100644 assets/voxygen/element/misc_bg/textbox.png delete mode 100644 assets/voxygen/element/misc_bg/textbox_bot.png delete mode 100644 assets/voxygen/element/misc_bg/textbox_mid.png delete mode 100644 assets/voxygen/element/misc_bg/textbox_top.png create mode 100644 voxygen/src/ui/ice/renderer/space.rs diff --git a/assets/voxygen/element/buttons/button.png b/assets/voxygen/element/buttons/button.png index 0f29f105f4ed78ef2c87d737174c570530c31a5e..ec740a7d4bcb1cc8a7f1d50eba6ec9b2cab2c831 100644 GIT binary patch literal 4971 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 LNkvXXu0mjf(r!Gi diff --git a/assets/voxygen/element/misc_bg/textbox.png b/assets/voxygen/element/misc_bg/textbox.png new file mode 100644 index 0000000000000000000000000000000000000000..222091e16c2405c8d064521bb11c99b55105ba0c GIT binary patch literal 7799 zcmeHLc|4SB`=3M#g-W7g3?<8~1~ZdM7;6SeQY2$G6K2LN3?)TVT2!bkWh=5JZHk0? zZ7tT2qL4_ov?xnQ-e)LB=k1*Hp3m?1{?~jy&y4%Nzt{KL?(4d5&tV5U%Y{Z`V)r;#FQScoG|BNFD0|T59MR3p)H)(wGiOn<_he6d&$wcUww62c zbi8ZL{*tGj4K?Nqb4&a~HinOkOfD?azE}_({|BFRcGX}>e11m{*N1U zlW#1(Zp%OG(&0LI^{#u;+gBsE5Ve)J>KqKYLa4~{led?rPH<&~O_l_&IYYqELtruU2nFDtYWNPcKG5XN@YP_L3gyYs@d&p{~Ez!{V zX_`tw?8~Y#!|Px|O{wecIeS%QCthy$2+Xj!h^IUuGH^52a1lHRh}B`YWQ=?A(roYhL& zy55C6tleF7Ape|3k4yzx>X;&Z-#+IaU8lpF60kkG=+#*D z!U{?{2W>v(3K}NMpLaJ-7{?YSzunqaO!5s(IBQ7G)#cfSMFv~aS6{O5bxw3?ijV z#VY&e{zr~bt;-S{qE(X>6;I?X_BkJ}?(Y5Yt;5xL_^p4v(dm`~`%!FZ_RF4}yRuw4 zUsmYt+$0Y=rQloI8Sk_ma*23@aqA4Z-10%*SsUmh^sP0!!TGVv{a0+O4OdyXL^H>O z*?Tli$UN|{E^3&%+e>G0Z2q+_wX#0QebcL(&%UhJZPQBTC|96HJX8;q)oKjN**4-M znk>pR*Cad*Nqp*xJ>V1GGkkKxz@>&kpS#{j>w;@->!Tc%OCvVR1Utl49PX~NRa0Lq zwA3!Rn{^<(HcQ8Q@rYzRG^xL6A+#|jV$bQJ7?ty?uwj!id3Ky4*5X{3l4bk;yE#|G z=(!(d(y@_BnMqBr`#N+USG|e}D$vHnKr@TnqRkH)h>h%XShAw*O9mvBpR131yxZjQ z?VL57P)~h)c~?bP@~~3wWxf#;m(hN2tn^W(<#2Cy4CC|Bi4f^@*2p^P=lkTD=Ue

r4PFx0;`Tc9F})<_saRq79}C`zSjQ0u^NYG zBb^h<84-m#i!dCwMatEo`->&tm!h@jjH9`x^9_|3*TuSbrfjtiyeVX!JgPWspuIAe<-k*mc695nA>K4Ph56k_GZ)W z`geXpcctvr7q$mM_Tp>sk*wIpHF(6nIyH@~!52gZ>(` z`uCwPvm3mY2YYCC_J+Bs}XM8%!xT zH}Zyz%KOSf#>)xx<5-@FLS~9x`JJt~H_}ZNHkNnaATWZPS2mS`r5+xo^;l4QGxXkD zJiw?y4h3!3l#JUXpvA&!zbfy*EI^svZTCmX~Vg5#ycRW{Ya_ZWpk%bw}UsEeg z_uJk`88pCme+j5mcY@?L;dbdf5@NrYQ_VfGCfk1PF z3^OwaYcsR&=P+>ko(qn|Ta|8DS-#UX&rxU90Ju}`3T~U6f?uQd(iGDSh3$Mq!LXUv z2iZJt+A(NP79(i z7c-Q*zg&muoV$g|?$-8hi!MQ&Yd#Z`Z^xCh*7cBwh~LlfoJ(~HZ^(-Dgb=?hke%H3 zO7X$!Xz#hlR({ObAnY2wH`x(zi$?N%PL$HzSEJbab@{x|!VMQ_t?iXJUts2l)!se* z0A_EU4mnfvRqT3xxS z>9QLR-nMxGl00t$)AudD*uo;U(1*2hT z7y@c8VECgT>!rX(910ERNU-=u0gUhvFCLGLgTn&?0$>4pFcyanM`E#9I06Mnp`ZW) z$_-@lNCGI6t1hCL<{(hHWDbMPW3ZTD5hux$#pmH65MUkrjT~4dn+t%+Zwf@?DS9rC z0=EVRXuux;07oGZ7$^b-MPcEy<$+ZqaYmZS{T4+ao^Sz)4M)Nda9`gaB)B|t|KIif zRD$aS+)v?-R4$9pAydu$sZ5^wY^7`;K6kcGK9?$*nzGx6LW2WFO=+GjV`)uvn2`~M zkI)dc1(=aP;Mra*9*gV6V$X>E;MC0Wf0zU) z{<+Ba$W2)_6=58KMdpjtTNCgQUIo>5)7Q z5D1i~p1~{%YbKXRVv?!fDHs3+5=}!QFcf_#35!8N(O3gAltiYGp=h!JMjwGS#Gp`= zSrqmh22cwmpZ}9JqD99~F#tET^ z(?jA=$Uo2lhBIh^e}@+J1F+GzO2^r<$Rtrmm^Pd;wK&}_Oi6V3G#36>;QwIS?8ORT z{&zgTLuXh_IJ^KB$J?G`@41Ug=KVF#&%iTGjzFX4@;HIkf8*32a7NP=W(CNyIDxbD zZ>IW9kEX-p!aHsSPk)B|w~~7a2dPwg0+YXIQu_8ZUsvp_I1xXswMo?MJNI-DzI<<7{Nv7-1o#8iB19{5jQ)vMxv7fXDw<6oB3?E4};42 zG3DP&;7@R~-Jf2jft{-xv}#rIdZeue8FMc^NSf9Ne7Q~9xGrHXLP@e}qzR}YvE zCWosk?2x?stoDLzD-`LXV`QSL%s9OMt%*F8lt&<4So>U7(HL zD>yHtBRl{8CvPUxlj2ifn_8ZJr;_w3(4nSjVF+lbs?pYAr)tSpXCoCE zFT2!+d{F``t4d9bZaK^D(c|Tq*Cj_EH2>s~QMkpa`pq-zb^JQJ1oTSby;N6xA>YyX kb6;vPOUnIdGH6ol_%000!0dQ@0+Qek%> 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 diff --git a/assets/voxygen/element/misc_bg/textbox_mid.png b/assets/voxygen/element/misc_bg/textbox_mid.png deleted file mode 100644 index 88bef50fecd3674e005d7d2b4d78b8c3223420a5..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2204 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 Y0E89YwfjLv^#A|>07*qoM6N<$f+d+20{{R3 diff --git a/voxygen/src/menu/char_selection/ui.rs b/voxygen/src/menu/char_selection/ui.rs index 25fc83a9c7..5e64c535fb 100644 --- a/voxygen/src/menu/char_selection/ui.rs +++ b/voxygen/src/menu/char_selection/ui.rs @@ -192,7 +192,7 @@ image_ids! { selection_hover: "voxygen.element.frames.selection_hover", selection_press: "voxygen.element.frames.selection_press", - name_input: "voxygen.element.misc_bg.textbox_mid", + name_input: "voxygen.element.misc_bg.textbox", slider_range: "voxygen.element.slider.track", slider_indicator: "voxygen.element.slider.indicator", diff --git a/voxygen/src/menu/main/ui.rs b/voxygen/src/menu/main/ui.rs index 2c6d02b128..4d8fd791eb 100644 --- a/voxygen/src/menu/main/ui.rs +++ b/voxygen/src/menu/main/ui.rs @@ -107,7 +107,7 @@ image_ids! { button: "voxygen.element.buttons.button", button_hover: "voxygen.element.buttons.button_hover", button_press: "voxygen.element.buttons.button_press", - input_bg: "voxygen.element.misc_bg.textbox_mid", + input_bg: "voxygen.element.misc_bg.textbox", //disclaimer: "voxygen.element.frames.disclaimer", loading_art: "voxygen.element.frames.loading_screen.loading_bg", loading_art_l: "voxygen.element.frames.loading_screen.loading_bg_l", @@ -140,9 +140,7 @@ image_ids_ice! { button: "voxygen.element.buttons.button", button_hover: "voxygen.element.buttons.button_hover", button_press: "voxygen.element.buttons.button_press", - input_bg_top: "voxygen.element.misc_bg.textbox_top", - input_bg_mid: "voxygen.element.misc_bg.textbox_mid", - input_bg_bot: "voxygen.element.misc_bg.textbox_bot", + input_bg: "voxygen.element.misc_bg.textbox", disclaimer: "voxygen.element.frames.disclaimer", @@ -193,7 +191,7 @@ struct IcedState { pub type Message = Event; impl IcedState { pub fn view(&mut self) -> Element { - use iced::{Align, Column, Container, Length, Row}; + use iced::{Align, Column, Container, Length, Row, Space}; use ui::ice::{ compound_graphic::{CompoundGraphic, Graphic}, BackgroundContainer, Image, Padding, @@ -216,23 +214,68 @@ impl IcedState { .align_y(Align::End) .padding(20); - let banner_content = - Column::with_children(vec![Image::new(self.imgs.v_logo).fix_aspect_ratio().into()]); - //.padding(15); + let banner_content = Column::with_children(vec![ + Image::new(self.imgs.v_logo) + .fix_aspect_ratio() + .height(Length::FillPortion(20)) + .into(), + Space::new(Length::Fill, Length::FillPortion(5)).into(), + Column::with_children(vec![ + BackgroundContainer::new( + CompoundGraphic::padded_image(self.imgs.input_bg, [169, 25], [0, 0, 0, 1]) + .fix_aspect_ratio(), + Space::new(Length::Fill, Length::Fill), + ) + .into(), + BackgroundContainer::new( + CompoundGraphic::padded_image(self.imgs.input_bg, [169, 25], [0, 1, 0, 1]) + .fix_aspect_ratio(), + Space::new(Length::Fill, Length::Fill), + ) + .into(), + BackgroundContainer::new( + CompoundGraphic::padded_image(self.imgs.input_bg, [169, 25], [0, 1, 0, 0]) + .fix_aspect_ratio(), + Space::new(Length::Fill, Length::Fill), + ) + .into(), + ]) + .height(Length::FillPortion(50)) + .into(), + Column::with_children(vec![ + BackgroundContainer::new( + CompoundGraphic::padded_image(self.imgs.button, [106, 26], [10, 0, 10, 2]) + .fix_aspect_ratio(), + Space::new(Length::Fill, Length::Fill), + ) + .into(), + BackgroundContainer::new( + CompoundGraphic::padded_image(self.imgs.button, [106, 26], [10, 2, 10, 0]) + .fix_aspect_ratio(), + Space::new(Length::Fill, Length::Fill), + ) + .into(), + ]) + .max_width(240) + .height(Length::FillPortion(25)) + .into(), + ]) + .width(Length::Fill) + .height(Length::Fill) + .align_items(Align::Center); let banner = BackgroundContainer::new( - CompoundGraphic::with_graphics(vec![ + CompoundGraphic::from_graphics(vec![ Graphic::image(self.imgs.banner_top, [138, 17], [0, 0]), - Graphic::rect(Rgba::new(0, 0, 0, 230), [130, 175], [4, 17]), - // Might need floats here - Graphic::image(self.imgs.banner, [130, 15], [4, 192]) + Graphic::rect(Rgba::new(0, 0, 0, 230), [130, 195], [4, 17]), + Graphic::image(self.imgs.banner, [130, 15], [4, 212]) .color(Rgba::new(255, 255, 255, 230)), ]) .fix_aspect_ratio() .height(Length::Fill), banner_content, ) - .padding(Padding::new().horizontal(16).top(21).bottom(4)) + .padding(Padding::new().horizontal(16).vertical(20)) .max_width(330); let central_column = Container::new(banner) diff --git a/voxygen/src/ui/ice/renderer.rs b/voxygen/src/ui/ice/renderer.rs index feb8580499..d8657de20e 100644 --- a/voxygen/src/ui/ice/renderer.rs +++ b/voxygen/src/ui/ice/renderer.rs @@ -4,6 +4,7 @@ mod compound_graphic; mod container; mod image; mod row; +mod space; use super::{ super::{ @@ -69,6 +70,7 @@ pub enum Primitive { bounds: iced::Rectangle, color: Rgba, }, + Nothing, } // Optimization idea inspired by what I think iced wgpu renderer may be doing @@ -347,6 +349,7 @@ impl IcedRenderer { UiMode::Geometry, )); }, + Primitive::Nothing => {}, } } diff --git a/voxygen/src/ui/ice/renderer/background_container.rs b/voxygen/src/ui/ice/renderer/background_container.rs index 599afb7806..c92f4f6986 100644 --- a/voxygen/src/ui/ice/renderer/background_container.rs +++ b/voxygen/src/ui/ice/renderer/background_container.rs @@ -1,7 +1,4 @@ -use super::{ - super::widget::{background_container, image}, - IcedRenderer, Primitive, -}; +use super::{super::widget::background_container, IcedRenderer, Primitive}; use iced::{Element, Layout, Point}; impl background_container::Renderer for IcedRenderer { diff --git a/voxygen/src/ui/ice/renderer/compound_graphic.rs b/voxygen/src/ui/ice/renderer/compound_graphic.rs index b3b43ce8da..166dbef484 100644 --- a/voxygen/src/ui/ice/renderer/compound_graphic.rs +++ b/voxygen/src/ui/ice/renderer/compound_graphic.rs @@ -4,7 +4,6 @@ use super::{ }; use compound_graphic::GraphicKind; use iced::{MouseCursor, Rectangle}; -use vek::Rgba; impl compound_graphic::Renderer for IcedRenderer { fn draw( diff --git a/voxygen/src/ui/ice/renderer/space.rs b/voxygen/src/ui/ice/renderer/space.rs new file mode 100644 index 0000000000..8c1f898e67 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/space.rs @@ -0,0 +1,8 @@ +use super::{IcedRenderer, Primitive}; +use iced::{space, MouseCursor, Rectangle}; + +impl space::Renderer for IcedRenderer { + fn draw(&mut self, _bounds: Rectangle) -> Self::Output { + (Primitive::Nothing, MouseCursor::OutOfBounds) + } +} diff --git a/voxygen/src/ui/ice/widget/background_container.rs b/voxygen/src/ui/ice/widget/background_container.rs index 7072e33d38..a4fa05f7a6 100644 --- a/voxygen/src/ui/ice/widget/background_container.rs +++ b/voxygen/src/ui/ice/widget/background_container.rs @@ -195,7 +195,6 @@ where } else { limits.max_height((max_width / aspect_ratio) as u32) }; - limits; // Account for padding at max size in the limits for the children let limits = limits.shrink({ let max = limits.max(); @@ -204,7 +203,6 @@ where max.height * vertical_pad_frac, ) }); - limits; // Get content size // again, why is loose() used here? @@ -221,7 +219,7 @@ where let size = if content_aspect_ratio > aspect_ratio { Size::new(content_size.width, content_size.width / aspect_ratio) } else { - Size::new(content_size.height * aspect_ratio, content_size.width) + Size::new(content_size.height * aspect_ratio, content_size.height) }; // Move content to account for padding @@ -229,7 +227,6 @@ where left_pad_frac * size.width, top_pad_frac * size.height, )); - size; (size, content) } else { diff --git a/voxygen/src/ui/ice/widget/compound_graphic.rs b/voxygen/src/ui/ice/widget/compound_graphic.rs index 40798b2dab..bae79a6bef 100644 --- a/voxygen/src/ui/ice/widget/compound_graphic.rs +++ b/voxygen/src/ui/ice/widget/compound_graphic.rs @@ -70,7 +70,7 @@ pub struct CompoundGraphic { } impl CompoundGraphic { - pub fn with_graphics(graphics: Vec) -> Self { + pub fn from_graphics(graphics: Vec) -> Self { let width = Length::Fill; let height = Length::Fill; let graphics_size = graphics @@ -89,6 +89,14 @@ impl CompoundGraphic { } } + pub fn padded_image(image: Handle, size: [u16; 2], pad: [u16; 4]) -> Self { + let image = Graphic::image(image, size, [pad[0], pad[1]]); + let mut this = Self::from_graphics(vec![image]); + this.graphics_size[0] += pad[2]; + this.graphics_size[1] += pad[3]; + this + } + pub fn width(mut self, width: Length) -> Self { self.width = width; self From a1f59cb2eac800fe79930f05863d9fc427a21b61 Mon Sep 17 00:00:00 2001 From: Imbris Date: Tue, 21 Apr 2020 21:29:59 -0400 Subject: [PATCH 08/61] Implement text renderering with glyph_brush (lifetime error) --- Cargo.lock | 32 +++ voxygen/Cargo.toml | 1 + voxygen/src/menu/main/ui.rs | 13 +- voxygen/src/render/mesh.rs | 15 ++ voxygen/src/ui/ice/cache.rs | 153 +++++++++++++ voxygen/src/ui/ice/mod.rs | 40 ++-- voxygen/src/ui/ice/renderer.rs | 210 +++++++++++++++--- .../ui/ice/renderer/background_container.rs | 11 +- voxygen/src/ui/ice/renderer/column.rs | 20 +- .../src/ui/ice/renderer/compound_graphic.rs | 10 +- voxygen/src/ui/ice/renderer/container.rs | 9 +- voxygen/src/ui/ice/renderer/image.rs | 13 +- voxygen/src/ui/ice/renderer/row.rs | 20 +- voxygen/src/ui/ice/renderer/space.rs | 8 +- voxygen/src/ui/ice/renderer/stack.rs | 23 +- voxygen/src/ui/ice/renderer/text.rs | 97 ++++++++ voxygen/src/ui/ice/winit_conversion.rs | 45 ++-- 17 files changed, 598 insertions(+), 122 deletions(-) create mode 100644 voxygen/src/ui/ice/cache.rs create mode 100644 voxygen/src/ui/ice/renderer/text.rs diff --git a/Cargo.lock b/Cargo.lock index e7f7f63ce1..d274d777a5 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1856,6 +1856,31 @@ dependencies = [ "gl_generator", ] +[[package]] +name = "glyph_brush" +version = "0.6.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5fca6f9d679bff1322c76c9a1ad4b8553b30a94f3f75bea6936e19032c2f2ec3" +dependencies = [ + "glyph_brush_layout", + "log", + "ordered-float", + "rustc-hash", + "rusttype 0.8.3", + "twox-hash", +] + +[[package]] +name = "glyph_brush_layout" +version = "0.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b70adc570f1dc71b6b32e241cbcc2b42175f5aea71951fbf41e68b04aec24c7" +dependencies = [ + "approx", + "rusttype 0.8.3", + "xi-unicode", +] + [[package]] name = "guillotiere" version = "0.5.2" @@ -5083,6 +5108,7 @@ dependencies = [ "git2", "glsl-include", "glutin", + "glyph_brush", "guillotiere", "hashbrown 0.7.2", "iced_native", @@ -5667,6 +5693,12 @@ version = "2.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d089681aa106a86fade1b0128fb5daf07d5867a509ab036d99988dec80429a57" +[[package]] +name = "xi-unicode" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e71b85d8b1b8bfaf4b5c834187554d201a8cd621c2bbfa33efd41a3ecabd48b2" + [[package]] name = "xml-rs" version = "0.8.3" diff --git a/voxygen/Cargo.toml b/voxygen/Cargo.toml index 7dfcd90ea8..c4b3208c1c 100644 --- a/voxygen/Cargo.toml +++ b/voxygen/Cargo.toml @@ -37,6 +37,7 @@ conrod_winit = {git = "https://gitlab.com/veloren/conrod.git", branch="copypasta euc = {git = "https://github.com/zesterer/euc.git"} iced = {package = "iced_native", git = "https://github.com/hecrj/iced"} window_clipboard = "0.1.1" +glyph_brush = "0.6.3" # ECS specs = {git = "https://github.com/amethyst/specs.git", rev = "7a2e348ab2223818bad487695c66c43db88050a5"} diff --git a/voxygen/src/menu/main/ui.rs b/voxygen/src/menu/main/ui.rs index 4d8fd791eb..a01ff43b15 100644 --- a/voxygen/src/menu/main/ui.rs +++ b/voxygen/src/menu/main/ui.rs @@ -378,7 +378,18 @@ impl<'a> MainMenuUi { let fonts = ConrodVoxygenFonts::load(&voxygen_i18n.fonts, &mut ui) .expect("Impossible to load fonts!"); - let mut ice_ui = IcedUi::new(window).unwrap(); + // TODO: newtype Font + let ice_font = { + use std::io::Read; + let mut buf = Vec::new(); + common::assets::load_file("voxygen.font.OpenSans-Regular", &["ttf"]) + .unwrap() + .read_to_end(&mut buf) + .unwrap(); + glyph_brush::rusttype::Font::from_bytes(buf).unwrap() + }; + + let mut ice_ui = IcedUi::new(window, ice_font).unwrap(); let ice_state = IcedState { imgs: IcedImgs::load(&mut ice_ui).expect("Failed to load images"), }; diff --git a/voxygen/src/render/mesh.rs b/voxygen/src/render/mesh.rs index cd38e88ae4..a8c6b6445e 100644 --- a/voxygen/src/render/mesh.rs +++ b/voxygen/src/render/mesh.rs @@ -57,6 +57,21 @@ impl Mesh

{ self.verts.push(quad.a); } + /// Overwrite a quad + pub fn replace_quad(&mut self, index: usize, quad: Quad

) { + debug_assert!(index % 3 == 0); + assert!(index + 5 < self.verts.len()); + // Tri 1 + self.verts[index] = quad.a.clone(); + self.verts[index + 1] = quad.b; + self.verts[index + 2] = quad.c.clone(); + + // Tri 2 + self.verts[index + 3] = quad.c; + self.verts[index + 4] = quad.d; + self.verts[index + 5] = quad.a; + } + /// Push the vertices of another mesh onto the end of this mesh. pub fn push_mesh(&mut self, other: &Mesh

) { self.verts.extend_from_slice(other.vertices()); } diff --git a/voxygen/src/ui/ice/cache.rs b/voxygen/src/ui/ice/cache.rs new file mode 100644 index 0000000000..c3d9e70f06 --- /dev/null +++ b/voxygen/src/ui/ice/cache.rs @@ -0,0 +1,153 @@ +use super::{ + graphic::{Graphic, GraphicCache, Id as GraphicId}, + renderer::{IcedRenderer, Primitive}, +}; +use crate::{ + render::{Renderer, Texture}, + Error, +}; +use glyph_brush::{ + GlyphBrushBuilder, GlyphCalculator, GlyphCalculatorBuilder, GlyphCalculatorGuard, +}; +use std::cell::RefCell; +use vek::*; + +// Multiplied by current window size +const GLYPH_CACHE_SIZE: u16 = 1; +// Glyph cache tolerances +const SCALE_TOLERANCE: f32 = 0.1; +const POSITION_TOLERANCE: f32 = 0.1; + +type GlyphBrush = glyph_brush::GlyphBrush<'static, (Aabr, Aabr)>; + +pub type Font = glyph_brush::rusttype::Font<'static>; + +pub struct Cache { + glyph_cache: GlyphBrush, + glyph_cache_tex: Texture, + graphic_cache: GraphicCache, +} + +// TODO: Should functions be returning UiError instead of Error? +impl Cache { + pub fn new(renderer: &mut Renderer, default_font: Font) -> Result { + let (w, h) = renderer.get_resolution().into_tuple(); + + let max_texture_size = renderer.max_texture_size(); + + let glyph_cache_dims = + Vec2::new(w, h).map(|e| (e * GLYPH_CACHE_SIZE).min(max_texture_size as u16).max(512)); + + Ok(Self { + glyph_cache: GlyphBrushBuilder::using_font(default_font) + .initial_cache_size((glyph_cache_dims.x as u32, glyph_cache_dims.y as u32)) + .gpu_cache_scale_tolerance(SCALE_TOLERANCE) + .gpu_cache_position_tolerance(POSITION_TOLERANCE) + .build(), + glyph_cache_tex: renderer.create_dynamic_texture(glyph_cache_dims.map(|e| e as u16))?, + graphic_cache: GraphicCache::new(renderer), + }) + } + + pub fn glyph_cache_tex(&self) -> &Texture { &self.glyph_cache_tex } + + pub fn glyph_cache_mut_and_tex(&mut self) -> (&mut GlyphBrush, &Texture) { + (&mut self.glyph_cache, &self.glyph_cache_tex) + } + + pub fn glyph_cache_mut(&mut self) -> &mut GlyphBrush { &mut self.glyph_cache } + + // TODO: add font fn + + pub fn graphic_cache(&self) -> &GraphicCache { &self.graphic_cache } + + pub fn graphic_cache_mut(&mut self) -> &mut GraphicCache { &mut self.graphic_cache } + + pub fn add_graphic(&mut self, graphic: Graphic) -> GraphicId { + self.graphic_cache.add_graphic(graphic) + } + + pub fn replace_graphic(&mut self, id: GraphicId, graphic: Graphic) { + self.graphic_cache.replace_graphic(id, graphic) + } + + // Resizes and clears the GraphicCache + pub fn resize_graphic_cache(&mut self, renderer: &mut Renderer) { + self.graphic_cache.clear_cache(renderer); + } + + // Resizes and clears the GlyphCache + pub fn resize_glyph_cache(&mut self, renderer: &mut Renderer) -> Result<(), Error> { + let max_texture_size = renderer.max_texture_size(); + let cache_dims = renderer + .get_resolution() + .map(|e| (e * GLYPH_CACHE_SIZE).min(max_texture_size as u16).max(512)); + self.glyph_cache = self + .glyph_cache + .to_builder() + .initial_cache_size((cache_dims.x as u32, cache_dims.y as u32)) + .build(); + self.glyph_cache_tex = renderer.create_dynamic_texture(cache_dims.map(|e| e as u16))?; + Ok(()) + } +} + +pub struct GlyphCalcCache { + // Hold one of these for adding new fonts + builder: GlyphCalculatorBuilder<'static>, + calculator: GlyphCalculator<'static>, +} + +impl GlyphCalcCache { + pub fn new(default_font: Font) -> Self { + // Multiple copies of the font in memory :/ (using Arc<[u8]> might help) + let builder = GlyphCalculatorBuilder::using_font(default_font); + let calculator = builder.clone().build(); + + Self { + builder, + calculator, + } + } + + pub fn frame_guard<'a>(&'a self) -> GlyphCalculatorGuard<'a, 'static> { + self.calculator.cache_scope() + } + + // add new font fn +} + +pub struct FrameRenderer<'a> { + pub renderer: &'a mut IcedRenderer, + pub glyph_calc: RefCell>, +} + +impl<'a> FrameRenderer<'a> { + pub fn new(renderer: &'a mut IcedRenderer, glyph_calc_cache: &'a mut GlyphCalcCache) -> Self { + Self { + renderer, + glyph_calc: RefCell::new(glyph_calc_cache.frame_guard()), + } + } +} + +impl iced::Renderer for FrameRenderer<'_> { + // Default styling + type Defaults = (); + // TODO: use graph of primitives to enable diffing??? + type Output = (Primitive, iced::mouse::Interaction); + + fn layout<'a, M>( + &mut self, + element: &iced::Element<'a, M, Self>, + limits: &iced::layout::Limits, + ) -> iced::layout::Node { + let node = element.layout(self, limits); + + // Trim text measurements cache? + + node + } +} + +// TODO: impl Debugger diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index ed3ac1418e..4fe0d7cf83 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -1,9 +1,11 @@ // tooltip_manager: TooltipManager, +mod cache; mod clipboard; mod renderer; mod widget; mod winit_conversion; +pub use cache::Font; pub use graphic::{Id, Rotation}; pub use iced::Event; pub use renderer::IcedRenderer; @@ -19,14 +21,16 @@ use super::{ scale::{Scale, ScaleMode}, }; use crate::{render::Renderer, window::Window, Error}; +use cache::{FrameRenderer, GlyphCalcCache}; use clipboard::Clipboard; -use iced::{Cache, MouseCursor, Size, UserInterface}; +use iced::{mouse, Cache, Size, UserInterface}; use vek::*; -pub type Element<'a, M> = iced::Element<'a, M, IcedRenderer>; +pub type Element<'a, 'b, M> = iced::Element<'a, M, FrameRenderer<'b>>; pub struct IcedUi { renderer: IcedRenderer, + glyph_calc_cache: GlyphCalcCache, cache: Option, events: Vec, clipboard: Clipboard, @@ -35,14 +39,16 @@ pub struct IcedUi { window_resized: Option>, } impl IcedUi { - pub fn new(window: &mut Window) -> Result { + pub fn new(window: &mut Window, default_font: Font) -> Result { let scale = Scale::new(window, ScaleMode::Absolute(1.0)); let renderer = window.renderer_mut(); let scaled_dims = scale.scaled_window_size().map(|e| e as f32); + // TODO: examine how much mem fonts take up and reduce clones if significant Ok(Self { - renderer: IcedRenderer::new(renderer, scaled_dims)?, + renderer: IcedRenderer::new(renderer, scaled_dims, default_font.clone())?, + glyph_calc_cache: GlyphCalcCache::new(default_font), cache: Some(Cache::new()), events: Vec::new(), // TODO: handle None @@ -58,7 +64,7 @@ impl IcedUi { } pub fn handle_event(&mut self, event: Event) { - use iced::{input::mouse, window}; + use iced::window; match event { // Intercept resizing events Event::Window(window::Event::Resized { width, height }) => { @@ -93,11 +99,12 @@ impl IcedUi { } // TODO: produce root internally??? - pub fn maintain<'a, M, E: Into>>( + // TODO: see if this lifetime soup can be simplified + pub fn maintain<'a, 'b, M, E: Into>>>( &mut self, root: E, renderer: &mut Renderer, - ) -> (Vec, MouseCursor) { + ) -> (Vec, mouse::Interaction) { // Handle window resizing if let Some(new_dims) = self.window_resized.take() { let old_scaled_dims = self.scale.scaled_window_size(); @@ -126,23 +133,28 @@ impl IcedUi { // TODO: convert to f32 at source let window_size = self.scale.scaled_window_size().map(|e| e as f32); + let mut frame_renderer = FrameRenderer::new(&mut self.renderer, &mut self.glyph_calc_cache); + let mut user_interface = UserInterface::build( root, Size::new(window_size.x, window_size.y), self.cache.take().unwrap(), - &mut self.renderer, + &mut frame_renderer, ); - let messages = - user_interface.update(self.events.drain(..), Some(&self.clipboard), &self.renderer); + let messages = user_interface.update( + self.events.drain(..), + Some(&self.clipboard), + &frame_renderer, + ); - let (primitive, mouse_cursor) = user_interface.draw(&mut self.renderer); - - self.renderer.draw(primitive, renderer); + let (primitive, mouse_interaction) = user_interface.draw(&mut frame_renderer); self.cache = Some(user_interface.into_cache()); - (messages, mouse_cursor) + self.renderer.draw(primitive, renderer); + + (messages, mouse_interaction) } pub fn render(&self, renderer: &mut Renderer) { self.renderer.render(renderer, None); } diff --git a/voxygen/src/ui/ice/renderer.rs b/voxygen/src/ui/ice/renderer.rs index d8657de20e..f94581d4d4 100644 --- a/voxygen/src/ui/ice/renderer.rs +++ b/voxygen/src/ui/ice/renderer.rs @@ -7,11 +7,9 @@ mod row; mod space; use super::{ - super::{ - cache::Cache, - graphic::{self, Graphic, TexId}, - }, - widget, Rotation, + super::graphic::{self, Graphic, TexId}, + cache::Cache, + widget, Font, Rotation, }; use crate::{ render::{ @@ -70,10 +68,23 @@ pub enum Primitive { bounds: iced::Rectangle, color: Rgba, }, + Text { + glyphs: Vec<( + glyph_brush::rusttype::PositionedGlyph<'static>, + glyph_brush::Color, + glyph_brush::FontId, + )>, + //size: f32, + bounds: iced::Rectangle, + linear_color: Rgba, + /*font: iced::Font, + *horizontal_alignment: iced::HorizontalAlignment, + *vertical_alignment: iced::VerticalAlignment, */ + }, Nothing, } -// Optimization idea inspired by what I think iced wgpu renderer may be doing +// Optimization idea inspired by what I think iced wgpu renderer may be doing: // Could have layers of things which don't intersect and thus can be reordered // arbitrarily @@ -90,35 +101,51 @@ pub struct IcedRenderer { // Used to delay cache resizing until after current frame is drawn //need_cache_resize: bool, + // Half of physical resolution half_res: Vec2, // Pixel perfection alignment align: Vec2, + // Scale factor between physical and win dims + p_scale: f32, // Pretend dims :) (i.e. scaled) win_dims: Vec2, // Per-frame/update current_state: State, mesh: Mesh, + glyphs: Vec<(usize, usize, Rgba)>, + // Output from glyph_brush in the previous frame + // It can sometimes ask you to redraw with these instead (idk if that is done with + // pre-positioned glyphs) + last_glyph_verts: Vec<(Aabr, Aabr)>, start: usize, // Draw commands for the next render draw_commands: Vec, //current_scissor: Aabr, } impl IcedRenderer { - pub fn new(renderer: &mut Renderer, scaled_dims: Vec2) -> Result { - let (half_res, align) = Self::calculate_resolution_dependents(renderer.get_resolution()); + pub fn new( + renderer: &mut Renderer, + scaled_dims: Vec2, + default_font: Font, + ) -> Result { + let (half_res, align, p_scale) = + Self::calculate_resolution_dependents(renderer.get_resolution(), scaled_dims); Ok(Self { - cache: Cache::new(renderer)?, + cache: Cache::new(renderer, default_font)?, draw_commands: Vec::new(), model: renderer.create_dynamic_model(100)?, interface_locals: renderer.create_consts(&[UiLocals::default()])?, default_globals: renderer.create_consts(&[Globals::default()])?, ingame_locals: Vec::new(), mesh: Mesh::new(), + glyphs: Vec::new(), + last_glyph_verts: Vec::new(), current_state: State::Plain, half_res, align, + p_scale, win_dims: scaled_dims, start: 0, //current_scissor: default_scissor(renderer), @@ -144,6 +171,7 @@ impl IcedRenderer { // Re-use memory self.draw_commands.clear(); self.mesh.clear(); + self.glyphs.clear(); self.current_state = State::Plain; self.start = 0; @@ -177,6 +205,80 @@ impl IcedRenderer { self.draw_commands .push(DrawCommand::plain(start..mesh.vertices().len()));*/ + // Fill in placeholder glyph quads + let (glyph_cache, cache_tex) = self.cache.glyph_cache_mut_and_tex(); + let half_res = self.half_res; + + let brush_result = glyph_cache.process_queued( + |rect, tex_data| { + let offset = [rect.min.x as u16, rect.min.y as u16]; + let size = [rect.width() as u16, rect.height() as u16]; + + let new_data = tex_data + .iter() + .map(|x| [255, 255, 255, *x]) + .collect::>(); + + if let Err(err) = renderer.update_texture(cache_tex, offset, size, &new_data) { + log::warn!("Failed to update glyph cache texture: {:?}", err); + } + }, + // Urgh more allocation we don't need + |vertex_data| { + let uv_rect = vertex_data.tex_coords; + let uv = Aabr { + min: Vec2::new(uv_rect.min.x, uv_rect.max.y), + max: Vec2::new(uv_rect.max.x, uv_rect.min.y), + }; + let pixel_coords = vertex_data.pixel_coords; + let rect = Aabr { + min: Vec2::new( + pixel_coords.min.x as f32 / half_res.x - 1.0, + pixel_coords.min.y as f32 / half_res.y - 1.0, + ), + max: Vec2::new( + pixel_coords.max.x as f32 / half_res.x - 1.0, + pixel_coords.max.y as f32 / half_res.y - 1.0, + ), + }; + (uv, rect) + }, + ); + + match brush_result { + Ok(brush_action) => { + match brush_action { + glyph_brush::BrushAction::Draw(verts) => self.last_glyph_verts = verts, + glyph_brush::BrushAction::ReDraw => {}, + } + + let glyphs = &self.glyphs; + let mesh = &mut self.mesh; + + glyphs + .iter() + .flat_map(|(mesh_index, glyph_count, linear_color)| { + let mesh_index = *mesh_index; + let linear_color = *linear_color; + (0..*glyph_count).map(move |i| (mesh_index + i * 6, linear_color)) + }) + .zip(self.last_glyph_verts.iter()) + .for_each(|((mesh_index, linear_color), (uv, rect))| { + mesh.replace_quad( + mesh_index, + create_ui_quad(*rect, *uv, linear_color, UiMode::Text), + ) + }); + }, + Err(glyph_brush::BrushError::TextureTooSmall { suggested: (x, y) }) => { + log::error!( + "Texture to small for all glyphs, would need one of the size: ({}, {})", + x, + y + ); + }, + } + // Create a larger dynamic model if the mesh is larger than the current model // size. if self.model.vbuf.len() < self.mesh.vertices().len() { @@ -203,17 +305,23 @@ impl IcedRenderer { } // Returns (half_res, align) - fn calculate_resolution_dependents(res: Vec2) -> (Vec2, Vec2) { + fn calculate_resolution_dependents( + res: Vec2, + win_dims: Vec2, + ) -> (Vec2, Vec2, f32) { let half_res = res.map(|e| e as f32 / 2.0); let align = align(res); + // Assume to be the same in x and y for now... + let p_scale = res.x as f32 / win_dims.x; - (half_res, align) + (half_res, align, p_scale) } fn update_resolution_dependents(&mut self, res: Vec2) { - let (half_res, align) = Self::calculate_resolution_dependents(res); + let (half_res, align, p_scale) = Self::calculate_resolution_dependents(res, self.win_dims); self.half_res = half_res; self.align = align; + self.p_scale = p_scale; } fn gl_aabr(&self, bounds: iced::Rectangle) -> Aabr { @@ -349,6 +457,63 @@ impl IcedRenderer { UiMode::Geometry, )); }, + Primitive::Text { + glyphs, + bounds, // iced::Rectangle + linear_color, + /*font, + *horizontal_alignment, + *vertical_alignment, */ + } => { + self.switch_state(State::Plain); + + // TODO: Scissor? + + // TODO: makes sure we are not doing all this work for hidden text + // e.g. in chat + let glyph_cache = self.cache.glyph_cache_mut(); + + // Count glyphs + let glyph_count = glyphs.len(); + + // Queue the glyphs to be cached. + glyph_cache.queue_pre_positioned( + glyphs, + // Since we already passed in `bounds` to position the glyphs some of this + // seems redundant... + glyph_brush::rusttype::Rect { + min: glyph_brush::rusttype::Point { + x: bounds.x, + y: bounds.y, + }, + max: glyph_brush::rusttype::Point { + x: bounds.x + bounds.width, + y: bounds.y + bounds.height, + }, + }, + 0.0, // z (we don't use this) + ); + + // Leave ui and verts blank to fill in when processing cached glyphs + let zero_aabr = Aabr { + min: Vec2::broadcast(0.0), + max: Vec2::broadcast(0.0), + }; + self.glyphs + .push((self.mesh.vertices().len(), glyph_count, linear_color)); + for _ in 0..glyph_count { + // Push placeholder quad + // Note: moving to some sort of layering / z based system would be an + // alternative to this (and might help with reducing draw + // calls) + self.mesh.push_quad(create_ui_quad( + zero_aabr, + zero_aabr, + linear_color, + UiMode::Text, + )); + } + }, Primitive::Nothing => {}, } } @@ -414,24 +579,3 @@ fn default_scissor(renderer: &Renderer) -> Aabr { }, } } - -impl iced::Renderer for IcedRenderer { - // Default styling - type Defaults = (); - // TODO: use graph of primitives to enable diffing??? - type Output = (Primitive, iced::MouseCursor); - - fn layout<'a, M>( - &mut self, - element: &iced::Element<'a, M, Self>, - limits: &iced::layout::Limits, - ) -> iced::layout::Node { - let node = element.layout(self, limits); - - // Trim text measurements cache? - - node - } -} - -// TODO: impl Debugger diff --git a/voxygen/src/ui/ice/renderer/background_container.rs b/voxygen/src/ui/ice/renderer/background_container.rs index c92f4f6986..3672d8e527 100644 --- a/voxygen/src/ui/ice/renderer/background_container.rs +++ b/voxygen/src/ui/ice/renderer/background_container.rs @@ -1,7 +1,10 @@ -use super::{super::widget::background_container, IcedRenderer, Primitive}; +use super::{ + super::{cache::FrameRenderer, widget::background_container}, + Primitive, +}; use iced::{Element, Layout, Point}; -impl background_container::Renderer for IcedRenderer { +impl background_container::Renderer for FrameRenderer<'_> { fn draw( &mut self, defaults: &Self::Defaults, @@ -17,13 +20,13 @@ impl background_container::Renderer for IcedRenderer { let back_primitive = background .draw(self, defaults, background_layout, cursor_position) .0; - let (content_primitive, mouse_cursor) = + let (content_primitive, mouse_interaction) = content.draw(self, defaults, content_layout, cursor_position); ( Primitive::Group { primitives: vec![back_primitive, content_primitive], }, - mouse_cursor, + mouse_interaction, ) } } diff --git a/voxygen/src/ui/ice/renderer/column.rs b/voxygen/src/ui/ice/renderer/column.rs index ab1a7abbd2..cec3c5cc20 100644 --- a/voxygen/src/ui/ice/renderer/column.rs +++ b/voxygen/src/ui/ice/renderer/column.rs @@ -1,34 +1,34 @@ -use super::{IcedRenderer, Primitive}; -use iced::{column, Element, Layout, MouseCursor, Point}; +use super::{super::cache::FrameRenderer, Primitive}; +use iced::{column, mouse, Element, Layout, Point}; -impl column::Renderer for IcedRenderer { +impl column::Renderer for FrameRenderer<'_> { fn draw( &mut self, defaults: &Self::Defaults, - children: &[Element<'_, M, Self>], + content: &[Element<'_, M, Self>], layout: Layout<'_>, cursor_position: Point, ) -> Self::Output { - let mut mouse_cursor = MouseCursor::OutOfBounds; + let mut mouse_interaction = mouse::Interaction::default(); ( Primitive::Group { - primitives: children + primitives: content .iter() .zip(layout.children()) .map(|(child, layout)| { - let (primitive, new_mouse_cursor) = + let (primitive, new_mouse_interaction) = child.draw(self, defaults, layout, cursor_position); - if new_mouse_cursor > mouse_cursor { - mouse_cursor = new_mouse_cursor; + if new_mouse_interaction > mouse_interaction { + mouse_interaction = new_mouse_interaction; } primitive }) .collect(), }, - mouse_cursor, + mouse_interaction, ) } } diff --git a/voxygen/src/ui/ice/renderer/compound_graphic.rs b/voxygen/src/ui/ice/renderer/compound_graphic.rs index 166dbef484..ecd3763541 100644 --- a/voxygen/src/ui/ice/renderer/compound_graphic.rs +++ b/voxygen/src/ui/ice/renderer/compound_graphic.rs @@ -1,11 +1,11 @@ use super::{ - super::{widget::compound_graphic, Rotation}, - IcedRenderer, Primitive, + super::{cache::FrameRenderer, widget::compound_graphic, Rotation}, + Primitive, }; use compound_graphic::GraphicKind; -use iced::{MouseCursor, Rectangle}; +use iced::{mouse, Rectangle}; -impl compound_graphic::Renderer for IcedRenderer { +impl compound_graphic::Renderer for FrameRenderer<'_> { fn draw( &mut self, graphics: I, @@ -28,7 +28,7 @@ impl compound_graphic::Renderer for IcedRenderer { }) .collect(), }, - MouseCursor::OutOfBounds, + mouse::Interaction::default(), ) } } diff --git a/voxygen/src/ui/ice/renderer/container.rs b/voxygen/src/ui/ice/renderer/container.rs index fd144f66ce..c029704c53 100644 --- a/voxygen/src/ui/ice/renderer/container.rs +++ b/voxygen/src/ui/ice/renderer/container.rs @@ -1,7 +1,7 @@ -use super::IcedRenderer; +use super::super::cache::FrameRenderer; use iced::{container, Element, Layout, Point, Rectangle}; -impl container::Renderer for IcedRenderer { +impl container::Renderer for FrameRenderer<'_> { type Style = (); fn draw( @@ -13,10 +13,11 @@ impl container::Renderer for IcedRenderer { content: &Element<'_, M, Self>, content_layout: Layout<'_>, ) -> Self::Output { - let (content, mouse_cursor) = content.draw(self, defaults, content_layout, cursor_position); + let (content, mouse_interaction) = + content.draw(self, defaults, content_layout, cursor_position); // We may have more stuff here if styles are used - (content, mouse_cursor) + (content, mouse_interaction) } } diff --git a/voxygen/src/ui/ice/renderer/image.rs b/voxygen/src/ui/ice/renderer/image.rs index 22b3f8a953..b66a492384 100644 --- a/voxygen/src/ui/ice/renderer/image.rs +++ b/voxygen/src/ui/ice/renderer/image.rs @@ -1,13 +1,14 @@ use super::{ - super::{widget::image, Rotation}, - IcedRenderer, Primitive, + super::{cache::FrameRenderer, widget::image, Rotation}, + Primitive, }; -use iced::MouseCursor; +use iced::mouse; use vek::Rgba; -impl image::Renderer for IcedRenderer { +impl image::Renderer for FrameRenderer<'_> { fn dimensions(&self, handle: image::Handle) -> (u32, u32) { - self.cache + self.renderer + .cache .graphic_cache() .get_graphic_dims((handle, Rotation::None)) // TODO: don't unwrap @@ -26,7 +27,7 @@ impl image::Renderer for IcedRenderer { bounds: layout.bounds(), color, }, - MouseCursor::OutOfBounds, + mouse::Interaction::default(), ) } } diff --git a/voxygen/src/ui/ice/renderer/row.rs b/voxygen/src/ui/ice/renderer/row.rs index 47d22004f4..06f9395193 100644 --- a/voxygen/src/ui/ice/renderer/row.rs +++ b/voxygen/src/ui/ice/renderer/row.rs @@ -1,34 +1,34 @@ -use super::{IcedRenderer, Primitive}; -use iced::{row, Element, Layout, MouseCursor, Point}; +use super::{super::cache::FrameRenderer, Primitive}; +use iced::{mouse, row, Element, Layout, Point}; -impl row::Renderer for IcedRenderer { +impl row::Renderer for FrameRenderer<'_> { fn draw( &mut self, defaults: &Self::Defaults, - children: &[Element<'_, M, Self>], + content: &[Element<'_, M, Self>], layout: Layout<'_>, cursor_position: Point, ) -> Self::Output { - let mut mouse_cursor = MouseCursor::OutOfBounds; + let mut mouse_interaction = mouse::Interaction::default(); ( Primitive::Group { - primitives: children + primitives: content .iter() .zip(layout.children()) .map(|(child, layout)| { - let (primitive, new_mouse_cursor) = + let (primitive, new_mouse_interaction) = child.draw(self, defaults, layout, cursor_position); - if new_mouse_cursor > mouse_cursor { - mouse_cursor = new_mouse_cursor; + if new_mouse_interaction > mouse_interaction { + mouse_interaction = new_mouse_interaction; } primitive }) .collect(), }, - mouse_cursor, + mouse_interaction, ) } } diff --git a/voxygen/src/ui/ice/renderer/space.rs b/voxygen/src/ui/ice/renderer/space.rs index 8c1f898e67..bdba27a577 100644 --- a/voxygen/src/ui/ice/renderer/space.rs +++ b/voxygen/src/ui/ice/renderer/space.rs @@ -1,8 +1,8 @@ -use super::{IcedRenderer, Primitive}; -use iced::{space, MouseCursor, Rectangle}; +use super::{super::cache::FrameRenderer, Primitive}; +use iced::{mouse, space, Rectangle}; -impl space::Renderer for IcedRenderer { +impl space::Renderer for FrameRenderer<'_> { fn draw(&mut self, _bounds: Rectangle) -> Self::Output { - (Primitive::Nothing, MouseCursor::OutOfBounds) + (Primitive::Nothing, mouse::Interaction::default()) } } diff --git a/voxygen/src/ui/ice/renderer/stack.rs b/voxygen/src/ui/ice/renderer/stack.rs index 3eb77d5ad7..9f295790ca 100644 --- a/voxygen/src/ui/ice/renderer/stack.rs +++ b/voxygen/src/ui/ice/renderer/stack.rs @@ -1,34 +1,37 @@ -use super::{super::widget::stack, IcedRenderer, Primitive}; -use iced::{Element, Layout, MouseCursor, Point}; +use super::{ + super::{cache::FrameRenderer, widget::stack}, + Primitive, +}; +use iced::{mouse, Element, Layout, Point}; -impl stack::Renderer for IcedRenderer { +impl stack::Renderer for FrameRenderer<'_> { fn draw( &mut self, defaults: &Self::Defaults, - children: &[Element<'_, M, Self>], + content: &[Element<'_, M, Self>], layout: Layout<'_>, cursor_position: Point, ) -> Self::Output { - let mut mouse_cursor = MouseCursor::OutOfBounds; + let mut mouse_interaction = mouse::Interaction::default(); ( Primitive::Group { - primitives: children + primitives: content .iter() .zip(layout.children()) .map(|(child, layout)| { - let (primitive, new_mouse_cursor) = + let (primitive, new_mouse_interaction) = child.draw(self, defaults, layout, cursor_position); - if new_mouse_cursor > mouse_cursor { - mouse_cursor = new_mouse_cursor; + if new_mouse_interaction > mouse_interaction { + mouse_interaction = new_mouse_interaction; } primitive }) .collect(), }, - mouse_cursor, + mouse_interaction, ) } } diff --git a/voxygen/src/ui/ice/renderer/text.rs b/voxygen/src/ui/ice/renderer/text.rs new file mode 100644 index 0000000000..14532de358 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/text.rs @@ -0,0 +1,97 @@ +use iced::{ + text, Color, Font, HorizontalAlignment, mouse, Rectangle, Size, VerticalAlignment, +}; +use super::{super::cache::FrameRenderer, Primitive}: + +struct FontId(glyph_brush::FontId); + +impl text::Renderer for FrameRenderer<'_> { + type Font = FontId; + const DEFAULT_SIZE: u16 = 20; + + fn measure( + &self, + content: &str, + size: u16, + font: Self::Font, + bounds: Size, + ) -> (f32, f32) { + // Using the physical scale might make these cached info usable below? + // Although we also have a position of the screen so this could be useless + let p_scale = self.p_scale; + // TODO: would be nice if the method was mut + let section = glyph_brush::Section { + text: content, + scale: glyph_brush::rusttype::Scale::uniform(size as f32 * p_scale), + font_id: font.0, + bounds: (size.width, size.height), + ..Default::default() + }; + + let maybe_rect = self.glyph_calc.borrow_mut().glyph_bounds(section); + maybe_rect.map_or((0.0, 0.0), |rect| (rect.width() / p_scale, rect.height() / p_scale)) + } + + fn draw( + &mut self, + defaults: &Self::Defaults, + bounds: Rectangle, + content: &str, + size: u16, + font: Self::Font, + color: Option, + horizontal_alignment: HorizontalAlignment, + vertical_alignment: VerticalAlignment, + ) -> Self::Output { + use glyph_brush::{HorizontalAlign, VerticalAlign}; + let h_align = match horizontal_alignment { + HorizontalAlignment::Left => HorizontalAlign::Left, + HorizontalAlignment::Center => HorizontalAlign::Center, + HorizontalAlignment::Right => HorizontalAlign::Right, + }; + + let v_align = match vertical_alignment { + VerticalAlignment::Top => VerticalAlign::Top, + VerticalAlignment::Center => VerticalAlign::Center, + VerticalAlignment::Bottom => VerticalAlign::Bottom, + }; + + let p_scale = self.p_scale; + + let section = glyph_brush::Section { + text: content, + // TODO: do snap to pixel thing here IF it is being done down the line + screen_position: (bounds.x * p_scale, bounds.y * p_scale), + bounds: (bounds.width * p_scale, bounds.height * p_scale), + scale: glyph_brush::rusttype::Scale::uniform(size as f32 * p_scale), + layout: glyph_brush::Layout::Wrap { + line_breaker: Default::default(), + h_align, + v_align, + }, + font_id: font.0, + ..Default::default() + }; + + let glyphs = self.glyph_calc.borrow_mut().glyphs(section).map(|positioned_glyph| + ( + positioned_glyph, + [0.0, 0.0, 0.0, 1.0], // Color + font.0, + ) + ).collect(); + + ( + Primitive::Text { + glyphs, + //size: size as f32, + bounds, + color: color.unwrap_or(Color::BLACK).into_linear().into(), + //font, + //horizontal_alignment, + //vertical_alignment, + }, + mouse::Interaction::default, + ) + } +} diff --git a/voxygen/src/ui/ice/winit_conversion.rs b/voxygen/src/ui/ice/winit_conversion.rs index 79538094ff..4b69e6ded1 100644 --- a/voxygen/src/ui/ice/winit_conversion.rs +++ b/voxygen/src/ui/ice/winit_conversion.rs @@ -1,10 +1,7 @@ // Using reference impl: https://github.com/hecrj/iced/blob/e1438774af809c2951c4c7446638500446c81111/winit/src/conversion.rs use iced::{ - input::{ - keyboard::{self, KeyCode, ModifiersState}, - mouse, ButtonState, - }, - window, Event, + keyboard::{self, KeyCode, ModifiersState}, + mouse, window, Event, }; /// Converts a winit event into an iced event. @@ -28,10 +25,14 @@ pub fn window_event(event: winit::WindowEvent) -> Option { y: position.y as f32, })) }, - WindowEvent::MouseInput { button, state, .. } => Some(Event::Mouse(mouse::Event::Input { - button: mouse_button(button), - state: button_state(state), - })), + WindowEvent::MouseInput { button, state, .. } => { + let button = mouse_button(button); + + Some(Event::Mouse(match state { + winit::ElementState::Pressed => mouse::Event::ButtonPressed(button), + winit::ElementState::Released => mouse::Event::ButtonReleased(button), + })) + }, WindowEvent::MouseWheel { delta, .. } => match delta { winit::MouseScrollDelta::LineDelta(delta_x, delta_y) => { Some(Event::Mouse(mouse::Event::WheelScrolled { @@ -63,10 +64,20 @@ pub fn window_event(event: winit::WindowEvent) -> Option { .. }, .. - } => Some(Event::Keyboard(keyboard::Event::Input { - key_code: key_code(virtual_keycode), - state: button_state(state), - modifiers: modifiers_state(modifiers), + } => Some(Event::Keyboard({ + let key_code = key_code(virtual_keycode); + let modifiers = modifiers_state(modifiers); + + match state { + winit::ElementState::Pressed => keyboard::Event::KeyPressed { + key_code, + modifiers, + }, + winit::ElementState::Released => keyboard::Event::KeyReleased { + key_code, + modifiers, + }, + } })), // iced also can use file hovering events but we don't need them right now _ => None, @@ -85,14 +96,6 @@ fn mouse_button(mouse_button: winit::MouseButton) -> mouse::Button { } } -/// Converts winit `ElementState` to an iced button state -fn button_state(element_state: winit::ElementState) -> ButtonState { - match element_state { - winit::ElementState::Pressed => ButtonState::Pressed, - winit::ElementState::Released => ButtonState::Released, - } -} - /// Converts winit `ModifiersState` to iced `ModifiersState` fn modifiers_state(modifiers: winit::ModifiersState) -> ModifiersState { ModifiersState { From 9dbabff887c2629304cdc19f7c713f6bfc928c3c Mon Sep 17 00:00:00 2001 From: Imbris Date: Mon, 18 May 2020 00:17:25 -0400 Subject: [PATCH 09/61] Add for<'b> lifetime annotation (almost worked) --- voxygen/src/ui/ice/mod.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index 4fe0d7cf83..437336e19e 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -100,7 +100,7 @@ impl IcedUi { // TODO: produce root internally??? // TODO: see if this lifetime soup can be simplified - pub fn maintain<'a, 'b, M, E: Into>>>( + pub fn maintain<'a, M, E: for<'b> Into>>>( &mut self, root: E, renderer: &mut Renderer, From bd694c5a810697dd60c9b69f5002e755aa630b21 Mon Sep 17 00:00:00 2001 From: Imbris Date: Mon, 18 May 2020 02:13:13 -0400 Subject: [PATCH 10/61] Text rendering for iced with glyph_brush works now --- voxygen/src/menu/main/ui.rs | 9 +- voxygen/src/ui/ice/cache.rs | 96 ++++--------------- voxygen/src/ui/ice/mod.rs | 23 +++-- voxygen/src/ui/ice/renderer.rs | 43 +++++++-- .../ui/ice/renderer/background_container.rs | 7 +- voxygen/src/ui/ice/renderer/column.rs | 4 +- .../src/ui/ice/renderer/compound_graphic.rs | 6 +- voxygen/src/ui/ice/renderer/container.rs | 4 +- voxygen/src/ui/ice/renderer/image.rs | 8 +- voxygen/src/ui/ice/renderer/row.rs | 4 +- voxygen/src/ui/ice/renderer/space.rs | 4 +- voxygen/src/ui/ice/renderer/stack.rs | 7 +- voxygen/src/ui/ice/renderer/text.rs | 62 ++++++------ 13 files changed, 119 insertions(+), 158 deletions(-) diff --git a/voxygen/src/menu/main/ui.rs b/voxygen/src/menu/main/ui.rs index a01ff43b15..49babfc263 100644 --- a/voxygen/src/menu/main/ui.rs +++ b/voxygen/src/menu/main/ui.rs @@ -191,7 +191,7 @@ struct IcedState { pub type Message = Event; impl IcedState { pub fn view(&mut self) -> Element { - use iced::{Align, Column, Container, Length, Row, Space}; + use iced::{Align, Column, Container, Length, Row, Space, Text}; use ui::ice::{ compound_graphic::{CompoundGraphic, Graphic}, BackgroundContainer, Image, Padding, @@ -201,7 +201,12 @@ impl IcedState { let buttons = Column::with_children(vec![ Image::new(self.imgs.button).fix_aspect_ratio().into(), Image::new(self.imgs.button).fix_aspect_ratio().into(), - Image::new(self.imgs.button).fix_aspect_ratio().into(), + /* BackgroundContainer::new( + Image::new(self.imgs.button).fix_aspect_ratio(), + Text::new("Quit"), + ) + .into(), */ + Text::new("Quit").size(20).into(), ]) .width(Length::Fill) .max_width(200) diff --git a/voxygen/src/ui/ice/cache.rs b/voxygen/src/ui/ice/cache.rs index c3d9e70f06..e061e6c2b5 100644 --- a/voxygen/src/ui/ice/cache.rs +++ b/voxygen/src/ui/ice/cache.rs @@ -1,15 +1,10 @@ -use super::{ - graphic::{Graphic, GraphicCache, Id as GraphicId}, - renderer::{IcedRenderer, Primitive}, -}; +use super::graphic::{Graphic, GraphicCache, Id as GraphicId}; use crate::{ render::{Renderer, Texture}, Error, }; -use glyph_brush::{ - GlyphBrushBuilder, GlyphCalculator, GlyphCalculatorBuilder, GlyphCalculatorGuard, -}; -use std::cell::RefCell; +use glyph_brush::GlyphBrushBuilder; +use std::cell::{RefCell, RefMut}; use vek::*; // Multiplied by current window size @@ -23,7 +18,7 @@ type GlyphBrush = glyph_brush::GlyphBrush<'static, (Aabr, Aabr)>; pub type Font = glyph_brush::rusttype::Font<'static>; pub struct Cache { - glyph_cache: GlyphBrush, + glyph_brush: RefCell, glyph_cache_tex: Texture, graphic_cache: GraphicCache, } @@ -38,12 +33,14 @@ impl Cache { let glyph_cache_dims = Vec2::new(w, h).map(|e| (e * GLYPH_CACHE_SIZE).min(max_texture_size as u16).max(512)); + let glyph_brush = GlyphBrushBuilder::using_font(default_font) + .initial_cache_size((glyph_cache_dims.x as u32, glyph_cache_dims.y as u32)) + .gpu_cache_scale_tolerance(SCALE_TOLERANCE) + .gpu_cache_position_tolerance(POSITION_TOLERANCE) + .build(); + Ok(Self { - glyph_cache: GlyphBrushBuilder::using_font(default_font) - .initial_cache_size((glyph_cache_dims.x as u32, glyph_cache_dims.y as u32)) - .gpu_cache_scale_tolerance(SCALE_TOLERANCE) - .gpu_cache_position_tolerance(POSITION_TOLERANCE) - .build(), + glyph_brush: RefCell::new(glyph_brush), glyph_cache_tex: renderer.create_dynamic_texture(glyph_cache_dims.map(|e| e as u16))?, graphic_cache: GraphicCache::new(renderer), }) @@ -52,10 +49,12 @@ impl Cache { pub fn glyph_cache_tex(&self) -> &Texture { &self.glyph_cache_tex } pub fn glyph_cache_mut_and_tex(&mut self) -> (&mut GlyphBrush, &Texture) { - (&mut self.glyph_cache, &self.glyph_cache_tex) + (self.glyph_brush.get_mut(), &self.glyph_cache_tex) } - pub fn glyph_cache_mut(&mut self) -> &mut GlyphBrush { &mut self.glyph_cache } + pub fn glyph_cache_mut(&mut self) -> &mut GlyphBrush { self.glyph_brush.get_mut() } + + pub fn glyph_calculator(&self) -> RefMut { self.glyph_brush.borrow_mut() } // TODO: add font fn @@ -82,72 +81,13 @@ impl Cache { let cache_dims = renderer .get_resolution() .map(|e| (e * GLYPH_CACHE_SIZE).min(max_texture_size as u16).max(512)); - self.glyph_cache = self - .glyph_cache + let glyph_brush = self.glyph_brush.get_mut(); + *glyph_brush = glyph_brush .to_builder() .initial_cache_size((cache_dims.x as u32, cache_dims.y as u32)) .build(); + self.glyph_cache_tex = renderer.create_dynamic_texture(cache_dims.map(|e| e as u16))?; Ok(()) } } - -pub struct GlyphCalcCache { - // Hold one of these for adding new fonts - builder: GlyphCalculatorBuilder<'static>, - calculator: GlyphCalculator<'static>, -} - -impl GlyphCalcCache { - pub fn new(default_font: Font) -> Self { - // Multiple copies of the font in memory :/ (using Arc<[u8]> might help) - let builder = GlyphCalculatorBuilder::using_font(default_font); - let calculator = builder.clone().build(); - - Self { - builder, - calculator, - } - } - - pub fn frame_guard<'a>(&'a self) -> GlyphCalculatorGuard<'a, 'static> { - self.calculator.cache_scope() - } - - // add new font fn -} - -pub struct FrameRenderer<'a> { - pub renderer: &'a mut IcedRenderer, - pub glyph_calc: RefCell>, -} - -impl<'a> FrameRenderer<'a> { - pub fn new(renderer: &'a mut IcedRenderer, glyph_calc_cache: &'a mut GlyphCalcCache) -> Self { - Self { - renderer, - glyph_calc: RefCell::new(glyph_calc_cache.frame_guard()), - } - } -} - -impl iced::Renderer for FrameRenderer<'_> { - // Default styling - type Defaults = (); - // TODO: use graph of primitives to enable diffing??? - type Output = (Primitive, iced::mouse::Interaction); - - fn layout<'a, M>( - &mut self, - element: &iced::Element<'a, M, Self>, - limits: &iced::layout::Limits, - ) -> iced::layout::Node { - let node = element.layout(self, limits); - - // Trim text measurements cache? - - node - } -} - -// TODO: impl Debugger diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index 437336e19e..7d24651b5c 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -21,16 +21,17 @@ use super::{ scale::{Scale, ScaleMode}, }; use crate::{render::Renderer, window::Window, Error}; -use cache::{FrameRenderer, GlyphCalcCache}; use clipboard::Clipboard; use iced::{mouse, Cache, Size, UserInterface}; use vek::*; -pub type Element<'a, 'b, M> = iced::Element<'a, M, FrameRenderer<'b>>; +pub type Element<'a, M> = iced::Element<'a, M, IcedRenderer>; + +#[derive(Clone, Copy, Default)] +pub struct FontId(glyph_brush::FontId); pub struct IcedUi { renderer: IcedRenderer, - glyph_calc_cache: GlyphCalcCache, cache: Option, events: Vec, clipboard: Clipboard, @@ -47,8 +48,7 @@ impl IcedUi { // TODO: examine how much mem fonts take up and reduce clones if significant Ok(Self { - renderer: IcedRenderer::new(renderer, scaled_dims, default_font.clone())?, - glyph_calc_cache: GlyphCalcCache::new(default_font), + renderer: IcedRenderer::new(renderer, scaled_dims, default_font)?, cache: Some(Cache::new()), events: Vec::new(), // TODO: handle None @@ -99,8 +99,9 @@ impl IcedUi { } // TODO: produce root internally??? - // TODO: see if this lifetime soup can be simplified - pub fn maintain<'a, M, E: for<'b> Into>>>( + // TODO: closure/trait for sending messages back? (take a look at higher level + // iced libs) + pub fn maintain<'a, M, E: Into>>( &mut self, root: E, renderer: &mut Renderer, @@ -133,22 +134,20 @@ impl IcedUi { // TODO: convert to f32 at source let window_size = self.scale.scaled_window_size().map(|e| e as f32); - let mut frame_renderer = FrameRenderer::new(&mut self.renderer, &mut self.glyph_calc_cache); - let mut user_interface = UserInterface::build( root, Size::new(window_size.x, window_size.y), self.cache.take().unwrap(), - &mut frame_renderer, + &mut self.renderer, ); let messages = user_interface.update( self.events.drain(..), Some(&self.clipboard), - &frame_renderer, + &mut self.renderer, ); - let (primitive, mouse_interaction) = user_interface.draw(&mut frame_renderer); + let (primitive, mouse_interaction) = user_interface.draw(&mut self.renderer); self.cache = Some(user_interface.into_cache()); diff --git a/voxygen/src/ui/ice/renderer.rs b/voxygen/src/ui/ice/renderer.rs index f94581d4d4..04d7f01830 100644 --- a/voxygen/src/ui/ice/renderer.rs +++ b/voxygen/src/ui/ice/renderer.rs @@ -5,6 +5,7 @@ mod container; mod image; mod row; mod space; +mod text; use super::{ super::graphic::{self, Graphic, TexId}, @@ -189,8 +190,8 @@ impl IcedRenderer { // Draw glyph cache (use for debugging). /*self.draw_commands .push(DrawCommand::Scissor(default_scissor(renderer))); - start = mesh.vertices().len(); - mesh.push_quad(create_ui_quad( + self.start = self.mesh.vertices().len(); + self.mesh.push_quad(create_ui_quad( Aabr { min: (-1.0, -1.0).into(), max: (1.0, 1.0).into(), @@ -199,11 +200,11 @@ impl IcedRenderer { min: (0.0, 1.0).into(), max: (1.0, 0.0).into(), }, - Rgba::new(1.0, 1.0, 1.0, 0.8), + Rgba::new(1.0, 1.0, 1.0, 0.3), UiMode::Text, )); self.draw_commands - .push(DrawCommand::plain(start..mesh.vertices().len()));*/ + .push(DrawCommand::plain(self.start..self.mesh.vertices().len()));*/ // Fill in placeholder glyph quads let (glyph_cache, cache_tex) = self.cache.glyph_cache_mut_and_tex(); @@ -234,11 +235,11 @@ impl IcedRenderer { let rect = Aabr { min: Vec2::new( pixel_coords.min.x as f32 / half_res.x - 1.0, - pixel_coords.min.y as f32 / half_res.y - 1.0, + 1.0 - pixel_coords.max.y as f32 / half_res.y, ), max: Vec2::new( pixel_coords.max.x as f32 / half_res.x - 1.0, - pixel_coords.max.y as f32 / half_res.y - 1.0, + 1.0 - pixel_coords.min.y as f32 / half_res.y, ), }; (uv, rect) @@ -483,12 +484,13 @@ impl IcedRenderer { // seems redundant... glyph_brush::rusttype::Rect { min: glyph_brush::rusttype::Point { - x: bounds.x, - y: bounds.y, + x: bounds.x * self.p_scale, + //y: (self.win_dims.y - bounds.y) * self.p_scale, + y: bounds.y * self.p_scale, }, max: glyph_brush::rusttype::Point { - x: bounds.x + bounds.width, - y: bounds.y + bounds.height, + x: (bounds.x + bounds.width) * self.p_scale, + y: (bounds.y + bounds.height) * self.p_scale, }, }, 0.0, // z (we don't use this) @@ -579,3 +581,24 @@ fn default_scissor(renderer: &Renderer) -> Aabr { }, } } + +impl iced::Renderer for IcedRenderer { + // Default styling + type Defaults = (); + // TODO: use graph of primitives to enable diffing??? + type Output = (Primitive, iced::mouse::Interaction); + + fn layout<'a, M>( + &mut self, + element: &iced::Element<'a, M, Self>, + limits: &iced::layout::Limits, + ) -> iced::layout::Node { + let node = element.layout(self, limits); + + // Trim text measurements cache? + + node + } +} + +// TODO: impl Debugger diff --git a/voxygen/src/ui/ice/renderer/background_container.rs b/voxygen/src/ui/ice/renderer/background_container.rs index 3672d8e527..7b90f5b3c0 100644 --- a/voxygen/src/ui/ice/renderer/background_container.rs +++ b/voxygen/src/ui/ice/renderer/background_container.rs @@ -1,10 +1,7 @@ -use super::{ - super::{cache::FrameRenderer, widget::background_container}, - Primitive, -}; +use super::{super::widget::background_container, IcedRenderer, Primitive}; use iced::{Element, Layout, Point}; -impl background_container::Renderer for FrameRenderer<'_> { +impl background_container::Renderer for IcedRenderer { fn draw( &mut self, defaults: &Self::Defaults, diff --git a/voxygen/src/ui/ice/renderer/column.rs b/voxygen/src/ui/ice/renderer/column.rs index cec3c5cc20..5a6a33bcc3 100644 --- a/voxygen/src/ui/ice/renderer/column.rs +++ b/voxygen/src/ui/ice/renderer/column.rs @@ -1,7 +1,7 @@ -use super::{super::cache::FrameRenderer, Primitive}; +use super::{IcedRenderer, Primitive}; use iced::{column, mouse, Element, Layout, Point}; -impl column::Renderer for FrameRenderer<'_> { +impl column::Renderer for IcedRenderer { fn draw( &mut self, defaults: &Self::Defaults, diff --git a/voxygen/src/ui/ice/renderer/compound_graphic.rs b/voxygen/src/ui/ice/renderer/compound_graphic.rs index ecd3763541..65ef251834 100644 --- a/voxygen/src/ui/ice/renderer/compound_graphic.rs +++ b/voxygen/src/ui/ice/renderer/compound_graphic.rs @@ -1,11 +1,11 @@ use super::{ - super::{cache::FrameRenderer, widget::compound_graphic, Rotation}, - Primitive, + super::{widget::compound_graphic, Rotation}, + IcedRenderer, Primitive, }; use compound_graphic::GraphicKind; use iced::{mouse, Rectangle}; -impl compound_graphic::Renderer for FrameRenderer<'_> { +impl compound_graphic::Renderer for IcedRenderer { fn draw( &mut self, graphics: I, diff --git a/voxygen/src/ui/ice/renderer/container.rs b/voxygen/src/ui/ice/renderer/container.rs index c029704c53..cd8a3c2748 100644 --- a/voxygen/src/ui/ice/renderer/container.rs +++ b/voxygen/src/ui/ice/renderer/container.rs @@ -1,7 +1,7 @@ -use super::super::cache::FrameRenderer; +use super::IcedRenderer; use iced::{container, Element, Layout, Point, Rectangle}; -impl container::Renderer for FrameRenderer<'_> { +impl container::Renderer for IcedRenderer { type Style = (); fn draw( diff --git a/voxygen/src/ui/ice/renderer/image.rs b/voxygen/src/ui/ice/renderer/image.rs index b66a492384..710bf386b7 100644 --- a/voxygen/src/ui/ice/renderer/image.rs +++ b/voxygen/src/ui/ice/renderer/image.rs @@ -1,13 +1,13 @@ use super::{ - super::{cache::FrameRenderer, widget::image, Rotation}, - Primitive, + super::{widget::image, Rotation}, + IcedRenderer, Primitive, }; use iced::mouse; use vek::Rgba; -impl image::Renderer for FrameRenderer<'_> { +impl image::Renderer for IcedRenderer { fn dimensions(&self, handle: image::Handle) -> (u32, u32) { - self.renderer + self .cache .graphic_cache() .get_graphic_dims((handle, Rotation::None)) diff --git a/voxygen/src/ui/ice/renderer/row.rs b/voxygen/src/ui/ice/renderer/row.rs index 06f9395193..b46747cdea 100644 --- a/voxygen/src/ui/ice/renderer/row.rs +++ b/voxygen/src/ui/ice/renderer/row.rs @@ -1,7 +1,7 @@ -use super::{super::cache::FrameRenderer, Primitive}; +use super::{IcedRenderer, Primitive}; use iced::{mouse, row, Element, Layout, Point}; -impl row::Renderer for FrameRenderer<'_> { +impl row::Renderer for IcedRenderer { fn draw( &mut self, defaults: &Self::Defaults, diff --git a/voxygen/src/ui/ice/renderer/space.rs b/voxygen/src/ui/ice/renderer/space.rs index bdba27a577..309d216209 100644 --- a/voxygen/src/ui/ice/renderer/space.rs +++ b/voxygen/src/ui/ice/renderer/space.rs @@ -1,7 +1,7 @@ -use super::{super::cache::FrameRenderer, Primitive}; +use super::{IcedRenderer, Primitive}; use iced::{mouse, space, Rectangle}; -impl space::Renderer for FrameRenderer<'_> { +impl space::Renderer for IcedRenderer { fn draw(&mut self, _bounds: Rectangle) -> Self::Output { (Primitive::Nothing, mouse::Interaction::default()) } diff --git a/voxygen/src/ui/ice/renderer/stack.rs b/voxygen/src/ui/ice/renderer/stack.rs index 9f295790ca..f94f4a2f6c 100644 --- a/voxygen/src/ui/ice/renderer/stack.rs +++ b/voxygen/src/ui/ice/renderer/stack.rs @@ -1,10 +1,7 @@ -use super::{ - super::{cache::FrameRenderer, widget::stack}, - Primitive, -}; +use super::{super::widget::stack, IcedRenderer, Primitive}; use iced::{mouse, Element, Layout, Point}; -impl stack::Renderer for FrameRenderer<'_> { +impl stack::Renderer for IcedRenderer { fn draw( &mut self, defaults: &Self::Defaults, diff --git a/voxygen/src/ui/ice/renderer/text.rs b/voxygen/src/ui/ice/renderer/text.rs index 14532de358..a8067c860c 100644 --- a/voxygen/src/ui/ice/renderer/text.rs +++ b/voxygen/src/ui/ice/renderer/text.rs @@ -1,21 +1,13 @@ -use iced::{ - text, Color, Font, HorizontalAlignment, mouse, Rectangle, Size, VerticalAlignment, -}; -use super::{super::cache::FrameRenderer, Primitive}: +use super::{super::FontId, IcedRenderer, Primitive}; +use glyph_brush::GlyphCruncher; +use iced::{mouse, text, Color, HorizontalAlignment, Rectangle, Size, VerticalAlignment}; -struct FontId(glyph_brush::FontId); - -impl text::Renderer for FrameRenderer<'_> { +impl text::Renderer for IcedRenderer { type Font = FontId; + const DEFAULT_SIZE: u16 = 20; - fn measure( - &self, - content: &str, - size: u16, - font: Self::Font, - bounds: Size, - ) -> (f32, f32) { + fn measure(&self, content: &str, size: u16, font: Self::Font, bounds: Size) -> (f32, f32) { // Using the physical scale might make these cached info usable below? // Although we also have a position of the screen so this could be useless let p_scale = self.p_scale; @@ -24,17 +16,19 @@ impl text::Renderer for FrameRenderer<'_> { text: content, scale: glyph_brush::rusttype::Scale::uniform(size as f32 * p_scale), font_id: font.0, - bounds: (size.width, size.height), + bounds: (bounds.width * p_scale, bounds.height * p_scale), ..Default::default() }; - let maybe_rect = self.glyph_calc.borrow_mut().glyph_bounds(section); - maybe_rect.map_or((0.0, 0.0), |rect| (rect.width() / p_scale, rect.height() / p_scale)) + let maybe_rect = self.cache.glyph_calculator().glyph_bounds(section); + maybe_rect.map_or((0.0, 0.0), |rect| { + (rect.width() / p_scale, rect.height() / p_scale) + }) } fn draw( &mut self, - defaults: &Self::Defaults, + _defaults: &Self::Defaults, bounds: Rectangle, content: &str, size: u16, @@ -61,37 +55,43 @@ impl text::Renderer for FrameRenderer<'_> { let section = glyph_brush::Section { text: content, // TODO: do snap to pixel thing here IF it is being done down the line + //screen_position: (bounds.x * p_scale, (self.win_dims.y - bounds.y) * p_scale), screen_position: (bounds.x * p_scale, bounds.y * p_scale), bounds: (bounds.width * p_scale, bounds.height * p_scale), scale: glyph_brush::rusttype::Scale::uniform(size as f32 * p_scale), layout: glyph_brush::Layout::Wrap { line_breaker: Default::default(), - h_align, + h_align, v_align, }, font_id: font.0, ..Default::default() }; - let glyphs = self.glyph_calc.borrow_mut().glyphs(section).map(|positioned_glyph| - ( - positioned_glyph, - [0.0, 0.0, 0.0, 1.0], // Color - font.0, - ) - ).collect(); + let glyphs = self + .cache + .glyph_cache_mut() + .glyphs(section) + .map(|positioned_glyph| { + ( + positioned_glyph.clone(), // :/ + [0.0, 0.0, 0.0, 1.0], // Color + font.0, + ) + }) + .collect::>(); ( Primitive::Text { glyphs, //size: size as f32, bounds, - color: color.unwrap_or(Color::BLACK).into_linear().into(), - //font, - //horizontal_alignment, - //vertical_alignment, + linear_color: color.unwrap_or(Color::BLACK).into_linear().into(), + /*font, + *horizontal_alignment, + *vertical_alignment, */ }, - mouse::Interaction::default, + mouse::Interaction::default(), ) } } From 04a56d6ec74b0ecbd45559de14bd95c328e49c02 Mon Sep 17 00:00:00 2001 From: Imbris Date: Mon, 25 May 2020 14:11:39 -0400 Subject: [PATCH 11/61] Rename localization/font types (e.g. VoxygenLocalization -> Localization) --- Cargo.lock | 2 +- assets/voxygen/i18n/PL.ron | 2 +- assets/voxygen/i18n/de_DE.ron | 2 +- assets/voxygen/i18n/en.ron | 2 +- assets/voxygen/i18n/es_ES.ron | 2 +- assets/voxygen/i18n/es_la.ron | 2 +- assets/voxygen/i18n/fr_FR.ron | 2 +- assets/voxygen/i18n/it_IT.ron | 2 +- assets/voxygen/i18n/nl.ron | 2 +- assets/voxygen/i18n/pt_BR.ron | 2 +- assets/voxygen/i18n/pt_PT.ron | 2 +- assets/voxygen/i18n/ru_RU.ron | 2 +- assets/voxygen/i18n/sv.ron | 2 +- assets/voxygen/i18n/tr_TR.ron | 2 +- assets/voxygen/i18n/zh_CN.ron | 2 +- assets/voxygen/i18n/zh_TW.ron | 2 +- voxygen/src/hud/bag.rs | 12 ++-- voxygen/src/hud/buffs.rs | 12 ++-- voxygen/src/hud/buttons.rs | 12 ++-- voxygen/src/hud/chat.rs | 17 ++--- voxygen/src/hud/crafting.rs | 12 ++-- voxygen/src/hud/esc_menu.rs | 12 ++-- voxygen/src/hud/group.rs | 12 ++-- voxygen/src/hud/map.rs | 12 ++-- voxygen/src/hud/minimap.rs | 6 +- voxygen/src/hud/mod.rs | 83 ++++++++++------------ voxygen/src/hud/overhead.rs | 19 +++-- voxygen/src/hud/overitem.rs | 6 +- voxygen/src/hud/popup.rs | 14 ++-- voxygen/src/hud/settings_window.rs | 12 ++-- voxygen/src/hud/skillbar.rs | 12 ++-- voxygen/src/hud/social.rs | 12 ++-- voxygen/src/hud/spell.rs | 10 +-- voxygen/src/i18n.rs | 30 ++++---- voxygen/src/main.rs | 6 +- voxygen/src/menu/char_selection/mod.rs | 4 +- voxygen/src/menu/char_selection/ui.rs | 96 +++++++++++++------------- voxygen/src/menu/main/mod.rs | 2 +- voxygen/src/menu/main/ui.rs | 88 ++++++++++++++--------- voxygen/src/session.rs | 30 ++++---- voxygen/src/ui/fonts.rs | 20 +++--- 41 files changed, 292 insertions(+), 291 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index d274d777a5..d6cde5f45c 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1864,7 +1864,7 @@ checksum = "5fca6f9d679bff1322c76c9a1ad4b8553b30a94f3f75bea6936e19032c2f2ec3" dependencies = [ "glyph_brush_layout", "log", - "ordered-float", + "ordered-float 1.1.0", "rustc-hash", "rusttype 0.8.3", "twox-hash", diff --git a/assets/voxygen/i18n/PL.ron b/assets/voxygen/i18n/PL.ron index 49b4d3173d..5069802cdc 100644 --- a/assets/voxygen/i18n/PL.ron +++ b/assets/voxygen/i18n/PL.ron @@ -1,5 +1,5 @@ /// Localization for Polish / Tłumaczenia dla języka polskiego -VoxygenLocalization( +Localization( metadata: ( language_name: "Polish", language_identifier: "PL", diff --git a/assets/voxygen/i18n/de_DE.ron b/assets/voxygen/i18n/de_DE.ron index a4f8b45b09..3fdd70ba3e 100644 --- a/assets/voxygen/i18n/de_DE.ron +++ b/assets/voxygen/i18n/de_DE.ron @@ -11,7 +11,7 @@ /// `assets/voxygen/i18n` and that's it! /// Lokalisation für Deutsch/Deutschland -VoxygenLocalization( +( metadata: ( language_name: "Deutsch", language_identifier: "de_DE", diff --git a/assets/voxygen/i18n/en.ron b/assets/voxygen/i18n/en.ron index daab2aa0ea..5ef6d90d71 100644 --- a/assets/voxygen/i18n/en.ron +++ b/assets/voxygen/i18n/en.ron @@ -13,7 +13,7 @@ /// WARNING: Localization files shall be saved in UTF-8 format without BOM /// Localization for "global" English -VoxygenLocalization( +( metadata: ( language_name: "English", language_identifier: "en", diff --git a/assets/voxygen/i18n/es_ES.ron b/assets/voxygen/i18n/es_ES.ron index 5c608525df..57a2dd3356 100644 --- a/assets/voxygen/i18n/es_ES.ron +++ b/assets/voxygen/i18n/es_ES.ron @@ -12,7 +12,7 @@ /// /// Localization for Spanish (Spain) -VoxygenLocalization( +Localization( metadata: ( language_name: "Español de España", language_identifier: "es_ES", diff --git a/assets/voxygen/i18n/es_la.ron b/assets/voxygen/i18n/es_la.ron index ffdb904665..b15f416386 100644 --- a/assets/voxygen/i18n/es_la.ron +++ b/assets/voxygen/i18n/es_la.ron @@ -13,7 +13,7 @@ /// WARNING: Localization files shall be saved in UTF-8 format without BOM /// Localization for "latinoamericano" Latin-American -VoxygenLocalization( +Localization( metadata: ( language_name: "Español Latino", language_identifier: "es_la", diff --git a/assets/voxygen/i18n/fr_FR.ron b/assets/voxygen/i18n/fr_FR.ron index dac58d4d68..4749bfe9d9 100644 --- a/assets/voxygen/i18n/fr_FR.ron +++ b/assets/voxygen/i18n/fr_FR.ron @@ -1,5 +1,5 @@ /// Localization for French (France locale) -VoxygenLocalization( +( metadata: ( language_name: "Français", language_identifier: "fr_FR", diff --git a/assets/voxygen/i18n/it_IT.ron b/assets/voxygen/i18n/it_IT.ron index 74a92c6d8b..d469582d34 100644 --- a/assets/voxygen/i18n/it_IT.ron +++ b/assets/voxygen/i18n/it_IT.ron @@ -14,7 +14,7 @@ /// Localization for "global" Italian -VoxygenLocalization( +( metadata: ( language_name: "Italiano", language_identifier: "it_IT", diff --git a/assets/voxygen/i18n/nl.ron b/assets/voxygen/i18n/nl.ron index 6c46761463..ed2c33f119 100644 --- a/assets/voxygen/i18n/nl.ron +++ b/assets/voxygen/i18n/nl.ron @@ -13,7 +13,7 @@ /// WARNING: Localization files shall be saved in UTF-8 format without BOM /// Localization for "global" English -VoxygenLocalization( +( metadata: ( language_name: "Nederlands", language_identifier: "nl", diff --git a/assets/voxygen/i18n/pt_BR.ron b/assets/voxygen/i18n/pt_BR.ron index dcd188b706..6918614aca 100644 --- a/assets/voxygen/i18n/pt_BR.ron +++ b/assets/voxygen/i18n/pt_BR.ron @@ -1,5 +1,5 @@ /// Localization for Portuguese (Brazil) -VoxygenLocalization( +Localization( metadata: ( language_name: "Português Brasileiro", language_identifier: "pt_BR", diff --git a/assets/voxygen/i18n/pt_PT.ron b/assets/voxygen/i18n/pt_PT.ron index 350d69377f..55f3420d25 100644 --- a/assets/voxygen/i18n/pt_PT.ron +++ b/assets/voxygen/i18n/pt_PT.ron @@ -1,5 +1,5 @@ /// Localization for portuguese (Portugal) -VoxygenLocalization( +( metadata: ( language_name: "Português", language_identifier: "pt_PT", diff --git a/assets/voxygen/i18n/ru_RU.ron b/assets/voxygen/i18n/ru_RU.ron index 8bc8479551..1269805096 100644 --- a/assets/voxygen/i18n/ru_RU.ron +++ b/assets/voxygen/i18n/ru_RU.ron @@ -1,5 +1,5 @@ /// Localization for "global" Russian -VoxygenLocalization( +( metadata: ( language_name: "Русский", language_identifier: "ru_RU", diff --git a/assets/voxygen/i18n/sv.ron b/assets/voxygen/i18n/sv.ron index f074769680..f1198c0e93 100644 --- a/assets/voxygen/i18n/sv.ron +++ b/assets/voxygen/i18n/sv.ron @@ -11,7 +11,7 @@ /// `assets/voxygen/i18n` and that's it! /// Localization for Swedish -VoxygenLocalization( +Localization( metadata: ( language_name: "Svenska", language_identifier: "sv", diff --git a/assets/voxygen/i18n/tr_TR.ron b/assets/voxygen/i18n/tr_TR.ron index 837b0ed7dc..15cd2e32b4 100644 --- a/assets/voxygen/i18n/tr_TR.ron +++ b/assets/voxygen/i18n/tr_TR.ron @@ -13,7 +13,7 @@ /// WARNING: Localization files shall be saved in UTF-8 format without BOM /// Localization for Turkish (Turkey) -VoxygenLocalization( +( metadata: ( language_name: "Türkçe (Türkiye)", language_identifier: "tr_TR", diff --git a/assets/voxygen/i18n/zh_CN.ron b/assets/voxygen/i18n/zh_CN.ron index a19f83765a..40622d9e0c 100644 --- a/assets/voxygen/i18n/zh_CN.ron +++ b/assets/voxygen/i18n/zh_CN.ron @@ -13,7 +13,7 @@ /// 注意: 本地化文件应以 UTF-8无BOM 格式保存 /// "全局"本地化 Simplified Chinese-简体中文 -VoxygenLocalization( +Localization( metadata: ( language_name: "Simplified Chinese", language_identifier: "zh_CN", diff --git a/assets/voxygen/i18n/zh_TW.ron b/assets/voxygen/i18n/zh_TW.ron index 657b2ba0fc..4d507c1a71 100644 --- a/assets/voxygen/i18n/zh_TW.ron +++ b/assets/voxygen/i18n/zh_TW.ron @@ -1,5 +1,5 @@ /// Localization for Traditional Chinese -VoxygenLocalization( +Localization( metadata: ( language_name: "繁體中文", language_identifier: "zh_TW", diff --git a/voxygen/src/hud/bag.rs b/voxygen/src/hud/bag.rs index f1252c9f0d..0e535cca69 100644 --- a/voxygen/src/hud/bag.rs +++ b/voxygen/src/hud/bag.rs @@ -8,9 +8,9 @@ use super::{ }; use crate::{ hud::get_quality_col, - i18n::VoxygenLocalization, + i18n::Localization, ui::{ - fonts::ConrodVoxygenFonts, + fonts::Fonts, slot::{ContentSize, SlotMaker}, ImageFrame, Tooltip, TooltipManager, Tooltipable, }, @@ -90,14 +90,14 @@ pub struct Bag<'a> { client: &'a Client, imgs: &'a Imgs, item_imgs: &'a ItemImgs, - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, #[conrod(common_builder)] common: widget::CommonBuilder, rot_imgs: &'a ImgsRot, tooltip_manager: &'a mut TooltipManager, slot_manager: &'a mut SlotManager, _pulse: f32, - localized_strings: &'a std::sync::Arc, + localized_strings: &'a Localization, stats: &'a Stats, show: &'a Show, @@ -109,12 +109,12 @@ impl<'a> Bag<'a> { client: &'a Client, imgs: &'a Imgs, item_imgs: &'a ItemImgs, - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, rot_imgs: &'a ImgsRot, tooltip_manager: &'a mut TooltipManager, slot_manager: &'a mut SlotManager, pulse: f32, - localized_strings: &'a std::sync::Arc, + localized_strings: &'a Localization, stats: &'a Stats, show: &'a Show, ) -> Self { diff --git a/voxygen/src/hud/buffs.rs b/voxygen/src/hud/buffs.rs index ad8aeed4cc..a1a5a994ca 100644 --- a/voxygen/src/hud/buffs.rs +++ b/voxygen/src/hud/buffs.rs @@ -4,8 +4,8 @@ use super::{ }; use crate::{ hud::{get_buff_info, BuffPosition}, - i18n::VoxygenLocalization, - ui::{fonts::ConrodVoxygenFonts, ImageFrame, Tooltip, TooltipManager, Tooltipable}, + i18n::Localization, + ui::{fonts::Fonts, ImageFrame, Tooltip, TooltipManager, Tooltipable}, GlobalState, }; @@ -34,12 +34,12 @@ widget_ids! { #[derive(WidgetCommon)] pub struct BuffsBar<'a> { imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, #[conrod(common_builder)] common: widget::CommonBuilder, rot_imgs: &'a ImgsRot, tooltip_manager: &'a mut TooltipManager, - localized_strings: &'a std::sync::Arc, + localized_strings: &'a Localization, buffs: &'a Buffs, pulse: f32, global_state: &'a GlobalState, @@ -49,10 +49,10 @@ impl<'a> BuffsBar<'a> { #[allow(clippy::too_many_arguments)] // TODO: Pending review in #587 pub fn new( imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, rot_imgs: &'a ImgsRot, tooltip_manager: &'a mut TooltipManager, - localized_strings: &'a std::sync::Arc, + localized_strings: &'a Localization, buffs: &'a Buffs, pulse: f32, global_state: &'a GlobalState, diff --git a/voxygen/src/hud/buttons.rs b/voxygen/src/hud/buttons.rs index e98cbebf19..874a83f192 100644 --- a/voxygen/src/hud/buttons.rs +++ b/voxygen/src/hud/buttons.rs @@ -3,8 +3,8 @@ use super::{ BLACK, CRITICAL_HP_COLOR, LOW_HP_COLOR, TEXT_COLOR, }; use crate::{ - i18n::VoxygenLocalization, - ui::{fonts::ConrodVoxygenFonts, ImageFrame, Tooltip, TooltipManager, Tooltipable}, + i18n::Localization, + ui::{fonts::Fonts, ImageFrame, Tooltip, TooltipManager, Tooltipable}, window::GameInput, GlobalState, }; @@ -49,13 +49,13 @@ pub struct Buttons<'a> { client: &'a Client, show_bag: bool, imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, #[conrod(common_builder)] common: widget::CommonBuilder, global_state: &'a GlobalState, rot_imgs: &'a ImgsRot, tooltip_manager: &'a mut TooltipManager, - localized_strings: &'a std::sync::Arc, + localized_strings: &'a Localization, stats: &'a Stats, } @@ -65,11 +65,11 @@ impl<'a> Buttons<'a> { client: &'a Client, show_bag: bool, imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, global_state: &'a GlobalState, rot_imgs: &'a ImgsRot, tooltip_manager: &'a mut TooltipManager, - localized_strings: &'a std::sync::Arc, + localized_strings: &'a Localization, stats: &'a Stats, ) -> Self { Self { diff --git a/voxygen/src/hud/chat.rs b/voxygen/src/hud/chat.rs index 36f3f7e94f..a102f92f01 100644 --- a/voxygen/src/hud/chat.rs +++ b/voxygen/src/hud/chat.rs @@ -2,7 +2,7 @@ use super::{ img_ids::Imgs, ERROR_COLOR, FACTION_COLOR, GROUP_COLOR, INFO_COLOR, KILL_COLOR, LOOT_COLOR, OFFLINE_COLOR, ONLINE_COLOR, REGION_COLOR, SAY_COLOR, TELL_COLOR, TEXT_COLOR, WORLD_COLOR, }; -use crate::{i18n::VoxygenLocalization, ui::fonts::ConrodVoxygenFonts, GlobalState}; +use crate::{ui::fonts::Fonts, GlobalState, Localization}; use client::{cmd, Client}; use common::{ comp::{ @@ -52,7 +52,7 @@ pub struct Chat<'a> { global_state: &'a GlobalState, imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, #[conrod(common_builder)] common: widget::CommonBuilder, @@ -60,7 +60,7 @@ pub struct Chat<'a> { // TODO: add an option to adjust this history_max: usize, - localized_strings: &'a std::sync::Arc, + localized_strings: &'a Localization, } impl<'a> Chat<'a> { @@ -69,8 +69,8 @@ impl<'a> Chat<'a> { client: &'a Client, global_state: &'a GlobalState, imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, - localized_strings: &'a std::sync::Arc, + fonts: &'a Fonts, + localized_strings: &'a Localization, ) -> Self { Self { new_messages, @@ -536,12 +536,7 @@ fn do_tab_completion(cursor: usize, input: &str, word: &str) -> (String, usize) } } -fn cursor_offset_to_index( - offset: usize, - text: &str, - ui: &Ui, - fonts: &ConrodVoxygenFonts, -) -> Option { +fn cursor_offset_to_index(offset: usize, text: &str, ui: &Ui, fonts: &Fonts) -> Option { // This moves the cursor to the given offset. Conrod is a pain. // // Width and font must match that of the chat TextEdit diff --git a/voxygen/src/hud/crafting.rs b/voxygen/src/hud/crafting.rs index d9fed33b44..4bdffff9a0 100644 --- a/voxygen/src/hud/crafting.rs +++ b/voxygen/src/hud/crafting.rs @@ -5,8 +5,8 @@ use super::{ }; use crate::{ hud::get_quality_col, - i18n::VoxygenLocalization, - ui::{fonts::ConrodVoxygenFonts, ImageFrame, Tooltip, TooltipManager, Tooltipable}, + i18n::Localization, + ui::{fonts::Fonts, ImageFrame, Tooltip, TooltipManager, Tooltipable}, }; use client::{self, Client}; use common::comp::{ @@ -55,8 +55,8 @@ pub enum Event { pub struct Crafting<'a> { client: &'a Client, imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, - localized_strings: &'a std::sync::Arc, + fonts: &'a Fonts, + localized_strings: &'a Localization, rot_imgs: &'a ImgsRot, tooltip_manager: &'a mut TooltipManager, item_imgs: &'a ItemImgs, @@ -69,8 +69,8 @@ impl<'a> Crafting<'a> { pub fn new( client: &'a Client, imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, - localized_strings: &'a std::sync::Arc, + fonts: &'a Fonts, + localized_strings: &'a Localization, rot_imgs: &'a ImgsRot, tooltip_manager: &'a mut TooltipManager, item_imgs: &'a ItemImgs, diff --git a/voxygen/src/hud/esc_menu.rs b/voxygen/src/hud/esc_menu.rs index 6ce036dfe6..fb949a249f 100644 --- a/voxygen/src/hud/esc_menu.rs +++ b/voxygen/src/hud/esc_menu.rs @@ -1,5 +1,5 @@ use super::{img_ids::Imgs, settings_window::SettingsTab, TEXT_COLOR}; -use crate::{i18n::VoxygenLocalization, ui::fonts::ConrodVoxygenFonts}; +use crate::{i18n::Localization, ui::fonts::Fonts}; use conrod_core::{ widget::{self, Button, Image}, widget_ids, Color, Labelable, Positionable, Sizeable, Widget, WidgetCommon, @@ -22,19 +22,15 @@ widget_ids! { #[derive(WidgetCommon)] pub struct EscMenu<'a> { imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, - localized_strings: &'a std::sync::Arc, + fonts: &'a Fonts, + localized_strings: &'a Localization, #[conrod(common_builder)] common: widget::CommonBuilder, } impl<'a> EscMenu<'a> { - pub fn new( - imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, - localized_strings: &'a std::sync::Arc, - ) -> Self { + pub fn new(imgs: &'a Imgs, fonts: &'a Fonts, localized_strings: &'a Localization) -> Self { Self { imgs, fonts, diff --git a/voxygen/src/hud/group.rs b/voxygen/src/hud/group.rs index 6e12c18fed..595a44ab99 100644 --- a/voxygen/src/hud/group.rs +++ b/voxygen/src/hud/group.rs @@ -6,9 +6,9 @@ use super::{ use crate::{ hud::get_buff_info, - i18n::VoxygenLocalization, + i18n::Localization, settings::Settings, - ui::{fonts::ConrodVoxygenFonts, ImageFrame, Tooltip, TooltipManager, Tooltipable}, + ui::{fonts::Fonts, ImageFrame, Tooltip, TooltipManager, Tooltipable}, window::GameInput, GlobalState, }; @@ -70,8 +70,8 @@ pub struct Group<'a> { settings: &'a Settings, imgs: &'a Imgs, rot_imgs: &'a ImgsRot, - fonts: &'a ConrodVoxygenFonts, - localized_strings: &'a std::sync::Arc, + fonts: &'a Fonts, + localized_strings: &'a Localization, pulse: f32, global_state: &'a GlobalState, tooltip_manager: &'a mut TooltipManager, @@ -88,8 +88,8 @@ impl<'a> Group<'a> { settings: &'a Settings, imgs: &'a Imgs, rot_imgs: &'a ImgsRot, - fonts: &'a ConrodVoxygenFonts, - localized_strings: &'a std::sync::Arc, + fonts: &'a Fonts, + localized_strings: &'a Localization, pulse: f32, global_state: &'a GlobalState, tooltip_manager: &'a mut TooltipManager, diff --git a/voxygen/src/hud/map.rs b/voxygen/src/hud/map.rs index ae34f752d0..046ca31e29 100644 --- a/voxygen/src/hud/map.rs +++ b/voxygen/src/hud/map.rs @@ -3,8 +3,8 @@ use super::{ Show, TEXT_COLOR, UI_HIGHLIGHT_0, UI_MAIN, }; use crate::{ - i18n::VoxygenLocalization, - ui::{fonts::ConrodVoxygenFonts, img_ids, ImageSlider}, + i18n::Localization, + ui::{fonts::Fonts, img_ids, ImageSlider}, GlobalState, }; use client::{self, Client}; @@ -41,11 +41,11 @@ pub struct Map<'a> { world_map: &'a (img_ids::Rotations, Vec2), imgs: &'a Imgs, rot_imgs: &'a ImgsRot, - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, #[conrod(common_builder)] common: widget::CommonBuilder, _pulse: f32, - localized_strings: &'a std::sync::Arc, + localized_strings: &'a Localization, global_state: &'a GlobalState, } impl<'a> Map<'a> { @@ -56,9 +56,9 @@ impl<'a> Map<'a> { imgs: &'a Imgs, rot_imgs: &'a ImgsRot, world_map: &'a (img_ids::Rotations, Vec2), - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, pulse: f32, - localized_strings: &'a std::sync::Arc, + localized_strings: &'a Localization, global_state: &'a GlobalState, ) -> Self { Self { diff --git a/voxygen/src/hud/minimap.rs b/voxygen/src/hud/minimap.rs index ca993285cb..d897fa3074 100644 --- a/voxygen/src/hud/minimap.rs +++ b/voxygen/src/hud/minimap.rs @@ -2,7 +2,7 @@ use super::{ img_ids::{Imgs, ImgsRot}, Show, TEXT_COLOR, UI_HIGHLIGHT_0, UI_MAIN, }; -use crate::ui::{fonts::ConrodVoxygenFonts, img_ids}; +use crate::ui::{fonts::Fonts, img_ids}; use client::{self, Client}; use common::{comp, terrain::TerrainChunkSize, vol::RectVolSize}; use conrod_core::{ @@ -40,7 +40,7 @@ pub struct MiniMap<'a> { imgs: &'a Imgs, rot_imgs: &'a ImgsRot, world_map: &'a (img_ids::Rotations, Vec2), - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, #[conrod(common_builder)] common: widget::CommonBuilder, ori: Vec3, @@ -53,7 +53,7 @@ impl<'a> MiniMap<'a> { imgs: &'a Imgs, rot_imgs: &'a ImgsRot, world_map: &'a (img_ids::Rotations, Vec2), - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, ori: Vec3, ) -> Self { Self { diff --git a/voxygen/src/hud/mod.rs b/voxygen/src/hud/mod.rs index 845a15c2ea..ffc344d3ac 100644 --- a/voxygen/src/hud/mod.rs +++ b/voxygen/src/hud/mod.rs @@ -46,13 +46,13 @@ use spell::Spell; use crate::{ ecs::{comp as vcomp, comp::HpFloaterList}, hud::img_ids::ImgsRot, - i18n::{i18n_asset_key, LanguageMetadata, VoxygenLocalization}, + i18n::{i18n_asset_key, LanguageMetadata, Localization}, render::{Consts, Globals, RenderMode, Renderer}, scene::{ camera::{self, Camera}, lod, }, - ui::{fonts::ConrodVoxygenFonts, img_ids::Rotations, slot, Graphic, Ingameable, ScaleMode, Ui}, + ui::{fonts::Fonts, img_ids::Rotations, slot, Graphic, Ingameable, ScaleMode, Ui}, window::{Event as WinEvent, FullScreenSettings, GameInput}, GlobalState, }; @@ -598,7 +598,7 @@ pub struct Hud { world_map: (/* Id */ Rotations, Vec2), imgs: Imgs, item_imgs: ItemImgs, - fonts: ConrodVoxygenFonts, + fonts: Fonts, rot_imgs: ImgsRot, new_messages: VecDeque, new_notifications: VecDeque, @@ -614,7 +614,7 @@ pub struct Hud { tab_complete: Option, pulse: f32, velocity: f32, - voxygen_i18n: std::sync::Arc, + i18n: std::sync::Arc, slot_manager: slots::SlotManager, hotbar: hotbar::State, events: Vec, @@ -649,12 +649,11 @@ impl Hud { // Load item images. let item_imgs = ItemImgs::new(&mut ui, imgs.not_found); // Load language. - let voxygen_i18n = VoxygenLocalization::load_expect(&i18n_asset_key( + let i18n = Localization::load_expect(&i18n_asset_key( &global_state.settings.language.selected_language, )); // Load fonts. - let fonts = ConrodVoxygenFonts::load(&voxygen_i18n.fonts, &mut ui) - .expect("Impossible to load fonts!"); + let fonts = Fonts::load(&i18n.fonts, &mut ui).expect("Impossible to load fonts!"); // Get the server name. let server = &client.server_info.name; // Get the id, unwrap is safe because this CANNOT be None at this @@ -715,7 +714,7 @@ impl Hud { tab_complete: None, pulse: 0.0, velocity: 0.0, - voxygen_i18n, + i18n, slot_manager, hotbar: hotbar_state, events: Vec::new(), @@ -723,10 +722,10 @@ impl Hud { } } - pub fn update_language(&mut self, voxygen_i18n: std::sync::Arc) { - self.voxygen_i18n = voxygen_i18n; - self.fonts = ConrodVoxygenFonts::load(&self.voxygen_i18n.fonts, &mut self.ui) - .expect("Impossible to load fonts!"); + pub fn update_language(&mut self, i18n: std::sync::Arc) { + self.i18n = i18n; + self.fonts = + Fonts::load(&self.i18n.fonts, &mut self.ui).expect("Impossible to load fonts!"); } #[allow(clippy::assign_op_pattern)] // TODO: Pending review in #587 @@ -1256,7 +1255,7 @@ impl Hud { in_group, &global_state.settings.gameplay, self.pulse, - &self.voxygen_i18n, + &self.i18n, &self.imgs, &self.fonts, ) @@ -1459,8 +1458,8 @@ impl Hud { Intro::Show => { if self.pulse > 20.0 { self.show.want_grab = false; - let quest_headline = &self.voxygen_i18n.get("hud.temp_quest_headline"); - let quest_text = &self.voxygen_i18n.get("hud.temp_quest_text"); + let quest_headline = &self.i18n.get("hud.temp_quest_headline"); + let quest_text = &self.i18n.get("hud.temp_quest_text"); Image::new(self.imgs.quest_bg) .w_h(404.0, 858.0) .middle_of(ui_widgets.window) @@ -1497,7 +1496,7 @@ impl Hud { .hover_image(self.imgs.button_hover) .press_image(self.imgs.button_press) .mid_bottom_with_margin_on(self.ids.q_text_bg, -120.0) - .label(&self.voxygen_i18n.get("common.accept")) + .label(&self.i18n.get("common.accept")) .label_font_id(self.fonts.cyri.conrod_id) .label_font_size(self.fonts.cyri.scale(22)) .label_color(TEXT_COLOR) @@ -1675,7 +1674,7 @@ impl Hud { if let Some(help_key) = global_state.settings.controls.get_binding(GameInput::Help) { Text::new( &self - .voxygen_i18n + .i18n .get("hud.press_key_to_toggle_keybindings_fmt") .replace("{key}", help_key.to_string().as_str()), ) @@ -1693,7 +1692,7 @@ impl Hud { { Text::new( &self - .voxygen_i18n + .i18n .get("hud.press_key_to_toggle_debug_info_fmt") .replace("{key}", toggle_debug_key.to_string().as_str()), ) @@ -1708,7 +1707,7 @@ impl Hud { if let Some(help_key) = global_state.settings.controls.get_binding(GameInput::Help) { Text::new( &self - .voxygen_i18n + .i18n .get("hud.press_key_to_show_keybindings_fmt") .replace("{key}", help_key.to_string().as_str()), ) @@ -1726,7 +1725,7 @@ impl Hud { { Text::new( &self - .voxygen_i18n + .i18n .get("hud.press_key_to_show_debug_info_fmt") .replace("{key}", toggle_debug_key.to_string().as_str()), ) @@ -1744,7 +1743,7 @@ impl Hud { { Text::new( &self - .voxygen_i18n + .i18n .get("hud.press_key_to_toggle_lantern_fmt") .replace("{key}", toggle_lantern_key.to_string().as_str()), ) @@ -1789,7 +1788,7 @@ impl Hud { global_state, &self.rot_imgs, tooltip_manager, - &self.voxygen_i18n, + &self.i18n, &player_stats, ) .set(self.ids.buttons, ui_widgets) @@ -1811,7 +1810,7 @@ impl Hud { &self.fonts, &self.rot_imgs, tooltip_manager, - &self.voxygen_i18n, + &self.i18n, &player_buffs, self.pulse, &global_state, @@ -1831,7 +1830,7 @@ impl Hud { &self.imgs, &self.rot_imgs, &self.fonts, - &self.voxygen_i18n, + &self.i18n, self.pulse, &global_state, tooltip_manager, @@ -1848,7 +1847,7 @@ impl Hud { } // Popup (waypoint saved and similar notifications) Popup::new( - &self.voxygen_i18n, + &self.i18n, client, &self.new_notifications, &self.fonts, @@ -1884,7 +1883,7 @@ impl Hud { tooltip_manager, &mut self.slot_manager, self.pulse, - &self.voxygen_i18n, + &self.i18n, &player_stats, &self.show, ) @@ -1951,7 +1950,7 @@ impl Hud { &self.hotbar, tooltip_manager, &mut self.slot_manager, - &self.voxygen_i18n, + &self.i18n, &self.show, ) .set(self.ids.skillbar, ui_widgets); @@ -1965,7 +1964,7 @@ impl Hud { client, &self.imgs, &self.fonts, - &self.voxygen_i18n, + &self.i18n, &self.rot_imgs, tooltip_manager, &self.item_imgs, @@ -2004,7 +2003,7 @@ impl Hud { global_state, &self.imgs, &self.fonts, - &self.voxygen_i18n, + &self.i18n, ) .and_then(self.force_chat_input.take(), |c, input| c.input(input)) .and_then(self.tab_complete.take(), |c, input| { @@ -2041,7 +2040,7 @@ impl Hud { &self.show, &self.imgs, &self.fonts, - &self.voxygen_i18n, + &self.i18n, fps as f32, ) .set(self.ids.settings_window, ui_widgets) @@ -2194,7 +2193,7 @@ impl Hud { client, &self.imgs, &self.fonts, - &self.voxygen_i18n, + &self.i18n, info.selected_entity, &self.rot_imgs, tooltip_manager, @@ -2222,14 +2221,8 @@ impl Hud { // Spellbook if self.show.spell { - match Spell::new( - &self.show, - client, - &self.imgs, - &self.fonts, - &self.voxygen_i18n, - ) - .set(self.ids.spell, ui_widgets) + match Spell::new(&self.show, client, &self.imgs, &self.fonts, &self.i18n) + .set(self.ids.spell, ui_widgets) { Some(spell::Event::Close) => { self.show.spell(false); @@ -2249,7 +2242,7 @@ impl Hud { &self.world_map, &self.fonts, self.pulse, - &self.voxygen_i18n, + &self.i18n, &global_state, ) .set(self.ids.map, ui_widgets) @@ -2268,7 +2261,7 @@ impl Hud { } if self.show.esc_menu { - match EscMenu::new(&self.imgs, &self.fonts, &self.voxygen_i18n) + match EscMenu::new(&self.imgs, &self.fonts, &self.i18n) .set(self.ids.esc_menu, ui_widgets) { Some(esc_menu::Event::OpenSettings(tab)) => { @@ -2311,7 +2304,7 @@ impl Hud { if self.show.free_look { Text::new( &self - .voxygen_i18n + .i18n .get("hud.free_look_indicator") .replace("{key}", freelook_key.to_string().as_str()), ) @@ -2322,7 +2315,7 @@ impl Hud { .set(self.ids.free_look_bg, ui_widgets); Text::new( &self - .voxygen_i18n + .i18n .get("hud.free_look_indicator") .replace("{key}", freelook_key.to_string().as_str()), ) @@ -2336,13 +2329,13 @@ impl Hud { // Auto walk indicator if self.show.auto_walk { - Text::new(&self.voxygen_i18n.get("hud.auto_walk_indicator")) + Text::new(&self.i18n.get("hud.auto_walk_indicator")) .color(TEXT_BG) .mid_top_with_margin_on(ui_widgets.window, 70.0) .font_id(self.fonts.cyri.conrod_id) .font_size(self.fonts.cyri.scale(20)) .set(self.ids.auto_walk_bg, ui_widgets); - Text::new(&self.voxygen_i18n.get("hud.auto_walk_indicator")) + Text::new(&self.i18n.get("hud.auto_walk_indicator")) .color(KILL_COLOR) .top_left_with_margins_on(self.ids.auto_walk_bg, -1.0, -1.0) .font_id(self.fonts.cyri.conrod_id) diff --git a/voxygen/src/hud/overhead.rs b/voxygen/src/hud/overhead.rs index 8d592a3501..170c6a8d29 100644 --- a/voxygen/src/hud/overhead.rs +++ b/voxygen/src/hud/overhead.rs @@ -4,9 +4,9 @@ use super::{ }; use crate::{ hud::get_buff_info, - i18n::VoxygenLocalization, + i18n::Localization, settings::GameplaySettings, - ui::{fonts::ConrodVoxygenFonts, Ingameable}, + ui::{fonts::Fonts, Ingameable}, }; use common::comp::{BuffKind, Buffs, Energy, Health, SpeechBubble, SpeechBubbleType, Stats}; use conrod_core::{ @@ -76,9 +76,9 @@ pub struct Overhead<'a> { in_group: bool, settings: &'a GameplaySettings, pulse: f32, - voxygen_i18n: &'a std::sync::Arc, + i18n: &'a Localization, imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, #[conrod(common_builder)] common: widget::CommonBuilder, @@ -93,9 +93,9 @@ impl<'a> Overhead<'a> { in_group: bool, settings: &'a GameplaySettings, pulse: f32, - voxygen_i18n: &'a std::sync::Arc, + i18n: &'a Localization, imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, ) -> Self { Self { info, @@ -104,7 +104,7 @@ impl<'a> Overhead<'a> { in_group, settings, pulse, - voxygen_i18n, + i18n, imgs, fonts, common: widget::CommonBuilder::default(), @@ -336,7 +336,7 @@ impl<'a> Widget for Overhead<'a> { .set(state.ids.health_bar, ui); let mut txt = format!("{}/{}", health_cur_txt, health_max_txt); if health.is_dead { - txt = self.voxygen_i18n.get("hud.group.dead").to_string() + txt = self.i18n.get("hud.group.dead").to_string() }; Text::new(&txt) .mid_top_with_margin_on(state.ids.health_bar_bg, 2.0) @@ -420,8 +420,7 @@ impl<'a> Widget for Overhead<'a> { // Speech bubble if let Some(bubble) = self.bubble { let dark_mode = self.settings.speech_bubble_dark_mode; - let localizer = - |s: &str, i| -> String { self.voxygen_i18n.get_variation(&s, i).to_string() }; + let localizer = |s: &str, i| -> String { self.i18n.get_variation(&s, i).to_string() }; let bubble_contents: String = bubble.message(localizer); let (text_color, shadow_color) = bubble_color(&bubble, dark_mode); let mut text = Text::new(&bubble_contents) diff --git a/voxygen/src/hud/overitem.rs b/voxygen/src/hud/overitem.rs index f45366176e..5c20a93576 100644 --- a/voxygen/src/hud/overitem.rs +++ b/voxygen/src/hud/overitem.rs @@ -1,6 +1,6 @@ use crate::{ settings::ControlSettings, - ui::{fonts::ConrodVoxygenFonts, Ingameable}, + ui::{fonts::Fonts, Ingameable}, window::GameInput, }; use conrod_core::{ @@ -25,7 +25,7 @@ widget_ids! { pub struct Overitem<'a> { name: &'a str, distance_from_player_sqr: &'a f32, - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, controls: &'a ControlSettings, #[conrod(common_builder)] common: widget::CommonBuilder, @@ -35,7 +35,7 @@ impl<'a> Overitem<'a> { pub fn new( name: &'a str, distance_from_player_sqr: &'a f32, - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, controls: &'a ControlSettings, ) -> Self { Self { diff --git a/voxygen/src/hud/popup.rs b/voxygen/src/hud/popup.rs index 2f0e33b005..b1237c53e4 100644 --- a/voxygen/src/hud/popup.rs +++ b/voxygen/src/hud/popup.rs @@ -1,5 +1,5 @@ use super::Show; -use crate::{i18n::VoxygenLocalization, ui::fonts::ConrodVoxygenFonts}; +use crate::{i18n::Localization, ui::fonts::Fonts}; use client::{self, Client}; use common::msg::Notification; use conrod_core::{ @@ -21,10 +21,10 @@ widget_ids! { #[derive(WidgetCommon)] pub struct Popup<'a> { - voxygen_i18n: &'a std::sync::Arc, + i18n: &'a Localization, client: &'a Client, new_notifications: &'a VecDeque, - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, #[conrod(common_builder)] common: widget::CommonBuilder, show: &'a Show, @@ -34,14 +34,14 @@ pub struct Popup<'a> { /// Dungeon Cleared (TODO), and Quest Completed (TODO) impl<'a> Popup<'a> { pub fn new( - voxygen_i18n: &'a std::sync::Arc, + i18n: &'a Localization, client: &'a Client, new_notifications: &'a VecDeque, - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, show: &'a Show, ) -> Self { Self { - voxygen_i18n, + i18n, client, new_notifications, fonts, @@ -126,7 +126,7 @@ impl<'a> Widget for Popup<'a> { if s.infos.is_empty() { s.last_info_update = Instant::now(); } - let text = self.voxygen_i18n.get("hud.waypoint_saved"); + let text = self.i18n.get("hud.waypoint_saved"); s.infos.push_back(text.to_string()); }); }, diff --git a/voxygen/src/hud/settings_window.rs b/voxygen/src/hud/settings_window.rs index daa5148219..25f987134b 100644 --- a/voxygen/src/hud/settings_window.rs +++ b/voxygen/src/hud/settings_window.rs @@ -5,9 +5,9 @@ use super::{ }; use crate::{ hud::BuffPosition, - i18n::{list_localizations, LanguageMetadata, VoxygenLocalization}, + i18n::{list_localizations, LanguageMetadata, Localization}, render::{AaMode, CloudMode, FluidMode, LightingMode, RenderMode, ShadowMapMode, ShadowMode}, - ui::{fonts::ConrodVoxygenFonts, ImageSlider, ScaleMode, ToggleButton}, + ui::{fonts::Fonts, ImageSlider, ScaleMode, ToggleButton}, window::{FullScreenSettings, FullscreenMode, GameInput}, GlobalState, }; @@ -227,8 +227,8 @@ pub struct SettingsWindow<'a> { global_state: &'a GlobalState, show: &'a Show, imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, - localized_strings: &'a std::sync::Arc, + fonts: &'a Fonts, + localized_strings: &'a Localization, fps: f32, #[conrod(common_builder)] common: widget::CommonBuilder, @@ -239,8 +239,8 @@ impl<'a> SettingsWindow<'a> { global_state: &'a GlobalState, show: &'a Show, imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, - localized_strings: &'a std::sync::Arc, + fonts: &'a Fonts, + localized_strings: &'a Localization, fps: f32, ) -> Self { Self { diff --git a/voxygen/src/hud/skillbar.rs b/voxygen/src/hud/skillbar.rs index e8afb91e18..1f8a48eda0 100644 --- a/voxygen/src/hud/skillbar.rs +++ b/voxygen/src/hud/skillbar.rs @@ -6,9 +6,9 @@ use super::{ STAMINA_COLOR, TEXT_COLOR, UI_HIGHLIGHT_0, UI_MAIN, XP_COLOR, }; use crate::{ - i18n::VoxygenLocalization, + i18n::Localization, ui::{ - fonts::ConrodVoxygenFonts, + fonts::Fonts, slot::{ContentSize, SlotMaker}, ImageFrame, Tooltip, TooltipManager, Tooltipable, }, @@ -121,7 +121,7 @@ pub struct Skillbar<'a> { global_state: &'a GlobalState, imgs: &'a Imgs, item_imgs: &'a ItemImgs, - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, rot_imgs: &'a ImgsRot, stats: &'a Stats, health: &'a Health, @@ -133,7 +133,7 @@ pub struct Skillbar<'a> { hotbar: &'a hotbar::State, tooltip_manager: &'a mut TooltipManager, slot_manager: &'a mut slots::SlotManager, - localized_strings: &'a std::sync::Arc, + localized_strings: &'a Localization, pulse: f32, #[conrod(common_builder)] common: widget::CommonBuilder, @@ -146,7 +146,7 @@ impl<'a> Skillbar<'a> { global_state: &'a GlobalState, imgs: &'a Imgs, item_imgs: &'a ItemImgs, - fonts: &'a ConrodVoxygenFonts, + fonts: &'a Fonts, rot_imgs: &'a ImgsRot, stats: &'a Stats, health: &'a Health, @@ -159,7 +159,7 @@ impl<'a> Skillbar<'a> { hotbar: &'a hotbar::State, tooltip_manager: &'a mut TooltipManager, slot_manager: &'a mut slots::SlotManager, - localized_strings: &'a std::sync::Arc, + localized_strings: &'a Localization, show: &'a Show, ) -> Self { Self { diff --git a/voxygen/src/hud/social.rs b/voxygen/src/hud/social.rs index 68680a7d01..20e7641373 100644 --- a/voxygen/src/hud/social.rs +++ b/voxygen/src/hud/social.rs @@ -4,8 +4,8 @@ use super::{ }; use crate::{ - i18n::VoxygenLocalization, - ui::{fonts::ConrodVoxygenFonts, ImageFrame, Tooltip, TooltipManager, Tooltipable}, + i18n::Localization, + ui::{fonts::Fonts, ImageFrame, Tooltip, TooltipManager, Tooltipable}, }; use client::{self, Client}; use common::{comp::group, sync::Uid}; @@ -66,8 +66,8 @@ pub struct Social<'a> { show: &'a Show, client: &'a Client, imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, - localized_strings: &'a std::sync::Arc, + fonts: &'a Fonts, + localized_strings: &'a Localization, selected_entity: Option<(specs::Entity, Instant)>, rot_imgs: &'a ImgsRot, tooltip_manager: &'a mut TooltipManager, @@ -82,8 +82,8 @@ impl<'a> Social<'a> { show: &'a Show, client: &'a Client, imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, - localized_strings: &'a std::sync::Arc, + fonts: &'a Fonts, + localized_strings: &'a Localization, selected_entity: Option<(specs::Entity, Instant)>, rot_imgs: &'a ImgsRot, tooltip_manager: &'a mut TooltipManager, diff --git a/voxygen/src/hud/spell.rs b/voxygen/src/hud/spell.rs index 4f1a3561ba..0fbf3ab91d 100644 --- a/voxygen/src/hud/spell.rs +++ b/voxygen/src/hud/spell.rs @@ -1,5 +1,5 @@ use super::{img_ids::Imgs, Show, TEXT_COLOR, UI_MAIN}; -use crate::{i18n::VoxygenLocalization, ui::fonts::ConrodVoxygenFonts}; +use crate::{i18n::Localization, ui::fonts::Fonts}; use conrod_core::{ color, widget::{self, Button, Image, Rectangle, Text}, @@ -24,8 +24,8 @@ pub struct Spell<'a> { _client: &'a Client, imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, - localized_strings: &'a std::sync::Arc, + fonts: &'a Fonts, + localized_strings: &'a Localization, #[conrod(common_builder)] common: widget::CommonBuilder, @@ -36,8 +36,8 @@ impl<'a> Spell<'a> { show: &'a Show, _client: &'a Client, imgs: &'a Imgs, - fonts: &'a ConrodVoxygenFonts, - localized_strings: &'a std::sync::Arc, + fonts: &'a Fonts, + localized_strings: &'a Localization, ) -> Self { Self { _show: show, diff --git a/voxygen/src/i18n.rs b/voxygen/src/i18n.rs index 479fef3121..13e34dc0c0 100644 --- a/voxygen/src/i18n.rs +++ b/voxygen/src/i18n.rs @@ -44,11 +44,11 @@ impl Font { } /// Store font metadata -pub type VoxygenFonts = HashMap; +pub type Fonts = HashMap; /// Store internationalization data #[derive(Clone, Debug, PartialEq, Serialize, Deserialize)] -pub struct VoxygenLocalization { +pub struct Localization { /// A map storing the localized texts /// /// Localized content can be accessed using a String key. @@ -64,12 +64,12 @@ pub struct VoxygenLocalization { pub convert_utf8_to_ascii: bool, /// Font configuration is stored here - pub fonts: VoxygenFonts, + pub fonts: Fonts, pub metadata: LanguageMetadata, } -impl VoxygenLocalization { +impl Localization { /// Get a localized text from the given key /// /// If the key is not present in the localization object @@ -97,7 +97,7 @@ impl VoxygenLocalization { /// Return the missing keys compared to the reference language pub fn list_missing_entries(&self) -> (HashSet, HashSet) { let reference_localization = - VoxygenLocalization::load_expect(i18n_asset_key(REFERENCE_LANG).as_ref()); + Localization::load_expect(i18n_asset_key(REFERENCE_LANG).as_ref()); let reference_string_keys: HashSet<_> = reference_localization.string_map.keys().cloned().collect(); @@ -136,14 +136,14 @@ impl VoxygenLocalization { } } -impl Asset for VoxygenLocalization { +impl Asset for Localization { const ENDINGS: &'static [&'static str] = &["ron"]; /// Load the translations located in the input buffer and convert them - /// into a `VoxygenLocalization` object. + /// into a `Localization` object. #[allow(clippy::into_iter_on_ref)] // TODO: Pending review in #587 fn parse(buf_reader: BufReader, _specifier: &str) -> Result { - let mut asked_localization: VoxygenLocalization = + let mut asked_localization: Localization = from_reader(buf_reader).map_err(assets::Error::parse_error)?; // Update the text if UTF-8 to ASCII conversion is enabled @@ -163,10 +163,10 @@ impl Asset for VoxygenLocalization { } } -/// Load all the available languages located in the Voxygen asset directory +/// Load all the available languages located in the voxygen asset directory pub fn list_localizations() -> Vec { let voxygen_locales_assets = "voxygen.i18n.*"; - let lang_list = VoxygenLocalization::load_glob(voxygen_locales_assets).unwrap(); + let lang_list = Localization::load_glob(voxygen_locales_assets).unwrap(); lang_list.iter().map(|e| (*e).metadata.clone()).collect() } @@ -175,7 +175,7 @@ pub fn i18n_asset_key(language_id: &str) -> String { "voxygen.i18n.".to_string() #[cfg(test)] mod tests { - use super::VoxygenLocalization; + use super::Localization; use git2::Repository; use ron::de::{from_bytes, from_reader}; use std::{ @@ -248,7 +248,7 @@ mod tests { fn generate_key_version<'a>( repo: &'a git2::Repository, - localization: &VoxygenLocalization, + localization: &Localization, path: &std::path::Path, file_blob: &git2::Blob, ) -> HashMap { @@ -348,7 +348,7 @@ mod tests { ); for path in i18n_files { let f = fs::File::open(&path).expect("Failed opening file"); - let _: VoxygenLocalization = match from_reader(f) { + let _: Localization = match from_reader(f) { Ok(v) => v, Err(e) => { panic!( @@ -387,7 +387,7 @@ mod tests { // Read HEAD for the reference language file let i18n_en_blob = read_file_from_path(&repo, &head_ref, &en_i18n_path); - let loc: VoxygenLocalization = from_bytes(i18n_en_blob.content()) + let loc: Localization = from_bytes(i18n_en_blob.content()) .expect("Expect to parse reference i18n RON file, can't proceed without it"); let i18n_references: HashMap = generate_key_version(&repo, &loc, &en_i18n_path, &i18n_en_blob); @@ -406,7 +406,7 @@ mod tests { // Find the localization entry state let current_blob = read_file_from_path(&repo, &head_ref, &relfile); - let current_loc: VoxygenLocalization = match from_bytes(current_blob.content()) { + let current_loc: Localization = match from_bytes(current_blob.content()) { Ok(v) => v, Err(e) => { eprintln!( diff --git a/voxygen/src/main.rs b/voxygen/src/main.rs index ae4be0453c..f87b32d35b 100644 --- a/voxygen/src/main.rs +++ b/voxygen/src/main.rs @@ -5,7 +5,7 @@ use veloren_voxygen::{ audio::{self, AudioFrontend}, - i18n::{self, i18n_asset_key, VoxygenLocalization}, + i18n::{self, i18n_asset_key, Localization}, logging, profile::Profile, run, @@ -157,7 +157,7 @@ fn main() { let profile = Profile::load(); let mut localization_watcher = watch::ReloadIndicator::new(); - let localized_strings = VoxygenLocalization::load_watched( + let localized_strings = Localization::load_watched( &i18n_asset_key(&settings.language.selected_language), &mut localization_watcher, ) @@ -169,7 +169,7 @@ fn main() { "Impossible to load language: change to the default language (English) instead.", ); settings.language.selected_language = i18n::REFERENCE_LANG.to_owned(); - VoxygenLocalization::load_watched( + Localization::load_watched( &i18n_asset_key(&settings.language.selected_language), &mut localization_watcher, ) diff --git a/voxygen/src/menu/char_selection/mod.rs b/voxygen/src/menu/char_selection/mod.rs index a50dc0c604..4cf356e5f8 100644 --- a/voxygen/src/menu/char_selection/mod.rs +++ b/voxygen/src/menu/char_selection/mod.rs @@ -1,7 +1,7 @@ mod ui; use crate::{ - i18n::{i18n_asset_key, VoxygenLocalization}, + i18n::{i18n_asset_key, Localization}, render::Renderer, scene::simple::{self as scene, Scene}, session::SessionState, @@ -149,7 +149,7 @@ impl PlayState for CharSelectionState { } // Tick the client (currently only to keep the connection alive). - let localized_strings = VoxygenLocalization::load_expect(&i18n_asset_key( + let localized_strings = Localization::load_expect(&i18n_asset_key( &global_state.settings.language.selected_language, )); diff --git a/voxygen/src/menu/char_selection/ui.rs b/voxygen/src/menu/char_selection/ui.rs index 5e64c535fb..bf3ae697af 100644 --- a/voxygen/src/menu/char_selection/ui.rs +++ b/voxygen/src/menu/char_selection/ui.rs @@ -1,8 +1,8 @@ use crate::{ - i18n::{i18n_asset_key, VoxygenLocalization}, + i18n::{i18n_asset_key, Localization}, render::{Consts, Globals, Renderer}, ui::{ - fonts::ConrodVoxygenFonts, + fonts::Fonts, img_ids::{BlankGraphic, ImageGraphic, VoxelGraphic, VoxelSs9Graphic}, ImageFrame, ImageSlider, Tooltip, Tooltipable, Ui, }, @@ -26,7 +26,6 @@ use conrod_core::{ widget_ids, Borderable, Color, Colorable, Labelable, Positionable, Sizeable, UiCell, Widget, }; use rand::{thread_rng, Rng}; -use std::sync::Arc; const STARTER_HAMMER: &str = "common.items.weapons.hammer.starter_hammer"; const STARTER_BOW: &str = "common.items.weapons.bow.starter_bow"; @@ -300,9 +299,9 @@ pub struct CharSelectionUi { ids: Ids, imgs: Imgs, rot_imgs: ImgsRot, - fonts: ConrodVoxygenFonts, + fonts: Fonts, info_content: InfoContent, - voxygen_i18n: Arc, + i18n: std::sync::Arc, enter: bool, pub mode: Mode, pub selected_character: usize, @@ -321,12 +320,11 @@ impl CharSelectionUi { let imgs = Imgs::load(&mut ui).expect("Failed to load images!"); let rot_imgs = ImgsRot::load(&mut ui).expect("Failed to load images!"); // Load language - let voxygen_i18n = VoxygenLocalization::load_expect(&i18n_asset_key( + let i18n = Localization::load_expect(&i18n_asset_key( &global_state.settings.language.selected_language, )); // Load fonts. - let fonts = ConrodVoxygenFonts::load(&voxygen_i18n.fonts, &mut ui) - .expect("Impossible to load fonts!"); + let fonts = Fonts::load(&i18n.fonts, &mut ui).expect("Impossible to load fonts!"); Self { ui, @@ -336,7 +334,7 @@ impl CharSelectionUi { fonts, info_content: InfoContent::LoadingCharacters, selected_character: 0, - voxygen_i18n, + i18n, mode: Mode::Select(None), enter: false, } @@ -483,7 +481,7 @@ impl CharSelectionUi { match self.info_content { InfoContent::None => unreachable!(), InfoContent::Deletion(character_index) => { - Text::new(&self.voxygen_i18n.get("char_selection.delete_permanently")) + Text::new(&self.i18n.get("char_selection.delete_permanently")) .mid_top_with_margin_on(self.ids.info_frame, 40.0) .font_size(self.fonts.cyri.scale(24)) .font_id(self.fonts.cyri.conrod_id) @@ -495,7 +493,7 @@ impl CharSelectionUi { .hover_image(self.imgs.button_hover) .press_image(self.imgs.button_press) .label_y(Relative::Scalar(2.0)) - .label(&self.voxygen_i18n.get("common.no")) + .label(&self.i18n.get("common.no")) .label_font_id(self.fonts.cyri.conrod_id) .label_font_size(self.fonts.cyri.scale(18)) .label_color(TEXT_COLOR) @@ -510,7 +508,7 @@ impl CharSelectionUi { .hover_image(self.imgs.button_hover) .press_image(self.imgs.button_press) .label_y(Relative::Scalar(2.0)) - .label(&self.voxygen_i18n.get("common.yes")) + .label(&self.i18n.get("common.yes")) .label_font_id(self.fonts.cyri.conrod_id) .label_font_size(self.fonts.cyri.scale(18)) .label_color(TEXT_COLOR) @@ -532,7 +530,7 @@ impl CharSelectionUi { }; }, InfoContent::LoadingCharacters => { - Text::new(&self.voxygen_i18n.get("char_selection.loading_characters")) + Text::new(&self.i18n.get("char_selection.loading_characters")) .mid_top_with_margin_on(self.ids.info_frame, 40.0) .font_size(self.fonts.cyri.scale(24)) .font_id(self.fonts.cyri.conrod_id) @@ -540,7 +538,7 @@ impl CharSelectionUi { .set(self.ids.loading_characters_text, ui_widgets); }, InfoContent::CreatingCharacter => { - Text::new(&self.voxygen_i18n.get("char_selection.creating_character")) + Text::new(&self.i18n.get("char_selection.creating_character")) .mid_top_with_margin_on(self.ids.info_frame, 40.0) .font_size(self.fonts.cyri.scale(24)) .font_id(self.fonts.cyri.conrod_id) @@ -548,7 +546,7 @@ impl CharSelectionUi { .set(self.ids.creating_character_text, ui_widgets); }, InfoContent::DeletingCharacter => { - Text::new(&self.voxygen_i18n.get("char_selection.deleting_character")) + Text::new(&self.i18n.get("char_selection.deleting_character")) .mid_top_with_margin_on(self.ids.info_frame, 40.0) .font_size(self.fonts.cyri.scale(24)) .font_id(self.fonts.cyri.conrod_id) @@ -559,7 +557,7 @@ impl CharSelectionUi { if let Some(error_message) = &client.character_list.error { Text::new(&format!( "{}: {}", - &self.voxygen_i18n.get("common.error"), + &self.i18n.get("common.error"), error_message )) .mid_top_with_margin_on(self.ids.info_frame, 40.0) @@ -574,7 +572,7 @@ impl CharSelectionUi { .hover_image(self.imgs.button_hover) .press_image(self.imgs.button_press) .label_y(Relative::Scalar(2.0)) - .label(&self.voxygen_i18n.get("common.close")) + .label(&self.i18n.get("common.close")) .label_font_id(self.fonts.cyri.conrod_id) .label_font_size(self.fonts.cyri.scale(18)) .label_color(TEXT_COLOR) @@ -644,7 +642,7 @@ impl CharSelectionUi { .parent(self.ids.charlist_bg) .hover_image(self.imgs.button_hover) .press_image(self.imgs.button_press) - .label(&self.voxygen_i18n.get("char_selection.change_server")) + .label(&self.i18n.get("char_selection.change_server")) .label_color(TEXT_COLOR) .label_font_id(self.fonts.cyri.conrod_id) .label_font_size(self.fonts.cyri.scale(18)) @@ -657,7 +655,7 @@ impl CharSelectionUi { // Enter World Button let character_count = client.character_list.characters.len(); - let enter_world_str = &self.voxygen_i18n.get("char_selection.enter_world"); + let enter_world_str = &self.i18n.get("char_selection.enter_world"); let enter_button = Button::image(self.imgs.button) .mid_bottom_with_margin_on(ui_widgets.window, 10.0) .w_h(250.0, 60.0) @@ -691,7 +689,7 @@ impl CharSelectionUi { .w_h(150.0, 40.0) .hover_image(self.imgs.button_hover) .press_image(self.imgs.button_press) - .label(&self.voxygen_i18n.get("char_selection.logout")) + .label(&self.i18n.get("char_selection.logout")) .label_font_id(self.fonts.cyri.conrod_id) .label_color(TEXT_COLOR) .label_font_size(self.fonts.cyri.scale(20)) @@ -772,7 +770,7 @@ impl CharSelectionUi { .press_image(self.imgs.delete_button_press) .with_tooltip( tooltip_manager, - &self.voxygen_i18n.get("char_selection.delete_permanently"), + &self.i18n.get("char_selection.delete_permanently"), "", &tooltip_human, TEXT_COLOR, @@ -791,7 +789,7 @@ impl CharSelectionUi { Text::new( &self - .voxygen_i18n + .i18n .get("char_selection.level_fmt") .replace("{level_nb}", &character_item.level.to_string()), ) @@ -801,7 +799,7 @@ impl CharSelectionUi { .color(TEXT_COLOR) .set(self.ids.character_levels[i], ui_widgets); - Text::new(&self.voxygen_i18n.get("char_selection.uncanny_valley")) + Text::new(&self.i18n.get("char_selection.uncanny_valley")) .down_from(self.ids.character_levels[i], 4.0) .font_size(self.fonts.cyri.scale(17)) .font_id(self.fonts.cyri.conrod_id) @@ -834,7 +832,7 @@ impl CharSelectionUi { .w_h(386.0, 80.0) .hover_image(self.imgs.selection_hover) .press_image(self.imgs.selection_press) - .label(&self.voxygen_i18n.get("char_selection.create_new_charater")) + .label(&self.i18n.get("char_selection.create_new_charater")) .label_color(color) .label_font_id(self.fonts.cyri.conrod_id) .image_color(color) @@ -869,7 +867,7 @@ impl CharSelectionUi { .w_h(150.0, 40.0) .hover_image(self.imgs.button_hover) .press_image(self.imgs.button_press) - .label(&self.voxygen_i18n.get("common.back")) + .label(&self.i18n.get("common.back")) .label_font_id(self.fonts.cyri.conrod_id) .label_color(TEXT_COLOR) .label_font_size(self.fonts.cyri.scale(20)) @@ -893,7 +891,7 @@ impl CharSelectionUi { } else { self.imgs.button }) - .label(&self.voxygen_i18n.get("common.create")) + .label(&self.i18n.get("common.create")) .label_font_id(self.fonts.cyri.conrod_id) .label_color(if *name != "Character Name" && *name != "" { TEXT_COLOR @@ -908,7 +906,7 @@ impl CharSelectionUi { if create_button .with_tooltip( tooltip_manager, - &self.voxygen_i18n.get("char_selection.create_info_name"), + &self.i18n.get("char_selection.create_info_name"), "", &tooltip_human, TEXT_COLOR, @@ -1081,7 +1079,7 @@ impl CharSelectionUi { .press_image(self.imgs.icon_border_press) .with_tooltip( tooltip_manager, - &self.voxygen_i18n.get("common.species.human"), + &self.i18n.get("common.species.human"), "", &tooltip_human, TEXT_COLOR, @@ -1108,7 +1106,7 @@ impl CharSelectionUi { .press_image(self.imgs.icon_border_press) .with_tooltip( tooltip_manager, - &self.voxygen_i18n.get("common.species.orc"), + &self.i18n.get("common.species.orc"), "", &tooltip_human, TEXT_COLOR, @@ -1134,7 +1132,7 @@ impl CharSelectionUi { .press_image(self.imgs.icon_border_press) .with_tooltip( tooltip_manager, - &self.voxygen_i18n.get("common.species.dwarf"), + &self.i18n.get("common.species.dwarf"), "", &tooltip_human, TEXT_COLOR, @@ -1160,7 +1158,7 @@ impl CharSelectionUi { .press_image(self.imgs.icon_border_press) .with_tooltip( tooltip_manager, - &self.voxygen_i18n.get("common.species.elf"), + &self.i18n.get("common.species.elf"), "", &tooltip_human, TEXT_COLOR, @@ -1187,7 +1185,7 @@ impl CharSelectionUi { .press_image(self.imgs.icon_border_press) .with_tooltip( tooltip_manager, - &self.voxygen_i18n.get("common.species.undead"), + &self.i18n.get("common.species.undead"), "", &tooltip_human, TEXT_COLOR, @@ -1213,7 +1211,7 @@ impl CharSelectionUi { .press_image(self.imgs.icon_border_press) .with_tooltip( tooltip_manager, - &self.voxygen_i18n.get("common.species.danari"), + &self.i18n.get("common.species.danari"), "", &tooltip_human, TEXT_COLOR, @@ -1239,7 +1237,7 @@ impl CharSelectionUi { .press_image(self.imgs.icon_border_press) .with_tooltip( tooltip_manager, - &self.voxygen_i18n.get("common.weapons.sceptre"), + &self.i18n.get("common.weapons.sceptre"), "", &tooltip_human, TEXT_COLOR, @@ -1265,7 +1263,7 @@ impl CharSelectionUi { .press_image(self.imgs.icon_border_press) .with_tooltip( tooltip_manager, - &self.voxygen_i18n.get("common.weapons.bow"), + &self.i18n.get("common.weapons.bow"), "", &tooltip_human, TEXT_COLOR, @@ -1290,7 +1288,7 @@ impl CharSelectionUi { .press_image(self.imgs.icon_border_press) .with_tooltip( tooltip_manager, - &self.voxygen_i18n.get("common.weapons.staff"), + &self.i18n.get("common.weapons.staff"), "", &tooltip_human, TEXT_COLOR, @@ -1315,7 +1313,7 @@ impl CharSelectionUi { .press_image(self.imgs.icon_border_press) .with_tooltip( tooltip_manager, - &self.voxygen_i18n.get("common.weapons.sword"), + &self.i18n.get("common.weapons.sword"), "", &tooltip_human, TEXT_COLOR, @@ -1341,7 +1339,7 @@ impl CharSelectionUi { .press_image(self.imgs.icon_border_press) .with_tooltip( tooltip_manager, - &self.voxygen_i18n.get("common.weapons.hammer"), + &self.i18n.get("common.weapons.hammer"), "", &tooltip_human, TEXT_COLOR, @@ -1367,7 +1365,7 @@ impl CharSelectionUi { .press_image(self.imgs.icon_border_press) .with_tooltip( tooltip_manager, - &self.voxygen_i18n.get("common.weapons.axe"), + &self.i18n.get("common.weapons.axe"), "", &tooltip_human, TEXT_COLOR, @@ -1385,7 +1383,7 @@ impl CharSelectionUi { .press_image(self.imgs.dice_press) .with_tooltip( tooltip_manager, - &self.voxygen_i18n.get("common.rand_appearance"), + &self.i18n.get("common.rand_appearance"), "", &tooltip_human, TEXT_COLOR, @@ -1436,7 +1434,7 @@ impl CharSelectionUi { // Hair Style if let Some(new_val) = char_slider( self.ids.creation_buttons_alignment_2, - self.voxygen_i18n.get("char_selection.hair_style"), + self.i18n.get("char_selection.hair_style"), self.ids.hairstyle_text, body.species.num_hair_styles(body.body_type) as usize - 1, body.hair_style as usize, @@ -1448,7 +1446,7 @@ impl CharSelectionUi { // Hair Color if let Some(new_val) = char_slider( self.ids.hairstyle_slider, - self.voxygen_i18n.get("char_selection.hair_color"), + self.i18n.get("char_selection.hair_color"), self.ids.haircolor_text, body.species.num_hair_colors() as usize - 1, body.hair_color as usize, @@ -1460,7 +1458,7 @@ impl CharSelectionUi { // Skin if let Some(new_val) = char_slider( self.ids.haircolor_slider, - self.voxygen_i18n.get("char_selection.skin"), + self.i18n.get("char_selection.skin"), self.ids.skin_text, body.species.num_skin_colors() as usize - 1, body.skin as usize, @@ -1472,7 +1470,7 @@ impl CharSelectionUi { // Eyebrows if let Some(new_val) = char_slider( self.ids.skin_slider, - self.voxygen_i18n.get("char_selection.eyeshape"), + self.i18n.get("char_selection.eyeshape"), self.ids.eyebrows_text, body.species.num_eyes(body.body_type) as usize - 1, body.eyes as usize, @@ -1484,7 +1482,7 @@ impl CharSelectionUi { // EyeColor if let Some(new_val) = char_slider( self.ids.eyebrows_slider, - self.voxygen_i18n.get("char_selection.eye_color"), + self.i18n.get("char_selection.eye_color"), self.ids.eyecolor_text, body.species.num_eye_colors() as usize - 1, body.eye_color as usize, @@ -1497,7 +1495,7 @@ impl CharSelectionUi { let _current_accessory = body.accessory; if let Some(new_val) = char_slider( self.ids.eyecolor_slider, - self.voxygen_i18n.get("char_selection.accessories"), + self.i18n.get("char_selection.accessories"), self.ids.accessories_text, body.species.num_accessories(body.body_type) as usize - 1, body.accessory as usize, @@ -1510,7 +1508,7 @@ impl CharSelectionUi { if body.species.num_beards(body.body_type) > 1 { if let Some(new_val) = char_slider( self.ids.accessories_slider, - self.voxygen_i18n.get("char_selection.beard"), + self.i18n.get("char_selection.beard"), self.ids.beard_text, body.species.num_beards(body.body_type) as usize - 1, body.beard as usize, @@ -1520,7 +1518,7 @@ impl CharSelectionUi { body.beard = new_val as u8; } } else { - Text::new(&self.voxygen_i18n.get("char_selection.beard")) + Text::new(&self.i18n.get("char_selection.beard")) .mid_bottom_with_margin_on(self.ids.accessories_slider, -40.0) .font_size(self.fonts.cyri.scale(18)) .font_id(self.fonts.cyri.conrod_id) @@ -1541,7 +1539,7 @@ impl CharSelectionUi { .expect("Unable to load armor!"); if let Some(new_val) = char_slider( self.ids.beard_slider, - self.voxygen_i18n.get("char_selection.chest_color"), + self.i18n.get("char_selection.chest_color"), self.ids.chest_text, armor.len() - 1, armor diff --git a/voxygen/src/menu/main/mod.rs b/voxygen/src/menu/main/mod.rs index b2d685b92d..8674aa8003 100644 --- a/voxygen/src/menu/main/mod.rs +++ b/voxygen/src/menu/main/mod.rs @@ -47,7 +47,7 @@ impl PlayState for MainMenuState { fn tick(&mut self, global_state: &mut GlobalState, events: Vec) -> PlayStateResult { span!(_guard, "tick", "::tick"); - let localized_strings = crate::i18n::VoxygenLocalization::load_expect( + let localized_strings = crate::i18n::Localization::load_expect( &crate::i18n::i18n_asset_key(&global_state.settings.language.selected_language), ); diff --git a/voxygen/src/menu/main/ui.rs b/voxygen/src/menu/main/ui.rs index 49babfc263..db29d8e19f 100644 --- a/voxygen/src/menu/main/ui.rs +++ b/voxygen/src/menu/main/ui.rs @@ -1,9 +1,9 @@ use crate::{ - i18n::{i18n_asset_key, VoxygenLocalization}, + i18n::{i18n_asset_key, Localization}, render::Renderer, ui::{ self, - fonts::ConrodVoxygenFonts, + fonts::Fonts, ice::{Element, IcedUi}, img_ids::{BlankGraphic, ImageGraphic, VoxelGraphic}, Graphic, Ui, @@ -190,7 +190,7 @@ struct IcedState { } pub type Message = Event; impl IcedState { - pub fn view(&mut self) -> Element { + pub fn view(&mut self, i18n: &Localization) -> Element { use iced::{Align, Column, Container, Length, Row, Space, Text}; use ui::ice::{ compound_graphic::{CompoundGraphic, Graphic}, @@ -206,13 +206,30 @@ impl IcedState { Text::new("Quit"), ) .into(), */ - Text::new("Quit").size(20).into(), + Text::new(i18n.get("common.quit")).size(40).into(), ]) .width(Length::Fill) .max_width(200) .spacing(5) .padding(10); + // Quit + /*if Button::image(self.imgs.button) + .w_h(190.0, 40.0) + .bottom_left_with_margins_on(ui_widgets.window, 60.0, 30.0) + .hover_image(self.imgs.button_hover) + .press_image(self.imgs.button_press) + .label(i18n.get("common.quit")) + .label_font_id(self.fonts.cyri.conrod_id) + .label_color(TEXT_COLOR) + .label_font_size(self.fonts.cyri.scale(20)) + .label_y(Relative::Scalar(3.0)) + .set(self.ids.quit_button, ui_widgets) + .was_clicked() + { + events.push(Event::Quit); + }*/ + let buttons = Container::new(buttons) .width(Length::Fill) .height(Length::Fill) @@ -331,8 +348,8 @@ pub struct MainMenuUi { time: f32, anim_timer: f32, bg_img_id: conrod_core::image::Id, - voxygen_i18n: std::sync::Arc, - fonts: ConrodVoxygenFonts, + i18n: std::sync::Arc, + fonts: Fonts, tip_no: u16, pub show_iced: bool, } @@ -376,21 +393,22 @@ impl<'a> MainMenuUi { )); //let chosen_tip = *tips.choose(&mut rng).unwrap(); // Load language - let voxygen_i18n = VoxygenLocalization::load_expect(&i18n_asset_key( + let i18n = Localization::load_expect(&i18n_asset_key( &global_state.settings.language.selected_language, )); // Load fonts. - let fonts = ConrodVoxygenFonts::load(&voxygen_i18n.fonts, &mut ui) - .expect("Impossible to load fonts!"); + let fonts = Fonts::load(&i18n.fonts, &mut ui).expect("Impossible to load fonts!"); // TODO: newtype Font let ice_font = { use std::io::Read; let mut buf = Vec::new(); - common::assets::load_file("voxygen.font.OpenSans-Regular", &["ttf"]) - .unwrap() - .read_to_end(&mut buf) - .unwrap(); + common::assets::load_file("voxygen.font.haxrcorp_4089_cyrillic_altgr_extended", &[ + "ttf", + ]) + .unwrap() + .read_to_end(&mut buf) + .unwrap(); glyph_brush::rusttype::Font::from_bytes(buf).unwrap() }; @@ -421,7 +439,7 @@ impl<'a> MainMenuUi { anim_timer: 0.0, //show_disclaimer: global_state.settings.show_disclaimer, bg_img_id, - voxygen_i18n, + i18n, fonts, tip_no: 0, show_iced: false, @@ -438,8 +456,8 @@ impl<'a> MainMenuUi { let (ref mut ui_widgets, ref mut _tooltip_manager) = self.ui.set_widgets(); let tip_msg = format!( "{} {}", - &self.voxygen_i18n.get("main.tip"), - &self.voxygen_i18n.get_variation("loading.tips", self.tip_no), + &self.i18n.get("main.tip"), + &self.i18n.get_variation("loading.tips", self.tip_no), ); let tip_show = global_state.settings.gameplay.loading_tips; let mut rng = thread_rng(); @@ -450,7 +468,7 @@ impl<'a> MainMenuUi { const TEXT_BG: Color = Color::Rgba(0.0, 0.0, 0.0, 1.0); //const INACTIVE: Color = Color::Rgba(0.47, 0.47, 0.47, 0.47); - let intro_text = &self.voxygen_i18n.get("main.login_process"); + let intro_text = &self.i18n.get("main.login_process"); // Tooltip /*let _tooltip = Tooltip::new({ @@ -614,9 +632,9 @@ impl<'a> MainMenuUi { .press_image(self.imgs.button_press) .label_y(Relative::Scalar(2.0)) .label(match popup_type { - PopupType::Error => self.voxygen_i18n.get("common.okay"), - PopupType::ConnectionInfo => self.voxygen_i18n.get("common.cancel"), - PopupType::AuthTrustPrompt(_) => self.voxygen_i18n.get("common.cancel"), + PopupType::Error => self.i18n.get("common.okay"), + PopupType::ConnectionInfo => self.i18n.get("common.cancel"), + PopupType::AuthTrustPrompt(_) => self.i18n.get("common.cancel"), }) .label_font_id(self.fonts.cyri.conrod_id) .label_font_size(self.fonts.cyri.scale(15)) @@ -652,7 +670,7 @@ impl<'a> MainMenuUi { { events.push(Event::AuthServerTrust(auth_server.clone(), true)); change_popup = Some(Some(PopupData { - msg: self.voxygen_i18n.get("main.connecting").into(), + msg: self.i18n.get("main.connecting").into(), popup_type: PopupType::ConnectionInfo, })); } @@ -690,13 +708,13 @@ impl<'a> MainMenuUi { .scroll_kids_vertically() .set(self.ids.disc_window, ui_widgets); - Text::new(&self.voxygen_i18n.get("common.disclaimer")) + Text::new(&self.i18n.get("common.disclaimer")) .top_left_with_margins_on(self.ids.disc_window, 30.0, 40.0) .font_size(self.fonts.cyri.scale(35)) .font_id(self.fonts.alkhemi.conrod_id) .color(TEXT_COLOR) .set(self.ids.disc_text_1, ui_widgets); - Text::new(&self.voxygen_i18n.get("main.notice")) + Text::new(&self.i18n.get("main.notice")) .top_left_with_margins_on(self.ids.disc_window, 110.0, 40.0) .font_size(self.fonts.cyri.scale(26)) .font_id(self.fonts.cyri.conrod_id) @@ -708,7 +726,7 @@ impl<'a> MainMenuUi { .hover_image(self.imgs.button_hover) .press_image(self.imgs.button_press) .label_y(Relative::Scalar(2.0)) - .label(&self.voxygen_i18n.get("common.accept")) + .label(&self.i18n.get("common.accept")) .label_font_size(self.fonts.cyri.scale(22)) .label_color(TEXT_COLOR) .label_font_id(self.fonts.cyri.conrod_id) @@ -727,7 +745,7 @@ impl<'a> MainMenuUi { self.connect = true; self.connecting = Some(std::time::Instant::now()); self.popup = Some(PopupData { - msg: [self.voxygen_i18n.get("main.connecting"), "..."].concat(), + msg: [self.i18n.get("main.connecting"), "..."].concat(), popup_type: PopupType::ConnectionInfo, }); @@ -764,7 +782,7 @@ impl<'a> MainMenuUi { self.connect = true; self.connecting = Some(std::time::Instant::now()); self.popup = Some(PopupData { - msg: [self.voxygen_i18n.get(""), ""].concat(), + msg: [self.i18n.get(""), ""].concat(), popup_type: PopupType::ConnectionInfo, }); }; @@ -893,7 +911,7 @@ impl<'a> MainMenuUi { .hover_image(self.imgs.button_hover) .press_image(self.imgs.button_press) .label_y(Relative::Scalar(2.0)) - .label(&self.voxygen_i18n.get("common.close")) + .label(&self.i18n.get("common.close")) .label_font_size(self.fonts.cyri.scale(20)) .label_font_id(self.fonts.cyri.conrod_id) .label_color(TEXT_COLOR) @@ -942,7 +960,7 @@ impl<'a> MainMenuUi { .w_h(258.0*scale, 55.0*scale) .down_from(self.ids.address_bg, 20.0*scale) .align_middle_x_of(self.ids.address_bg) - .label(&self.voxygen_i18n.get("common.multiplayer")) + .label(&self.i18n.get("common.multiplayer")) .label_font_id(self.fonts.cyri.conrod_id) .label_color(TEXT_COLOR) .label_font_size(self.fonts.cyri.scale(18)) @@ -970,7 +988,7 @@ impl<'a> MainMenuUi { .w_h(258.0 * scale, 55.0 * scale) .down_from(self.ids.login_button, 20.0 * scale) .align_middle_x_of(self.ids.address_bg) - .label(&self.voxygen_i18n.get("common.singleplayer")) + .label(&self.i18n.get("common.singleplayer")) .label_font_id(self.fonts.cyri.conrod_id) .label_color(TEXT_COLOR) .label_font_size(self.fonts.cyri.scale(18)) @@ -989,7 +1007,7 @@ impl<'a> MainMenuUi { .bottom_left_with_margins_on(ui_widgets.window, 60.0 * scale, 30.0 * scale) .hover_image(self.imgs.button_hover) .press_image(self.imgs.button_press) - .label(&self.voxygen_i18n.get("common.quit")) + .label(&self.i18n.get("common.quit")) .label_font_id(self.fonts.cyri.conrod_id) .label_color(TEXT_COLOR) .label_font_size(self.fonts.cyri.scale(16)) @@ -1006,7 +1024,7 @@ impl<'a> MainMenuUi { .up_from(self.ids.quit_button, 8.0*scale) //.hover_image(self.imgs.button_hover) //.press_image(self.imgs.button_press) - .label(&self.voxygen_i18n.get("common.settings")) + .label(&self.i18n.get("common.settings")) .label_font_id(self.fonts.cyri.conrod_id) .label_color(TEXT_COLOR_2) .label_font_size(self.fonts.cyri.scale(16)) @@ -1023,7 +1041,7 @@ impl<'a> MainMenuUi { .up_from(self.ids.settings_button, 8.0 * scale) .hover_image(self.imgs.button_hover) .press_image(self.imgs.button_press) - .label(&self.voxygen_i18n.get("common.servers")) + .label(&self.i18n.get("common.servers")) .label_font_id(self.fonts.cyri.conrod_id) .label_color(TEXT_COLOR) .label_font_size(self.fonts.cyri.scale(16)) @@ -1079,8 +1097,10 @@ impl<'a> MainMenuUi { pub fn maintain(&mut self, global_state: &mut GlobalState, dt: Duration) -> Vec { let events = self.update_layout(global_state, dt); self.ui.maintain(global_state.window.renderer_mut(), None); - self.ice_ui - .maintain(self.ice_state.view(), global_state.window.renderer_mut()); + self.ice_ui.maintain( + self.ice_state.view(&self.i18n), + global_state.window.renderer_mut(), + ); events } diff --git a/voxygen/src/session.rs b/voxygen/src/session.rs index c94d93071c..0a5f8e5503 100644 --- a/voxygen/src/session.rs +++ b/voxygen/src/session.rs @@ -2,7 +2,7 @@ use crate::{ audio::sfx::{SfxEvent, SfxEventItem}, ecs::MyEntity, hud::{DebugInfo, Event as HudEvent, Hud, HudInfo, PressBehavior}, - i18n::{i18n_asset_key, VoxygenLocalization}, + i18n::{i18n_asset_key, Localization}, key_state::KeyState, menu::char_selection::CharSelectionState, render::Renderer, @@ -46,7 +46,7 @@ pub struct SessionState { key_state: KeyState, inputs: comp::ControllerInputs, selected_block: Block, - voxygen_i18n: std::sync::Arc, + i18n: std::sync::Arc, walk_forward_dir: Vec2, walk_right_dir: Vec2, freefly_vel: Vec3, @@ -72,7 +72,7 @@ impl SessionState { .camera_mut() .set_fov_deg(global_state.settings.graphics.fov); let hud = Hud::new(global_state, &client.borrow()); - let voxygen_i18n = VoxygenLocalization::load_expect(&i18n_asset_key( + let i18n = Localization::load_expect(&i18n_asset_key( &global_state.settings.language.selected_language, )); @@ -86,7 +86,7 @@ impl SessionState { inputs: comp::ControllerInputs::default(), hud, selected_block: Block::new(BlockKind::Misc, Rgb::broadcast(255)), - voxygen_i18n, + i18n, walk_forward_dir, walk_right_dir, freefly_vel: Vec3::zero(), @@ -131,14 +131,14 @@ impl SessionState { match inv_event { InventoryUpdateEvent::CollectFailed => { self.hud.new_message(ChatMsg { - message: self.voxygen_i18n.get("hud.chat.loot_fail").to_string(), + message: self.i18n.get("hud.chat.loot_fail").to_string(), chat_type: ChatType::CommandError, }); }, InventoryUpdateEvent::Collected(item) => { self.hud.new_message(ChatMsg { message: self - .voxygen_i18n + .i18n .get("hud.chat.loot_msg") .replace("{item}", item.name()), chat_type: ChatType::Loot, @@ -150,9 +150,9 @@ impl SessionState { client::Event::Disconnect => return Ok(TickAction::Disconnect), client::Event::DisconnectionNotification(time) => { let message = match time { - 0 => String::from(self.voxygen_i18n.get("hud.chat.goodbye")), + 0 => String::from(self.i18n.get("hud.chat.goodbye")), _ => self - .voxygen_i18n + .i18n .get("hud.chat.connection_lost") .replace("{time}", time.to_string().as_str()), }; @@ -165,7 +165,7 @@ impl SessionState { client::Event::Kicked(reason) => { global_state.info_message = Some(format!( "{}: {}", - self.voxygen_i18n.get("main.login.kicked").to_string(), + self.i18n.get("main.login.kicked").to_string(), reason )); return Ok(TickAction::Disconnect); @@ -207,7 +207,7 @@ impl PlayState for SessionState { span!(_guard, "tick", "::tick"); // TODO: let mut client = self.client.borrow_mut(); // NOTE: Not strictly necessary, but useful for hotloading translation changes. - self.voxygen_i18n = VoxygenLocalization::load_expect(&i18n_asset_key( + self.i18n = Localization::load_expect(&i18n_asset_key( &global_state.settings.language.selected_language, )); @@ -673,7 +673,7 @@ impl PlayState for SessionState { Ok(TickAction::Disconnect) => return PlayStateResult::Pop, // Go to main menu Err(err) => { global_state.info_message = - Some(self.voxygen_i18n.get("common.connection_lost").to_owned()); + Some(self.i18n.get("common.connection_lost").to_owned()); error!("[session] Failed to tick the scene: {:?}", err); return PlayStateResult::Pop; @@ -752,7 +752,7 @@ impl PlayState for SessionState { // Look for changes in the localization files if global_state.localization_watcher.reloaded() { hud_events.push(HudEvent::ChangeLanguage(Box::new( - self.voxygen_i18n.metadata.clone(), + self.i18n.metadata.clone(), ))); } @@ -980,13 +980,13 @@ impl PlayState for SessionState { HudEvent::ChangeLanguage(new_language) => { global_state.settings.language.selected_language = new_language.language_identifier; - self.voxygen_i18n = VoxygenLocalization::load_watched( + self.i18n = Localization::load_watched( &i18n_asset_key(&global_state.settings.language.selected_language), &mut global_state.localization_watcher, ) .unwrap(); - self.voxygen_i18n.log_missing_entries(); - self.hud.update_language(Arc::clone(&self.voxygen_i18n)); + self.i18n.log_missing_entries(); + self.hud.update_language(Arc::clone(&self.i18n)); }, HudEvent::ChangeFullscreenMode(new_fullscreen_settings) => { global_state diff --git a/voxygen/src/ui/fonts.rs b/voxygen/src/ui/fonts.rs index 9c726eadba..05ba6d3aab 100644 --- a/voxygen/src/ui/fonts.rs +++ b/voxygen/src/ui/fonts.rs @@ -1,14 +1,14 @@ -use crate::i18n::{Font, VoxygenFonts}; +use crate::i18n; use common::assets::Asset; -pub struct ConrodVoxygenFont { - metadata: Font, +pub struct Font { + metadata: i18n::Font, pub conrod_id: conrod_core::text::font::Id, } -impl ConrodVoxygenFont { +impl Font { #[allow(clippy::needless_return)] // TODO: Pending review in #587 - pub fn new(font: &Font, ui: &mut crate::ui::Ui) -> ConrodVoxygenFont { + pub fn new(font: &i18n::Font, ui: &mut crate::ui::Ui) -> Font { return Self { metadata: font.clone(), conrod_id: ui.new_font(crate::ui::Font::load_expect(&font.asset_key)), @@ -22,14 +22,14 @@ impl ConrodVoxygenFont { macro_rules! conrod_fonts { ($([ $( $name:ident$(,)? )* ])*) => { $( - pub struct ConrodVoxygenFonts { - $(pub $name: ConrodVoxygenFont,)* + pub struct Fonts { + $(pub $name: Font,)* } - impl ConrodVoxygenFonts { - pub fn load(voxygen_fonts: &VoxygenFonts, ui: &mut crate::ui::Ui) -> Result { + impl Fonts { + pub fn load(fonts: &i18n::Fonts, ui: &mut crate::ui::Ui) -> Result { Ok(Self { - $( $name: ConrodVoxygenFont::new(voxygen_fonts.get(stringify!($name)).unwrap(), ui),)* + $( $name: Font::new(fonts.get(stringify!($name)).unwrap(), ui),)* }) } } From 937e1e696cd923be68664e0b7b3e93331c60b6c0 Mon Sep 17 00:00:00 2001 From: Imbris Date: Tue, 26 May 2020 18:26:16 -0400 Subject: [PATCH 12/61] Add support for Button widget, new custom widget AspectRatioContainer --- voxygen/src/menu/main/ui.rs | 60 ++++-- voxygen/src/ui/ice/mod.rs | 3 +- .../ui/ice/renderer/aspect_ratio_container.rs | 25 +++ voxygen/src/ui/ice/renderer/button.rs | 165 +++++++++++++++ voxygen/src/ui/ice/renderer/image.rs | 9 +- .../ui/ice/{renderer.rs => renderer/mod.rs} | 29 ++- voxygen/src/ui/ice/renderer/text.rs | 28 +-- voxygen/src/ui/ice/widget.rs | 1 + .../ui/ice/widget/aspect_ratio_container.rs | 189 ++++++++++++++++++ .../src/ui/ice/widget/background_container.rs | 29 ++- voxygen/src/ui/ice/widget/compound_graphic.rs | 4 +- voxygen/src/ui/ice/widget/image.rs | 4 +- 12 files changed, 500 insertions(+), 46 deletions(-) create mode 100644 voxygen/src/ui/ice/renderer/aspect_ratio_container.rs create mode 100644 voxygen/src/ui/ice/renderer/button.rs rename voxygen/src/ui/ice/{renderer.rs => renderer/mod.rs} (97%) create mode 100644 voxygen/src/ui/ice/widget/aspect_ratio_container.rs diff --git a/voxygen/src/menu/main/ui.rs b/voxygen/src/menu/main/ui.rs index db29d8e19f..b10e64aad1 100644 --- a/voxygen/src/menu/main/ui.rs +++ b/voxygen/src/menu/main/ui.rs @@ -158,6 +158,7 @@ rotation_image_ids! { } } +#[derive(Clone)] // TODO: why does iced require Clone? pub enum Event { LoginAttempt { username: String, @@ -187,26 +188,59 @@ pub struct PopupData { // No state currently struct IcedState { imgs: IcedImgs, + quit_button: iced::button::State, } pub type Message = Event; impl IcedState { + pub fn new(imgs: IcedImgs) -> Self { + Self { + imgs, + quit_button: iced::button::State::new(), + } + } + pub fn view(&mut self, i18n: &Localization) -> Element { - use iced::{Align, Column, Container, Length, Row, Space, Text}; + // TODO: scale with window size + let button_font_size = 30; + const TEXT_COLOR: iced::Color = iced::Color::from_rgb(1.0, 1.0, 1.0); + const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1.0, 0.2); + + use iced::{ + Align, Button, Column, Container, HorizontalAlignment, Length, Row, Space, Text, + VerticalAlignment, + }; use ui::ice::{ compound_graphic::{CompoundGraphic, Graphic}, - BackgroundContainer, Image, Padding, + AspectRatioContainer, BackgroundContainer, ButtonStyle, Image, Padding, }; use vek::*; + let button_style = ButtonStyle::new(self.imgs.button) + .hover_image(self.imgs.button_hover) + .press_image(self.imgs.button_press) + .text_color(TEXT_COLOR) + .disabled_text_color(DISABLED_TEXT_COLOR); + let buttons = Column::with_children(vec![ Image::new(self.imgs.button).fix_aspect_ratio().into(), Image::new(self.imgs.button).fix_aspect_ratio().into(), - /* BackgroundContainer::new( - Image::new(self.imgs.button).fix_aspect_ratio(), - Text::new("Quit"), + AspectRatioContainer::new( + Button::new( + &mut self.quit_button, + Text::new(i18n.get("common.quit")) + .size(button_font_size) + .height(Length::Fill) + .width(Length::Fill) + .horizontal_alignment(HorizontalAlignment::Center) + .vertical_alignment(VerticalAlignment::Center), + ) + .height(Length::Fill) + .width(Length::Fill) + .style(button_style) + .on_press(Message::Quit), ) - .into(), */ - Text::new(i18n.get("common.quit")).size(40).into(), + .ratio_of_image(self.imgs.button) + .into(), ]) .width(Length::Fill) .max_width(200) @@ -314,13 +348,7 @@ impl IcedState { .height(Length::Fill) .spacing(10); - BackgroundContainer::new( - Image::new(self.imgs.bg) - .width(Length::Fill) - .height(Length::Fill), - content, - ) - .into() + BackgroundContainer::new(Image::new(self.imgs.bg), content).into() } pub fn update(message: Message) { @@ -413,9 +441,7 @@ impl<'a> MainMenuUi { }; let mut ice_ui = IcedUi::new(window, ice_font).unwrap(); - let ice_state = IcedState { - imgs: IcedImgs::load(&mut ice_ui).expect("Failed to load images"), - }; + let ice_state = IcedState::new(IcedImgs::load(&mut ice_ui).expect("Failed to load images")); Self { ui, diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index 7d24651b5c..a3e8745b98 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -8,8 +8,9 @@ mod winit_conversion; pub use cache::Font; pub use graphic::{Id, Rotation}; pub use iced::Event; -pub use renderer::IcedRenderer; +pub use renderer::{ButtonStyle, IcedRenderer}; pub use widget::{ + aspect_ratio_container::AspectRatioContainer, background_container::{BackgroundContainer, Padding}, compound_graphic, image::Image, diff --git a/voxygen/src/ui/ice/renderer/aspect_ratio_container.rs b/voxygen/src/ui/ice/renderer/aspect_ratio_container.rs new file mode 100644 index 0000000000..71c0f13f3b --- /dev/null +++ b/voxygen/src/ui/ice/renderer/aspect_ratio_container.rs @@ -0,0 +1,25 @@ +use super::{ + super::widget::{aspect_ratio_container, image}, + IcedRenderer, +}; +use iced::{Element, Layout, Point, Rectangle}; + +impl aspect_ratio_container::Renderer for IcedRenderer { + //type Style + type ImageHandle = image::Handle; + + fn dimensions(&self, handle: &Self::ImageHandle) -> (u32, u32) { self.image_dims(*handle) } + + fn draw( + &mut self, + defaults: &Self::Defaults, + _bounds: Rectangle, + cursor_position: Point, + //style: &Self::Style, + content: &Element<'_, M, Self>, + content_layout: Layout<'_>, + ) -> Self::Output { + // TODO: stlying to add a background image and such + content.draw(self, defaults, content_layout, cursor_position) + } +} diff --git a/voxygen/src/ui/ice/renderer/button.rs b/voxygen/src/ui/ice/renderer/button.rs new file mode 100644 index 0000000000..30754d512f --- /dev/null +++ b/voxygen/src/ui/ice/renderer/button.rs @@ -0,0 +1,165 @@ +use super::{super::Rotation, widget::image, Defaults, IcedRenderer, Primitive}; +use iced::{button, mouse, Color, Element, Layout, Point, Rectangle}; +use vek::Rgba; + +#[derive(Clone, Copy)] +struct Background { + default: image::Handle, + hover: image::Handle, + press: image::Handle, +} + +impl Background { + fn new(image: image::Handle) -> Self { + Self { + default: image, + hover: image, + press: image, + } + } +} +// TODO: consider a different place for this +// Note: for now all buttons have an image background +#[derive(Clone, Copy)] +pub struct Style { + background: Option, + enabled_text: Color, + disabled_text: Color, + /* greying out / changing text color + *disabled: , */ +} + +impl Style { + pub fn new(image: image::Handle) -> Self { + Self { + background: Some(Background::new(image)), + ..Default::default() + } + } + + pub fn hover_image(mut self, image: image::Handle) -> Self { + self.background = Some(match self.background { + Some(mut background) => { + background.hover = image; + background + }, + None => Background::new(image), + }); + self + } + + pub fn press_image(mut self, image: image::Handle) -> Self { + self.background = Some(match self.background { + Some(mut background) => { + background.press = image; + background + }, + None => Background::new(image), + }); + self + } + + pub fn text_color(mut self, color: Color) -> Self { + self.enabled_text = color; + self + } + + pub fn disabled_text_color(mut self, color: Color) -> Self { + self.disabled_text = color; + self + } + + fn disabled(&self) -> (Option, Color) { + ( + self.background.as_ref().map(|b| b.default), + self.disabled_text, + ) + } + + fn pressed(&self) -> (Option, Color) { + (self.background.as_ref().map(|b| b.press), self.enabled_text) + } + + fn hovered(&self) -> (Option, Color) { + (self.background.as_ref().map(|b| b.hover), self.enabled_text) + } + + fn active(&self) -> (Option, Color) { + ( + self.background.as_ref().map(|b| b.default), + self.enabled_text, + ) + } +} + +impl Default for Style { + fn default() -> Self { + Self { + background: None, + enabled_text: Color::WHITE, + disabled_text: Color::from_rgb(0.5, 0.5, 0.5), + } + } +} + +impl button::Renderer for IcedRenderer { + // TODO: what if this gets large enough to not be copied around? + type Style = Style; + + const DEFAULT_PADDING: u16 = 0; + + fn draw( + &mut self, + defaults: &Self::Defaults, + bounds: Rectangle, + cursor_position: Point, + is_disabled: bool, + is_pressed: bool, + style: &Self::Style, + content: &Element<'_, M, Self>, + content_layout: Layout<'_>, + ) -> Self::Output { + let is_mouse_over = bounds.contains(cursor_position); + + let (maybe_image, text_color) = if is_disabled { + style.disabled() + } else if is_mouse_over { + if is_pressed { + style.pressed() + } else { + style.hovered() + } + } else { + style.active() + }; + + let (content, _) = content.draw( + self, + &Defaults { text_color }, + content_layout, + cursor_position, + ); + + let primitive = if let Some(handle) = maybe_image { + let background = Primitive::Image { + handle: (handle, Rotation::None), + bounds, + color: Rgba::broadcast(255), + }; + + Primitive::Group { + primitives: vec![background, content], + } + } else { + content + }; + + let mouse_interaction = if is_mouse_over { + mouse::Interaction::Pointer + } else { + mouse::Interaction::default() + }; + + (primitive, mouse_interaction) + } +} diff --git a/voxygen/src/ui/ice/renderer/image.rs b/voxygen/src/ui/ice/renderer/image.rs index 710bf386b7..829bef748b 100644 --- a/voxygen/src/ui/ice/renderer/image.rs +++ b/voxygen/src/ui/ice/renderer/image.rs @@ -6,14 +6,7 @@ use iced::mouse; use vek::Rgba; impl image::Renderer for IcedRenderer { - fn dimensions(&self, handle: image::Handle) -> (u32, u32) { - self - .cache - .graphic_cache() - .get_graphic_dims((handle, Rotation::None)) - // TODO: don't unwrap - .unwrap() - } + fn dimensions(&self, handle: image::Handle) -> (u32, u32) { self.image_dims(handle) } fn draw( &mut self, diff --git a/voxygen/src/ui/ice/renderer.rs b/voxygen/src/ui/ice/renderer/mod.rs similarity index 97% rename from voxygen/src/ui/ice/renderer.rs rename to voxygen/src/ui/ice/renderer/mod.rs index 04d7f01830..3ed49040c3 100644 --- a/voxygen/src/ui/ice/renderer.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -1,4 +1,7 @@ +// TODO: reorganize modules (e.g. put all these in a widget submodule) +mod aspect_ratio_container; mod background_container; +mod button; mod column; mod compound_graphic; mod container; @@ -7,6 +10,8 @@ mod row; mod space; mod text; +pub use button::Style as ButtonStyle; + use super::{ super::graphic::{self, Graphic, TexId}, cache::Cache, @@ -157,6 +162,15 @@ impl IcedRenderer { self.cache.add_graphic(graphic) } + fn image_dims(&self, handle: super::widget::image::Handle) -> (u32, u32) { + self + .cache + .graphic_cache() + .get_graphic_dims((handle, Rotation::None)) + // TODO: don't unwrap + .unwrap() + } + pub fn resize(&mut self, scaled_dims: Vec2, renderer: &mut Renderer) { self.win_dims = scaled_dims; @@ -582,9 +596,22 @@ fn default_scissor(renderer: &Renderer) -> Aabr { } } +// TODO: expose to user +pub struct Defaults { + pub text_color: iced::Color, +} + +impl Default for Defaults { + fn default() -> Self { + Self { + text_color: iced::Color::WHITE, + } + } +} + impl iced::Renderer for IcedRenderer { // Default styling - type Defaults = (); + type Defaults = Defaults; // TODO: use graph of primitives to enable diffing??? type Output = (Primitive, iced::mouse::Interaction); diff --git a/voxygen/src/ui/ice/renderer/text.rs b/voxygen/src/ui/ice/renderer/text.rs index a8067c860c..5ffb25d44d 100644 --- a/voxygen/src/ui/ice/renderer/text.rs +++ b/voxygen/src/ui/ice/renderer/text.rs @@ -28,7 +28,7 @@ impl text::Renderer for IcedRenderer { fn draw( &mut self, - _defaults: &Self::Defaults, + defaults: &Self::Defaults, bounds: Rectangle, content: &str, size: u16, @@ -38,25 +38,27 @@ impl text::Renderer for IcedRenderer { vertical_alignment: VerticalAlignment, ) -> Self::Output { use glyph_brush::{HorizontalAlign, VerticalAlign}; - let h_align = match horizontal_alignment { - HorizontalAlignment::Left => HorizontalAlign::Left, - HorizontalAlignment::Center => HorizontalAlign::Center, - HorizontalAlignment::Right => HorizontalAlign::Right, + // glyph_brush thought it would be a great idea to change what the bounds and + // position mean based on the alignment + // TODO: add option to align based on the geometry of the rendered glyphs + // instead of all possible glyphs + let (x, h_align) = match horizontal_alignment { + HorizontalAlignment::Left => (bounds.x, HorizontalAlign::Left), + HorizontalAlignment::Center => (bounds.center_x(), HorizontalAlign::Center), + HorizontalAlignment::Right => (bounds.x + bounds.width, HorizontalAlign::Right), }; - let v_align = match vertical_alignment { - VerticalAlignment::Top => VerticalAlign::Top, - VerticalAlignment::Center => VerticalAlign::Center, - VerticalAlignment::Bottom => VerticalAlign::Bottom, + let (y, v_align) = match vertical_alignment { + VerticalAlignment::Top => (bounds.y, VerticalAlign::Top), + VerticalAlignment::Center => (bounds.center_y(), VerticalAlign::Center), + VerticalAlignment::Bottom => (bounds.y + bounds.height, VerticalAlign::Bottom), }; let p_scale = self.p_scale; let section = glyph_brush::Section { text: content, - // TODO: do snap to pixel thing here IF it is being done down the line - //screen_position: (bounds.x * p_scale, (self.win_dims.y - bounds.y) * p_scale), - screen_position: (bounds.x * p_scale, bounds.y * p_scale), + screen_position: (x * p_scale, y * p_scale), bounds: (bounds.width * p_scale, bounds.height * p_scale), scale: glyph_brush::rusttype::Scale::uniform(size as f32 * p_scale), layout: glyph_brush::Layout::Wrap { @@ -86,7 +88,7 @@ impl text::Renderer for IcedRenderer { glyphs, //size: size as f32, bounds, - linear_color: color.unwrap_or(Color::BLACK).into_linear().into(), + linear_color: color.unwrap_or(defaults.text_color).into_linear().into(), /*font, *horizontal_alignment, *vertical_alignment, */ diff --git a/voxygen/src/ui/ice/widget.rs b/voxygen/src/ui/ice/widget.rs index ebf8ee659b..85b4968588 100644 --- a/voxygen/src/ui/ice/widget.rs +++ b/voxygen/src/ui/ice/widget.rs @@ -1,3 +1,4 @@ +pub mod aspect_ratio_container; pub mod background_container; pub mod compound_graphic; pub mod image; diff --git a/voxygen/src/ui/ice/widget/aspect_ratio_container.rs b/voxygen/src/ui/ice/widget/aspect_ratio_container.rs new file mode 100644 index 0000000000..f92832f987 --- /dev/null +++ b/voxygen/src/ui/ice/widget/aspect_ratio_container.rs @@ -0,0 +1,189 @@ +use iced::{layout, Clipboard, Element, Event, Hasher, Layout, Length, Point, Rectangle, Widget}; +use std::{hash::Hash, u32}; + +// Note: it might be more efficient to make this generic over the content type? + +enum AspectRatio { + /// Image Id + Image(I), + /// width / height + Ratio(f32), +} + +impl Hash for AspectRatio { + fn hash(&self, state: &mut H) { + match self { + Self::Image(i) => i.hash(state), + Self::Ratio(r) => r.to_bits().hash(state), + } + } +} + +/// Provides a container that takes on a fixed aspect ratio +/// Thus, can be used to fix the aspect ratio of content if it is set to +/// Length::Fill The aspect ratio may be based on that of an image, in which +/// case the ratio is obtained from the renderer +pub struct AspectRatioContainer<'a, M, R: self::Renderer> { + max_width: u32, + max_height: u32, + aspect_ratio: AspectRatio, + content: Element<'a, M, R>, +} + +impl<'a, M, R> AspectRatioContainer<'a, M, R> +where + R: self::Renderer, +{ + pub fn new(content: impl Into>) -> Self { + Self { + max_width: u32::MAX, + max_height: u32::MAX, + aspect_ratio: AspectRatio::Ratio(1.0), + content: content.into(), + } + } + + /// Set the ratio (width/height) + pub fn ratio(mut self, ratio: f32) -> Self { + self.aspect_ratio = AspectRatio::Ratio(ratio); + self + } + + /// Use the ratio of the provided image + pub fn ratio_of_image(mut self, handle: R::ImageHandle) -> Self { + self.aspect_ratio = AspectRatio::Image(handle); + self + } + + pub fn max_width(mut self, max_width: u32) -> Self { + self.max_width = max_width; + self + } + + pub fn max_height(mut self, max_height: u32) -> Self { + self.max_height = max_height; + self + } +} + +impl<'a, M, R> Widget for AspectRatioContainer<'a, M, R> +where + R: self::Renderer, +{ + fn width(&self) -> Length { Length::Fill } + + fn height(&self) -> Length { Length::Fill } + + fn layout(&self, renderer: &R, limits: &layout::Limits) -> layout::Node { + let limits = limits + .loose() + .max_width(self.max_width) + .max_height(self.max_height) + .width(self.width()) + .height(self.height()); + + let aspect_ratio = match &self.aspect_ratio { + AspectRatio::Image(handle) => { + let (pixel_w, pixel_h) = renderer.dimensions(handle); + + // Just in case + // could convert to gracefully handling + debug_assert!(pixel_w != 0); + debug_assert!(pixel_h != 0); + + pixel_w as f32 / pixel_h as f32 + }, + AspectRatio::Ratio(ratio) => *ratio, + }; + + // We need to figure out the max width/height of the limits + // and then adjust one down to meet the aspect ratio + let max_size = limits.max(); + let (max_width, max_height) = (max_size.width as f32, max_size.height as f32); + let max_aspect_ratio = max_width / max_height; + let limits = if max_aspect_ratio > aspect_ratio { + limits.max_width((max_height * aspect_ratio) as u32) + } else { + limits.max_height((max_width / aspect_ratio) as u32) + }; + + let content = self.content.layout(renderer, &limits.loose()); + + layout::Node::with_children(limits.max(), vec![content]) + } + + fn on_event( + &mut self, + event: Event, + layout: Layout<'_>, + cursor_position: Point, + messages: &mut Vec, + renderer: &R, + clipboard: Option<&dyn Clipboard>, + ) { + self.content.on_event( + event, + layout.children().next().unwrap(), + cursor_position, + messages, + renderer, + clipboard, + ); + } + + fn draw( + &self, + renderer: &mut R, + defaults: &R::Defaults, + layout: Layout<'_>, + cursor_position: Point, + ) -> R::Output { + renderer.draw( + defaults, + layout.bounds(), + cursor_position, + &self.content, + layout.children().next().unwrap(), + ) + } + + fn hash_layout(&self, state: &mut Hasher) { + struct Marker; + std::any::TypeId::of::().hash(state); + + self.max_width.hash(state); + self.max_height.hash(state); + self.aspect_ratio.hash(state); + // TODO: add pixel dims (need renderer) + + self.content.hash_layout(state); + } +} + +pub trait Renderer: iced::Renderer { + /// The style supported by this renderer. + //type Style: Default; + /// The handle used by this renderer for images. + type ImageHandle: Hash; + + fn dimensions(&self, handle: &Self::ImageHandle) -> (u32, u32); + + fn draw( + &mut self, + defaults: &Self::Defaults, + bounds: Rectangle, + cursor_position: Point, + //style: &Self::Style, + content: &Element<'_, M, Self>, + content_layout: Layout<'_>, + ) -> Self::Output; +} + +// They got to live ¯\_(ツ)_/¯ +impl<'a, M, R> From> for Element<'a, M, R> +where + R: 'a + self::Renderer, + M: 'a, +{ + fn from(widget: AspectRatioContainer<'a, M, R>) -> Element<'a, M, R> { Element::new(widget) } +} diff --git a/voxygen/src/ui/ice/widget/background_container.rs b/voxygen/src/ui/ice/widget/background_container.rs index a4fa05f7a6..af94fbdedb 100644 --- a/voxygen/src/ui/ice/widget/background_container.rs +++ b/voxygen/src/ui/ice/widget/background_container.rs @@ -1,4 +1,4 @@ -use iced::{layout, Element, Hasher, Layout, Length, Point, Size, Widget}; +use iced::{layout, Clipboard, Element, Event, Hasher, Layout, Length, Point, Size, Widget}; use std::{hash::Hash, u32}; // Note: it might be more efficient to make this generic over the content type @@ -63,7 +63,7 @@ pub trait Background: Sized { fn width(&self) -> Length; fn height(&self) -> Length; fn aspect_ratio_fixed(&self) -> bool; - fn pixel_dims(&self, renderer: &R) -> [u16; 2]; + fn pixel_dims(&self, renderer: &R) -> (u16, u16); fn draw( &self, renderer: &mut R, @@ -158,7 +158,7 @@ where .width(self.width()) .height(self.height()); - let [pixel_w, pixel_h] = self.background.pixel_dims(renderer); + let (pixel_w, pixel_h) = self.background.pixel_dims(renderer); let (horizontal_pad_frac, vertical_pad_frac, top_pad_frac, left_pad_frac) = { let Padding { top, @@ -208,6 +208,10 @@ where // again, why is loose() used here? let mut content = self.content.layout(renderer, &limits.loose()); + // TODO: handle cases where self and/or children are not Length::Fill + // If fill use max_size + //if match self.width(), self.height() + // This time we need to adjust up to meet the aspect ratio // so that the container is larger than the contents let mut content_size = content.size(); @@ -261,6 +265,25 @@ where layout::Node::with_children(size, vec![content]) } + fn on_event( + &mut self, + event: Event, + layout: Layout<'_>, + cursor_position: Point, + messages: &mut Vec, + renderer: &R, + clipboard: Option<&dyn Clipboard>, + ) { + self.content.on_event( + event, + layout.children().next().unwrap(), + cursor_position, + messages, + renderer, + clipboard, + ); + } + fn draw( &self, renderer: &mut R, diff --git a/voxygen/src/ui/ice/widget/compound_graphic.rs b/voxygen/src/ui/ice/widget/compound_graphic.rs index bae79a6bef..7382edabde 100644 --- a/voxygen/src/ui/ice/widget/compound_graphic.rs +++ b/voxygen/src/ui/ice/widget/compound_graphic.rs @@ -230,7 +230,9 @@ where fn aspect_ratio_fixed(&self) -> bool { self.fix_aspect_ratio } - fn pixel_dims(&self, _renderer: &R) -> [u16; 2] { self.graphics_size } + fn pixel_dims(&self, _renderer: &R) -> (u16, u16) { + (self.graphics_size[0], self.graphics_size[1]) + } fn draw( &self, diff --git a/voxygen/src/ui/ice/widget/image.rs b/voxygen/src/ui/ice/widget/image.rs index 22eb73e15d..4a74bf0c81 100644 --- a/voxygen/src/ui/ice/widget/image.rs +++ b/voxygen/src/ui/ice/widget/image.rs @@ -123,9 +123,9 @@ where fn aspect_ratio_fixed(&self) -> bool { self.fix_aspect_ratio } - fn pixel_dims(&self, renderer: &R) -> [u16; 2] { + fn pixel_dims(&self, renderer: &R) -> (u16, u16) { let (w, h) = renderer.dimensions(self.handle); - [w as u16, h as u16] + (w as u16, h as u16) } fn draw( From d8e1af835dfc95053d62e6536dd96c602b96a8fc Mon Sep 17 00:00:00 2001 From: Imbris Date: Tue, 26 May 2020 20:59:16 -0400 Subject: [PATCH 13/61] Rearrange iced renderer modules, wire up events for the main menu iced ui --- voxygen/src/menu/main/ui.rs | 20 +++-- voxygen/src/ui/ice/renderer/defaults.rs | 12 +++ voxygen/src/ui/ice/renderer/mod.rs | 31 ++------ .../src/ui/ice/renderer/{ => style}/button.rs | 75 ++----------------- voxygen/src/ui/ice/renderer/style/mod.rs | 3 + .../{ => widgets}/aspect_ratio_container.rs | 2 +- .../{ => widgets}/background_container.rs | 2 +- voxygen/src/ui/ice/renderer/widgets/button.rs | 65 ++++++++++++++++ .../ui/ice/renderer/{ => widgets}/column.rs | 2 +- .../{ => widgets}/compound_graphic.rs | 2 +- .../ice/renderer/{ => widgets}/container.rs | 2 +- .../ui/ice/renderer/{ => widgets}/image.rs | 2 +- voxygen/src/ui/ice/renderer/widgets/mod.rs | 10 +++ .../src/ui/ice/renderer/{ => widgets}/row.rs | 2 +- .../ui/ice/renderer/{ => widgets}/space.rs | 2 +- .../ui/ice/renderer/{ => widgets}/stack.rs | 2 +- .../src/ui/ice/renderer/{ => widgets}/text.rs | 2 +- 17 files changed, 125 insertions(+), 111 deletions(-) create mode 100644 voxygen/src/ui/ice/renderer/defaults.rs rename voxygen/src/ui/ice/renderer/{ => style}/button.rs (52%) create mode 100644 voxygen/src/ui/ice/renderer/style/mod.rs rename voxygen/src/ui/ice/renderer/{ => widgets}/aspect_ratio_container.rs (97%) rename voxygen/src/ui/ice/renderer/{ => widgets}/background_container.rs (91%) create mode 100644 voxygen/src/ui/ice/renderer/widgets/button.rs rename voxygen/src/ui/ice/renderer/{ => widgets}/column.rs (95%) rename voxygen/src/ui/ice/renderer/{ => widgets}/compound_graphic.rs (98%) rename voxygen/src/ui/ice/renderer/{ => widgets}/container.rs (95%) rename voxygen/src/ui/ice/renderer/{ => widgets}/image.rs (96%) create mode 100644 voxygen/src/ui/ice/renderer/widgets/mod.rs rename voxygen/src/ui/ice/renderer/{ => widgets}/row.rs (95%) rename voxygen/src/ui/ice/renderer/{ => widgets}/space.rs (82%) rename voxygen/src/ui/ice/renderer/{ => widgets}/stack.rs (93%) rename voxygen/src/ui/ice/renderer/{ => widgets}/text.rs (98%) diff --git a/voxygen/src/menu/main/ui.rs b/voxygen/src/menu/main/ui.rs index b10e64aad1..69bcaf6b77 100644 --- a/voxygen/src/menu/main/ui.rs +++ b/voxygen/src/menu/main/ui.rs @@ -158,7 +158,6 @@ rotation_image_ids! { } } -#[derive(Clone)] // TODO: why does iced require Clone? pub enum Event { LoginAttempt { username: String, @@ -190,7 +189,12 @@ struct IcedState { imgs: IcedImgs, quit_button: iced::button::State, } -pub type Message = Event; + +#[derive(Clone)] // TODO: why does iced require Clone? +enum Message { + Quit, +} + impl IcedState { pub fn new(imgs: IcedImgs) -> Self { Self { @@ -351,9 +355,9 @@ impl IcedState { BackgroundContainer::new(Image::new(self.imgs.bg), content).into() } - pub fn update(message: Message) { + pub fn update(&mut self, message: Message, events: &mut Vec) { match message { - _ => unimplemented!(), + Message::Quit => events.push(Event::Quit), } } } @@ -1121,12 +1125,16 @@ impl<'a> MainMenuUi { pub fn handle_iced_event(&mut self, event: ui::ice::Event) { self.ice_ui.handle_event(event); } pub fn maintain(&mut self, global_state: &mut GlobalState, dt: Duration) -> Vec { - let events = self.update_layout(global_state, dt); + let mut events = self.update_layout(global_state, dt); self.ui.maintain(global_state.window.renderer_mut(), None); - self.ice_ui.maintain( + let (messages, _) = self.ice_ui.maintain( self.ice_state.view(&self.i18n), global_state.window.renderer_mut(), ); + messages + .into_iter() + .for_each(|message| self.ice_state.update(message, &mut events)); + events } diff --git a/voxygen/src/ui/ice/renderer/defaults.rs b/voxygen/src/ui/ice/renderer/defaults.rs new file mode 100644 index 0000000000..2fdf5215f6 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/defaults.rs @@ -0,0 +1,12 @@ +// TODO: expose to user +pub struct Defaults { + pub text_color: iced::Color, +} + +impl Default for Defaults { + fn default() -> Self { + Self { + text_color: iced::Color::WHITE, + } + } +} diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index 3ed49040c3..f94e576660 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -1,16 +1,9 @@ -// TODO: reorganize modules (e.g. put all these in a widget submodule) -mod aspect_ratio_container; -mod background_container; -mod button; -mod column; -mod compound_graphic; -mod container; -mod image; -mod row; -mod space; -mod text; +mod defaults; +mod style; +mod widgets; -pub use button::Style as ButtonStyle; +pub use defaults::Defaults; +pub use style::ButtonStyle; use super::{ super::graphic::{self, Graphic, TexId}, @@ -24,7 +17,6 @@ use crate::{ Error, }; use common::util::srgba_to_linear; -//use log::warn; use std::ops::Range; use vek::*; @@ -596,19 +588,6 @@ fn default_scissor(renderer: &Renderer) -> Aabr { } } -// TODO: expose to user -pub struct Defaults { - pub text_color: iced::Color, -} - -impl Default for Defaults { - fn default() -> Self { - Self { - text_color: iced::Color::WHITE, - } - } -} - impl iced::Renderer for IcedRenderer { // Default styling type Defaults = Defaults; diff --git a/voxygen/src/ui/ice/renderer/button.rs b/voxygen/src/ui/ice/renderer/style/button.rs similarity index 52% rename from voxygen/src/ui/ice/renderer/button.rs rename to voxygen/src/ui/ice/renderer/style/button.rs index 30754d512f..c4f3681771 100644 --- a/voxygen/src/ui/ice/renderer/button.rs +++ b/voxygen/src/ui/ice/renderer/style/button.rs @@ -1,6 +1,5 @@ -use super::{super::Rotation, widget::image, Defaults, IcedRenderer, Primitive}; -use iced::{button, mouse, Color, Element, Layout, Point, Rectangle}; -use vek::Rgba; +use super::super::widget::image; +use iced::Color; #[derive(Clone, Copy)] struct Background { @@ -69,22 +68,22 @@ impl Style { self } - fn disabled(&self) -> (Option, Color) { + pub fn disabled(&self) -> (Option, Color) { ( self.background.as_ref().map(|b| b.default), self.disabled_text, ) } - fn pressed(&self) -> (Option, Color) { + pub fn pressed(&self) -> (Option, Color) { (self.background.as_ref().map(|b| b.press), self.enabled_text) } - fn hovered(&self) -> (Option, Color) { + pub fn hovered(&self) -> (Option, Color) { (self.background.as_ref().map(|b| b.hover), self.enabled_text) } - fn active(&self) -> (Option, Color) { + pub fn active(&self) -> (Option, Color) { ( self.background.as_ref().map(|b| b.default), self.enabled_text, @@ -101,65 +100,3 @@ impl Default for Style { } } } - -impl button::Renderer for IcedRenderer { - // TODO: what if this gets large enough to not be copied around? - type Style = Style; - - const DEFAULT_PADDING: u16 = 0; - - fn draw( - &mut self, - defaults: &Self::Defaults, - bounds: Rectangle, - cursor_position: Point, - is_disabled: bool, - is_pressed: bool, - style: &Self::Style, - content: &Element<'_, M, Self>, - content_layout: Layout<'_>, - ) -> Self::Output { - let is_mouse_over = bounds.contains(cursor_position); - - let (maybe_image, text_color) = if is_disabled { - style.disabled() - } else if is_mouse_over { - if is_pressed { - style.pressed() - } else { - style.hovered() - } - } else { - style.active() - }; - - let (content, _) = content.draw( - self, - &Defaults { text_color }, - content_layout, - cursor_position, - ); - - let primitive = if let Some(handle) = maybe_image { - let background = Primitive::Image { - handle: (handle, Rotation::None), - bounds, - color: Rgba::broadcast(255), - }; - - Primitive::Group { - primitives: vec![background, content], - } - } else { - content - }; - - let mouse_interaction = if is_mouse_over { - mouse::Interaction::Pointer - } else { - mouse::Interaction::default() - }; - - (primitive, mouse_interaction) - } -} diff --git a/voxygen/src/ui/ice/renderer/style/mod.rs b/voxygen/src/ui/ice/renderer/style/mod.rs new file mode 100644 index 0000000000..1e5fe43f73 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/style/mod.rs @@ -0,0 +1,3 @@ +mod button; + +pub use button::Style as ButtonStyle; diff --git a/voxygen/src/ui/ice/renderer/aspect_ratio_container.rs b/voxygen/src/ui/ice/renderer/widgets/aspect_ratio_container.rs similarity index 97% rename from voxygen/src/ui/ice/renderer/aspect_ratio_container.rs rename to voxygen/src/ui/ice/renderer/widgets/aspect_ratio_container.rs index 71c0f13f3b..6732f73adc 100644 --- a/voxygen/src/ui/ice/renderer/aspect_ratio_container.rs +++ b/voxygen/src/ui/ice/renderer/widgets/aspect_ratio_container.rs @@ -1,4 +1,4 @@ -use super::{ +use super::super::{ super::widget::{aspect_ratio_container, image}, IcedRenderer, }; diff --git a/voxygen/src/ui/ice/renderer/background_container.rs b/voxygen/src/ui/ice/renderer/widgets/background_container.rs similarity index 91% rename from voxygen/src/ui/ice/renderer/background_container.rs rename to voxygen/src/ui/ice/renderer/widgets/background_container.rs index 7b90f5b3c0..8ddfa0e26b 100644 --- a/voxygen/src/ui/ice/renderer/background_container.rs +++ b/voxygen/src/ui/ice/renderer/widgets/background_container.rs @@ -1,4 +1,4 @@ -use super::{super::widget::background_container, IcedRenderer, Primitive}; +use super::super::{super::widget::background_container, IcedRenderer, Primitive}; use iced::{Element, Layout, Point}; impl background_container::Renderer for IcedRenderer { diff --git a/voxygen/src/ui/ice/renderer/widgets/button.rs b/voxygen/src/ui/ice/renderer/widgets/button.rs new file mode 100644 index 0000000000..ecba1fb37b --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widgets/button.rs @@ -0,0 +1,65 @@ +use super::super::{super::Rotation, Defaults, IcedRenderer, Primitive}; +use iced::{button, mouse, Element, Layout, Point, Rectangle}; +use vek::Rgba; + +impl button::Renderer for IcedRenderer { + // TODO: what if this gets large enough to not be copied around? + type Style = super::super::style::ButtonStyle; + + const DEFAULT_PADDING: u16 = 0; + + fn draw( + &mut self, + _defaults: &Self::Defaults, + bounds: Rectangle, + cursor_position: Point, + is_disabled: bool, + is_pressed: bool, + style: &Self::Style, + content: &Element<'_, M, Self>, + content_layout: Layout<'_>, + ) -> Self::Output { + let is_mouse_over = bounds.contains(cursor_position); + + let (maybe_image, text_color) = if is_disabled { + style.disabled() + } else if is_mouse_over { + if is_pressed { + style.pressed() + } else { + style.hovered() + } + } else { + style.active() + }; + + let (content, _) = content.draw( + self, + &Defaults { text_color }, + content_layout, + cursor_position, + ); + + let primitive = if let Some(handle) = maybe_image { + let background = Primitive::Image { + handle: (handle, Rotation::None), + bounds, + color: Rgba::broadcast(255), + }; + + Primitive::Group { + primitives: vec![background, content], + } + } else { + content + }; + + let mouse_interaction = if is_mouse_over { + mouse::Interaction::Pointer + } else { + mouse::Interaction::default() + }; + + (primitive, mouse_interaction) + } +} diff --git a/voxygen/src/ui/ice/renderer/column.rs b/voxygen/src/ui/ice/renderer/widgets/column.rs similarity index 95% rename from voxygen/src/ui/ice/renderer/column.rs rename to voxygen/src/ui/ice/renderer/widgets/column.rs index 5a6a33bcc3..7dbbbde20f 100644 --- a/voxygen/src/ui/ice/renderer/column.rs +++ b/voxygen/src/ui/ice/renderer/widgets/column.rs @@ -1,4 +1,4 @@ -use super::{IcedRenderer, Primitive}; +use super::super::{IcedRenderer, Primitive}; use iced::{column, mouse, Element, Layout, Point}; impl column::Renderer for IcedRenderer { diff --git a/voxygen/src/ui/ice/renderer/compound_graphic.rs b/voxygen/src/ui/ice/renderer/widgets/compound_graphic.rs similarity index 98% rename from voxygen/src/ui/ice/renderer/compound_graphic.rs rename to voxygen/src/ui/ice/renderer/widgets/compound_graphic.rs index 65ef251834..3923f19d1b 100644 --- a/voxygen/src/ui/ice/renderer/compound_graphic.rs +++ b/voxygen/src/ui/ice/renderer/widgets/compound_graphic.rs @@ -1,4 +1,4 @@ -use super::{ +use super::super::{ super::{widget::compound_graphic, Rotation}, IcedRenderer, Primitive, }; diff --git a/voxygen/src/ui/ice/renderer/container.rs b/voxygen/src/ui/ice/renderer/widgets/container.rs similarity index 95% rename from voxygen/src/ui/ice/renderer/container.rs rename to voxygen/src/ui/ice/renderer/widgets/container.rs index cd8a3c2748..f1a4a24355 100644 --- a/voxygen/src/ui/ice/renderer/container.rs +++ b/voxygen/src/ui/ice/renderer/widgets/container.rs @@ -1,4 +1,4 @@ -use super::IcedRenderer; +use super::super::IcedRenderer; use iced::{container, Element, Layout, Point, Rectangle}; impl container::Renderer for IcedRenderer { diff --git a/voxygen/src/ui/ice/renderer/image.rs b/voxygen/src/ui/ice/renderer/widgets/image.rs similarity index 96% rename from voxygen/src/ui/ice/renderer/image.rs rename to voxygen/src/ui/ice/renderer/widgets/image.rs index 829bef748b..8f7c9deaf2 100644 --- a/voxygen/src/ui/ice/renderer/image.rs +++ b/voxygen/src/ui/ice/renderer/widgets/image.rs @@ -1,4 +1,4 @@ -use super::{ +use super::super::{ super::{widget::image, Rotation}, IcedRenderer, Primitive, }; diff --git a/voxygen/src/ui/ice/renderer/widgets/mod.rs b/voxygen/src/ui/ice/renderer/widgets/mod.rs new file mode 100644 index 0000000000..eff22e5c8a --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widgets/mod.rs @@ -0,0 +1,10 @@ +mod aspect_ratio_container; +mod background_container; +mod button; +mod column; +mod compound_graphic; +mod container; +mod image; +mod row; +mod space; +mod text; diff --git a/voxygen/src/ui/ice/renderer/row.rs b/voxygen/src/ui/ice/renderer/widgets/row.rs similarity index 95% rename from voxygen/src/ui/ice/renderer/row.rs rename to voxygen/src/ui/ice/renderer/widgets/row.rs index b46747cdea..3de2f1ea26 100644 --- a/voxygen/src/ui/ice/renderer/row.rs +++ b/voxygen/src/ui/ice/renderer/widgets/row.rs @@ -1,4 +1,4 @@ -use super::{IcedRenderer, Primitive}; +use super::super::{IcedRenderer, Primitive}; use iced::{mouse, row, Element, Layout, Point}; impl row::Renderer for IcedRenderer { diff --git a/voxygen/src/ui/ice/renderer/space.rs b/voxygen/src/ui/ice/renderer/widgets/space.rs similarity index 82% rename from voxygen/src/ui/ice/renderer/space.rs rename to voxygen/src/ui/ice/renderer/widgets/space.rs index 309d216209..61abfcb1b2 100644 --- a/voxygen/src/ui/ice/renderer/space.rs +++ b/voxygen/src/ui/ice/renderer/widgets/space.rs @@ -1,4 +1,4 @@ -use super::{IcedRenderer, Primitive}; +use super::super::{IcedRenderer, Primitive}; use iced::{mouse, space, Rectangle}; impl space::Renderer for IcedRenderer { diff --git a/voxygen/src/ui/ice/renderer/stack.rs b/voxygen/src/ui/ice/renderer/widgets/stack.rs similarity index 93% rename from voxygen/src/ui/ice/renderer/stack.rs rename to voxygen/src/ui/ice/renderer/widgets/stack.rs index f94f4a2f6c..0594a69ae3 100644 --- a/voxygen/src/ui/ice/renderer/stack.rs +++ b/voxygen/src/ui/ice/renderer/widgets/stack.rs @@ -1,4 +1,4 @@ -use super::{super::widget::stack, IcedRenderer, Primitive}; +use super::super::{super::widget::stack, IcedRenderer, Primitive}; use iced::{mouse, Element, Layout, Point}; impl stack::Renderer for IcedRenderer { diff --git a/voxygen/src/ui/ice/renderer/text.rs b/voxygen/src/ui/ice/renderer/widgets/text.rs similarity index 98% rename from voxygen/src/ui/ice/renderer/text.rs rename to voxygen/src/ui/ice/renderer/widgets/text.rs index 5ffb25d44d..b35d471fa2 100644 --- a/voxygen/src/ui/ice/renderer/text.rs +++ b/voxygen/src/ui/ice/renderer/widgets/text.rs @@ -1,4 +1,4 @@ -use super::{super::FontId, IcedRenderer, Primitive}; +use super::super::{super::FontId, IcedRenderer, Primitive}; use glyph_brush::GlyphCruncher; use iced::{mouse, text, Color, HorizontalAlignment, Rectangle, Size, VerticalAlignment}; From 93b5376cb993733c379aeb085a7c0e7b223c200c Mon Sep 17 00:00:00 2001 From: Imbris Date: Wed, 27 May 2020 03:42:55 -0400 Subject: [PATCH 14/61] More buttons, FillText custom widget that adjusts text size based on available space, created button component thing (a reusable composition of widgets), fixed log to tracing rebase error --- Cargo.lock | 8 +- voxygen/Cargo.toml | 2 +- voxygen/src/menu/main/ui.rs | 124 ++++++++++---------- voxygen/src/ui/ice/component/mod.rs | 1 + voxygen/src/ui/ice/component/neat_button.rs | 44 +++++++ voxygen/src/ui/ice/mod.rs | 9 +- voxygen/src/ui/ice/renderer/mod.rs | 4 +- voxygen/src/ui/ice/widget.rs | 5 - voxygen/src/ui/ice/widget/fill_text.rs | 124 ++++++++++++++++++++ voxygen/src/ui/ice/widget/mod.rs | 13 ++ 10 files changed, 255 insertions(+), 79 deletions(-) create mode 100644 voxygen/src/ui/ice/component/mod.rs create mode 100644 voxygen/src/ui/ice/component/neat_button.rs delete mode 100644 voxygen/src/ui/ice/widget.rs create mode 100644 voxygen/src/ui/ice/widget/fill_text.rs create mode 100644 voxygen/src/ui/ice/widget/mod.rs diff --git a/Cargo.lock b/Cargo.lock index d6cde5f45c..929b764240 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2066,12 +2066,12 @@ dependencies = [ [[package]] name = "iced_core" version = "0.2.1" -source = "git+https://github.com/hecrj/iced#f46431600cb61d4e83e0ded1ca79525478436be3" +source = "git+https://github.com/Imberflur/iced?branch=text-clone#0a775191abad5787af3aaa302d5599ef12060264" [[package]] name = "iced_futures" version = "0.1.2" -source = "git+https://github.com/hecrj/iced#f46431600cb61d4e83e0ded1ca79525478436be3" +source = "git+https://github.com/Imberflur/iced?branch=text-clone#0a775191abad5787af3aaa302d5599ef12060264" dependencies = [ "futures 0.3.5", "log", @@ -2081,11 +2081,11 @@ dependencies = [ [[package]] name = "iced_native" version = "0.2.2" -source = "git+https://github.com/hecrj/iced#f46431600cb61d4e83e0ded1ca79525478436be3" +source = "git+https://github.com/Imberflur/iced?branch=text-clone#0a775191abad5787af3aaa302d5599ef12060264" dependencies = [ "iced_core", "iced_futures", - "num-traits 0.2.12", + "raw-window-handle", "twox-hash", "unicode-segmentation", ] diff --git a/voxygen/Cargo.toml b/voxygen/Cargo.toml index c4b3208c1c..435d395271 100644 --- a/voxygen/Cargo.toml +++ b/voxygen/Cargo.toml @@ -35,7 +35,7 @@ winit = {version = "0.22.2", features = ["serde"]} conrod_core = {git = "https://gitlab.com/veloren/conrod.git", branch="copypasta_0.7"} conrod_winit = {git = "https://gitlab.com/veloren/conrod.git", branch="copypasta_0.7"} euc = {git = "https://github.com/zesterer/euc.git"} -iced = {package = "iced_native", git = "https://github.com/hecrj/iced"} +iced = {package = "iced_native", git = "https://github.com/Imberflur/iced", branch = "text-clone"} window_clipboard = "0.1.1" glyph_brush = "0.6.3" diff --git a/voxygen/src/menu/main/ui.rs b/voxygen/src/menu/main/ui.rs index 69bcaf6b77..c3d6b25163 100644 --- a/voxygen/src/menu/main/ui.rs +++ b/voxygen/src/menu/main/ui.rs @@ -184,38 +184,55 @@ pub struct PopupData { popup_type: PopupType, } -// No state currently +use ui::ice::component::neat_button; struct IcedState { imgs: IcedImgs, - quit_button: iced::button::State, + quit_button: neat_button::State, + settings_button: neat_button::State, + servers_button: neat_button::State, + multiplayer_button: neat_button::State, + #[cfg(feature = "singleplayer")] + singleplayer_button: neat_button::State, + show_servers: bool, } #[derive(Clone)] // TODO: why does iced require Clone? enum Message { Quit, + ShowServers, + #[cfg(feature = "singleplayer")] + Singleplayer, + Multiplayer, } impl IcedState { pub fn new(imgs: IcedImgs) -> Self { Self { imgs, - quit_button: iced::button::State::new(), + servers_button: Default::default(), + settings_button: Default::default(), + quit_button: Default::default(), + multiplayer_button: Default::default(), + #[cfg(feature = "singleplayer")] + singleplayer_button: Default::default(), + show_servers: false, } } pub fn view(&mut self, i18n: &Localization) -> Element { - // TODO: scale with window size - let button_font_size = 30; + //let button_font_size = 30; const TEXT_COLOR: iced::Color = iced::Color::from_rgb(1.0, 1.0, 1.0); const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1.0, 0.2); + const FILL_FRAC_ONE: f32 = 0.77; + const FILL_FRAC_TWO: f32 = 0.53; - use iced::{ - Align, Button, Column, Container, HorizontalAlignment, Length, Row, Space, Text, - VerticalAlignment, - }; + use iced::{Align, Column, Container, Length, Row, Space}; use ui::ice::{ - compound_graphic::{CompoundGraphic, Graphic}, - AspectRatioContainer, BackgroundContainer, ButtonStyle, Image, Padding, + widget::{ + compound_graphic::{CompoundGraphic, Graphic}, + BackgroundContainer, Image, Padding, + }, + ButtonStyle, }; use vek::*; @@ -224,50 +241,31 @@ impl IcedState { .press_image(self.imgs.button_press) .text_color(TEXT_COLOR) .disabled_text_color(DISABLED_TEXT_COLOR); - let buttons = Column::with_children(vec![ - Image::new(self.imgs.button).fix_aspect_ratio().into(), - Image::new(self.imgs.button).fix_aspect_ratio().into(), - AspectRatioContainer::new( - Button::new( - &mut self.quit_button, - Text::new(i18n.get("common.quit")) - .size(button_font_size) - .height(Length::Fill) - .width(Length::Fill) - .horizontal_alignment(HorizontalAlignment::Center) - .vertical_alignment(VerticalAlignment::Center), - ) - .height(Length::Fill) - .width(Length::Fill) - .style(button_style) - .on_press(Message::Quit), - ) - .ratio_of_image(self.imgs.button) - .into(), + self.servers_button.view( + i18n.get("common.servers"), + FILL_FRAC_ONE, + button_style, + Some(Message::ShowServers), + ), + self.settings_button.view( + i18n.get("common.settings"), + FILL_FRAC_ONE, + button_style, + None, + ), + self.quit_button.view( + i18n.get("common.quit"), + FILL_FRAC_ONE, + button_style, + Some(Message::Quit), + ), ]) .width(Length::Fill) .max_width(200) .spacing(5) .padding(10); - // Quit - /*if Button::image(self.imgs.button) - .w_h(190.0, 40.0) - .bottom_left_with_margins_on(ui_widgets.window, 60.0, 30.0) - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .label(i18n.get("common.quit")) - .label_font_id(self.fonts.cyri.conrod_id) - .label_color(TEXT_COLOR) - .label_font_size(self.fonts.cyri.scale(20)) - .label_y(Relative::Scalar(3.0)) - .set(self.ids.quit_button, ui_widgets) - .was_clicked() - { - events.push(Event::Quit); - }*/ - let buttons = Container::new(buttons) .width(Length::Fill) .height(Length::Fill) @@ -303,20 +301,22 @@ impl IcedState { .height(Length::FillPortion(50)) .into(), Column::with_children(vec![ - BackgroundContainer::new( - CompoundGraphic::padded_image(self.imgs.button, [106, 26], [10, 0, 10, 2]) - .fix_aspect_ratio(), - Space::new(Length::Fill, Length::Fill), - ) - .into(), - BackgroundContainer::new( - CompoundGraphic::padded_image(self.imgs.button, [106, 26], [10, 2, 10, 0]) - .fix_aspect_ratio(), - Space::new(Length::Fill, Length::Fill), - ) - .into(), + self.multiplayer_button.view( + i18n.get("common.multiplayer"), + FILL_FRAC_TWO, + button_style, + Some(Message::Multiplayer), + ), + #[cfg(feature = "singleplayer")] + self.singleplayer_button.view( + i18n.get("common.singleplayer"), + FILL_FRAC_TWO, + button_style, + Some(Message::Singleplayer), + ), ]) .max_width(240) + .spacing(8) // TODO scale with available window size, awkward because both width and height can become limiting factors, might need custom column, could also just use fill portion .height(Length::FillPortion(25)) .into(), ]) @@ -358,6 +358,10 @@ impl IcedState { pub fn update(&mut self, message: Message, events: &mut Vec) { match message { Message::Quit => events.push(Event::Quit), + Message::ShowServers => self.show_servers = true, + #[cfg(feature = "singleplayer")] + Message::Singleplayer => events.push(Event::StartSingleplayer), + Message::Multiplayer => (), //TODO } } } diff --git a/voxygen/src/ui/ice/component/mod.rs b/voxygen/src/ui/ice/component/mod.rs new file mode 100644 index 0000000000..7f44faef0b --- /dev/null +++ b/voxygen/src/ui/ice/component/mod.rs @@ -0,0 +1 @@ +pub mod neat_button; diff --git a/voxygen/src/ui/ice/component/neat_button.rs b/voxygen/src/ui/ice/component/neat_button.rs new file mode 100644 index 0000000000..f0ce7d576c --- /dev/null +++ b/voxygen/src/ui/ice/component/neat_button.rs @@ -0,0 +1,44 @@ +use crate::ui::ice as ui; +use iced::{Button, Element, Length}; +use ui::{ + widget::{AspectRatioContainer, FillText}, + ButtonStyle, +}; + +#[derive(Default)] +pub struct State { + state: iced::button::State, +} + +impl State { + pub fn new() -> Self { Self::default() } + + pub fn view( + &mut self, + label: impl Into, + fill_fraction: f32, + button_style: ButtonStyle, + message: Option, + ) -> Element { + let button = Button::new( + &mut self.state, + FillText::new(label).fill_fraction(fill_fraction), + ) + .height(Length::Fill) + .width(Length::Fill) + .style(button_style); + + let button = match message { + Some(message) => button.on_press(message), + None => button, + }; + + let container = AspectRatioContainer::new(button); + let container = match button_style.active().0 { + Some(img) => container.ratio_of_image(img), + None => container, + }; + + container.into() + } +} diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index a3e8745b98..b7791e4638 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -1,20 +1,15 @@ // tooltip_manager: TooltipManager, mod cache; mod clipboard; +pub mod component; mod renderer; -mod widget; +pub mod widget; mod winit_conversion; pub use cache::Font; pub use graphic::{Id, Rotation}; pub use iced::Event; pub use renderer::{ButtonStyle, IcedRenderer}; -pub use widget::{ - aspect_ratio_container::AspectRatioContainer, - background_container::{BackgroundContainer, Padding}, - compound_graphic, - image::Image, -}; pub use winit_conversion::window_event; use super::{ diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index f94e576660..193e778842 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -227,7 +227,7 @@ impl IcedRenderer { .collect::>(); if let Err(err) = renderer.update_texture(cache_tex, offset, size, &new_data) { - log::warn!("Failed to update glyph cache texture: {:?}", err); + tracing::warn!("Failed to update glyph cache texture: {:?}", err); } }, // Urgh more allocation we don't need @@ -278,7 +278,7 @@ impl IcedRenderer { }); }, Err(glyph_brush::BrushError::TextureTooSmall { suggested: (x, y) }) => { - log::error!( + tracing::error!( "Texture to small for all glyphs, would need one of the size: ({}, {})", x, y diff --git a/voxygen/src/ui/ice/widget.rs b/voxygen/src/ui/ice/widget.rs deleted file mode 100644 index 85b4968588..0000000000 --- a/voxygen/src/ui/ice/widget.rs +++ /dev/null @@ -1,5 +0,0 @@ -pub mod aspect_ratio_container; -pub mod background_container; -pub mod compound_graphic; -pub mod image; -pub mod stack; diff --git a/voxygen/src/ui/ice/widget/fill_text.rs b/voxygen/src/ui/ice/widget/fill_text.rs new file mode 100644 index 0000000000..7cd5b8d103 --- /dev/null +++ b/voxygen/src/ui/ice/widget/fill_text.rs @@ -0,0 +1,124 @@ +use iced::{layout, Element, Hasher, Layout, Length, Point, Size, Widget}; +use std::hash::Hash; + +const DEFAULT_FILL_FRACTION: f32 = 1.0; +const DEFAULT_VERTICAL_ADJUSTMENT: f32 = 0.05; + +/// Wraps the existing Text widget giving it more advanced layouting +/// capabilities +/// Centers child text widget and adjust the font size depending on the height +/// of the limits Assumes single line text is being used +pub struct FillText +where + R: iced::text::Renderer, +{ + //max_font_size: u16, uncomment if there is a use case for this + /// Portion of the height of the limits which the font size should be + fill_fraction: f32, + /// Adjustment factor to center the text vertically + /// Multiplied by font size and used to move the text up if positive + // TODO: use the produced glyph geometry directly to do this and/or add support to + // layouting library + vertical_adjustment: f32, + text: iced::Text, +} + +impl FillText +where + R: iced::text::Renderer, +{ + pub fn new(label: impl Into) -> Self { + Self { + //max_font_size: u16::MAX, + fill_fraction: DEFAULT_FILL_FRACTION, + vertical_adjustment: DEFAULT_VERTICAL_ADJUSTMENT, + text: iced::Text::new(label), + } + } + + pub fn fill_fraction(mut self, fraction: f32) -> Self { + self.fill_fraction = fraction; + self + } + + pub fn vertical_adjustment(mut self, adjustment: f32) -> Self { + self.vertical_adjustment = adjustment; + self + } + + pub fn color(mut self, color: impl Into) -> Self { + self.text = self.text.color(color); + self + } + + pub fn font(mut self, font: impl Into) -> Self { + self.text = self.text.font(font); + self + } +} + +impl Widget for FillText +where + R: iced::text::Renderer, +{ + fn width(&self) -> Length { Length::Fill } + + fn height(&self) -> Length { Length::Fill } + + fn layout(&self, renderer: &R, limits: &layout::Limits) -> layout::Node { + let limits = limits.width(Length::Fill).height(Length::Fill); + + let size = limits.max(); + + let font_size = (size.height * self.fill_fraction) as u16; + + let mut text = + Widget::::layout(&self.text.clone().size(font_size), renderer, &limits); + + // Size adjusted for centering + text.align( + iced::Align::Center, + iced::Align::Center, + Size::new( + size.width, + size.height - 2.0 * font_size as f32 * self.vertical_adjustment, + ), + ); + + layout::Node::with_children(size, vec![text]) + } + + fn draw( + &self, + renderer: &mut R, + defaults: &R::Defaults, + layout: Layout<'_>, + cursor_position: Point, + ) -> R::Output { + // Note: this breaks if the parent widget adjusts the bounds height + let font_size = (layout.bounds().height * self.fill_fraction) as u16; + Widget::::draw( + &self.text.clone().size(font_size), + renderer, + defaults, + layout.children().next().unwrap(), + cursor_position, + ) + } + + fn hash_layout(&self, state: &mut Hasher) { + struct Marker; + std::any::TypeId::of::().hash(state); + + self.fill_fraction.to_bits().hash(state); + self.vertical_adjustment.to_bits().hash(state); + Widget::::hash_layout(&self.text, state); + } +} + +impl<'a, M, R> From> for Element<'a, M, R> +where + R: 'a + iced::text::Renderer, +{ + fn from(fill_text: FillText) -> Element<'a, M, R> { Element::new(fill_text) } +} diff --git a/voxygen/src/ui/ice/widget/mod.rs b/voxygen/src/ui/ice/widget/mod.rs new file mode 100644 index 0000000000..18034b58ec --- /dev/null +++ b/voxygen/src/ui/ice/widget/mod.rs @@ -0,0 +1,13 @@ +pub mod aspect_ratio_container; +pub mod background_container; +pub mod compound_graphic; +pub mod fill_text; +pub mod image; +pub mod stack; + +pub use self::{ + aspect_ratio_container::AspectRatioContainer, + background_container::{BackgroundContainer, Padding}, + fill_text::FillText, + image::Image, +}; From 4bd5349a807031f14158d4d9cf3d875972b4a36c Mon Sep 17 00:00:00 2001 From: Imbris Date: Fri, 29 May 2020 00:40:22 -0400 Subject: [PATCH 15/61] Add support for TextInput widget, rename widgets module to widget, update to glyph_brush 0.7.0 --- Cargo.lock | 154 ++++++++-- voxygen/Cargo.toml | 4 +- voxygen/src/menu/main/ui.rs | 3 +- voxygen/src/ui/ice/cache.rs | 10 +- voxygen/src/ui/ice/renderer/mod.rs | 152 ++++++---- voxygen/src/ui/ice/renderer/primitive.rs | 31 ++ voxygen/src/ui/ice/renderer/style/button.rs | 2 +- .../aspect_ratio_container.rs | 0 .../background_container.rs | 0 .../renderer/{widgets => widget}/button.rs | 0 .../renderer/{widgets => widget}/column.rs | 0 .../{widgets => widget}/compound_graphic.rs | 6 +- .../renderer/{widgets => widget}/container.rs | 0 .../ice/renderer/{widgets => widget}/image.rs | 0 .../ice/renderer/{widgets => widget}/mod.rs | 1 + .../ice/renderer/{widgets => widget}/row.rs | 0 .../ice/renderer/{widgets => widget}/space.rs | 0 .../ice/renderer/{widgets => widget}/stack.rs | 0 .../ice/renderer/{widgets => widget}/text.rs | 30 +- .../src/ui/ice/renderer/widget/text_input.rs | 273 ++++++++++++++++++ 20 files changed, 557 insertions(+), 109 deletions(-) create mode 100644 voxygen/src/ui/ice/renderer/primitive.rs rename voxygen/src/ui/ice/renderer/{widgets => widget}/aspect_ratio_container.rs (100%) rename voxygen/src/ui/ice/renderer/{widgets => widget}/background_container.rs (100%) rename voxygen/src/ui/ice/renderer/{widgets => widget}/button.rs (100%) rename voxygen/src/ui/ice/renderer/{widgets => widget}/column.rs (100%) rename voxygen/src/ui/ice/renderer/{widgets => widget}/compound_graphic.rs (83%) rename voxygen/src/ui/ice/renderer/{widgets => widget}/container.rs (100%) rename voxygen/src/ui/ice/renderer/{widgets => widget}/image.rs (100%) rename voxygen/src/ui/ice/renderer/{widgets => widget}/mod.rs (90%) rename voxygen/src/ui/ice/renderer/{widgets => widget}/row.rs (100%) rename voxygen/src/ui/ice/renderer/{widgets => widget}/space.rs (100%) rename voxygen/src/ui/ice/renderer/{widgets => widget}/stack.rs (100%) rename voxygen/src/ui/ice/renderer/{widgets => widget}/text.rs (84%) create mode 100644 voxygen/src/ui/ice/renderer/widget/text_input.rs diff --git a/Cargo.lock b/Cargo.lock index 929b764240..1032afa1ce 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1,5 +1,15 @@ # This file is automatically @generated by Cargo. # It is not intended for manual editing. +[[package]] +name = "ab_glyph" +version = "0.2.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "26a685fe66654266f321a8b572660953f4df36a2135706503a4c89981d76e1a2" +dependencies = [ + "ab_glyph_rasterizer", + "owned_ttf_parser 0.8.0", +] + [[package]] name = "ab_glyph_rasterizer" version = "0.1.3" @@ -117,6 +127,15 @@ dependencies = [ "num-traits 0.2.12", ] +[[package]] +name = "approx" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f2a05fd1bd10b2527e20a2cd32d8873d115b8b39fe219ee25f42a8aca6ba278" +dependencies = [ + "num-traits 0.2.12", +] + [[package]] name = "arc-swap" version = "0.4.7" @@ -192,7 +211,7 @@ dependencies = [ "async-task", "broadcaster", "crossbeam-channel 0.4.4", - "crossbeam-deque", + "crossbeam-deque 0.7.3", "crossbeam-utils 0.7.2", "futures-core", "futures-io", @@ -717,6 +736,12 @@ dependencies = [ "syn 1.0.42", ] +[[package]] +name = "const_fn" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c478836e029dcef17fb47c89023448c64f781a046e0300e257ad8225ae59afab" + [[package]] name = "constant_time_eq" version = "0.1.5" @@ -943,7 +968,7 @@ checksum = "2d818a4990769aac0c7ff1360e233ef3a41adcb009ebb2036bf6915eb0f6b23c" dependencies = [ "cfg-if 0.1.10", "crossbeam-channel 0.3.9", - "crossbeam-deque", + "crossbeam-deque 0.7.3", "crossbeam-epoch 0.7.2", "crossbeam-queue 0.1.2", "crossbeam-utils 0.6.6", @@ -968,6 +993,16 @@ dependencies = [ "maybe-uninit", ] +[[package]] +name = "crossbeam-channel" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dca26ee1f8d361640700bde38b2c37d8c22b3ce2d360e1fc1c74ea4b0aa7d775" +dependencies = [ + "cfg-if 1.0.0", + "crossbeam-utils 0.8.0", +] + [[package]] name = "crossbeam-deque" version = "0.7.3" @@ -979,6 +1014,17 @@ dependencies = [ "maybe-uninit", ] +[[package]] +name = "crossbeam-deque" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "94af6efb46fef72616855b036a624cf27ba656ffc9be1b9a3c931cfc7749a9a9" +dependencies = [ + "cfg-if 1.0.0", + "crossbeam-epoch 0.9.0", + "crossbeam-utils 0.8.0", +] + [[package]] name = "crossbeam-epoch" version = "0.7.2" @@ -1008,6 +1054,20 @@ dependencies = [ "scopeguard", ] +[[package]] +name = "crossbeam-epoch" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ec0f606a85340376eef0d6d8fec399e6d4a544d648386c6645eb6d0653b27d9f" +dependencies = [ + "cfg-if 1.0.0", + "const_fn", + "crossbeam-utils 0.8.0", + "lazy_static", + "memoffset", + "scopeguard", +] + [[package]] name = "crossbeam-queue" version = "0.1.2" @@ -1049,6 +1109,18 @@ dependencies = [ "lazy_static", ] +[[package]] +name = "crossbeam-utils" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ec91540d98355f690a86367e566ecad2e9e579f230230eb7c21398372be73ea5" +dependencies = [ + "autocfg 1.0.1", + "cfg-if 1.0.0", + "const_fn", + "lazy_static", +] + [[package]] name = "crossterm" version = "0.17.7" @@ -1858,26 +1930,40 @@ dependencies = [ [[package]] name = "glyph_brush" -version = "0.6.3" +version = "0.7.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5fca6f9d679bff1322c76c9a1ad4b8553b30a94f3f75bea6936e19032c2f2ec3" +checksum = "afd3e2cfd503a5218dd56172a8bf7c8655a4a7cf745737c606a6edfeea1b343f" dependencies = [ + "glyph_brush_draw_cache", "glyph_brush_layout", "log", "ordered-float 1.1.0", "rustc-hash", - "rusttype 0.8.3", "twox-hash", ] [[package]] -name = "glyph_brush_layout" -version = "0.1.9" +name = "glyph_brush_draw_cache" +version = "0.1.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8b70adc570f1dc71b6b32e241cbcc2b42175f5aea71951fbf41e68b04aec24c7" +checksum = "8cef969a091be5565c2c10b31fd2f115cbeed9f783a27c96ae240ff8ceee067c" dependencies = [ - "approx", - "rusttype 0.8.3", + "ab_glyph", + "crossbeam-channel 0.5.0", + "crossbeam-deque 0.8.0", + "linked-hash-map", + "rayon", + "rustc-hash", +] + +[[package]] +name = "glyph_brush_layout" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "10bc06d530bf20c1902f1b02799ab7372ff43f6119770c49b0bc3f21bd148820" +dependencies = [ + "ab_glyph", + "approx 0.4.0", "xi-unicode", ] @@ -2066,12 +2152,12 @@ dependencies = [ [[package]] name = "iced_core" version = "0.2.1" -source = "git+https://github.com/Imberflur/iced?branch=text-clone#0a775191abad5787af3aaa302d5599ef12060264" +source = "git+https://github.com/Imberflur/iced#cf514910c2c0db8633377d2719b244e388774cee" [[package]] name = "iced_futures" version = "0.1.2" -source = "git+https://github.com/Imberflur/iced?branch=text-clone#0a775191abad5787af3aaa302d5599ef12060264" +source = "git+https://github.com/Imberflur/iced#cf514910c2c0db8633377d2719b244e388774cee" dependencies = [ "futures 0.3.5", "log", @@ -2081,11 +2167,10 @@ dependencies = [ [[package]] name = "iced_native" version = "0.2.2" -source = "git+https://github.com/Imberflur/iced?branch=text-clone#0a775191abad5787af3aaa302d5599ef12060264" +source = "git+https://github.com/Imberflur/iced#cf514910c2c0db8633377d2719b244e388774cee" dependencies = [ "iced_core", "iced_futures", - "raw-window-handle", "twox-hash", "unicode-segmentation", ] @@ -3112,7 +3197,16 @@ version = "0.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9f923fb806c46266c02ab4a5b239735c144bdeda724a50ed058e5226f594cde3" dependencies = [ - "ttf-parser", + "ttf-parser 0.6.2", +] + +[[package]] +name = "owned_ttf_parser" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fb477c7fd2a3a6e04e1dc6ca2e4e9b04f2df702021dc5a5d1cf078c587dc59f7" +dependencies = [ + "ttf-parser 0.8.2", ] [[package]] @@ -3627,7 +3721,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "cfd016f0c045ad38b5251be2c9c0ab806917f82da4d36b2a327e5166adad9270" dependencies = [ "autocfg 1.0.1", - "crossbeam-deque", + "crossbeam-deque 0.7.3", "either", "rayon-core", ] @@ -3639,7 +3733,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e8c4fec834fb6e6d2dd5eece3c7b432a52f0ba887cf40e595190c4107edc08bf" dependencies = [ "crossbeam-channel 0.4.4", - "crossbeam-deque", + "crossbeam-deque 0.7.3", "crossbeam-utils 0.7.2", "lazy_static", "num_cpus", @@ -3810,8 +3904,8 @@ version = "0.8.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9f61411055101f7b60ecf1041d87fb74205fb20b0c7a723f07ef39174cf6b4c0" dependencies = [ - "approx", - "crossbeam-deque", + "approx 0.3.2", + "crossbeam-deque 0.7.3", "crossbeam-utils 0.7.2", "linked-hash-map", "num_cpus", @@ -3827,7 +3921,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "dc7c727aded0be18c5b80c1640eae0ac8e396abf6fa8477d96cb37d18ee5ec59" dependencies = [ "ab_glyph_rasterizer", - "owned_ttf_parser", + "owned_ttf_parser 0.6.0", ] [[package]] @@ -4590,7 +4684,7 @@ version = "0.1.18" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "df720b6581784c118f0eb4310796b12b1d242a7eb95f716a8367855325c25f89" dependencies = [ - "crossbeam-deque", + "crossbeam-deque 0.7.3", "crossbeam-queue 0.2.3", "crossbeam-utils 0.7.2", "futures 0.1.29", @@ -4764,6 +4858,12 @@ version = "0.6.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3e5d7cd7ab3e47dda6e56542f4bbf3824c15234958c6e1bd6aaa347e93499fdc" +[[package]] +name = "ttf-parser" +version = "0.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d973cfa0e6124166b50a1105a67c85de40bbc625082f35c0f56f84cb1fb0a827" + [[package]] name = "tui" version = "0.10.0" @@ -4788,7 +4888,7 @@ version = "1.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3bfd5b7557925ce778ff9b9ef90e3ade34c524b5ff10e239c69a42d546d2af56" dependencies = [ - "rand 0.4.6", + "rand 0.7.3", ] [[package]] @@ -4936,7 +5036,7 @@ version = "0.11.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2657d8704e5e0be82b60157c8dbc71a269273ad766984508fdc54030a0690c4d" dependencies = [ - "approx", + "approx 0.3.2", "num-integer", "num-traits 0.2.12", "rustc_version", @@ -4949,7 +5049,7 @@ name = "vek" version = "0.12.0" source = "git+https://gitlab.com/veloren/vek.git?branch=fix_intrinsics#237a78528b505f34f6dde5dc77db3b642388fe4a" dependencies = [ - "approx", + "approx 0.3.2", "num-integer", "num-traits 0.2.12", "rustc_version", @@ -5301,7 +5401,7 @@ version = "0.4.18" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b7866cab0aa01de1edf8b5d7936938a7e397ee50ce24119aef3e1eaa3b6171da" dependencies = [ - "cfg-if", + "cfg-if 0.1.10", "js-sys", "wasm-bindgen", "web-sys", @@ -5695,9 +5795,9 @@ checksum = "d089681aa106a86fade1b0128fb5daf07d5867a509ab036d99988dec80429a57" [[package]] name = "xi-unicode" -version = "0.2.1" +version = "0.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e71b85d8b1b8bfaf4b5c834187554d201a8cd621c2bbfa33efd41a3ecabd48b2" +checksum = "a67300977d3dc3f8034dae89778f502b6ba20b269527b3223ba59c0cf393bb8a" [[package]] name = "xml-rs" diff --git a/voxygen/Cargo.toml b/voxygen/Cargo.toml index 435d395271..ceab09c555 100644 --- a/voxygen/Cargo.toml +++ b/voxygen/Cargo.toml @@ -35,9 +35,9 @@ winit = {version = "0.22.2", features = ["serde"]} conrod_core = {git = "https://gitlab.com/veloren/conrod.git", branch="copypasta_0.7"} conrod_winit = {git = "https://gitlab.com/veloren/conrod.git", branch="copypasta_0.7"} euc = {git = "https://github.com/zesterer/euc.git"} -iced = {package = "iced_native", git = "https://github.com/Imberflur/iced", branch = "text-clone"} +iced = {package = "iced_native", git = "https://github.com/Imberflur/iced"} window_clipboard = "0.1.1" -glyph_brush = "0.6.3" +glyph_brush = "0.7.0" # ECS specs = {git = "https://github.com/amethyst/specs.git", rev = "7a2e348ab2223818bad487695c66c43db88050a5"} diff --git a/voxygen/src/menu/main/ui.rs b/voxygen/src/menu/main/ui.rs index c3d6b25163..b6032cf29a 100644 --- a/voxygen/src/menu/main/ui.rs +++ b/voxygen/src/menu/main/ui.rs @@ -241,6 +241,7 @@ impl IcedState { .press_image(self.imgs.button_press) .text_color(TEXT_COLOR) .disabled_text_color(DISABLED_TEXT_COLOR); + let buttons = Column::with_children(vec![ self.servers_button.view( i18n.get("common.servers"), @@ -445,7 +446,7 @@ impl<'a> MainMenuUi { .unwrap() .read_to_end(&mut buf) .unwrap(); - glyph_brush::rusttype::Font::from_bytes(buf).unwrap() + ui::ice::Font::try_from_vec(buf).unwrap() }; let mut ice_ui = IcedUi::new(window, ice_font).unwrap(); diff --git a/voxygen/src/ui/ice/cache.rs b/voxygen/src/ui/ice/cache.rs index e061e6c2b5..8e20b3c4ab 100644 --- a/voxygen/src/ui/ice/cache.rs +++ b/voxygen/src/ui/ice/cache.rs @@ -10,12 +10,12 @@ use vek::*; // Multiplied by current window size const GLYPH_CACHE_SIZE: u16 = 1; // Glyph cache tolerances -const SCALE_TOLERANCE: f32 = 0.1; +const SCALE_TOLERANCE: f32 = 0.5; // Note: Changed from 0.1 change back if not decent const POSITION_TOLERANCE: f32 = 0.1; -type GlyphBrush = glyph_brush::GlyphBrush<'static, (Aabr, Aabr)>; +type GlyphBrush = glyph_brush::GlyphBrush<(Aabr, Aabr), ()>; -pub type Font = glyph_brush::rusttype::Font<'static>; +pub type Font = glyph_brush::ab_glyph::FontArc; pub struct Cache { glyph_brush: RefCell, @@ -35,8 +35,8 @@ impl Cache { let glyph_brush = GlyphBrushBuilder::using_font(default_font) .initial_cache_size((glyph_cache_dims.x as u32, glyph_cache_dims.y as u32)) - .gpu_cache_scale_tolerance(SCALE_TOLERANCE) - .gpu_cache_position_tolerance(POSITION_TOLERANCE) + .draw_cache_scale_tolerance(SCALE_TOLERANCE) + .draw_cache_position_tolerance(POSITION_TOLERANCE) .build(); Ok(Self { diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index 193e778842..2bc64764c9 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -1,14 +1,18 @@ mod defaults; +mod primitive; mod style; -mod widgets; +mod widget; pub use defaults::Defaults; pub use style::ButtonStyle; +pub(self) use primitive::Primitive; + use super::{ super::graphic::{self, Graphic, TexId}, cache::Cache, - widget, Font, Rotation, + widget::image, + Font, Rotation, }; use crate::{ render::{ @@ -52,36 +56,6 @@ enum State { Plain, } -pub enum Primitive { - // Allocation :( - Group { - primitives: Vec, - }, - Image { - handle: (widget::image::Handle, Rotation), - bounds: iced::Rectangle, - color: Rgba, - }, - Rectangle { - bounds: iced::Rectangle, - color: Rgba, - }, - Text { - glyphs: Vec<( - glyph_brush::rusttype::PositionedGlyph<'static>, - glyph_brush::Color, - glyph_brush::FontId, - )>, - //size: f32, - bounds: iced::Rectangle, - linear_color: Rgba, - /*font: iced::Font, - *horizontal_alignment: iced::HorizontalAlignment, - *vertical_alignment: iced::VerticalAlignment, */ - }, - Nothing, -} - // Optimization idea inspired by what I think iced wgpu renderer may be doing: // Could have layers of things which don't intersect and thus can be reordered // arbitrarily @@ -107,6 +81,8 @@ pub struct IcedRenderer { p_scale: f32, // Pretend dims :) (i.e. scaled) win_dims: Vec2, + // Scissor for the whole window + window_scissor: Aabr, // Per-frame/update current_state: State, @@ -145,6 +121,7 @@ impl IcedRenderer { align, p_scale, win_dims: scaled_dims, + window_scissor: default_scissor(renderer), start: 0, //current_scissor: default_scissor(renderer), }) @@ -154,7 +131,7 @@ impl IcedRenderer { self.cache.add_graphic(graphic) } - fn image_dims(&self, handle: super::widget::image::Handle) -> (u32, u32) { + fn image_dims(&self, handle: image::Handle) -> (u32, u32) { self .cache .graphic_cache() @@ -218,7 +195,7 @@ impl IcedRenderer { let brush_result = glyph_cache.process_queued( |rect, tex_data| { - let offset = [rect.min.x as u16, rect.min.y as u16]; + let offset = [rect.min[0] as u16, rect.min[1] as u16]; let size = [rect.width() as u16, rect.height() as u16]; let new_data = tex_data @@ -377,6 +354,12 @@ impl IcedRenderer { bounds, color, } => { + let color = srgba_to_linear(color.map(|e| e as f32 / 255.0)); + // Don't draw a transparent image. + if color[3] == 0.0 { + return; + } + let (graphic_id, rotation) = handle; let gl_aabr = self.gl_aabr(bounds); @@ -387,12 +370,6 @@ impl IcedRenderer { _ => {}, } - let color = srgba_to_linear(color.map(|e| e as f32 / 255.0)); - // Don't draw a transparent image. - if color[3] == 0.0 { - return; - } - let resolution = Vec2::new( (gl_aabr.size().w * self.half_res.x).round() as u16, (gl_aabr.size().h * self.half_res.y).round() as u16, @@ -445,10 +422,12 @@ impl IcedRenderer { self.mesh .push_quad(create_ui_quad(gl_aabr, uv_aabr, color, UiMode::Image)); }, - Primitive::Rectangle { bounds, color } => { - let color = srgba_to_linear(color.map(|e| e as f32 / 255.0)); + Primitive::Rectangle { + bounds, + linear_color, + } => { // Don't draw a transparent rectangle. - if color[3] == 0.0 { + if linear_color[3] == 0.0 { return; } @@ -460,7 +439,7 @@ impl IcedRenderer { min: Vec2::zero(), max: Vec2::zero(), }, - color, + linear_color, UiMode::Geometry, )); }, @@ -474,8 +453,6 @@ impl IcedRenderer { } => { self.switch_state(State::Plain); - // TODO: Scissor? - // TODO: makes sure we are not doing all this work for hidden text // e.g. in chat let glyph_cache = self.cache.glyph_cache_mut(); @@ -486,20 +463,21 @@ impl IcedRenderer { // Queue the glyphs to be cached. glyph_cache.queue_pre_positioned( glyphs, + // TODO: glyph_brush should document that these need to be the same length + vec![(); glyph_count], // Since we already passed in `bounds` to position the glyphs some of this // seems redundant... - glyph_brush::rusttype::Rect { - min: glyph_brush::rusttype::Point { - x: bounds.x * self.p_scale, - //y: (self.win_dims.y - bounds.y) * self.p_scale, - y: bounds.y * self.p_scale, - }, - max: glyph_brush::rusttype::Point { - x: (bounds.x + bounds.width) * self.p_scale, - y: (bounds.y + bounds.height) * self.p_scale, - }, + glyph_brush::ab_glyph::Rect { + min: glyph_brush::ab_glyph::point( + bounds.x * self.p_scale, + //(self.win_dims.y - bounds.y) * self.p_scale, + bounds.y * self.p_scale, + ), + max: glyph_brush::ab_glyph::point( + (bounds.x + bounds.width) * self.p_scale, + (bounds.y + bounds.height) * self.p_scale, + ), }, - 0.0, // z (we don't use this) ); // Leave ui and verts blank to fill in when processing cached glyphs @@ -522,6 +500,66 @@ impl IcedRenderer { )); } }, + Primitive::Clip { bounds, content } => { + // Check for a change in the scissor. + let new_scissor = { + // Calculate minimum x and y coordinates while + // flipping y axis (from +down to +uo) and + // moving origin from top-left corner to bottom-left + let min_x = bounds.x; + let min_y = self.win_dims.y - bounds.y; + let intersection = Aabr { + min: Vec2 { + x: (min_x * self.p_scale) as u16, + y: (min_y * self.p_scale) as u16, + }, + max: Vec2 { + x: ((min_x + bounds.width) * self.p_scale) as u16, + y: ((min_y + bounds.height) * self.p_scale) as u16, + }, + } + .intersection(self.window_scissor); + + if intersection.is_valid() { + intersection + } else { + Aabr::new_empty(Vec2::zero()) + } + }; + // Not expecting this case: new_cursor == current_scissor + + // Finish the current command. + // TODO: ensure we never push empty commands (make fields private & debug assert + // in constructors?) + self.draw_commands.push(match self.current_state { + State::Plain => DrawCommand::plain(self.start..self.mesh.vertices().len()), + State::Image(id) => { + DrawCommand::image(self.start..self.mesh.vertices().len(), id) + }, + }); + self.start = self.mesh.vertices().len(); + + self.draw_commands.push(DrawCommand::Scissor(new_scissor)); + + // TODO: support nested clips? + // TODO: if last command is a clip changing back to the default replace it with + // this + + // Renderer child + self.draw_primitive(*content, renderer); + + // Reset scissor + self.draw_commands.push(match self.current_state { + State::Plain => DrawCommand::plain(self.start..self.mesh.vertices().len()), + State::Image(id) => { + DrawCommand::image(self.start..self.mesh.vertices().len(), id) + }, + }); + self.start = self.mesh.vertices().len(); + + self.draw_commands + .push(DrawCommand::Scissor(self.window_scissor)); + }, Primitive::Nothing => {}, } } diff --git a/voxygen/src/ui/ice/renderer/primitive.rs b/voxygen/src/ui/ice/renderer/primitive.rs new file mode 100644 index 0000000000..46734998e1 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/primitive.rs @@ -0,0 +1,31 @@ +use crate::ui::{graphic, ice::widget::image}; + +pub enum Primitive { + // Allocation :( + Group { + primitives: Vec, + }, + Image { + handle: (image::Handle, graphic::Rotation), + bounds: iced::Rectangle, + color: vek::Rgba, + }, + Rectangle { + bounds: iced::Rectangle, + linear_color: vek::Rgba, + }, + Text { + glyphs: Vec, + //size: f32, + bounds: iced::Rectangle, + linear_color: vek::Rgba, + /*font: iced::Font, + *horizontal_alignment: iced::HorizontalAlignment, + *vertical_alignment: iced::VerticalAlignment, */ + }, + Clip { + bounds: iced::Rectangle, + content: Box, + }, + Nothing, +} diff --git a/voxygen/src/ui/ice/renderer/style/button.rs b/voxygen/src/ui/ice/renderer/style/button.rs index c4f3681771..20f02a5264 100644 --- a/voxygen/src/ui/ice/renderer/style/button.rs +++ b/voxygen/src/ui/ice/renderer/style/button.rs @@ -1,4 +1,4 @@ -use super::super::widget::image; +use super::super::super::widget::image; use iced::Color; #[derive(Clone, Copy)] diff --git a/voxygen/src/ui/ice/renderer/widgets/aspect_ratio_container.rs b/voxygen/src/ui/ice/renderer/widget/aspect_ratio_container.rs similarity index 100% rename from voxygen/src/ui/ice/renderer/widgets/aspect_ratio_container.rs rename to voxygen/src/ui/ice/renderer/widget/aspect_ratio_container.rs diff --git a/voxygen/src/ui/ice/renderer/widgets/background_container.rs b/voxygen/src/ui/ice/renderer/widget/background_container.rs similarity index 100% rename from voxygen/src/ui/ice/renderer/widgets/background_container.rs rename to voxygen/src/ui/ice/renderer/widget/background_container.rs diff --git a/voxygen/src/ui/ice/renderer/widgets/button.rs b/voxygen/src/ui/ice/renderer/widget/button.rs similarity index 100% rename from voxygen/src/ui/ice/renderer/widgets/button.rs rename to voxygen/src/ui/ice/renderer/widget/button.rs diff --git a/voxygen/src/ui/ice/renderer/widgets/column.rs b/voxygen/src/ui/ice/renderer/widget/column.rs similarity index 100% rename from voxygen/src/ui/ice/renderer/widgets/column.rs rename to voxygen/src/ui/ice/renderer/widget/column.rs diff --git a/voxygen/src/ui/ice/renderer/widgets/compound_graphic.rs b/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs similarity index 83% rename from voxygen/src/ui/ice/renderer/widgets/compound_graphic.rs rename to voxygen/src/ui/ice/renderer/widget/compound_graphic.rs index 3923f19d1b..a3a47af190 100644 --- a/voxygen/src/ui/ice/renderer/widgets/compound_graphic.rs +++ b/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs @@ -2,6 +2,7 @@ use super::super::{ super::{widget::compound_graphic, Rotation}, IcedRenderer, Primitive, }; +use common::util::srgba_to_linear; use compound_graphic::GraphicKind; use iced::{mouse, Rectangle}; @@ -24,7 +25,10 @@ impl compound_graphic::Renderer for IcedRenderer { bounds, color, }, - GraphicKind::Color(color) => Primitive::Rectangle { bounds, color }, + GraphicKind::Color(color) => Primitive::Rectangle { + bounds, + linear_color: srgba_to_linear(color.map(|e| e as f32 * 255.0)), + }, }) .collect(), }, diff --git a/voxygen/src/ui/ice/renderer/widgets/container.rs b/voxygen/src/ui/ice/renderer/widget/container.rs similarity index 100% rename from voxygen/src/ui/ice/renderer/widgets/container.rs rename to voxygen/src/ui/ice/renderer/widget/container.rs diff --git a/voxygen/src/ui/ice/renderer/widgets/image.rs b/voxygen/src/ui/ice/renderer/widget/image.rs similarity index 100% rename from voxygen/src/ui/ice/renderer/widgets/image.rs rename to voxygen/src/ui/ice/renderer/widget/image.rs diff --git a/voxygen/src/ui/ice/renderer/widgets/mod.rs b/voxygen/src/ui/ice/renderer/widget/mod.rs similarity index 90% rename from voxygen/src/ui/ice/renderer/widgets/mod.rs rename to voxygen/src/ui/ice/renderer/widget/mod.rs index eff22e5c8a..f60c6eb0c5 100644 --- a/voxygen/src/ui/ice/renderer/widgets/mod.rs +++ b/voxygen/src/ui/ice/renderer/widget/mod.rs @@ -8,3 +8,4 @@ mod image; mod row; mod space; mod text; +mod text_input; diff --git a/voxygen/src/ui/ice/renderer/widgets/row.rs b/voxygen/src/ui/ice/renderer/widget/row.rs similarity index 100% rename from voxygen/src/ui/ice/renderer/widgets/row.rs rename to voxygen/src/ui/ice/renderer/widget/row.rs diff --git a/voxygen/src/ui/ice/renderer/widgets/space.rs b/voxygen/src/ui/ice/renderer/widget/space.rs similarity index 100% rename from voxygen/src/ui/ice/renderer/widgets/space.rs rename to voxygen/src/ui/ice/renderer/widget/space.rs diff --git a/voxygen/src/ui/ice/renderer/widgets/stack.rs b/voxygen/src/ui/ice/renderer/widget/stack.rs similarity index 100% rename from voxygen/src/ui/ice/renderer/widgets/stack.rs rename to voxygen/src/ui/ice/renderer/widget/stack.rs diff --git a/voxygen/src/ui/ice/renderer/widgets/text.rs b/voxygen/src/ui/ice/renderer/widget/text.rs similarity index 84% rename from voxygen/src/ui/ice/renderer/widgets/text.rs rename to voxygen/src/ui/ice/renderer/widget/text.rs index b35d471fa2..d9f6b1629e 100644 --- a/voxygen/src/ui/ice/renderer/widgets/text.rs +++ b/voxygen/src/ui/ice/renderer/widget/text.rs @@ -13,11 +13,15 @@ impl text::Renderer for IcedRenderer { let p_scale = self.p_scale; // TODO: would be nice if the method was mut let section = glyph_brush::Section { - text: content, - scale: glyph_brush::rusttype::Scale::uniform(size as f32 * p_scale), - font_id: font.0, + screen_position: (0.0, 0.0), bounds: (bounds.width * p_scale, bounds.height * p_scale), - ..Default::default() + layout: Default::default(), + text: vec![glyph_brush::Text { + text: content, + scale: (size as f32 * p_scale).into(), + font_id: font.0, + extra: (), + }], }; let maybe_rect = self.cache.glyph_calculator().glyph_bounds(section); @@ -57,30 +61,26 @@ impl text::Renderer for IcedRenderer { let p_scale = self.p_scale; let section = glyph_brush::Section { - text: content, screen_position: (x * p_scale, y * p_scale), bounds: (bounds.width * p_scale, bounds.height * p_scale), - scale: glyph_brush::rusttype::Scale::uniform(size as f32 * p_scale), layout: glyph_brush::Layout::Wrap { line_breaker: Default::default(), h_align, v_align, }, - font_id: font.0, - ..Default::default() + text: vec![glyph_brush::Text { + text: content, + scale: (size as f32 * p_scale).into(), + font_id: font.0, + extra: (), + }], }; let glyphs = self .cache .glyph_cache_mut() .glyphs(section) - .map(|positioned_glyph| { - ( - positioned_glyph.clone(), // :/ - [0.0, 0.0, 0.0, 1.0], // Color - font.0, - ) - }) + .cloned() .collect::>(); ( diff --git a/voxygen/src/ui/ice/renderer/widget/text_input.rs b/voxygen/src/ui/ice/renderer/widget/text_input.rs new file mode 100644 index 0000000000..9c6ad3f93f --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/text_input.rs @@ -0,0 +1,273 @@ +use super::super::{super::FontId, IcedRenderer, Primitive}; +use glyph_brush::GlyphCruncher; +use iced::{ + mouse, + text_input::{self, cursor}, + Color, Point, Rectangle, +}; + +const CURSOR_WIDTH: f32 = 1.0; +// Extra scroll offset past the cursor +const EXTRA_OFFSET: f32 = 5.0; + +impl text_input::Renderer for IcedRenderer { + type Font = FontId; + type Style = (); + + fn default_size(&self) -> u16 { + // TODO: make configurable + 20 + } + + fn measure_value(&self, value: &str, size: u16, font: Self::Font) -> f32 { + // Using the physical scale might make this cached info usable below? + // Although we also have a position of the screen there so this could be useless + let p_scale = self.p_scale; + + let section = glyph_brush::Section { + screen_position: (0.0, 0.0), + bounds: (f32::INFINITY, f32::INFINITY), + layout: Default::default(), + text: vec![glyph_brush::Text { + text: value, + scale: (size as f32 * p_scale).into(), + font_id: font.0, + extra: (), + }], + }; + + let mut glyph_calculator = self.cache.glyph_calculator(); + let mut width = glyph_calculator + .glyph_bounds(section) + .map_or(0.0, |rect| rect.width() / p_scale); + + // glyph_brush ignores the exterior spaces + // TODO: need better layout lib + let exterior_spaces = value.len() - value.trim().len(); + + if exterior_spaces > 0 { + use glyph_brush::ab_glyph::{Font, ScaleFont}; + // Could cache this if it is slow + let font = glyph_calculator.fonts()[font.0].as_scaled(size as f32); + let space_width = font.h_advance(font.glyph_id(' ')); + width += exterior_spaces as f32 * space_width; + } + + width + } + + fn offset( + &self, + text_bounds: Rectangle, + font: Self::Font, + size: u16, + value: &text_input::Value, + state: &text_input::State, + ) -> f32 { + // Only need to offset if focused with cursor somewhere in the text + if state.is_focused() { + let cursor = state.cursor(); + + let focus_position = match cursor.state(value) { + cursor::State::Index(i) => i, + cursor::State::Selection { end, .. } => end, + }; + + let (_, offset) = measure_cursor_and_scroll_offset( + self, + text_bounds, + value, + size, + focus_position, + font, + ); + + offset + } else { + 0.0 + } + } + + fn draw( + &mut self, + bounds: Rectangle, + text_bounds: Rectangle, + //defaults: &Self::Defaults, No defaults!! + cursor_position: Point, + font: Self::Font, + size: u16, + placeholder: &str, + value: &text_input::Value, + state: &text_input::State, + _style_sheet: &Self::Style, + ) -> Self::Output { + let is_mouse_over = bounds.contains(cursor_position); + + /* + let style = if state.is_focused() { + style.focused() + } else if is_mouse_over { + style.hovered() + } else { + style.active() + }; */ + + let p_scale = self.p_scale; + + // Allocation :( + let text = value.to_string(); + let text = if text.is_empty() { Some(&*text) } else { None }; + + // TODO: background from style, image? + + // TODO: color from style + let color = if text.is_some() { + Color::WHITE + } else { + Color::from_rgba(1.0, 1.0, 1.0, 0.4) + }; + let linear_color = color.into_linear().into(); + + let (cursor_primitive, scroll_offset) = if state.is_focused() { + let cursor = state.cursor(); + + let cursor_and_scroll_offset = |position| { + measure_cursor_and_scroll_offset(self, text_bounds, value, size, position, font) + }; + + let (cursor_primitive, offset) = match cursor.state(value) { + cursor::State::Index(position) => { + let (position, offset) = cursor_and_scroll_offset(position); + ( + Primitive::Rectangle { + bounds: Rectangle { + x: text_bounds.x + position, + y: text_bounds.y, + width: CURSOR_WIDTH / p_scale, + height: text_bounds.height, + }, + linear_color, + }, + offset, + ) + }, + cursor::State::Selection { start, end } => { + let left = start.min(end); + let right = end.max(start); + + let (left_position, left_offset) = cursor_and_scroll_offset(left); + let (right_position, right_offset) = cursor_and_scroll_offset(right); + + let width = right_position - left_position; + + ( + Primitive::Rectangle { + bounds: Rectangle { + x: text_bounds.x + left_position, + y: text_bounds.y, + width, + height: text_bounds.height, + }, + // TODO: selection color from stlye + linear_color: Color::from_rgba(1.0, 0.0, 1.0, 0.2).into_linear().into(), + }, + if end == right { + right_offset + } else { + left_offset + }, + ) + }, + }; + + (Some(cursor_primitive), offset) + } else { + (None, 0.0) + }; + + let section = glyph_brush::Section { + screen_position: ( + text_bounds.x * p_scale + scroll_offset, + text_bounds.center_y() * p_scale, + ), + bounds: (text_bounds.width * p_scale, text_bounds.height * p_scale), + layout: glyph_brush::Layout::SingleLine { + line_breaker: Default::default(), + h_align: glyph_brush::HorizontalAlign::Left, + v_align: glyph_brush::VerticalAlign::Center, + }, + text: vec![glyph_brush::Text { + text: text.unwrap_or(placeholder), + scale: (size as f32 * p_scale).into(), + font_id: font.0, + extra: (), + }], + }; + + let glyphs = self + .cache + .glyph_cache_mut() + .glyphs(section) + .cloned() + .collect::>(); + + let text_primitive = Primitive::Text { + glyphs, + //size: size as f32, + bounds, + linear_color, + /*font, + *horizontal_alignment, + *vertical_alignment, */ + }; + + let primitive = match cursor_primitive { + Some(cursor_primitive) => Primitive::Group { + primitives: vec![cursor_primitive, text_primitive], + }, + None => text_primitive, + }; + + // Probably already computed this somewhere + let text_width = self.measure_value(text.unwrap_or(placeholder), size, font); + + let primitive = if text_width > text_bounds.width { + Primitive::Clip { + bounds: text_bounds, + content: Box::new(primitive), + /* Note: iced_wgpu uses offset here but we can't do that since we pass the text + * to the glyph_brush here */ + } + } else { + primitive + }; + + ( + primitive, + if is_mouse_over { + mouse::Interaction::Text + } else { + mouse::Interaction::default() + }, + ) + } +} + +fn measure_cursor_and_scroll_offset( + renderer: &IcedRenderer, + text_bounds: Rectangle, + value: &text_input::Value, + size: u16, + cursor_index: usize, + font: FontId, +) -> (f32, f32) { + use text_input::Renderer; + + // TODO: so much allocation (fyi .until() allocates) + let text_before_cursor = value.until(cursor_index).to_string(); + + let text_value_width = renderer.measure_value(&text_before_cursor, size, font); + let offset = ((text_value_width + EXTRA_OFFSET) - text_bounds.width).max(0.0); + + (text_value_width, offset) +} From 45dd2b08964ea6015653871cb3edb782eb500528 Mon Sep 17 00:00:00 2001 From: Imbris Date: Fri, 29 May 2020 04:04:03 -0400 Subject: [PATCH 16/61] Add input fields to main menu attempt, fix a few bugs, add bullet point to the font (might have done this wrong). (just a few glitches left when text contains spaces) --- .../haxrcorp_4089_cyrillic_altgr_extended.ttf | Bin 26316 -> 26392 bytes voxygen/src/menu/main/ui.rs | 63 ++++++++++++++++-- voxygen/src/ui/ice/renderer/mod.rs | 27 ++++---- .../ice/renderer/widget/compound_graphic.rs | 2 +- .../src/ui/ice/renderer/widget/text_input.rs | 30 +++++---- 5 files changed, 88 insertions(+), 34 deletions(-) diff --git a/assets/voxygen/font/haxrcorp_4089_cyrillic_altgr_extended.ttf b/assets/voxygen/font/haxrcorp_4089_cyrillic_altgr_extended.ttf index 7d3477d3a1755fdebfcc1bd81d122ca5022c337f..6a63f5fb04762ea032e083bb13e08690244cdf7f 100644 GIT binary patch delta 528 zcmYLGO-LI-6#m}qva2LgkSbAw4Vt!yf?^XR&c-cW-_RTgKsM%KK#$Ai?V1_*Dgy$C z+_k#fJ6tHI66UMSOS09#H26v%KR7*2g0#oWqWgF z2gtGmLTyBuh*JeRS$Fb(TU3eoKi)SIfb0cK7xCEG_+5)(ma|zo)M$D+UhwMWyk;<` z@drkc$AHVW#IueM2ZHBpjhe`!Hd8+pYg;reR_Y4xkS=T50%T&QU&;y*Nk%7Z{IaO6 zl0%)6>eVC3sV1ccttvw%0nipR-vk}Q#!t7lE5Dh@qAmOEf_5@gBo=XgzLSu4G_jkR VOJ_udYBEre7j?{Y=U+_E*&i}HXPW>3 diff --git a/voxygen/src/menu/main/ui.rs b/voxygen/src/menu/main/ui.rs index b6032cf29a..a34318d27a 100644 --- a/voxygen/src/menu/main/ui.rs +++ b/voxygen/src/menu/main/ui.rs @@ -193,6 +193,12 @@ struct IcedState { multiplayer_button: neat_button::State, #[cfg(feature = "singleplayer")] singleplayer_button: neat_button::State, + username_input: iced::text_input::State, + password_input: iced::text_input::State, + server_input: iced::text_input::State, + username_value: String, + password_value: String, + server_value: String, show_servers: bool, } @@ -203,6 +209,10 @@ enum Message { #[cfg(feature = "singleplayer")] Singleplayer, Multiplayer, + Username(String), + Password(String), + Server(String), + FocusPassword, } impl IcedState { @@ -215,6 +225,12 @@ impl IcedState { multiplayer_button: Default::default(), #[cfg(feature = "singleplayer")] singleplayer_button: Default::default(), + username_input: Default::default(), + password_input: Default::default(), + server_input: Default::default(), + username_value: String::new(), + password_value: String::new(), + server_value: String::new(), show_servers: false, } } @@ -226,7 +242,7 @@ impl IcedState { const FILL_FRAC_ONE: f32 = 0.77; const FILL_FRAC_TWO: f32 = 0.53; - use iced::{Align, Column, Container, Length, Row, Space}; + use iced::{Align, Column, Container, Length, Row, Space, TextInput}; use ui::ice::{ widget::{ compound_graphic::{CompoundGraphic, Graphic}, @@ -283,19 +299,45 @@ impl IcedState { BackgroundContainer::new( CompoundGraphic::padded_image(self.imgs.input_bg, [169, 25], [0, 0, 0, 1]) .fix_aspect_ratio(), - Space::new(Length::Fill, Length::Fill), + TextInput::new( + &mut self.username_input, + "Username", + &self.username_value, + Message::Username, + ) + // TODO: wrap in fill text thing to auto adjust the size + .size(22) + .padding(25) + .on_submit(Message::FocusPassword), ) .into(), BackgroundContainer::new( CompoundGraphic::padded_image(self.imgs.input_bg, [169, 25], [0, 1, 0, 1]) .fix_aspect_ratio(), - Space::new(Length::Fill, Length::Fill), + TextInput::new( + &mut self.password_input, + "Password", + &self.password_value, + Message::Password, + ) + .size(22) + .padding(25) + .password() + .on_submit(Message::Multiplayer), ) .into(), BackgroundContainer::new( CompoundGraphic::padded_image(self.imgs.input_bg, [169, 25], [0, 1, 0, 0]) .fix_aspect_ratio(), - Space::new(Length::Fill, Length::Fill), + TextInput::new( + &mut self.server_input, + "Server", + &self.server_value, + Message::Server, + ) + .size(22) + .padding(25) + .on_submit(Message::Multiplayer), ) .into(), ]) @@ -363,6 +405,13 @@ impl IcedState { #[cfg(feature = "singleplayer")] Message::Singleplayer => events.push(Event::StartSingleplayer), Message::Multiplayer => (), //TODO + Message::Username(new_value) => self.username_value = new_value, + Message::Password(new_value) => self.password_value = new_value, + Message::Server(new_value) => self.server_value = new_value, + Message::FocusPassword => { + self.password_input = iced::text_input::State::focused(); + self.username_input = iced::text_input::State::new(); + }, } } } @@ -1125,7 +1174,11 @@ impl<'a> MainMenuUi { self.connect = false; } - pub fn handle_event(&mut self, event: ui::Event) { self.ui.handle_event(event); } + pub fn handle_event(&mut self, event: ui::Event) { + if !self.show_iced { + self.ui.handle_event(event); + } + } pub fn handle_iced_event(&mut self, event: ui::ice::Event) { self.ice_ui.handle_event(event); } diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index 2bc64764c9..2812b98c3c 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -142,6 +142,7 @@ impl IcedRenderer { pub fn resize(&mut self, scaled_dims: Vec2, renderer: &mut Renderer) { self.win_dims = scaled_dims; + self.window_scissor = default_scissor(renderer); self.update_resolution_dependents(renderer.get_resolution()); @@ -467,15 +468,17 @@ impl IcedRenderer { vec![(); glyph_count], // Since we already passed in `bounds` to position the glyphs some of this // seems redundant... + // Note: we can't actually use this because dropping glyphs messeses up the + // counting and there is not a convenient method provided to drop out of bounds + // glyphs while positioning them glyph_brush::ab_glyph::Rect { min: glyph_brush::ab_glyph::point( - bounds.x * self.p_scale, - //(self.win_dims.y - bounds.y) * self.p_scale, - bounds.y * self.p_scale, + -10000.0, //bounds.x * self.p_scale, + -10000.0, //bounds.y * self.p_scale, ), max: glyph_brush::ab_glyph::point( - (bounds.x + bounds.width) * self.p_scale, - (bounds.y + bounds.height) * self.p_scale, + 10000.0, //(bounds.x + bounds.width) * self.p_scale, + 10000.0, //(bounds.y + bounds.height) * self.p_scale, ), }, ); @@ -501,21 +504,17 @@ impl IcedRenderer { } }, Primitive::Clip { bounds, content } => { - // Check for a change in the scissor. let new_scissor = { - // Calculate minimum x and y coordinates while - // flipping y axis (from +down to +uo) and - // moving origin from top-left corner to bottom-left let min_x = bounds.x; - let min_y = self.win_dims.y - bounds.y; + let min_y = bounds.y; let intersection = Aabr { min: Vec2 { - x: (min_x * self.p_scale) as u16, - y: (min_y * self.p_scale) as u16, + x: (bounds.x * self.p_scale) as u16, + y: (bounds.y * self.p_scale) as u16, }, max: Vec2 { - x: ((min_x + bounds.width) * self.p_scale) as u16, - y: ((min_y + bounds.height) * self.p_scale) as u16, + x: ((bounds.x + bounds.width) * self.p_scale) as u16, + y: ((bounds.y + bounds.height) * self.p_scale) as u16, }, } .intersection(self.window_scissor); diff --git a/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs b/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs index a3a47af190..f28adb7bfc 100644 --- a/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs +++ b/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs @@ -27,7 +27,7 @@ impl compound_graphic::Renderer for IcedRenderer { }, GraphicKind::Color(color) => Primitive::Rectangle { bounds, - linear_color: srgba_to_linear(color.map(|e| e as f32 * 255.0)), + linear_color: srgba_to_linear(color.map(|e| e as f32 / 255.0)), }, }) .collect(), diff --git a/voxygen/src/ui/ice/renderer/widget/text_input.rs b/voxygen/src/ui/ice/renderer/widget/text_input.rs index 9c6ad3f93f..3d5ade3dba 100644 --- a/voxygen/src/ui/ice/renderer/widget/text_input.rs +++ b/voxygen/src/ui/ice/renderer/widget/text_input.rs @@ -6,9 +6,9 @@ use iced::{ Color, Point, Rectangle, }; -const CURSOR_WIDTH: f32 = 1.0; +const CURSOR_WIDTH: f32 = 2.0; // Extra scroll offset past the cursor -const EXTRA_OFFSET: f32 = 5.0; +const EXTRA_OFFSET: f32 = 10.0; impl text_input::Renderer for IcedRenderer { type Font = FontId; @@ -116,7 +116,7 @@ impl text_input::Renderer for IcedRenderer { // Allocation :( let text = value.to_string(); - let text = if text.is_empty() { Some(&*text) } else { None }; + let text = if !text.is_empty() { Some(&*text) } else { None }; // TODO: background from style, image? @@ -124,7 +124,7 @@ impl text_input::Renderer for IcedRenderer { let color = if text.is_some() { Color::WHITE } else { - Color::from_rgba(1.0, 1.0, 1.0, 0.4) + Color::from_rgba(1.0, 1.0, 1.0, 0.3) }; let linear_color = color.into_linear().into(); @@ -141,7 +141,7 @@ impl text_input::Renderer for IcedRenderer { ( Primitive::Rectangle { bounds: Rectangle { - x: text_bounds.x + position, + x: text_bounds.x + position - offset, y: text_bounds.y, width: CURSOR_WIDTH / p_scale, height: text_bounds.height, @@ -158,12 +158,17 @@ impl text_input::Renderer for IcedRenderer { let (left_position, left_offset) = cursor_and_scroll_offset(left); let (right_position, right_offset) = cursor_and_scroll_offset(right); + let offset = if end == right { + right_offset + } else { + left_offset + }; let width = right_position - left_position; ( Primitive::Rectangle { bounds: Rectangle { - x: text_bounds.x + left_position, + x: text_bounds.x + left_position - left_offset, y: text_bounds.y, width, height: text_bounds.height, @@ -171,11 +176,7 @@ impl text_input::Renderer for IcedRenderer { // TODO: selection color from stlye linear_color: Color::from_rgba(1.0, 0.0, 1.0, 0.2).into_linear().into(), }, - if end == right { - right_offset - } else { - left_offset - }, + offset, ) }, }; @@ -185,9 +186,10 @@ impl text_input::Renderer for IcedRenderer { (None, 0.0) }; + let display_text = text.unwrap_or(if state.is_focused() { "" } else { placeholder }); let section = glyph_brush::Section { screen_position: ( - text_bounds.x * p_scale + scroll_offset, + text_bounds.x * p_scale - scroll_offset, text_bounds.center_y() * p_scale, ), bounds: (text_bounds.width * p_scale, text_bounds.height * p_scale), @@ -197,7 +199,7 @@ impl text_input::Renderer for IcedRenderer { v_align: glyph_brush::VerticalAlign::Center, }, text: vec![glyph_brush::Text { - text: text.unwrap_or(placeholder), + text: display_text, scale: (size as f32 * p_scale).into(), font_id: font.0, extra: (), @@ -229,7 +231,7 @@ impl text_input::Renderer for IcedRenderer { }; // Probably already computed this somewhere - let text_width = self.measure_value(text.unwrap_or(placeholder), size, font); + let text_width = self.measure_value(display_text, size, font); let primitive = if text_width > text_bounds.width { Primitive::Clip { From 1e1e3a66accd420ddd23a14c23f1269daf71f4a4 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sat, 30 May 2020 13:21:36 -0400 Subject: [PATCH 17/61] Fix whitespace glitches, center text cursor --- voxygen/src/ui/graphic/mod.rs | 1 + .../src/ui/ice/renderer/widget/text_input.rs | 53 +++++++++++++++++-- 2 files changed, 49 insertions(+), 5 deletions(-) diff --git a/voxygen/src/ui/graphic/mod.rs b/voxygen/src/ui/graphic/mod.rs index b4e52c4b5c..0e0703ad13 100644 --- a/voxygen/src/ui/graphic/mod.rs +++ b/voxygen/src/ui/graphic/mod.rs @@ -58,6 +58,7 @@ pub struct Id(u32); pub struct TexId(usize); type Parameters = (Id, Vec2); +// TODO replace with slab/slotmap type GraphicMap = HashMap; enum CachedDetails { diff --git a/voxygen/src/ui/ice/renderer/widget/text_input.rs b/voxygen/src/ui/ice/renderer/widget/text_input.rs index 3d5ade3dba..2d98756629 100644 --- a/voxygen/src/ui/ice/renderer/widget/text_input.rs +++ b/voxygen/src/ui/ice/renderer/widget/text_input.rs @@ -42,16 +42,44 @@ impl text_input::Renderer for IcedRenderer { .map_or(0.0, |rect| rect.width() / p_scale); // glyph_brush ignores the exterior spaces + // or does it!!! // TODO: need better layout lib - let exterior_spaces = value.len() - value.trim().len(); + /*let exterior_spaces = value.len() - value.trim().len(); if exterior_spaces > 0 { use glyph_brush::ab_glyph::{Font, ScaleFont}; // Could cache this if it is slow + let sur = format!("x{}x", value); + let section = glyph_brush::Section { + screen_position: (0.0, 0.0), + bounds: (f32::INFINITY, f32::INFINITY), + layout: Default::default(), + text: vec![glyph_brush::Text { + text: &sur, + scale: (size as f32 * p_scale).into(), + font_id: font.0, + extra: (), + }], + }; let font = glyph_calculator.fonts()[font.0].as_scaled(size as f32); - let space_width = font.h_advance(font.glyph_id(' ')); + let space_id = font.glyph_id(' '); + let x_id = font.glyph_id('x'); + let space_width = font.h_advance(space_id); + let x_width = font.h_advance(x_id); + let kern1 = font.kern(x_id, space_id); + let kern2 = font.kern(space_id, x_id); + dbg!(font.kern(x_id, x_id)); + let sur_width = glyph_calculator + .glyph_bounds(section) + .map_or(0.0, |rect| rect.width() / p_scale); + dbg!(space_width); + dbg!(width); + dbg!(sur_width); + let extra = x_width * 2.0 + dbg!(kern1) + dbg!(kern2); + dbg!(extra); + dbg!(sur_width - extra); width += exterior_spaces as f32 * space_width; - } + }*/ width } @@ -141,7 +169,7 @@ impl text_input::Renderer for IcedRenderer { ( Primitive::Rectangle { bounds: Rectangle { - x: text_bounds.x + position - offset, + x: text_bounds.x + position - offset - CURSOR_WIDTH / p_scale / 2.0, y: text_bounds.y, width: CURSOR_WIDTH / p_scale, height: text_bounds.height, @@ -192,7 +220,10 @@ impl text_input::Renderer for IcedRenderer { text_bounds.x * p_scale - scroll_offset, text_bounds.center_y() * p_scale, ), - bounds: (text_bounds.width * p_scale, text_bounds.height * p_scale), + bounds: ( + 10000.0, /* text_bounds.width * p_scale */ + text_bounds.height * p_scale, + ), layout: glyph_brush::Layout::SingleLine { line_breaker: Default::default(), h_align: glyph_brush::HorizontalAlign::Left, @@ -210,6 +241,18 @@ impl text_input::Renderer for IcedRenderer { .cache .glyph_cache_mut() .glyphs(section) + // We would still have to generate vertices for these even if they have no pixels + // Note: this is somewhat hacky and could fail if there is a non-whitespace character + // that is not visible (to solve this we could use the extra values in + // queue_pre_positioned to keep track of which glyphs are actually returned by + // proccess_queued) + .filter(|g| { + !display_text[g.byte_index..] + .chars() + .next() + .unwrap() + .is_whitespace() + }) .cloned() .collect::>(); From f4a704eaa61090111d31ed507a124f55358128b4 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sat, 30 May 2020 17:10:12 -0400 Subject: [PATCH 18/61] Input fields layout tweaks --- voxygen/src/menu/main/ui.rs | 27 +++++++++++++++------------ 1 file changed, 15 insertions(+), 12 deletions(-) diff --git a/voxygen/src/menu/main/ui.rs b/voxygen/src/menu/main/ui.rs index a34318d27a..6c7719b13a 100644 --- a/voxygen/src/menu/main/ui.rs +++ b/voxygen/src/menu/main/ui.rs @@ -288,7 +288,8 @@ impl IcedState { .height(Length::Fill) .align_y(Align::End) .padding(20); - + const INPUT_WIDTH: u16 = 250; + const INPUT_TEXT_SIZE: u16 = 24; let banner_content = Column::with_children(vec![ Image::new(self.imgs.v_logo) .fix_aspect_ratio() @@ -297,7 +298,8 @@ impl IcedState { Space::new(Length::Fill, Length::FillPortion(5)).into(), Column::with_children(vec![ BackgroundContainer::new( - CompoundGraphic::padded_image(self.imgs.input_bg, [169, 25], [0, 0, 0, 1]) + Image::new(self.imgs.input_bg) + .width(Length::Units(INPUT_WIDTH)) .fix_aspect_ratio(), TextInput::new( &mut self.username_input, @@ -305,14 +307,14 @@ impl IcedState { &self.username_value, Message::Username, ) - // TODO: wrap in fill text thing to auto adjust the size - .size(22) - .padding(25) + .size(INPUT_TEXT_SIZE) .on_submit(Message::FocusPassword), ) + .padding(Padding::new().horizontal(10).top(10)) .into(), BackgroundContainer::new( - CompoundGraphic::padded_image(self.imgs.input_bg, [169, 25], [0, 1, 0, 1]) + Image::new(self.imgs.input_bg) + .width(Length::Units(INPUT_WIDTH)) .fix_aspect_ratio(), TextInput::new( &mut self.password_input, @@ -320,14 +322,15 @@ impl IcedState { &self.password_value, Message::Password, ) - .size(22) - .padding(25) + .size(INPUT_TEXT_SIZE) .password() .on_submit(Message::Multiplayer), ) + .padding(Padding::new().horizontal(10).top(8)) .into(), BackgroundContainer::new( - CompoundGraphic::padded_image(self.imgs.input_bg, [169, 25], [0, 1, 0, 0]) + Image::new(self.imgs.input_bg) + .width(Length::Units(INPUT_WIDTH)) .fix_aspect_ratio(), TextInput::new( &mut self.server_input, @@ -335,12 +338,13 @@ impl IcedState { &self.server_value, Message::Server, ) - .size(22) - .padding(25) + .size(INPUT_TEXT_SIZE) .on_submit(Message::Multiplayer), ) + .padding(Padding::new().horizontal(10).top(8)) .into(), ]) + .spacing(2) .height(Length::FillPortion(50)) .into(), Column::with_children(vec![ @@ -360,7 +364,6 @@ impl IcedState { ]) .max_width(240) .spacing(8) // TODO scale with available window size, awkward because both width and height can become limiting factors, might need custom column, could also just use fill portion - .height(Length::FillPortion(25)) .into(), ]) .width(Length::Fill) From 7d64ce93bdf24825930acd063db8d04c49280500 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sun, 31 May 2020 14:50:57 -0400 Subject: [PATCH 19/61] More main menu work, split main menu ui into multiple modules, misc fixes in the renderer --- voxygen/src/menu/main/ui/connecting.rs | 36 ++ voxygen/src/menu/main/ui/login.rs | 237 ++++++++ voxygen/src/menu/main/{ui.rs => ui/mod.rs} | 535 +++++++----------- voxygen/src/ui/ice/component/mod.rs | 1 + voxygen/src/ui/ice/component/neat_button.rs | 50 +- voxygen/src/ui/ice/renderer/mod.rs | 4 +- voxygen/src/ui/ice/renderer/widget/text.rs | 7 + .../src/ui/ice/renderer/widget/text_input.rs | 2 +- 8 files changed, 505 insertions(+), 367 deletions(-) create mode 100644 voxygen/src/menu/main/ui/connecting.rs create mode 100644 voxygen/src/menu/main/ui/login.rs rename voxygen/src/menu/main/{ui.rs => ui/mod.rs} (79%) diff --git a/voxygen/src/menu/main/ui/connecting.rs b/voxygen/src/menu/main/ui/connecting.rs new file mode 100644 index 0000000000..4297ba4035 --- /dev/null +++ b/voxygen/src/menu/main/ui/connecting.rs @@ -0,0 +1,36 @@ +use super::{IcedImgs as Imgs, Message}; +use crate::{ + i18n::Localization, + ui::ice::{ + component::neat_button, + widget::{image, BackgroundContainer, Image}, + Element, + }, +}; +use iced::{button, Length, Space}; + +/// Connecting screen for the main menu +pub struct Screen { + cancel_button: button::State, +} + +impl Screen { + pub fn new() -> Self { + Self { + cancel_button: Default::default(), + } + } + + pub(super) fn view( + &mut self, + imgs: &Imgs, + bg_img: image::Handle, + start: &std::time::Instant, + i18n: &Localization, + ) -> Element { + let content = Space::new(Length::Fill, Length::Fill); + // Note: could replace this with styling on iced's container since we aren't + // using fixed aspect ratio + BackgroundContainer::new(Image::new(bg_img), content).into() + } +} diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs new file mode 100644 index 0000000000..de21d09597 --- /dev/null +++ b/voxygen/src/menu/main/ui/login.rs @@ -0,0 +1,237 @@ +use super::{IcedImgs as Imgs, LoginInfo, Message}; +use crate::{ + i18n::Localization, + ui::ice::{ + component::neat_button, + widget::{ + compound_graphic::{CompoundGraphic, Graphic}, + BackgroundContainer, Image, Padding, + }, + ButtonStyle, Element, + }, +}; +use iced::{button, text_input, Align, Column, Container, Length, Row, Space, TextInput}; +use vek::*; + +const TEXT_COLOR: iced::Color = iced::Color::from_rgb(1.0, 1.0, 1.0); +const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1.0, 0.2); +const FILL_FRAC_ONE: f32 = 0.77; +const FILL_FRAC_TWO: f32 = 0.53; +const INPUT_WIDTH: u16 = 250; +const INPUT_TEXT_SIZE: u16 = 24; + +/// Login screen for the main menu +pub struct Screen { + quit_button: button::State, + settings_button: button::State, + servers_button: button::State, + + pub banner: Banner, +} + +impl Screen { + pub fn new() -> Self { + Self { + servers_button: Default::default(), + settings_button: Default::default(), + quit_button: Default::default(), + + banner: Banner::new(), + } + } + + pub(super) fn view( + &mut self, + imgs: &Imgs, + login_info: &LoginInfo, + i18n: &Localization, + ) -> Element { + let button_style = ButtonStyle::new(imgs.button) + .hover_image(imgs.button_hover) + .press_image(imgs.button_press) + .text_color(TEXT_COLOR) + .disabled_text_color(DISABLED_TEXT_COLOR); + + let buttons = Column::with_children(vec![ + neat_button( + &mut self.servers_button, + i18n.get("common.servers"), + FILL_FRAC_ONE, + button_style, + Some(Message::ShowServers), + ), + neat_button( + &mut self.settings_button, + i18n.get("common.settings"), + FILL_FRAC_ONE, + button_style, + None, + ), + neat_button( + &mut self.quit_button, + i18n.get("common.quit"), + FILL_FRAC_ONE, + button_style, + Some(Message::Quit), + ), + ]) + .width(Length::Fill) + .max_width(200) + .spacing(5) + .padding(10); + + let buttons = Container::new(buttons) + .width(Length::Fill) + .height(Length::Fill) + .align_y(Align::End) + .padding(20); + + let banner = self.banner.view(imgs, login_info, i18n, button_style); + + let central_column = Container::new(banner) + .width(Length::Fill) + .height(Length::Fill) + .align_x(Align::Center) + .align_y(Align::Center); + + let image3 = Image::new(imgs.banner_bottom).fix_aspect_ratio(); + + let content = + Row::with_children(vec![buttons.into(), central_column.into(), image3.into()]) + .width(Length::Fill) + .height(Length::Fill) + .spacing(10); + + BackgroundContainer::new(Image::new(imgs.bg), content).into() + } +} + +pub struct Banner { + pub username: text_input::State, + pub password: text_input::State, + pub server: text_input::State, + + multiplayer_button: button::State, + #[cfg(feature = "singleplayer")] + singleplayer_button: button::State, +} + +impl Banner { + fn new() -> Self { + Self { + username: Default::default(), + password: Default::default(), + server: Default::default(), + + multiplayer_button: Default::default(), + #[cfg(feature = "singleplayer")] + singleplayer_button: Default::default(), + } + } + + fn view( + &mut self, + imgs: &Imgs, + login_info: &LoginInfo, + i18n: &Localization, + button_style: ButtonStyle, + ) -> Element { + let banner_content = Column::with_children(vec![ + Image::new(imgs.v_logo) + .fix_aspect_ratio() + .height(Length::FillPortion(20)) + .into(), + Space::new(Length::Fill, Length::FillPortion(5)).into(), + Column::with_children(vec![ + BackgroundContainer::new( + Image::new(imgs.input_bg) + .width(Length::Units(INPUT_WIDTH)) + .fix_aspect_ratio(), + TextInput::new( + &mut self.username, + "Username", + &login_info.username, + Message::Username, + ) + .size(INPUT_TEXT_SIZE) + .on_submit(Message::FocusPassword), + ) + .padding(Padding::new().horizontal(10).top(10)) + .into(), + BackgroundContainer::new( + Image::new(imgs.input_bg) + .width(Length::Units(INPUT_WIDTH)) + .fix_aspect_ratio(), + TextInput::new( + &mut self.password, + "Password", + &login_info.password, + Message::Password, + ) + .size(INPUT_TEXT_SIZE) + .password() + .on_submit(Message::Multiplayer), + ) + .padding(Padding::new().horizontal(10).top(8)) + .into(), + BackgroundContainer::new( + Image::new(imgs.input_bg) + .width(Length::Units(INPUT_WIDTH)) + .fix_aspect_ratio(), + TextInput::new( + &mut self.server, + "Server", + &login_info.server, + Message::Server, + ) + .size(INPUT_TEXT_SIZE) + .on_submit(Message::Multiplayer), + ) + .padding(Padding::new().horizontal(10).top(8)) + .into(), + ]) + .spacing(2) + .height(Length::FillPortion(50)) + .into(), + Column::with_children(vec![ + neat_button( + &mut self.multiplayer_button, + i18n.get("common.multiplayer"), + FILL_FRAC_TWO, + button_style, + Some(Message::Multiplayer), + ), + #[cfg(feature = "singleplayer")] + neat_button( + &mut self.singleplayer_button, + i18n.get("common.singleplayer"), + FILL_FRAC_TWO, + button_style, + Some(Message::Singleplayer), + ), + ]) + .max_width(240) + .spacing(8) + .into(), + ]) + .width(Length::Fill) + .height(Length::Fill) + .align_items(Align::Center); + + let banner = BackgroundContainer::new( + CompoundGraphic::from_graphics(vec![ + Graphic::image(imgs.banner_top, [138, 17], [0, 0]), + Graphic::rect(Rgba::new(0, 0, 0, 230), [130, 195], [4, 17]), + Graphic::image(imgs.banner, [130, 15], [4, 212]) + .color(Rgba::new(255, 255, 255, 230)), + ]) + .fix_aspect_ratio() + .height(Length::Fill), + banner_content, + ) + .padding(Padding::new().horizontal(16).vertical(20)) + .max_width(330); + + banner.into() + } +} diff --git a/voxygen/src/menu/main/ui.rs b/voxygen/src/menu/main/ui/mod.rs similarity index 79% rename from voxygen/src/menu/main/ui.rs rename to voxygen/src/menu/main/ui/mod.rs index 6c7719b13a..43d363c573 100644 --- a/voxygen/src/menu/main/ui.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -1,3 +1,6 @@ +mod connecting; +mod login; + use crate::{ i18n::{i18n_asset_key, Localization}, render::Renderer, @@ -22,6 +25,7 @@ use conrod_core::{ use image::DynamicImage; use rand::{seq::SliceRandom, thread_rng, Rng}; use std::time::Duration; +use ui::ice::widget; const COL1: Color = Color::Rgba(0.07, 0.1, 0.1, 0.9); @@ -29,6 +33,194 @@ const COL1: Color = Color::Rgba(0.07, 0.1, 0.1, 0.9); /*const UI_MAIN: Color = Color::Rgba(0.61, 0.70, 0.70, 1.0); // Greenish Blue const UI_HIGHLIGHT_0: Color = Color::Rgba(0.79, 1.09, 1.09, 1.0);*/ +use iced::text_input; +image_ids_ice! { + struct IcedImgs { + + v_logo: "voxygen.element.v_logo", + + info_frame: "voxygen.element.frames.info_frame_2", + + //banner: "voxygen.element.frames.banner", + + bg: "voxygen.background.bg_main", + banner: "voxygen.element.frames.banner_png", + banner_bottom: "voxygen.element.frames.banner_bottom_png", + banner_top: "voxygen.element.frames.banner_top", + button: "voxygen.element.buttons.button", + button_hover: "voxygen.element.buttons.button_hover", + button_press: "voxygen.element.buttons.button_press", + input_bg: "voxygen.element.misc_bg.textbox", + disclaimer: "voxygen.element.frames.disclaimer", + loading_art: "voxygen.element.frames.loading_screen.loading_bg", + loading_art_l: "voxygen.element.frames.loading_screen.loading_bg_l", + loading_art_r: "voxygen.element.frames.loading_screen.loading_bg_r", + + + nothing: (), + } +} + +// Randomly loaded background images +const BG_IMGS: [&str; 16] = [ + "voxygen.background.bg_1", + "voxygen.background.bg_2", + "voxygen.background.bg_3", + "voxygen.background.bg_4", + "voxygen.background.bg_5", + "voxygen.background.bg_6", + "voxygen.background.bg_7", + "voxygen.background.bg_8", + "voxygen.background.bg_9", + //"voxygen.background.bg_10", + "voxygen.background.bg_11", + //"voxygen.background.bg_12", + "voxygen.background.bg_13", + //"voxygen.background.bg_14", + "voxygen.background.bg_15", + "voxygen.background.bg_16", +]; + +pub enum Event { + LoginAttempt { + username: String, + password: String, + server_address: String, + }, + CancelLoginAttempt, + #[cfg(feature = "singleplayer")] + StartSingleplayer, + Quit, + Settings, + //DisclaimerClosed, + AuthServerTrust(String, bool), +} + +pub enum PopupType { + Error, + ConnectionInfo, + AuthTrustPrompt(String), +} + +pub struct PopupData { + msg: String, + popup_type: PopupType, +} + +pub struct LoginInfo { + pub username: String, + pub password: String, + pub server: String, +} + +enum Screen { + Login { + screen: login::Screen, + }, + Connecting { + screen: connecting::Screen, + // TODO: why instant? + start: std::time::Instant, + }, +} + +// TODO: use i18n font scale thing +struct IcedState { + imgs: IcedImgs, + bg_img: widget::image::Handle, + i18n: std::sync::Arc, + + // TODO: not sure if this should be used for connecting + popup: Option, + show_servers: bool, + show_disclaimer: bool, + login_info: LoginInfo, + + screen: Screen, +} + +#[derive(Clone)] +enum Message { + Quit, + ShowServers, + #[cfg(feature = "singleplayer")] + Singleplayer, + Multiplayer, + Username(String), + Password(String), + Server(String), + FocusPassword, +} + +impl IcedState { + fn new( + imgs: IcedImgs, + bg_img: widget::image::Handle, + i18n: std::sync::Arc, + ) -> Self { + Self { + imgs, + bg_img, + i18n, + popup: None, + show_servers: false, + show_disclaimer: false, + login_info: LoginInfo { + username: String::new(), + password: String::new(), + server: String::new(), + }, + + screen: Screen::Login { + screen: login::Screen::new(), + }, + } + } + + fn view(&mut self) -> Element { + match &mut self.screen { + Screen::Login { screen } => screen.view(&self.imgs, &self.login_info, &self.i18n), + Screen::Connecting { screen, start } => { + screen.view(&self.imgs, self.bg_img, &start, &self.i18n) + }, + } + } + + fn update(&mut self, message: Message, events: &mut Vec) { + match message { + Message::Quit => events.push(Event::Quit), + Message::ShowServers => self.show_servers = true, + #[cfg(feature = "singleplayer")] + Message::Singleplayer => events.push(Event::StartSingleplayer), + Message::Multiplayer => { + self.screen = Screen::Connecting { + screen: connecting::Screen::new(), + start: std::time::Instant::now(), + }; + self.popup = Some(PopupData { + msg: [self.i18n.get("main.connecting"), "..."].concat(), + popup_type: PopupType::ConnectionInfo, + }); + + events.push(Event::LoginAttempt { + username: self.login_info.username.clone(), + password: self.login_info.password.clone(), + server_address: self.login_info.server.clone(), + }); + }, + Message::Username(new_value) => self.login_info.username = new_value, + Message::Password(new_value) => self.login_info.password = new_value, + Message::Server(new_value) => self.login_info.server = new_value, + Message::FocusPassword => { + if let Screen::Login { screen } = &mut self.screen { + screen.banner.password = text_input::State::focused(); + screen.banner.username = text_input::State::new(); + } + }, + } + } +} + widget_ids! { struct Ids { // Background and logo @@ -38,7 +230,6 @@ widget_ids! { alpha_text, banner, banner_top, - gears, // Disclaimer //disc_window, //disc_text_1, @@ -82,13 +273,6 @@ widget_ids! { info_bottom, // Auth Trust Prompt button_add_auth_trust, - // Loading Screen Tips - tip_txt_bg, - tip_txt, - // Loading Screen Artwork - mid, - left, - right, } } @@ -109,39 +293,7 @@ image_ids! { button_press: "voxygen.element.buttons.button_press", input_bg: "voxygen.element.misc_bg.textbox", //disclaimer: "voxygen.element.frames.disclaimer", - loading_art: "voxygen.element.frames.loading_screen.loading_bg", - loading_art_l: "voxygen.element.frames.loading_screen.loading_bg_l", - loading_art_r: "voxygen.element.frames.loading_screen.loading_bg_r", - // Animation - f1: "voxygen.element.animation.gears.1", - f2: "voxygen.element.animation.gears.2", - f3: "voxygen.element.animation.gears.3", - f4: "voxygen.element.animation.gears.4", - f5: "voxygen.element.animation.gears.5", - - nothing: (), - } -} - -image_ids_ice! { - struct IcedImgs { - - v_logo: "voxygen.element.v_logo", - - info_frame: "voxygen.element.frames.info_frame_2", - - //banner: "voxygen.element.frames.banner", - - bg: "voxygen.background.bg_main", - banner: "voxygen.element.frames.banner_png", - banner_bottom: "voxygen.element.frames.banner_bottom_png", - banner_top: "voxygen.element.frames.banner_top", - button: "voxygen.element.buttons.button", - button_hover: "voxygen.element.buttons.button_hover", - button_press: "voxygen.element.buttons.button_press", - input_bg: "voxygen.element.misc_bg.textbox", - disclaimer: "voxygen.element.frames.disclaimer", nothing: (), @@ -150,7 +302,7 @@ image_ids_ice! { rotation_image_ids! { pub struct ImgsRot { - + // Tooltip Test tt_side: "voxygen/element/frames/tt_test_edge", @@ -158,267 +310,6 @@ rotation_image_ids! { } } -pub enum Event { - LoginAttempt { - username: String, - password: String, - server_address: String, - }, - CancelLoginAttempt, - #[cfg(feature = "singleplayer")] - StartSingleplayer, - Quit, - Settings, - //DisclaimerClosed, - AuthServerTrust(String, bool), -} - -pub enum PopupType { - Error, - ConnectionInfo, - AuthTrustPrompt(String), -} - -pub struct PopupData { - msg: String, - popup_type: PopupType, -} - -use ui::ice::component::neat_button; -struct IcedState { - imgs: IcedImgs, - quit_button: neat_button::State, - settings_button: neat_button::State, - servers_button: neat_button::State, - multiplayer_button: neat_button::State, - #[cfg(feature = "singleplayer")] - singleplayer_button: neat_button::State, - username_input: iced::text_input::State, - password_input: iced::text_input::State, - server_input: iced::text_input::State, - username_value: String, - password_value: String, - server_value: String, - show_servers: bool, -} - -#[derive(Clone)] // TODO: why does iced require Clone? -enum Message { - Quit, - ShowServers, - #[cfg(feature = "singleplayer")] - Singleplayer, - Multiplayer, - Username(String), - Password(String), - Server(String), - FocusPassword, -} - -impl IcedState { - pub fn new(imgs: IcedImgs) -> Self { - Self { - imgs, - servers_button: Default::default(), - settings_button: Default::default(), - quit_button: Default::default(), - multiplayer_button: Default::default(), - #[cfg(feature = "singleplayer")] - singleplayer_button: Default::default(), - username_input: Default::default(), - password_input: Default::default(), - server_input: Default::default(), - username_value: String::new(), - password_value: String::new(), - server_value: String::new(), - show_servers: false, - } - } - - pub fn view(&mut self, i18n: &Localization) -> Element { - //let button_font_size = 30; - const TEXT_COLOR: iced::Color = iced::Color::from_rgb(1.0, 1.0, 1.0); - const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1.0, 0.2); - const FILL_FRAC_ONE: f32 = 0.77; - const FILL_FRAC_TWO: f32 = 0.53; - - use iced::{Align, Column, Container, Length, Row, Space, TextInput}; - use ui::ice::{ - widget::{ - compound_graphic::{CompoundGraphic, Graphic}, - BackgroundContainer, Image, Padding, - }, - ButtonStyle, - }; - use vek::*; - - let button_style = ButtonStyle::new(self.imgs.button) - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .text_color(TEXT_COLOR) - .disabled_text_color(DISABLED_TEXT_COLOR); - - let buttons = Column::with_children(vec![ - self.servers_button.view( - i18n.get("common.servers"), - FILL_FRAC_ONE, - button_style, - Some(Message::ShowServers), - ), - self.settings_button.view( - i18n.get("common.settings"), - FILL_FRAC_ONE, - button_style, - None, - ), - self.quit_button.view( - i18n.get("common.quit"), - FILL_FRAC_ONE, - button_style, - Some(Message::Quit), - ), - ]) - .width(Length::Fill) - .max_width(200) - .spacing(5) - .padding(10); - - let buttons = Container::new(buttons) - .width(Length::Fill) - .height(Length::Fill) - .align_y(Align::End) - .padding(20); - const INPUT_WIDTH: u16 = 250; - const INPUT_TEXT_SIZE: u16 = 24; - let banner_content = Column::with_children(vec![ - Image::new(self.imgs.v_logo) - .fix_aspect_ratio() - .height(Length::FillPortion(20)) - .into(), - Space::new(Length::Fill, Length::FillPortion(5)).into(), - Column::with_children(vec![ - BackgroundContainer::new( - Image::new(self.imgs.input_bg) - .width(Length::Units(INPUT_WIDTH)) - .fix_aspect_ratio(), - TextInput::new( - &mut self.username_input, - "Username", - &self.username_value, - Message::Username, - ) - .size(INPUT_TEXT_SIZE) - .on_submit(Message::FocusPassword), - ) - .padding(Padding::new().horizontal(10).top(10)) - .into(), - BackgroundContainer::new( - Image::new(self.imgs.input_bg) - .width(Length::Units(INPUT_WIDTH)) - .fix_aspect_ratio(), - TextInput::new( - &mut self.password_input, - "Password", - &self.password_value, - Message::Password, - ) - .size(INPUT_TEXT_SIZE) - .password() - .on_submit(Message::Multiplayer), - ) - .padding(Padding::new().horizontal(10).top(8)) - .into(), - BackgroundContainer::new( - Image::new(self.imgs.input_bg) - .width(Length::Units(INPUT_WIDTH)) - .fix_aspect_ratio(), - TextInput::new( - &mut self.server_input, - "Server", - &self.server_value, - Message::Server, - ) - .size(INPUT_TEXT_SIZE) - .on_submit(Message::Multiplayer), - ) - .padding(Padding::new().horizontal(10).top(8)) - .into(), - ]) - .spacing(2) - .height(Length::FillPortion(50)) - .into(), - Column::with_children(vec![ - self.multiplayer_button.view( - i18n.get("common.multiplayer"), - FILL_FRAC_TWO, - button_style, - Some(Message::Multiplayer), - ), - #[cfg(feature = "singleplayer")] - self.singleplayer_button.view( - i18n.get("common.singleplayer"), - FILL_FRAC_TWO, - button_style, - Some(Message::Singleplayer), - ), - ]) - .max_width(240) - .spacing(8) // TODO scale with available window size, awkward because both width and height can become limiting factors, might need custom column, could also just use fill portion - .into(), - ]) - .width(Length::Fill) - .height(Length::Fill) - .align_items(Align::Center); - - let banner = BackgroundContainer::new( - CompoundGraphic::from_graphics(vec![ - Graphic::image(self.imgs.banner_top, [138, 17], [0, 0]), - Graphic::rect(Rgba::new(0, 0, 0, 230), [130, 195], [4, 17]), - Graphic::image(self.imgs.banner, [130, 15], [4, 212]) - .color(Rgba::new(255, 255, 255, 230)), - ]) - .fix_aspect_ratio() - .height(Length::Fill), - banner_content, - ) - .padding(Padding::new().horizontal(16).vertical(20)) - .max_width(330); - - let central_column = Container::new(banner) - .width(Length::Fill) - .height(Length::Fill) - .align_x(Align::Center) - .align_y(Align::Center); - - let image3 = Image::new(self.imgs.banner_bottom).fix_aspect_ratio(); - - let content = - Row::with_children(vec![buttons.into(), central_column.into(), image3.into()]) - .width(Length::Fill) - .height(Length::Fill) - .spacing(10); - - BackgroundContainer::new(Image::new(self.imgs.bg), content).into() - } - - pub fn update(&mut self, message: Message, events: &mut Vec) { - match message { - Message::Quit => events.push(Event::Quit), - Message::ShowServers => self.show_servers = true, - #[cfg(feature = "singleplayer")] - Message::Singleplayer => events.push(Event::StartSingleplayer), - Message::Multiplayer => (), //TODO - Message::Username(new_value) => self.username_value = new_value, - Message::Password(new_value) => self.password_value = new_value, - Message::Server(new_value) => self.server_value = new_value, - Message::FocusPassword => { - self.password_input = iced::text_input::State::focused(); - self.username_input = iced::text_input::State::new(); - }, - } - } -} - pub struct MainMenuUi { ui: Ui, ice_ui: IcedUi, @@ -448,26 +339,6 @@ impl<'a> MainMenuUi { let window = &mut global_state.window; let networking = &global_state.settings.networking; let gameplay = &global_state.settings.gameplay; - // Randomly loaded background images - let bg_imgs = [ - "voxygen.background.bg_1", - "voxygen.background.bg_2", - "voxygen.background.bg_3", - "voxygen.background.bg_4", - "voxygen.background.bg_5", - "voxygen.background.bg_6", - "voxygen.background.bg_7", - "voxygen.background.bg_8", - "voxygen.background.bg_9", - //"voxygen.background.bg_10", - "voxygen.background.bg_11", - //"voxygen.background.bg_12", - "voxygen.background.bg_13", - //"voxygen.background.bg_14", - "voxygen.background.bg_15", - "voxygen.background.bg_16", - ]; - let mut rng = thread_rng(); let mut ui = Ui::new(window).unwrap(); ui.set_scaling_mode(gameplay.ui_scale); @@ -476,11 +347,8 @@ impl<'a> MainMenuUi { // Load images let imgs = Imgs::load(&mut ui).expect("Failed to load images"); let rot_imgs = ImgsRot::load(&mut ui).expect("Failed to load images!"); - let bg_img_id = ui.add_graphic(Graphic::Image( - DynamicImage::load_expect(bg_imgs.choose(&mut rng).unwrap()), - None, - )); - //let chosen_tip = *tips.choose(&mut rng).unwrap(); + let bg_img_spec = BG_IMGS.choose(&mut thread_rng()).unwrap(); + let bg_img_id = ui.add_graphic(Graphic::Image(DynamicImage::load_expect(bg_img_spec))); // Load language let i18n = Localization::load_expect(&i18n_asset_key( &global_state.settings.language.selected_language, @@ -502,7 +370,11 @@ impl<'a> MainMenuUi { }; let mut ice_ui = IcedUi::new(window, ice_font).unwrap(); - let ice_state = IcedState::new(IcedImgs::load(&mut ice_ui).expect("Failed to load images")); + let ice_state = IcedState::new( + IcedImgs::load(&mut ice_ui).expect("Failed to load images"), + ice_ui.add_graphic(Graphic::Image(load_expect(bg_img_spec))), + i18n.clone(), + ); Self { ui, @@ -1188,10 +1060,9 @@ impl<'a> MainMenuUi { pub fn maintain(&mut self, global_state: &mut GlobalState, dt: Duration) -> Vec { let mut events = self.update_layout(global_state, dt); self.ui.maintain(global_state.window.renderer_mut(), None); - let (messages, _) = self.ice_ui.maintain( - self.ice_state.view(&self.i18n), - global_state.window.renderer_mut(), - ); + let (messages, _) = self + .ice_ui + .maintain(self.ice_state.view(), global_state.window.renderer_mut()); messages .into_iter() .for_each(|message| self.ice_state.update(message, &mut events)); diff --git a/voxygen/src/ui/ice/component/mod.rs b/voxygen/src/ui/ice/component/mod.rs index 7f44faef0b..de0a8293eb 100644 --- a/voxygen/src/ui/ice/component/mod.rs +++ b/voxygen/src/ui/ice/component/mod.rs @@ -1 +1,2 @@ pub mod neat_button; +pub use neat_button::neat_button; diff --git a/voxygen/src/ui/ice/component/neat_button.rs b/voxygen/src/ui/ice/component/neat_button.rs index f0ce7d576c..6b68988b46 100644 --- a/voxygen/src/ui/ice/component/neat_button.rs +++ b/voxygen/src/ui/ice/component/neat_button.rs @@ -1,44 +1,32 @@ use crate::ui::ice as ui; -use iced::{Button, Element, Length}; +use iced::{button::State, Button, Element, Length}; use ui::{ widget::{AspectRatioContainer, FillText}, ButtonStyle, }; -#[derive(Default)] -pub struct State { - state: iced::button::State, -} - -impl State { - pub fn new() -> Self { Self::default() } - - pub fn view( - &mut self, - label: impl Into, - fill_fraction: f32, - button_style: ButtonStyle, - message: Option, - ) -> Element { - let button = Button::new( - &mut self.state, - FillText::new(label).fill_fraction(fill_fraction), - ) +pub fn neat_button( + state: &mut State, + label: impl Into, + fill_fraction: f32, + button_style: ButtonStyle, + message: Option, +) -> Element { + let button = Button::new(state, FillText::new(label).fill_fraction(fill_fraction)) .height(Length::Fill) .width(Length::Fill) .style(button_style); - let button = match message { - Some(message) => button.on_press(message), - None => button, - }; + let button = match message { + Some(message) => button.on_press(message), + None => button, + }; - let container = AspectRatioContainer::new(button); - let container = match button_style.active().0 { - Some(img) => container.ratio_of_image(img), - None => container, - }; + let container = AspectRatioContainer::new(button); + let container = match button_style.active().0 { + Some(img) => container.ratio_of_image(img), + None => container, + }; - container.into() - } + container.into() } diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index 2812b98c3c..a7fdd24645 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -446,7 +446,7 @@ impl IcedRenderer { }, Primitive::Text { glyphs, - bounds, // iced::Rectangle + bounds: _bounds, // iced::Rectangle linear_color, /*font, *horizontal_alignment, @@ -505,8 +505,6 @@ impl IcedRenderer { }, Primitive::Clip { bounds, content } => { let new_scissor = { - let min_x = bounds.x; - let min_y = bounds.y; let intersection = Aabr { min: Vec2 { x: (bounds.x * self.p_scale) as u16, diff --git a/voxygen/src/ui/ice/renderer/widget/text.rs b/voxygen/src/ui/ice/renderer/widget/text.rs index d9f6b1629e..9caf5c11c1 100644 --- a/voxygen/src/ui/ice/renderer/widget/text.rs +++ b/voxygen/src/ui/ice/renderer/widget/text.rs @@ -80,6 +80,13 @@ impl text::Renderer for IcedRenderer { .cache .glyph_cache_mut() .glyphs(section) + .filter(|g| { + !content[g.byte_index..] + .chars() + .next() + .unwrap() + .is_whitespace() + }) .cloned() .collect::>(); diff --git a/voxygen/src/ui/ice/renderer/widget/text_input.rs b/voxygen/src/ui/ice/renderer/widget/text_input.rs index 2d98756629..f267f7f451 100644 --- a/voxygen/src/ui/ice/renderer/widget/text_input.rs +++ b/voxygen/src/ui/ice/renderer/widget/text_input.rs @@ -37,7 +37,7 @@ impl text_input::Renderer for IcedRenderer { }; let mut glyph_calculator = self.cache.glyph_calculator(); - let mut width = glyph_calculator + let width = glyph_calculator .glyph_bounds(section) .map_or(0.0, |rect| rect.width() / p_scale); From 53fe8eec83b260494bdf8dc5275d4e2c46636d38 Mon Sep 17 00:00:00 2001 From: Imbris Date: Mon, 1 Jun 2020 04:05:10 -0400 Subject: [PATCH 20/61] Add parts of main menu such as version text and info text banner, make connection screen --- voxygen/src/menu/main/ui/connecting.rs | 54 ++++++- voxygen/src/menu/main/ui/login.rs | 62 +++++-- voxygen/src/menu/main/ui/mod.rs | 153 +++++++++++------- voxygen/src/ui/ice/renderer/mod.rs | 1 + .../src/ui/ice/widget/background_container.rs | 4 + 5 files changed, 204 insertions(+), 70 deletions(-) diff --git a/voxygen/src/menu/main/ui/connecting.rs b/voxygen/src/menu/main/ui/connecting.rs index 4297ba4035..59cee400e1 100644 --- a/voxygen/src/menu/main/ui/connecting.rs +++ b/voxygen/src/menu/main/ui/connecting.rs @@ -4,16 +4,20 @@ use crate::{ ui::ice::{ component::neat_button, widget::{image, BackgroundContainer, Image}, - Element, + ButtonStyle, Element, }, }; -use iced::{button, Length, Space}; +use iced::{button, Color, Column, Container, HorizontalAlignment, Length, Row, Space, Text}; /// Connecting screen for the main menu pub struct Screen { cancel_button: button::State, } +// TODO: move to super and unify with identical login consts +const TEXT_COLOR: iced::Color = iced::Color::from_rgb(1.0, 1.0, 1.0); +const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1.0, 0.2); + impl Screen { pub fn new() -> Self { Self { @@ -26,9 +30,53 @@ impl Screen { imgs: &Imgs, bg_img: image::Handle, start: &std::time::Instant, + status_text: &str, + version: &str, + time: f32, i18n: &Localization, ) -> Element { - let content = Space::new(Length::Fill, Length::Fill); + let fade_msg = (time * 2.0).sin() * 0.5 + 0.51; + let button_style = ButtonStyle::new(imgs.button) + .hover_image(imgs.button_hover) + .press_image(imgs.button_press) + .text_color(TEXT_COLOR) + .disabled_text_color(DISABLED_TEXT_COLOR); + + let version = Text::new(version) + .size(15) // move version text size to const + .width(Length::Fill) + .height(Length::Fill) + .horizontal_alignment(HorizontalAlignment::Right); + + let status = Text::new(status_text) + .size(80) + .color(Color::from_rgba(1.0, 1.0, 1.0, fade_msg)) + .width(Length::Fill); + + let status = Row::with_children(vec![ + Space::new(Length::Units(80), Length::Shrink).into(), + status.into(), + ]); + + let cancel = neat_button( + &mut self.cancel_button, + i18n.get("common.cancel"), + 0.7, + button_style, + Some(Message::CancelConnect), + ); + + let cancel = Container::new(cancel) + .width(Length::Fill) + .height(Length::Units(50)) + .center_x() + .padding(3); + + let content = Column::with_children(vec![version.into(), status.into(), cancel.into()]) + .width(Length::Fill) + .height(Length::Fill) + .padding(3); + // Note: could replace this with styling on iced's container since we aren't // using fixed aspect ratio BackgroundContainer::new(Image::new(bg_img), content).into() diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index de21d09597..1d46028638 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -1,4 +1,4 @@ -use super::{IcedImgs as Imgs, LoginInfo, Message}; +use super::{IcedImgs as Imgs, Info, LoginInfo, Message}; use crate::{ i18n::Localization, ui::ice::{ @@ -10,7 +10,10 @@ use crate::{ ButtonStyle, Element, }, }; -use iced::{button, text_input, Align, Column, Container, Length, Row, Space, TextInput}; +use iced::{ + button, text_input, Align, Column, Container, HorizontalAlignment, Length, Row, Space, Text, + TextInput, +}; use vek::*; const TEXT_COLOR: iced::Color = iced::Color::from_rgb(1.0, 1.0, 1.0); @@ -44,6 +47,9 @@ impl Screen { &mut self, imgs: &Imgs, login_info: &LoginInfo, + info: &Info, + version: &str, + show_servers: bool, i18n: &Localization, ) -> Element { let button_style = ButtonStyle::new(imgs.button) @@ -77,14 +83,39 @@ impl Screen { ]) .width(Length::Fill) .max_width(200) - .spacing(5) - .padding(10); + .spacing(5); let buttons = Container::new(buttons) .width(Length::Fill) .height(Length::Fill) - .align_y(Align::End) - .padding(20); + .align_y(Align::End); + + let left_column = if matches!(info, Info::Intro) { + let intro_text = i18n.get("main.login_process"); + + let info_window = BackgroundContainer::new( + CompoundGraphic::from_graphics(vec![ + Graphic::rect(Rgba::new(0, 0, 0, 240), [500, 300], [0, 0]), + // Note: a way to tell it to keep the height of this one piece constant and + // unstreched would be nice, I suppose we could just break this out into a + // column and use Length::Units + Graphic::image(imgs.banner_bottom, [500, 30], [0, 300]) + .color(Rgba::new(255, 255, 255, 240)), + ]) + .height(Length::Shrink), + Text::new(intro_text).size(21), + ) + .max_width(450) + .padding(Padding::new().horizontal(20).top(10).bottom(60)); + + Column::with_children(vec![info_window.into(), buttons.into()]) + .width(Length::Fill) + .height(Length::Fill) + .padding(27) + .into() + } else { + buttons.into() + }; let banner = self.banner.view(imgs, login_info, i18n, button_style); @@ -94,13 +125,20 @@ impl Screen { .align_x(Align::Center) .align_y(Align::Center); - let image3 = Image::new(imgs.banner_bottom).fix_aspect_ratio(); + let right_column = Text::new(version) + .size(15) + .width(Length::Fill) + .horizontal_alignment(HorizontalAlignment::Right); - let content = - Row::with_children(vec![buttons.into(), central_column.into(), image3.into()]) - .width(Length::Fill) - .height(Length::Fill) - .spacing(10); + let content = Row::with_children(vec![ + left_column, + central_column.into(), + right_column.into(), + ]) + .width(Length::Fill) + .height(Length::Fill) + .spacing(10) + .padding(3); BackgroundContainer::new(Image::new(imgs.bg), content).into() } diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 43d363c573..e6774244b6 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -14,6 +14,7 @@ use crate::{ GlobalState, }; //ImageFrame, Tooltip, +use crate::settings::Settings; use common::assets::Asset; use conrod_core::{ color, @@ -113,6 +114,11 @@ pub struct LoginInfo { pub server: String, } +enum Info { + Disclaimer, + Intro, +} + enum Screen { Login { screen: login::Screen, @@ -121,6 +127,7 @@ enum Screen { screen: connecting::Screen, // TODO: why instant? start: std::time::Instant, + status_text: String, }, } @@ -129,12 +136,16 @@ struct IcedState { imgs: IcedImgs, bg_img: widget::image::Handle, i18n: std::sync::Arc, + // Voxygen version + version: String, + + login_info: LoginInfo, // TODO: not sure if this should be used for connecting popup: Option, show_servers: bool, - show_disclaimer: bool, - login_info: LoginInfo, + info: Info, + time: f32, screen: Screen, } @@ -150,6 +161,7 @@ enum Message { Password(String), Server(String), FocusPassword, + CancelConnect, } impl IcedState { @@ -157,32 +169,68 @@ impl IcedState { imgs: IcedImgs, bg_img: widget::image::Handle, i18n: std::sync::Arc, + settings: &Settings, ) -> Self { + let version = format!( + "{}-{}", + env!("CARGO_PKG_VERSION"), + common::util::GIT_VERSION.to_string() + ); + + let info = if settings.show_disclaimer { + Info::Disclaimer + } else { + Info::Intro + }; + Self { imgs, bg_img, i18n, - popup: None, - show_servers: false, - show_disclaimer: false, + version, + login_info: LoginInfo { username: String::new(), password: String::new(), server: String::new(), }, + popup: None, + show_servers: false, + info, + time: 0.0, + screen: Screen::Login { screen: login::Screen::new(), }, } } - fn view(&mut self) -> Element { + fn view(&mut self, dt: f32) -> Element { + self.time = self.time + dt; + match &mut self.screen { - Screen::Login { screen } => screen.view(&self.imgs, &self.login_info, &self.i18n), - Screen::Connecting { screen, start } => { - screen.view(&self.imgs, self.bg_img, &start, &self.i18n) - }, + Screen::Login { screen } => screen.view( + &self.imgs, + &self.login_info, + &self.info, + &self.version, + self.show_servers, + &self.i18n, + ), + Screen::Connecting { + screen, + start, + status_text, + } => screen.view( + &self.imgs, + self.bg_img, + &start, + &status_text, + &self.version, + self.time, + &self.i18n, + ), } } @@ -191,16 +239,21 @@ impl IcedState { Message::Quit => events.push(Event::Quit), Message::ShowServers => self.show_servers = true, #[cfg(feature = "singleplayer")] - Message::Singleplayer => events.push(Event::StartSingleplayer), + Message::Singleplayer => { + self.screen = Screen::Connecting { + screen: connecting::Screen::new(), + start: std::time::Instant::now(), + status_text: [self.i18n.get("main.creating_world"), "..."].concat(), + }; + + events.push(Event::StartSingleplayer); + }, Message::Multiplayer => { self.screen = Screen::Connecting { screen: connecting::Screen::new(), start: std::time::Instant::now(), + status_text: [self.i18n.get("main.connecting"), "..."].concat(), }; - self.popup = Some(PopupData { - msg: [self.i18n.get("main.connecting"), "..."].concat(), - popup_type: PopupType::ConnectionInfo, - }); events.push(Event::LoginAttempt { username: self.login_info.username.clone(), @@ -212,11 +265,23 @@ impl IcedState { Message::Password(new_value) => self.login_info.password = new_value, Message::Server(new_value) => self.login_info.server = new_value, Message::FocusPassword => { - if let Screen::Login { screen } = &mut self.screen { + if let Screen::Login { screen, .. } = &mut self.screen { screen.banner.password = text_input::State::focused(); screen.banner.username = text_input::State::new(); } }, + Message::CancelConnect => { + self.cancel_connection(); + events.push(Event::CancelLoginAttempt); + }, + } + } + + fn cancel_connection(&mut self) { + if matches!(&self.screen, Screen::Connecting {..}) { + self.screen = Screen::Login { + screen: login::Screen::new(), + } } } } @@ -300,23 +365,12 @@ image_ids! { } } -rotation_image_ids! { - pub struct ImgsRot { - - - // Tooltip Test - tt_side: "voxygen/element/frames/tt_test_edge", - tt_corner: "voxygen/element/frames/tt_test_corner_tr", - } -} - pub struct MainMenuUi { ui: Ui, ice_ui: IcedUi, ice_state: IcedState, ids: Ids, imgs: Imgs, - rot_imgs: ImgsRot, username: String, password: String, server_address: String, @@ -346,7 +400,6 @@ impl<'a> MainMenuUi { let ids = Ids::new(ui.id_generator()); // Load images let imgs = Imgs::load(&mut ui).expect("Failed to load images"); - let rot_imgs = ImgsRot::load(&mut ui).expect("Failed to load images!"); let bg_img_spec = BG_IMGS.choose(&mut thread_rng()).unwrap(); let bg_img_id = ui.add_graphic(Graphic::Image(DynamicImage::load_expect(bg_img_spec))); // Load language @@ -372,8 +425,9 @@ impl<'a> MainMenuUi { let mut ice_ui = IcedUi::new(window, ice_font).unwrap(); let ice_state = IcedState::new( IcedImgs::load(&mut ice_ui).expect("Failed to load images"), - ice_ui.add_graphic(Graphic::Image(load_expect(bg_img_spec))), + ice_ui.add_graphic(Graphic::Image(DynamicImage::load_expect(bg_img_spec))), i18n.clone(), + &global_state.settings, ); Self { @@ -382,7 +436,6 @@ impl<'a> MainMenuUi { ice_state, ids, imgs, - rot_imgs, username: networking.username.clone(), password: "".to_owned(), server_address: networking @@ -429,24 +482,6 @@ impl<'a> MainMenuUi { let intro_text = &self.i18n.get("main.login_process"); - // Tooltip - /*let _tooltip = Tooltip::new({ - // Edge images [t, b, r, l] - // Corner images [tr, tl, br, bl] - let edge = &self.rot_imgs.tt_side; - let corner = &self.rot_imgs.tt_corner; - ImageFrame::new( - [edge.cw180, edge.none, edge.cw270, edge.cw90], - [corner.none, corner.cw270, corner.cw90, corner.cw180], - Color::Rgba(0.08, 0.07, 0.04, 1.0), - 5.0, - ) - }) - .title_font_size(self.fonts.cyri.scale(15)) - .desc_font_size(self.fonts.cyri.scale(10)) - .font_id(self.fonts.cyri.conrod_id) - .desc_text_color(TEXT_COLOR_2);*/ - // Background image, Veloren logo, Alpha-Version Label Image::new(if self.connect { self.bg_img_id @@ -1047,6 +1082,7 @@ impl<'a> MainMenuUi { self.popup = None; self.connecting = None; self.connect = false; + self.ice_state.cancel_connection(); } pub fn handle_event(&mut self, event: ui::Event) { @@ -1055,17 +1091,24 @@ impl<'a> MainMenuUi { } } - pub fn handle_iced_event(&mut self, event: ui::ice::Event) { self.ice_ui.handle_event(event); } + pub fn handle_iced_event(&mut self, event: ui::ice::Event) { + if self.show_iced { + self.ice_ui.handle_event(event); + } + } pub fn maintain(&mut self, global_state: &mut GlobalState, dt: Duration) -> Vec { let mut events = self.update_layout(global_state, dt); self.ui.maintain(global_state.window.renderer_mut(), None); - let (messages, _) = self - .ice_ui - .maintain(self.ice_state.view(), global_state.window.renderer_mut()); - messages - .into_iter() - .for_each(|message| self.ice_state.update(message, &mut events)); + if self.show_iced { + let (messages, _) = self.ice_ui.maintain( + self.ice_state.view(dt.as_secs_f32()), + global_state.window.renderer_mut(), + ); + messages + .into_iter() + .for_each(|message| self.ice_state.update(message, &mut events)); + } events } diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index a7fdd24645..02774e09a5 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -541,6 +541,7 @@ impl IcedRenderer { // TODO: support nested clips? // TODO: if last command is a clip changing back to the default replace it with // this + // TODO: cull primitives outside the current scissor // Renderer child self.draw_primitive(*content, renderer); diff --git a/voxygen/src/ui/ice/widget/background_container.rs b/voxygen/src/ui/ice/widget/background_container.rs index af94fbdedb..aa8a10583d 100644 --- a/voxygen/src/ui/ice/widget/background_container.rs +++ b/voxygen/src/ui/ice/widget/background_container.rs @@ -1,6 +1,10 @@ use iced::{layout, Clipboard, Element, Event, Hasher, Layout, Length, Point, Size, Widget}; use std::{hash::Hash, u32}; +// TODO: decouple from image/compound graphic widgets (they could still use +// common types, but the info we want here for the background is a subset of +// what a fullblown widget might need) + // Note: it might be more efficient to make this generic over the content type // Note: maybe we could just use the container styling for this (not really with From a6f04e0da5a7c12ff5d48bdf69edc27e1438376e Mon Sep 17 00:00:00 2001 From: Imbris Date: Tue, 2 Jun 2020 00:55:26 -0400 Subject: [PATCH 21/61] Make all the fonts available in iced ui rework --- voxygen/src/menu/main/ui/connecting.rs | 19 +++++++---- voxygen/src/menu/main/ui/login.rs | 33 +++++++++++------- voxygen/src/menu/main/ui/mod.rs | 14 ++++++-- voxygen/src/ui/fonts.rs | 46 ++++++++++++++++++++++++-- voxygen/src/ui/ice/cache.rs | 27 ++++++++++++++- voxygen/src/ui/ice/mod.rs | 10 +++--- voxygen/src/ui/ice/renderer/mod.rs | 4 ++- voxygen/src/ui/mod.rs | 23 +++---------- 8 files changed, 126 insertions(+), 50 deletions(-) diff --git a/voxygen/src/menu/main/ui/connecting.rs b/voxygen/src/menu/main/ui/connecting.rs index 59cee400e1..2c463ca985 100644 --- a/voxygen/src/menu/main/ui/connecting.rs +++ b/voxygen/src/menu/main/ui/connecting.rs @@ -1,10 +1,13 @@ use super::{IcedImgs as Imgs, Message}; use crate::{ i18n::Localization, - ui::ice::{ - component::neat_button, - widget::{image, BackgroundContainer, Image}, - ButtonStyle, Element, + ui::{ + fonts::IcedFonts as Fonts, + ice::{ + component::neat_button, + widget::{image, BackgroundContainer, Image}, + ButtonStyle, Element, + }, }, }; use iced::{button, Color, Column, Container, HorizontalAlignment, Length, Row, Space, Text}; @@ -27,6 +30,7 @@ impl Screen { pub(super) fn view( &mut self, + fonts: &Fonts, imgs: &Imgs, bg_img: image::Handle, start: &std::time::Instant, @@ -43,13 +47,14 @@ impl Screen { .disabled_text_color(DISABLED_TEXT_COLOR); let version = Text::new(version) - .size(15) // move version text size to const + .size(fonts.cyri.scale(15)) // move version text size to const .width(Length::Fill) .height(Length::Fill) .horizontal_alignment(HorizontalAlignment::Right); let status = Text::new(status_text) - .size(80) + .size(fonts.alkhemi.scale(80)) + .font(fonts.alkhemi.id) .color(Color::from_rgba(1.0, 1.0, 1.0, fade_msg)) .width(Length::Fill); @@ -68,7 +73,7 @@ impl Screen { let cancel = Container::new(cancel) .width(Length::Fill) - .height(Length::Units(50)) + .height(Length::Units(fonts.cyri.scale(50))) .center_x() .padding(3); diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index 1d46028638..59725601d5 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -1,13 +1,16 @@ use super::{IcedImgs as Imgs, Info, LoginInfo, Message}; use crate::{ i18n::Localization, - ui::ice::{ - component::neat_button, - widget::{ - compound_graphic::{CompoundGraphic, Graphic}, - BackgroundContainer, Image, Padding, + ui::{ + fonts::IcedFonts as Fonts, + ice::{ + component::neat_button, + widget::{ + compound_graphic::{CompoundGraphic, Graphic}, + BackgroundContainer, Image, Padding, + }, + ButtonStyle, Element, }, - ButtonStyle, Element, }, }; use iced::{ @@ -45,6 +48,7 @@ impl Screen { pub(super) fn view( &mut self, + fonts: &Fonts, imgs: &Imgs, login_info: &LoginInfo, info: &Info, @@ -103,7 +107,7 @@ impl Screen { .color(Rgba::new(255, 255, 255, 240)), ]) .height(Length::Shrink), - Text::new(intro_text).size(21), + Text::new(intro_text).size(fonts.cyri.scale(21)), ) .max_width(450) .padding(Padding::new().horizontal(20).top(10).bottom(60)); @@ -117,7 +121,9 @@ impl Screen { buttons.into() }; - let banner = self.banner.view(imgs, login_info, i18n, button_style); + let banner = self + .banner + .view(fonts, imgs, login_info, i18n, button_style); let central_column = Container::new(banner) .width(Length::Fill) @@ -126,7 +132,7 @@ impl Screen { .align_y(Align::Center); let right_column = Text::new(version) - .size(15) + .size(fonts.cyri.scale(15)) .width(Length::Fill) .horizontal_alignment(HorizontalAlignment::Right); @@ -169,11 +175,14 @@ impl Banner { fn view( &mut self, + fonts: &Fonts, imgs: &Imgs, login_info: &LoginInfo, i18n: &Localization, button_style: ButtonStyle, ) -> Element { + let input_text_size = fonts.cyri.scale(INPUT_TEXT_SIZE); + let banner_content = Column::with_children(vec![ Image::new(imgs.v_logo) .fix_aspect_ratio() @@ -191,7 +200,7 @@ impl Banner { &login_info.username, Message::Username, ) - .size(INPUT_TEXT_SIZE) + .size(input_text_size) .on_submit(Message::FocusPassword), ) .padding(Padding::new().horizontal(10).top(10)) @@ -206,7 +215,7 @@ impl Banner { &login_info.password, Message::Password, ) - .size(INPUT_TEXT_SIZE) + .size(input_text_size) .password() .on_submit(Message::Multiplayer), ) @@ -222,7 +231,7 @@ impl Banner { &login_info.server, Message::Server, ) - .size(INPUT_TEXT_SIZE) + .size(input_text_size) .on_submit(Message::Multiplayer), ) .padding(Padding::new().horizontal(10).top(8)) diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index e6774244b6..96c8fbed0c 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -6,7 +6,7 @@ use crate::{ render::Renderer, ui::{ self, - fonts::Fonts, + fonts::{Fonts, IcedFonts}, ice::{Element, IcedUi}, img_ids::{BlankGraphic, ImageGraphic, VoxelGraphic}, Graphic, Ui, @@ -133,6 +133,7 @@ enum Screen { // TODO: use i18n font scale thing struct IcedState { + fonts: IcedFonts, imgs: IcedImgs, bg_img: widget::image::Handle, i18n: std::sync::Arc, @@ -166,6 +167,7 @@ enum Message { impl IcedState { fn new( + fonts: IcedFonts, imgs: IcedImgs, bg_img: widget::image::Handle, i18n: std::sync::Arc, @@ -184,6 +186,7 @@ impl IcedState { }; Self { + fonts, imgs, bg_img, i18n, @@ -211,6 +214,7 @@ impl IcedState { match &mut self.screen { Screen::Login { screen } => screen.view( + &self.fonts, &self.imgs, &self.login_info, &self.info, @@ -223,6 +227,7 @@ impl IcedState { start, status_text, } => screen.view( + &self.fonts, &self.imgs, self.bg_img, &start, @@ -409,7 +414,7 @@ impl<'a> MainMenuUi { // Load fonts. let fonts = Fonts::load(&i18n.fonts, &mut ui).expect("Impossible to load fonts!"); - // TODO: newtype Font + // TODO: don't add default font twice let ice_font = { use std::io::Read; let mut buf = Vec::new(); @@ -423,7 +428,12 @@ impl<'a> MainMenuUi { }; let mut ice_ui = IcedUi::new(window, ice_font).unwrap(); + + let ice_fonts = + IcedFonts::load(&i18n.fonts, &mut ice_ui).expect("Impossible to load fonts"); + let ice_state = IcedState::new( + ice_fonts, IcedImgs::load(&mut ice_ui).expect("Failed to load images"), ice_ui.add_graphic(Graphic::Image(DynamicImage::load_expect(bg_img_spec))), i18n.clone(), diff --git a/voxygen/src/ui/fonts.rs b/voxygen/src/ui/fonts.rs index 05ba6d3aab..d8afb3cdce 100644 --- a/voxygen/src/ui/fonts.rs +++ b/voxygen/src/ui/fonts.rs @@ -8,11 +8,11 @@ pub struct Font { impl Font { #[allow(clippy::needless_return)] // TODO: Pending review in #587 - pub fn new(font: &i18n::Font, ui: &mut crate::ui::Ui) -> Font { - return Self { + pub fn new(font: &i18n::Font, ui: &mut crate::ui::Ui) -> Self { + Self { metadata: font.clone(), conrod_id: ui.new_font(crate::ui::Font::load_expect(&font.asset_key)), - }; + } } /// Scale input size to final UI size @@ -40,3 +40,43 @@ macro_rules! conrod_fonts { conrod_fonts! { [opensans, metamorph, alkhemi, cyri, wizard] } + +pub struct IcedFont { + metadata: i18n::Font, + pub id: crate::ui::ice::FontId, +} + +impl IcedFont { + pub fn new(font: &i18n::Font, ui: &mut crate::ui::ice::IcedUi) -> Self { + Self { + metadata: font.clone(), + id: ui.add_font((*crate::ui::ice::RawFont::load_expect(&font.asset_key)).clone()), + } + } + + /// Scale input size to final UI size + /// TODO: change metadata to use u16 + pub fn scale(&self, value: u16) -> u16 { self.metadata.scale(value as u32) as u16 } +} + +macro_rules! iced_fonts { + ($([ $( $name:ident$(,)? )* ])*) => { + $( + pub struct IcedFonts { + $(pub $name: IcedFont,)* + } + + impl IcedFonts { + pub fn load(fonts: &i18n::Fonts, ui: &mut crate::ui::ice::IcedUi) -> Result { + Ok(Self { + $( $name: IcedFont::new(fonts.get(stringify!($name)).unwrap(), ui),)* + }) + } + } + )* + }; +} + +iced_fonts! { + [opensans, metamorph, alkhemi, cyri, wizard] +} diff --git a/voxygen/src/ui/ice/cache.rs b/voxygen/src/ui/ice/cache.rs index 8e20b3c4ab..3dfc25f777 100644 --- a/voxygen/src/ui/ice/cache.rs +++ b/voxygen/src/ui/ice/cache.rs @@ -15,8 +15,12 @@ const POSITION_TOLERANCE: f32 = 0.1; type GlyphBrush = glyph_brush::GlyphBrush<(Aabr, Aabr), ()>; +// TODO: might not need pub pub type Font = glyph_brush::ab_glyph::FontArc; +#[derive(Clone, Copy, Default)] +pub struct FontId(pub(super) glyph_brush::FontId); + pub struct Cache { glyph_brush: RefCell, glyph_cache_tex: Texture, @@ -56,7 +60,12 @@ impl Cache { pub fn glyph_calculator(&self) -> RefMut { self.glyph_brush.borrow_mut() } - // TODO: add font fn + // TODO: consider not re-adding default font + pub fn add_font(&mut self, font: RawFont) -> FontId { + let font = Font::try_from_vec(font.0).unwrap(); + let id = self.glyph_brush.get_mut().add_font(font); + FontId(id) + } pub fn graphic_cache(&self) -> &GraphicCache { &self.graphic_cache } @@ -91,3 +100,19 @@ impl Cache { Ok(()) } } + +// TODO: use font type instead of raw vec once we convert to full iced +#[derive(Clone)] +pub struct RawFont(pub Vec); +impl common::assets::Asset for RawFont { + const ENDINGS: &'static [&'static str] = &["ttf"]; + + fn parse( + mut buf_reader: std::io::BufReader, + ) -> Result { + use std::io::Read; + let mut buf = Vec::new(); + buf_reader.read_to_end(&mut buf)?; + Ok(Self(buf)) + } +} diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index b7791e4638..1a45117781 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -6,7 +6,7 @@ mod renderer; pub mod widget; mod winit_conversion; -pub use cache::Font; +pub use cache::{Font, FontId, RawFont}; pub use graphic::{Id, Rotation}; pub use iced::Event; pub use renderer::{ButtonStyle, IcedRenderer}; @@ -23,9 +23,6 @@ use vek::*; pub type Element<'a, M> = iced::Element<'a, M, IcedRenderer>; -#[derive(Clone, Copy, Default)] -pub struct FontId(glyph_brush::FontId); - pub struct IcedUi { renderer: IcedRenderer, cache: Option, @@ -54,7 +51,10 @@ impl IcedUi { }) } - // Add an new graphic that is referencable via the returned Id + /// Add a new font that is referncable via the returned Id + pub fn add_font(&mut self, font: RawFont) -> FontId { self.renderer.add_font(font) } + + /// Add a new graphic that is referencable via the returned Id pub fn add_graphic(&mut self, graphic: Graphic) -> graphic::Id { self.renderer.add_graphic(graphic) } diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index 02774e09a5..a5cf9aa4da 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -12,7 +12,7 @@ use super::{ super::graphic::{self, Graphic, TexId}, cache::Cache, widget::image, - Font, Rotation, + Font, FontId, RawFont, Rotation, }; use crate::{ render::{ @@ -127,6 +127,8 @@ impl IcedRenderer { }) } + pub fn add_font(&mut self, font: RawFont) -> FontId { self.cache.add_font(font) } + pub fn add_graphic(&mut self, graphic: Graphic) -> graphic::Id { self.cache.add_graphic(graphic) } diff --git a/voxygen/src/ui/mod.rs b/voxygen/src/ui/mod.rs index 158bcd31d4..6cf668eeb7 100644 --- a/voxygen/src/ui/mod.rs +++ b/voxygen/src/ui/mod.rs @@ -47,12 +47,7 @@ use conrod_core::{ use core::{convert::TryInto, f32, f64, ops::Range}; use graphic::TexId; use hashbrown::hash_map::Entry; -use std::{ - fs::File, - io::{BufReader, Read}, - sync::Arc, - time::Duration, -}; +use std::{sync::Arc, time::Duration}; use tracing::{error, warn}; use vek::*; @@ -102,17 +97,6 @@ impl DrawCommand { } } -pub struct Font(text::Font); -impl assets::Asset for Font { - const ENDINGS: &'static [&'static str] = &["ttf"]; - - fn parse(mut buf_reader: BufReader, _specifier: &str) -> Result { - let mut buf = Vec::new(); - buf_reader.read_to_end(&mut buf)?; - Ok(Font(text::Font::from_bytes(buf).unwrap())) - } -} - pub struct Ui { pub ui: conrod_core::Ui, image_map: Map<(graphic::Id, Rotation)>, @@ -224,8 +208,9 @@ impl Ui { self.image_map.replace(id, (graphic_id, Rotation::None)); } - pub fn new_font(&mut self, font: Arc) -> font::Id { - self.ui.fonts.insert(font.as_ref().0.clone()) + pub fn new_font(&mut self, font: Arc) -> font::Id { + let font = text::Font::from_bytes(font.0.clone()).unwrap(); + self.ui.fonts.insert(font) } pub fn id_generator(&mut self) -> Generator { self.ui.widget_id_generator() } From f576c38c050c841f5c0da46b7c2e05567ae55d4f Mon Sep 17 00:00:00 2001 From: Imbris Date: Wed, 3 Jun 2020 01:23:51 -0400 Subject: [PATCH 22/61] Add styling for Container, implement auth trust prompt, misc additions --- assets/voxygen/i18n/en.ron | 1 + voxygen/src/menu/main/ui/connecting.rs | 124 +++++++++++++----- voxygen/src/menu/main/ui/login.rs | 12 +- voxygen/src/menu/main/ui/mod.rs | 112 ++++++++++++++-- voxygen/src/ui/ice/component/neat_button.rs | 4 +- voxygen/src/ui/ice/mod.rs | 2 +- voxygen/src/ui/ice/renderer/mod.rs | 3 +- .../src/ui/ice/renderer/style/container.rs | 18 +++ voxygen/src/ui/ice/renderer/style/mod.rs | 5 +- voxygen/src/ui/ice/renderer/widget/button.rs | 4 +- .../src/ui/ice/renderer/widget/container.rs | 36 ++++- 11 files changed, 257 insertions(+), 64 deletions(-) create mode 100644 voxygen/src/ui/ice/renderer/style/container.rs diff --git a/assets/voxygen/i18n/en.ron b/assets/voxygen/i18n/en.ron index 5ef6d90d71..fb12fe4233 100644 --- a/assets/voxygen/i18n/en.ron +++ b/assets/voxygen/i18n/en.ron @@ -65,6 +65,7 @@ "common.back": "Back", "common.create": "Create", "common.okay": "Okay", + "common.add": "Add", "common.accept": "Accept", "common.decline": "Decline", "common.disclaimer": "Disclaimer", diff --git a/voxygen/src/menu/main/ui/connecting.rs b/voxygen/src/menu/main/ui/connecting.rs index 2c463ca985..63f1012565 100644 --- a/voxygen/src/menu/main/ui/connecting.rs +++ b/voxygen/src/menu/main/ui/connecting.rs @@ -1,20 +1,19 @@ -use super::{IcedImgs as Imgs, Message}; +use super::{ConnectionState, IcedImgs as Imgs, Message}; use crate::{ i18n::Localization, ui::{ fonts::IcedFonts as Fonts, - ice::{ - component::neat_button, - widget::{image, BackgroundContainer, Image}, - ButtonStyle, Element, - }, + ice::{component::neat_button, style, widget::image, Element}, }, }; -use iced::{button, Color, Column, Container, HorizontalAlignment, Length, Row, Space, Text}; +use iced::{ + button, Align, Color, Column, Container, HorizontalAlignment, Length, Row, Space, Text, +}; /// Connecting screen for the main menu pub struct Screen { cancel_button: button::State, + add_button: button::State, } // TODO: move to super and unify with identical login consts @@ -25,6 +24,7 @@ impl Screen { pub fn new() -> Self { Self { cancel_button: Default::default(), + add_button: Default::default(), } } @@ -34,13 +34,13 @@ impl Screen { imgs: &Imgs, bg_img: image::Handle, start: &std::time::Instant, - status_text: &str, + connection_state: &ConnectionState, version: &str, time: f32, i18n: &Localization, ) -> Element { let fade_msg = (time * 2.0).sin() * 0.5 + 0.51; - let button_style = ButtonStyle::new(imgs.button) + let button_style = style::button::Style::new(imgs.button) .hover_image(imgs.button_hover) .press_image(imgs.button_press) .text_color(TEXT_COLOR) @@ -49,41 +49,99 @@ impl Screen { let version = Text::new(version) .size(fonts.cyri.scale(15)) // move version text size to const .width(Length::Fill) - .height(Length::Fill) + .height(if matches!(connection_state, ConnectionState::InProgress {..}){Length::Fill}else{Length::Shrink}) .horizontal_alignment(HorizontalAlignment::Right); - let status = Text::new(status_text) - .size(fonts.alkhemi.scale(80)) - .font(fonts.alkhemi.id) - .color(Color::from_rgba(1.0, 1.0, 1.0, fade_msg)) - .width(Length::Fill); + let (middle, bottom) = match connection_state { + ConnectionState::InProgress { status } => { + let status = Text::new(status) + .size(fonts.alkhemi.scale(80)) + .font(fonts.alkhemi.id) + .color(Color::from_rgba(1.0, 1.0, 1.0, fade_msg)) + .width(Length::Fill); - let status = Row::with_children(vec![ - Space::new(Length::Units(80), Length::Shrink).into(), - status.into(), - ]); + let status = Row::with_children(vec![ + Space::new(Length::Units(80), Length::Shrink).into(), + status.into(), + ]); - let cancel = neat_button( - &mut self.cancel_button, - i18n.get("common.cancel"), - 0.7, - button_style, - Some(Message::CancelConnect), - ); + let cancel = neat_button( + &mut self.cancel_button, + i18n.get("common.cancel"), + 0.7, + button_style, + Some(Message::CancelConnect), + ); - let cancel = Container::new(cancel) - .width(Length::Fill) - .height(Length::Units(fonts.cyri.scale(50))) - .center_x() - .padding(3); + let cancel = Container::new(cancel) + .width(Length::Fill) + .height(Length::Units(fonts.cyri.scale(50))) + .center_x() + .padding(3); - let content = Column::with_children(vec![version.into(), status.into(), cancel.into()]) + (status.into(), cancel.into()) + }, + ConnectionState::AuthTrustPrompt { msg, .. } => { + let text = Text::new(msg).size(fonts.cyri.scale(25)); + + let cancel = neat_button( + &mut self.cancel_button, + i18n.get("common.cancel"), + 0.7, + button_style, + Some(Message::TrustPromptCancel), + ); + let add = neat_button( + &mut self.add_button, + i18n.get("common.add"), + 0.7, + button_style, + Some(Message::TrustPromptAdd), + ); + + let content = Column::with_children(vec![ + text.into(), + Container::new( + Row::with_children(vec![cancel.into(), add.into()]) + .spacing(20) + .height(Length::Units(25)), + ) + .align_x(Align::End) + .width(Length::Fill) + .into(), + ]) + .spacing(4) + .max_width(500) + .width(Length::Fill) + .height(Length::Fill); + + let prompt_window = Container::new(content) + // TODO: add borders + .style(style::container::Style::Color((10, 10, 0, 255).into())) + .padding(10); + + let container = Container::new(prompt_window) + .width(Length::Fill) + .height(Length::Fill) + .center_x() + .center_y(); + + ( + container.into(), + Space::new(Length::Fill, Length::Units(fonts.cyri.scale(15))).into(), + ) + }, + }; + + let content = Column::with_children(vec![version.into(), middle, bottom]) .width(Length::Fill) .height(Length::Fill) .padding(3); // Note: could replace this with styling on iced's container since we aren't // using fixed aspect ratio - BackgroundContainer::new(Image::new(bg_img), content).into() + Container::new(content) + .style(style::container::Style::image(bg_img)) + .into() } } diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index 59725601d5..c88ea2c74c 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -5,11 +5,12 @@ use crate::{ fonts::IcedFonts as Fonts, ice::{ component::neat_button, + style, widget::{ compound_graphic::{CompoundGraphic, Graphic}, BackgroundContainer, Image, Padding, }, - ButtonStyle, Element, + Element, }, }, }; @@ -52,11 +53,12 @@ impl Screen { imgs: &Imgs, login_info: &LoginInfo, info: &Info, + error: Option<&str>, version: &str, show_servers: bool, i18n: &Localization, ) -> Element { - let button_style = ButtonStyle::new(imgs.button) + let button_style = style::button::Style::new(imgs.button) .hover_image(imgs.button_hover) .press_image(imgs.button_press) .text_color(TEXT_COLOR) @@ -146,7 +148,9 @@ impl Screen { .spacing(10) .padding(3); - BackgroundContainer::new(Image::new(imgs.bg), content).into() + Container::new(content) + .style(style::container::Style::image(imgs.bg)) + .into() } } @@ -179,7 +183,7 @@ impl Banner { imgs: &Imgs, login_info: &LoginInfo, i18n: &Localization, - button_style: ButtonStyle, + button_style: style::button::Style, ) -> Element { let input_text_size = fonts.cyri.scale(INPUT_TEXT_SIZE); diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 96c8fbed0c..1bb3bef40d 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -119,15 +119,35 @@ enum Info { Intro, } +enum ConnectionState { + InProgress { + status: String, + }, + AuthTrustPrompt { + auth_server: String, + msg: String, + // To remember when we switch back + status: String, + }, +} +impl ConnectionState { + fn take_status_string(&mut self) -> String { + std::mem::take(match self { + Self::InProgress { status } => status, + Self::AuthTrustPrompt { status, .. } => status, + }) + } +} + enum Screen { Login { screen: login::Screen, + // Error to display in a box + error: Option, }, Connecting { screen: connecting::Screen, - // TODO: why instant? - start: std::time::Instant, - status_text: String, + connection_state: ConnectionState, }, } @@ -142,8 +162,6 @@ struct IcedState { login_info: LoginInfo, - // TODO: not sure if this should be used for connecting - popup: Option, show_servers: bool, info: Info, time: f32, @@ -163,6 +181,10 @@ enum Message { Server(String), FocusPassword, CancelConnect, + TrustPromptAdd, + TrustPromptCancel, + CloseError, + CloseDisclaimer, } impl IcedState { @@ -198,13 +220,13 @@ impl IcedState { server: String::new(), }, - popup: None, show_servers: false, info, time: 0.0, screen: Screen::Login { screen: login::Screen::new(), + error: None, }, } } @@ -212,12 +234,14 @@ impl IcedState { fn view(&mut self, dt: f32) -> Element { self.time = self.time + dt; + // TODO: make disclaimer it's own screen? match &mut self.screen { - Screen::Login { screen } => screen.view( + Screen::Login { screen, error } => screen.view( &self.fonts, &self.imgs, &self.login_info, &self.info, + error.as_deref(), &self.version, self.show_servers, &self.i18n, @@ -225,13 +249,13 @@ impl IcedState { Screen::Connecting { screen, start, - status_text, + connection_state, } => screen.view( &self.fonts, &self.imgs, self.bg_img, &start, - &status_text, + &connection_state, &self.version, self.time, &self.i18n, @@ -248,7 +272,9 @@ impl IcedState { self.screen = Screen::Connecting { screen: connecting::Screen::new(), start: std::time::Instant::now(), - status_text: [self.i18n.get("main.creating_world"), "..."].concat(), + connection_state: ConnectionState::InProgress { + status: [self.i18n.get("main.creating_world"), "..."].concat(), + }, }; events.push(Event::StartSingleplayer); @@ -257,7 +283,9 @@ impl IcedState { self.screen = Screen::Connecting { screen: connecting::Screen::new(), start: std::time::Instant::now(), - status_text: [self.i18n.get("main.connecting"), "..."].concat(), + connection_state: ConnectionState::InProgress { + status: [self.i18n.get("main.connecting"), "..."].concat(), + }, }; events.push(Event::LoginAttempt { @@ -279,6 +307,34 @@ impl IcedState { self.cancel_connection(); events.push(Event::CancelLoginAttempt); }, + msg @ Message::TrustPromptAdd | msg @ Message::TrustPromptCancel => { + if let Screen::Connecting { + connection_state, .. + } = &mut self.screen + { + if let ConnectionState::AuthTrustPrompt { + auth_server, + status, + .. + } = connection_state + { + let auth_server = std::mem::take(auth_server); + let status = std::mem::take(status); + let added = matches!(msg, Message::TrustPromptAdd); + + *connection_state = ConnectionState::InProgress { status }; + events.push(Event::AuthServerTrust(auth_server, added)); + } + } + }, + Message::CloseError => { + if let Screen::Login { error, .. } = &mut self.screen { + *error = None; + } + }, + Message::CloseDisclaimer => { + events.push(Event::DisclaimerClosed); + }, } } @@ -286,6 +342,37 @@ impl IcedState { if matches!(&self.screen, Screen::Connecting {..}) { self.screen = Screen::Login { screen: login::Screen::new(), + error: None, + } + } + } + + fn auth_trust_prompt(&mut self, auth_server: String) { + if let Screen::Connecting { + connection_state, .. + } = &mut self.screen + { + let msg = format!( + "Warning: The server you are trying to connect to has provided this \ + authentication server address:\n\n{}\n\nbut it is not in your list of trusted \ + authentication servers.\n\nMake sure that you trust this site and owner to not \ + try and bruteforce your password!", + &auth_server + ); + + *connection_state = ConnectionState::AuthTrustPrompt { + auth_server, + msg, + status: connection_state.take_status_string(), + }; + } + } + + fn connection_error(&mut self, error: String) { + if matches!(&self.screen, Screen::Connecting {..}) { + self.screen = Screen::Login { + screen: login::Screen::new(), + error: Some(error), } } } @@ -1061,6 +1148,7 @@ impl<'a> MainMenuUi { } pub fn auth_trust_prompt(&mut self, auth_server: String) { + self.ice_state.auth_trust_prompt(auth_server.clone()); self.popup = Some(PopupData { msg: format!( "Warning: The server you are trying to connect to has provided this \ @@ -1074,6 +1162,8 @@ impl<'a> MainMenuUi { } pub fn show_info(&mut self, msg: String) { + self.ice_state.connection_error(msg.clone()); + self.popup = Some(PopupData { msg, popup_type: PopupType::Error, diff --git a/voxygen/src/ui/ice/component/neat_button.rs b/voxygen/src/ui/ice/component/neat_button.rs index 6b68988b46..5cc3b53f6e 100644 --- a/voxygen/src/ui/ice/component/neat_button.rs +++ b/voxygen/src/ui/ice/component/neat_button.rs @@ -1,15 +1,15 @@ use crate::ui::ice as ui; use iced::{button::State, Button, Element, Length}; use ui::{ + style::button::Style, widget::{AspectRatioContainer, FillText}, - ButtonStyle, }; pub fn neat_button( state: &mut State, label: impl Into, fill_fraction: f32, - button_style: ButtonStyle, + button_style: Style, message: Option, ) -> Element { let button = Button::new(state, FillText::new(label).fill_fraction(fill_fraction)) diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index 1a45117781..d882a6d9c8 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -9,7 +9,7 @@ mod winit_conversion; pub use cache::{Font, FontId, RawFont}; pub use graphic::{Id, Rotation}; pub use iced::Event; -pub use renderer::{ButtonStyle, IcedRenderer}; +pub use renderer::{style, IcedRenderer}; pub use winit_conversion::window_event; use super::{ diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index a5cf9aa4da..3655dd20c0 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -1,10 +1,9 @@ mod defaults; mod primitive; -mod style; +pub mod style; mod widget; pub use defaults::Defaults; -pub use style::ButtonStyle; pub(self) use primitive::Primitive; diff --git a/voxygen/src/ui/ice/renderer/style/container.rs b/voxygen/src/ui/ice/renderer/style/container.rs new file mode 100644 index 0000000000..c430a271bd --- /dev/null +++ b/voxygen/src/ui/ice/renderer/style/container.rs @@ -0,0 +1,18 @@ +use super::super::super::widget::image; +use vek::Rgba; + +/// Background of the container +pub enum Style { + Image(image::Handle, Rgba), + Color(Rgba), + None, +} + +impl Style { + /// Shorthand for common case where the color of the image is not modified + pub fn image(image: image::Handle) -> Self { Self::Image(image, Rgba::broadcast(255)) } +} + +impl Default for Style { + fn default() -> Self { Self::None } +} diff --git a/voxygen/src/ui/ice/renderer/style/mod.rs b/voxygen/src/ui/ice/renderer/style/mod.rs index 1e5fe43f73..c6f343012b 100644 --- a/voxygen/src/ui/ice/renderer/style/mod.rs +++ b/voxygen/src/ui/ice/renderer/style/mod.rs @@ -1,3 +1,2 @@ -mod button; - -pub use button::Style as ButtonStyle; +pub mod button; +pub mod container; diff --git a/voxygen/src/ui/ice/renderer/widget/button.rs b/voxygen/src/ui/ice/renderer/widget/button.rs index ecba1fb37b..541b1a66c7 100644 --- a/voxygen/src/ui/ice/renderer/widget/button.rs +++ b/voxygen/src/ui/ice/renderer/widget/button.rs @@ -1,10 +1,10 @@ -use super::super::{super::Rotation, Defaults, IcedRenderer, Primitive}; +use super::super::{super::Rotation, style, Defaults, IcedRenderer, Primitive}; use iced::{button, mouse, Element, Layout, Point, Rectangle}; use vek::Rgba; impl button::Renderer for IcedRenderer { // TODO: what if this gets large enough to not be copied around? - type Style = super::super::style::ButtonStyle; + type Style = style::button::Style; const DEFAULT_PADDING: u16 = 0; diff --git a/voxygen/src/ui/ice/renderer/widget/container.rs b/voxygen/src/ui/ice/renderer/widget/container.rs index f1a4a24355..4a30ede9a9 100644 --- a/voxygen/src/ui/ice/renderer/widget/container.rs +++ b/voxygen/src/ui/ice/renderer/widget/container.rs @@ -1,23 +1,47 @@ -use super::super::IcedRenderer; +use super::super::{super::Rotation, style, IcedRenderer, Primitive}; +use common::util::srgba_to_linear; use iced::{container, Element, Layout, Point, Rectangle}; impl container::Renderer for IcedRenderer { - type Style = (); + type Style = style::container::Style; fn draw( &mut self, defaults: &Self::Defaults, - _bounds: Rectangle, + bounds: Rectangle, cursor_position: Point, - _style_sheet: &Self::Style, + style_sheet: &Self::Style, content: &Element<'_, M, Self>, content_layout: Layout<'_>, ) -> Self::Output { let (content, mouse_interaction) = content.draw(self, defaults, content_layout, cursor_position); - // We may have more stuff here if styles are used + let prim = match style_sheet { + Self::Style::Image(handle, color) => { + let background = Primitive::Image { + handle: (*handle, Rotation::None), + bounds, + color: *color, + }; - (content, mouse_interaction) + Primitive::Group { + primitives: vec![background, content], + } + }, + Self::Style::Color(color) => { + let background = Primitive::Rectangle { + bounds, + linear_color: srgba_to_linear(color.map(|e| e as f32 / 255.0)), + }; + + Primitive::Group { + primitives: vec![background, content], + } + }, + Self::Style::None => content, + }; + + (prim, mouse_interaction) } } From 64a5391f4137b043509fd2e63b2f38c66fc5ea0d Mon Sep 17 00:00:00 2001 From: Imbris Date: Wed, 3 Jun 2020 01:48:33 -0400 Subject: [PATCH 23/61] Fill in username & server fields in ui reimplementation, stop saving password in the settings file --- voxygen/src/menu/main/ui/connecting.rs | 1 - voxygen/src/menu/main/ui/mod.rs | 39 ++++++++++---------------- voxygen/src/settings.rs | 6 ++-- 3 files changed, 18 insertions(+), 28 deletions(-) diff --git a/voxygen/src/menu/main/ui/connecting.rs b/voxygen/src/menu/main/ui/connecting.rs index 63f1012565..cdab71b879 100644 --- a/voxygen/src/menu/main/ui/connecting.rs +++ b/voxygen/src/menu/main/ui/connecting.rs @@ -33,7 +33,6 @@ impl Screen { fonts: &Fonts, imgs: &Imgs, bg_img: image::Handle, - start: &std::time::Instant, connection_state: &ConnectionState, version: &str, time: f32, diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 1bb3bef40d..f94a5b1100 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -93,7 +93,7 @@ pub enum Event { StartSingleplayer, Quit, Settings, - //DisclaimerClosed, + //DisclaimerClosed, TODO: remove all traces? AuthServerTrust(String, bool), } @@ -115,7 +115,7 @@ pub struct LoginInfo { } enum Info { - Disclaimer, + //Disclaimer, Intro, } @@ -151,7 +151,6 @@ enum Screen { }, } -// TODO: use i18n font scale thing struct IcedState { fonts: IcedFonts, imgs: IcedImgs, @@ -184,7 +183,7 @@ enum Message { TrustPromptAdd, TrustPromptCancel, CloseError, - CloseDisclaimer, + //CloseDisclaimer, } impl IcedState { @@ -201,11 +200,11 @@ impl IcedState { common::util::GIT_VERSION.to_string() ); - let info = if settings.show_disclaimer { - Info::Disclaimer - } else { - Info::Intro - }; + let info = Info::Intro; // if settings.show_disclaimer { + //Info::Disclaimer + //} else { + //Info::Intro + //}; Self { fonts, @@ -215,9 +214,9 @@ impl IcedState { version, login_info: LoginInfo { - username: String::new(), + username: settings.networking.username.clone(), password: String::new(), - server: String::new(), + server: settings.networking.default_server.clone(), }, show_servers: false, @@ -248,13 +247,11 @@ impl IcedState { ), Screen::Connecting { screen, - start, connection_state, } => screen.view( &self.fonts, &self.imgs, self.bg_img, - &start, &connection_state, &self.version, self.time, @@ -271,7 +268,6 @@ impl IcedState { Message::Singleplayer => { self.screen = Screen::Connecting { screen: connecting::Screen::new(), - start: std::time::Instant::now(), connection_state: ConnectionState::InProgress { status: [self.i18n.get("main.creating_world"), "..."].concat(), }, @@ -282,7 +278,6 @@ impl IcedState { Message::Multiplayer => { self.screen = Screen::Connecting { screen: connecting::Screen::new(), - start: std::time::Instant::now(), connection_state: ConnectionState::InProgress { status: [self.i18n.get("main.connecting"), "..."].concat(), }, @@ -332,9 +327,9 @@ impl IcedState { *error = None; } }, - Message::CloseDisclaimer => { - events.push(Event::DisclaimerClosed); - }, + //Message::CloseDisclaimer => { + // events.push(Event::DisclaimerClosed); + //}, } } @@ -535,11 +530,7 @@ impl<'a> MainMenuUi { imgs, username: networking.username.clone(), password: "".to_owned(), - server_address: networking - .servers - .get(networking.default_server) - .cloned() - .unwrap_or_default(), + server_address: networking.default_server.clone(), popup: None, connecting: None, show_servers: false, @@ -992,7 +983,7 @@ impl<'a> MainMenuUi { .was_clicked() { self.server_address = net_settings.servers[item.i].clone(); - net_settings.default_server = item.i; + net_settings.default_server = self.server_address.clone(); } } diff --git a/voxygen/src/settings.rs b/voxygen/src/settings.rs index b6f23dd828..e74be75932 100644 --- a/voxygen/src/settings.rs +++ b/voxygen/src/settings.rs @@ -558,16 +558,16 @@ impl Default for GameplaySettings { pub struct NetworkingSettings { pub username: String, pub servers: Vec, - pub default_server: usize, + pub default_server: String, pub trusted_auth_servers: HashSet, } impl Default for NetworkingSettings { fn default() -> Self { Self { - username: "Username".to_string(), + username: "".to_string(), servers: vec!["server.veloren.net".to_string()], - default_server: 0, + default_server: "server.veloren.net".to_string(), trusted_auth_servers: ["https://auth.veloren.net"] .iter() .map(|s| s.to_string()) From 1ae4bb696358912d1b13592c106cfef2e59081b4 Mon Sep 17 00:00:00 2001 From: Imbris Date: Fri, 5 Jun 2020 02:21:23 -0400 Subject: [PATCH 24/61] Add Scrollable widget support, implement disclaimer screen, rearrangements of main menu ui code --- voxygen/src/menu/main/mod.rs | 2 +- voxygen/src/menu/main/ui/connecting.rs | 44 ++--- voxygen/src/menu/main/ui/disclaimer.rs | 78 ++++++++ voxygen/src/menu/main/ui/login.rs | 77 +++----- voxygen/src/menu/main/ui/mod.rs | 91 ++++++--- voxygen/src/ui/ice/mod.rs | 2 + voxygen/src/ui/ice/renderer/mod.rs | 59 ++++-- voxygen/src/ui/ice/renderer/primitive.rs | 1 + voxygen/src/ui/ice/renderer/style/mod.rs | 1 + .../src/ui/ice/renderer/style/scrollable.rs | 33 ++++ voxygen/src/ui/ice/renderer/widget/mod.rs | 1 + .../src/ui/ice/renderer/widget/scrollable.rs | 178 ++++++++++++++++++ .../src/ui/ice/renderer/widget/text_input.rs | 14 +- 13 files changed, 450 insertions(+), 131 deletions(-) create mode 100644 voxygen/src/menu/main/ui/disclaimer.rs create mode 100644 voxygen/src/ui/ice/renderer/style/scrollable.rs create mode 100644 voxygen/src/ui/ice/renderer/widget/scrollable.rs diff --git a/voxygen/src/menu/main/mod.rs b/voxygen/src/menu/main/mod.rs index 8674aa8003..87d06381e0 100644 --- a/voxygen/src/menu/main/mod.rs +++ b/voxygen/src/menu/main/mod.rs @@ -255,7 +255,7 @@ impl PlayState for MainMenuState { }, MainMenuEvent::Settings => {}, // TODO MainMenuEvent::Quit => return PlayStateResult::Shutdown, - /*MainMenuEvent::DisclaimerClosed => { + /*MainMenuEvent::DisclaimerAccepted => { global_state.settings.show_disclaimer = false },*/ MainMenuEvent::AuthServerTrust(auth_server, trust) => { diff --git a/voxygen/src/menu/main/ui/connecting.rs b/voxygen/src/menu/main/ui/connecting.rs index cdab71b879..b75080f3e8 100644 --- a/voxygen/src/menu/main/ui/connecting.rs +++ b/voxygen/src/menu/main/ui/connecting.rs @@ -3,12 +3,10 @@ use crate::{ i18n::Localization, ui::{ fonts::IcedFonts as Fonts, - ice::{component::neat_button, style, widget::image, Element}, + ice::{component::neat_button, style, Element}, }, }; -use iced::{ - button, Align, Color, Column, Container, HorizontalAlignment, Length, Row, Space, Text, -}; +use iced::{button, Align, Color, Column, Container, Length, Row, Space, Text, VerticalAlignment}; /// Connecting screen for the main menu pub struct Screen { @@ -16,10 +14,6 @@ pub struct Screen { add_button: button::State, } -// TODO: move to super and unify with identical login consts -const TEXT_COLOR: iced::Color = iced::Color::from_rgb(1.0, 1.0, 1.0); -const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1.0, 0.2); - impl Screen { pub fn new() -> Self { Self { @@ -32,32 +26,22 @@ impl Screen { &mut self, fonts: &Fonts, imgs: &Imgs, - bg_img: image::Handle, connection_state: &ConnectionState, - version: &str, time: f32, i18n: &Localization, + button_style: style::button::Style, ) -> Element { let fade_msg = (time * 2.0).sin() * 0.5 + 0.51; - let button_style = style::button::Style::new(imgs.button) - .hover_image(imgs.button_hover) - .press_image(imgs.button_press) - .text_color(TEXT_COLOR) - .disabled_text_color(DISABLED_TEXT_COLOR); - let version = Text::new(version) - .size(fonts.cyri.scale(15)) // move version text size to const - .width(Length::Fill) - .height(if matches!(connection_state, ConnectionState::InProgress {..}){Length::Fill}else{Length::Shrink}) - .horizontal_alignment(HorizontalAlignment::Right); - - let (middle, bottom) = match connection_state { + let children = match connection_state { ConnectionState::InProgress { status } => { let status = Text::new(status) .size(fonts.alkhemi.scale(80)) .font(fonts.alkhemi.id) .color(Color::from_rgba(1.0, 1.0, 1.0, fade_msg)) - .width(Length::Fill); + .vertical_alignment(VerticalAlignment::Bottom) + .width(Length::Fill) + .height(Length::Fill); let status = Row::with_children(vec![ Space::new(Length::Units(80), Length::Shrink).into(), @@ -78,7 +62,7 @@ impl Screen { .center_x() .padding(3); - (status.into(), cancel.into()) + vec![status.into(), cancel.into()] }, ConnectionState::AuthTrustPrompt { msg, .. } => { let text = Text::new(msg).size(fonts.cyri.scale(25)); @@ -125,22 +109,16 @@ impl Screen { .center_x() .center_y(); - ( + vec![ container.into(), Space::new(Length::Fill, Length::Units(fonts.cyri.scale(15))).into(), - ) + ] }, }; - let content = Column::with_children(vec![version.into(), middle, bottom]) + Column::with_children(children) .width(Length::Fill) .height(Length::Fill) - .padding(3); - - // Note: could replace this with styling on iced's container since we aren't - // using fixed aspect ratio - Container::new(content) - .style(style::container::Style::image(bg_img)) .into() } } diff --git a/voxygen/src/menu/main/ui/disclaimer.rs b/voxygen/src/menu/main/ui/disclaimer.rs new file mode 100644 index 0000000000..ac5401772f --- /dev/null +++ b/voxygen/src/menu/main/ui/disclaimer.rs @@ -0,0 +1,78 @@ +use super::{IcedImgs as Imgs, Message}; +use crate::{ + i18n::Localization, + ui::{ + fonts::IcedFonts as Fonts, + ice::{component::neat_button, style, Element}, + }, +}; +use iced::{button, scrollable, Column, Container, Length, Scrollable, Space}; +use vek::Rgba; + +/// Connecting screen for the main menu +pub struct Screen { + accept_button: button::State, + scroll: scrollable::State, +} + +impl Screen { + pub fn new() -> Self { + Self { + accept_button: Default::default(), + scroll: Default::default(), + } + } + + pub(super) fn view( + &mut self, + fonts: &Fonts, + imgs: &Imgs, + i18n: &Localization, + button_style: style::button::Style, + ) -> Element { + Container::new( + Container::new( + Column::with_children(vec![ + iced::Text::new(i18n.get("common.disclaimer")) + .font(fonts.alkhemi.id) + .size(fonts.alkhemi.scale(35)) + .into(), + Space::new(Length::Fill, Length::Units(20)).into(), + Scrollable::new(&mut self.scroll) + .push( + iced::Text::new(i18n.get("main.notice")) + .font(fonts.cyri.id) + .size(fonts.cyri.scale(23)), + ) + .height(Length::FillPortion(1)) + .into(), + Container::new( + Container::new(neat_button( + &mut self.accept_button, + i18n.get("common.accept"), + 0.7, + button_style, + Some(Message::AcceptDisclaimer), + )) + .height(Length::Units(fonts.cyri.scale(50))), + ) + .center_x() + .height(Length::Shrink) + .width(Length::Fill) + .into(), + ]) + .spacing(5) + .padding(20) + .width(Length::Fill) + .height(Length::Fill), + ) + .style(style::container::Style::Color(Rgba::new(22, 19, 17, 255))), + ) + .center_x() + .center_y() + .padding(70) + .width(Length::Fill) + .height(Length::Fill) + .into() + } +} diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index c88ea2c74c..dc43361f0a 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -1,4 +1,4 @@ -use super::{IcedImgs as Imgs, Info, LoginInfo, Message}; +use super::{IcedImgs as Imgs, LoginInfo, Message}; use crate::{ i18n::Localization, ui::{ @@ -14,14 +14,11 @@ use crate::{ }, }, }; -use iced::{ - button, text_input, Align, Column, Container, HorizontalAlignment, Length, Row, Space, Text, - TextInput, -}; +use iced::{button, text_input, Align, Column, Container, Length, Row, Space, Text, TextInput}; use vek::*; -const TEXT_COLOR: iced::Color = iced::Color::from_rgb(1.0, 1.0, 1.0); -const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1.0, 0.2); +pub const TEXT_COLOR: iced::Color = iced::Color::from_rgb(1.0, 1.0, 1.0); +pub const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1.0, 0.2); const FILL_FRAC_ONE: f32 = 0.77; const FILL_FRAC_TWO: f32 = 0.53; const INPUT_WIDTH: u16 = 250; @@ -52,18 +49,11 @@ impl Screen { fonts: &Fonts, imgs: &Imgs, login_info: &LoginInfo, - info: &Info, error: Option<&str>, - version: &str, show_servers: bool, i18n: &Localization, + button_style: style::button::Style, ) -> Element { - let button_style = style::button::Style::new(imgs.button) - .hover_image(imgs.button_hover) - .press_image(imgs.button_press) - .text_color(TEXT_COLOR) - .disabled_text_color(DISABLED_TEXT_COLOR); - let buttons = Column::with_children(vec![ neat_button( &mut self.servers_button, @@ -96,32 +86,28 @@ impl Screen { .height(Length::Fill) .align_y(Align::End); - let left_column = if matches!(info, Info::Intro) { - let intro_text = i18n.get("main.login_process"); + let intro_text = i18n.get("main.login_process"); - let info_window = BackgroundContainer::new( - CompoundGraphic::from_graphics(vec![ - Graphic::rect(Rgba::new(0, 0, 0, 240), [500, 300], [0, 0]), - // Note: a way to tell it to keep the height of this one piece constant and - // unstreched would be nice, I suppose we could just break this out into a - // column and use Length::Units - Graphic::image(imgs.banner_bottom, [500, 30], [0, 300]) - .color(Rgba::new(255, 255, 255, 240)), - ]) - .height(Length::Shrink), - Text::new(intro_text).size(fonts.cyri.scale(21)), - ) - .max_width(450) - .padding(Padding::new().horizontal(20).top(10).bottom(60)); + let info_window = BackgroundContainer::new( + CompoundGraphic::from_graphics(vec![ + Graphic::rect(Rgba::new(0, 0, 0, 240), [500, 300], [0, 0]), + // Note: a way to tell it to keep the height of this one piece constant and + // unstreched would be nice, I suppose we could just break this out into a + // column and use Length::Units + Graphic::image(imgs.banner_bottom, [500, 30], [0, 300]) + .color(Rgba::new(255, 255, 255, 240)), + ]) + .height(Length::Shrink), + Text::new(intro_text).size(fonts.cyri.scale(21)), + ) + .max_width(450) + .padding(Padding::new().horizontal(20).top(10).bottom(60)); - Column::with_children(vec![info_window.into(), buttons.into()]) - .width(Length::Fill) - .height(Length::Fill) - .padding(27) - .into() - } else { - buttons.into() - }; + let left_column = Column::with_children(vec![info_window.into(), buttons.into()]) + .width(Length::Fill) + .height(Length::Fill) + .padding(27) + .into(); let banner = self .banner @@ -133,12 +119,9 @@ impl Screen { .align_x(Align::Center) .align_y(Align::Center); - let right_column = Text::new(version) - .size(fonts.cyri.scale(15)) - .width(Length::Fill) - .horizontal_alignment(HorizontalAlignment::Right); + let right_column = Space::new(Length::Fill, Length::Fill); - let content = Row::with_children(vec![ + Row::with_children(vec![ left_column, central_column.into(), right_column.into(), @@ -146,11 +129,7 @@ impl Screen { .width(Length::Fill) .height(Length::Fill) .spacing(10) - .padding(3); - - Container::new(content) - .style(style::container::Style::image(imgs.bg)) - .into() + .into() } } diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index f94a5b1100..7811f7f5d3 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -1,4 +1,5 @@ mod connecting; +//mod disclaimer; mod login; use crate::{ @@ -34,7 +35,8 @@ const COL1: Color = Color::Rgba(0.07, 0.1, 0.1, 0.9); /*const UI_MAIN: Color = Color::Rgba(0.61, 0.70, 0.70, 1.0); // Greenish Blue const UI_HIGHLIGHT_0: Color = Color::Rgba(0.79, 1.09, 1.09, 1.0);*/ -use iced::text_input; +use iced::{text_input, Column, Container, HorizontalAlignment, Length}; +use ui::ice::style; image_ids_ice! { struct IcedImgs { @@ -94,6 +96,7 @@ pub enum Event { Quit, Settings, //DisclaimerClosed, TODO: remove all traces? + //DisclaimerAccepted, AuthServerTrust(String, bool), } @@ -114,11 +117,6 @@ pub struct LoginInfo { pub server: String, } -enum Info { - //Disclaimer, - Intro, -} - enum ConnectionState { InProgress { status: String, @@ -140,6 +138,9 @@ impl ConnectionState { } enum Screen { + /*Disclaimer { + screen: disclaimer::Screen, + },*/ Login { screen: login::Screen, // Error to display in a box @@ -162,7 +163,6 @@ struct IcedState { login_info: LoginInfo, show_servers: bool, - info: Info, time: f32, screen: Screen, @@ -183,7 +183,8 @@ enum Message { TrustPromptAdd, TrustPromptCancel, CloseError, - //CloseDisclaimer, + /*CloseDisclaimer, + *AcceptDisclaimer, */ } impl IcedState { @@ -200,10 +201,15 @@ impl IcedState { common::util::GIT_VERSION.to_string() ); - let info = Info::Intro; // if settings.show_disclaimer { - //Info::Disclaimer - //} else { - //Info::Intro + let screen = /* if settings.show_disclaimer { + Screen::Disclaimer { + screen: disclaimer::Screen::new(), + } + } else { */ + Screen::Login { + screen: login::Screen::new(), + error: None, + }; //}; Self { @@ -220,30 +226,47 @@ impl IcedState { }, show_servers: false, - info, time: 0.0, - screen: Screen::Login { - screen: login::Screen::new(), - error: None, - }, + screen, } } fn view(&mut self, dt: f32) -> Element { self.time = self.time + dt; - // TODO: make disclaimer it's own screen? - match &mut self.screen { + // TODO: consider setting this as the default in the renderer + let button_style = style::button::Style::new(self.imgs.button) + .hover_image(self.imgs.button_hover) + .press_image(self.imgs.button_press) + .text_color(login::TEXT_COLOR) + .disabled_text_color(login::DISABLED_TEXT_COLOR); + + let version = iced::Text::new(&self.version) + .size(self.fonts.cyri.scale(15)) + .width(Length::Fill) + .horizontal_alignment(HorizontalAlignment::Right); + + let bg_img = if matches!(&self.screen, Screen::Connecting {..}) { + self.bg_img + } else { + self.imgs.bg + }; + + // TODO: make any large text blocks scrollable so that if the area is to + // small they can still be read + let content = match &mut self.screen { + /*Screen::Disclaimer { screen } => { + screen.view(&self.fonts, &self.imgs, &self.i18n, button_style) + },*/ Screen::Login { screen, error } => screen.view( &self.fonts, &self.imgs, &self.login_info, - &self.info, error.as_deref(), - &self.version, self.show_servers, &self.i18n, + button_style, ), Screen::Connecting { screen, @@ -251,13 +274,22 @@ impl IcedState { } => screen.view( &self.fonts, &self.imgs, - self.bg_img, &connection_state, - &self.version, self.time, &self.i18n, + button_style, ), - } + }; + + Container::new( + Column::with_children(vec![version.into(), content]) + .spacing(3) + .width(Length::Fill) + .height(Length::Fill), + ) + .style(style::container::Style::image(bg_img)) + .padding(3) + .into() } fn update(&mut self, message: Message, events: &mut Vec) { @@ -330,6 +362,15 @@ impl IcedState { //Message::CloseDisclaimer => { // events.push(Event::DisclaimerClosed); //}, + /*Message::AcceptDisclaimer => { + if let Screen::Disclaimer { .. } = &self.screen { + events.push(Event::DisclaimerAccepted); + self.screen = Screen::Login { + screen: login::Screen::new(), + error: None, + }; + } + },*/ } } @@ -816,7 +857,7 @@ impl<'a> MainMenuUi { .was_clicked() { self.show_disclaimer = false; - events.push(Event::DisclaimerClosed); + events.push(Event::DisclaimerAccepted); } } else {*/ // TODO: Don't use macros for this? diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index d882a6d9c8..89ab43cc37 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -63,6 +63,8 @@ impl IcedUi { use iced::window; match event { // Intercept resizing events + // TODO: examine if we are handling dpi properly here + // ideally these values should be the logical ones Event::Window(window::Event::Resized { width, height }) => { self.window_resized = Some(Vec2::new(width, height)); }, diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index 3655dd20c0..da6568e9e2 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -86,7 +86,7 @@ pub struct IcedRenderer { // Per-frame/update current_state: State, mesh: Mesh, - glyphs: Vec<(usize, usize, Rgba)>, + glyphs: Vec<(usize, usize, Rgba, Vec2)>, // Output from glyph_brush in the previous frame // It can sometimes ask you to redraw with these instead (idk if that is done with // pre-positioned glyphs) @@ -164,7 +164,7 @@ impl IcedRenderer { //self.current_scissor = default_scissor(renderer); - self.draw_primitive(primitive, renderer); + self.draw_primitive(primitive, Vec2::zero(), renderer); // Enter the final command. self.draw_commands.push(match self.current_state { @@ -240,19 +240,29 @@ impl IcedRenderer { let glyphs = &self.glyphs; let mesh = &mut self.mesh; + let p_scale = self.p_scale; + let half_res = self.half_res; glyphs .iter() - .flat_map(|(mesh_index, glyph_count, linear_color)| { + .flat_map(|(mesh_index, glyph_count, linear_color, offset)| { let mesh_index = *mesh_index; let linear_color = *linear_color; - (0..*glyph_count).map(move |i| (mesh_index + i * 6, linear_color)) + // Could potentially pass this in as part of the extras + let offset = offset.map(|e| e as f32 * p_scale) / half_res; + (0..*glyph_count).map(move |i| (mesh_index + i * 6, linear_color, offset)) }) .zip(self.last_glyph_verts.iter()) - .for_each(|((mesh_index, linear_color), (uv, rect))| { + .for_each(|((mesh_index, linear_color, offset), (uv, rect))| { + // TODO: add function to vek for this + let rect = Aabr { + min: rect.min + offset, + max: rect.max + offset, + }; + mesh.replace_quad( mesh_index, - create_ui_quad(*rect, *uv, linear_color, UiMode::Text), + create_ui_quad(rect, *uv, linear_color, UiMode::Text), ) }); }, @@ -344,12 +354,12 @@ impl IcedRenderer { Aabr { min, max } } - fn draw_primitive(&mut self, primitive: Primitive, renderer: &mut Renderer) { + fn draw_primitive(&mut self, primitive: Primitive, offset: Vec2, renderer: &mut Renderer) { match primitive { Primitive::Group { primitives } => { primitives .into_iter() - .for_each(|p| self.draw_primitive(p, renderer)); + .for_each(|p| self.draw_primitive(p, offset, renderer)); }, Primitive::Image { handle, @@ -363,7 +373,11 @@ impl IcedRenderer { } let (graphic_id, rotation) = handle; - let gl_aabr = self.gl_aabr(bounds); + let gl_aabr = self.gl_aabr(iced::Rectangle { + x: bounds.x + offset.x as f32, + y: bounds.y + offset.y as f32, + ..bounds + }); let graphic_cache = self.cache.graphic_cache_mut(); @@ -435,8 +449,14 @@ impl IcedRenderer { self.switch_state(State::Plain); + let gl_aabr = self.gl_aabr(iced::Rectangle { + x: bounds.x + offset.x as f32, + y: bounds.y + offset.y as f32, + ..bounds + }); + self.mesh.push_quad(create_ui_quad( - self.gl_aabr(bounds), + gl_aabr, Aabr { min: Vec2::zero(), max: Vec2::zero(), @@ -470,7 +490,7 @@ impl IcedRenderer { // Since we already passed in `bounds` to position the glyphs some of this // seems redundant... // Note: we can't actually use this because dropping glyphs messeses up the - // counting and there is not a convenient method provided to drop out of bounds + // counting and there is not a method provided to drop out of bounds // glyphs while positioning them glyph_brush::ab_glyph::Rect { min: glyph_brush::ab_glyph::point( @@ -489,8 +509,12 @@ impl IcedRenderer { min: Vec2::broadcast(0.0), max: Vec2::broadcast(0.0), }; - self.glyphs - .push((self.mesh.vertices().len(), glyph_count, linear_color)); + self.glyphs.push(( + self.mesh.vertices().len(), + glyph_count, + linear_color, + offset, + )); for _ in 0..glyph_count { // Push placeholder quad // Note: moving to some sort of layering / z based system would be an @@ -504,8 +528,13 @@ impl IcedRenderer { )); } }, - Primitive::Clip { bounds, content } => { + Primitive::Clip { + bounds, + offset: clip_offset, + content, + } => { let new_scissor = { + // TODO: incorporate current offset for nested Clips let intersection = Aabr { min: Vec2 { x: (bounds.x * self.p_scale) as u16, @@ -545,7 +574,7 @@ impl IcedRenderer { // TODO: cull primitives outside the current scissor // Renderer child - self.draw_primitive(*content, renderer); + self.draw_primitive(*content, offset + clip_offset, renderer); // Reset scissor self.draw_commands.push(match self.current_state { diff --git a/voxygen/src/ui/ice/renderer/primitive.rs b/voxygen/src/ui/ice/renderer/primitive.rs index 46734998e1..65125f22a6 100644 --- a/voxygen/src/ui/ice/renderer/primitive.rs +++ b/voxygen/src/ui/ice/renderer/primitive.rs @@ -25,6 +25,7 @@ pub enum Primitive { }, Clip { bounds: iced::Rectangle, + offset: vek::Vec2, content: Box, }, Nothing, diff --git a/voxygen/src/ui/ice/renderer/style/mod.rs b/voxygen/src/ui/ice/renderer/style/mod.rs index c6f343012b..370210eaaa 100644 --- a/voxygen/src/ui/ice/renderer/style/mod.rs +++ b/voxygen/src/ui/ice/renderer/style/mod.rs @@ -1,2 +1,3 @@ pub mod button; pub mod container; +pub mod scrollable; diff --git a/voxygen/src/ui/ice/renderer/style/scrollable.rs b/voxygen/src/ui/ice/renderer/style/scrollable.rs new file mode 100644 index 0000000000..7b43d6c65f --- /dev/null +++ b/voxygen/src/ui/ice/renderer/style/scrollable.rs @@ -0,0 +1,33 @@ +use super::super::super::widget::image; +use vek::Rgba; + +#[derive(Clone, Copy)] +pub struct Style { + pub track: Option, + pub scroller: Scroller, +} + +impl Default for Style { + fn default() -> Self { + Self { + track: None, + scroller: Scroller::Color(Rgba::new(128, 128, 128, 255)), + } + } +} + +#[derive(Clone, Copy)] +pub enum Track { + Color(Rgba), + Image(image::Handle, Rgba), +} + +#[derive(Clone, Copy)] +pub enum Scroller { + Color(Rgba), + Image { + ends: image::Handle, + mid: image::Handle, + color: Rgba, + }, +} diff --git a/voxygen/src/ui/ice/renderer/widget/mod.rs b/voxygen/src/ui/ice/renderer/widget/mod.rs index f60c6eb0c5..6fe01dac05 100644 --- a/voxygen/src/ui/ice/renderer/widget/mod.rs +++ b/voxygen/src/ui/ice/renderer/widget/mod.rs @@ -6,6 +6,7 @@ mod compound_graphic; mod container; mod image; mod row; +mod scrollable; mod space; mod text; mod text_input; diff --git a/voxygen/src/ui/ice/renderer/widget/scrollable.rs b/voxygen/src/ui/ice/renderer/widget/scrollable.rs new file mode 100644 index 0000000000..014ccd12e3 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/scrollable.rs @@ -0,0 +1,178 @@ +use super::super::{super::Rotation, style, IcedRenderer, Primitive}; +use common::util::srgba_to_linear; +use iced::{mouse, scrollable, Rectangle}; +use style::scrollable::{Scroller, Track}; + +const SCROLLBAR_WIDTH: u16 = 10; +const SCROLLBAR_MIN_HEIGHT: u16 = 6; +const SCROLLBAR_MARGIN: u16 = 2; + +impl scrollable::Renderer for IcedRenderer { + type Style = style::scrollable::Style; + + // Interesting that this is here + // I guess we can take advantage of this to keep a constant size despite + // scaling? + fn scrollbar( + &self, + bounds: Rectangle, + content_bounds: Rectangle, + offset: u32, + ) -> Option { + // TODO: might actually want to divide by p_scale here (same in text&ext_input) + // (or just not use it) (or at least only account for dpi but not any + // additional scaling) + let width = (SCROLLBAR_WIDTH + 2 * SCROLLBAR_MARGIN) as f32 * self.p_scale; + if content_bounds.height > bounds.height { + let scrollbar_bounds = Rectangle { + x: bounds.x + bounds.width - width, + width, + ..bounds + }; + + let visible_fraction = bounds.height / content_bounds.height; + let scrollbar_height = (bounds.height * visible_fraction) + .max((2 * SCROLLBAR_MIN_HEIGHT) as f32 * self.p_scale); + let y_offset = offset as f32 * visible_fraction; + + let scroller_bounds = Rectangle { + x: scrollbar_bounds.x + SCROLLBAR_MARGIN as f32 * self.p_scale, + // TODO: check this behavior + y: scrollbar_bounds.y + y_offset, + width: scrollbar_bounds.width - (2 * SCROLLBAR_MARGIN) as f32 * self.p_scale, + height: scrollbar_height, + }; + Some(scrollable::Scrollbar { + bounds: scrollbar_bounds, + scroller: scrollable::Scroller { + bounds: scroller_bounds, + }, + }) + } else { + None + } + } + + fn draw( + &mut self, + state: &scrollable::State, + bounds: Rectangle, + _content_bounds: Rectangle, + is_mouse_over: bool, + is_mouse_over_scrollbar: bool, + scrollbar: Option, + offset: u32, + style_sheet: &Self::Style, + (content, mouse_interaction): Self::Output, + ) -> Self::Output { + ( + if let Some(scrollbar) = scrollbar { + let mut primitives = Vec::with_capacity(5); + + // Scrolled content + primitives.push(Primitive::Clip { + bounds, + offset: (0, offset).into(), + content: Box::new(content), + }); + + let style = style_sheet; + //let style = if state.is_scroller_grabbed() { + // style_sheet.dragging() + //} else if is_mouse_over_scrollbar { + // style_sheet.hovered() + //} else { + // style_sheet.active(); + //}; + + let is_scrollbar_visible = style.track.is_some(); + + if is_mouse_over || state.is_scroller_grabbed() || is_scrollbar_visible { + let bounds = scrollbar.scroller.bounds; + + match style.scroller { + Scroller::Color(color) => primitives.push(Primitive::Rectangle { + bounds, + linear_color: srgba_to_linear(color.map(|e| e as f32 / 255.0)), + }), + Scroller::Image { ends, mid, color } => { + // Calculate sizes of ends pieces based on image aspect ratio + let (img_w, img_h) = self.image_dims(ends); + let end_height = bounds.width * img_h as f32 / img_w as f32; + + // Calcutate size of middle piece based on available space + // Note: Might want to scale into real pixels for parts of this + let (end_height, middle_height) = + if end_height * 2.0 + 1.0 <= bounds.height { + (end_height, bounds.height - end_height * 2.0) + } else { + // Take 1 logical pixel for the middle height + let remaining_height = bounds.height - 1.0; + (remaining_height / 2.0, 1.0) + }; + + // Top + primitives.push(Primitive::Image { + handle: (ends, Rotation::None), + bounds: Rectangle { + height: end_height, + ..bounds + }, + color, + }); + // Middle + primitives.push(Primitive::Image { + handle: (mid, Rotation::None), + bounds: Rectangle { + y: bounds.y + end_height, + height: middle_height, + ..bounds + }, + color, + }); + // Bottom + primitives.push(Primitive::Image { + handle: (ends, Rotation::Cw180), + bounds: Rectangle { + y: bounds.y + end_height + middle_height, + height: end_height, + ..bounds + }, + color, + }); + }, + } + } + + if let Some(track) = style.track { + let bounds = Rectangle { + x: scrollbar.bounds.x + SCROLLBAR_MARGIN as f32 * self.p_scale, + width: scrollbar.bounds.width + - (2 * SCROLLBAR_MARGIN) as f32 * self.p_scale, + ..scrollbar.bounds + }; + primitives.push(match track { + Track::Color(color) => Primitive::Rectangle { + bounds, + linear_color: srgba_to_linear(color.map(|e| e as f32 / 255.0)), + }, + Track::Image(handle, color) => Primitive::Image { + handle: (handle, Rotation::None), + bounds, + color, + }, + }); + } + + Primitive::Group { primitives } + } else { + content + }, + if is_mouse_over_scrollbar || state.is_scroller_grabbed() { + mouse::Interaction::Idle + } else { + mouse_interaction + }, + ) + } +} diff --git a/voxygen/src/ui/ice/renderer/widget/text_input.rs b/voxygen/src/ui/ice/renderer/widget/text_input.rs index f267f7f451..802874c925 100644 --- a/voxygen/src/ui/ice/renderer/widget/text_input.rs +++ b/voxygen/src/ui/ice/renderer/widget/text_input.rs @@ -169,7 +169,7 @@ impl text_input::Renderer for IcedRenderer { ( Primitive::Rectangle { bounds: Rectangle { - x: text_bounds.x + position - offset - CURSOR_WIDTH / p_scale / 2.0, + x: text_bounds.x + position - CURSOR_WIDTH / p_scale / 2.0, y: text_bounds.y, width: CURSOR_WIDTH / p_scale, height: text_bounds.height, @@ -196,7 +196,7 @@ impl text_input::Renderer for IcedRenderer { ( Primitive::Rectangle { bounds: Rectangle { - x: text_bounds.x + left_position - left_offset, + x: text_bounds.x + left_position, y: text_bounds.y, width, height: text_bounds.height, @@ -216,10 +216,9 @@ impl text_input::Renderer for IcedRenderer { let display_text = text.unwrap_or(if state.is_focused() { "" } else { placeholder }); let section = glyph_brush::Section { - screen_position: ( - text_bounds.x * p_scale - scroll_offset, - text_bounds.center_y() * p_scale, - ), + // Note: clip offset is an integer so we don't have to worry about not accounting for + // that here where the alignment of the glyphs with pixels affects rasterization + screen_position: (text_bounds.x * p_scale, text_bounds.center_y() * p_scale), bounds: ( 10000.0, /* text_bounds.width * p_scale */ text_bounds.height * p_scale, @@ -279,9 +278,8 @@ impl text_input::Renderer for IcedRenderer { let primitive = if text_width > text_bounds.width { Primitive::Clip { bounds: text_bounds, + offset: (scroll_offset as u32, 0).into(), content: Box::new(primitive), - /* Note: iced_wgpu uses offset here but we can't do that since we pass the text - * to the glyph_brush here */ } } else { primitive From 39a2fe13153ec945cd12061b1675b88fd13baee8 Mon Sep 17 00:00:00 2001 From: Yusuf Bera Ertan Date: Mon, 8 Jun 2020 00:26:21 +0300 Subject: [PATCH 25/61] Implemented server selection screen --- voxygen/src/menu/main/ui/login.rs | 1 - voxygen/src/menu/main/ui/mod.rs | 56 ++++++++++++++--- voxygen/src/menu/main/ui/servers.rs | 98 +++++++++++++++++++++++++++++ 3 files changed, 144 insertions(+), 11 deletions(-) create mode 100644 voxygen/src/menu/main/ui/servers.rs diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index dc43361f0a..9a58989621 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -50,7 +50,6 @@ impl Screen { imgs: &Imgs, login_info: &LoginInfo, error: Option<&str>, - show_servers: bool, i18n: &Localization, button_style: style::button::Style, ) -> Element { diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 7811f7f5d3..b218f0a89b 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -1,6 +1,7 @@ mod connecting; //mod disclaimer; mod login; +mod servers; use crate::{ i18n::{i18n_asset_key, Localization}, @@ -146,6 +147,9 @@ enum Screen { // Error to display in a box error: Option, }, + Servers { + screen: servers::Screen, + }, Connecting { screen: connecting::Screen, connection_state: ConnectionState, @@ -160,9 +164,10 @@ struct IcedState { // Voxygen version version: String, + servers: Vec, + selected_server_index: Option, login_info: LoginInfo, - show_servers: bool, time: f32, screen: Screen, @@ -171,6 +176,7 @@ struct IcedState { #[derive(Clone)] enum Message { Quit, + Back, ShowServers, #[cfg(feature = "singleplayer")] Singleplayer, @@ -178,6 +184,7 @@ enum Message { Username(String), Password(String), Server(String), + ServerChanged(usize), FocusPassword, CancelConnect, TrustPromptAdd, @@ -212,6 +219,14 @@ impl IcedState { }; //}; + let servers = settings.networking.servers.clone(); + let login_info = LoginInfo { + username: settings.networking.username.clone(), + password: String::new(), + server: settings.networking.default_server.clone(), + }; + let selected_server_index = servers.iter().position(|f| f == &login_info.server); + Self { fonts, imgs, @@ -219,13 +234,10 @@ impl IcedState { i18n, version, - login_info: LoginInfo { - username: settings.networking.username.clone(), - password: String::new(), - server: settings.networking.default_server.clone(), - }, + servers, + selected_server_index, + login_info, - show_servers: false, time: 0.0, screen, @@ -264,7 +276,14 @@ impl IcedState { &self.imgs, &self.login_info, error.as_deref(), - self.show_servers, + &self.i18n, + button_style, + ), + Screen::Servers { screen } => screen.view( + &self.fonts, + &self.imgs, + &self.servers, + self.selected_server_index, &self.i18n, button_style, ), @@ -295,7 +314,17 @@ impl IcedState { fn update(&mut self, message: Message, events: &mut Vec) { match message { Message::Quit => events.push(Event::Quit), - Message::ShowServers => self.show_servers = true, + Message::Back => { + self.screen = Screen::Login { + screen: login::Screen::new(), + error: None, + }; + }, + Message::ShowServers => { + self.screen = Screen::Servers { + screen: servers::Screen::new(), + }; + }, #[cfg(feature = "singleplayer")] Message::Singleplayer => { self.screen = Screen::Connecting { @@ -323,7 +352,14 @@ impl IcedState { }, Message::Username(new_value) => self.login_info.username = new_value, Message::Password(new_value) => self.login_info.password = new_value, - Message::Server(new_value) => self.login_info.server = new_value, + Message::Server(new_value) => { + self.selected_server_index = self.servers.iter().position(|f| f == &new_value); + self.login_info.server = new_value; + }, + Message::ServerChanged(new_value) => { + self.login_info.server = self.servers[new_value].clone(); + self.selected_server_index = Some(new_value); + }, Message::FocusPassword => { if let Screen::Login { screen, .. } = &mut self.screen { screen.banner.password = text_input::State::focused(); diff --git a/voxygen/src/menu/main/ui/servers.rs b/voxygen/src/menu/main/ui/servers.rs new file mode 100644 index 0000000000..817596a003 --- /dev/null +++ b/voxygen/src/menu/main/ui/servers.rs @@ -0,0 +1,98 @@ +use super::{IcedImgs as Imgs, Message}; +use crate::{ + i18n::Localization, + ui::{ + fonts::IcedFonts as Fonts, + ice::{ + component::neat_button, + style, + widget::{ + background_container::Padding, + compound_graphic::{CompoundGraphic, Graphic}, + BackgroundContainer, + }, + Element, + }, + }, +}; +use iced::{button, scrollable, Align, Button, Column, Container, Length, Scrollable, Text}; + +pub struct Screen { + back_button: button::State, + server_buttons: Vec, + servers_list: scrollable::State, +} + +impl Screen { + pub fn new() -> Self { + Self { + back_button: Default::default(), + server_buttons: vec![], + servers_list: Default::default(), + } + } + + pub(super) fn view( + &mut self, + fonts: &Fonts, + imgs: &Imgs, + servers: &Vec, + selected_server_index: Option, + i18n: &Localization, + button_style: style::button::Style, + ) -> Element { + let button = neat_button( + &mut self.back_button, + i18n.get("common.back"), + 0.77_f32, + button_style, + Some(Message::Back), + ); + + let button = Container::new(Container::new(button).max_width(200)) + .width(Length::Fill) + .align_x(Align::Center); + + let mut list = Scrollable::new(&mut self.servers_list) + .align_items(Align::Start) + .width(Length::Fill) + .height(Length::Fill) + .spacing(10); + + if self.server_buttons.len() != servers.len() { + self.server_buttons = vec![Default::default(); servers.len()]; + } + + for (i, state) in self.server_buttons.iter_mut().enumerate() { + let server = servers.get(i).unwrap(); + let text = format!( + "{}{}", + if i == selected_server_index.unwrap_or(std::usize::MAX) { + "-> " + } else { + " " + }, + server + ); + let button = Button::new(state, Text::new(text).size(fonts.cyri.scale(25))) + .on_press(Message::ServerChanged(i)); + list = list.push(button); + } + + Container::new( + BackgroundContainer::new( + CompoundGraphic::padded_image(imgs.info_frame, [500, 300], [0; 4]), + Column::with_children(vec![list.into(), button.into()]) + .width(Length::Fill) + .height(Length::Fill) + .spacing(10) + .padding(20), + ) + .max_width(500), + ) + .width(Length::Fill) + .align_x(Align::Center) + .padding(80) + .into() + } +} From cc76b8ef8f951ab03271f38f96f7357f48c03ce7 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sat, 6 Jun 2020 00:37:29 -0400 Subject: [PATCH 26/61] Make cancel button in connecting screen visble again, don't save username & server for singleplayer login, make iced version go back to login screen properly. --- voxygen/src/menu/main/mod.rs | 18 +++++++++--------- voxygen/src/menu/main/ui/connecting.rs | 12 ++++++------ voxygen/src/menu/main/ui/mod.rs | 8 +++++--- 3 files changed, 20 insertions(+), 18 deletions(-) diff --git a/voxygen/src/menu/main/mod.rs b/voxygen/src/menu/main/mod.rs index 87d06381e0..f9d81b3244 100644 --- a/voxygen/src/menu/main/mod.rs +++ b/voxygen/src/menu/main/mod.rs @@ -226,6 +226,15 @@ impl PlayState for MainMenuState { password, server_address, } => { + let mut net_settings = &mut global_state.settings.networking; + net_settings.username = username.clone(); + if !net_settings.servers.contains(&server_address) { + net_settings.servers.push(server_address.clone()); + } + if let Err(err) = global_state.settings.save_to_file() { + warn!("Failed to save settings: {:?}", err); + } + attempt_login( &mut global_state.settings, &mut global_state.info_message, @@ -298,15 +307,6 @@ fn attempt_login( server_port: u16, client_init: &mut Option, ) { - let mut net_settings = &mut settings.networking; - net_settings.username = username.clone(); - if !net_settings.servers.contains(&server_address) { - net_settings.servers.push(server_address.clone()); - } - if let Err(e) = settings.save_to_file() { - warn!(?e, "Failed to save settings"); - } - if comp::Player::alias_is_valid(&username) { // Don't try to connect if there is already a connection in progress. if client_init.is_none() { diff --git a/voxygen/src/menu/main/ui/connecting.rs b/voxygen/src/menu/main/ui/connecting.rs index b75080f3e8..36647f7d9c 100644 --- a/voxygen/src/menu/main/ui/connecting.rs +++ b/voxygen/src/menu/main/ui/connecting.rs @@ -38,15 +38,15 @@ impl Screen { let status = Text::new(status) .size(fonts.alkhemi.scale(80)) .font(fonts.alkhemi.id) - .color(Color::from_rgba(1.0, 1.0, 1.0, fade_msg)) - .vertical_alignment(VerticalAlignment::Bottom) - .width(Length::Fill) - .height(Length::Fill); + .color(Color::from_rgba(1.0, 1.0, 1.0, fade_msg)); - let status = Row::with_children(vec![ + let status = Container::new(Row::with_children(vec![ Space::new(Length::Units(80), Length::Shrink).into(), status.into(), - ]); + ])) + .width(Length::Fill) + .height(Length::Fill) + .align_y(Align::End); let cancel = neat_button( &mut self.cancel_button, diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index b218f0a89b..b9dd5171d1 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -367,7 +367,7 @@ impl IcedState { } }, Message::CancelConnect => { - self.cancel_connection(); + self.exit_connect_screen(); events.push(Event::CancelLoginAttempt); }, msg @ Message::TrustPromptAdd | msg @ Message::TrustPromptCancel => { @@ -410,7 +410,8 @@ impl IcedState { } } - fn cancel_connection(&mut self) { + // Connection successful of failed + fn exit_connect_screen(&mut self) { if matches!(&self.screen, Screen::Connecting {..}) { self.screen = Screen::Login { screen: login::Screen::new(), @@ -1244,13 +1245,14 @@ impl<'a> MainMenuUi { self.popup = None; self.connecting = None; self.connect = false; + self.ice_state.exit_connect_screen(); } pub fn cancel_connection(&mut self) { self.popup = None; self.connecting = None; self.connect = false; - self.ice_state.cancel_connection(); + self.ice_state.exit_connect_screen(); } pub fn handle_event(&mut self, event: ui::Event) { From 1e82f4ff410ecddd7c5ed2f40943f84a03651fcc Mon Sep 17 00:00:00 2001 From: Imbris Date: Wed, 10 Jun 2020 01:08:08 -0400 Subject: [PATCH 27/61] Add border styling to container --- voxygen/src/menu/main/ui/connecting.rs | 10 +- voxygen/src/menu/main/ui/disclaimer.rs | 8 +- voxygen/src/menu/main/ui/mod.rs | 38 +++--- voxygen/src/menu/main/ui/servers.rs | 42 +++--- .../src/ui/ice/renderer/style/container.rs | 20 ++- .../src/ui/ice/renderer/widget/container.rs | 127 +++++++++++++++++- 6 files changed, 190 insertions(+), 55 deletions(-) diff --git a/voxygen/src/menu/main/ui/connecting.rs b/voxygen/src/menu/main/ui/connecting.rs index 36647f7d9c..7316ce3e47 100644 --- a/voxygen/src/menu/main/ui/connecting.rs +++ b/voxygen/src/menu/main/ui/connecting.rs @@ -6,7 +6,7 @@ use crate::{ ice::{component::neat_button, style, Element}, }, }; -use iced::{button, Align, Color, Column, Container, Length, Row, Space, Text, VerticalAlignment}; +use iced::{button, Align, Color, Column, Container, Length, Row, Space, Text}; /// Connecting screen for the main menu pub struct Screen { @@ -25,7 +25,6 @@ impl Screen { pub(super) fn view( &mut self, fonts: &Fonts, - imgs: &Imgs, connection_state: &ConnectionState, time: f32, i18n: &Localization, @@ -99,8 +98,11 @@ impl Screen { .height(Length::Fill); let prompt_window = Container::new(content) - // TODO: add borders - .style(style::container::Style::Color((10, 10, 0, 255).into())) + .style(style::container::Style::color_double_cornerless_border( + (22, 18, 16, 255).into(), + (11, 11, 11, 255).into(), + (54, 46, 38, 255).into(), + )) .padding(10); let container = Container::new(prompt_window) diff --git a/voxygen/src/menu/main/ui/disclaimer.rs b/voxygen/src/menu/main/ui/disclaimer.rs index ac5401772f..198544c012 100644 --- a/voxygen/src/menu/main/ui/disclaimer.rs +++ b/voxygen/src/menu/main/ui/disclaimer.rs @@ -7,7 +7,6 @@ use crate::{ }, }; use iced::{button, scrollable, Column, Container, Length, Scrollable, Space}; -use vek::Rgba; /// Connecting screen for the main menu pub struct Screen { @@ -26,7 +25,6 @@ impl Screen { pub(super) fn view( &mut self, fonts: &Fonts, - imgs: &Imgs, i18n: &Localization, button_style: style::button::Style, ) -> Element { @@ -66,7 +64,11 @@ impl Screen { .width(Length::Fill) .height(Length::Fill), ) - .style(style::container::Style::Color(Rgba::new(22, 19, 17, 255))), + .style(style::container::Style::color_double_cornerless_border( + (22, 19, 17, 255).into(), + (11, 11, 11, 255).into(), + (54, 46, 38, 255).into(), + )), ) .center_x() .center_y() diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index b9dd5171d1..2c660b8714 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -164,7 +164,6 @@ struct IcedState { // Voxygen version version: String, - servers: Vec, selected_server_index: Option, login_info: LoginInfo, @@ -219,13 +218,16 @@ impl IcedState { }; //}; - let servers = settings.networking.servers.clone(); let login_info = LoginInfo { username: settings.networking.username.clone(), password: String::new(), server: settings.networking.default_server.clone(), }; - let selected_server_index = servers.iter().position(|f| f == &login_info.server); + let selected_server_index = settings + .networking + .servers + .iter() + .position(|f| f == &login_info.server); Self { fonts, @@ -234,7 +236,6 @@ impl IcedState { i18n, version, - servers, selected_server_index, login_info, @@ -244,7 +245,7 @@ impl IcedState { } } - fn view(&mut self, dt: f32) -> Element { + fn view(&mut self, settings: &Settings, dt: f32) -> Element { self.time = self.time + dt; // TODO: consider setting this as the default in the renderer @@ -268,9 +269,7 @@ impl IcedState { // TODO: make any large text blocks scrollable so that if the area is to // small they can still be read let content = match &mut self.screen { - /*Screen::Disclaimer { screen } => { - screen.view(&self.fonts, &self.imgs, &self.i18n, button_style) - },*/ + //Screen::Disclaimer { screen } => screen.view(&self.fonts, &self.i18n, button_style), Screen::Login { screen, error } => screen.view( &self.fonts, &self.imgs, @@ -281,8 +280,7 @@ impl IcedState { ), Screen::Servers { screen } => screen.view( &self.fonts, - &self.imgs, - &self.servers, + &settings.networking.servers, self.selected_server_index, &self.i18n, button_style, @@ -292,7 +290,6 @@ impl IcedState { connection_state, } => screen.view( &self.fonts, - &self.imgs, &connection_state, self.time, &self.i18n, @@ -311,7 +308,9 @@ impl IcedState { .into() } - fn update(&mut self, message: Message, events: &mut Vec) { + fn update(&mut self, message: Message, events: &mut Vec, settings: &Settings) { + let servers = &settings.networking.servers; + match message { Message::Quit => events.push(Event::Quit), Message::Back => { @@ -321,6 +320,8 @@ impl IcedState { }; }, Message::ShowServers => { + self.selected_server_index = + servers.iter().position(|f| f == &self.login_info.server); self.screen = Screen::Servers { screen: servers::Screen::new(), }; @@ -353,12 +354,11 @@ impl IcedState { Message::Username(new_value) => self.login_info.username = new_value, Message::Password(new_value) => self.login_info.password = new_value, Message::Server(new_value) => { - self.selected_server_index = self.servers.iter().position(|f| f == &new_value); self.login_info.server = new_value; }, Message::ServerChanged(new_value) => { - self.login_info.server = self.servers[new_value].clone(); self.selected_server_index = Some(new_value); + self.login_info.server = servers[new_value].clone(); }, Message::FocusPassword => { if let Screen::Login { screen, .. } = &mut self.screen { @@ -1272,12 +1272,14 @@ impl<'a> MainMenuUi { self.ui.maintain(global_state.window.renderer_mut(), None); if self.show_iced { let (messages, _) = self.ice_ui.maintain( - self.ice_state.view(dt.as_secs_f32()), + self.ice_state + .view(&global_state.settings, dt.as_secs_f32()), global_state.window.renderer_mut(), ); - messages - .into_iter() - .for_each(|message| self.ice_state.update(message, &mut events)); + messages.into_iter().for_each(|message| { + self.ice_state + .update(message, &mut events, &global_state.settings) + }); } events diff --git a/voxygen/src/menu/main/ui/servers.rs b/voxygen/src/menu/main/ui/servers.rs index 817596a003..0443566a41 100644 --- a/voxygen/src/menu/main/ui/servers.rs +++ b/voxygen/src/menu/main/ui/servers.rs @@ -3,16 +3,7 @@ use crate::{ i18n::Localization, ui::{ fonts::IcedFonts as Fonts, - ice::{ - component::neat_button, - style, - widget::{ - background_container::Padding, - compound_graphic::{CompoundGraphic, Graphic}, - BackgroundContainer, - }, - Element, - }, + ice::{component::neat_button, style, Element}, }, }; use iced::{button, scrollable, Align, Button, Column, Container, Length, Scrollable, Text}; @@ -35,8 +26,7 @@ impl Screen { pub(super) fn view( &mut self, fonts: &Fonts, - imgs: &Imgs, - servers: &Vec, + servers: &[impl AsRef], selected_server_index: Option, i18n: &Localization, button_style: style::button::Style, @@ -56,38 +46,46 @@ impl Screen { let mut list = Scrollable::new(&mut self.servers_list) .align_items(Align::Start) .width(Length::Fill) - .height(Length::Fill) - .spacing(10); + .height(Length::Fill); if self.server_buttons.len() != servers.len() { self.server_buttons = vec![Default::default(); servers.len()]; } - for (i, state) in self.server_buttons.iter_mut().enumerate() { - let server = servers.get(i).unwrap(); + for (i, (state, server)) in self.server_buttons.iter_mut().zip(servers).enumerate() { let text = format!( "{}{}", - if i == selected_server_index.unwrap_or(std::usize::MAX) { + if Some(i) == selected_server_index { "-> " } else { " " }, - server + server.as_ref(), ); - let button = Button::new(state, Text::new(text).size(fonts.cyri.scale(25))) - .on_press(Message::ServerChanged(i)); + let button = Button::new( + state, + Container::new(Text::new(text).size(fonts.cyri.scale(25))) + .padding(5) + .center_y(), + ) + .width(Length::Fill) + .on_press(Message::ServerChanged(i)); list = list.push(button); } Container::new( - BackgroundContainer::new( - CompoundGraphic::padded_image(imgs.info_frame, [500, 300], [0; 4]), + Container::new( Column::with_children(vec![list.into(), button.into()]) .width(Length::Fill) .height(Length::Fill) .spacing(10) .padding(20), ) + .style(style::container::Style::color_double_cornerless_border( + (22, 18, 16, 255).into(), + (11, 11, 11, 255).into(), + (54, 46, 38, 255).into(), + )) .max_width(500), ) .width(Length::Fill) diff --git a/voxygen/src/ui/ice/renderer/style/container.rs b/voxygen/src/ui/ice/renderer/style/container.rs index c430a271bd..dde0865216 100644 --- a/voxygen/src/ui/ice/renderer/style/container.rs +++ b/voxygen/src/ui/ice/renderer/style/container.rs @@ -1,16 +1,34 @@ use super::super::super::widget::image; use vek::Rgba; +/// Container Border +pub enum Border { + DoubleCornerless { inner: Rgba, outer: Rgba }, + None, +} + /// Background of the container pub enum Style { Image(image::Handle, Rgba), - Color(Rgba), + Color(Rgba, Border), None, } impl Style { /// Shorthand for common case where the color of the image is not modified pub fn image(image: image::Handle) -> Self { Self::Image(image, Rgba::broadcast(255)) } + + /// Shorthand for a color background with no border + pub fn color(color: Rgba) -> Self { Self::Color(color, Border::None) } + + /// Shorthand for a color background with a cornerless border + pub fn color_double_cornerless_border( + color: Rgba, + inner: Rgba, + outer: Rgba, + ) -> Self { + Self::Color(color, Border::DoubleCornerless { inner, outer }) + } } impl Default for Style { diff --git a/voxygen/src/ui/ice/renderer/widget/container.rs b/voxygen/src/ui/ice/renderer/widget/container.rs index 4a30ede9a9..6b1559294f 100644 --- a/voxygen/src/ui/ice/renderer/widget/container.rs +++ b/voxygen/src/ui/ice/renderer/widget/container.rs @@ -1,6 +1,9 @@ use super::super::{super::Rotation, style, IcedRenderer, Primitive}; use common::util::srgba_to_linear; use iced::{container, Element, Layout, Point, Rectangle}; +use style::container::Border; + +const BORDER_SIZE: u16 = 8; impl container::Renderer for IcedRenderer { type Style = style::container::Style; @@ -29,15 +32,125 @@ impl container::Renderer for IcedRenderer { primitives: vec![background, content], } }, - Self::Style::Color(color) => { - let background = Primitive::Rectangle { - bounds, - linear_color: srgba_to_linear(color.map(|e| e as f32 / 255.0)), + Self::Style::Color(color, border) => { + let linear_color = srgba_to_linear(color.map(|e| e as f32 / 255.0)); + + let primitives = match border { + Border::None => { + let background = Primitive::Rectangle { + bounds, + linear_color, + }; + + vec![background, content] + }, + Border::DoubleCornerless { inner, outer } => { + let border_size = f32::from(BORDER_SIZE) + .min(bounds.width / 4.0) + .min(bounds.height / 4.0); + + let center = Primitive::Rectangle { + bounds: Rectangle { + x: bounds.x + border_size * 2.0, + y: bounds.y + border_size * 2.0, + width: bounds.width - border_size * 4.0, + height: bounds.height - border_size * 4.0, + }, + linear_color, + }; + + let linear_color = srgba_to_linear(outer.map(|e| e as f32 / 255.0)); + let top = Primitive::Rectangle { + bounds: Rectangle { + x: bounds.x + border_size, + y: bounds.y, + width: bounds.width - border_size * 2.0, + height: border_size, + }, + linear_color, + }; + let bottom = Primitive::Rectangle { + bounds: Rectangle { + x: bounds.x + border_size, + y: bounds.y + bounds.height - border_size, + width: bounds.width - border_size * 2.0, + height: border_size, + }, + linear_color, + }; + let left = Primitive::Rectangle { + bounds: Rectangle { + x: bounds.x, + y: bounds.y + border_size, + width: border_size, + height: bounds.height - border_size * 2.0, + }, + linear_color, + }; + let right = Primitive::Rectangle { + bounds: Rectangle { + x: bounds.x + bounds.width - border_size, + y: bounds.y + border_size, + width: border_size, + height: bounds.height - border_size * 2.0, + }, + linear_color, + }; + + let linear_color = srgba_to_linear(inner.map(|e| e as f32 / 255.0)); + let top_inner = Primitive::Rectangle { + bounds: Rectangle { + x: bounds.x + border_size, + y: bounds.y + border_size, + width: bounds.width - border_size * 2.0, + height: border_size, + }, + linear_color, + }; + let bottom_inner = Primitive::Rectangle { + bounds: Rectangle { + x: bounds.x + border_size, + y: bounds.y + bounds.height - border_size * 2.0, + width: bounds.width - border_size * 2.0, + height: border_size, + }, + linear_color, + }; + let left_inner = Primitive::Rectangle { + bounds: Rectangle { + x: bounds.x + border_size, + y: bounds.y + border_size * 2.0, + width: border_size, + height: bounds.height - border_size * 4.0, + }, + linear_color, + }; + let right_inner = Primitive::Rectangle { + bounds: Rectangle { + x: bounds.x + bounds.width - border_size * 2.0, + y: bounds.y + border_size * 2.0, + width: border_size, + height: bounds.height - border_size * 4.0, + }, + linear_color, + }; + + vec![ + center, + top, + bottom, + left, + right, + top_inner, + bottom_inner, + left_inner, + right_inner, + content, + ] + }, }; - Primitive::Group { - primitives: vec![background, content], - } + Primitive::Group { primitives } }, Self::Style::None => content, }; From c04d2102e152c5bea2f7b5d6d4db4c49fb9a3ddd Mon Sep 17 00:00:00 2001 From: Imbris Date: Wed, 10 Jun 2020 01:56:49 -0400 Subject: [PATCH 28/61] Add error message box (is the main menu finished?) --- voxygen/src/menu/main/ui/login.rs | 45 ++++++++++++++++++++++++++----- 1 file changed, 39 insertions(+), 6 deletions(-) diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index 9a58989621..ef3e833097 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -30,6 +30,8 @@ pub struct Screen { settings_button: button::State, servers_button: button::State, + error_okay_button: button::State, + pub banner: Banner, } @@ -40,6 +42,8 @@ impl Screen { settings_button: Default::default(), quit_button: Default::default(), + error_okay_button: Default::default(), + banner: Banner::new(), } } @@ -108,15 +112,44 @@ impl Screen { .padding(27) .into(); - let banner = self - .banner - .view(fonts, imgs, login_info, i18n, button_style); + let central_content = if let Some(error) = error { + Container::new( + Column::with_children(vec![ + Container::new(Text::new(error)).height(Length::Fill).into(), + Container::new(neat_button( + &mut self.error_okay_button, + i18n.get("common.okay"), + FILL_FRAC_ONE, + button_style, + Some(Message::CloseError), + )) + .width(Length::Fill) + .height(Length::Units(30)) + .center_x() + .into(), + ]) + .height(Length::Fill) + .width(Length::Fill), + ) + .style(style::container::Style::color_double_cornerless_border( + (22, 18, 16, 255).into(), + (11, 11, 11, 255).into(), + (54, 46, 38, 255).into(), + )) + .width(Length::Units(400)) + .height(Length::Units(180)) + .padding(20) + .into() + } else { + self.banner + .view(fonts, imgs, login_info, i18n, button_style) + }; - let central_column = Container::new(banner) + let central_column = Container::new(central_content) .width(Length::Fill) .height(Length::Fill) - .align_x(Align::Center) - .align_y(Align::Center); + .center_x() + .center_y(); let right_column = Space::new(Length::Fill, Length::Fill); From 78fc52b9ec67967668c42cbc113dba90208bed2b Mon Sep 17 00:00:00 2001 From: Imbris Date: Wed, 10 Jun 2020 02:37:52 -0400 Subject: [PATCH 29/61] Delete conrod main menu code!!!!!!, small tweaks --- .../frames/{banner_png.png => banner.png} | Bin ...anner_bottom_png.png => banner_bottom.png} | Bin .../voxygen/element/frames/info_frame_2.vox | Bin 121052 -> 0 bytes voxygen/src/menu/main/mod.rs | 8 - voxygen/src/menu/main/ui/connecting.rs | 6 +- voxygen/src/menu/main/ui/disclaimer.rs | 2 +- voxygen/src/menu/main/ui/login.rs | 2 +- voxygen/src/menu/main/ui/mod.rs | 895 ++---------------- voxygen/src/menu/main/ui/servers.rs | 2 +- 9 files changed, 59 insertions(+), 856 deletions(-) rename assets/voxygen/element/frames/{banner_png.png => banner.png} (100%) rename assets/voxygen/element/frames/{banner_bottom_png.png => banner_bottom.png} (100%) delete mode 100644 assets/voxygen/element/frames/info_frame_2.vox diff --git a/assets/voxygen/element/frames/banner_png.png b/assets/voxygen/element/frames/banner.png similarity index 100% rename from assets/voxygen/element/frames/banner_png.png rename to assets/voxygen/element/frames/banner.png diff --git a/assets/voxygen/element/frames/banner_bottom_png.png b/assets/voxygen/element/frames/banner_bottom.png similarity index 100% rename from assets/voxygen/element/frames/banner_bottom_png.png rename to assets/voxygen/element/frames/banner_bottom.png diff --git a/assets/voxygen/element/frames/info_frame_2.vox b/assets/voxygen/element/frames/info_frame_2.vox deleted file mode 100644 index 0682f36efe9a43f374c22e67a5fa4958aff55931..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 121052 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 diff --git a/voxygen/src/menu/main/mod.rs b/voxygen/src/menu/main/mod.rs index f9d81b3244..7e1b12dc32 100644 --- a/voxygen/src/menu/main/mod.rs +++ b/voxygen/src/menu/main/mod.rs @@ -84,16 +84,9 @@ impl PlayState for MainMenuState { match event { Event::Close => return PlayStateResult::Shutdown, // Pass events to ui. - Event::Ui(event) => { - self.main_menu_ui.handle_event(event); - }, - // Pass events to iced ui. Event::IcedUi(event) => { self.main_menu_ui.handle_iced_event(event); }, - Event::InputUpdate(crate::window::GameInput::Jump, true) => { - self.main_menu_ui.show_iced ^= true; - }, // Ignore all other events. _ => {}, } @@ -262,7 +255,6 @@ impl PlayState for MainMenuState { global_state.singleplayer = Some(singleplayer); }, - MainMenuEvent::Settings => {}, // TODO MainMenuEvent::Quit => return PlayStateResult::Shutdown, /*MainMenuEvent::DisclaimerAccepted => { global_state.settings.show_disclaimer = false diff --git a/voxygen/src/menu/main/ui/connecting.rs b/voxygen/src/menu/main/ui/connecting.rs index 7316ce3e47..42bc40a20d 100644 --- a/voxygen/src/menu/main/ui/connecting.rs +++ b/voxygen/src/menu/main/ui/connecting.rs @@ -1,4 +1,4 @@ -use super::{ConnectionState, IcedImgs as Imgs, Message}; +use super::{ConnectionState, Message}; use crate::{ i18n::Localization, ui::{ @@ -93,7 +93,7 @@ impl Screen { .into(), ]) .spacing(4) - .max_width(500) + .max_width(520) .width(Length::Fill) .height(Length::Fill); @@ -103,7 +103,7 @@ impl Screen { (11, 11, 11, 255).into(), (54, 46, 38, 255).into(), )) - .padding(10); + .padding(20); let container = Container::new(prompt_window) .width(Length::Fill) diff --git a/voxygen/src/menu/main/ui/disclaimer.rs b/voxygen/src/menu/main/ui/disclaimer.rs index 198544c012..4215498d91 100644 --- a/voxygen/src/menu/main/ui/disclaimer.rs +++ b/voxygen/src/menu/main/ui/disclaimer.rs @@ -1,4 +1,4 @@ -use super::{IcedImgs as Imgs, Message}; +use super::Message; use crate::{ i18n::Localization, ui::{ diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index ef3e833097..a55ac6f5cb 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -1,4 +1,4 @@ -use super::{IcedImgs as Imgs, LoginInfo, Message}; +use super::{Imgs, LoginInfo, Message}; use crate::{ i18n::Localization, ui::{ diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 2c660b8714..bca5c332fc 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -8,29 +8,23 @@ use crate::{ render::Renderer, ui::{ self, - fonts::{Fonts, IcedFonts}, - ice::{Element, IcedUi}, - img_ids::{BlankGraphic, ImageGraphic, VoxelGraphic}, - Graphic, Ui, + fonts::IcedFonts as Fonts, + ice::{Element, IcedUi as Ui}, + img_ids::{ImageGraphic, VoxelGraphic}, + Graphic, }, GlobalState, }; //ImageFrame, Tooltip, use crate::settings::Settings; use common::assets::Asset; -use conrod_core::{ - color, - color::TRANSPARENT, - position::Relative, - widget::{text_box::Event as TextBoxEvent, Button, Image, List, Rectangle, Text, TextBox}, - widget_ids, Borderable, Color, Colorable, Labelable, Positionable, Sizeable, Widget, -}; use image::DynamicImage; use rand::{seq::SliceRandom, thread_rng, Rng}; use std::time::Duration; use ui::ice::widget; -const COL1: Color = Color::Rgba(0.07, 0.1, 0.1, 0.9); +// TODO: what is this? (showed up in rebase) +//const COL1: Color = Color::Rgba(0.07, 0.1, 0.1, 0.9); // UI Color-Theme /*const UI_MAIN: Color = Color::Rgba(0.61, 0.70, 0.70, 1.0); // Greenish Blue @@ -39,29 +33,22 @@ const UI_HIGHLIGHT_0: Color = Color::Rgba(0.79, 1.09, 1.09, 1.0);*/ use iced::{text_input, Column, Container, HorizontalAlignment, Length}; use ui::ice::style; image_ids_ice! { - struct IcedImgs { + struct Imgs { v_logo: "voxygen.element.v_logo", - info_frame: "voxygen.element.frames.info_frame_2", - - //banner: "voxygen.element.frames.banner", bg: "voxygen.background.bg_main", - banner: "voxygen.element.frames.banner_png", - banner_bottom: "voxygen.element.frames.banner_bottom_png", + banner: "voxygen.element.frames.banner", + banner_bottom: "voxygen.element.frames.banner_bottom", banner_top: "voxygen.element.frames.banner_top", button: "voxygen.element.buttons.button", button_hover: "voxygen.element.buttons.button_hover", button_press: "voxygen.element.buttons.button_press", input_bg: "voxygen.element.misc_bg.textbox", - disclaimer: "voxygen.element.frames.disclaimer", loading_art: "voxygen.element.frames.loading_screen.loading_bg", loading_art_l: "voxygen.element.frames.loading_screen.loading_bg_l", loading_art_r: "voxygen.element.frames.loading_screen.loading_bg_r", - - - nothing: (), } } @@ -95,23 +82,11 @@ pub enum Event { #[cfg(feature = "singleplayer")] StartSingleplayer, Quit, - Settings, //DisclaimerClosed, TODO: remove all traces? //DisclaimerAccepted, AuthServerTrust(String, bool), } -pub enum PopupType { - Error, - ConnectionInfo, - AuthTrustPrompt(String), -} - -pub struct PopupData { - msg: String, - popup_type: PopupType, -} - pub struct LoginInfo { pub username: String, pub password: String, @@ -156,9 +131,9 @@ enum Screen { }, } -struct IcedState { - fonts: IcedFonts, - imgs: IcedImgs, +struct Controls { + fonts: Fonts, + imgs: Imgs, bg_img: widget::image::Handle, i18n: std::sync::Arc, // Voxygen version @@ -193,10 +168,10 @@ enum Message { *AcceptDisclaimer, */ } -impl IcedState { +impl Controls { fn new( - fonts: IcedFonts, - imgs: IcedImgs, + fonts: Fonts, + imgs: Imgs, bg_img: widget::image::Handle, i18n: std::sync::Arc, settings: &Settings, @@ -320,11 +295,13 @@ impl IcedState { }; }, Message::ShowServers => { - self.selected_server_index = - servers.iter().position(|f| f == &self.login_info.server); - self.screen = Screen::Servers { - screen: servers::Screen::new(), - }; + if matches!(&self.screen, Screen::Login {..}) { + self.selected_server_index = + servers.iter().position(|f| f == &self.login_info.server); + self.screen = Screen::Servers { + screen: servers::Screen::new(), + }; + } }, #[cfg(feature = "singleplayer")] Message::Singleplayer => { @@ -451,131 +428,23 @@ impl IcedState { } } -widget_ids! { - struct Ids { - // Background and logo - bg, - v_logo, - alpha_version, - alpha_text, - banner, - banner_top, - // Disclaimer - //disc_window, - //disc_text_1, - //disc_text_2, - //disc_button, - //disc_scrollbar, - // Login, Singleplayer - login_button, - login_text, - login_error, - login_error_bg, - address_text, - address_bg, - address_field, - username_text, - username_bg, - username_field, - password_text, - password_bg, - password_field, - singleplayer_button, - singleplayer_text, - usrnm_bg, - srvr_bg, - passwd_bg, - // Server list - servers_button, - servers_frame, - servers_text, - servers_close, - // Buttons - settings_button, - quit_button, - // Error - error_frame, - button_ok, - version, - // Info Window - info_frame, - info_text, - info_bottom, - // Auth Trust Prompt - button_add_auth_trust, - } -} - -image_ids! { - struct Imgs { - - v_logo: "voxygen.element.v_logo", - - info_frame: "voxygen.element.frames.info_frame_2", - - - bg: "voxygen.background.bg_main", - banner_top: "voxygen.element.frames.banner_top", - banner: "voxygen.element.frames.banner", - banner_bottom: "voxygen.element.frames.banner_bottom", - button: "voxygen.element.buttons.button", - button_hover: "voxygen.element.buttons.button_hover", - button_press: "voxygen.element.buttons.button_press", - input_bg: "voxygen.element.misc_bg.textbox", - //disclaimer: "voxygen.element.frames.disclaimer", - - - - nothing: (), - } -} - pub struct MainMenuUi { ui: Ui, - ice_ui: IcedUi, - ice_state: IcedState, - ids: Ids, - imgs: Imgs, - username: String, - password: String, - server_address: String, - popup: Option, - connecting: Option, - connect: bool, - show_servers: bool, - //show_disclaimer: bool, - time: f32, - anim_timer: f32, - bg_img_id: conrod_core::image::Id, - i18n: std::sync::Arc, - fonts: Fonts, - tip_no: u16, + // TODO: re add this + // tip_no: u16, pub show_iced: bool, + controls: Controls, } impl<'a> MainMenuUi { pub fn new(global_state: &mut GlobalState) -> Self { - let window = &mut global_state.window; - let networking = &global_state.settings.networking; - let gameplay = &global_state.settings.gameplay; - - let mut ui = Ui::new(window).unwrap(); - ui.set_scaling_mode(gameplay.ui_scale); - // Generate ids - let ids = Ids::new(ui.id_generator()); - // Load images - let imgs = Imgs::load(&mut ui).expect("Failed to load images"); - let bg_img_spec = BG_IMGS.choose(&mut thread_rng()).unwrap(); - let bg_img_id = ui.add_graphic(Graphic::Image(DynamicImage::load_expect(bg_img_spec))); // Load language let i18n = Localization::load_expect(&i18n_asset_key( &global_state.settings.language.selected_language, )); - // Load fonts. - let fonts = Fonts::load(&i18n.fonts, &mut ui).expect("Impossible to load fonts!"); // TODO: don't add default font twice - let ice_font = { + let font = { use std::io::Read; let mut buf = Vec::new(); common::assets::load_file("voxygen.font.haxrcorp_4089_cyrillic_altgr_extended", &[ @@ -587,708 +456,50 @@ impl<'a> MainMenuUi { ui::ice::Font::try_from_vec(buf).unwrap() }; - let mut ice_ui = IcedUi::new(window, ice_font).unwrap(); + let mut ui = Ui::new(&mut global_state.window, font).unwrap(); - let ice_fonts = - IcedFonts::load(&i18n.fonts, &mut ice_ui).expect("Impossible to load fonts"); + let fonts = Fonts::load(&i18n.fonts, &mut ui).expect("Impossible to load fonts"); - let ice_state = IcedState::new( - ice_fonts, - IcedImgs::load(&mut ice_ui).expect("Failed to load images"), - ice_ui.add_graphic(Graphic::Image(DynamicImage::load_expect(bg_img_spec))), - i18n.clone(), + let bg_img_spec = BG_IMGS.choose(&mut thread_rng()).unwrap(); + + let controls = Controls::new( + fonts, + Imgs::load(&mut ui).expect("Failed to load images"), + ui.add_graphic(Graphic::Image(DynamicImage::load_expect(bg_img_spec))), + i18n, &global_state.settings, ); - Self { - ui, - ice_ui, - ice_state, - ids, - imgs, - username: networking.username.clone(), - password: "".to_owned(), - server_address: networking.default_server.clone(), - popup: None, - connecting: None, - show_servers: false, - connect: false, - time: 0.0, - anim_timer: 0.0, - //show_disclaimer: global_state.settings.show_disclaimer, - bg_img_id, - i18n, - fonts, - tip_no: 0, - show_iced: false, - } - } - - #[allow(clippy::assign_op_pattern)] // TODO: Pending review in #587 - #[allow(clippy::op_ref)] // TODO: Pending review in #587 - #[allow(clippy::toplevel_ref_arg)] // TODO: Pending review in #587 - fn update_layout(&mut self, global_state: &mut GlobalState, dt: Duration) -> Vec { - let mut events = Vec::new(); - self.time = self.time + dt.as_secs_f32(); - let fade_msg = (self.time * 2.0).sin() * 0.5 + 0.51; - let (ref mut ui_widgets, ref mut _tooltip_manager) = self.ui.set_widgets(); - let tip_msg = format!( - "{} {}", - &self.i18n.get("main.tip"), - &self.i18n.get_variation("loading.tips", self.tip_no), - ); - let tip_show = global_state.settings.gameplay.loading_tips; - let mut rng = thread_rng(); - let version = common::util::DISPLAY_VERSION_LONG.clone(); - let scale = 0.8; - const TEXT_COLOR: Color = Color::Rgba(1.0, 1.0, 1.0, 1.0); - const TEXT_COLOR_2: Color = Color::Rgba(1.0, 1.0, 1.0, 0.2); - const TEXT_BG: Color = Color::Rgba(0.0, 0.0, 0.0, 1.0); - //const INACTIVE: Color = Color::Rgba(0.47, 0.47, 0.47, 0.47); - - let intro_text = &self.i18n.get("main.login_process"); - - // Background image, Veloren logo, Alpha-Version Label - Image::new(if self.connect { - self.bg_img_id - } else { - self.imgs.bg - }) - .middle_of(ui_widgets.window) - .set(self.ids.bg, ui_widgets); - - if self.connect { - // Artwork - Image::new(self.imgs.loading_art) - .h(100.0) - .w_of(self.ids.bg) - .mid_bottom_of(self.ids.bg) - .set(self.ids.mid, ui_widgets); - Image::new(self.imgs.loading_art_l) - .w_h(12.0, 10.0) - .top_left_with_margins_on(self.ids.mid, 2.0, 0.0) - .set(self.ids.left, ui_widgets); - Image::new(self.imgs.loading_art_r) - .w_h(12.0, 10.0) - .top_right_with_margins_on(self.ids.mid, 2.0, 0.0) - .set(self.ids.right, ui_widgets); - // Gears Animation - self.anim_timer = (self.anim_timer + dt.as_secs_f32()) * 1.05; // Linear time function with Anim-Speed Factor - if self.anim_timer >= 4.0 { - self.anim_timer = 0.0 // Reset timer at last frame to loop - }; - Image::new(match self.anim_timer.round() as i32 { - 0 => self.imgs.f1, - 1 => self.imgs.f2, - 2 => self.imgs.f3, - 3 => self.imgs.f4, - _ => self.imgs.f5, - }) - .w_h(74.0, 62.0) - .bottom_right_with_margins_on(self.ids.mid, 10.0, 10.0) - .set(self.ids.gears, ui_widgets); - if tip_show { - // Tips - Text::new(&tip_msg) - .color(TEXT_BG) - .mid_bottom_with_margin_on(self.ids.mid, 60.0) - .font_id(self.fonts.cyri.conrod_id) - .font_size(self.fonts.cyri.scale(20)) - .set(self.ids.tip_txt_bg, ui_widgets); - Text::new(&tip_msg) - .color(TEXT_COLOR) - .bottom_left_with_margins_on(self.ids.tip_txt_bg, 2.0, 2.0) - .font_id(self.fonts.cyri.conrod_id) - .font_size(self.fonts.cyri.scale(20)) - .set(self.ids.tip_txt, ui_widgets); - }; - }; - - // Version displayed top right corner - let pos = if self.connect { 5.0 } else { 98.0 }; - Text::new(&version) - .color(TEXT_COLOR) - .top_right_with_margins_on(ui_widgets.window, pos * scale, 10.0 * scale) - .font_id(self.fonts.cyri.conrod_id) - .font_size(self.fonts.cyri.scale(14)) - .set(self.ids.version, ui_widgets); - // Alpha Disclaimer - Text::new(&format!( - "Veloren {}", - common::util::DISPLAY_VERSION.as_str() - )) - .font_id(self.fonts.cyri.conrod_id) - .font_size(self.fonts.cyri.scale(10)) - .color(TEXT_COLOR) - .mid_top_with_margin_on(ui_widgets.window, 2.0) - .set(self.ids.alpha_text, ui_widgets); - // Popup (Error/Info/AuthTrustPrompt) - let mut change_popup = None; - if let Some(PopupData { msg, popup_type }) = &self.popup { - let text = Text::new(msg) - .rgba( - 1.0, - 1.0, - 1.0, - if let PopupType::ConnectionInfo = popup_type { - fade_msg - } else { - 1.0 - }, - ) - .font_id(self.fonts.cyri.conrod_id); - let (frame_w, frame_h) = if let PopupType::AuthTrustPrompt(_) = popup_type { - (65.0 * 8.0, 370.0) - } else { - (65.0 * 6.0, 140.0) - }; - let error_bg = Rectangle::fill_with([frame_w, frame_h], color::TRANSPARENT) - .rgba(0.1, 0.1, 0.1, if self.connect { 0.0 } else { 1.0 }) - .parent(ui_widgets.window); - if let PopupType::AuthTrustPrompt(_) = popup_type { - error_bg.middle_of(ui_widgets.window) - } else { - error_bg.up_from(self.ids.banner_top, 15.0) - } - .set(self.ids.login_error_bg, ui_widgets); - Image::new(self.imgs.info_frame) - .w_h(frame_w, frame_h) - .color(Some(Color::Rgba( - 1.0, - 1.0, - 1.0, - if let PopupType::ConnectionInfo = popup_type { - 0.0 - } else { - 1.0 - }, - ))) - .middle_of(self.ids.login_error_bg) - .set(self.ids.error_frame, ui_widgets); - if let PopupType::ConnectionInfo = popup_type { - /*text.mid_top_with_margin_on(self.ids.error_frame, 10.0) - .font_id(self.fonts.cyri.conrod_id) - .bottom_left_with_margins_on(self.ids.bg, 30.0, 95.0) - .font_size(self.fonts.cyri.scale(35)) - .set(self.ids.login_error, ui_widgets);*/ - } else { - text.mid_top_with_margin_on(self.ids.error_frame, 10.0) - .w(frame_w - 10.0 * 2.0) - .font_id(self.fonts.cyri.conrod_id) - .font_size(self.fonts.cyri.scale(20)) - .set(self.ids.login_error, ui_widgets); - }; - if Button::image(self.imgs.button) - .w_h(100.0, 30.0) - .mid_bottom_with_margin_on( - if let PopupType::ConnectionInfo = popup_type { - ui_widgets.window - } else { - self.ids.login_error_bg - }, - 10.0, - ) - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .label_y(Relative::Scalar(2.0)) - .label(match popup_type { - PopupType::Error => self.i18n.get("common.okay"), - PopupType::ConnectionInfo => self.i18n.get("common.cancel"), - PopupType::AuthTrustPrompt(_) => self.i18n.get("common.cancel"), - }) - .label_font_id(self.fonts.cyri.conrod_id) - .label_font_size(self.fonts.cyri.scale(15)) - .label_color(TEXT_COLOR) - .set(self.ids.button_ok, ui_widgets) - .was_clicked() - { - match &popup_type { - PopupType::Error => (), - PopupType::ConnectionInfo => { - events.push(Event::CancelLoginAttempt); - }, - PopupType::AuthTrustPrompt(auth_server) => { - events.push(Event::AuthServerTrust(auth_server.clone(), false)); - }, - }; - change_popup = Some(None); - } - - if let PopupType::AuthTrustPrompt(auth_server) = popup_type { - if Button::image(self.imgs.button) - .w_h(100.0, 30.0) - .right_from(self.ids.button_ok, 10.0) - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .label_y(Relative::Scalar(2.0)) - .label("Add") // TODO: localize - .label_font_id(self.fonts.cyri.conrod_id) - .label_font_size(self.fonts.cyri.scale(15)) - .label_color(TEXT_COLOR) - .set(self.ids.button_add_auth_trust, ui_widgets) - .was_clicked() - { - events.push(Event::AuthServerTrust(auth_server.clone(), true)); - change_popup = Some(Some(PopupData { - msg: self.i18n.get("main.connecting").into(), - popup_type: PopupType::ConnectionInfo, - })); - } - } - } - if let Some(p) = change_popup { - self.popup = p; - } - - if !self.connect { - Image::new(self.imgs.banner) - .w_h(65.0 * 6.0 * scale, 100.0 * 6.0 * scale) - .middle_of(self.ids.bg) - .color(Some(Color::Rgba(0.0, 0.0, 0.0, 0.0))) - .set(self.ids.banner, ui_widgets); - - Image::new(self.imgs.banner_top) - .w_h(70.0 * 6.0 * scale, 34.0 * scale) - .mid_top_with_margin_on(self.ids.banner, -34.0) - .color(Some(Color::Rgba(0.0, 0.0, 0.0, 0.0))) - .set(self.ids.banner_top, ui_widgets); - - // Logo - Image::new(self.imgs.v_logo) - .w_h(123.0 * 2.5 * scale, 35.0 * 2.5 * scale) - .top_right_with_margins_on(self.ids.bg, 10.0, 10.0) - .color(Some(Color::Rgba(1.0, 1.0, 1.0, 0.95))) - .set(self.ids.v_logo, ui_widgets); - - /*if self.show_disclaimer { - Image::new(self.imgs.disclaimer) - .w_h(1800.0, 800.0) - .middle_of(ui_widgets.window) - .scroll_kids() - .scroll_kids_vertically() - .set(self.ids.disc_window, ui_widgets); - - Text::new(&self.i18n.get("common.disclaimer")) - .top_left_with_margins_on(self.ids.disc_window, 30.0, 40.0) - .font_size(self.fonts.cyri.scale(35)) - .font_id(self.fonts.alkhemi.conrod_id) - .color(TEXT_COLOR) - .set(self.ids.disc_text_1, ui_widgets); - Text::new(&self.i18n.get("main.notice")) - .top_left_with_margins_on(self.ids.disc_window, 110.0, 40.0) - .font_size(self.fonts.cyri.scale(26)) - .font_id(self.fonts.cyri.conrod_id) - .color(TEXT_COLOR) - .set(self.ids.disc_text_2, ui_widgets); - if Button::image(self.imgs.button) - .w_h(300.0, 50.0) - .mid_bottom_with_margin_on(self.ids.disc_window, 30.0) - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .label_y(Relative::Scalar(2.0)) - .label(&self.i18n.get("common.accept")) - .label_font_size(self.fonts.cyri.scale(22)) - .label_color(TEXT_COLOR) - .label_font_id(self.fonts.cyri.conrod_id) - .set(self.ids.disc_button, ui_widgets) - .was_clicked() - { - self.show_disclaimer = false; - events.push(Event::DisclaimerAccepted); - } - } else {*/ - // TODO: Don't use macros for this? - // Input fields - // Used when the login button is pressed, or enter is pressed within input field - macro_rules! login { - () => { - self.connect = true; - self.connecting = Some(std::time::Instant::now()); - self.popup = Some(PopupData { - msg: [self.i18n.get("main.connecting"), "..."].concat(), - popup_type: PopupType::ConnectionInfo, - }); - - events.push(Event::LoginAttempt { - username: self.username.clone(), - password: self.password.clone(), - server_address: self.server_address.clone(), - }); - }; - } - // Info Window - Rectangle::fill_with([550.0 * scale, 250.0 * scale], COL1) - .top_left_with_margins_on(ui_widgets.window, 40.0 * scale, 40.0 * scale) - .color(Color::Rgba(0.0, 0.0, 0.0, 0.80)) - .set(self.ids.info_frame, ui_widgets); - Image::new(self.imgs.banner_bottom) - .mid_bottom_with_margin_on(self.ids.info_frame, -50.0 * scale) - .w_h(550.0 * scale, 50.0 * scale) - .color(Some(Color::Rgba(0.0, 0.0, 0.0, 0.80))) - .set(self.ids.info_bottom, ui_widgets); - Text::new(intro_text) - .top_left_with_margins_on(self.ids.info_frame, 15.0 * scale, 15.0 * scale) - .font_size(self.fonts.cyri.scale(16)) - .font_id(self.fonts.cyri.conrod_id) - .color(TEXT_COLOR) - .set(self.ids.info_text, ui_widgets); - - // Singleplayer - // Used when the singleplayer button is pressed - #[cfg(feature = "singleplayer")] - macro_rules! singleplayer { - () => { - events.push(Event::StartSingleplayer); - self.connect = true; - self.connecting = Some(std::time::Instant::now()); - self.popup = Some(PopupData { - msg: [self.i18n.get(""), ""].concat(), - popup_type: PopupType::ConnectionInfo, - }); - }; - } - - // Username - Rectangle::fill_with( - [320.0 * scale, 50.0 * scale], - color::rgba(0.0, 0.0, 0.0, 0.0), - ) - .mid_top_with_margin_on(self.ids.banner_top, 150.0) - .set(self.ids.usrnm_bg, ui_widgets); - Image::new(self.imgs.input_bg) - .w_h(338.0 * scale, 50.0 * scale) - .middle_of(self.ids.usrnm_bg) - .set(self.ids.username_bg, ui_widgets); - for event in TextBox::new(&self.username) - .w_h(290.0* scale, 30.0* scale) - .mid_bottom_with_margin_on(self.ids.username_bg, 14.0* scale) - .font_size(self.fonts.cyri.scale(18)) - .font_id(self.fonts.cyri.conrod_id) - .text_color(TEXT_COLOR) - // transparent background - .color(TRANSPARENT) - .border_color(TRANSPARENT) - .set(self.ids.username_field, ui_widgets) - { - match event { - TextBoxEvent::Update(username) => { - // Note: TextBox limits the input string length to what fits in it - self.username = username.to_string(); - }, - TextBoxEvent::Enter => { - login!(); - }, - } - } - // Password - Rectangle::fill_with( - [320.0 * scale, 50.0 * scale], - color::rgba(0.0, 0.0, 0.0, 0.0), - ) - .down_from(self.ids.usrnm_bg, 10.0 * scale) - .set(self.ids.passwd_bg, ui_widgets); - Image::new(self.imgs.input_bg) - .w_h(338.0 * scale, 50.0 * scale) - .middle_of(self.ids.passwd_bg) - .set(self.ids.password_bg, ui_widgets); - for event in TextBox::new(&self.password) - .w_h(290.0 * scale, 30.0* scale) - .mid_bottom_with_margin_on(self.ids.password_bg, 10.0* scale) - // The text is smaller to allow longer passwords, conrod limits text length - // Basically the lower the scale of the font, the smaller and more characters we can fit - // At the time of this commit change, scale of 10 should fit 34 characters - .font_size(self.fonts.cyri.scale(10)) - .font_id(self.fonts.cyri.conrod_id) - .text_color(TEXT_COLOR) - // transparent background - .color(TRANSPARENT) - .border_color(TRANSPARENT) - .hide_text("*") - .set(self.ids.password_field, ui_widgets) - { - match event { - TextBoxEvent::Update(password) => { - // Note: TextBox limits the input string length to what fits in it - self.password = password; - }, - TextBoxEvent::Enter => { - self.password.pop(); - login!(); - }, - } - } - - if self.show_servers { - Image::new(self.imgs.info_frame) - .mid_top_with_margin_on(self.ids.username_bg, -320.0) - .w_h(400.0, 300.0) - .set(self.ids.servers_frame, ui_widgets); - - let ref mut net_settings = global_state.settings.networking; - - // TODO: Draw scroll bar or remove it. - let (mut items, _scrollbar) = List::flow_down(net_settings.servers.len()) - .top_left_with_margins_on(self.ids.servers_frame, 0.0, 5.0) - .w_h(400.0, 300.0) - .scrollbar_next_to() - .scrollbar_thickness(18.0) - .scrollbar_color(TEXT_COLOR) - .set(self.ids.servers_text, ui_widgets); - - while let Some(item) = items.next(ui_widgets) { - let mut text = "".to_string(); - if &net_settings.servers[item.i] == &self.server_address { - text.push_str("-> ") - } else { - text.push_str(" ") - } - text.push_str(&net_settings.servers[item.i]); - - if item - .set( - Button::image(self.imgs.nothing) - .w_h(100.0, 50.0) - .mid_top_with_margin_on(self.ids.servers_frame, 10.0) - //.hover_image(self.imgs.button_hover) - //.press_image(self.imgs.button_press) - .label_y(Relative::Scalar(2.0)) - .label(&text) - .label_font_size(self.fonts.cyri.scale(20)) - .label_font_id(self.fonts.cyri.conrod_id) - .label_color(TEXT_COLOR), - ui_widgets, - ) - .was_clicked() - { - self.server_address = net_settings.servers[item.i].clone(); - net_settings.default_server = self.server_address.clone(); - } - } - - if Button::image(self.imgs.button) - .w_h(200.0, 53.0) - .mid_bottom_with_margin_on(self.ids.servers_frame, 5.0) - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .label_y(Relative::Scalar(2.0)) - .label(&self.i18n.get("common.close")) - .label_font_size(self.fonts.cyri.scale(20)) - .label_font_id(self.fonts.cyri.conrod_id) - .label_color(TEXT_COLOR) - .set(self.ids.servers_close, ui_widgets) - .was_clicked() - { - self.show_servers = false - }; - } - // Server address - Rectangle::fill_with( - [320.0 * scale, 50.0 * scale], - color::rgba(0.0, 0.0, 0.0, 0.0), - ) - .down_from(self.ids.passwd_bg, 8.0 * scale) - .set(self.ids.srvr_bg, ui_widgets); - Image::new(self.imgs.input_bg) - .w_h(338.0 * scale, 50.0 * scale) - .middle_of(self.ids.srvr_bg) - .set(self.ids.address_bg, ui_widgets); - for event in TextBox::new(&self.server_address) - .w_h(290.0*scale, 30.0*scale) - .mid_top_with_margin_on(self.ids.address_bg, 8.0*scale) - .font_size(self.fonts.cyri.scale(18)) - .font_id(self.fonts.cyri.conrod_id) - .text_color(TEXT_COLOR) - // transparent background - .color(TRANSPARENT) - .border_color(TRANSPARENT) - .set(self.ids.address_field, ui_widgets) - { - match event { - TextBoxEvent::Update(server_address) => { - self.server_address = server_address.to_string(); - }, - TextBoxEvent::Enter => { - login!(); - }, - } - } - - // Login button - if Button::image(self.imgs.button) - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .w_h(258.0*scale, 55.0*scale) - .down_from(self.ids.address_bg, 20.0*scale) - .align_middle_x_of(self.ids.address_bg) - .label(&self.i18n.get("common.multiplayer")) - .label_font_id(self.fonts.cyri.conrod_id) - .label_color(TEXT_COLOR) - .label_font_size(self.fonts.cyri.scale(18)) - .label_y(Relative::Scalar(4.0)) - /*.with_tooltip( - tooltip_manager, - "Login", - "Click to login with the entered details", - &tooltip, - ) - .tooltip_image(self.imgs.v_logo)*/ - .set(self.ids.login_button, ui_widgets) - .was_clicked() - { - self.tip_no = rng.gen(); - login!(); - } - - // Singleplayer button - #[cfg(feature = "singleplayer")] - { - if Button::image(self.imgs.button) - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .w_h(258.0 * scale, 55.0 * scale) - .down_from(self.ids.login_button, 20.0 * scale) - .align_middle_x_of(self.ids.address_bg) - .label(&self.i18n.get("common.singleplayer")) - .label_font_id(self.fonts.cyri.conrod_id) - .label_color(TEXT_COLOR) - .label_font_size(self.fonts.cyri.scale(18)) - .label_y(Relative::Scalar(4.0)) - .label_x(Relative::Scalar(2.0)) - .set(self.ids.singleplayer_button, ui_widgets) - .was_clicked() - { - self.tip_no = rng.gen(); - singleplayer!(); - } - } - // Quit - if Button::image(self.imgs.button) - .w_h(190.0 * scale, 40.0 * scale) - .bottom_left_with_margins_on(ui_widgets.window, 60.0 * scale, 30.0 * scale) - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .label(&self.i18n.get("common.quit")) - .label_font_id(self.fonts.cyri.conrod_id) - .label_color(TEXT_COLOR) - .label_font_size(self.fonts.cyri.scale(16)) - .label_y(Relative::Scalar(3.0)) - .set(self.ids.quit_button, ui_widgets) - .was_clicked() - { - events.push(Event::Quit); - } - - // Settings - if Button::image(self.imgs.button) - .w_h(190.0*scale, 40.0*scale) - .up_from(self.ids.quit_button, 8.0*scale) - //.hover_image(self.imgs.button_hover) - //.press_image(self.imgs.button_press) - .label(&self.i18n.get("common.settings")) - .label_font_id(self.fonts.cyri.conrod_id) - .label_color(TEXT_COLOR_2) - .label_font_size(self.fonts.cyri.scale(16)) - .label_y(Relative::Scalar(3.0)) - .set(self.ids.settings_button, ui_widgets) - .was_clicked() - { - events.push(Event::Settings); - } - - // Servers - if Button::image(self.imgs.button) - .w_h(190.0 * scale, 40.0 * scale) - .up_from(self.ids.settings_button, 8.0 * scale) - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .label(&self.i18n.get("common.servers")) - .label_font_id(self.fonts.cyri.conrod_id) - .label_color(TEXT_COLOR) - .label_font_size(self.fonts.cyri.scale(16)) - .label_y(Relative::Scalar(3.0)) - .set(self.ids.servers_button, ui_widgets) - .was_clicked() - { - self.show_servers = !self.show_servers; - }; - } - - events + Self { ui, controls } } pub fn auth_trust_prompt(&mut self, auth_server: String) { - self.ice_state.auth_trust_prompt(auth_server.clone()); - self.popup = Some(PopupData { - msg: format!( - "Warning: The server you are trying to connect to has provided this \ - authentication server address:\n\n{}\n\nbut it is not in your list of trusted \ - authentication servers.\n\nMake sure that you trust this site and owner to not \ - try and bruteforce your password!", - &auth_server - ), - popup_type: PopupType::AuthTrustPrompt(auth_server), - }) + self.controls.auth_trust_prompt(auth_server); } - pub fn show_info(&mut self, msg: String) { - self.ice_state.connection_error(msg.clone()); + pub fn show_info(&mut self, msg: String) { self.controls.connection_error(msg); } - self.popup = Some(PopupData { - msg, - popup_type: PopupType::Error, - }); - self.connecting = None; - self.connect = false; - } + pub fn connected(&mut self) { self.controls.exit_connect_screen(); } - pub fn connected(&mut self) { - self.popup = None; - self.connecting = None; - self.connect = false; - self.ice_state.exit_connect_screen(); - } + pub fn cancel_connection(&mut self) { self.controls.exit_connect_screen(); } - pub fn cancel_connection(&mut self) { - self.popup = None; - self.connecting = None; - self.connect = false; - self.ice_state.exit_connect_screen(); - } - - pub fn handle_event(&mut self, event: ui::Event) { - if !self.show_iced { - self.ui.handle_event(event); - } - } - - pub fn handle_iced_event(&mut self, event: ui::ice::Event) { - if self.show_iced { - self.ice_ui.handle_event(event); - } - } + pub fn handle_event(&mut self, event: ui::ice::Event) { self.ui.handle_event(event); } pub fn maintain(&mut self, global_state: &mut GlobalState, dt: Duration) -> Vec { - let mut events = self.update_layout(global_state, dt); - self.ui.maintain(global_state.window.renderer_mut(), None); - if self.show_iced { - let (messages, _) = self.ice_ui.maintain( - self.ice_state - .view(&global_state.settings, dt.as_secs_f32()), - global_state.window.renderer_mut(), - ); - messages.into_iter().for_each(|message| { - self.ice_state - .update(message, &mut events, &global_state.settings) - }); - } + let mut events = Vec::new(); + + let (messages, _) = self.ui.maintain( + self.controls.view(&global_state.settings, dt.as_secs_f32()), + global_state.window.renderer_mut(), + ); + + messages.into_iter().for_each(|message| { + self.controls + .update(message, &mut events, &global_state.settings) + }); events } - pub fn render(&self, renderer: &mut Renderer) { - self.ui.render(renderer, None); - if self.show_iced { - self.ice_ui.render(renderer); - } - } + pub fn render(&self, renderer: &mut Renderer) { self.ui.render(renderer); } } diff --git a/voxygen/src/menu/main/ui/servers.rs b/voxygen/src/menu/main/ui/servers.rs index 0443566a41..e0fd31a478 100644 --- a/voxygen/src/menu/main/ui/servers.rs +++ b/voxygen/src/menu/main/ui/servers.rs @@ -1,4 +1,4 @@ -use super::{IcedImgs as Imgs, Message}; +use super::Message; use crate::{ i18n::Localization, ui::{ From aa726f56e1fbe707ee395c11b2f58a87ab635b10 Mon Sep 17 00:00:00 2001 From: CapsizeGlimmer <> Date: Wed, 10 Jun 2020 12:04:48 -0400 Subject: [PATCH 30/61] Implement renderer for slider --- voxygen/src/ui/ice/renderer/style/mod.rs | 1 + voxygen/src/ui/ice/renderer/style/slider.rs | 32 ++++++++ voxygen/src/ui/ice/renderer/widget/mod.rs | 1 + voxygen/src/ui/ice/renderer/widget/slider.rs | 79 ++++++++++++++++++++ 4 files changed, 113 insertions(+) create mode 100644 voxygen/src/ui/ice/renderer/style/slider.rs create mode 100644 voxygen/src/ui/ice/renderer/widget/slider.rs diff --git a/voxygen/src/ui/ice/renderer/style/mod.rs b/voxygen/src/ui/ice/renderer/style/mod.rs index 370210eaaa..04297da836 100644 --- a/voxygen/src/ui/ice/renderer/style/mod.rs +++ b/voxygen/src/ui/ice/renderer/style/mod.rs @@ -1,3 +1,4 @@ pub mod button; pub mod container; pub mod scrollable; +pub mod slider; diff --git a/voxygen/src/ui/ice/renderer/style/slider.rs b/voxygen/src/ui/ice/renderer/style/slider.rs new file mode 100644 index 0000000000..6c6da652ac --- /dev/null +++ b/voxygen/src/ui/ice/renderer/style/slider.rs @@ -0,0 +1,32 @@ +use super::super::super::widget::image; +use vek::Rgba; + +#[derive(Clone, Copy)] +pub struct Style { + pub cursor: Cursor, + pub bar: Bar, + pub labels: bool, +} + +impl Default for Style { + fn default() -> Self { + Self { + cursor: Cursor::Color(Rgba::new(0.5, 0.5, 0.5, 1.0)), + bar: Bar::Color(Rgba::new(0.5, 0.5, 0.5, 1.0)), + labels: false, + } + } +} + +#[derive(Clone, Copy)] +pub enum Cursor { + Color(Rgba), + Image(image::Handle, Rgba), +} + +#[derive(Clone, Copy)] +pub enum Bar { + Color(Rgba), + Image(image::Handle, Rgba), +} + diff --git a/voxygen/src/ui/ice/renderer/widget/mod.rs b/voxygen/src/ui/ice/renderer/widget/mod.rs index 6fe01dac05..9cdb7b5e81 100644 --- a/voxygen/src/ui/ice/renderer/widget/mod.rs +++ b/voxygen/src/ui/ice/renderer/widget/mod.rs @@ -7,6 +7,7 @@ mod container; mod image; mod row; mod scrollable; +mod slider; mod space; mod text; mod text_input; diff --git a/voxygen/src/ui/ice/renderer/widget/slider.rs b/voxygen/src/ui/ice/renderer/widget/slider.rs new file mode 100644 index 0000000000..056c60b423 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/slider.rs @@ -0,0 +1,79 @@ +use super::super::{super::Rotation, style, IcedRenderer, Primitive}; +use common::util::srgba_to_linear; +use iced::{slider, mouse, Rectangle, Point}; +use core::ops::RangeInclusive; +use style::slider::{Bar, Cursor, Style}; + +const CURSOR_WIDTH: f32 = 10.0; +const CURSOR_HEIGHT: f32 = 16.0; +const BAR_HEIGHT: f32 = 18.0; + +impl slider::Renderer for IcedRenderer { + type Style = Style; + fn height(&self) -> u32 { 20 } + fn draw( + &mut self, + bounds: Rectangle, + cursor_position: Point, + range: RangeInclusive, + value: f32, + is_dragging: bool, + style: &Self::Style + ) -> Self::Output { + + let bar_bounds = Rectangle { + height: BAR_HEIGHT, + ..bounds + }; + let bar = match style.bar { + Bar::Color(color) => Primitive::Rectangle { + bounds: bar_bounds, + linear_color: srgba_to_linear(color), + }, + Bar::Image(handle, color) => Primitive::Image { + handle: (handle, Rotation::None), + bounds: bar_bounds, + color, + }, + }; + + let (max, min) = range.into_inner(); + let offset = bounds.width as f32 * (max - min ) / (value - min); + let cursor_bounds = Rectangle { + x: bounds.x + offset - CURSOR_WIDTH / 2.0, + y: bounds.y + if is_dragging { 2.0 } else { 0.0 }, + width: CURSOR_WIDTH, + height: CURSOR_HEIGHT, + }; + let cursor = match style.cursor { + Cursor::Color(color) => Primitive::Rectangle { + bounds: cursor_bounds, + linear_color: srgba_to_linear(color), + }, + Cursor::Image(handle, color) => Primitive::Image { + handle: (handle, Rotation::None), + bounds: cursor_bounds, + color, + }, + }; + + let interaction = if is_dragging { + mouse::Interaction::Grabbing + } else if cursor_bounds.contains(cursor_position) { + mouse::Interaction::Grab + } else if bar_bounds.contains(cursor_position) { + mouse::Interaction::Pointer + } else { + mouse::Interaction::Idle + }; + + let primitives = if style.labels { + // TODO text label on left and right ends + vec![bar, cursor] + } else { + // TODO Cursor text label + vec![bar, cursor] + }; + (Primitive::Group{primitives}, interaction) + } +} From 84d09ac92ea6e7f7322871fab5a44a1d1877e2f4 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sat, 27 Jun 2020 01:09:04 -0400 Subject: [PATCH 31/61] Start rewrite of character selection with iced --- assets/voxygen/element/buttons/x_red.png | Bin 0 -> 9714 bytes assets/voxygen/element/buttons/x_red.vox | Bin 58652 -> 0 bytes .../voxygen/element/buttons/x_red_hover.png | Bin 0 -> 10485 bytes .../voxygen/element/buttons/x_red_hover.vox | Bin 58652 -> 0 bytes .../voxygen/element/buttons/x_red_press.png | Bin 0 -> 8668 bytes .../voxygen/element/buttons/x_red_press.vox | Bin 59020 -> 0 bytes voxygen/src/menu/char_selection/mod.rs | 32 +- .../menu/char_selection/{ui.rs => old_ui.rs} | 0 voxygen/src/menu/char_selection/ui/mod.rs | 513 ++++++++++++++++++ voxygen/src/menu/main/ui/login.rs | 2 - voxygen/src/menu/main/ui/mod.rs | 16 +- voxygen/src/ui/ice/renderer/style/button.rs | 11 + voxygen/src/ui/ice/renderer/widget/mod.rs | 1 + voxygen/src/ui/ice/renderer/widget/overlay.rs | 36 ++ voxygen/src/ui/ice/widget/mod.rs | 2 + voxygen/src/ui/ice/widget/overlay.rs | 222 ++++++++ 16 files changed, 802 insertions(+), 33 deletions(-) create mode 100644 assets/voxygen/element/buttons/x_red.png delete mode 100644 assets/voxygen/element/buttons/x_red.vox create mode 100644 assets/voxygen/element/buttons/x_red_hover.png delete mode 100644 assets/voxygen/element/buttons/x_red_hover.vox create mode 100644 assets/voxygen/element/buttons/x_red_press.png delete mode 100644 assets/voxygen/element/buttons/x_red_press.vox rename voxygen/src/menu/char_selection/{ui.rs => old_ui.rs} (100%) create mode 100644 voxygen/src/menu/char_selection/ui/mod.rs create mode 100644 voxygen/src/ui/ice/renderer/widget/overlay.rs create mode 100644 voxygen/src/ui/ice/widget/overlay.rs diff --git a/assets/voxygen/element/buttons/x_red.png b/assets/voxygen/element/buttons/x_red.png new file mode 100644 index 0000000000000000000000000000000000000000..87533b70a6a1bedb4f8ff9a975d920aa4779e9a0 GIT binary patch literal 9714 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 A=>Px# literal 0 HcmV?d00001 diff --git a/assets/voxygen/element/buttons/x_red.vox b/assets/voxygen/element/buttons/x_red.vox deleted file mode 100644 index 19ec8922f79b3d3823e6db52adde0b964b895242..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 58652 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 Y*qeEn-c3qQe;W6E zaB^>EX>4U6ba`-PAZ2)IW&i+q+O3>xdK@{Atp9TrvxMZGTn?|#Gdq~&&l_ZxEO*=P z{?3_JTP2mMGD!jmM1WwK^*{f;&42M<&AFHmb4fL&hyOwi)ir)7&;75S@Ae7j`~4-p zf3rV7Z@&KEc*^tW@2|PNuWvjbf4t#y58Ho!-jw$*uDlnrUl%_AdEHq*&+mied3T?; z;k{5@_lHzV{yxveqT8E1Hi{&O)F&~u)#EPcZqt0U6&t-UFXUY9}TjDG8FY-% zBJLd-jP+RHN!CZ3?310#*0PQT12snOd~h8L*cjbpkS=(ioXM^@-JADhdtU3JOHNON zO&B5)SsB|@-!N7P5&y}d;6p9?7-Ebm=2%Ruu_d2EiYcX>1ad9=9CFMlgM7K>R(uI1 zmQ->nrIuD*^8|@8dhDs^UV80q;65XaIMT?Yj5^wM z(@$VxrkQ7%b+%>A0x4Elaix`4S#`CIuWi4>jyvtV%dWe9S?#sz-@g6>tJ&9T?pR8% z%U@RGS(EeU5>{{$Wiu@1g2m!l7QmsMZ1xlzf|KQBv!^ zWxF4i`&V`|vHUB$xqq^pk?H;~EN3v?-|hAbt4-y(aV&PDP;2^lqHpKYTsOnzy=?QF zr&fzWkD~#EJf_xgk|!_E?`d)|zn7~MC>nKQP`7*Uz7FmK9@$l+XH`q%$=GagEuJ|} z&nk@_XWRH(pH{a_RkxVIc0IbpLx0-#rnN;9oS}?0M;p7XQT7w#e)?h+dSDG$Er!Z_ z($k#gQJK%sPo*hG>(P9E`2T&>h59P>ucNQ+LXHmzMKb+xMLJ=!+%bt-3+rnr-5(dw3J_UA%S{ zu|{68W{h)NT+a1|XQV!#R)gFvM;mJ+Yq;JO9mmEaU`#X51=hf|xJqQeB%HB{lx{It zajgAn4ARFwTU@xG!wF~Xy+kXTHfPEqnX<9T!iVjQwJxR-?TS0++F3Z>!kAMeaD|Wt2J20ShLxyPF6Tf{bi@KN@?|m{g0E5l3LpMcINwhh{KdiT!F}mW3l=x zv%ssHK+q9cagQ^_$Kt?@m`Nt`!Wq3t**`v-zx^SOC(C9F+<%TCa&PA*g&1Wmb0Q;m zmNDX)+s}dq{AM9|i^17u$XJX;#g&LUc^rQm zb1kCM&EDo_S*u$EQ!Y%p;hSfyofw}r^)j2emklkK$_lnJ%VK#J?iCw&(pX8yy|1FBfK7rNe;;Y{osZZv=Qtsa9}eKxAS9 z#%yhi1X-i-HL}RFwXiZ^*mIjXAM-y1$8xMmdkQhRfjZo5FJbZGZ|AlUVRKfEF=YW9 z6Jvov34NdvqitjV-v>`vfM z^gO+gWn3!*PQi%60x$@??E*{U!g)pzg$FVcEp}pwu}&b0R5HCg2>_iNvaI{q#i0gEK;j0tgh6 zKuA8A$sIgXb7>4LOz5~mXs1$o4=22x-QhWoAD`bpn)e6Z;C-8X4Mwt_8cJpCPh7nS z5BamaxsD_Qh{iIaU1UE3AmwP7VSQxY8&x0%hSq$_fmb-~3}Fssx3RLfZIi2k$iO`t zDbk+=R)7)wcR}6lK|3V`gv-HH2UE^)sd^*!SwUUiug6Q5>JIAXk2IVi4bhp_{g0djr;P1NZ_-w zVaRSd3et{!3EU>1^2Ggw8%3BBM#mvoL0BH3RdO=}OOY9C++lP$6O0VtC6>|T_6SO4N zYyt7h%W&j zFeYKoD*DCkM#&*UDy(cq<|r_LRBFzM|L*4?uP99$u;(O1bS@?jC{HvEG`tLW>?lM& z)XM|g#6*TI*s;3xgldossgkl|X3%&ROUiO$K~sIGNI$19= zjYK>UVT+P1dLJys>%uxgRSg$8c^tZX5QoX5_e+Z|`#xZTm?WeG`zB`j3D1SB$?8fk z##w_3WPjtxE}+2{w*yV*S9 z#XWh)ii`&IjEo;anM5w2U-AJ5d92nW8WMWURUFgeKA@8|WolWQ6sW`3Syt<8d8`z9 zA+T`zOcF`J)?acX{QG1_pr1J!)Dm7M;)>;+kP}crTR}NVZtx+T2G`^PDCVat#o(vKJiCcxjxNA#jo2YN(4Kex3xKmH@={S5U#GZ|S zL{Gq>7><&1vb~vs;`soJ;CwXOMq?Hn7I%EQBj9^=K zOxMKV0>lAYZyt^a68y5xyv7LwvS+VB83PPhowOGP;u~mLE;2(DHyaUd+^3~ka9v(*0C(H8BReLpL zliEroFLeCRG{Ai00slw?y#HgSg^(}u*n=CW3USIYQYJ)US3YmY$#?S^H^Raw8HZIy zV#!qi>i9~8)_}_uXteMR*x-{i)5T&Zn@j2uuqDC+SYoj?gcZ+)MzPYMI#EZG6Qg=e z>p&~y4*(|TB$n29fPk3gMOlH6cdvGl2oCYI;wL|NCot?ztQzpINmtxDN3*~Z@k34u zWH!L);U}chLZkxaYm6hpaLoGT^dt<(OnhIaLwQ(yfE^vklC19Psm&-HQ6Swa_A~fS z{)RHaNp~yuFcvuDlsJGpSA(610j$UZvDhr?IBXTqERP)7?{t$oYlnP1L1TlWB&Y^C zukdgNL2EG|q0s}tfw&#?%yFdDxpkI{6l9IMx=RCc#62xf#M7XR$ZvMIgBVg z5!Ky6QQ~YtddEh%-!jr-*eU#@HV)?FqByVf2h)C(Jpc#&WAWzN{P9fDwULcM`#Jf zJqZIf1Y#2kqa5sn@c|H|R~EHOHXW?8m#dKEi@Xz=g=~mZ+Kq{o*;M_Bb%<7Z?Sqk} z;mQOwPE1VTIL$Kk982wi3=dNSI+97H%>e=S+%Tztta6GpZ`QU~vWQujOMSKMCsCl6!6_dNyqpCZApS2;M*e_a6eUAK~!?Q+qi&Y0%?LE?KqWV&TzVjF>oGJR%`NV zLHm}VnV_5?og8}!x7aD4gIs8y&2xRiH}Da~^d58*!zLZzfCx3HphSQQEaLRMD&c`f2?QqGcJ>Qun4|NU#wXlzEEU6_g&ym5A$EbM z0W`keK_bLElxhK}ZEz20R|9bfQnG6Y=}AbQh%agkSPiv?!t48qZwbhqdStqVAVdS} zvZi5x5SgZu71jl)5ZP$s2}tirrWqcQq%y@K(R~$pNI>E%--xU4_BfGOX&2ih3$jQm zafO%QHf9$`VgS^WvPmPm0k!a8l2vEcq|c_!jtzqxHG?FH1nxRsr)?B9r(~Gr#w@wIu1K|nOoWN!^xU{UN6^9}xg|>i7<4@QQLP^HNaupz6N}>> z8iK-WfC4DetcdCGI3Ps?;K@bi>o`r88ZPDzrUxrV6-jz`DnJ1%yU)f&PS_Rjhjt5U z>k>wR*%7glUFnb@TaD{J+&!WaMS||GL=iWY)7%PWntYDL+e5qWM3s=z~MXO3-4p5hpMt9Tu3%2 z=k0roYxO>HMS%9$9EG=3sZFXFJF)>}L&u?N!d8&U^| z16i9)2;W5t%gZ`U5IVxWs%==}kxVSw`RVu(+>Jq$3tm&;1vd#uI&dIGw5R+yjjhth za;Q$b;T%=F4_2jYe78EEP)ESt$P#o?p29Zm(gJ)=d7GJ8NXAqf2igNoNbb42!=(21 zO#qmGPzHZ~1W7ZuC7&k~Rpcy>WR@%O?{r0*E{lYFK)oE$=grClDi9JVe)Y2suBjGo-t6eL^#xE!&JhLLro7ur&B465 z3K@gHdDTPrxV{sx$Rdw;Q3QakNhdDtj;>P5>b1XBiP3*lC9FDUP`qvnn3Vyy_L{WO z%yn9v=87`FqJ`&QZ|?u%(fr{Bgm?YJ^8UQK<}b@T&`GyVAf$M!)_*hUPX?7bkVy?m z1UX(1{un~rloN5Ht*E_{xwrE%{t)yD42L`UdAA%7 zb#nbur=71bb_b|L)={Tcg-6s8A!Tb}?UCfxZba>f)mrEndW8HBL&BNWHjq8iCVlaQ zOYvToV6I9NO(;wXoe>C3#KgYt&>$)@^vKGT+*Mbo; z?o~xpQiD9h;JPK|GWRVe*^5LVpWvuFV(9Rr;I+*!VY7I7}OS*{lR$&+F z+LyJZd0c$kAhj;R;2B~qv6qXdm?!G9qlQB{Q3<1QWFU)=A~an0m7 zoVk4bX}+ikt-WB?u3m1GtGLkvLp$6ki>|sYYK@im;YM?l4Urf|fe`}SNQAA9*0oth zuF-v+JTS{a0#c4eMYQ8?T2-}OiQ^$J)B_hiA)+FH%A++FD+dcIifKSZq~47B1?71f z_g_)EFp<+PJ4+BRdnRJ4woRd|Ma&Fix5|Yojuy`@Rg(G1 zlMc~RgxG`-d)CB8Qw1(!k--SjPBwkix1pp&n5b17#}T>n#2J8_@@e)`UV(@acDOEn zIzV6~`L2QZN(**}$9k_!S=C-$A@#l632dptNqER+>=_XONv{6Ile?aq-7oAFYn@H9ezScLoUpuO!wSHcYcy5G}$dxBgs zU>52?4WN$IwkAKJuv`hwokod-?$e8IdI{QBMqr?v4E%KA&Plsqzb~NP`f0zNZFMjK z{RCdCIES#loh%3zMSO(Y?;pjRtgVVnD0~%fRR6R|yk`fyF+RD{ZIp zUESI$Tmf9RtLF+0pR>_m)E+Z%U-IhHU&i88nmb!$hZDj?JH)8~1GaYbhN_sB#R7EH znVs-#r&dU0PPZb>!7YLumYH1;7R9ppC5XejK`1P>5oamh*45<@y!1iYhNbFJEv>;4 z83g&G?MNKfP8B%urKYBs&RbSIY~QMom*>ioS>)8BSOVCDR?6n@o*v*7-W}gl55=if z4^5{lex53B^0#{em=735-4k^f3xr&)*`@Z>=`rOgQ;mCG`*121X(2NyE~N(O0uCVN z-Hvt0KR-VS}hWF}~OX?n1 z2R$=|`8068J_|!83$@J=o*@L*a z7r0te1fo2Aww)~+e8*|$cpQ1b+sPcKw?_TR`?*d`67KKrH@bhQWlX!i+9v>5`azFC zCh{;?R7l?!JbgXjsSRKdDh{WJ?5+j`#1qT|dRnHduJ*3R)ux^F)8)>U^gr*f&#dKF z+WN<%`Gq#Xm@|6G84zdP-uM6)$DiUuVhh^gJY5A3S&{I6RM2b=@5Fndhd7NoYHAo@ zgw8WXP5yL?JP;3-9m=C9DCr`*?4richZQuy_~}I!{fp0a{-0Wrr(h~YZQEpRNZMW} zk=U({TRag4ZzTtfnhy<)lD`Gdu9Mx=OdmZ&4ySuTwRSS)H9mENVMbLJ&6d_!BHIZ50 zEnlIVwtTIBw|u2$Pj6a&w|jr;b6Do9&4H!46~ER5Y1(twrY;~AKYQ*tkT#+r0C&$l zsHh+Bp1Z+sA!CKC(57$i7Jma5)vSHC_&2|daJTpgVSu9k-1l&bkLI+Gp5})Fu>&=b zv*Z+Y)D^%6g29mC;ea4&zLUD_FIo|DHGqf+`rG*cjtu3O+xg(NkK#|~1B?QsQx8rk zpFRuNz=N~K(`RAaMwX0TPzj$t3sNgY>GWA(a=3Cn8S4}!-p+?}*g-C%-b=D>=w|1H zb2v~Fl0bg-NCMWzcP+5e*HFyfPTJcqGX3or8P44%S~B~ZCg!&prKumW?@uoZ%%P@9 z3OKzeYWo+Ca(rvMy4(pLfMD)`%SrA2^jmK zg<{^j+=Exaseo-F&eEh-*X=}r>8foBA;s4qVPJ`JYm>-e_uDP4Q>?psUp!1@**YcS05fLl{Qkc3`qZ8%o~mKnDc!*$i*EroP3~ zwg@z=s?Id+_n*E+^||srynTx*J%PaLTRc6c#LYioxA}ng-?001ho4Ba3$D7fTdYWQ zkl~%&K#dWJL>UuP)in0KdAV2}}fD*|$rY*hU*pxTwYLrn;{^ zJloldx*)rvA<6$m*VP8>tox7P@pg5?kx!?rQCqhNO8n6(i?K>eYDRfGW$_)9b3UE2 z{GX%PH&6^=Q`agY#5c9X;IU77JAEN;$SH>Y)#-~UB-5AM>8naETKIJOp6*dtKvFX? zessHu^QW75UeKs*oPKW*_B19lRTOx*jmhOfnWod2?6ifTes6OqdKbh7tZ|yL;|>lO zA$DatHDjMXY-Wh1(ZeuW9RUR@$^GLLGQ&1-Dd^ihZk5zmcw4cp#2reFF#207_HDzK zkWf2_)A59{jQT-vV3x-&InZ7;ihiOtg7y;YPzP);It4P+M(`JzLS8(K-@J`xN||tg zD6xa@CKgHU+CPp)DTU7P9*uHVM?c&-8l?r&mQQr2;TtM6ZyUYUX%&|Bw9!8f?UPO$ zJl(dWR=jH*n{Al zT%zt@XN}cvR=5q)M7i3!<*erj0y#wg#UDoy7&UGLDf%2iP%^4z-&4@CUii`0{p+Z> zvc7Y5a;I9cacUJllD2cPfqCRC_sqq!S_UuY`kc7{3?mU-?LBglbbMfxbL0Zm*0lHd zr3uXMMZv$+1Z_)2H6y&l#E2-ENZvZG%pk!OHOjzBtN%vOf5MAKC6ECvdmK+`Boe;_jC{It6kUkY)@6Y zOuc7&pkt&do$VpH*K@X~NZadjk zRJ5RZKl)zP&A3p)dce@krhw3rif;cW!9^B!v}w};vQ#W|}X z>r|1nf!M>T`!M+DoC1GaK3$8d3aRGhZYzFJ+q70n`+kO`g$o?(o*j6e)qFjMa;C@dv zo@CKx7qR?L4q|@fubb~aqb}Id0hIl|l zO||+62xHY$yUGoK-?Y)G^4Zh5P4v9ZQA4cuw)27Dt^8E)u8 z8k2lJCMu*BwagZs9B%r29s|gm-dFJx^i?oVx-G_!Of?!09U zu=d6pN>f|%=a;Tt;~9p|KSYU&0?>~wSd6N?>H|S`hG6St`Ib=pItpq{MA|pmgYQQ(tv6(xEe+8wGhVuIVJHIzNWe-PiH{_i(>- zNq!G>YLg6cs#E?Na@QGk9riFmMSuJcJr@Xd0DQft`v0>N{^oC+{68P>zlZz3j`nLc zbQo1_j3Qz~j09_{0cERg2n1k`>mGUy-pHuV4%i>9V{8ZqTRf8Y7qC3KkP%t{O2v)nfILk)LFoe9#Xlfs@jC2j6wo!pXxTE zHflOjchET+uTo7rbzV|1P@CraSs)L;CYSGLfwWVhIOk%uwXE&$az+8KkzXyHEXw(F z(0x~9o3deiP6G5`ZrK8eBC$!%Imke0ZWYkF>hoV8yixQ~NRa<7&IhYii80Od`5LeV ziIAqYPp8itC_jBp!5=$5&GYYCLVxwRL%dt|8oZGMscc#=_I?}x`D73sBHcruWui|8 z(a5!-%!;djRXyBZa&&BG^pOj9opr+e<%;fp>I&A;I_pzcz$PEm5m)B>$p{CXqVQo! z5*i|(OLaLcY`dLf8|RZj)D@-=D5z2U7b-|p^S?cw99}p);Cq)-2@xG?eBE}1k_i^4mhxhISglds#R@*qB>9(1O zMa5KlRSdkM1O4z}6x}kjj5$e)!?V8bsgvq1!n3^l{;VD~V==%d63;Tjw23!}r#Eeb z^FDEiwUd^i64??-uAyQ( zhsYP`9kMfWbnd_%*efsTM3)*Rr4SoLJ%Mmu3~@rpIRgM*eLv%5+=n3mK!`D7O6fGH z@fgPto?SP8Zk7jAN*-i2j>-fPfrwOzEl|glLFFvWyx=WZK>%Rp>g}pZt?(YWLM?C$ z{E6HIihtA_D@e7DMK-J^00000NkvXXu0mjfl`IZ( literal 0 HcmV?d00001 diff --git a/assets/voxygen/element/buttons/x_red_hover.vox b/assets/voxygen/element/buttons/x_red_hover.vox deleted file mode 100644 index 59608cc7eb9241aa11a03491d1ff333ce032a780..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 58652 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 aHtX|d9;SDblGEMAUtaQ#`tYy>bpH>;PHE`? diff --git a/assets/voxygen/element/buttons/x_red_press.png b/assets/voxygen/element/buttons/x_red_press.png new file mode 100644 index 0000000000000000000000000000000000000000..7044cb22ba1168090948f1c7785f162ac2ffb5f0 GIT binary patch literal 8668 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 diff --git a/voxygen/src/menu/char_selection/mod.rs b/voxygen/src/menu/char_selection/mod.rs index 4cf356e5f8..91dbabd584 100644 --- a/voxygen/src/menu/char_selection/mod.rs +++ b/voxygen/src/menu/char_selection/mod.rs @@ -39,16 +39,10 @@ impl CharSelectionState { fn get_humanoid_body(&self) -> Option { self.char_selection_ui - .get_character_list() - .and_then(|data| { - if let Some(character) = data.get(self.char_selection_ui.selected_character) { - match character.body { - comp::Body::Humanoid(body) => Some(body), - _ => None, - } - } else { - None - } + .selected_character() + .and_then(|character| match character.body { + comp::Body::Humanoid(body) => Some(body), + _ => None, }) } } @@ -98,18 +92,10 @@ impl PlayState for CharSelectionState { ui::Event::DeleteCharacter(character_id) => { self.client.borrow_mut().delete_character(character_id); }, - ui::Event::Play => { - let char_data = self - .char_selection_ui - .get_character_list() - .expect("Character data is required to play"); - - if let Some(selected_character) = - char_data.get(self.char_selection_ui.selected_character) - { - if let Some(character_id) = selected_character.character.id { - self.client.borrow_mut().request_character(character_id); - } + ui::Event::Play(character) => { + // TODO: eliminate option in character id + if let Some(character_id) = character.character.id { + self.client.borrow_mut().request_character(character_id); } return PlayStateResult::Switch(Box::new(SessionState::new( @@ -212,6 +198,6 @@ impl PlayState for CharSelectionState { // Draw the UI to the screen. self.char_selection_ui - .render(renderer, self.scene.globals()); + .render(global_state.window.renderer_mut()); } } diff --git a/voxygen/src/menu/char_selection/ui.rs b/voxygen/src/menu/char_selection/old_ui.rs similarity index 100% rename from voxygen/src/menu/char_selection/ui.rs rename to voxygen/src/menu/char_selection/old_ui.rs diff --git a/voxygen/src/menu/char_selection/ui/mod.rs b/voxygen/src/menu/char_selection/ui/mod.rs new file mode 100644 index 0000000000..42bf044b2a --- /dev/null +++ b/voxygen/src/menu/char_selection/ui/mod.rs @@ -0,0 +1,513 @@ +use crate::{ + i18n::{i18n_asset_key, Localization}, + render::Renderer, + ui::{ + self, + fonts::IcedFonts as Fonts, + ice::{component::neat_button, style, widget::Overlay, Element, IcedUi as Ui}, + img_ids::{ImageGraphic, VoxelGraphic}, + }, + window, GlobalState, +}; +use client::Client; +use common::{ + character::{CharacterItem, MAX_CHARACTERS_PER_PLAYER}, + comp, + comp::humanoid, +}; +//ImageFrame, Tooltip, +use crate::settings::Settings; +use common::assets::load_expect; +//use std::time::Duration; +//use ui::ice::widget; +use iced::{ + button, text_input, Align, Button, Column, Container, HorizontalAlignment, Length, Row, Space, + Text, +}; + +pub const TEXT_COLOR: iced::Color = iced::Color::from_rgb(1.0, 1.0, 1.0); +pub const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1.0, 0.2); +const FILL_FRAC_ONE: f32 = 0.77; +const FILL_FRAC_TWO: f32 = 0.53; + +image_ids_ice! { + struct Imgs { + + // TODO: convert large frames into borders + charlist_frame: "voxygen.element.frames.window_4", + server_frame: "voxygen.element.frames.server_frame", + slider_range: "voxygen.element.slider.track", + slider_indicator: "voxygen.element.slider.indicator", + + + selection: "voxygen.element.frames.selection", + selection_hover: "voxygen.element.frames.selection_hover", + selection_press: "voxygen.element.frames.selection_press", + + delete_button: "voxygen.element.buttons.x_red", + delete_button_hover: "voxygen.element.buttons.x_red_hover", + delete_button_press: "voxygen.element.buttons.x_red_press", + + + name_input: "voxygen.element.misc_bg.textbox_mid", + + // Tool Icons + daggers: "voxygen.element.icons.daggers", + sword: "voxygen.element.icons.sword", + axe: "voxygen.element.icons.axe", + hammer: "voxygen.element.icons.hammer", + bow: "voxygen.element.icons.bow", + staff: "voxygen.element.icons.staff", + + // Species Icons + male: "voxygen.element.icons.male", + female: "voxygen.element.icons.female", + human_m: "voxygen.element.icons.human_m", + human_f: "voxygen.element.icons.human_f", + orc_m: "voxygen.element.icons.orc_m", + orc_f: "voxygen.element.icons.orc_f", + dwarf_m: "voxygen.element.icons.dwarf_m", + dwarf_f: "voxygen.element.icons.dwarf_f", + undead_m: "voxygen.element.icons.ud_m", + undead_f: "voxygen.element.icons.ud_f", + elf_m: "voxygen.element.icons.elf_m", + elf_f: "voxygen.element.icons.elf_f", + danari_m: "voxygen.element.icons.danari_m", + danari_f: "voxygen.element.icons.danari_f", + // Icon Borders + icon_border: "voxygen.element.buttons.border", + icon_border_mo: "voxygen.element.buttons.border_mo", + icon_border_press: "voxygen.element.buttons.border_press", + icon_border_pressed: "voxygen.element.buttons.border_pressed", + + + button: "voxygen.element.buttons.button", + button_hover: "voxygen.element.buttons.button_hover", + button_press: "voxygen.element.buttons.button_press", + } +} + +// TODO: do rotation in widget renderer +/*rotation_image_ids! { + pub struct ImgsRot { + + + // Tooltip Test + tt_side: "voxygen/element/frames/tt_test_edge", + tt_corner: "voxygen/element/frames/tt_test_corner_tr", + } +}*/ + +pub enum Event { + Logout, + Play(CharacterItem), + AddCharacter { + alias: String, + tool: Option, + body: comp::Body, + }, + DeleteCharacter(i32), +} + +struct CharacterList { + characters: Vec, + selected_character: usize, +} + +enum Mode { + Select { + list: Option, + character_buttons: Vec, + new_character_button: button::State, + + logout_button: button::State, + enter_world_button: button::State, + change_server_button: button::State, + }, + Create { + name: String, // TODO: default to username + body: humanoid::Body, + loadout: comp::Loadout, + tool: Option<&'static str>, + + name_input: text_input::State, + back_button: button::State, + create_button: button::State, + }, +} + +#[derive(PartialEq)] +enum InfoContent { + Deletion(usize), + LoadingCharacters, + CreatingCharacter, + DeletingCharacter, + CharacterError, +} + +/* +impl InfoContent { + pub fn has_content(&self, character_list_loading: &bool) -> bool { + match self { + Self::None => false, + Self::CreatingCharacter | Self::DeletingCharacter | Self::LoadingCharacters => { + *character_list_loading + }, + _ => true, + } + } +} +*/ + +struct Controls { + fonts: Fonts, + imgs: Imgs, + i18n: std::sync::Arc, + // Voxygen version + version: String, + + info_content: Option, + // enter: bool, + mode: Mode, +} + +#[derive(Clone)] +enum Message { + Back, + Logout, + EnterWorld, + Delete(usize), + ChangeServer, + NewCharacter, + CreateCharacter, + Name(String), +} + +impl Controls { + fn new(fonts: Fonts, imgs: Imgs, i18n: std::sync::Arc) -> Self { + let version = format!( + "{}-{}", + env!("CARGO_PKG_VERSION"), + common::util::GIT_VERSION.to_string() + ); + + Self { + fonts, + imgs, + i18n, + version, + + info_content: None, + mode: Mode::Select { + list: None, + character_buttons: Vec::new(), + new_character_button: Default::default(), + logout_button: Default::default(), + enter_world_button: Default::default(), + change_server_button: Default::default(), + }, + } + } + + fn view(&mut self, settings: &Settings) -> Element { + // TODO: if enter key pressed and character is selected then enter the world + // TODO: tooltip widget + + let imgs = &self.imgs; + let i18n = &self.i18n; + + let button_style = style::button::Style::new(imgs.button) + .hover_image(imgs.button_hover) + .press_image(imgs.button_press) + .text_color(TEXT_COLOR) + .disabled_text_color(DISABLED_TEXT_COLOR); + + let version = iced::Text::new(&self.version) + .size(self.fonts.cyri.scale(15)) + .width(Length::Fill) + .horizontal_alignment(HorizontalAlignment::Right); + + let content = match &mut self.mode { + Mode::Select { + list, + ref mut character_buttons, + ref mut new_character_button, + ref mut logout_button, + ref mut enter_world_button, + ref mut change_server_button, + } => { + // TODO: impl delete prompt as overlay + let change_server = Space::new(Length::Units(100), Length::Units(40)); + let characters = if let Some(list) = list { + let num = list.characters.len(); + // Ensure we have enough button states + character_buttons.resize_with(num * 2, Default::default); + + let mut characters = list + .characters + .iter() + .zip(character_buttons.chunks_exact_mut(2)) + .map(|(character, buttons)| { + let mut buttons = buttons.iter_mut(); + ( + character, + (buttons.next().unwrap(), buttons.next().unwrap()), + ) + }) + .enumerate() + .map(|(i, (character, (select_button, delete_button)))| { + Overlay::new( + Button::new(select_button, Space::new(Length::Fill, Length::Fill)) + .width(Length::Units(20)) + .height(Length::Units(20)) + .style( + style::button::Style::new(imgs.delete_button) + .hover_image(imgs.delete_button_hover) + .press_image(imgs.delete_button_press), + ), + Button::new( + delete_button, + Column::with_children(vec![ + Text::new("Hi").into(), + Text::new("Hi").into(), + Text::new("Hi").into(), + ]), + ) + .style( + style::button::Style::new(imgs.selection) + .hover_image(imgs.selection_hover) + .press_image(imgs.selection_press), + ), + ) + .into() + }) + .collect::>(); + + // Add create new character button + let color = if num >= MAX_CHARACTERS_PER_PLAYER { + iced::Color::from_rgb8(97, 97, 25) + } else { + iced::Color::from_rgb8(97, 255, 18) + }; + characters.push( + Button::new( + new_character_button, + Text::new(i18n.get("char_selection.create_new_character")), + ) + .style( + style::button::Style::new(imgs.selection) + .hover_image(imgs.selection_hover) + .press_image(imgs.selection_press) + .image_color(color) + .text_color(color), + ) + .into(), + ); + characters + } else { + Vec::new() + }; + + let characters = Column::with_children(characters); + + let right_column = + Column::with_children(vec![change_server.into(), characters.into()]) + .spacing(10) + .width(Length::Fill) + .height(Length::Fill) + .max_width(300); + + let top = Container::new(right_column) + .width(Length::Fill) + .height(Length::Fill); + + let logout = neat_button( + logout_button, + i18n.get("char_selection.logout"), + FILL_FRAC_ONE, + button_style, + Some(Message::Logout), + ); + + let enter_world = neat_button( + enter_world_button, + i18n.get("char_selection.enter_world"), + FILL_FRAC_ONE, + button_style, + Some(Message::EnterWorld), + ); + + let bottom = Row::with_children(vec![ + Container::new(logout) + .width(Length::Fill) + .height(Length::Units(40)) + .align_y(Align::End) + .into(), + Container::new(enter_world) + .width(Length::Fill) + .height(Length::Units(60)) + .center_x() + .align_y(Align::End) + .into(), + Space::new(Length::Fill, Length::Shrink).into(), + ]); + + Column::with_children(vec![top.into(), bottom.into()]) + .width(Length::Fill) + .height(Length::Fill) + }, + Mode::Create { + name, + body, + loadout, + tool, + name_input, + back_button, + create_button, + } => { + let top_row = Row::with_children(vec![]); + let bottom_row = Row::with_children(vec![]); + + Column::with_children(vec![top_row.into(), bottom_row.into()]) + .width(Length::Fill) + .height(Length::Fill) + }, + }; + + Container::new( + Column::with_children(vec![version.into(), content.into()]) + .spacing(3) + .width(Length::Fill) + .height(Length::Fill), + ) + .padding(3) + .into() + } + + fn update(&mut self, message: Message, events: &mut Vec, settings: &Settings) { + let servers = &settings.networking.servers; + + //match message { } + } + + pub fn selected_character(&self) -> Option<&CharacterItem> { + match self.mode { + // TODO + Mode::Select { .. } => None, + // TODO + Mode::Create { .. } => None, + } + } +} + +pub struct CharSelectionUi { + ui: Ui, + controls: Controls, +} + +impl CharSelectionUi { + pub fn new(global_state: &mut GlobalState) -> Self { + // Load language + let i18n = load_expect::(&i18n_asset_key( + &global_state.settings.language.selected_language, + )); + + // TODO: don't add default font twice + let font = { + use std::io::Read; + let mut buf = Vec::new(); + common::assets::load_file("voxygen.font.haxrcorp_4089_cyrillic_altgr_extended", &[ + "ttf", + ]) + .unwrap() + .read_to_end(&mut buf) + .unwrap(); + ui::ice::Font::try_from_vec(buf).unwrap() + }; + + let mut ui = Ui::new(&mut global_state.window, font).unwrap(); + + let fonts = Fonts::load(&i18n.fonts, &mut ui).expect("Impossible to load fonts"); + + let controls = Controls::new( + fonts, + Imgs::load(&mut ui).expect("Failed to load images"), + i18n, + ); + + Self { ui, controls } + } + + pub fn selected_character(&self) -> Option<&CharacterItem> { + self.controls.selected_character() + } + + // TODO + pub fn get_loadout(&mut self) -> Option { + // TODO: don't clone + /*match &mut self.mode { + Mode::Select(character_list) => { + if let Some(data) = character_list { + data.get(self.selected_character).map(|c| c.loadout.clone()) + } else { + None + } + }, + Mode::Create { loadout, tool, .. } => { + loadout.active_item = tool.map(|tool| comp::ItemConfig { + item: (*load_expect::(tool)).clone(), + ability1: None, + ability2: None, + ability3: None, + block_ability: None, + dodge_ability: None, + }); + loadout.chest = Some(assets::load_expect_cloned( + "common.items.armor.starter.rugged_chest", + )); + loadout.pants = Some(assets::load_expect_cloned( + "common.items.armor.starter.rugged_pants", + )); + loadout.foot = Some(assets::load_expect_cloned( + "common.items.armor.starter.sandals_0", + )); + Some(loadout.clone()) + }, + }*/ + None + } + + pub fn handle_event(&mut self, event: window::Event) -> bool { + match event { + window::Event::IcedUi(event) => { + self.ui.handle_event(event); + true + }, + window::Event::MouseButton(_, window::PressState::Pressed) => { + // TODO: implement this with iced + // !self.ui.no_widget_capturing_mouse() + false + }, + _ => false, + } + } + + pub fn maintain(&mut self, global_state: &mut GlobalState, client: &mut Client) -> Vec { + let mut events = Vec::new(); + + let (messages, _) = self.ui.maintain( + self.controls.view(&global_state.settings), + global_state.window.renderer_mut(), + ); + + messages.into_iter().for_each(|message| { + self.controls + .update(message, &mut events, &global_state.settings) + }); + + events + } + + // TODO: do we need globals + pub fn render(&self, renderer: &mut Renderer) { self.ui.render(renderer); } +} diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index a55ac6f5cb..acd374119a 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -17,8 +17,6 @@ use crate::{ use iced::{button, text_input, Align, Column, Container, Length, Row, Space, Text, TextInput}; use vek::*; -pub const TEXT_COLOR: iced::Color = iced::Color::from_rgb(1.0, 1.0, 1.0); -pub const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1.0, 0.2); const FILL_FRAC_ONE: f32 = 0.77; const FILL_FRAC_TWO: f32 = 0.53; const INPUT_WIDTH: u16 = 250; diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index bca5c332fc..629803e53c 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -9,19 +9,19 @@ use crate::{ ui::{ self, fonts::IcedFonts as Fonts, - ice::{Element, IcedUi as Ui}, + ice::{style, widget, Element, Font, IcedUi as Ui}, img_ids::{ImageGraphic, VoxelGraphic}, Graphic, }, GlobalState, }; +use iced::{text_input, Column, Container, HorizontalAlignment, Length}; //ImageFrame, Tooltip, use crate::settings::Settings; use common::assets::Asset; use image::DynamicImage; use rand::{seq::SliceRandom, thread_rng, Rng}; use std::time::Duration; -use ui::ice::widget; // TODO: what is this? (showed up in rebase) //const COL1: Color = Color::Rgba(0.07, 0.1, 0.1, 0.9); @@ -30,8 +30,9 @@ use ui::ice::widget; /*const UI_MAIN: Color = Color::Rgba(0.61, 0.70, 0.70, 1.0); // Greenish Blue const UI_HIGHLIGHT_0: Color = Color::Rgba(0.79, 1.09, 1.09, 1.0);*/ -use iced::{text_input, Column, Container, HorizontalAlignment, Length}; -use ui::ice::style; +pub const TEXT_COLOR: iced::Color = iced::Color::from_rgb(1.0, 1.0, 1.0); +pub const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1.0, 0.2); + image_ids_ice! { struct Imgs { @@ -227,8 +228,8 @@ impl Controls { let button_style = style::button::Style::new(self.imgs.button) .hover_image(self.imgs.button_hover) .press_image(self.imgs.button_press) - .text_color(login::TEXT_COLOR) - .disabled_text_color(login::DISABLED_TEXT_COLOR); + .text_color(TEXT_COLOR) + .disabled_text_color(DISABLED_TEXT_COLOR); let version = iced::Text::new(&self.version) .size(self.fonts.cyri.scale(15)) @@ -432,7 +433,6 @@ pub struct MainMenuUi { ui: Ui, // TODO: re add this // tip_no: u16, - pub show_iced: bool, controls: Controls, } @@ -453,7 +453,7 @@ impl<'a> MainMenuUi { .unwrap() .read_to_end(&mut buf) .unwrap(); - ui::ice::Font::try_from_vec(buf).unwrap() + Font::try_from_vec(buf).unwrap() }; let mut ui = Ui::new(&mut global_state.window, font).unwrap(); diff --git a/voxygen/src/ui/ice/renderer/style/button.rs b/voxygen/src/ui/ice/renderer/style/button.rs index 20f02a5264..4c4de0e58b 100644 --- a/voxygen/src/ui/ice/renderer/style/button.rs +++ b/voxygen/src/ui/ice/renderer/style/button.rs @@ -6,6 +6,7 @@ struct Background { default: image::Handle, hover: image::Handle, press: image::Handle, + color: Color, } impl Background { @@ -14,6 +15,7 @@ impl Background { default: image, hover: image, press: image, + color: Color::WHITE, } } } @@ -58,6 +60,15 @@ impl Style { self } + // TODO: this needs to be refactored since the color isn't used if there is no + // background + pub fn image_color(mut self, color: Color) -> Self { + if let Some(background) = &mut self.background { + background.color = color; + } + self + } + pub fn text_color(mut self, color: Color) -> Self { self.enabled_text = color; self diff --git a/voxygen/src/ui/ice/renderer/widget/mod.rs b/voxygen/src/ui/ice/renderer/widget/mod.rs index 9cdb7b5e81..3cd501dcf3 100644 --- a/voxygen/src/ui/ice/renderer/widget/mod.rs +++ b/voxygen/src/ui/ice/renderer/widget/mod.rs @@ -5,6 +5,7 @@ mod column; mod compound_graphic; mod container; mod image; +mod overlay; mod row; mod scrollable; mod slider; diff --git a/voxygen/src/ui/ice/renderer/widget/overlay.rs b/voxygen/src/ui/ice/renderer/widget/overlay.rs new file mode 100644 index 0000000000..8fac3ed675 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/overlay.rs @@ -0,0 +1,36 @@ +use super::super::{super::widget::overlay, IcedRenderer, Primitive}; +use iced::{mouse::Interaction, Element, Layout, Point, Rectangle}; + +const BORDER_SIZE: u16 = 8; + +impl overlay::Renderer for IcedRenderer { + fn draw( + &mut self, + defaults: &Self::Defaults, + _bounds: Rectangle, + cursor_position: Point, + over: &Element<'_, M, Self>, + over_layout: Layout<'_>, + under: &Element<'_, M, Self>, + under_layout: Layout<'_>, + ) -> Self::Output { + let (under, under_mouse_interaction) = + under.draw(self, defaults, under_layout, cursor_position); + + let (over, over_mouse_interaction) = + over.draw(self, defaults, over_layout, cursor_position); + + // TODO: this isn't perfect but should be obselete when iced gets layer support + let mouse_interaction = if over_mouse_interaction == Interaction::Idle { + under_mouse_interaction + } else { + over_mouse_interaction + }; + + let prim = Primitive::Group { + primitives: vec![under, over], + }; + + (prim, mouse_interaction) + } +} diff --git a/voxygen/src/ui/ice/widget/mod.rs b/voxygen/src/ui/ice/widget/mod.rs index 18034b58ec..b7db600fea 100644 --- a/voxygen/src/ui/ice/widget/mod.rs +++ b/voxygen/src/ui/ice/widget/mod.rs @@ -3,6 +3,7 @@ pub mod background_container; pub mod compound_graphic; pub mod fill_text; pub mod image; +pub mod overlay; pub mod stack; pub use self::{ @@ -10,4 +11,5 @@ pub use self::{ background_container::{BackgroundContainer, Padding}, fill_text::FillText, image::Image, + overlay::Overlay, }; diff --git a/voxygen/src/ui/ice/widget/overlay.rs b/voxygen/src/ui/ice/widget/overlay.rs new file mode 100644 index 0000000000..5cc0cc96ba --- /dev/null +++ b/voxygen/src/ui/ice/widget/overlay.rs @@ -0,0 +1,222 @@ +use iced::{ + layout, mouse, Align, Clipboard, Element, Event, Hasher, Layout, Length, Point, Rectangle, + Size, Widget, +}; +use std::hash::Hash; + +/// A widget used to overlay one widget on top of another +/// Layout behaves similar to the iced::Container widget +/// Manages filtering out mouse input for the back widget if the mouse is over +/// the front widget +/// Alignment and padding is used for the front widget +pub struct Overlay<'a, M, R: self::Renderer> { + padding: u16, + width: Length, + height: Length, + max_width: u32, + max_height: u32, + horizontal_alignment: Align, + vertical_alignment: Align, + over: Element<'a, M, R>, + under: Element<'a, M, R>, + // add style etc as needed +} + +impl<'a, M, R> Overlay<'a, M, R> +where + R: self::Renderer, +{ + pub fn new(over: O, under: U) -> Self + where + O: Into>, + U: Into>, + { + Self { + padding: 0, + width: Length::Shrink, + height: Length::Shrink, + max_width: u32::MAX, + max_height: u32::MAX, + horizontal_alignment: Align::Start, + vertical_alignment: Align::Start, + over: over.into(), + under: under.into(), + } + } + + pub fn padding(mut self, pad: u16) -> Self { + self.padding = pad; + self + } + + pub fn width(mut self, width: Length) -> Self { + self.width = width; + self + } + + pub fn height(mut self, height: Length) -> Self { + self.height = height; + self + } + + pub fn max_width(mut self, max_width: u32) -> Self { + self.max_width = max_width; + self + } + + pub fn max_height(mut self, max_height: u32) -> Self { + self.max_height = max_height; + self + } + + pub fn align_x(mut self, align_x: Align) -> Self { + self.horizontal_alignment = align_x; + self + } + + pub fn align_y(mut self, align_y: Align) -> Self { + self.vertical_alignment = align_y; + self + } + + pub fn center_x(mut self) -> Self { + self.horizontal_alignment = Align::Center; + self + } + + pub fn center_y(mut self) -> Self { + self.vertical_alignment = Align::Center; + self + } +} + +impl<'a, M, R> Widget for Overlay<'a, M, R> +where + R: self::Renderer, +{ + fn width(&self) -> Length { self.width } + + fn height(&self) -> Length { self.height } + + fn layout(&self, renderer: &R, limits: &layout::Limits) -> layout::Node { + let padding = self.padding as f32; + + let limits = limits + .loose() + .max_width(self.max_width) + .max_height(self.max_height) + .width(self.width) + .height(self.height); + + let under = self.under.layout(renderer, &limits.loose()); + let under_size = under.size(); + + let limits = limits.pad(padding); + let mut over = self.under.layout(renderer, &limits.loose()); + let over_size = over.size(); + + let size = limits.resolve(Size { + width: under_size.width.max(over_size.width), + height: under_size.width.max(over_size.width), + }); + + over.move_to(Point::new(padding, padding)); + over.align(self.horizontal_alignment, self.vertical_alignment, size); + + layout::Node::with_children(size.pad(padding), vec![over, under]) + } + + fn on_event( + &mut self, + event: Event, + layout: Layout<'_>, + cursor_position: Point, + messages: &mut Vec, + renderer: &R, + clipboard: Option<&dyn Clipboard>, + ) { + self.over.on_event( + event.clone(), + layout, + cursor_position, + messages, + renderer, + clipboard, + ); + + // If mouse press check if over the overlay widget before sending to under + // widget + if !matches!(&event, Event::Mouse(mouse::Event::ButtonPressed(_))) + || !layout + .children() + .next() + .unwrap() + .bounds() + .contains(cursor_position) + { + self.under.on_event( + event, + layout, + cursor_position, + messages, + renderer, + clipboard, + ); + } + } + + fn draw( + &self, + renderer: &mut R, + defaults: &R::Defaults, + layout: Layout<'_>, + cursor_position: Point, + ) -> R::Output { + let mut children = layout.children(); + renderer.draw( + defaults, + layout.bounds(), + cursor_position, + &self.over, + children.next().unwrap(), + &self.under, + children.next().unwrap(), + ) + } + + fn hash_layout(&self, state: &mut Hasher) { + struct Marker; + std::any::TypeId::of::().hash(state); + + self.padding.hash(state); + self.width.hash(state); + self.height.hash(state); + self.max_width.hash(state); + self.max_height.hash(state); + + self.over.hash_layout(state); + self.under.hash_layout(state); + } +} + +pub trait Renderer: iced::Renderer { + fn draw( + &mut self, + defaults: &Self::Defaults, + bounds: Rectangle, + cursor_position: Point, + //style: &self::Style, + over: &Element<'_, M, Self>, + over_layout: Layout<'_>, + under: &Element<'_, M, Self>, + under_layout: Layout<'_>, + ) -> Self::Output; +} + +impl<'a, M, R> From> for Element<'a, M, R> +where + R: 'a + self::Renderer, + M: 'a, +{ + fn from(overlay: Overlay<'a, M, R>) -> Element<'a, M, R> { Element::new(overlay) } +} From c4eb90d6f84e253720a1f0617ca7256177d54379 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sat, 27 Jun 2020 13:44:43 -0400 Subject: [PATCH 32/61] Update to latest iced git, rebase fixes --- Cargo.lock | 47 +++- voxygen/Cargo.toml | 3 +- voxygen/src/hud/chat.rs | 2 +- voxygen/src/menu/char_selection/mod.rs | 3 +- voxygen/src/menu/char_selection/ui/mod.rs | 28 ++- voxygen/src/menu/main/mod.rs | 2 +- voxygen/src/menu/main/ui/mod.rs | 24 +- voxygen/src/run.rs | 9 +- voxygen/src/ui/event.rs | 4 +- voxygen/src/ui/fonts.rs | 2 +- voxygen/src/ui/graphic/mod.rs | 2 +- voxygen/src/ui/ice/cache.rs | 1 + voxygen/src/ui/ice/clipboard.rs | 12 - voxygen/src/ui/ice/mod.rs | 24 +- voxygen/src/ui/ice/renderer/mod.rs | 26 +- voxygen/src/ui/ice/renderer/widget/text.rs | 3 +- voxygen/src/ui/ice/winit_conversion.rs | 274 --------------------- voxygen/src/ui/mod.rs | 2 +- voxygen/src/window.rs | 14 +- 19 files changed, 156 insertions(+), 326 deletions(-) delete mode 100644 voxygen/src/ui/ice/clipboard.rs delete mode 100644 voxygen/src/ui/ice/winit_conversion.rs diff --git a/Cargo.lock b/Cargo.lock index 1032afa1ce..9dd0bbee7d 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1845,6 +1845,12 @@ dependencies = [ "xml-rs", ] +[[package]] +name = "glam" +version = "0.8.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b00572b5b10070ac495be20a25b4c8d379d20bcdec8ea0c870022b620ec79b20" + [[package]] name = "glob" version = "0.3.0" @@ -2152,29 +2158,63 @@ dependencies = [ [[package]] name = "iced_core" version = "0.2.1" -source = "git+https://github.com/Imberflur/iced#cf514910c2c0db8633377d2719b244e388774cee" +source = "git+https://github.com/hecrj/iced?rev=b5d842f#b5d842f877145c78f5d595a87cc1927bb6f5b86a" [[package]] name = "iced_futures" version = "0.1.2" -source = "git+https://github.com/Imberflur/iced#cf514910c2c0db8633377d2719b244e388774cee" +source = "git+https://github.com/hecrj/iced?rev=b5d842f#b5d842f877145c78f5d595a87cc1927bb6f5b86a" dependencies = [ "futures 0.3.5", "log", "wasm-bindgen-futures", ] +[[package]] +name = "iced_graphics" +version = "0.1.0" +source = "git+https://github.com/hecrj/iced?rev=b5d842f#b5d842f877145c78f5d595a87cc1927bb6f5b86a" +dependencies = [ + "bytemuck", + "glam", + "iced_native", + "iced_style", + "raw-window-handle", +] + [[package]] name = "iced_native" version = "0.2.2" -source = "git+https://github.com/Imberflur/iced#cf514910c2c0db8633377d2719b244e388774cee" +source = "git+https://github.com/hecrj/iced?rev=b5d842f#b5d842f877145c78f5d595a87cc1927bb6f5b86a" dependencies = [ "iced_core", "iced_futures", + "num-traits 0.2.12", "twox-hash", "unicode-segmentation", ] +[[package]] +name = "iced_style" +version = "0.1.0" +source = "git+https://github.com/hecrj/iced?rev=b5d842f#b5d842f877145c78f5d595a87cc1927bb6f5b86a" +dependencies = [ + "iced_core", +] + +[[package]] +name = "iced_winit" +version = "0.1.1" +source = "git+https://github.com/hecrj/iced?rev=b5d842f#b5d842f877145c78f5d595a87cc1927bb6f5b86a" +dependencies = [ + "iced_graphics", + "iced_native", + "log", + "winapi 0.3.9", + "window_clipboard", + "winit", +] + [[package]] name = "ident_case" version = "1.0.1" @@ -5212,6 +5252,7 @@ dependencies = [ "guillotiere", "hashbrown 0.7.2", "iced_native", + "iced_winit", "image", "inline_tweak", "itertools", diff --git a/voxygen/Cargo.toml b/voxygen/Cargo.toml index ceab09c555..21fd6d0e76 100644 --- a/voxygen/Cargo.toml +++ b/voxygen/Cargo.toml @@ -35,7 +35,8 @@ winit = {version = "0.22.2", features = ["serde"]} conrod_core = {git = "https://gitlab.com/veloren/conrod.git", branch="copypasta_0.7"} conrod_winit = {git = "https://gitlab.com/veloren/conrod.git", branch="copypasta_0.7"} euc = {git = "https://github.com/zesterer/euc.git"} -iced = {package = "iced_native", git = "https://github.com/Imberflur/iced"} +iced = {package = "iced_native", git = "https://github.com/hecrj/iced", rev = "b5d842f"} +iced_winit = {git = "https://github.com/hecrj/iced", rev = "b5d842f"} window_clipboard = "0.1.1" glyph_brush = "0.7.0" diff --git a/voxygen/src/hud/chat.rs b/voxygen/src/hud/chat.rs index a102f92f01..9fe39ed1cf 100644 --- a/voxygen/src/hud/chat.rs +++ b/voxygen/src/hud/chat.rs @@ -2,7 +2,7 @@ use super::{ img_ids::Imgs, ERROR_COLOR, FACTION_COLOR, GROUP_COLOR, INFO_COLOR, KILL_COLOR, LOOT_COLOR, OFFLINE_COLOR, ONLINE_COLOR, REGION_COLOR, SAY_COLOR, TELL_COLOR, TEXT_COLOR, WORLD_COLOR, }; -use crate::{ui::fonts::Fonts, GlobalState, Localization}; +use crate::{i18n::Localization, ui::fonts::Fonts, GlobalState}; use client::{cmd, Client}; use common::{ comp::{ diff --git a/voxygen/src/menu/char_selection/mod.rs b/voxygen/src/menu/char_selection/mod.rs index 91dbabd584..9de731f887 100644 --- a/voxygen/src/menu/char_selection/mod.rs +++ b/voxygen/src/menu/char_selection/mod.rs @@ -197,7 +197,6 @@ impl PlayState for CharSelectionState { ); // Draw the UI to the screen. - self.char_selection_ui - .render(global_state.window.renderer_mut()); + self.char_selection_ui.render(renderer); } } diff --git a/voxygen/src/menu/char_selection/ui/mod.rs b/voxygen/src/menu/char_selection/ui/mod.rs index 42bf044b2a..c7d7fe3d2d 100644 --- a/voxygen/src/menu/char_selection/ui/mod.rs +++ b/voxygen/src/menu/char_selection/ui/mod.rs @@ -11,13 +11,13 @@ use crate::{ }; use client::Client; use common::{ - character::{CharacterItem, MAX_CHARACTERS_PER_PLAYER}, + assets::Asset, + character::{CharacterId, CharacterItem, MAX_CHARACTERS_PER_PLAYER}, comp, comp::humanoid, }; //ImageFrame, Tooltip, use crate::settings::Settings; -use common::assets::load_expect; //use std::time::Duration; //use ui::ice::widget; use iced::{ @@ -49,7 +49,7 @@ image_ids_ice! { delete_button_press: "voxygen.element.buttons.x_red_press", - name_input: "voxygen.element.misc_bg.textbox_mid", + name_input: "voxygen.element.misc_bg.textbox", // Tool Icons daggers: "voxygen.element.icons.daggers", @@ -106,7 +106,7 @@ pub enum Event { tool: Option, body: comp::Body, }, - DeleteCharacter(i32), + DeleteCharacter(CharacterId), } struct CharacterList { @@ -165,6 +165,8 @@ struct Controls { i18n: std::sync::Arc, // Voxygen version version: String, + // Alpha disclaimer + alpha: String, info_content: Option, // enter: bool, @@ -190,12 +192,14 @@ impl Controls { env!("CARGO_PKG_VERSION"), common::util::GIT_VERSION.to_string() ); + let alpha = format!("Veloren Pre-Alpha {}", env!("CARGO_PKG_VERSION"),); Self { fonts, imgs, i18n, version, + alpha, info_content: None, mode: Mode::Select { @@ -227,6 +231,18 @@ impl Controls { .width(Length::Fill) .horizontal_alignment(HorizontalAlignment::Right); + let alpha = iced::Text::new(&self.alpha) + .size(self.fonts.cyri.scale(15)) + .width(Length::Fill) + .horizontal_alignment(HorizontalAlignment::Center); + + let top_text = Row::with_children(vec![ + Space::new(Length::Fill, Length::Shrink).into(), + alpha.into(), + version.into(), + ]) + .width(Length::Fill); + let content = match &mut self.mode { Mode::Select { list, @@ -375,7 +391,7 @@ impl Controls { }; Container::new( - Column::with_children(vec![version.into(), content.into()]) + Column::with_children(vec![top_text.into(), content.into()]) .spacing(3) .width(Length::Fill) .height(Length::Fill), @@ -408,7 +424,7 @@ pub struct CharSelectionUi { impl CharSelectionUi { pub fn new(global_state: &mut GlobalState) -> Self { // Load language - let i18n = load_expect::(&i18n_asset_key( + let i18n = Localization::load_expect(&i18n_asset_key( &global_state.settings.language.selected_language, )); diff --git a/voxygen/src/menu/main/mod.rs b/voxygen/src/menu/main/mod.rs index 7e1b12dc32..2abeb9d017 100644 --- a/voxygen/src/menu/main/mod.rs +++ b/voxygen/src/menu/main/mod.rs @@ -85,7 +85,7 @@ impl PlayState for MainMenuState { Event::Close => return PlayStateResult::Shutdown, // Pass events to ui. Event::IcedUi(event) => { - self.main_menu_ui.handle_iced_event(event); + self.main_menu_ui.handle_event(event); }, // Ignore all other events. _ => {}, diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 629803e53c..4ba0a367e7 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -15,12 +15,12 @@ use crate::{ }, GlobalState, }; -use iced::{text_input, Column, Container, HorizontalAlignment, Length}; +use iced::{text_input, Column, Container, HorizontalAlignment, Length, Row, Space}; //ImageFrame, Tooltip, use crate::settings::Settings; use common::assets::Asset; use image::DynamicImage; -use rand::{seq::SliceRandom, thread_rng, Rng}; +use rand::{seq::SliceRandom, thread_rng}; use std::time::Duration; // TODO: what is this? (showed up in rebase) @@ -139,6 +139,8 @@ struct Controls { i18n: std::sync::Arc, // Voxygen version version: String, + // Alpha disclaimer + alpha: String, selected_server_index: Option, login_info: LoginInfo, @@ -182,6 +184,7 @@ impl Controls { env!("CARGO_PKG_VERSION"), common::util::GIT_VERSION.to_string() ); + let alpha = format!("Veloren Pre-Alpha {}", env!("CARGO_PKG_VERSION"),); let screen = /* if settings.show_disclaimer { Screen::Disclaimer { @@ -211,6 +214,7 @@ impl Controls { bg_img, i18n, version, + alpha, selected_server_index, login_info, @@ -236,6 +240,18 @@ impl Controls { .width(Length::Fill) .horizontal_alignment(HorizontalAlignment::Right); + let alpha = iced::Text::new(&self.alpha) + .size(self.fonts.cyri.scale(15)) + .width(Length::Fill) + .horizontal_alignment(HorizontalAlignment::Center); + + let top_text = Row::with_children(vec![ + Space::new(Length::Fill, Length::Shrink).into(), + alpha.into(), + version.into(), + ]) + .width(Length::Fill); + let bg_img = if matches!(&self.screen, Screen::Connecting {..}) { self.bg_img } else { @@ -274,7 +290,7 @@ impl Controls { }; Container::new( - Column::with_children(vec![version.into(), content]) + Column::with_children(vec![top_text.into(), content]) .spacing(3) .width(Length::Fill) .height(Length::Fill), @@ -465,7 +481,7 @@ impl<'a> MainMenuUi { let controls = Controls::new( fonts, Imgs::load(&mut ui).expect("Failed to load images"), - ui.add_graphic(Graphic::Image(DynamicImage::load_expect(bg_img_spec))), + ui.add_graphic(Graphic::Image(DynamicImage::load_expect(bg_img_spec), None)), i18n, &global_state.settings, ); diff --git a/voxygen/src/run.rs b/voxygen/src/run.rs index 8253440906..1c75aced18 100644 --- a/voxygen/src/run.rs +++ b/voxygen/src/run.rs @@ -36,9 +36,12 @@ pub fn run(mut global_state: GlobalState, event_loop: EventLoop) { } // iced ui events // TODO: no clone - if let winit::Event::WindowEvent { event, .. } = event.clone() { - if let Some(event) = ui::ice::window_event(event) { - global_state.window.send_event(Event::IcedUi(event)); + if let winit::event::Event::WindowEvent { event, .. } = &event { + let window = &mut global_state.window; + if let Some(event) = + ui::ice::window_event(event, window.scale_factor(), window.modifiers()) + { + window.send_event(Event::IcedUi(event)); } } diff --git a/voxygen/src/ui/event.rs b/voxygen/src/ui/event.rs index e84deca87f..56bc07e2a8 100644 --- a/voxygen/src/ui/event.rs +++ b/voxygen/src/ui/event.rs @@ -6,7 +6,7 @@ pub struct Event(pub Input); impl Event { pub fn try_from( event: &winit::event::Event<()>, - window: &glutin::ContextWrapper, + window: &winit::window::Window, ) -> Option { use conrod_winit::*; // A wrapper around the winit window that allows us to implement the trait @@ -26,7 +26,7 @@ impl Event { fn hidpi_factor(&self) -> f64 { winit::window::Window::scale_factor(&self.0) } } - convert_event!(event, &WindowRef(window.window())).map(Self) + convert_event!(event, &WindowRef(window)).map(Self) } pub fn is_keyboard_or_mouse(&self) -> bool { diff --git a/voxygen/src/ui/fonts.rs b/voxygen/src/ui/fonts.rs index d8afb3cdce..583bee4282 100644 --- a/voxygen/src/ui/fonts.rs +++ b/voxygen/src/ui/fonts.rs @@ -11,7 +11,7 @@ impl Font { pub fn new(font: &i18n::Font, ui: &mut crate::ui::Ui) -> Self { Self { metadata: font.clone(), - conrod_id: ui.new_font(crate::ui::Font::load_expect(&font.asset_key)), + conrod_id: ui.new_font(crate::ui::ice::RawFont::load_expect(&font.asset_key)), } } diff --git a/voxygen/src/ui/graphic/mod.rs b/voxygen/src/ui/graphic/mod.rs index 0e0703ad13..365ec2b7d3 100644 --- a/voxygen/src/ui/graphic/mod.rs +++ b/voxygen/src/ui/graphic/mod.rs @@ -192,7 +192,7 @@ impl GraphicCache { use image::GenericImageView; self.get_graphic(id) .and_then(|graphic| match graphic { - Graphic::Image(image) => Some(image.dimensions()), + Graphic::Image(image, _) => Some(image.dimensions()), Graphic::Voxel(segment, _, _) => { use common::vol::SizedVol; let size = segment.size(); diff --git a/voxygen/src/ui/ice/cache.rs b/voxygen/src/ui/ice/cache.rs index 3dfc25f777..778ab35bb2 100644 --- a/voxygen/src/ui/ice/cache.rs +++ b/voxygen/src/ui/ice/cache.rs @@ -109,6 +109,7 @@ impl common::assets::Asset for RawFont { fn parse( mut buf_reader: std::io::BufReader, + _specifier: &str, ) -> Result { use std::io::Read; let mut buf = Vec::new(); diff --git a/voxygen/src/ui/ice/clipboard.rs b/voxygen/src/ui/ice/clipboard.rs deleted file mode 100644 index feb78983ab..0000000000 --- a/voxygen/src/ui/ice/clipboard.rs +++ /dev/null @@ -1,12 +0,0 @@ -// Taken from https://github.com/hecrj/iced/blob/e1438774af809c2951c4c7446638500446c81111/winit/src/clipboard.rs -pub struct Clipboard(window_clipboard::Clipboard); - -impl Clipboard { - pub fn new(window: &winit::Window) -> Option { - window_clipboard::Clipboard::new(window).map(Clipboard).ok() - } -} - -impl iced::Clipboard for Clipboard { - fn content(&self) -> Option { self.0.read().ok() } -} diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index 89ab43cc37..b5464d432d 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -1,24 +1,22 @@ // tooltip_manager: TooltipManager, mod cache; -mod clipboard; pub mod component; mod renderer; pub mod widget; -mod winit_conversion; pub use cache::{Font, FontId, RawFont}; pub use graphic::{Id, Rotation}; pub use iced::Event; +pub use iced_winit::conversion::window_event; pub use renderer::{style, IcedRenderer}; -pub use winit_conversion::window_event; use super::{ graphic::{self, Graphic}, scale::{Scale, ScaleMode}, }; use crate::{render::Renderer, window::Window, Error}; -use clipboard::Clipboard; use iced::{mouse, Cache, Size, UserInterface}; +use iced_winit::Clipboard; use vek::*; pub type Element<'a, M> = iced::Element<'a, M, IcedRenderer>; @@ -28,6 +26,7 @@ pub struct IcedUi { cache: Option, events: Vec, clipboard: Clipboard, + cursor_position: Vec2, // Scaling of the ui scale: Scale, window_resized: Option>, @@ -46,6 +45,7 @@ impl IcedUi { events: Vec::new(), // TODO: handle None clipboard: Clipboard::new(window.window()).unwrap(), + cursor_position: Vec2::zero(), scale, window_resized: None, }) @@ -74,6 +74,13 @@ impl IcedUi { Event::Mouse(mouse::Event::CursorMoved { x, y }) => { // TODO: return f32 here let scale = self.scale.scale_factor_logical() as f32; + // TODO: determine why iced moved cursor position out of the `Cache` and if we + // may need to handle this in a different way to address + // whatever issue iced was trying to address + self.cursor_position = Vec2 { + x: x * scale, + y: y * scale, + }; self.events.push(Event::Mouse(mouse::Event::CursorMoved { x: x * scale, y: y * scale, @@ -129,6 +136,11 @@ impl IcedUi { } } + let cursor_position = iced::Point { + x: self.cursor_position.x, + y: self.cursor_position.y, + }; + // TODO: convert to f32 at source let window_size = self.scale.scaled_window_size().map(|e| e as f32); @@ -141,11 +153,13 @@ impl IcedUi { let messages = user_interface.update( self.events.drain(..), + cursor_position, Some(&self.clipboard), &mut self.renderer, ); - let (primitive, mouse_interaction) = user_interface.draw(&mut self.renderer); + let (primitive, mouse_interaction) = + user_interface.draw(&mut self.renderer, cursor_position); self.cache = Some(user_interface.into_cache()); diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index da6568e9e2..37b7703aa2 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -20,7 +20,7 @@ use crate::{ Error, }; use common::util::srgba_to_linear; -use std::ops::Range; +use std::{convert::TryInto, ops::Range}; use vek::*; enum DrawKind { @@ -29,7 +29,7 @@ enum DrawKind { Plain, } enum DrawCommand { - Draw { kind: DrawKind, verts: Range }, + Draw { kind: DrawKind, verts: Range }, Scissor(Aabr), WorldPos(Option), } @@ -37,14 +37,30 @@ impl DrawCommand { fn image(verts: Range, id: TexId) -> DrawCommand { DrawCommand::Draw { kind: DrawKind::Image(id), - verts, + // TODO: move conversion into helper method so we don't have to write it out so many + // times + verts: verts + .start + .try_into() + .expect("Vertex count for UI rendering does not fit in a u32!") + ..verts + .end + .try_into() + .expect("Vertex count for UI rendering does not fit in a u32!"), } } fn plain(verts: Range) -> DrawCommand { DrawCommand::Draw { kind: DrawKind::Plain, - verts, + verts: verts + .start + .try_into() + .expect("Vertex count for UI rendering does not fit in a u32!") + ..verts + .end + .try_into() + .expect("Vertex count for UI rendering does not fit in a u32!"), } } } @@ -625,7 +641,7 @@ impl IcedRenderer { DrawKind::Plain => self.cache.glyph_cache_tex(), }; let model = self.model.submodel(verts.clone()); - renderer.render_ui_element(&model, tex, scissor, globals, locals); + renderer.render_ui_element(model, tex, scissor, globals, locals); }, } } diff --git a/voxygen/src/ui/ice/renderer/widget/text.rs b/voxygen/src/ui/ice/renderer/widget/text.rs index 9caf5c11c1..7360aeaa6e 100644 --- a/voxygen/src/ui/ice/renderer/widget/text.rs +++ b/voxygen/src/ui/ice/renderer/widget/text.rs @@ -5,7 +5,8 @@ use iced::{mouse, text, Color, HorizontalAlignment, Rectangle, Size, VerticalAli impl text::Renderer for IcedRenderer { type Font = FontId; - const DEFAULT_SIZE: u16 = 20; + // TODO: expose as setting + fn default_size(&self) -> u16 { 20 } fn measure(&self, content: &str, size: u16, font: Self::Font, bounds: Size) -> (f32, f32) { // Using the physical scale might make these cached info usable below? diff --git a/voxygen/src/ui/ice/winit_conversion.rs b/voxygen/src/ui/ice/winit_conversion.rs deleted file mode 100644 index 4b69e6ded1..0000000000 --- a/voxygen/src/ui/ice/winit_conversion.rs +++ /dev/null @@ -1,274 +0,0 @@ -// Using reference impl: https://github.com/hecrj/iced/blob/e1438774af809c2951c4c7446638500446c81111/winit/src/conversion.rs -use iced::{ - keyboard::{self, KeyCode, ModifiersState}, - mouse, window, Event, -}; - -/// Converts a winit event into an iced event. -pub fn window_event(event: winit::WindowEvent) -> Option { - use winit::WindowEvent; - - match event { - WindowEvent::Resized(new_size) => { - let logical_size: winit::dpi::LogicalSize = new_size; - - Some(Event::Window(window::Event::Resized { - width: logical_size.width as u32, - height: logical_size.height as u32, - })) - }, - WindowEvent::CursorMoved { position, .. } => { - let position: winit::dpi::LogicalPosition = position; - - Some(Event::Mouse(mouse::Event::CursorMoved { - x: position.x as f32, - y: position.y as f32, - })) - }, - WindowEvent::MouseInput { button, state, .. } => { - let button = mouse_button(button); - - Some(Event::Mouse(match state { - winit::ElementState::Pressed => mouse::Event::ButtonPressed(button), - winit::ElementState::Released => mouse::Event::ButtonReleased(button), - })) - }, - WindowEvent::MouseWheel { delta, .. } => match delta { - winit::MouseScrollDelta::LineDelta(delta_x, delta_y) => { - Some(Event::Mouse(mouse::Event::WheelScrolled { - delta: mouse::ScrollDelta::Lines { - x: delta_x, - y: delta_y, - }, - })) - }, - winit::MouseScrollDelta::PixelDelta(position) => { - let position: winit::dpi::LogicalPosition = position; - Some(Event::Mouse(mouse::Event::WheelScrolled { - delta: mouse::ScrollDelta::Pixels { - x: position.x as f32, - y: position.y as f32, - }, - })) - }, - }, - WindowEvent::ReceivedCharacter(c) => { - Some(Event::Keyboard(keyboard::Event::CharacterReceived(c))) - }, - WindowEvent::KeyboardInput { - input: - winit::KeyboardInput { - virtual_keycode: Some(virtual_keycode), - state, - modifiers, - .. - }, - .. - } => Some(Event::Keyboard({ - let key_code = key_code(virtual_keycode); - let modifiers = modifiers_state(modifiers); - - match state { - winit::ElementState::Pressed => keyboard::Event::KeyPressed { - key_code, - modifiers, - }, - winit::ElementState::Released => keyboard::Event::KeyReleased { - key_code, - modifiers, - }, - } - })), - // iced also can use file hovering events but we don't need them right now - _ => None, - } -} - -// iced has a function for converting mouse cursors here - -/// Converts winit mouse button to iced mouse button -fn mouse_button(mouse_button: winit::MouseButton) -> mouse::Button { - match mouse_button { - winit::MouseButton::Left => mouse::Button::Left, - winit::MouseButton::Right => mouse::Button::Right, - winit::MouseButton::Middle => mouse::Button::Middle, - winit::MouseButton::Other(other) => mouse::Button::Other(other), - } -} - -/// Converts winit `ModifiersState` to iced `ModifiersState` -fn modifiers_state(modifiers: winit::ModifiersState) -> ModifiersState { - ModifiersState { - shift: modifiers.shift, - control: modifiers.ctrl, - alt: modifiers.alt, - logo: modifiers.logo, - } -} - -/// Converts winit `VirtualKeyCode` to iced `KeyCode` -fn key_code(virtual_keycode: winit::VirtualKeyCode) -> KeyCode { - match virtual_keycode { - winit::VirtualKeyCode::Key1 => KeyCode::Key1, - winit::VirtualKeyCode::Key2 => KeyCode::Key2, - winit::VirtualKeyCode::Key3 => KeyCode::Key3, - winit::VirtualKeyCode::Key4 => KeyCode::Key4, - winit::VirtualKeyCode::Key5 => KeyCode::Key5, - winit::VirtualKeyCode::Key6 => KeyCode::Key6, - winit::VirtualKeyCode::Key7 => KeyCode::Key7, - winit::VirtualKeyCode::Key8 => KeyCode::Key8, - winit::VirtualKeyCode::Key9 => KeyCode::Key9, - winit::VirtualKeyCode::Key0 => KeyCode::Key0, - winit::VirtualKeyCode::A => KeyCode::A, - winit::VirtualKeyCode::B => KeyCode::B, - winit::VirtualKeyCode::C => KeyCode::C, - winit::VirtualKeyCode::D => KeyCode::D, - winit::VirtualKeyCode::E => KeyCode::E, - winit::VirtualKeyCode::F => KeyCode::F, - winit::VirtualKeyCode::G => KeyCode::G, - winit::VirtualKeyCode::H => KeyCode::H, - winit::VirtualKeyCode::I => KeyCode::I, - winit::VirtualKeyCode::J => KeyCode::J, - winit::VirtualKeyCode::K => KeyCode::K, - winit::VirtualKeyCode::L => KeyCode::L, - winit::VirtualKeyCode::M => KeyCode::M, - winit::VirtualKeyCode::N => KeyCode::N, - winit::VirtualKeyCode::O => KeyCode::O, - winit::VirtualKeyCode::P => KeyCode::P, - winit::VirtualKeyCode::Q => KeyCode::Q, - winit::VirtualKeyCode::R => KeyCode::R, - winit::VirtualKeyCode::S => KeyCode::S, - winit::VirtualKeyCode::T => KeyCode::T, - winit::VirtualKeyCode::U => KeyCode::U, - winit::VirtualKeyCode::V => KeyCode::V, - winit::VirtualKeyCode::W => KeyCode::W, - winit::VirtualKeyCode::X => KeyCode::X, - winit::VirtualKeyCode::Y => KeyCode::Y, - winit::VirtualKeyCode::Z => KeyCode::Z, - winit::VirtualKeyCode::Escape => KeyCode::Escape, - winit::VirtualKeyCode::F1 => KeyCode::F1, - winit::VirtualKeyCode::F2 => KeyCode::F2, - winit::VirtualKeyCode::F3 => KeyCode::F3, - winit::VirtualKeyCode::F4 => KeyCode::F4, - winit::VirtualKeyCode::F5 => KeyCode::F5, - winit::VirtualKeyCode::F6 => KeyCode::F6, - winit::VirtualKeyCode::F7 => KeyCode::F7, - winit::VirtualKeyCode::F8 => KeyCode::F8, - winit::VirtualKeyCode::F9 => KeyCode::F9, - winit::VirtualKeyCode::F10 => KeyCode::F10, - winit::VirtualKeyCode::F11 => KeyCode::F11, - winit::VirtualKeyCode::F12 => KeyCode::F12, - winit::VirtualKeyCode::F13 => KeyCode::F13, - winit::VirtualKeyCode::F14 => KeyCode::F14, - winit::VirtualKeyCode::F15 => KeyCode::F15, - winit::VirtualKeyCode::F16 => KeyCode::F16, - winit::VirtualKeyCode::F17 => KeyCode::F17, - winit::VirtualKeyCode::F18 => KeyCode::F18, - winit::VirtualKeyCode::F19 => KeyCode::F19, - winit::VirtualKeyCode::F20 => KeyCode::F20, - winit::VirtualKeyCode::F21 => KeyCode::F21, - winit::VirtualKeyCode::F22 => KeyCode::F22, - winit::VirtualKeyCode::F23 => KeyCode::F23, - winit::VirtualKeyCode::F24 => KeyCode::F24, - winit::VirtualKeyCode::Snapshot => KeyCode::Snapshot, - winit::VirtualKeyCode::Scroll => KeyCode::Scroll, - winit::VirtualKeyCode::Pause => KeyCode::Pause, - winit::VirtualKeyCode::Insert => KeyCode::Insert, - winit::VirtualKeyCode::Home => KeyCode::Home, - winit::VirtualKeyCode::Delete => KeyCode::Delete, - winit::VirtualKeyCode::End => KeyCode::End, - winit::VirtualKeyCode::PageDown => KeyCode::PageDown, - winit::VirtualKeyCode::PageUp => KeyCode::PageUp, - winit::VirtualKeyCode::Left => KeyCode::Left, - winit::VirtualKeyCode::Up => KeyCode::Up, - winit::VirtualKeyCode::Right => KeyCode::Right, - winit::VirtualKeyCode::Down => KeyCode::Down, - winit::VirtualKeyCode::Back => KeyCode::Backspace, - winit::VirtualKeyCode::Return => KeyCode::Enter, - winit::VirtualKeyCode::Space => KeyCode::Space, - winit::VirtualKeyCode::Compose => KeyCode::Compose, - winit::VirtualKeyCode::Caret => KeyCode::Caret, - winit::VirtualKeyCode::Numlock => KeyCode::Numlock, - winit::VirtualKeyCode::Numpad0 => KeyCode::Numpad0, - winit::VirtualKeyCode::Numpad1 => KeyCode::Numpad1, - winit::VirtualKeyCode::Numpad2 => KeyCode::Numpad2, - winit::VirtualKeyCode::Numpad3 => KeyCode::Numpad3, - winit::VirtualKeyCode::Numpad4 => KeyCode::Numpad4, - winit::VirtualKeyCode::Numpad5 => KeyCode::Numpad5, - winit::VirtualKeyCode::Numpad6 => KeyCode::Numpad6, - winit::VirtualKeyCode::Numpad7 => KeyCode::Numpad7, - winit::VirtualKeyCode::Numpad8 => KeyCode::Numpad8, - winit::VirtualKeyCode::Numpad9 => KeyCode::Numpad9, - winit::VirtualKeyCode::AbntC1 => KeyCode::AbntC1, - winit::VirtualKeyCode::AbntC2 => KeyCode::AbntC2, - winit::VirtualKeyCode::Add => KeyCode::Add, - winit::VirtualKeyCode::Apostrophe => KeyCode::Apostrophe, - winit::VirtualKeyCode::Apps => KeyCode::Apps, - winit::VirtualKeyCode::At => KeyCode::At, - winit::VirtualKeyCode::Ax => KeyCode::Ax, - winit::VirtualKeyCode::Backslash => KeyCode::Backslash, - winit::VirtualKeyCode::Calculator => KeyCode::Calculator, - winit::VirtualKeyCode::Capital => KeyCode::Capital, - winit::VirtualKeyCode::Colon => KeyCode::Colon, - winit::VirtualKeyCode::Comma => KeyCode::Comma, - winit::VirtualKeyCode::Convert => KeyCode::Convert, - winit::VirtualKeyCode::Decimal => KeyCode::Decimal, - winit::VirtualKeyCode::Divide => KeyCode::Divide, - winit::VirtualKeyCode::Equals => KeyCode::Equals, - winit::VirtualKeyCode::Grave => KeyCode::Grave, - winit::VirtualKeyCode::Kana => KeyCode::Kana, - winit::VirtualKeyCode::Kanji => KeyCode::Kanji, - winit::VirtualKeyCode::LAlt => KeyCode::LAlt, - winit::VirtualKeyCode::LBracket => KeyCode::LBracket, - winit::VirtualKeyCode::LControl => KeyCode::LControl, - winit::VirtualKeyCode::LShift => KeyCode::LShift, - winit::VirtualKeyCode::LWin => KeyCode::LWin, - winit::VirtualKeyCode::Mail => KeyCode::Mail, - winit::VirtualKeyCode::MediaSelect => KeyCode::MediaSelect, - winit::VirtualKeyCode::MediaStop => KeyCode::MediaStop, - winit::VirtualKeyCode::Minus => KeyCode::Minus, - winit::VirtualKeyCode::Multiply => KeyCode::Multiply, - winit::VirtualKeyCode::Mute => KeyCode::Mute, - winit::VirtualKeyCode::MyComputer => KeyCode::MyComputer, - winit::VirtualKeyCode::NavigateForward => KeyCode::NavigateForward, - winit::VirtualKeyCode::NavigateBackward => KeyCode::NavigateBackward, - winit::VirtualKeyCode::NextTrack => KeyCode::NextTrack, - winit::VirtualKeyCode::NoConvert => KeyCode::NoConvert, - winit::VirtualKeyCode::NumpadComma => KeyCode::NumpadComma, - winit::VirtualKeyCode::NumpadEnter => KeyCode::NumpadEnter, - winit::VirtualKeyCode::NumpadEquals => KeyCode::NumpadEquals, - winit::VirtualKeyCode::OEM102 => KeyCode::OEM102, - winit::VirtualKeyCode::Period => KeyCode::Period, - winit::VirtualKeyCode::PlayPause => KeyCode::PlayPause, - winit::VirtualKeyCode::Power => KeyCode::Power, - winit::VirtualKeyCode::PrevTrack => KeyCode::PrevTrack, - winit::VirtualKeyCode::RAlt => KeyCode::RAlt, - winit::VirtualKeyCode::RBracket => KeyCode::RBracket, - winit::VirtualKeyCode::RControl => KeyCode::RControl, - winit::VirtualKeyCode::RShift => KeyCode::RShift, - winit::VirtualKeyCode::RWin => KeyCode::RWin, - winit::VirtualKeyCode::Semicolon => KeyCode::Semicolon, - winit::VirtualKeyCode::Slash => KeyCode::Slash, - winit::VirtualKeyCode::Sleep => KeyCode::Sleep, - winit::VirtualKeyCode::Stop => KeyCode::Stop, - winit::VirtualKeyCode::Subtract => KeyCode::Subtract, - winit::VirtualKeyCode::Sysrq => KeyCode::Sysrq, - winit::VirtualKeyCode::Tab => KeyCode::Tab, - winit::VirtualKeyCode::Underline => KeyCode::Underline, - winit::VirtualKeyCode::Unlabeled => KeyCode::Unlabeled, - winit::VirtualKeyCode::VolumeDown => KeyCode::VolumeDown, - winit::VirtualKeyCode::VolumeUp => KeyCode::VolumeUp, - winit::VirtualKeyCode::Wake => KeyCode::Wake, - winit::VirtualKeyCode::WebBack => KeyCode::WebBack, - winit::VirtualKeyCode::WebFavorites => KeyCode::WebFavorites, - winit::VirtualKeyCode::WebForward => KeyCode::WebForward, - winit::VirtualKeyCode::WebHome => KeyCode::WebHome, - winit::VirtualKeyCode::WebRefresh => KeyCode::WebRefresh, - winit::VirtualKeyCode::WebSearch => KeyCode::WebSearch, - winit::VirtualKeyCode::WebStop => KeyCode::WebStop, - winit::VirtualKeyCode::Yen => KeyCode::Yen, - winit::VirtualKeyCode::Copy => KeyCode::Copy, - winit::VirtualKeyCode::Paste => KeyCode::Paste, - winit::VirtualKeyCode::Cut => KeyCode::Cut, - } -} diff --git a/voxygen/src/ui/mod.rs b/voxygen/src/ui/mod.rs index 6cf668eeb7..6ff7d67c8c 100644 --- a/voxygen/src/ui/mod.rs +++ b/voxygen/src/ui/mod.rs @@ -33,7 +33,7 @@ use crate::{ #[rustfmt::skip] use ::image::GenericImageView; use cache::Cache; -use common::{assets, span, util::srgba_to_linear}; +use common::{span, util::srgba_to_linear}; use conrod_core::{ event::Input, graph::{self, Graph}, diff --git a/voxygen/src/window.rs b/voxygen/src/window.rs index 499b40777a..eba4675f8c 100644 --- a/voxygen/src/window.rs +++ b/voxygen/src/window.rs @@ -499,6 +499,7 @@ pub struct Window { pub mouse_y_inversion: bool, fullscreen: FullScreenSettings, modifiers: winit::event::ModifiersState, + scale_factor: f64, needs_refresh_resize: bool, keypress_map: HashMap, pub remapping_keybindings: Option, @@ -587,6 +588,8 @@ impl Window { channel::Receiver, ) = channel::unbounded::(); + let scale_factor = window.window().scale_factor(); + let mut this = Self { renderer: Renderer::new( device, @@ -603,6 +606,7 @@ impl Window { mouse_y_inversion: settings.gameplay.mouse_y_inversion, fullscreen: FullScreenSettings::default(), modifiers: Default::default(), + scale_factor, needs_refresh_resize: false, keypress_map, remapping_keybindings: None, @@ -922,8 +926,8 @@ impl Window { self.events .push(Event::Resize(Vec2::new(width as u32, height as u32))); }, - WindowEvent::ScaleFactorChanged { .. } => { - // TODO: Handle properly! + WindowEvent::ScaleFactorChanged { scale_factor, .. } => { + self.scale_factor = scale_factor }, WindowEvent::ReceivedCharacter(c) => self.events.push(Event::Char(c)), WindowEvent::MouseInput { button, state, .. } => { @@ -1374,7 +1378,11 @@ impl Window { self.remapping_keybindings = Some(game_input); } - pub fn window(&self) -> &winit::Window { self.window.window() } + pub fn window(&self) -> &winit::window::Window { self.window.window() } + + pub fn modifiers(&self) -> winit::event::ModifiersState { self.modifiers } + + pub fn scale_factor(&self) -> f64 { self.scale_factor } } #[derive(Copy, Clone, Hash, Eq, PartialEq, Debug, Serialize, Deserialize)] From 803a5ecea6e264456bc2986d7f28c96ce98b4c8c Mon Sep 17 00:00:00 2001 From: Imbris Date: Sat, 27 Jun 2020 16:30:43 -0400 Subject: [PATCH 33/61] Add gradient primitive to ui, adjust textbox aspect ratio, misc tweaks --- assets/voxygen/element/frames/banner.png | Bin 221 -> 0 bytes .../voxygen/element/frames/banner_bottom.png | Bin 305 -> 0 bytes voxygen/src/menu/main/ui/login.rs | 36 +++++++-------- voxygen/src/menu/main/ui/mod.rs | 2 - voxygen/src/render/mod.rs | 5 +- voxygen/src/render/pipelines/ui.rs | 43 ++++++++++++------ voxygen/src/ui/ice/renderer/mod.rs | 33 +++++++++++++- voxygen/src/ui/ice/renderer/primitive.rs | 7 +++ .../ice/renderer/widget/compound_graphic.rs | 7 +++ voxygen/src/ui/ice/widget/compound_graphic.rs | 15 +++++- 10 files changed, 108 insertions(+), 40 deletions(-) delete mode 100644 assets/voxygen/element/frames/banner.png delete mode 100644 assets/voxygen/element/frames/banner_bottom.png diff --git a/assets/voxygen/element/frames/banner.png b/assets/voxygen/element/frames/banner.png deleted file mode 100644 index 748f0d7f0ce1a2ab8a16e83bd36f87090fc7f1e5..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 221 zcmeAS@N?(olHy`uVBq!ia0vp^j0_A+5gcql)?)6S`9O-Zz$3Dlfq`2Xgc%uT&5-~K zDkOUZ`7$t6sWLD$G&3;#{12pGGBA`HFfhDIU|_JC!N4G%KPmpG8&GYsr;B4q#jQ6J zt+@^u@Hn^jUoiZ>KUH>j^om}NEpxe|kJdQ)4%=cVA=(= Oj=|H_&t;ucLK6Tlo=A8A diff --git a/assets/voxygen/element/frames/banner_bottom.png b/assets/voxygen/element/frames/banner_bottom.png deleted file mode 100644 index c82073520aa7f0a74855d2d80e9b0ec43c60d3ad..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 305 zcmeAS@N?(olHy`uVBq!ia0vp^MnIg)!3HEBKR22Tq&N#aB8wRqxP?KOkzv*x380`t zvPY0F14ET614BbI1H;e%K>8&EL#Y7+!>a@a2CEqi4C48d;*Yuk)o%B6aSW-r_2%|Q z-Ub667Ki2;4MxB7IS=MA2#L5n+jQ{dUp~zl*RC0H&9Ur~5deV+=INasYnUumn3z8G z_cgrW=;&*xVR|XFajVKAWub{3E?jHx^D!l5OX@CG6$I-5>WR7^wcAC7=}}_p78NF- zvcmp5j%JHljndqUL^{55GchqG{qa#l`&*lTYz~JfX=d#Wzp$PzO%3zTI diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index acd374119a..5b2cccc8da 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -19,7 +19,7 @@ use vek::*; const FILL_FRAC_ONE: f32 = 0.77; const FILL_FRAC_TWO: f32 = 0.53; -const INPUT_WIDTH: u16 = 250; +const INPUT_WIDTH: u16 = 280; const INPUT_TEXT_SIZE: u16 = 24; /// Login screen for the main menu @@ -95,8 +95,7 @@ impl Screen { // Note: a way to tell it to keep the height of this one piece constant and // unstreched would be nice, I suppose we could just break this out into a // column and use Length::Units - Graphic::image(imgs.banner_bottom, [500, 30], [0, 300]) - .color(Rgba::new(255, 255, 255, 240)), + Graphic::gradient(Rgba::new(0, 0, 0, 240), Rgba::zero(), [500, 30], [0, 300]), ]) .height(Length::Shrink), Text::new(intro_text).size(fonts.cyri.scale(21)), @@ -197,11 +196,10 @@ impl Banner { let input_text_size = fonts.cyri.scale(INPUT_TEXT_SIZE); let banner_content = Column::with_children(vec![ - Image::new(imgs.v_logo) - .fix_aspect_ratio() - .height(Length::FillPortion(20)) + Container::new(Image::new(imgs.v_logo).fix_aspect_ratio()) + .padding(10) + .height(Length::FillPortion(25)) .into(), - Space::new(Length::Fill, Length::FillPortion(5)).into(), Column::with_children(vec![ BackgroundContainer::new( Image::new(imgs.input_bg) @@ -216,7 +214,7 @@ impl Banner { .size(input_text_size) .on_submit(Message::FocusPassword), ) - .padding(Padding::new().horizontal(10).top(10)) + .padding(Padding::new().horizontal(7).top(5)) .into(), BackgroundContainer::new( Image::new(imgs.input_bg) @@ -232,7 +230,7 @@ impl Banner { .password() .on_submit(Message::Multiplayer), ) - .padding(Padding::new().horizontal(10).top(8)) + .padding(Padding::new().horizontal(7).top(5)) .into(), BackgroundContainer::new( Image::new(imgs.input_bg) @@ -247,12 +245,13 @@ impl Banner { .size(input_text_size) .on_submit(Message::Multiplayer), ) - .padding(Padding::new().horizontal(10).top(8)) + .padding(Padding::new().horizontal(7).top(5)) .into(), ]) - .spacing(2) - .height(Length::FillPortion(50)) + .spacing(10) + .height(Length::FillPortion(35)) .into(), + Space::new(Length::Fill, Length::FillPortion(2)).into(), Column::with_children(vec![ neat_button( &mut self.multiplayer_button, @@ -270,7 +269,8 @@ impl Banner { Some(Message::Singleplayer), ), ]) - .max_width(240) + .max_width(200) + .height(Length::FillPortion(38)) .spacing(8) .into(), ]) @@ -281,16 +281,16 @@ impl Banner { let banner = BackgroundContainer::new( CompoundGraphic::from_graphics(vec![ Graphic::image(imgs.banner_top, [138, 17], [0, 0]), - Graphic::rect(Rgba::new(0, 0, 0, 230), [130, 195], [4, 17]), - Graphic::image(imgs.banner, [130, 15], [4, 212]) - .color(Rgba::new(255, 255, 255, 230)), + Graphic::rect(Rgba::new(0, 0, 0, 230), [130, 165], [4, 17]), + // TODO: use non image gradient + Graphic::gradient(Rgba::new(0, 0, 0, 230), Rgba::zero(), [130, 50], [4, 182]), ]) .fix_aspect_ratio() .height(Length::Fill), banner_content, ) - .padding(Padding::new().horizontal(16).vertical(20)) - .max_width(330); + .padding(Padding::new().horizontal(8).vertical(15)) + .max_width(350); banner.into() } diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 4ba0a367e7..85f821e6a1 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -40,8 +40,6 @@ image_ids_ice! { bg: "voxygen.background.bg_main", - banner: "voxygen.element.frames.banner", - banner_bottom: "voxygen.element.frames.banner_bottom", banner_top: "voxygen.element.frames.banner_top", button: "voxygen.element.buttons.button", button_hover: "voxygen.element.buttons.button_hover", diff --git a/voxygen/src/render/mod.rs b/voxygen/src/render/mod.rs index c03b91d00d..61077b19a6 100644 --- a/voxygen/src/render/mod.rs +++ b/voxygen/src/render/mod.rs @@ -31,8 +31,9 @@ pub use self::{ sprite::{Instance as SpriteInstance, Locals as SpriteLocals, SpritePipeline}, terrain::{Locals as TerrainLocals, TerrainPipeline}, ui::{ - create_quad as create_ui_quad, create_tri as create_ui_tri, Locals as UiLocals, - Mode as UiMode, UiPipeline, + create_quad as create_ui_quad, + create_quad_vert_gradient as create_ui_quad_vert_gradient, create_tri as create_ui_tri, + Locals as UiLocals, Mode as UiMode, UiPipeline, }, GlobalModel, Globals, Light, Shadow, }, diff --git a/voxygen/src/render/pipelines/ui.rs b/voxygen/src/render/pipelines/ui.rs index aed98810dd..8a39625c6e 100644 --- a/voxygen/src/render/pipelines/ui.rs +++ b/voxygen/src/render/pipelines/ui.rs @@ -87,24 +87,37 @@ impl Mode { } } -#[allow(clippy::many_single_char_names)] pub fn create_quad( rect: Aabr, uv_rect: Aabr, color: Rgba, mode: Mode, ) -> Quad { + create_quad_vert_gradient(rect, uv_rect, color, color, mode) +} + +#[allow(clippy::many_single_char_names)] +pub fn create_quad_vert_gradient( + rect: Aabr, + uv_rect: Aabr, + top_color: Rgba, + bottom_color: Rgba, + mode: Mode, +) -> Quad { + let top_color = top_color.into_array(); + let bottom_color = bottom_color.into_array(); + let center = if let Mode::ImageSourceNorth = mode { uv_rect.center().into_array() } else { rect.center().into_array() }; let mode_val = mode.value(); - let v = |pos, uv| Vertex { + let v = |pos, uv, color| Vertex { pos, uv, center, - color: color.into_array(), + color, mode: mode_val, }; let aabr_to_lbrt = |aabr: Aabr| (aabr.min.x, aabr.min.y, aabr.max.x, aabr.max.y); @@ -114,22 +127,22 @@ pub fn create_quad( match (uv_b > uv_t, uv_l > uv_r) { (true, true) => Quad::new( - v([r, t], [uv_l, uv_b]), - v([l, t], [uv_l, uv_t]), - v([l, b], [uv_r, uv_t]), - v([r, b], [uv_r, uv_b]), + v([r, t], [uv_l, uv_b], top_color), + v([l, t], [uv_l, uv_t], top_color), + v([l, b], [uv_r, uv_t], bottom_color), + v([r, b], [uv_r, uv_b], bottom_color), ), (false, false) => Quad::new( - v([r, t], [uv_l, uv_b]), - v([l, t], [uv_l, uv_t]), - v([l, b], [uv_r, uv_t]), - v([r, b], [uv_r, uv_b]), + v([r, t], [uv_l, uv_b], top_color), + v([l, t], [uv_l, uv_t], top_color), + v([l, b], [uv_r, uv_t], bottom_color), + v([r, b], [uv_r, uv_b], bottom_color), ), _ => Quad::new( - v([r, t], [uv_r, uv_t]), - v([l, t], [uv_l, uv_t]), - v([l, b], [uv_l, uv_b]), - v([r, b], [uv_r, uv_b]), + v([r, t], [uv_r, uv_t], top_color), + v([l, t], [uv_l, uv_t], top_color), + v([l, b], [uv_l, uv_b], bottom_color), + v([r, b], [uv_r, uv_b], bottom_color), ), } } diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index 37b7703aa2..9a7b50bc51 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -15,7 +15,8 @@ use super::{ }; use crate::{ render::{ - create_ui_quad, Consts, DynamicModel, Globals, Mesh, Renderer, UiLocals, UiMode, UiPipeline, + create_ui_quad, create_ui_quad_vert_gradient, Consts, DynamicModel, Globals, Mesh, + Renderer, UiLocals, UiMode, UiPipeline, }, Error, }; @@ -454,6 +455,36 @@ impl IcedRenderer { self.mesh .push_quad(create_ui_quad(gl_aabr, uv_aabr, color, UiMode::Image)); }, + Primitive::Gradient { + bounds, + top_linear_color, + bottom_linear_color, + } => { + // Don't draw a transparent rectangle. + if top_linear_color[3] == 0.0 && bottom_linear_color[3] == 0.0 { + return; + } + + self.switch_state(State::Plain); + + let gl_aabr = self.gl_aabr(iced::Rectangle { + x: bounds.x + offset.x as f32, + y: bounds.y + offset.y as f32, + ..bounds + }); + + self.mesh.push_quad(create_ui_quad_vert_gradient( + gl_aabr, + Aabr { + min: Vec2::zero(), + max: Vec2::zero(), + }, + top_linear_color, + bottom_linear_color, + UiMode::Geometry, + )); + }, + Primitive::Rectangle { bounds, linear_color, diff --git a/voxygen/src/ui/ice/renderer/primitive.rs b/voxygen/src/ui/ice/renderer/primitive.rs index 65125f22a6..05c08c9969 100644 --- a/voxygen/src/ui/ice/renderer/primitive.rs +++ b/voxygen/src/ui/ice/renderer/primitive.rs @@ -10,6 +10,13 @@ pub enum Primitive { bounds: iced::Rectangle, color: vek::Rgba, }, + // A vertical gradient + // TODO: could be combined with rectangle + Gradient { + bounds: iced::Rectangle, + top_linear_color: vek::Rgba, + bottom_linear_color: vek::Rgba, + }, Rectangle { bounds: iced::Rectangle, linear_color: vek::Rgba, diff --git a/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs b/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs index f28adb7bfc..797fd916d7 100644 --- a/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs +++ b/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs @@ -29,6 +29,13 @@ impl compound_graphic::Renderer for IcedRenderer { bounds, linear_color: srgba_to_linear(color.map(|e| e as f32 / 255.0)), }, + GraphicKind::Gradient(top_color, bottom_color) => Primitive::Gradient { + bounds, + top_linear_color: srgba_to_linear(top_color.map(|e| e as f32 / 255.0)), + bottom_linear_color: srgba_to_linear( + bottom_color.map(|e| e as f32 / 255.0), + ), + }, }) .collect(), }, diff --git a/voxygen/src/ui/ice/widget/compound_graphic.rs b/voxygen/src/ui/ice/widget/compound_graphic.rs index 7382edabde..167e175734 100644 --- a/voxygen/src/ui/ice/widget/compound_graphic.rs +++ b/voxygen/src/ui/ice/widget/compound_graphic.rs @@ -11,9 +11,10 @@ use vek::{Aabr, Rgba, Vec2}; // TODO: design trait to interface with background container #[derive(Copy, Clone)] pub enum GraphicKind { - // TODO: if there is a use case, allow coloring individual images Image(Handle, Rgba), Color(Rgba), + /// Vertical gradient + Gradient(Rgba, Rgba), } // TODO: consider faculties for composing compound graphics (if a use case pops @@ -48,11 +49,21 @@ impl Graphic { match &mut self.kind { GraphicKind::Image(_, c) => *c = color, GraphicKind::Color(c) => *c = color, + // Not relevant here + GraphicKind::Gradient(_, _) => (), } self } - // TODO: consider removing color here + pub fn gradient( + top_color: Rgba, + bottom_color: Rgba, + size: [u16; 2], + offset: [u16; 2], + ) -> Self { + Self::new(GraphicKind::Gradient(top_color, bottom_color), size, offset) + } + pub fn rect(color: Rgba, size: [u16; 2], offset: [u16; 2]) -> Self { Self::new(GraphicKind::Color(color), size, offset) } From 1583929e00a322e8a674273e233bb53107a4e1dc Mon Sep 17 00:00:00 2001 From: Imbris Date: Sat, 27 Jun 2020 17:43:13 -0400 Subject: [PATCH 34/61] Add tabbing between fields in login screen --- voxygen/src/menu/main/ui/mod.rs | 35 ++++++++++++++++++++++++++++++++- 1 file changed, 34 insertions(+), 1 deletion(-) diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 85f821e6a1..8b8b28bdcd 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -441,6 +441,25 @@ impl Controls { } } } + + fn tab(&mut self) { + if let Screen::Login { screen, .. } = &mut self.screen { + // TODO: add select all function in iced + if screen.banner.username.is_focused() { + screen.banner.username = iced::text_input::State::new(); + screen.banner.password = iced::text_input::State::focused(); + screen.banner.password.move_cursor_to_end(); + } else if screen.banner.password.is_focused() { + screen.banner.password = iced::text_input::State::new(); + screen.banner.server = iced::text_input::State::focused(); + screen.banner.server.move_cursor_to_end(); + } else if screen.banner.server.is_focused() { + screen.banner.server = iced::text_input::State::new(); + screen.banner.username = iced::text_input::State::focused(); + screen.banner.username.move_cursor_to_end(); + } + } + } } pub struct MainMenuUi { @@ -497,7 +516,21 @@ impl<'a> MainMenuUi { pub fn cancel_connection(&mut self) { self.controls.exit_connect_screen(); } - pub fn handle_event(&mut self, event: ui::ice::Event) { self.ui.handle_event(event); } + pub fn handle_event(&mut self, event: ui::ice::Event) { + // Tab for input fields + use iced::keyboard; + if matches!( + &event, + iced::Event::Keyboard(keyboard::Event::KeyPressed { + key_code: keyboard::KeyCode::Tab, + .. + }) + ) { + self.controls.tab(); + } + + self.ui.handle_event(event); + } pub fn maintain(&mut self, global_state: &mut GlobalState, dt: Duration) -> Vec { let mut events = Vec::new(); From 5e913d46c7318102c2c206c8963943f72d333f3c Mon Sep 17 00:00:00 2001 From: Imbris Date: Sun, 28 Jun 2020 13:46:11 -0400 Subject: [PATCH 35/61] Basic structure of character selection --- assets/voxygen/element/frames/gray/corner.png | Bin 0 -> 142 bytes assets/voxygen/element/frames/gray/edge.png | Bin 0 -> 133 bytes .../voxygen/element/frames/server_frame.vox | Bin 56107 -> 0 bytes assets/voxygen/element/frames/window_4.vox | Bin 2440 -> 0 bytes assets/voxygen/i18n/de_DE.ron | 4 +- assets/voxygen/i18n/en.ron | 2 +- assets/voxygen/i18n/es_ES.ron | 2 +- assets/voxygen/i18n/fr_FR.ron | 2 +- assets/voxygen/i18n/it_IT.ron | 2 +- assets/voxygen/i18n/pt_PT.ron | 2 +- assets/voxygen/i18n/ru_RU.ron | 2 +- voxygen/src/menu/char_selection/mod.rs | 28 +- voxygen/src/menu/char_selection/ui/mod.rs | 339 +++++++++++++----- voxygen/src/menu/main/ui/connecting.rs | 12 +- voxygen/src/menu/main/ui/disclaimer.rs | 12 +- voxygen/src/menu/main/ui/login.rs | 14 +- voxygen/src/menu/main/ui/mod.rs | 2 +- voxygen/src/menu/main/ui/servers.rs | 12 +- voxygen/src/ui/ice/component/neat_button.rs | 2 +- voxygen/src/ui/ice/renderer/style/button.rs | 37 +- .../src/ui/ice/renderer/style/container.rs | 21 +- voxygen/src/ui/ice/renderer/widget/button.rs | 5 +- .../src/ui/ice/renderer/widget/container.rs | 121 +++++++ voxygen/src/ui/ice/widget/overlay.rs | 4 +- 24 files changed, 467 insertions(+), 158 deletions(-) create mode 100644 assets/voxygen/element/frames/gray/corner.png create mode 100644 assets/voxygen/element/frames/gray/edge.png delete mode 100644 assets/voxygen/element/frames/server_frame.vox delete mode 100644 assets/voxygen/element/frames/window_4.vox diff --git a/assets/voxygen/element/frames/gray/corner.png b/assets/voxygen/element/frames/gray/corner.png new file mode 100644 index 0000000000000000000000000000000000000000..2a44e690e0cd90512c6357a1d642558ee6cc12a9 GIT binary patch literal 142 zcmeAS@N?(olHy`uVBq!ia0vp^Od!m`1|*BN@u~nRwj^(N7l!{JxM1({$v_d#0*}aI z1_o|n5N2eUHAey{$X?><>&pIwO@`S@xcvTkU!ah*r;B3<$MxiY|Nq+`?iSboXS~1S e<0E@!AYdpu!&S6N?uIB(8H1;*pUXO@geCyUekGLv literal 0 HcmV?d00001 diff --git a/assets/voxygen/element/frames/gray/edge.png b/assets/voxygen/element/frames/gray/edge.png new file mode 100644 index 0000000000000000000000000000000000000000..4dad2e753f595c9fb67a0c0015233865aef90c0c GIT binary patch literal 133 zcmeAS@N?(olHy`uVBq!ia0vp^j6lr9!3HE}Hf~b~Qfx`y?k)@*44MpFa#bnkfg+p* z9+AZi4BWyX%*Zfnjs#GUy~NYkmHi2u3=4-~;k!3efkHx_E{-7_*ONW`{r{goaNt0J ZFoQrVGoPIZyB<)M!PC{xWt~$(697UW9IgNW literal 0 HcmV?d00001 diff --git a/assets/voxygen/element/frames/server_frame.vox b/assets/voxygen/element/frames/server_frame.vox deleted file mode 100644 index 2478bcce6c317714647b7f6e83484f8edf6be5fc..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 56107 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 diff --git a/assets/voxygen/i18n/de_DE.ron b/assets/voxygen/i18n/de_DE.ron index 3fdd70ba3e..2c95575941 100644 --- a/assets/voxygen/i18n/de_DE.ron +++ b/assets/voxygen/i18n/de_DE.ron @@ -470,10 +470,10 @@ magischen Gegenstände ergattern?"#, "char_selection.change_server": "Server wechseln.", "char_selection.enter_world": "Betreten", "char_selection.logout": "Ausloggen", - "char_selection.create_charater": "Charakter erstellen", + "char_selection.create_character": "Charakter erstellen", + "char_selection.create_new_character": "Neuen Charakter erstellen", "char_selection.creating_character": "Erstelle Charakter...", "char_selection.character_creation": "Charaktererstellung", - "char_selection.create_new_charater": "Neuen Charakter erstellen", "char_selection.human_default": "Human Default", "char_selection.level_fmt": "Level {level_nb}", diff --git a/assets/voxygen/i18n/en.ron b/assets/voxygen/i18n/en.ron index fb12fe4233..be9600261e 100644 --- a/assets/voxygen/i18n/en.ron +++ b/assets/voxygen/i18n/en.ron @@ -477,7 +477,7 @@ magically infused items?"#, "char_selection.change_server": "Change Server", "char_selection.enter_world": "Enter World", "char_selection.logout": "Logout", - "char_selection.create_new_charater": "Create New Character", + "char_selection.create_new_character": "Create New Character", "char_selection.creating_character": "Creating Character...", "char_selection.character_creation": "Character Creation", diff --git a/assets/voxygen/i18n/es_ES.ron b/assets/voxygen/i18n/es_ES.ron index 57a2dd3356..8b688abe05 100644 --- a/assets/voxygen/i18n/es_ES.ron +++ b/assets/voxygen/i18n/es_ES.ron @@ -358,7 +358,7 @@ objetos imbuidos de magia?"#, "char_selection.change_server": "Cambiar de servidor", "char_selection.enter_world": "Entrar al mundo", "char_selection.logout": "Salir", - "char_selection.create_new_charater": "Crear nuevo personaje", + "char_selection.create_new_character": "Crear nuevo personaje", "char_selection.creating_character": "Creando personaje...", "char_selection.character_creation": "Creación de personaje", diff --git a/assets/voxygen/i18n/fr_FR.ron b/assets/voxygen/i18n/fr_FR.ron index 4749bfe9d9..2649daefaf 100644 --- a/assets/voxygen/i18n/fr_FR.ron +++ b/assets/voxygen/i18n/fr_FR.ron @@ -372,7 +372,7 @@ objets magiques ?"#, "char_selection.change_server": "Changer de serveur", "char_selection.enter_world": "Entrer dans le monde", "char_selection.logout": "Se déconnecter", - "char_selection.create_new_charater": "Créer un nouveau personnage", + "char_selection.create_new_character": "Créer un nouveau personnage", "char_selection.creating_character": "Création du personnage...", "char_selection.character_creation": "Création de personnage", diff --git a/assets/voxygen/i18n/it_IT.ron b/assets/voxygen/i18n/it_IT.ron index d469582d34..5a37d39d36 100644 --- a/assets/voxygen/i18n/it_IT.ron +++ b/assets/voxygen/i18n/it_IT.ron @@ -462,7 +462,7 @@ oggetti infusi di magia?"#, "char_selection.change_server": "Cambia Server", "char_selection.enter_world": "Unisciti al Mondo", "char_selection.logout": "Disconnettiti", - "char_selection.create_new_charater": "Crea un nuovo Personaggio", + "char_selection.create_new_character": "Crea un nuovo Personaggio", "char_selection.creating_character": "Creazione Personaggio...", "char_selection.character_creation": "Creazione Personaggio", diff --git a/assets/voxygen/i18n/pt_PT.ron b/assets/voxygen/i18n/pt_PT.ron index 55f3420d25..87bd22bcc7 100644 --- a/assets/voxygen/i18n/pt_PT.ron +++ b/assets/voxygen/i18n/pt_PT.ron @@ -343,7 +343,7 @@ Comandos de chat: "char_selection.change_server": "Mudar de servidor", "char_selection.enter_world": "Entrar no mundo", "char_selection.logout": "Desconectar", - "char_selection.create_new_charater": "Criar nova personagem", + "char_selection.create_new_character": "Criar nova personagem", "char_selection.character_creation": "Criação de personagem", "char_selection.human_default": "Humano padrão", diff --git a/assets/voxygen/i18n/ru_RU.ron b/assets/voxygen/i18n/ru_RU.ron index 1269805096..0b26362297 100644 --- a/assets/voxygen/i18n/ru_RU.ron +++ b/assets/voxygen/i18n/ru_RU.ron @@ -400,7 +400,7 @@ https://veloren.net/account/."#, "char_selection.change_server": "Сменить сервер", "char_selection.enter_world": "Войти в мир", "char_selection.logout": "Выйти в меню", - "char_selection.create_new_charater": "Создать нового персонажа", + "char_selection.create_new_character": "Создать нового персонажа", "char_selection.creating_character": "Создание персонажа...", "char_selection.character_creation": "Создание персонажа", diff --git a/voxygen/src/menu/char_selection/mod.rs b/voxygen/src/menu/char_selection/mod.rs index 9de731f887..6140b68135 100644 --- a/voxygen/src/menu/char_selection/mod.rs +++ b/voxygen/src/menu/char_selection/mod.rs @@ -37,9 +37,9 @@ impl CharSelectionState { } } - fn get_humanoid_body(&self) -> Option { + fn get_humanoid_body(&self, client: &Client) -> Option { self.char_selection_ui - .selected_character() + .display_character(&client.character_list.characters) .and_then(|character| match character.body { comp::Body::Humanoid(body) => Some(body), _ => None, @@ -92,11 +92,8 @@ impl PlayState for CharSelectionState { ui::Event::DeleteCharacter(character_id) => { self.client.borrow_mut().delete_character(character_id); }, - ui::Event::Play(character) => { - // TODO: eliminate option in character id - if let Some(character_id) = character.character.id { - self.client.borrow_mut().request_character(character_id); - } + ui::Event::Play(character_id) => { + self.client.borrow_mut().request_character(character_id); return PlayStateResult::Switch(Box::new(SessionState::new( global_state, @@ -106,12 +103,15 @@ impl PlayState for CharSelectionState { } } - let humanoid_body = self.get_humanoid_body(); let loadout = self.char_selection_ui.get_loadout(); // Maintain the scene. { let client = self.client.borrow(); + let humanoid_body = self.get_humanoid_body(&client); + let loadout = self.char_selection_ui.get_loadout(); + + // Maintain the scene. let scene_data = scene::SceneData { time: client.state().get_time(), delta_time: client.state().ecs().read_resource::().0, @@ -127,6 +127,7 @@ impl PlayState for CharSelectionState { .figure_lod_render_distance as f32, }; + self.scene.maintain( global_state.window.renderer_mut(), scene_data, @@ -185,16 +186,13 @@ impl PlayState for CharSelectionState { fn name(&self) -> &'static str { "Title" } fn render(&mut self, renderer: &mut Renderer, _: &Settings) { - let humanoid_body = self.get_humanoid_body(); + let client = self.client.borrow(); + let humanoid_body = self.get_humanoid_body(&client); let loadout = self.char_selection_ui.get_loadout(); // Render the scene. - self.scene.render( - renderer, - self.client.borrow().get_tick(), - humanoid_body, - loadout.as_ref(), - ); + self.scene + .render(renderer, client.get_tick(), humanoid_body, loadout.as_ref()); // Draw the UI to the screen. self.char_selection_ui.render(renderer); diff --git a/voxygen/src/menu/char_selection/ui/mod.rs b/voxygen/src/menu/char_selection/ui/mod.rs index c7d7fe3d2d..f39c6bd338 100644 --- a/voxygen/src/menu/char_selection/ui/mod.rs +++ b/voxygen/src/menu/char_selection/ui/mod.rs @@ -4,7 +4,12 @@ use crate::{ ui::{ self, fonts::IcedFonts as Fonts, - ice::{component::neat_button, style, widget::Overlay, Element, IcedUi as Ui}, + ice::{ + component::neat_button, + style, + widget::{AspectRatioContainer, Overlay}, + Element, IcedUi as Ui, + }, img_ids::{ImageGraphic, VoxelGraphic}, }, window, GlobalState, @@ -13,33 +18,42 @@ use client::Client; use common::{ assets::Asset, character::{CharacterId, CharacterItem, MAX_CHARACTERS_PER_PLAYER}, - comp, - comp::humanoid, + comp::{self, humanoid}, }; //ImageFrame, Tooltip, use crate::settings::Settings; //use std::time::Duration; //use ui::ice::widget; use iced::{ - button, text_input, Align, Button, Column, Container, HorizontalAlignment, Length, Row, Space, - Text, + button, scrollable, text_input, Align, Button, Column, Container, HorizontalAlignment, Length, + Row, Scrollable, Space, Text, }; +use vek::Rgba; pub const TEXT_COLOR: iced::Color = iced::Color::from_rgb(1.0, 1.0, 1.0); pub const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1.0, 0.2); const FILL_FRAC_ONE: f32 = 0.77; -const FILL_FRAC_TWO: f32 = 0.53; +const FILL_FRAC_TWO: f32 = 0.60; + +const STARTER_HAMMER: &str = "common.items.weapons.hammer.starter_hammer"; +const STARTER_BOW: &str = "common.items.weapons.bow.starter_bow"; +const STARTER_AXE: &str = "common.items.weapons.axe.starter_axe"; +const STARTER_STAFF: &str = "common.items.weapons.staff.starter_staff"; +const STARTER_SWORD: &str = "common.items.weapons.sword.starter_sword"; + +// TODO: look into what was using this in old ui +const UI_MAIN: iced::Color = iced::Color::from_rgba(0.61, 0.70, 0.70, 1.0); // Greenish Blue image_ids_ice! { struct Imgs { - // TODO: convert large frames into borders - charlist_frame: "voxygen.element.frames.window_4", - server_frame: "voxygen.element.frames.server_frame", slider_range: "voxygen.element.slider.track", slider_indicator: "voxygen.element.slider.indicator", + gray_corner: "voxygen.element.frames.gray.corner", + gray_edge: "voxygen.element.frames.gray.edge", + selection: "voxygen.element.frames.selection", selection_hover: "voxygen.element.frames.selection_hover", selection_press: "voxygen.element.frames.selection_press", @@ -48,7 +62,6 @@ image_ids_ice! { delete_button_hover: "voxygen.element.buttons.x_red_hover", delete_button_press: "voxygen.element.buttons.x_red_press", - name_input: "voxygen.element.misc_bg.textbox", // Tool Icons @@ -80,7 +93,6 @@ image_ids_ice! { icon_border_press: "voxygen.element.buttons.border_press", icon_border_pressed: "voxygen.element.buttons.border_pressed", - button: "voxygen.element.buttons.button", button_hover: "voxygen.element.buttons.button_hover", button_press: "voxygen.element.buttons.button_press", @@ -100,7 +112,7 @@ image_ids_ice! { pub enum Event { Logout, - Play(CharacterItem), + Play(CharacterId), AddCharacter { alias: String, tool: Option, @@ -109,17 +121,14 @@ pub enum Event { DeleteCharacter(CharacterId), } -struct CharacterList { - characters: Vec, - selected_character: usize, -} - enum Mode { Select { - list: Option, + // Index of selected character + selected: Option, + + characters_scroll: scrollable::State, character_buttons: Vec, new_character_button: button::State, - logout_button: button::State, enter_world_button: button::State, change_server_button: button::State, @@ -136,6 +145,33 @@ enum Mode { }, } +impl Mode { + pub fn select() -> Self { + Self::Select { + selected: None, + characters_scroll: Default::default(), + character_buttons: Vec::new(), + new_character_button: Default::default(), + logout_button: Default::default(), + enter_world_button: Default::default(), + change_server_button: Default::default(), + } + } + + pub fn create(name: String) -> Self { + Self::Create { + name, + body: humanoid::Body::random(), + loadout: comp::Loadout::default(), + tool: Some(STARTER_SWORD), + + name_input: Default::default(), + back_button: Default::default(), + create_button: Default::default(), + } + } +} + #[derive(PartialEq)] enum InfoContent { Deletion(usize), @@ -202,22 +238,16 @@ impl Controls { alpha, info_content: None, - mode: Mode::Select { - list: None, - character_buttons: Vec::new(), - new_character_button: Default::default(), - logout_button: Default::default(), - enter_world_button: Default::default(), - change_server_button: Default::default(), - }, + mode: Mode::select(), } } - fn view(&mut self, settings: &Settings) -> Element { + fn view(&mut self, settings: &Settings, client: &Client) -> Element { // TODO: if enter key pressed and character is selected then enter the world // TODO: tooltip widget let imgs = &self.imgs; + let fonts = &self.fonts; let i18n = &self.i18n; let button_style = style::button::Style::new(imgs.button) @@ -232,7 +262,7 @@ impl Controls { .horizontal_alignment(HorizontalAlignment::Right); let alpha = iced::Text::new(&self.alpha) - .size(self.fonts.cyri.scale(15)) + .size(self.fonts.cyri.scale(12)) .width(Length::Fill) .horizontal_alignment(HorizontalAlignment::Center); @@ -245,7 +275,8 @@ impl Controls { let content = match &mut self.mode { Mode::Select { - list, + selected, + ref mut characters_scroll, ref mut character_buttons, ref mut new_character_button, ref mut logout_button, @@ -253,14 +284,41 @@ impl Controls { ref mut change_server_button, } => { // TODO: impl delete prompt as overlay - let change_server = Space::new(Length::Units(100), Length::Units(40)); - let characters = if let Some(list) = list { - let num = list.characters.len(); + let server = Container::new( + Column::with_children(vec![ + Text::new(&client.server_info.name) + .size(fonts.cyri.scale(25)) + //.horizontal_alignment(HorizontalAlignment::Center) + .into(), + Container::new(neat_button( + change_server_button, + i18n.get("char_selection.change_server"), + FILL_FRAC_TWO, + button_style, + Some(Message::ChangeServer), + )) + .height(Length::Units(35)) + .into(), + ]) + .spacing(5) + .align_items(Align::Center), + ) + .style(style::container::Style::color_with_image_border( + Rgba::new(0, 0, 0, 217), + imgs.gray_corner, + imgs.gray_edge, + )) + .padding(12) + .center_x() + .width(Length::Fill); + + let characters = { + let characters = &client.character_list.characters; + let num = characters.len(); // Ensure we have enough button states character_buttons.resize_with(num * 2, Default::default); - let mut characters = list - .characters + let mut characters = characters .iter() .zip(character_buttons.chunks_exact_mut(2)) .map(|(character, buttons)| { @@ -273,65 +331,102 @@ impl Controls { .enumerate() .map(|(i, (character, (select_button, delete_button)))| { Overlay::new( - Button::new(select_button, Space::new(Length::Fill, Length::Fill)) - .width(Length::Units(20)) - .height(Length::Units(20)) - .style( - style::button::Style::new(imgs.delete_button) - .hover_image(imgs.delete_button_hover) - .press_image(imgs.delete_button_press), - ), Button::new( - delete_button, - Column::with_children(vec![ - Text::new("Hi").into(), - Text::new("Hi").into(), - Text::new("Hi").into(), - ]), + select_button, + Space::new(Length::Units(20), Length::Units(20)), ) .style( - style::button::Style::new(imgs.selection) - .hover_image(imgs.selection_hover) - .press_image(imgs.selection_press), + style::button::Style::new(imgs.delete_button) + .hover_image(imgs.delete_button_hover) + .press_image(imgs.delete_button_press), ), + AspectRatioContainer::new( + Button::new( + delete_button, + Column::with_children(vec![ + Text::new("Hi").width(Length::Fill).into(), + Text::new("Hi").into(), + Text::new("Hi").into(), + ]), + ) + .style( + style::button::Style::new(imgs.selection) + .hover_image(imgs.selection_hover) + .press_image(imgs.selection_press), + ), + ) + .ratio_of_image(imgs.selection), ) + .padding(15) + .align_x(Align::End) .into() }) .collect::>(); // Add create new character button let color = if num >= MAX_CHARACTERS_PER_PLAYER { - iced::Color::from_rgb8(97, 97, 25) + (97, 97, 25) } else { - iced::Color::from_rgb8(97, 255, 18) + (97, 255, 18) }; characters.push( - Button::new( - new_character_button, - Text::new(i18n.get("char_selection.create_new_character")), - ) - .style( - style::button::Style::new(imgs.selection) - .hover_image(imgs.selection_hover) - .press_image(imgs.selection_press) - .image_color(color) - .text_color(color), - ) + AspectRatioContainer::new({ + let button = Button::new( + new_character_button, + Container::new(Text::new( + i18n.get("char_selection.create_new_character"), + )) + .width(Length::Fill) + .height(Length::Fill) + .center_x() + .center_y(), + ) + .style( + style::button::Style::new(imgs.selection) + .hover_image(imgs.selection_hover) + .press_image(imgs.selection_press) + .image_color(Rgba::new(color.0, color.1, color.2, 255)) + .text_color(iced::Color::from_rgb8(color.0, color.1, color.2)) + .disabled_text_color(iced::Color::from_rgb8( + color.0, color.1, color.2, + )), + ) + .width(Length::Fill) + .height(Length::Fill); + // TODO: try to get better interface for this in iced + if num < MAX_CHARACTERS_PER_PLAYER { + button.on_press(Message::NewCharacter) + } else { + button + } + }) + .ratio_of_image(imgs.selection) .into(), ); characters - } else { - Vec::new() }; - let characters = Column::with_children(characters); + // TODO: could replace column with scrollable completely if it had a with + // children method + let characters = Container::new( + Scrollable::new(characters_scroll) + .push(Column::with_children(characters).spacing(2)), + ) + .style(style::container::Style::color_with_image_border( + Rgba::new(0, 0, 0, 217), + imgs.gray_corner, + imgs.gray_edge, + )) + .padding(9) + .width(Length::Fill) + .height(Length::Fill); - let right_column = - Column::with_children(vec![change_server.into(), characters.into()]) - .spacing(10) - .width(Length::Fill) - .height(Length::Fill) - .max_width(300); + let right_column = Column::with_children(vec![server.into(), characters.into()]) + .padding(15) + .spacing(10) + .width(Length::Fill) + .height(Length::Fill) + .max_width(360); let top = Container::new(right_column) .width(Length::Fill) @@ -350,23 +445,22 @@ impl Controls { i18n.get("char_selection.enter_world"), FILL_FRAC_ONE, button_style, - Some(Message::EnterWorld), + selected.map(|_| Message::EnterWorld), ); let bottom = Row::with_children(vec![ Container::new(logout) .width(Length::Fill) .height(Length::Units(40)) - .align_y(Align::End) .into(), Container::new(enter_world) .width(Length::Fill) .height(Length::Units(60)) .center_x() - .align_y(Align::End) .into(), Space::new(Length::Fill, Length::Shrink).into(), - ]); + ]) + .align_items(Align::End); Column::with_children(vec![top.into(), bottom.into()]) .width(Length::Fill) @@ -400,13 +494,72 @@ impl Controls { .into() } - fn update(&mut self, message: Message, events: &mut Vec, settings: &Settings) { + fn update( + &mut self, + message: Message, + events: &mut Vec, + settings: &Settings, + characters: &[CharacterItem], + ) { let servers = &settings.networking.servers; - //match message { } + match message { + Message::Back => { + if matches!(&self.mode, Mode::Create { .. }) { + self.mode = Mode::select(); + } + }, + Message::Logout => { + events.push(Event::Logout); + }, + Message::EnterWorld => { + if let Mode::Select { + selected: Some(selected), + .. + } = &self.mode + { + // TODO: eliminate option in character id + if let Some(id) = characters.get(*selected).and_then(|i| i.character.id) { + events.push(Event::Play(id)); + } + } + }, + Message::Delete(idx) => { + if let Some(id) = characters.get(idx).and_then(|i| i.character.id) { + events.push(Event::DeleteCharacter(id)); + } + }, + Message::ChangeServer => { + events.push(Event::Logout); + }, + Message::NewCharacter => { + if matches!(&self.mode, Mode::Select { .. }) { + self.mode = Mode::create(String::new()); + } + }, + Message::CreateCharacter => { + if let Mode::Create { + name, body, tool, .. + } = &self.mode + { + events.push(Event::AddCharacter { + alias: name.clone(), + tool: tool.map(String::from), + body: comp::Body::Humanoid(*body), + }); + self.mode = Mode::select(); + } + }, + Message::Name(value) => { + if let Mode::Create { name, .. } = &mut self.mode { + *name = value; + } + }, + } } - pub fn selected_character(&self) -> Option<&CharacterItem> { + /// Get the character to display + pub fn display_character(&self, characters: &[CharacterItem]) -> Option<&CharacterItem> { match self.mode { // TODO Mode::Select { .. } => None, @@ -454,12 +607,13 @@ impl CharSelectionUi { Self { ui, controls } } - pub fn selected_character(&self) -> Option<&CharacterItem> { - self.controls.selected_character() + pub fn display_character(&self, characters: &[CharacterItem]) -> Option<&CharacterItem> { + self.controls.display_character(characters) } // TODO pub fn get_loadout(&mut self) -> Option { + // TODO: error gracefully // TODO: don't clone /*match &mut self.mode { Mode::Select(character_list) => { @@ -471,20 +625,20 @@ impl CharSelectionUi { }, Mode::Create { loadout, tool, .. } => { loadout.active_item = tool.map(|tool| comp::ItemConfig { - item: (*load_expect::(tool)).clone(), + item: comp::Item::new_from_asset_expect(tool), ability1: None, ability2: None, ability3: None, block_ability: None, dodge_ability: None, }); - loadout.chest = Some(assets::load_expect_cloned( + loadout.chest = Some(comp::Item::new_from_asset_expect( "common.items.armor.starter.rugged_chest", )); - loadout.pants = Some(assets::load_expect_cloned( + loadout.pants = Some(comp::Item::new_from_asset_expect( "common.items.armor.starter.rugged_pants", )); - loadout.foot = Some(assets::load_expect_cloned( + loadout.foot = Some(comp::Item::new_from_asset_expect( "common.items.armor.starter.sandals_0", )); Some(loadout.clone()) @@ -508,17 +662,22 @@ impl CharSelectionUi { } } + // TODO: do we need whole client here or just character list pub fn maintain(&mut self, global_state: &mut GlobalState, client: &mut Client) -> Vec { let mut events = Vec::new(); let (messages, _) = self.ui.maintain( - self.controls.view(&global_state.settings), + self.controls.view(&global_state.settings, &client), global_state.window.renderer_mut(), ); messages.into_iter().for_each(|message| { - self.controls - .update(message, &mut events, &global_state.settings) + self.controls.update( + message, + &mut events, + &global_state.settings, + &client.character_list.characters, + ) }); events diff --git a/voxygen/src/menu/main/ui/connecting.rs b/voxygen/src/menu/main/ui/connecting.rs index 42bc40a20d..f063eaae5a 100644 --- a/voxygen/src/menu/main/ui/connecting.rs +++ b/voxygen/src/menu/main/ui/connecting.rs @@ -98,11 +98,13 @@ impl Screen { .height(Length::Fill); let prompt_window = Container::new(content) - .style(style::container::Style::color_double_cornerless_border( - (22, 18, 16, 255).into(), - (11, 11, 11, 255).into(), - (54, 46, 38, 255).into(), - )) + .style( + style::container::Style::color_with_double_cornerless_border( + (22, 18, 16, 255).into(), + (11, 11, 11, 255).into(), + (54, 46, 38, 255).into(), + ), + ) .padding(20); let container = Container::new(prompt_window) diff --git a/voxygen/src/menu/main/ui/disclaimer.rs b/voxygen/src/menu/main/ui/disclaimer.rs index 4215498d91..b7225a0edf 100644 --- a/voxygen/src/menu/main/ui/disclaimer.rs +++ b/voxygen/src/menu/main/ui/disclaimer.rs @@ -64,11 +64,13 @@ impl Screen { .width(Length::Fill) .height(Length::Fill), ) - .style(style::container::Style::color_double_cornerless_border( - (22, 19, 17, 255).into(), - (11, 11, 11, 255).into(), - (54, 46, 38, 255).into(), - )), + .style( + style::container::Style::color_with_double_cornerless_border( + (22, 19, 17, 255).into(), + (11, 11, 11, 255).into(), + (54, 46, 38, 255).into(), + ), + ), ) .center_x() .center_y() diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index 5b2cccc8da..1ac637a4e7 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -95,7 +95,7 @@ impl Screen { // Note: a way to tell it to keep the height of this one piece constant and // unstreched would be nice, I suppose we could just break this out into a // column and use Length::Units - Graphic::gradient(Rgba::new(0, 0, 0, 240), Rgba::zero(), [500, 30], [0, 300]), + Graphic::gradient(Rgba::new(0, 0, 0, 240), Rgba::zero(), [500, 50], [0, 300]), ]) .height(Length::Shrink), Text::new(intro_text).size(fonts.cyri.scale(21)), @@ -128,11 +128,13 @@ impl Screen { .height(Length::Fill) .width(Length::Fill), ) - .style(style::container::Style::color_double_cornerless_border( - (22, 18, 16, 255).into(), - (11, 11, 11, 255).into(), - (54, 46, 38, 255).into(), - )) + .style( + style::container::Style::color_with_double_cornerless_border( + (22, 18, 16, 255).into(), + (11, 11, 11, 255).into(), + (54, 46, 38, 255).into(), + ), + ) .width(Length::Units(400)) .height(Length::Units(180)) .padding(20) diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 8b8b28bdcd..0cb9960e76 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -239,7 +239,7 @@ impl Controls { .horizontal_alignment(HorizontalAlignment::Right); let alpha = iced::Text::new(&self.alpha) - .size(self.fonts.cyri.scale(15)) + .size(self.fonts.cyri.scale(12)) .width(Length::Fill) .horizontal_alignment(HorizontalAlignment::Center); diff --git a/voxygen/src/menu/main/ui/servers.rs b/voxygen/src/menu/main/ui/servers.rs index e0fd31a478..fe7bdb1631 100644 --- a/voxygen/src/menu/main/ui/servers.rs +++ b/voxygen/src/menu/main/ui/servers.rs @@ -81,11 +81,13 @@ impl Screen { .spacing(10) .padding(20), ) - .style(style::container::Style::color_double_cornerless_border( - (22, 18, 16, 255).into(), - (11, 11, 11, 255).into(), - (54, 46, 38, 255).into(), - )) + .style( + style::container::Style::color_with_double_cornerless_border( + (22, 18, 16, 255).into(), + (11, 11, 11, 255).into(), + (54, 46, 38, 255).into(), + ), + ) .max_width(500), ) .width(Length::Fill) diff --git a/voxygen/src/ui/ice/component/neat_button.rs b/voxygen/src/ui/ice/component/neat_button.rs index 5cc3b53f6e..7b5508c4fd 100644 --- a/voxygen/src/ui/ice/component/neat_button.rs +++ b/voxygen/src/ui/ice/component/neat_button.rs @@ -24,7 +24,7 @@ pub fn neat_button( let container = AspectRatioContainer::new(button); let container = match button_style.active().0 { - Some(img) => container.ratio_of_image(img), + Some((img, _)) => container.ratio_of_image(img), None => container, }; diff --git a/voxygen/src/ui/ice/renderer/style/button.rs b/voxygen/src/ui/ice/renderer/style/button.rs index 4c4de0e58b..beea30781b 100644 --- a/voxygen/src/ui/ice/renderer/style/button.rs +++ b/voxygen/src/ui/ice/renderer/style/button.rs @@ -1,12 +1,13 @@ use super::super::super::widget::image; use iced::Color; +use vek::Rgba; #[derive(Clone, Copy)] struct Background { default: image::Handle, hover: image::Handle, press: image::Handle, - color: Color, + color: Rgba, } impl Background { @@ -15,7 +16,7 @@ impl Background { default: image, hover: image, press: image, - color: Color::WHITE, + color: Rgba::white(), } } } @@ -62,7 +63,7 @@ impl Style { // TODO: this needs to be refactored since the color isn't used if there is no // background - pub fn image_color(mut self, color: Color) -> Self { + pub fn image_color(mut self, color: Rgba) -> Self { if let Some(background) = &mut self.background { background.color = color; } @@ -79,24 +80,30 @@ impl Style { self } - pub fn disabled(&self) -> (Option, Color) { + pub fn disabled(&self) -> (Option<(image::Handle, Rgba)>, Color) { ( - self.background.as_ref().map(|b| b.default), + self.background.as_ref().map(|b| (b.default, b.color)), self.disabled_text, ) } - pub fn pressed(&self) -> (Option, Color) { - (self.background.as_ref().map(|b| b.press), self.enabled_text) - } - - pub fn hovered(&self) -> (Option, Color) { - (self.background.as_ref().map(|b| b.hover), self.enabled_text) - } - - pub fn active(&self) -> (Option, Color) { + pub fn pressed(&self) -> (Option<(image::Handle, Rgba)>, Color) { ( - self.background.as_ref().map(|b| b.default), + self.background.as_ref().map(|b| (b.press, b.color)), + self.enabled_text, + ) + } + + pub fn hovered(&self) -> (Option<(image::Handle, Rgba)>, Color) { + ( + self.background.as_ref().map(|b| (b.hover, b.color)), + self.enabled_text, + ) + } + + pub fn active(&self) -> (Option<(image::Handle, Rgba)>, Color) { + ( + self.background.as_ref().map(|b| (b.default, b.color)), self.enabled_text, ) } diff --git a/voxygen/src/ui/ice/renderer/style/container.rs b/voxygen/src/ui/ice/renderer/style/container.rs index dde0865216..f30ca8cff9 100644 --- a/voxygen/src/ui/ice/renderer/style/container.rs +++ b/voxygen/src/ui/ice/renderer/style/container.rs @@ -3,7 +3,14 @@ use vek::Rgba; /// Container Border pub enum Border { - DoubleCornerless { inner: Rgba, outer: Rgba }, + DoubleCornerless { + inner: Rgba, + outer: Rgba, + }, + Image { + corner: image::Handle, + edge: image::Handle, + }, None, } @@ -22,13 +29,23 @@ impl Style { pub fn color(color: Rgba) -> Self { Self::Color(color, Border::None) } /// Shorthand for a color background with a cornerless border - pub fn color_double_cornerless_border( + pub fn color_with_double_cornerless_border( color: Rgba, inner: Rgba, outer: Rgba, ) -> Self { Self::Color(color, Border::DoubleCornerless { inner, outer }) } + + /// Shorthand for a color background with image borders where the corners + /// are inset + pub fn color_with_image_border( + color: Rgba, + corner: image::Handle, + edge: image::Handle, + ) -> Self { + Self::Color(color, Border::Image { corner, edge }) + } } impl Default for Style { diff --git a/voxygen/src/ui/ice/renderer/widget/button.rs b/voxygen/src/ui/ice/renderer/widget/button.rs index 541b1a66c7..5f9c17c783 100644 --- a/voxygen/src/ui/ice/renderer/widget/button.rs +++ b/voxygen/src/ui/ice/renderer/widget/button.rs @@ -1,6 +1,5 @@ use super::super::{super::Rotation, style, Defaults, IcedRenderer, Primitive}; use iced::{button, mouse, Element, Layout, Point, Rectangle}; -use vek::Rgba; impl button::Renderer for IcedRenderer { // TODO: what if this gets large enough to not be copied around? @@ -40,11 +39,11 @@ impl button::Renderer for IcedRenderer { cursor_position, ); - let primitive = if let Some(handle) = maybe_image { + let primitive = if let Some((handle, color)) = maybe_image { let background = Primitive::Image { handle: (handle, Rotation::None), bounds, - color: Rgba::broadcast(255), + color, }; Primitive::Group { diff --git a/voxygen/src/ui/ice/renderer/widget/container.rs b/voxygen/src/ui/ice/renderer/widget/container.rs index 6b1559294f..a15436e160 100644 --- a/voxygen/src/ui/ice/renderer/widget/container.rs +++ b/voxygen/src/ui/ice/renderer/widget/container.rs @@ -2,7 +2,9 @@ use super::super::{super::Rotation, style, IcedRenderer, Primitive}; use common::util::srgba_to_linear; use iced::{container, Element, Layout, Point, Rectangle}; use style::container::Border; +use vek::Rgba; +// TODO: move to style const BORDER_SIZE: u16 = 8; impl container::Renderer for IcedRenderer { @@ -148,6 +150,125 @@ impl container::Renderer for IcedRenderer { content, ] }, + Border::Image { corner, edge } => { + let border_size = f32::from(BORDER_SIZE) + .min(bounds.width / 4.0) + .min(bounds.height / 4.0); + + let center = Primitive::Rectangle { + bounds: Rectangle { + x: bounds.x + border_size, + y: bounds.y + border_size, + width: bounds.width - border_size * 2.0, + height: bounds.height - border_size * 2.0, + }, + linear_color, + }; + + let color = Rgba::white(); + + let tl_corner = Primitive::Image { + handle: (*corner, Rotation::None), + bounds: Rectangle { + x: bounds.x, + y: bounds.y, + width: border_size, + height: border_size, + }, + color, + }; + + let tr_corner = Primitive::Image { + handle: (*corner, Rotation::Cw90), + bounds: Rectangle { + x: bounds.x + bounds.width - border_size, + y: bounds.y, + width: border_size, + height: border_size, + }, + color, + }; + + let bl_corner = Primitive::Image { + handle: (*corner, Rotation::Cw270), + bounds: Rectangle { + x: bounds.x, + y: bounds.y + bounds.height - border_size, + width: border_size, + height: border_size, + }, + color, + }; + + let br_corner = Primitive::Image { + handle: (*corner, Rotation::Cw180), + bounds: Rectangle { + x: bounds.x + bounds.width - border_size, + y: bounds.y + bounds.height - border_size, + width: border_size, + height: border_size, + }, + color, + }; + + let top_edge = Primitive::Image { + handle: (*edge, Rotation::None), + bounds: Rectangle { + x: bounds.x + border_size, + y: bounds.y, + width: bounds.width - 2.0 * border_size, + height: border_size, + }, + color, + }; + + let bottom_edge = Primitive::Image { + handle: (*edge, Rotation::Cw180), + bounds: Rectangle { + x: bounds.x + border_size, + y: bounds.y + bounds.height - border_size, + width: bounds.width - 2.0 * border_size, + height: border_size, + }, + color, + }; + + let left_edge = Primitive::Image { + handle: (*edge, Rotation::Cw270), + bounds: Rectangle { + x: bounds.x, + y: bounds.y + border_size, + width: border_size, + height: bounds.height - 2.0 * border_size, + }, + color, + }; + + let right_edge = Primitive::Image { + handle: (*edge, Rotation::Cw90), + bounds: Rectangle { + x: bounds.x + bounds.width - border_size, + y: bounds.y + border_size, + width: border_size, + height: bounds.height - 2.0 * border_size, + }, + color, + }; + + // Is this worth it as opposed to using a giant image? (Probably) + vec![ + center, + tl_corner, + tr_corner, + bl_corner, + br_corner, + top_edge, + bottom_edge, + left_edge, + right_edge, + content, + ] + }, }; Primitive::Group { primitives } diff --git a/voxygen/src/ui/ice/widget/overlay.rs b/voxygen/src/ui/ice/widget/overlay.rs index 5cc0cc96ba..b168e1f99b 100644 --- a/voxygen/src/ui/ice/widget/overlay.rs +++ b/voxygen/src/ui/ice/widget/overlay.rs @@ -112,12 +112,12 @@ where let under_size = under.size(); let limits = limits.pad(padding); - let mut over = self.under.layout(renderer, &limits.loose()); + let mut over = self.over.layout(renderer, &limits.loose()); let over_size = over.size(); let size = limits.resolve(Size { width: under_size.width.max(over_size.width), - height: under_size.width.max(over_size.width), + height: under_size.height.max(over_size.height), }); over.move_to(Point::new(padding, padding)); From 09fc05db66441db2b0521ebd987b8a15fb2148d8 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sun, 30 Aug 2020 18:32:26 -0400 Subject: [PATCH 36/61] Add more parts of the character selection screen, add mouse detector widget, misc tweaks --- assets/voxygen/i18n/en.ron | 5 +- voxygen/src/menu/char_selection/mod.rs | 44 +- voxygen/src/menu/char_selection/ui/mod.rs | 584 ++++++++++++++---- voxygen/src/menu/main/mod.rs | 7 +- voxygen/src/menu/main/ui/login.rs | 7 +- voxygen/src/ui/ice/renderer/mod.rs | 77 ++- voxygen/src/ui/ice/renderer/widget/mod.rs | 1 + .../ui/ice/renderer/widget/mouse_detector.rs | 9 + voxygen/src/ui/ice/renderer/widget/text.rs | 56 +- .../src/ui/ice/renderer/widget/text_input.rs | 42 +- .../ui/ice/widget/aspect_ratio_container.rs | 16 +- voxygen/src/ui/ice/widget/mod.rs | 2 + voxygen/src/ui/ice/widget/mouse_detector.rs | 112 ++++ voxygen/src/ui/ice/widget/overlay.rs | 20 +- 14 files changed, 749 insertions(+), 233 deletions(-) create mode 100644 voxygen/src/ui/ice/renderer/widget/mouse_detector.rs create mode 100644 voxygen/src/ui/ice/widget/mouse_detector.rs diff --git a/assets/voxygen/i18n/en.ron b/assets/voxygen/i18n/en.ron index be9600261e..2c9c5e3f2d 100644 --- a/assets/voxygen/i18n/en.ron +++ b/assets/voxygen/i18n/en.ron @@ -108,6 +108,9 @@ Is the client up to date?"#, /// Start Main screen section + "main.username": "Username", + "main.server": "Server", + "main.password": "Password", "main.connecting": "Connecting", "main.creating_world": "Creating world", "main.tip": "Tip:", @@ -494,7 +497,7 @@ magically infused items?"#, "char_selection.accessories": "Accessories", "char_selection.create_info_name": "Your Character needs a name!", - /// End chracter selection section + /// End character selection section /// Start character window section diff --git a/voxygen/src/menu/char_selection/mod.rs b/voxygen/src/menu/char_selection/mod.rs index 6140b68135..f312da2705 100644 --- a/voxygen/src/menu/char_selection/mod.rs +++ b/voxygen/src/menu/char_selection/mod.rs @@ -37,13 +37,22 @@ impl CharSelectionState { } } - fn get_humanoid_body(&self, client: &Client) -> Option { - self.char_selection_ui - .display_character(&client.character_list.characters) - .and_then(|character| match character.body { - comp::Body::Humanoid(body) => Some(body), - _ => None, + fn get_humanoid_body_loadout<'a>( + char_selection_ui: &'a CharSelectionUi, + client: &'a Client, + ) -> (Option, Option<&'a comp::Loadout>) { + char_selection_ui + .display_body_loadout(&client.character_list.characters) + .map(|(body, loadout)| { + ( + match body { + comp::Body::Humanoid(body) => Some(body), + _ => None, + }, + Some(loadout), + ) }) + .unwrap_or_default() } } @@ -87,7 +96,9 @@ impl PlayState for CharSelectionState { return PlayStateResult::Pop; }, ui::Event::AddCharacter { alias, tool, body } => { - self.client.borrow_mut().create_character(alias, tool, body); + self.client + .borrow_mut() + .create_character(alias, Some(tool), body); }, ui::Event::DeleteCharacter(character_id) => { self.client.borrow_mut().delete_character(character_id); @@ -103,13 +114,11 @@ impl PlayState for CharSelectionState { } } - let loadout = self.char_selection_ui.get_loadout(); - // Maintain the scene. { let client = self.client.borrow(); - let humanoid_body = self.get_humanoid_body(&client); - let loadout = self.char_selection_ui.get_loadout(); + let (humanoid_body, loadout) = + Self::get_humanoid_body_loadout(&self.char_selection_ui, &client); // Maintain the scene. let scene_data = scene::SceneData { @@ -128,11 +137,8 @@ impl PlayState for CharSelectionState { as f32, }; - self.scene.maintain( - global_state.window.renderer_mut(), - scene_data, - loadout.as_ref(), - ); + self.scene + .maintain(global_state.window.renderer_mut(), scene_data, loadout); } // Tick the client (currently only to keep the connection alive). @@ -187,12 +193,12 @@ impl PlayState for CharSelectionState { fn render(&mut self, renderer: &mut Renderer, _: &Settings) { let client = self.client.borrow(); - let humanoid_body = self.get_humanoid_body(&client); - let loadout = self.char_selection_ui.get_loadout(); + let (humanoid_body, loadout) = + Self::get_humanoid_body_loadout(&self.char_selection_ui, &client); // Render the scene. self.scene - .render(renderer, client.get_tick(), humanoid_body, loadout.as_ref()); + .render(renderer, client.get_tick(), humanoid_body, loadout); // Draw the UI to the screen. self.char_selection_ui.render(renderer); diff --git a/voxygen/src/menu/char_selection/ui/mod.rs b/voxygen/src/menu/char_selection/ui/mod.rs index f39c6bd338..e2b2e32333 100644 --- a/voxygen/src/menu/char_selection/ui/mod.rs +++ b/voxygen/src/menu/char_selection/ui/mod.rs @@ -7,8 +7,11 @@ use crate::{ ice::{ component::neat_button, style, - widget::{AspectRatioContainer, Overlay}, - Element, IcedUi as Ui, + widget::{ + mouse_detector, AspectRatioContainer, BackgroundContainer, Image, MouseDetector, + Overlay, Padding, + }, + Element, IcedRenderer, IcedUi as Ui, }, img_ids::{ImageGraphic, VoxelGraphic}, }, @@ -19,6 +22,7 @@ use common::{ assets::Asset, character::{CharacterId, CharacterItem, MAX_CHARACTERS_PER_PLAYER}, comp::{self, humanoid}, + LoadoutBuilder, }; //ImageFrame, Tooltip, use crate::settings::Settings; @@ -26,7 +30,7 @@ use crate::settings::Settings; //use ui::ice::widget; use iced::{ button, scrollable, text_input, Align, Button, Column, Container, HorizontalAlignment, Length, - Row, Scrollable, Space, Text, + Row, Scrollable, Space, Text, TextInput, }; use vek::Rgba; @@ -40,17 +44,17 @@ const STARTER_BOW: &str = "common.items.weapons.bow.starter_bow"; const STARTER_AXE: &str = "common.items.weapons.axe.starter_axe"; const STARTER_STAFF: &str = "common.items.weapons.staff.starter_staff"; const STARTER_SWORD: &str = "common.items.weapons.sword.starter_sword"; +const STARTER_DAGGER: &str = "common.items.weapons.dagger.starter_dagger"; // TODO: look into what was using this in old ui const UI_MAIN: iced::Color = iced::Color::from_rgba(0.61, 0.70, 0.70, 1.0); // Greenish Blue image_ids_ice! { struct Imgs { - + slider_range: "voxygen.element.slider.track", slider_indicator: "voxygen.element.slider.indicator", - gray_corner: "voxygen.element.frames.gray.corner", gray_edge: "voxygen.element.frames.gray.edge", @@ -73,8 +77,6 @@ image_ids_ice! { staff: "voxygen.element.icons.staff", // Species Icons - male: "voxygen.element.icons.male", - female: "voxygen.element.icons.female", human_m: "voxygen.element.icons.human_m", human_f: "voxygen.element.icons.human_f", orc_m: "voxygen.element.icons.orc_m", @@ -115,7 +117,7 @@ pub enum Event { Play(CharacterId), AddCharacter { alias: String, - tool: Option, + tool: String, body: comp::Body, }, DeleteCharacter(CharacterId), @@ -123,6 +125,7 @@ pub enum Event { enum Mode { Select { + info_content: Option, // Index of selected character selected: Option, @@ -132,13 +135,20 @@ enum Mode { logout_button: button::State, enter_world_button: button::State, change_server_button: button::State, + yes_button: button::State, + no_button: button::State, }, Create { name: String, // TODO: default to username body: humanoid::Body, loadout: comp::Loadout, - tool: Option<&'static str>, + // TODO: does this need to be an option, never seems to be none + tool: &'static str, + body_type_buttons: [button::State; 2], + species_buttons: [button::State; 6], + tool_buttons: [button::State; 6], + scroll: scrollable::State, name_input: text_input::State, back_button: button::State, create_button: button::State, @@ -148,6 +158,7 @@ enum Mode { impl Mode { pub fn select() -> Self { Self::Select { + info_content: None, selected: None, characters_scroll: Default::default(), character_buttons: Vec::new(), @@ -155,16 +166,29 @@ impl Mode { logout_button: Default::default(), enter_world_button: Default::default(), change_server_button: Default::default(), + yes_button: Default::default(), + no_button: Default::default(), } } pub fn create(name: String) -> Self { + let tool = STARTER_SWORD; + + let loadout = LoadoutBuilder::new() + .defaults() + .active_item(Some(LoadoutBuilder::default_item_config_from_str(tool))) + .build(); + Self::Create { name, body: humanoid::Body::random(), - loadout: comp::Loadout::default(), - tool: Some(STARTER_SWORD), + loadout, + tool, + body_type_buttons: Default::default(), + species_buttons: Default::default(), + tool_buttons: Default::default(), + scroll: Default::default(), name_input: Default::default(), back_button: Default::default(), create_button: Default::default(), @@ -204,7 +228,8 @@ struct Controls { // Alpha disclaimer alpha: String, - info_content: Option, + // Zone for rotating the character with the mouse + mouse_detector: mouse_detector::State, // enter: bool, mode: Mode, } @@ -214,11 +239,17 @@ enum Message { Back, Logout, EnterWorld, + Select(usize), Delete(usize), ChangeServer, NewCharacter, CreateCharacter, Name(String), + BodyType(humanoid::BodyType), + Species(humanoid::Species), + Tool(&'static str), + CancelDeletion, + ConfirmDeletion, } impl Controls { @@ -237,14 +268,15 @@ impl Controls { version, alpha, - info_content: None, + mouse_detector: Default::default(), mode: Mode::select(), } } fn view(&mut self, settings: &Settings, client: &Client) -> Element { - // TODO: if enter key pressed and character is selected then enter the world - // TODO: tooltip widget + // TODO: use font scale thing for text size (use on button size for buttons with + // text) TODO: if enter key pressed and character is selected then enter + // the world TODO: tooltip widget let imgs = &self.imgs; let fonts = &self.fonts; @@ -275,6 +307,7 @@ impl Controls { let content = match &mut self.mode { Mode::Select { + info_content, selected, ref mut characters_scroll, ref mut character_buttons, @@ -282,14 +315,14 @@ impl Controls { ref mut logout_button, ref mut enter_world_button, ref mut change_server_button, + ref mut yes_button, + ref mut no_button, } => { - // TODO: impl delete prompt as overlay let server = Container::new( Column::with_children(vec![ Text::new(&client.server_info.name) - .size(fonts.cyri.scale(25)) - //.horizontal_alignment(HorizontalAlignment::Center) - .into(), + .size(fonts.cyri.scale(25)) + .into(), Container::new(neat_button( change_server_button, i18n.get("char_selection.change_server"), @@ -333,18 +366,19 @@ impl Controls { Overlay::new( Button::new( select_button, - Space::new(Length::Units(20), Length::Units(20)), + Space::new(Length::Units(16), Length::Units(16)), ) .style( style::button::Style::new(imgs.delete_button) .hover_image(imgs.delete_button_hover) .press_image(imgs.delete_button_press), - ), + ) + .on_press(Message::Delete(i)), AspectRatioContainer::new( Button::new( delete_button, Column::with_children(vec![ - Text::new("Hi").width(Length::Fill).into(), + Text::new("Hi").into(), Text::new("Hi").into(), Text::new("Hi").into(), ]), @@ -353,11 +387,14 @@ impl Controls { style::button::Style::new(imgs.selection) .hover_image(imgs.selection_hover) .press_image(imgs.selection_press), - ), + ) + .width(Length::Fill) + .height(Length::Fill) + .on_press(Message::Select(i)), ) .ratio_of_image(imgs.selection), ) - .padding(15) + .padding(12) .align_x(Align::End) .into() }) @@ -393,7 +430,6 @@ impl Controls { ) .width(Length::Fill) .height(Length::Fill); - // TODO: try to get better interface for this in iced if num < MAX_CHARACTERS_PER_PLAYER { button.on_press(Message::NewCharacter) } else { @@ -410,7 +446,7 @@ impl Controls { // children method let characters = Container::new( Scrollable::new(characters_scroll) - .push(Column::with_children(characters).spacing(2)), + .push(Column::with_children(characters).spacing(4)), ) .style(style::container::Style::color_with_image_border( Rgba::new(0, 0, 0, 217), @@ -424,14 +460,19 @@ impl Controls { let right_column = Column::with_children(vec![server.into(), characters.into()]) .padding(15) .spacing(10) - .width(Length::Fill) - .height(Length::Fill) - .max_width(360); - - let top = Container::new(right_column) - .width(Length::Fill) + .width(Length::Units(360)) // TODO: see if we can get iced to work with settings below + //.max_width(360) + //.width(Length::Fill) .height(Length::Fill); + let top = Row::with_children(vec![ + right_column.into(), + MouseDetector::new(&mut self.mouse_detector, Length::Fill, Length::Fill).into(), + ]) + .padding(15) + .width(Length::Fill) + .height(Length::Fill); + let logout = neat_button( logout_button, i18n.get("char_selection.logout"), @@ -462,30 +503,359 @@ impl Controls { ]) .align_items(Align::End); - Column::with_children(vec![top.into(), bottom.into()]) + let content = Column::with_children(vec![top.into(), bottom.into()]) .width(Length::Fill) - .height(Length::Fill) + .height(Length::Fill); + + // Overlay delete prompt + if let Some(info_content) = info_content { + let over: Element<_> = match info_content { + InfoContent::Deletion(_) => Container::new( + Column::with_children(vec![ + Text::new(self.i18n.get("char_selection.delete_permanently")) + .size(fonts.cyri.scale(24)) + .into(), + Row::with_children(vec![ + neat_button( + no_button, + i18n.get("common.no"), + FILL_FRAC_ONE, + button_style, + Some(Message::CancelDeletion), + ), + neat_button( + yes_button, + i18n.get("common.yes"), + FILL_FRAC_ONE, + button_style, + Some(Message::ConfirmDeletion), + ), + ]) + .height(Length::Units(28)) + .spacing(30) + .into(), + ]) + .align_items(Align::Center) + .spacing(10), + ) + .style( + style::container::Style::color_with_double_cornerless_border( + (0, 0, 0, 200).into(), + (3, 4, 4, 255).into(), + (28, 28, 22, 255).into(), + ), + ) + .width(Length::Fill) + .height(Length::Fill) + .max_width(400) + .max_height(130) + .padding(16) + .center_x() + .center_y() + .into(), + // TODO + _ => Space::new(Length::Shrink, Length::Shrink).into(), + }; + + Overlay::new(over, content) + .width(Length::Fill) + .height(Length::Fill) + .center_x() + .center_y() + .into() + } else { + content.into() + } }, Mode::Create { name, body, loadout, tool, - name_input, - back_button, - create_button, + ref mut scroll, + ref mut body_type_buttons, + ref mut species_buttons, + ref mut tool_buttons, + ref mut name_input, + ref mut back_button, + ref mut create_button, } => { - let top_row = Row::with_children(vec![]); - let bottom_row = Row::with_children(vec![]); + let unselected_style = style::button::Style::new(imgs.icon_border) + .hover_image(imgs.icon_border_mo) + .press_image(imgs.icon_border_press); - Column::with_children(vec![top_row.into(), bottom_row.into()]) + let selected_style = style::button::Style::new(imgs.icon_border_pressed) + .hover_image(imgs.icon_border_mo) + .press_image(imgs.icon_border_press); + + let icon_button = |button, selected, msg, img| { + Container::new( + Button::<_, IcedRenderer>::new( + button, + Space::new(Length::Units(70), Length::Units(70)), + ) + .style(if selected { + selected_style + } else { + unselected_style + }) + .on_press(msg), + ) + .style(style::container::Style::image(img)) + }; + + let (body_m_ico, body_f_ico) = match body.species { + humanoid::Species::Human => (imgs.human_m, imgs.human_f), + humanoid::Species::Orc => (imgs.orc_m, imgs.orc_f), + humanoid::Species::Dwarf => (imgs.dwarf_m, imgs.dwarf_f), + humanoid::Species::Elf => (imgs.elf_m, imgs.elf_f), + humanoid::Species::Undead => (imgs.undead_m, imgs.undead_f), + humanoid::Species::Danari => (imgs.danari_m, imgs.danari_f), + }; + + let [ref mut body_m_button, ref mut body_f_button] = body_type_buttons; + let body_type = Row::with_children(vec![ + icon_button( + body_m_button, + matches!(body.body_type, humanoid::BodyType::Male), + Message::BodyType(humanoid::BodyType::Male), + body_m_ico, + ) + .into(), + icon_button( + body_f_button, + matches!(body.body_type, humanoid::BodyType::Female), + Message::BodyType(humanoid::BodyType::Female), + body_f_ico, + ) + .into(), + ]) + .spacing(1); + + let (human_icon, orc_icon, dwarf_icon, elf_icon, undead_icon, danari_icon) = + match body.body_type { + humanoid::BodyType::Male => ( + self.imgs.human_m, + self.imgs.orc_m, + self.imgs.dwarf_m, + self.imgs.elf_m, + self.imgs.undead_m, + self.imgs.danari_m, + ), + humanoid::BodyType::Female => ( + self.imgs.human_f, + self.imgs.orc_f, + self.imgs.dwarf_f, + self.imgs.elf_f, + self.imgs.undead_f, + self.imgs.danari_f, + ), + }; + + // TODO: tooltips + let [ref mut human_button, ref mut orc_button, ref mut dwarf_button, ref mut elf_button, ref mut undead_button, ref mut danari_button] = + species_buttons; + let species = Column::with_children(vec![ + Row::with_children(vec![ + icon_button( + human_button, + matches!(body.species, humanoid::Species::Human), + Message::Species(humanoid::Species::Human), + human_icon, + ) + .into(), + icon_button( + orc_button, + matches!(body.species, humanoid::Species::Orc), + Message::Species(humanoid::Species::Orc), + orc_icon, + ) + .into(), + icon_button( + dwarf_button, + matches!(body.species, humanoid::Species::Dwarf), + Message::Species(humanoid::Species::Dwarf), + dwarf_icon, + ) + .into(), + ]) + .spacing(1) + .into(), + Row::with_children(vec![ + icon_button( + elf_button, + matches!(body.species, humanoid::Species::Elf), + Message::Species(humanoid::Species::Elf), + elf_icon, + ) + .into(), + icon_button( + undead_button, + matches!(body.species, humanoid::Species::Undead), + Message::Species(humanoid::Species::Undead), + undead_icon, + ) + .into(), + icon_button( + danari_button, + matches!(body.species, humanoid::Species::Danari), + Message::Species(humanoid::Species::Danari), + danari_icon, + ) + .into(), + ]) + .spacing(1) + .into(), + ]) + .spacing(1); + + let [ref mut sword_button, ref mut daggers_button, ref mut axe_button, ref mut hammer_button, ref mut bow_button, ref mut staff_button] = + tool_buttons; + let tool = Column::with_children(vec![ + Row::with_children(vec![ + icon_button( + sword_button, + *tool == STARTER_SWORD, + Message::Tool(STARTER_SWORD), + imgs.sword, + ) + .into(), + icon_button( + daggers_button, + *tool == STARTER_DAGGER, + Message::Tool(STARTER_DAGGER), + imgs.daggers, + ) + .into(), + icon_button( + axe_button, + *tool == STARTER_AXE, + Message::Tool(STARTER_AXE), + imgs.axe, + ) + .into(), + ]) + .spacing(1) + .into(), + Row::with_children(vec![ + icon_button( + hammer_button, + *tool == STARTER_HAMMER, + Message::Tool(STARTER_HAMMER), + imgs.hammer, + ) + .into(), + icon_button( + bow_button, + *tool == STARTER_BOW, + Message::Tool(STARTER_BOW), + imgs.bow, + ) + .into(), + icon_button( + staff_button, + *tool == STARTER_STAFF, + Message::Tool(STARTER_STAFF), + imgs.staff, + ) + .into(), + ]) + .spacing(1) + .into(), + ]) + .spacing(1); + + let column_content = vec![body_type.into(), species.into(), tool.into()]; + + let right_column = Container::new( + Column::with_children(vec![ + Text::new(i18n.get("char_selection.character_creation")) + .size(fonts.cyri.scale(26)) + .into(), + Scrollable::new(scroll) + .push( + Column::with_children(column_content) + .align_items(Align::Center) + .spacing(16), + ) + .into(), + ]) + .spacing(20) + .padding(10) + .width(Length::Fill) + .align_items(Align::Center), + ) + .style(style::container::Style::color_with_image_border( + Rgba::new(0, 0, 0, 217), + imgs.gray_corner, + imgs.gray_edge, + )) + .padding(10) + .width(Length::Units(360)) // TODO: see if we can get iced to work with settings below + //.max_width(360) + //.width(Length::Fill) + .height(Length::Fill); + + let top = Row::with_children(vec![ + right_column.into(), + MouseDetector::new(&mut self.mouse_detector, Length::Fill, Length::Fill).into(), + ]) + .padding(15) + .width(Length::Fill) + .height(Length::Fill); + + let back = neat_button( + back_button, + i18n.get("common.back"), + FILL_FRAC_ONE, + button_style, + Some(Message::Back), + ); + + let name_input = BackgroundContainer::new( + Image::new(imgs.name_input) + .height(Length::Units(40)) + .fix_aspect_ratio(), + TextInput::new(name_input, "Character Name", &name, Message::Name) + .size(25) + .on_submit(Message::CreateCharacter), + ) + .padding(Padding::new().horizontal(7).top(5)); + + let create = neat_button( + create_button, + i18n.get("common.create"), + FILL_FRAC_ONE, + button_style, + (!name.is_empty()).then_some(Message::CreateCharacter), + ); + + let bottom = Row::with_children(vec![ + Container::new(back) + .width(Length::Fill) + .height(Length::Units(40)) + .into(), + Container::new(name_input) + .width(Length::Fill) + .center_x() + .into(), + Container::new(create) + .width(Length::Fill) + .height(Length::Units(40)) + .align_x(Align::End) + .into(), + ]) + .align_items(Align::End); + + Column::with_children(vec![top.into(), bottom.into()]) .width(Length::Fill) .height(Length::Fill) + .into() }, }; Container::new( - Column::with_children(vec![top_text.into(), content.into()]) + Column::with_children(vec![top_text.into(), content]) .spacing(3) .width(Length::Fill) .height(Length::Fill), @@ -494,15 +864,7 @@ impl Controls { .into() } - fn update( - &mut self, - message: Message, - events: &mut Vec, - settings: &Settings, - characters: &[CharacterItem], - ) { - let servers = &settings.networking.servers; - + fn update(&mut self, message: Message, events: &mut Vec, characters: &[CharacterItem]) { match message { Message::Back => { if matches!(&self.mode, Mode::Create { .. }) { @@ -518,15 +880,20 @@ impl Controls { .. } = &self.mode { - // TODO: eliminate option in character id + // TODO: eliminate option in character id? if let Some(id) = characters.get(*selected).and_then(|i| i.character.id) { events.push(Event::Play(id)); } } }, + Message::Select(idx) => { + if let Mode::Select { selected, .. } = &mut self.mode { + *selected = Some(idx); + } + }, Message::Delete(idx) => { - if let Some(id) = characters.get(idx).and_then(|i| i.character.id) { - events.push(Event::DeleteCharacter(id)); + if let Mode::Select { info_content, .. } = &mut self.mode { + *info_content = Some(InfoContent::Deletion(idx)); } }, Message::ChangeServer => { @@ -544,7 +911,7 @@ impl Controls { { events.push(Event::AddCharacter { alias: name.clone(), - tool: tool.map(String::from), + tool: String::from(*tool), body: comp::Body::Humanoid(*body), }); self.mode = Mode::select(); @@ -555,16 +922,54 @@ impl Controls { *name = value; } }, + Message::BodyType(value) => { + if let Mode::Create { body, .. } = &mut self.mode { + body.body_type = value; + body.validate(); + } + }, + Message::Species(value) => { + if let Mode::Create { body, .. } = &mut self.mode { + body.species = value; + body.validate(); + } + }, + Message::Tool(value) => { + if let Mode::Create { tool, loadout, .. } = &mut self.mode { + *tool = value; + loadout.active_item = Some(LoadoutBuilder::default_item_config_from_str(*tool)); + } + }, + Message::ConfirmDeletion => { + if let Mode::Select { info_content, .. } = &mut self.mode { + if let Some(InfoContent::Deletion(idx)) = info_content { + if let Some(id) = characters.get(*idx).and_then(|i| i.character.id) { + events.push(Event::DeleteCharacter(id)); + } + *info_content = None; + } + } + }, + Message::CancelDeletion => { + if let Mode::Select { info_content, .. } = &mut self.mode { + if let Some(InfoContent::Deletion(idx)) = info_content { + *info_content = None; + } + } + }, } } /// Get the character to display - pub fn display_character(&self, characters: &[CharacterItem]) -> Option<&CharacterItem> { - match self.mode { - // TODO - Mode::Select { .. } => None, - // TODO - Mode::Create { .. } => None, + pub fn display_body_loadout<'a>( + &'a self, + characters: &'a [CharacterItem], + ) -> Option<(comp::Body, &'a comp::Loadout)> { + match &self.mode { + Mode::Select { selected, .. } => selected + .and_then(|idx| characters.get(idx)) + .map(|i| (i.body, &i.loadout)), + Mode::Create { loadout, body, .. } => Some((comp::Body::Humanoid(*body), loadout)), } } } @@ -607,44 +1012,11 @@ impl CharSelectionUi { Self { ui, controls } } - pub fn display_character(&self, characters: &[CharacterItem]) -> Option<&CharacterItem> { - self.controls.display_character(characters) - } - - // TODO - pub fn get_loadout(&mut self) -> Option { - // TODO: error gracefully - // TODO: don't clone - /*match &mut self.mode { - Mode::Select(character_list) => { - if let Some(data) = character_list { - data.get(self.selected_character).map(|c| c.loadout.clone()) - } else { - None - } - }, - Mode::Create { loadout, tool, .. } => { - loadout.active_item = tool.map(|tool| comp::ItemConfig { - item: comp::Item::new_from_asset_expect(tool), - ability1: None, - ability2: None, - ability3: None, - block_ability: None, - dodge_ability: None, - }); - loadout.chest = Some(comp::Item::new_from_asset_expect( - "common.items.armor.starter.rugged_chest", - )); - loadout.pants = Some(comp::Item::new_from_asset_expect( - "common.items.armor.starter.rugged_pants", - )); - loadout.foot = Some(comp::Item::new_from_asset_expect( - "common.items.armor.starter.sandals_0", - )); - Some(loadout.clone()) - }, - }*/ - None + pub fn display_body_loadout<'a>( + &'a self, + characters: &'a [CharacterItem], + ) -> Option<(comp::Body, &'a comp::Loadout)> { + self.controls.display_body_loadout(characters) } pub fn handle_event(&mut self, event: window::Event) -> bool { @@ -654,15 +1026,13 @@ impl CharSelectionUi { true }, window::Event::MouseButton(_, window::PressState::Pressed) => { - // TODO: implement this with iced - // !self.ui.no_widget_capturing_mouse() - false + !self.controls.mouse_detector.mouse_over() }, _ => false, } } - // TODO: do we need whole client here or just character list + // TODO: do we need whole client here or just character list? pub fn maintain(&mut self, global_state: &mut GlobalState, client: &mut Client) -> Vec { let mut events = Vec::new(); @@ -672,17 +1042,13 @@ impl CharSelectionUi { ); messages.into_iter().for_each(|message| { - self.controls.update( - message, - &mut events, - &global_state.settings, - &client.character_list.characters, - ) + self.controls + .update(message, &mut events, &client.character_list.characters) }); events } - // TODO: do we need globals + // TODO: do we need globals? pub fn render(&self, renderer: &mut Renderer) { self.ui.render(renderer); } } diff --git a/voxygen/src/menu/main/mod.rs b/voxygen/src/menu/main/mod.rs index 2abeb9d017..ed94928f19 100644 --- a/voxygen/src/menu/main/mod.rs +++ b/voxygen/src/menu/main/mod.rs @@ -10,7 +10,7 @@ use crate::{ }; use client_init::{ClientInit, Error as InitError, Msg as InitMsg}; use common::{assets::Asset, comp, span}; -use tracing::{error, warn}; +use tracing::error; use ui::{Event as MainMenuEvent, MainMenuUi}; pub struct MainMenuState { @@ -221,12 +221,11 @@ impl PlayState for MainMenuState { } => { let mut net_settings = &mut global_state.settings.networking; net_settings.username = username.clone(); + net_settings.default_server = server_address.clone(); if !net_settings.servers.contains(&server_address) { net_settings.servers.push(server_address.clone()); } - if let Err(err) = global_state.settings.save_to_file() { - warn!("Failed to save settings: {:?}", err); - } + global_state.settings.save_to_file_warn(); attempt_login( &mut global_state.settings, diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index 1ac637a4e7..b64e3bfd8f 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -202,6 +202,7 @@ impl Banner { .padding(10) .height(Length::FillPortion(25)) .into(), + // TODO: i18n Column::with_children(vec![ BackgroundContainer::new( Image::new(imgs.input_bg) @@ -209,7 +210,7 @@ impl Banner { .fix_aspect_ratio(), TextInput::new( &mut self.username, - "Username", + i18n.get("main.username"), &login_info.username, Message::Username, ) @@ -224,7 +225,7 @@ impl Banner { .fix_aspect_ratio(), TextInput::new( &mut self.password, - "Password", + i18n.get("main.password"), &login_info.password, Message::Password, ) @@ -240,7 +241,7 @@ impl Banner { .fix_aspect_ratio(), TextInput::new( &mut self.server, - "Server", + i18n.get("main.server"), &login_info.server, Message::Server, ) diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index 9a7b50bc51..092a5e5c5d 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -266,7 +266,8 @@ impl IcedRenderer { let mesh_index = *mesh_index; let linear_color = *linear_color; // Could potentially pass this in as part of the extras - let offset = offset.map(|e| e as f32 * p_scale) / half_res; + let offset = + offset.map(|e| e as f32 * p_scale) / half_res * Vec2::new(-1.0, 1.0); (0..*glyph_count).map(move |i| (mesh_index + i * 6, linear_color, offset)) }) .zip(self.last_glyph_verts.iter()) @@ -371,6 +372,68 @@ impl IcedRenderer { Aabr { min, max } } + fn position_glyphs( + &mut self, + bounds: iced::Rectangle, + horizontal_alignment: iced::HorizontalAlignment, + vertical_alignment: iced::VerticalAlignment, + text: &str, + size: u16, + font: FontId, + ) -> Vec { + use glyph_brush::{GlyphCruncher, HorizontalAlign, VerticalAlign}; + // TODO: add option to align based on the geometry of the rendered glyphs + // instead of all possible glyphs + let (x, h_align) = match horizontal_alignment { + iced::HorizontalAlignment::Left => (bounds.x, HorizontalAlign::Left), + iced::HorizontalAlignment::Center => (bounds.center_x(), HorizontalAlign::Center), + iced::HorizontalAlignment::Right => (bounds.x + bounds.width, HorizontalAlign::Right), + }; + + let (y, v_align) = match vertical_alignment { + iced::VerticalAlignment::Top => (bounds.y, VerticalAlign::Top), + iced::VerticalAlignment::Center => (bounds.center_y(), VerticalAlign::Center), + iced::VerticalAlignment::Bottom => (bounds.y + bounds.height, VerticalAlign::Bottom), + }; + + let p_scale = self.p_scale; + + let section = glyph_brush::Section { + screen_position: (x * p_scale, y * p_scale), + bounds: (bounds.width * p_scale, bounds.height * p_scale), + layout: glyph_brush::Layout::Wrap { + line_breaker: Default::default(), + h_align, + v_align, + }, + text: vec![glyph_brush::Text { + text, + scale: (size as f32 * p_scale).into(), + font_id: font.0, + extra: (), + }], + }; + + self + .cache + .glyph_cache_mut() + .glyphs(section) + // We would still have to generate vertices for these even if they have no pixels + // Note: this is somewhat hacky and could fail if there is a non-whitespace character + // that is not visible (to solve this we could use the extra values in + // queue_pre_positioned to keep track of which glyphs are actually returned by + // proccess_queued) + .filter(|g| { + !text[g.byte_index..] + .chars() + .next() + .unwrap() + .is_whitespace() + }) + .cloned() + .collect() + } + fn draw_primitive(&mut self, primitive: Primitive, offset: Vec2, renderer: &mut Renderer) { match primitive { Primitive::Group { primitives } => { @@ -391,8 +454,8 @@ impl IcedRenderer { let (graphic_id, rotation) = handle; let gl_aabr = self.gl_aabr(iced::Rectangle { - x: bounds.x + offset.x as f32, - y: bounds.y + offset.y as f32, + x: bounds.x - offset.x as f32, + y: bounds.y - offset.y as f32, ..bounds }); @@ -468,8 +531,8 @@ impl IcedRenderer { self.switch_state(State::Plain); let gl_aabr = self.gl_aabr(iced::Rectangle { - x: bounds.x + offset.x as f32, - y: bounds.y + offset.y as f32, + x: bounds.x - offset.x as f32, + y: bounds.y - offset.y as f32, ..bounds }); @@ -497,8 +560,8 @@ impl IcedRenderer { self.switch_state(State::Plain); let gl_aabr = self.gl_aabr(iced::Rectangle { - x: bounds.x + offset.x as f32, - y: bounds.y + offset.y as f32, + x: bounds.x - offset.x as f32, + y: bounds.y - offset.y as f32, ..bounds }); diff --git a/voxygen/src/ui/ice/renderer/widget/mod.rs b/voxygen/src/ui/ice/renderer/widget/mod.rs index 3cd501dcf3..d165b6b2a8 100644 --- a/voxygen/src/ui/ice/renderer/widget/mod.rs +++ b/voxygen/src/ui/ice/renderer/widget/mod.rs @@ -5,6 +5,7 @@ mod column; mod compound_graphic; mod container; mod image; +mod mouse_detector; mod overlay; mod row; mod scrollable; diff --git a/voxygen/src/ui/ice/renderer/widget/mouse_detector.rs b/voxygen/src/ui/ice/renderer/widget/mouse_detector.rs new file mode 100644 index 0000000000..218828c16a --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/mouse_detector.rs @@ -0,0 +1,9 @@ +use super::super::{super::widget::mouse_detector, IcedRenderer, Primitive}; +use iced::{mouse, Rectangle}; + +impl mouse_detector::Renderer for IcedRenderer { + fn draw(&mut self, _bounds: Rectangle) -> Self::Output { + // TODO: mouse interaction if in bounds?? + (Primitive::Nothing, mouse::Interaction::default()) + } +} diff --git a/voxygen/src/ui/ice/renderer/widget/text.rs b/voxygen/src/ui/ice/renderer/widget/text.rs index 7360aeaa6e..99af86aa5e 100644 --- a/voxygen/src/ui/ice/renderer/widget/text.rs +++ b/voxygen/src/ui/ice/renderer/widget/text.rs @@ -42,54 +42,14 @@ impl text::Renderer for IcedRenderer { horizontal_alignment: HorizontalAlignment, vertical_alignment: VerticalAlignment, ) -> Self::Output { - use glyph_brush::{HorizontalAlign, VerticalAlign}; - // glyph_brush thought it would be a great idea to change what the bounds and - // position mean based on the alignment - // TODO: add option to align based on the geometry of the rendered glyphs - // instead of all possible glyphs - let (x, h_align) = match horizontal_alignment { - HorizontalAlignment::Left => (bounds.x, HorizontalAlign::Left), - HorizontalAlignment::Center => (bounds.center_x(), HorizontalAlign::Center), - HorizontalAlignment::Right => (bounds.x + bounds.width, HorizontalAlign::Right), - }; - - let (y, v_align) = match vertical_alignment { - VerticalAlignment::Top => (bounds.y, VerticalAlign::Top), - VerticalAlignment::Center => (bounds.center_y(), VerticalAlign::Center), - VerticalAlignment::Bottom => (bounds.y + bounds.height, VerticalAlign::Bottom), - }; - - let p_scale = self.p_scale; - - let section = glyph_brush::Section { - screen_position: (x * p_scale, y * p_scale), - bounds: (bounds.width * p_scale, bounds.height * p_scale), - layout: glyph_brush::Layout::Wrap { - line_breaker: Default::default(), - h_align, - v_align, - }, - text: vec![glyph_brush::Text { - text: content, - scale: (size as f32 * p_scale).into(), - font_id: font.0, - extra: (), - }], - }; - - let glyphs = self - .cache - .glyph_cache_mut() - .glyphs(section) - .filter(|g| { - !content[g.byte_index..] - .chars() - .next() - .unwrap() - .is_whitespace() - }) - .cloned() - .collect::>(); + let glyphs = self.position_glyphs( + bounds, + horizontal_alignment, + vertical_alignment, + content, + size, + font, + ); ( Primitive::Text { diff --git a/voxygen/src/ui/ice/renderer/widget/text_input.rs b/voxygen/src/ui/ice/renderer/widget/text_input.rs index 802874c925..ce62bf7277 100644 --- a/voxygen/src/ui/ice/renderer/widget/text_input.rs +++ b/voxygen/src/ui/ice/renderer/widget/text_input.rs @@ -60,7 +60,7 @@ impl text_input::Renderer for IcedRenderer { font_id: font.0, extra: (), }], - }; + i; let font = glyph_calculator.fonts()[font.0].as_scaled(size as f32); let space_id = font.glyph_id(' '); let x_id = font.glyph_id('x'); @@ -215,9 +215,22 @@ impl text_input::Renderer for IcedRenderer { }; let display_text = text.unwrap_or(if state.is_focused() { "" } else { placeholder }); - let section = glyph_brush::Section { - // Note: clip offset is an integer so we don't have to worry about not accounting for - // that here where the alignment of the glyphs with pixels affects rasterization + // Note: clip offset is an integer so we don't have to worry about not + // accounting for that here where the alignment of the glyphs with + // pixels affects rasterization + let glyphs = self.position_glyphs( + Rectangle { + width: 1000.0, // hacky + ..bounds + }, + iced::HorizontalAlignment::Left, + iced::VerticalAlignment::Center, + display_text, + size, + font, + ); + // TODO: delete if new arrangment behaves nicely + /*let section = glyph_brush::Section { screen_position: (text_bounds.x * p_scale, text_bounds.center_y() * p_scale), bounds: ( 10000.0, /* text_bounds.width * p_scale */ @@ -234,26 +247,7 @@ impl text_input::Renderer for IcedRenderer { font_id: font.0, extra: (), }], - }; - - let glyphs = self - .cache - .glyph_cache_mut() - .glyphs(section) - // We would still have to generate vertices for these even if they have no pixels - // Note: this is somewhat hacky and could fail if there is a non-whitespace character - // that is not visible (to solve this we could use the extra values in - // queue_pre_positioned to keep track of which glyphs are actually returned by - // proccess_queued) - .filter(|g| { - !display_text[g.byte_index..] - .chars() - .next() - .unwrap() - .is_whitespace() - }) - .cloned() - .collect::>(); + };*/ let text_primitive = Primitive::Text { glyphs, diff --git a/voxygen/src/ui/ice/widget/aspect_ratio_container.rs b/voxygen/src/ui/ice/widget/aspect_ratio_container.rs index f92832f987..e4ad81ea72 100644 --- a/voxygen/src/ui/ice/widget/aspect_ratio_container.rs +++ b/voxygen/src/ui/ice/widget/aspect_ratio_container.rs @@ -1,4 +1,6 @@ -use iced::{layout, Clipboard, Element, Event, Hasher, Layout, Length, Point, Rectangle, Widget}; +use iced::{ + layout, Clipboard, Element, Event, Hasher, Layout, Length, Point, Rectangle, Size, Widget, +}; use std::{hash::Hash, u32}; // Note: it might be more efficient to make this generic over the content type? @@ -70,17 +72,15 @@ impl<'a, M, R> Widget for AspectRatioContainer<'a, M, R> where R: self::Renderer, { - fn width(&self) -> Length { Length::Fill } + fn width(&self) -> Length { Length::Shrink } - fn height(&self) -> Length { Length::Fill } + fn height(&self) -> Length { Length::Shrink } fn layout(&self, renderer: &R, limits: &layout::Limits) -> layout::Node { let limits = limits .loose() .max_width(self.max_width) - .max_height(self.max_height) - .width(self.width()) - .height(self.height()); + .max_height(self.max_height); let aspect_ratio = match &self.aspect_ratio { AspectRatio::Image(handle) => { @@ -107,7 +107,9 @@ where limits.max_height((max_width / aspect_ratio) as u32) }; - let content = self.content.layout(renderer, &limits.loose()); + // Remove fill limits in case one of the parents was Shrink + let limits = layout::Limits::new(Size::ZERO, limits.max()); + let content = self.content.layout(renderer, &limits); layout::Node::with_children(limits.max(), vec![content]) } diff --git a/voxygen/src/ui/ice/widget/mod.rs b/voxygen/src/ui/ice/widget/mod.rs index b7db600fea..d771b95138 100644 --- a/voxygen/src/ui/ice/widget/mod.rs +++ b/voxygen/src/ui/ice/widget/mod.rs @@ -3,6 +3,7 @@ pub mod background_container; pub mod compound_graphic; pub mod fill_text; pub mod image; +pub mod mouse_detector; pub mod overlay; pub mod stack; @@ -11,5 +12,6 @@ pub use self::{ background_container::{BackgroundContainer, Padding}, fill_text::FillText, image::Image, + mouse_detector::MouseDetector, overlay::Overlay, }; diff --git a/voxygen/src/ui/ice/widget/mouse_detector.rs b/voxygen/src/ui/ice/widget/mouse_detector.rs new file mode 100644 index 0000000000..742fbb6f34 --- /dev/null +++ b/voxygen/src/ui/ice/widget/mouse_detector.rs @@ -0,0 +1,112 @@ +use iced::{ + layout, mouse, Clipboard, Element, Event, Hasher, Layout, Length, Point, Rectangle, Size, + Widget, +}; +use std::hash::Hash; + +#[derive(Debug, Default)] +pub struct State { + mouse_over: bool, +} +impl State { + pub fn mouse_over(&self) -> bool { self.mouse_over } +} + +#[derive(Debug)] +pub struct MouseDetector<'a> { + width: Length, + height: Length, + //on_enter: M, + //on_exit: M, + state: &'a mut State, +} + +impl<'a> MouseDetector<'a> { + pub fn new( + state: &'a mut State, + //on_enter: M, + //on_exit: M, + width: Length, + height: Length, + ) -> Self { + Self { + state, + //on_enter, + //on_exit, + width, + height, + } + } +} + +impl<'a, M, R> Widget for MouseDetector<'a> +where + R: self::Renderer, +{ + fn width(&self) -> Length { self.width } + + fn height(&self) -> Length { self.height } + + fn layout(&self, _renderer: &R, limits: &layout::Limits) -> layout::Node { + let limits = limits.width(self.width).height(self.height); + + layout::Node::new(limits.resolve(Size::ZERO)) + } + + fn on_event( + &mut self, + event: Event, + layout: Layout<'_>, + _cursor_position: Point, + _messages: &mut Vec, + _renderer: &R, + _clipboard: Option<&dyn Clipboard>, + ) { + if let Event::Mouse(mouse::Event::CursorMoved { x, y }) = event { + let bounds = layout.bounds(); + let mouse_over = x > bounds.x + && x < bounds.x + bounds.width + && y > bounds.y + && y < bounds.y + bounds.height; + if mouse_over != self.state.mouse_over { + self.state.mouse_over = mouse_over; + + /*messages.push(if mouse_over { + self.on_enter.clone() + } else { + self.on_exit.clone() + });*/ + } + } + } + + fn draw( + &self, + renderer: &mut R, + _defaults: &R::Defaults, + layout: Layout<'_>, + _cursor_position: Point, + ) -> R::Output { + renderer.draw(layout.bounds()) + } + + fn hash_layout(&self, state: &mut Hasher) { + struct Marker; + std::any::TypeId::of::().hash(state); + + self.width.hash(state); + self.height.hash(state); + } +} + +pub trait Renderer: iced::Renderer { + fn draw(&mut self, bounds: Rectangle) -> Self::Output; +} + +impl<'a, M, R> From> for Element<'a, M, R> +where + R: self::Renderer, + M: 'a, +{ + fn from(mouse_detector: MouseDetector<'a>) -> Element<'a, M, R> { Element::new(mouse_detector) } +} diff --git a/voxygen/src/ui/ice/widget/overlay.rs b/voxygen/src/ui/ice/widget/overlay.rs index b168e1f99b..7a85d4e8ee 100644 --- a/voxygen/src/ui/ice/widget/overlay.rs +++ b/voxygen/src/ui/ice/widget/overlay.rs @@ -116,14 +116,14 @@ where let over_size = over.size(); let size = limits.resolve(Size { - width: under_size.width.max(over_size.width), - height: under_size.height.max(over_size.height), + width: under_size.width.max(over_size.width + padding * 2.0), + height: under_size.height.max(over_size.height + padding * 2.0), }); over.move_to(Point::new(padding, padding)); over.align(self.horizontal_alignment, self.vertical_alignment, size); - layout::Node::with_children(size.pad(padding), vec![over, under]) + layout::Node::with_children(size, vec![over, under]) } fn on_event( @@ -135,9 +135,12 @@ where renderer: &R, clipboard: Option<&dyn Clipboard>, ) { + let mut children = layout.children(); + let over_layout = children.next().unwrap(); + self.over.on_event( event.clone(), - layout, + over_layout, cursor_position, messages, renderer, @@ -147,16 +150,11 @@ where // If mouse press check if over the overlay widget before sending to under // widget if !matches!(&event, Event::Mouse(mouse::Event::ButtonPressed(_))) - || !layout - .children() - .next() - .unwrap() - .bounds() - .contains(cursor_position) + || !over_layout.bounds().contains(cursor_position) { self.under.on_event( event, - layout, + children.next().unwrap(), cursor_position, messages, renderer, From 9348f5915a676f9c7618814c99cee8b6f965b39e Mon Sep 17 00:00:00 2001 From: Imbris Date: Mon, 14 Sep 2020 00:58:43 -0400 Subject: [PATCH 37/61] Update to the latest git iced --- Cargo.lock | 19 ++++++++------- voxygen/Cargo.toml | 4 ++-- voxygen/src/ui/ice/mod.rs | 4 +++- voxygen/src/ui/ice/renderer/mod.rs | 23 +++++++++++++++++++ voxygen/src/ui/ice/renderer/widget/slider.rs | 13 ++++++----- .../src/ui/ice/renderer/widget/text_input.rs | 6 ----- 6 files changed, 46 insertions(+), 23 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 9dd0bbee7d..2316f712f2 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1847,9 +1847,9 @@ dependencies = [ [[package]] name = "glam" -version = "0.8.7" +version = "0.9.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b00572b5b10070ac495be20a25b4c8d379d20bcdec8ea0c870022b620ec79b20" +checksum = "8637c7ec4fd0776c51eeab3e0d5d1aa7e440ece3fc2ee7d674e13c957287bfc1" [[package]] name = "glob" @@ -2158,12 +2158,12 @@ dependencies = [ [[package]] name = "iced_core" version = "0.2.1" -source = "git+https://github.com/hecrj/iced?rev=b5d842f#b5d842f877145c78f5d595a87cc1927bb6f5b86a" +source = "git+https://github.com/hecrj/iced?rev=4f2962d#4f2962d73f3bdeeca8a11817e404c45e91e2c2cc" [[package]] name = "iced_futures" version = "0.1.2" -source = "git+https://github.com/hecrj/iced?rev=b5d842f#b5d842f877145c78f5d595a87cc1927bb6f5b86a" +source = "git+https://github.com/hecrj/iced?rev=4f2962d#4f2962d73f3bdeeca8a11817e404c45e91e2c2cc" dependencies = [ "futures 0.3.5", "log", @@ -2173,19 +2173,20 @@ dependencies = [ [[package]] name = "iced_graphics" version = "0.1.0" -source = "git+https://github.com/hecrj/iced?rev=b5d842f#b5d842f877145c78f5d595a87cc1927bb6f5b86a" +source = "git+https://github.com/hecrj/iced?rev=4f2962d#4f2962d73f3bdeeca8a11817e404c45e91e2c2cc" dependencies = [ "bytemuck", "glam", "iced_native", "iced_style", "raw-window-handle", + "thiserror", ] [[package]] name = "iced_native" version = "0.2.2" -source = "git+https://github.com/hecrj/iced?rev=b5d842f#b5d842f877145c78f5d595a87cc1927bb6f5b86a" +source = "git+https://github.com/hecrj/iced?rev=4f2962d#4f2962d73f3bdeeca8a11817e404c45e91e2c2cc" dependencies = [ "iced_core", "iced_futures", @@ -2197,7 +2198,7 @@ dependencies = [ [[package]] name = "iced_style" version = "0.1.0" -source = "git+https://github.com/hecrj/iced?rev=b5d842f#b5d842f877145c78f5d595a87cc1927bb6f5b86a" +source = "git+https://github.com/hecrj/iced?rev=4f2962d#4f2962d73f3bdeeca8a11817e404c45e91e2c2cc" dependencies = [ "iced_core", ] @@ -2205,11 +2206,13 @@ dependencies = [ [[package]] name = "iced_winit" version = "0.1.1" -source = "git+https://github.com/hecrj/iced?rev=b5d842f#b5d842f877145c78f5d595a87cc1927bb6f5b86a" +source = "git+https://github.com/hecrj/iced?rev=4f2962d#4f2962d73f3bdeeca8a11817e404c45e91e2c2cc" dependencies = [ + "iced_futures", "iced_graphics", "iced_native", "log", + "thiserror", "winapi 0.3.9", "window_clipboard", "winit", diff --git a/voxygen/Cargo.toml b/voxygen/Cargo.toml index 21fd6d0e76..3a0bf5d634 100644 --- a/voxygen/Cargo.toml +++ b/voxygen/Cargo.toml @@ -35,8 +35,8 @@ winit = {version = "0.22.2", features = ["serde"]} conrod_core = {git = "https://gitlab.com/veloren/conrod.git", branch="copypasta_0.7"} conrod_winit = {git = "https://gitlab.com/veloren/conrod.git", branch="copypasta_0.7"} euc = {git = "https://github.com/zesterer/euc.git"} -iced = {package = "iced_native", git = "https://github.com/hecrj/iced", rev = "b5d842f"} -iced_winit = {git = "https://github.com/hecrj/iced", rev = "b5d842f"} +iced = {package = "iced_native", git = "https://github.com/hecrj/iced", rev = "4f2962d"} +iced_winit = {git = "https://github.com/hecrj/iced", rev = "4f2962d"} window_clipboard = "0.1.1" glyph_brush = "0.7.0" diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index b5464d432d..6538cef61a 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -152,11 +152,13 @@ impl IcedUi { ); let messages = user_interface.update( - self.events.drain(..), + &self.events, cursor_position, Some(&self.clipboard), &mut self.renderer, ); + // Clear events + self.events.clear(); let (primitive, mouse_interaction) = user_interface.draw(&mut self.renderer, cursor_position); diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index 092a5e5c5d..b1f200aa45 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -781,6 +781,29 @@ impl iced::Renderer for IcedRenderer { node } + + fn overlay( + &mut self, + (base_primitive, base_interaction): Self::Output, + (overlay_primitive, overlay_interaction): Self::Output, + overlay_bounds: iced::Rectangle, + ) -> Self::Output { + ( + Primitive::Group { + primitives: vec![base_primitive, Primitive::Clip { + bounds: iced::Rectangle { + // TODO: do we need this + 0.5? + width: overlay_bounds.width + 0.5, + height: overlay_bounds.height + 0.5, + ..overlay_bounds + }, + offset: Vec2::new(0, 0), + content: Box::new(overlay_primitive), + }], + }, + base_interaction.max(overlay_interaction), + ) + } } // TODO: impl Debugger diff --git a/voxygen/src/ui/ice/renderer/widget/slider.rs b/voxygen/src/ui/ice/renderer/widget/slider.rs index 056c60b423..e58c6acc80 100644 --- a/voxygen/src/ui/ice/renderer/widget/slider.rs +++ b/voxygen/src/ui/ice/renderer/widget/slider.rs @@ -1,7 +1,7 @@ use super::super::{super::Rotation, style, IcedRenderer, Primitive}; use common::util::srgba_to_linear; -use iced::{slider, mouse, Rectangle, Point}; use core::ops::RangeInclusive; +use iced::{mouse, slider, Point, Rectangle}; use style::slider::{Bar, Cursor, Style}; const CURSOR_WIDTH: f32 = 10.0; @@ -10,7 +10,9 @@ const BAR_HEIGHT: f32 = 18.0; impl slider::Renderer for IcedRenderer { type Style = Style; - fn height(&self) -> u32 { 20 } + + const DEFAULT_HEIGHT: u16 = 20; + fn draw( &mut self, bounds: Rectangle, @@ -18,9 +20,8 @@ impl slider::Renderer for IcedRenderer { range: RangeInclusive, value: f32, is_dragging: bool, - style: &Self::Style + style: &Self::Style, ) -> Self::Output { - let bar_bounds = Rectangle { height: BAR_HEIGHT, ..bounds @@ -38,7 +39,7 @@ impl slider::Renderer for IcedRenderer { }; let (max, min) = range.into_inner(); - let offset = bounds.width as f32 * (max - min ) / (value - min); + let offset = bounds.width as f32 * (max - min) / (value - min); let cursor_bounds = Rectangle { x: bounds.x + offset - CURSOR_WIDTH / 2.0, y: bounds.y + if is_dragging { 2.0 } else { 0.0 }, @@ -74,6 +75,6 @@ impl slider::Renderer for IcedRenderer { // TODO Cursor text label vec![bar, cursor] }; - (Primitive::Group{primitives}, interaction) + (Primitive::Group { primitives }, interaction) } } diff --git a/voxygen/src/ui/ice/renderer/widget/text_input.rs b/voxygen/src/ui/ice/renderer/widget/text_input.rs index ce62bf7277..ec833214b3 100644 --- a/voxygen/src/ui/ice/renderer/widget/text_input.rs +++ b/voxygen/src/ui/ice/renderer/widget/text_input.rs @@ -11,14 +11,8 @@ const CURSOR_WIDTH: f32 = 2.0; const EXTRA_OFFSET: f32 = 10.0; impl text_input::Renderer for IcedRenderer { - type Font = FontId; type Style = (); - fn default_size(&self) -> u16 { - // TODO: make configurable - 20 - } - fn measure_value(&self, value: &str, size: u16, font: Self::Font) -> f32 { // Using the physical scale might make this cached info usable below? // Although we also have a position of the screen there so this could be useless From de7c9a9858279f241b96cdb2f4a14e744b47ab8d Mon Sep 17 00:00:00 2001 From: Yusuf Bera Ertan Date: Mon, 20 Jul 2020 17:07:08 +0300 Subject: [PATCH 38/61] add language selection menu to main menu screen --- assets/voxygen/i18n/en.ron | 1 + voxygen/src/menu/main/mod.rs | 18 +++- voxygen/src/menu/main/ui/login.rs | 140 ++++++++++++++++++++++++++++-- voxygen/src/menu/main/ui/mod.rs | 33 ++++++- 4 files changed, 181 insertions(+), 11 deletions(-) diff --git a/assets/voxygen/i18n/en.ron b/assets/voxygen/i18n/en.ron index 2c9c5e3f2d..6501b36463 100644 --- a/assets/voxygen/i18n/en.ron +++ b/assets/voxygen/i18n/en.ron @@ -114,6 +114,7 @@ Is the client up to date?"#, "main.connecting": "Connecting", "main.creating_world": "Creating world", "main.tip": "Tip:", + "main.select_language": "Select a language", // Welcome notice that appears the first time Veloren is started "main.notice": r#"Welcome to the alpha version of Veloren! diff --git a/voxygen/src/menu/main/mod.rs b/voxygen/src/menu/main/mod.rs index ed94928f19..b26213a3cb 100644 --- a/voxygen/src/menu/main/mod.rs +++ b/voxygen/src/menu/main/mod.rs @@ -5,8 +5,11 @@ use super::char_selection::CharSelectionState; #[cfg(feature = "singleplayer")] use crate::singleplayer::Singleplayer; use crate::{ - render::Renderer, settings::Settings, window::Event, Direction, GlobalState, PlayState, - PlayStateResult, + i18n::{i18n_asset_key, Localization}, + render::Renderer, + settings::Settings, + window::Event, + Direction, GlobalState, PlayState, PlayStateResult, }; use client_init::{ClientInit, Error as InitError, Msg as InitMsg}; use common::{assets::Asset, comp, span}; @@ -47,7 +50,7 @@ impl PlayState for MainMenuState { fn tick(&mut self, global_state: &mut GlobalState, events: Vec) -> PlayStateResult { span!(_guard, "tick", "::tick"); - let localized_strings = crate::i18n::Localization::load_expect( + let mut localized_strings = crate::i18n::Localization::load_expect( &crate::i18n::i18n_asset_key(&global_state.settings.language.selected_language), ); @@ -248,6 +251,15 @@ impl PlayState for MainMenuState { self.client_init = None; self.main_menu_ui.cancel_connection(); }, + MainMenuEvent::ChangeLanguage(new_language) => { + global_state.settings.language.selected_language = + new_language.language_identifier; + localized_strings = Localization::load_expect(&i18n_asset_key( + &global_state.settings.language.selected_language, + )); + localized_strings.log_missing_entries(); + self.main_menu_ui.update_language(localized_strings.clone()); + }, #[cfg(feature = "singleplayer")] MainMenuEvent::StartSingleplayer => { let singleplayer = Singleplayer::new(None); // TODO: Make client and server use the same thread pool diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index b64e3bfd8f..494e53561e 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -14,7 +14,10 @@ use crate::{ }, }, }; -use iced::{button, text_input, Align, Column, Container, Length, Row, Space, Text, TextInput}; +use iced::{ + button, scrollable, text_input, Align, Button, Column, Container, Length, Row, Scrollable, + Space, Text, TextInput, +}; use vek::*; const FILL_FRAC_ONE: f32 = 0.77; @@ -27,10 +30,12 @@ pub struct Screen { quit_button: button::State, settings_button: button::State, servers_button: button::State, + language_select_button: button::State, error_okay_button: button::State, - pub banner: Banner, + pub banner: LoginBanner, + language_selection: LanguageSelectBanner, } impl Screen { @@ -39,10 +44,12 @@ impl Screen { servers_button: Default::default(), settings_button: Default::default(), quit_button: Default::default(), + language_select_button: Default::default(), error_okay_button: Default::default(), - banner: Banner::new(), + banner: LoginBanner::new(), + language_selection: LanguageSelectBanner::new(), } } @@ -53,6 +60,9 @@ impl Screen { login_info: &LoginInfo, error: Option<&str>, i18n: &Localization, + is_selecting_language: bool, + selected_language_index: Option, + language_metadatas: &[crate::i18n::LanguageMetadata], button_style: style::button::Style, ) -> Element { let buttons = Column::with_children(vec![ @@ -77,6 +87,13 @@ impl Screen { button_style, Some(Message::Quit), ), + neat_button( + &mut self.language_select_button, + i18n.get("common.languages"), + FILL_FRAC_ONE, + button_style, + Some(Message::OpenLanguageMenu), + ), ]) .width(Length::Fill) .max_width(200) @@ -140,8 +157,19 @@ impl Screen { .padding(20) .into() } else { - self.banner - .view(fonts, imgs, login_info, i18n, button_style) + if is_selecting_language { + self.language_selection.view( + fonts, + imgs, + i18n, + language_metadatas, + selected_language_index, + button_style, + ) + } else { + self.banner + .view(fonts, imgs, login_info, i18n, button_style) + } }; let central_column = Container::new(central_content) @@ -164,7 +192,105 @@ impl Screen { } } -pub struct Banner { +pub struct LanguageSelectBanner { + okay_button: button::State, + language_buttons: Vec, + + selection_list: scrollable::State, +} + +impl LanguageSelectBanner { + fn new() -> Self { + Self { + okay_button: Default::default(), + language_buttons: Default::default(), + selection_list: Default::default(), + } + } + + fn view( + &mut self, + fonts: &Fonts, + imgs: &Imgs, + i18n: &Localization, + language_metadatas: &[crate::i18n::LanguageMetadata], + selected_language_index: Option, + button_style: style::button::Style, + ) -> Element { + let title = + Container::new(Text::new(i18n.get("main.select_language")).size(fonts.cyri.scale(35))) + .center_x(); + + let mut list = Scrollable::new(&mut self.selection_list) + .height(Length::Fill) + .width(Length::Fill) + .align_items(Align::Start); + + if self.language_buttons.len() != language_metadatas.len() { + self.language_buttons = vec![Default::default(); language_metadatas.len()]; + } + + for (i, (state, lang)) in self + .language_buttons + .iter_mut() + .zip(language_metadatas) + .enumerate() + { + let text = format!( + "{}{}", + if Some(i) == selected_language_index { + "-> " + } else { + " " + }, + lang.language_name, + ); + let button = Button::new( + state, + Container::new(Text::new(text).size(fonts.cyri.scale(25))) + .padding(5) + .center_y(), + ) + .width(Length::Fill) + .on_press(Message::LangaugeChanged(i)); + list = list.push(button); + } + + let okay_button = Container::new(neat_button( + &mut self.okay_button, + i18n.get("common.okay"), + FILL_FRAC_TWO, + button_style, + Some(Message::OpenLanguageMenu), + )) + .center_x() + .max_width(200); + + let content = Column::with_children(vec![title.into(), list.into(), okay_button.into()]) + .spacing(8) + .width(Length::Fill) + .height(Length::FillPortion(38)) + .align_items(Align::Center); + + let selection_menu = BackgroundContainer::new( + CompoundGraphic::from_graphics(vec![ + Graphic::image(imgs.banner_top, [138, 17], [0, 0]), + Graphic::rect(Rgba::new(0, 0, 0, 230), [130, 165], [4, 17]), + // TODO: use non image gradient + Graphic::gradient(Rgba::new(0, 0, 0, 230), Rgba::zero(), [130, 50], [4, 182]), + ]) + .fix_aspect_ratio() + .height(Length::Fill), + content, + ) + .padding(Padding::new().horizontal(8).vertical(15).bottom(50)) + .max_width(350); + + selection_menu.into() + } +} + +pub struct LoginBanner { pub username: text_input::State, pub password: text_input::State, pub server: text_input::State, @@ -174,7 +300,7 @@ pub struct Banner { singleplayer_button: button::State, } -impl Banner { +impl LoginBanner { fn new() -> Self { Self { username: Default::default(), diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 0cb9960e76..2b04d69469 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -4,7 +4,7 @@ mod login; mod servers; use crate::{ - i18n::{i18n_asset_key, Localization}, + i18n::{i18n_asset_key, LanguageMetadata, Localization}, render::Renderer, ui::{ self, @@ -78,6 +78,7 @@ pub enum Event { server_address: String, }, CancelLoginAttempt, + ChangeLanguage(LanguageMetadata), #[cfg(feature = "singleplayer")] StartSingleplayer, Quit, @@ -143,6 +144,9 @@ struct Controls { selected_server_index: Option, login_info: LoginInfo, + is_selecting_language: bool, + selected_language_index: Option, + time: f32, screen: Screen, @@ -156,6 +160,8 @@ enum Message { #[cfg(feature = "singleplayer")] Singleplayer, Multiplayer, + LangaugeChanged(usize), + OpenLanguageMenu, Username(String), Password(String), Server(String), @@ -206,6 +212,11 @@ impl Controls { .iter() .position(|f| f == &login_info.server); + let language_metadatas = crate::i18n::list_localizations(); + let selected_language_index = language_metadatas + .iter() + .position(|f| &f.language_identifier == &settings.language.selected_language); + Self { fonts, imgs, @@ -217,6 +228,9 @@ impl Controls { selected_server_index, login_info, + is_selecting_language: false, + selected_language_index, + time: 0.0, screen, @@ -256,6 +270,8 @@ impl Controls { self.imgs.bg }; + let language_metadatas = crate::i18n::list_localizations(); + // TODO: make any large text blocks scrollable so that if the area is to // small they can still be read let content = match &mut self.screen { @@ -266,6 +282,9 @@ impl Controls { &self.login_info, error.as_deref(), &self.i18n, + self.is_selecting_language, + self.selected_language_index, + &language_metadatas, button_style, ), Screen::Servers { screen } => screen.view( @@ -300,6 +319,7 @@ impl Controls { fn update(&mut self, message: Message, events: &mut Vec, settings: &Settings) { let servers = &settings.networking.servers; + let mut language_metadatas = crate::i18n::list_localizations(); match message { Message::Quit => events.push(Event::Quit), @@ -344,6 +364,11 @@ impl Controls { }); }, Message::Username(new_value) => self.login_info.username = new_value, + Message::LangaugeChanged(new_value) => { + self.selected_language_index = Some(new_value); + events.push(Event::ChangeLanguage(language_metadatas.remove(new_value))); + }, + Message::OpenLanguageMenu => self.is_selecting_language = !self.is_selecting_language, Message::Password(new_value) => self.login_info.password = new_value, Message::Server(new_value) => { self.login_info.server = new_value; @@ -506,6 +531,12 @@ impl<'a> MainMenuUi { Self { ui, controls } } + pub fn update_language(&mut self, i18n: std::sync::Arc) { + self.controls.i18n = i18n; + self.controls.fonts = Fonts::load(&self.controls.i18n.fonts, &mut self.ui) + .expect("Impossible to load fonts!"); + } + pub fn auth_trust_prompt(&mut self, auth_server: String) { self.controls.auth_trust_prompt(auth_server); } From 319c744611fc6fde727e8a60c78431ad073a5bda Mon Sep 17 00:00:00 2001 From: Yusuf Bera Ertan Date: Fri, 24 Jul 2020 22:19:57 +0300 Subject: [PATCH 39/61] change the text arrows in language select and server select with proper ui elements --- voxygen/src/menu/main/ui/login.rs | 28 +++++++++++-------- voxygen/src/menu/main/ui/mod.rs | 10 +++++-- voxygen/src/menu/main/ui/servers.rs | 43 +++++++++++++++++++---------- 3 files changed, 51 insertions(+), 30 deletions(-) diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index 494e53561e..44ff562128 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -222,6 +222,7 @@ impl LanguageSelectBanner { .center_x(); let mut list = Scrollable::new(&mut self.selection_list) + .spacing(8) .height(Length::Fill) .width(Length::Fill) .align_items(Align::Start); @@ -236,23 +237,26 @@ impl LanguageSelectBanner { .zip(language_metadatas) .enumerate() { - let text = format!( - "{}{}", - if Some(i) == selected_language_index { - "-> " - } else { - " " - }, - lang.language_name, - ); + let color = if Some(i) == selected_language_index { + (97, 255, 18) + } else { + (97, 97, 25) + }; let button = Button::new( state, - Container::new(Text::new(text).size(fonts.cyri.scale(25))) - .padding(5) + Container::new(Text::new(lang.language_name.clone()).size(fonts.cyri.scale(25))) + .padding(16) .center_y(), ) + .style( + style::button::Style::new(imgs.selection) + .hover_image(imgs.selection_hover) + .press_image(imgs.selection_press) + .image_color(vek::Rgba::new(color.0, color.1, color.2, 192)), + ) .width(Length::Fill) - .on_press(Message::LangaugeChanged(i)); + .min_height(56) + .on_press(Message::LanguageChanged(i)); list = list.push(button); } diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 2b04d69469..7e0144d70f 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -48,11 +48,14 @@ image_ids_ice! { loading_art: "voxygen.element.frames.loading_screen.loading_bg", loading_art_l: "voxygen.element.frames.loading_screen.loading_bg_l", loading_art_r: "voxygen.element.frames.loading_screen.loading_bg_r", + selection: "voxygen.element.frames.selection", + selection_hover: "voxygen.element.frames.selection_hover", + selection_press: "voxygen.element.frames.selection_press", } } // Randomly loaded background images -const BG_IMGS: [&str; 16] = [ +const BG_IMGS: [&str; 13] = [ "voxygen.background.bg_1", "voxygen.background.bg_2", "voxygen.background.bg_3", @@ -160,7 +163,7 @@ enum Message { #[cfg(feature = "singleplayer")] Singleplayer, Multiplayer, - LangaugeChanged(usize), + LanguageChanged(usize), OpenLanguageMenu, Username(String), Password(String), @@ -289,6 +292,7 @@ impl Controls { ), Screen::Servers { screen } => screen.view( &self.fonts, + &self.imgs, &settings.networking.servers, self.selected_server_index, &self.i18n, @@ -364,7 +368,7 @@ impl Controls { }); }, Message::Username(new_value) => self.login_info.username = new_value, - Message::LangaugeChanged(new_value) => { + Message::LanguageChanged(new_value) => { self.selected_language_index = Some(new_value); events.push(Event::ChangeLanguage(language_metadatas.remove(new_value))); }, diff --git a/voxygen/src/menu/main/ui/servers.rs b/voxygen/src/menu/main/ui/servers.rs index fe7bdb1631..5564c16cfa 100644 --- a/voxygen/src/menu/main/ui/servers.rs +++ b/voxygen/src/menu/main/ui/servers.rs @@ -1,12 +1,19 @@ -use super::Message; +use super::{Imgs, Message}; use crate::{ i18n::Localization, ui::{ fonts::IcedFonts as Fonts, - ice::{component::neat_button, style, Element}, + ice::{ + component::neat_button, + style, + widget::background_container::{BackgroundContainer, Padding}, + Element, + }, }, }; -use iced::{button, scrollable, Align, Button, Column, Container, Length, Scrollable, Text}; +use iced::{ + button, scrollable, Align, Button, Column, Container, Length, Row, Scrollable, Space, Text, +}; pub struct Screen { back_button: button::State, @@ -26,6 +33,7 @@ impl Screen { pub(super) fn view( &mut self, fonts: &Fonts, + imgs: &Imgs, servers: &[impl AsRef], selected_server_index: Option, i18n: &Localization, @@ -44,6 +52,7 @@ impl Screen { .align_x(Align::Center); let mut list = Scrollable::new(&mut self.servers_list) + .spacing(8) .align_items(Align::Start) .width(Length::Fill) .height(Length::Fill); @@ -53,22 +62,26 @@ impl Screen { } for (i, (state, server)) in self.server_buttons.iter_mut().zip(servers).enumerate() { - let text = format!( - "{}{}", - if Some(i) == selected_server_index { - "-> " - } else { - " " - }, - server.as_ref(), - ); + let color = if Some(i) == selected_server_index { + (97, 255, 18) + } else { + (97, 97, 25) + }; let button = Button::new( state, - Container::new(Text::new(text).size(fonts.cyri.scale(25))) - .padding(5) - .center_y(), + Container::new(Text::new(server.as_ref()).size(fonts.cyri.scale(30))) + .padding(24) + .center_y() + .height(Length::Fill), + ) + .style( + style::button::Style::new(imgs.selection) + .hover_image(imgs.selection_hover) + .press_image(imgs.selection_press) + .image_color(vek::Rgba::new(color.0, color.1, color.2, 255)), ) .width(Length::Fill) + .min_height(100) .on_press(Message::ServerChanged(i)); list = list.push(button); } From cc4c6b746a5118277826be0c74209b7cbbb36308 Mon Sep 17 00:00:00 2001 From: Yusuf Bera Ertan Date: Sat, 25 Jul 2020 11:57:02 +0300 Subject: [PATCH 40/61] add title to server select menu, don't use unneeded Container in language select menu --- assets/voxygen/i18n/en.ron | 2 ++ voxygen/src/menu/main/ui/login.rs | 6 +++--- voxygen/src/menu/main/ui/servers.rs | 28 +++++++++++++++++----------- 3 files changed, 22 insertions(+), 14 deletions(-) diff --git a/assets/voxygen/i18n/en.ron b/assets/voxygen/i18n/en.ron index 6501b36463..fe67fe335d 100644 --- a/assets/voxygen/i18n/en.ron +++ b/assets/voxygen/i18n/en.ron @@ -160,6 +160,8 @@ https://veloren.net/account/."#, "main.login.banned": "You have been banned with the following reason", "main.login.kicked": "You have been kicked with the following reason", + "main.servers.select_server": "Select a server", + /// End Main screen section diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index 44ff562128..f7039b400a 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -217,9 +217,9 @@ impl LanguageSelectBanner { selected_language_index: Option, button_style: style::button::Style, ) -> Element { - let title = - Container::new(Text::new(i18n.get("main.select_language")).size(fonts.cyri.scale(35))) - .center_x(); + let title = Text::new(i18n.get("main.select_language")) + .size(fonts.cyri.scale(35)) + .horizontal_alignment(iced::HorizontalAlignment::Center); let mut list = Scrollable::new(&mut self.selection_list) .spacing(8) diff --git a/voxygen/src/menu/main/ui/servers.rs b/voxygen/src/menu/main/ui/servers.rs index 5564c16cfa..7cf11cdf15 100644 --- a/voxygen/src/menu/main/ui/servers.rs +++ b/voxygen/src/menu/main/ui/servers.rs @@ -39,17 +39,23 @@ impl Screen { i18n: &Localization, button_style: style::button::Style, ) -> Element { - let button = neat_button( - &mut self.back_button, - i18n.get("common.back"), - 0.77_f32, - button_style, - Some(Message::Back), - ); - - let button = Container::new(Container::new(button).max_width(200)) + let title = Text::new(i18n.get("main.servers.select_server")) + .size(fonts.cyri.scale(35)) .width(Length::Fill) - .align_x(Align::Center); + .horizontal_alignment(iced::HorizontalAlignment::Center); + + let back_button = Container::new( + Container::new(neat_button( + &mut self.back_button, + i18n.get("common.back"), + 0.77_f32, + button_style, + Some(Message::Back), + )) + .max_width(200), + ) + .width(Length::Fill) + .align_x(Align::Center); let mut list = Scrollable::new(&mut self.servers_list) .spacing(8) @@ -88,7 +94,7 @@ impl Screen { Container::new( Container::new( - Column::with_children(vec![list.into(), button.into()]) + Column::with_children(vec![title.into(), list.into(), back_button.into()]) .width(Length::Fill) .height(Length::Fill) .spacing(10) From 79ee8dcc2a706add0d9e29f84baeb3009cbcd47a Mon Sep 17 00:00:00 2001 From: Yusuf Bera Ertan Date: Sat, 25 Jul 2020 16:52:04 +0300 Subject: [PATCH 41/61] add padding to language select and server select menu buttons so that the scrollbar doesn't overlap them --- assets/voxygen/i18n/en.ron | 2 +- voxygen/src/menu/main/ui/login.rs | 76 ++++++++++++++++------------- voxygen/src/menu/main/ui/mod.rs | 5 +- voxygen/src/menu/main/ui/servers.rs | 67 +++++++++++++++---------- 4 files changed, 90 insertions(+), 60 deletions(-) diff --git a/assets/voxygen/i18n/en.ron b/assets/voxygen/i18n/en.ron index fe67fe335d..8a298e3850 100644 --- a/assets/voxygen/i18n/en.ron +++ b/assets/voxygen/i18n/en.ron @@ -114,7 +114,6 @@ Is the client up to date?"#, "main.connecting": "Connecting", "main.creating_world": "Creating world", "main.tip": "Tip:", - "main.select_language": "Select a language", // Welcome notice that appears the first time Veloren is started "main.notice": r#"Welcome to the alpha version of Veloren! @@ -159,6 +158,7 @@ https://veloren.net/account/."#, "main.login.not_on_whitelist": "You need a Whitelist entry by an Admin to join", "main.login.banned": "You have been banned with the following reason", "main.login.kicked": "You have been kicked with the following reason", + "main.login.select_language": "Select a language", "main.servers.select_server": "Select a server", diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index f7039b400a..dd5112ed57 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -1,4 +1,4 @@ -use super::{Imgs, LoginInfo, Message}; +use super::{Imgs, LoginInfo, Message, FILL_FRAC_ONE, FILL_FRAC_TWO}; use crate::{ i18n::Localization, ui::{ @@ -20,8 +20,6 @@ use iced::{ }; use vek::*; -const FILL_FRAC_ONE: f32 = 0.77; -const FILL_FRAC_TWO: f32 = 0.53; const INPUT_WIDTH: u16 = 280; const INPUT_TEXT_SIZE: u16 = 24; @@ -217,47 +215,59 @@ impl LanguageSelectBanner { selected_language_index: Option, button_style: style::button::Style, ) -> Element { - let title = Text::new(i18n.get("main.select_language")) + // Reset button states if languages were added / removed + if self.language_buttons.len() != language_metadatas.len() { + self.language_buttons = vec![Default::default(); language_metadatas.len()]; + } + + let title = Text::new(i18n.get("main.login.select_language")) .size(fonts.cyri.scale(35)) .horizontal_alignment(iced::HorizontalAlignment::Center); let mut list = Scrollable::new(&mut self.selection_list) .spacing(8) .height(Length::Fill) - .width(Length::Fill) .align_items(Align::Start); - if self.language_buttons.len() != language_metadatas.len() { - self.language_buttons = vec![Default::default(); language_metadatas.len()]; - } - - for (i, (state, lang)) in self + let list_items = self .language_buttons .iter_mut() .zip(language_metadatas) .enumerate() - { - let color = if Some(i) == selected_language_index { - (97, 255, 18) - } else { - (97, 97, 25) - }; - let button = Button::new( - state, - Container::new(Text::new(lang.language_name.clone()).size(fonts.cyri.scale(25))) - .padding(16) - .center_y(), - ) - .style( - style::button::Style::new(imgs.selection) - .hover_image(imgs.selection_hover) - .press_image(imgs.selection_press) - .image_color(vek::Rgba::new(color.0, color.1, color.2, 192)), - ) - .width(Length::Fill) - .min_height(56) - .on_press(Message::LanguageChanged(i)); - list = list.push(button); + .map(|(i, (state, lang))| { + let color = if Some(i) == selected_language_index { + (97, 255, 18) + } else { + (97, 97, 25) + }; + let button = Button::new( + state, + Row::with_children(vec![ + Space::new(Length::FillPortion(5), Length::Units(0)).into(), + Text::new(lang.language_name.clone()) + .width(Length::FillPortion(95)) + .size(fonts.cyri.scale(25)) + .vertical_alignment(iced::VerticalAlignment::Center) + .into(), + ]), + ) + .style( + style::button::Style::new(imgs.selection) + .hover_image(imgs.selection_hover) + .press_image(imgs.selection_press) + .image_color(vek::Rgba::new(color.0, color.1, color.2, 192)), + ) + .min_height(56) + .on_press(Message::LanguageChanged(i)); + Row::with_children(vec![ + Space::new(Length::FillPortion(3), Length::Units(0)).into(), + button.width(Length::FillPortion(92)).into(), + Space::new(Length::FillPortion(5), Length::Units(0)).into(), + ]) + }); + + for item in list_items { + list = list.push(item); } let okay_button = Container::new(neat_button( @@ -287,7 +297,7 @@ impl LanguageSelectBanner { .height(Length::Fill), content, ) - .padding(Padding::new().horizontal(8).vertical(15).bottom(50)) + .padding(Padding::new().horizontal(5).top(15).bottom(50)) .max_width(350); selection_menu.into() diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 7e0144d70f..2b1a04760a 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -33,6 +33,9 @@ const UI_HIGHLIGHT_0: Color = Color::Rgba(0.79, 1.09, 1.09, 1.0);*/ pub const TEXT_COLOR: iced::Color = iced::Color::from_rgb(1.0, 1.0, 1.0); pub const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1.0, 0.2); +pub const FILL_FRAC_ONE: f32 = 0.77; +pub const FILL_FRAC_TWO: f32 = 0.53; + image_ids_ice! { struct Imgs { @@ -218,7 +221,7 @@ impl Controls { let language_metadatas = crate::i18n::list_localizations(); let selected_language_index = language_metadatas .iter() - .position(|f| &f.language_identifier == &settings.language.selected_language); + .position(|f| f.language_identifier == settings.language.selected_language); Self { fonts, diff --git a/voxygen/src/menu/main/ui/servers.rs b/voxygen/src/menu/main/ui/servers.rs index 7cf11cdf15..e84ed5fe27 100644 --- a/voxygen/src/menu/main/ui/servers.rs +++ b/voxygen/src/menu/main/ui/servers.rs @@ -1,4 +1,4 @@ -use super::{Imgs, Message}; +use super::{Imgs, Message, FILL_FRAC_ONE}; use crate::{ i18n::Localization, ui::{ @@ -48,7 +48,7 @@ impl Screen { Container::new(neat_button( &mut self.back_button, i18n.get("common.back"), - 0.77_f32, + FILL_FRAC_ONE, button_style, Some(Message::Back), )) @@ -63,33 +63,50 @@ impl Screen { .width(Length::Fill) .height(Length::Fill); + // Reset button states if servers were added / removed if self.server_buttons.len() != servers.len() { self.server_buttons = vec![Default::default(); servers.len()]; } - for (i, (state, server)) in self.server_buttons.iter_mut().zip(servers).enumerate() { - let color = if Some(i) == selected_server_index { - (97, 255, 18) - } else { - (97, 97, 25) - }; - let button = Button::new( - state, - Container::new(Text::new(server.as_ref()).size(fonts.cyri.scale(30))) - .padding(24) - .center_y() - .height(Length::Fill), - ) - .style( - style::button::Style::new(imgs.selection) - .hover_image(imgs.selection_hover) - .press_image(imgs.selection_press) - .image_color(vek::Rgba::new(color.0, color.1, color.2, 255)), - ) - .width(Length::Fill) - .min_height(100) - .on_press(Message::ServerChanged(i)); - list = list.push(button); + let list_items = + self.server_buttons + .iter_mut() + .zip(servers) + .enumerate() + .map(|(i, (state, server))| { + let color = if Some(i) == selected_server_index { + (97, 255, 18) + } else { + (97, 97, 25) + }; + let button = Button::new( + state, + Row::with_children(vec![ + Space::new(Length::FillPortion(5), Length::Units(0)).into(), + Text::new(server.as_ref()) + .size(fonts.cyri.scale(30)) + .width(Length::FillPortion(95)) + .vertical_alignment(iced::VerticalAlignment::Center) + .into(), + ]), + ) + .style( + style::button::Style::new(imgs.selection) + .hover_image(imgs.selection_hover) + .press_image(imgs.selection_press) + .image_color(vek::Rgba::new(color.0, color.1, color.2, 255)), + ) + .min_height(100) + .on_press(Message::ServerChanged(i)); + Row::with_children(vec![ + Space::new(Length::FillPortion(3), Length::Units(0)).into(), + button.width(Length::FillPortion(92)).into(), + Space::new(Length::FillPortion(5), Length::Units(0)).into(), + ]) + }); + + for item in list_items { + list = list.push(item); } Container::new( From fcc95d41362f2844c0a75c81a47ad86f01101cf6 Mon Sep 17 00:00:00 2001 From: Imbris Date: Wed, 16 Sep 2020 03:16:41 -0400 Subject: [PATCH 42/61] Remove background banner from login screen input boxes and move logo to right corner --- voxygen/src/menu/main/ui/login.rs | 51 +++++++++++++++---------------- voxygen/src/menu/main/ui/mod.rs | 17 +++++++---- 2 files changed, 35 insertions(+), 33 deletions(-) diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index dd5112ed57..d9ca8bc8e4 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -62,6 +62,7 @@ impl Screen { selected_language_index: Option, language_metadatas: &[crate::i18n::LanguageMetadata], button_style: style::button::Style, + version: &str, ) -> Element { let buttons = Column::with_children(vec![ neat_button( @@ -94,7 +95,7 @@ impl Screen { ), ]) .width(Length::Fill) - .max_width(200) + .max_width(160) .spacing(5); let buttons = Container::new(buttons) @@ -176,7 +177,18 @@ impl Screen { .center_x() .center_y(); - let right_column = Space::new(Length::Fill, Length::Fill); + let v_logo = Container::new(Image::new(imgs.v_logo).fix_aspect_ratio()) + .padding(3) + .width(Length::Units(230)); + + let version = iced::Text::new(version).size(fonts.cyri.scale(15)); + + let right_column = Container::new( + Column::with_children(vec![v_logo.into(), version.into()]).align_items(Align::Center), + ) + .width(Length::Fill) + .height(Length::Fill) + .align_x(Align::End); Row::with_children(vec![ left_column, @@ -338,11 +350,6 @@ impl LoginBanner { let input_text_size = fonts.cyri.scale(INPUT_TEXT_SIZE); let banner_content = Column::with_children(vec![ - Container::new(Image::new(imgs.v_logo).fix_aspect_ratio()) - .padding(10) - .height(Length::FillPortion(25)) - .into(), - // TODO: i18n Column::with_children(vec![ BackgroundContainer::new( Image::new(imgs.input_bg) @@ -391,10 +398,10 @@ impl LoginBanner { .padding(Padding::new().horizontal(7).top(5)) .into(), ]) - .spacing(10) - .height(Length::FillPortion(35)) + .spacing(5) + //.height(Length::Units(200)) .into(), - Space::new(Length::Fill, Length::FillPortion(2)).into(), + Space::new(Length::Fill, Length::Units(8)).into(), Column::with_children(vec![ neat_button( &mut self.multiplayer_button, @@ -413,28 +420,18 @@ impl LoginBanner { ), ]) .max_width(200) - .height(Length::FillPortion(38)) + .height(Length::Units(200)) .spacing(8) .into(), ]) .width(Length::Fill) - .height(Length::Fill) .align_items(Align::Center); - let banner = BackgroundContainer::new( - CompoundGraphic::from_graphics(vec![ - Graphic::image(imgs.banner_top, [138, 17], [0, 0]), - Graphic::rect(Rgba::new(0, 0, 0, 230), [130, 165], [4, 17]), - // TODO: use non image gradient - Graphic::gradient(Rgba::new(0, 0, 0, 230), Rgba::zero(), [130, 50], [4, 182]), - ]) - .fix_aspect_ratio() - .height(Length::Fill), - banner_content, - ) - .padding(Padding::new().horizontal(8).vertical(15)) - .max_width(350); - - banner.into() + Container::new(banner_content) + .height(Length::Fill) + .center_y() + .into() + //.padding(Padding::new().horizontal(8).vertical(15)) + //.max_width(350); } } diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 2b1a04760a..72deaf34b8 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -253,11 +253,6 @@ impl Controls { .text_color(TEXT_COLOR) .disabled_text_color(DISABLED_TEXT_COLOR); - let version = iced::Text::new(&self.version) - .size(self.fonts.cyri.scale(15)) - .width(Length::Fill) - .horizontal_alignment(HorizontalAlignment::Right); - let alpha = iced::Text::new(&self.alpha) .size(self.fonts.cyri.scale(12)) .width(Length::Fill) @@ -266,7 +261,16 @@ impl Controls { let top_text = Row::with_children(vec![ Space::new(Length::Fill, Length::Shrink).into(), alpha.into(), - version.into(), + if matches!(&self.screen, Screen::Login { .. }) { + // Login screen shows the Velroen logo over the version + Space::new(Length::Fill, Length::Shrink).into() + } else { + iced::Text::new(&self.version) + .size(self.fonts.cyri.scale(15)) + .width(Length::Fill) + .horizontal_alignment(HorizontalAlignment::Right) + .into() + }, ]) .width(Length::Fill); @@ -292,6 +296,7 @@ impl Controls { self.selected_language_index, &language_metadatas, button_style, + &self.version, ), Screen::Servers { screen } => screen.view( &self.fonts, From ecf3d0d1905107f7d98dfe1a79a86fe22444c5eb Mon Sep 17 00:00:00 2001 From: Imbris Date: Fri, 18 Sep 2020 01:44:28 -0400 Subject: [PATCH 43/61] Loading screen: show animated gears instead of status message, show tips --- voxygen/src/menu/main/ui/connecting.rs | 79 ++++++++++++++++++-------- voxygen/src/menu/main/ui/mod.rs | 51 ++++++----------- 2 files changed, 71 insertions(+), 59 deletions(-) diff --git a/voxygen/src/menu/main/ui/connecting.rs b/voxygen/src/menu/main/ui/connecting.rs index f063eaae5a..07a14fa221 100644 --- a/voxygen/src/menu/main/ui/connecting.rs +++ b/voxygen/src/menu/main/ui/connecting.rs @@ -1,17 +1,19 @@ -use super::{ConnectionState, Message}; +use super::{ConnectionState, Imgs, Message}; use crate::{ i18n::Localization, ui::{ fonts::IcedFonts as Fonts, - ice::{component::neat_button, style, Element}, + ice::{component::neat_button, style, widget::Image, Element}, }, }; -use iced::{button, Align, Color, Column, Container, Length, Row, Space, Text}; +use iced::{button, Align, Column, Container, Length, Row, Space, Text}; +const GEAR_ANIMATION_SPEED_FACTOR: f64 = 10.0; /// Connecting screen for the main menu pub struct Screen { cancel_button: button::State, add_button: button::State, + tip_number: u16, } impl Screen { @@ -19,49 +21,76 @@ impl Screen { Self { cancel_button: Default::default(), add_button: Default::default(), + tip_number: rand::random(), } } pub(super) fn view( &mut self, fonts: &Fonts, + imgs: &Imgs, connection_state: &ConnectionState, - time: f32, + time: f64, i18n: &Localization, button_style: style::button::Style, + show_tip: bool, ) -> Element { - let fade_msg = (time * 2.0).sin() * 0.5 + 0.51; + let gear_anim_time = time * GEAR_ANIMATION_SPEED_FACTOR; + // TODO: add built in support for animated images + let gear_anim_image = match (gear_anim_time % 5.0).trunc() as u8 { + 0 => imgs.f1, + 1 => imgs.f2, + 2 => imgs.f3, + 3 => imgs.f4, + _ => imgs.f5, + }; let children = match connection_state { - ConnectionState::InProgress { status } => { - let status = Text::new(status) - .size(fonts.alkhemi.scale(80)) - .font(fonts.alkhemi.id) - .color(Color::from_rgba(1.0, 1.0, 1.0, fade_msg)); + ConnectionState::InProgress => { + let tip = if show_tip { + let tip = format!( + "{} {}", + &i18n.get("main.tip"), + &i18n.get_variation("loading.tips", self.tip_number) + ); + Container::new(Text::new(tip).size(fonts.cyri.scale(25))) + .width(Length::Fill) + .height(Length::Fill) + .center_x() + .align_y(Align::End) + .into() + } else { + Space::new(Length::Fill, Length::Fill).into() + }; - let status = Container::new(Row::with_children(vec![ - Space::new(Length::Units(80), Length::Shrink).into(), - status.into(), - ])) - .width(Length::Fill) - .height(Length::Fill) - .align_y(Align::End); + let gear = Container::new( + Image::new(gear_anim_image) + .width(Length::Units(74)) + .height(Length::Units(62)), + ) + .width(Length::Fill); - let cancel = neat_button( + let cancel = Container::new(neat_button( &mut self.cancel_button, i18n.get("common.cancel"), 0.7, button_style, Some(Message::CancelConnect), - ); + )) + .width(Length::Fill) + .height(Length::Units(fonts.cyri.scale(50))) + .center_x() + .padding(3); - let cancel = Container::new(cancel) - .width(Length::Fill) - .height(Length::Units(fonts.cyri.scale(50))) - .center_x() - .padding(3); + let gear_cancel = Row::with_children(vec![ + gear.into(), + cancel.into(), + Space::new(Length::Fill, Length::Shrink).into(), + ]) + .width(Length::Fill) + .align_items(Align::End); - vec![status.into(), cancel.into()] + vec![tip, gear_cancel.into()] }, ConnectionState::AuthTrustPrompt { msg, .. } => { let text = Text::new(msg).size(fonts.cyri.scale(25)); diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 72deaf34b8..7ef53d8c93 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -54,6 +54,13 @@ image_ids_ice! { selection: "voxygen.element.frames.selection", selection_hover: "voxygen.element.frames.selection_hover", selection_press: "voxygen.element.frames.selection_press", + + // Animation + f1: "voxygen.element.animation.gears.1", + f2: "voxygen.element.animation.gears.2", + f3: "voxygen.element.animation.gears.3", + f4: "voxygen.element.animation.gears.4", + f5: "voxygen.element.animation.gears.5", } } @@ -100,23 +107,8 @@ pub struct LoginInfo { } enum ConnectionState { - InProgress { - status: String, - }, - AuthTrustPrompt { - auth_server: String, - msg: String, - // To remember when we switch back - status: String, - }, -} -impl ConnectionState { - fn take_status_string(&mut self) -> String { - std::mem::take(match self { - Self::InProgress { status } => status, - Self::AuthTrustPrompt { status, .. } => status, - }) - } + InProgress, + AuthTrustPrompt { auth_server: String, msg: String }, } enum Screen { @@ -153,7 +145,7 @@ struct Controls { is_selecting_language: bool, selected_language_index: Option, - time: f32, + time: f64, screen: Screen, } @@ -244,7 +236,7 @@ impl Controls { } fn view(&mut self, settings: &Settings, dt: f32) -> Element { - self.time = self.time + dt; + self.time = self.time + dt as f64; // TODO: consider setting this as the default in the renderer let button_style = style::button::Style::new(self.imgs.button) @@ -311,10 +303,12 @@ impl Controls { connection_state, } => screen.view( &self.fonts, + &self.imgs, &connection_state, self.time, &self.i18n, button_style, + settings.gameplay.loading_tips, ), }; @@ -354,19 +348,14 @@ impl Controls { Message::Singleplayer => { self.screen = Screen::Connecting { screen: connecting::Screen::new(), - connection_state: ConnectionState::InProgress { - status: [self.i18n.get("main.creating_world"), "..."].concat(), - }, + connection_state: ConnectionState::InProgress, }; - events.push(Event::StartSingleplayer); }, Message::Multiplayer => { self.screen = Screen::Connecting { screen: connecting::Screen::new(), - connection_state: ConnectionState::InProgress { - status: [self.i18n.get("main.connecting"), "..."].concat(), - }, + connection_state: ConnectionState::InProgress, }; events.push(Event::LoginAttempt { @@ -406,15 +395,13 @@ impl Controls { { if let ConnectionState::AuthTrustPrompt { auth_server, - status, .. } = connection_state { let auth_server = std::mem::take(auth_server); - let status = std::mem::take(status); let added = matches!(msg, Message::TrustPromptAdd); - *connection_state = ConnectionState::InProgress { status }; + *connection_state = ConnectionState::InProgress; events.push(Event::AuthServerTrust(auth_server, added)); } } @@ -462,11 +449,7 @@ impl Controls { &auth_server ); - *connection_state = ConnectionState::AuthTrustPrompt { - auth_server, - msg, - status: connection_state.take_status_string(), - }; + *connection_state = ConnectionState::AuthTrustPrompt { auth_server, msg }; } } From 20096e8a09bce5eb18e3d370c95d3d1baf9f0af0 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sun, 11 Oct 2020 16:30:16 -0400 Subject: [PATCH 44/61] Update to reflect changes in master ui: switched daggers with sceptre, added background at the bottom of the loading screen, added randomize button in the char creation screen --- .../frames/loading_screen/loading_bg_l.png | Bin 1882 -> 7802 bytes .../frames/loading_screen/loading_bg_r.png | Bin 1879 -> 7814 bytes voxygen/src/menu/char_selection/ui/mod.rs | 76 +++++++++++++----- voxygen/src/menu/main/ui/connecting.rs | 58 +++++++++---- voxygen/src/menu/main/ui/mod.rs | 10 +-- 5 files changed, 101 insertions(+), 43 deletions(-) diff --git a/assets/voxygen/element/frames/loading_screen/loading_bg_l.png b/assets/voxygen/element/frames/loading_screen/loading_bg_l.png index d91a841eaf6687a1e2bf60059f6573b2d1e7dfe5..41124a5daa79daf33725fde0cb602b48cb34695f 100644 GIT binary patch literal 7802 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 a47ol0Q(7HsUjQ{Qc)I$ztaD0WYytpOsJKD^ diff --git a/assets/voxygen/element/frames/loading_screen/loading_bg_r.png b/assets/voxygen/element/frames/loading_screen/loading_bg_r.png index 6431e7bb85836257a6e3246a8cbf41c1cd54502e..a3b9e843f16613cd0ebacddcfc6697f6aca72450 100644 GIT binary patch literal 7814 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 St5)(IO$(P#gfhd delta 596 zcmZp(z0Nm5T$z=FiGhLPr4`pkAjMhW5n0T@z%2~Ij105pNH8$4-JUGO<~AS1sdzc?cws5d_+ zzsNT~CDm5RSZT5$pJsqUO0rd2eo<~>iLFv*Zen_>enDQklAVGL#3YZ*VxV1>Kz>n5 znpJRWA(WMqsZdx;tXf;xwwIt6;?o-a#KqZfgJmag2d$P)Dnf% zJfNG3lM;%K4abR(3pmszQUgW+5y W|CEPQBUyo}89ZJ6T-G@yGywpZcebkl diff --git a/voxygen/src/menu/char_selection/ui/mod.rs b/voxygen/src/menu/char_selection/ui/mod.rs index e2b2e32333..5f750c850f 100644 --- a/voxygen/src/menu/char_selection/ui/mod.rs +++ b/voxygen/src/menu/char_selection/ui/mod.rs @@ -13,7 +13,7 @@ use crate::{ }, Element, IcedRenderer, IcedUi as Ui, }, - img_ids::{ImageGraphic, VoxelGraphic}, + img_ids::ImageGraphic, }, window, GlobalState, }; @@ -44,7 +44,8 @@ const STARTER_BOW: &str = "common.items.weapons.bow.starter_bow"; const STARTER_AXE: &str = "common.items.weapons.axe.starter_axe"; const STARTER_STAFF: &str = "common.items.weapons.staff.starter_staff"; const STARTER_SWORD: &str = "common.items.weapons.sword.starter_sword"; -const STARTER_DAGGER: &str = "common.items.weapons.dagger.starter_dagger"; +const STARTER_SCEPTRE: &str = "common.items.weapons.sceptre.starter_sceptre"; +// // Use in future MR to make this a starter weapon // TODO: look into what was using this in old ui const UI_MAIN: iced::Color = iced::Color::from_rgba(0.61, 0.70, 0.70, 1.0); // Greenish Blue @@ -69,13 +70,18 @@ image_ids_ice! { name_input: "voxygen.element.misc_bg.textbox", // Tool Icons - daggers: "voxygen.element.icons.daggers", + sceptre: "voxygen.element.icons.sceptre", sword: "voxygen.element.icons.sword", axe: "voxygen.element.icons.axe", hammer: "voxygen.element.icons.hammer", bow: "voxygen.element.icons.bow", staff: "voxygen.element.icons.staff", + // Dice icons + dice: "voxygen.element.icons.dice", + dice_hover: "voxygen.element.icons.dice_hover", + dice_press: "voxygen.element.icons.dice_press", + // Species Icons human_m: "voxygen.element.icons.human_m", human_f: "voxygen.element.icons.human_f", @@ -142,7 +148,6 @@ enum Mode { name: String, // TODO: default to username body: humanoid::Body, loadout: comp::Loadout, - // TODO: does this need to be an option, never seems to be none tool: &'static str, body_type_buttons: [button::State; 2], @@ -152,6 +157,7 @@ enum Mode { name_input: text_input::State, back_button: button::State, create_button: button::State, + randomize_button: button::State, }, } @@ -192,6 +198,7 @@ impl Mode { name_input: Default::default(), back_button: Default::default(), create_button: Default::default(), + randomize_button: Default::default(), } } } @@ -248,18 +255,15 @@ enum Message { BodyType(humanoid::BodyType), Species(humanoid::Species), Tool(&'static str), + RandomizeCharacter, CancelDeletion, ConfirmDeletion, } impl Controls { fn new(fonts: Fonts, imgs: Imgs, i18n: std::sync::Arc) -> Self { - let version = format!( - "{}-{}", - env!("CARGO_PKG_VERSION"), - common::util::GIT_VERSION.to_string() - ); - let alpha = format!("Veloren Pre-Alpha {}", env!("CARGO_PKG_VERSION"),); + let version = common::util::DISPLAY_VERSION_LONG.clone(); + let alpha = format!("Veloren {}", common::util::DISPLAY_VERSION.as_str()); Self { fonts, @@ -579,6 +583,7 @@ impl Controls { ref mut name_input, ref mut back_button, ref mut create_button, + ref mut randomize_button, } => { let unselected_style = style::button::Style::new(imgs.icon_border) .hover_image(imgs.icon_border_mo) @@ -709,7 +714,7 @@ impl Controls { ]) .spacing(1); - let [ref mut sword_button, ref mut daggers_button, ref mut axe_button, ref mut hammer_button, ref mut bow_button, ref mut staff_button] = + let [ref mut sword_button, ref mut sceptre_button, ref mut axe_button, ref mut hammer_button, ref mut bow_button, ref mut staff_button] = tool_buttons; let tool = Column::with_children(vec![ Row::with_children(vec![ @@ -721,10 +726,10 @@ impl Controls { ) .into(), icon_button( - daggers_button, - *tool == STARTER_DAGGER, - Message::Tool(STARTER_DAGGER), - imgs.daggers, + hammer_button, + *tool == STARTER_HAMMER, + Message::Tool(STARTER_HAMMER), + imgs.hammer, ) .into(), icon_button( @@ -739,10 +744,10 @@ impl Controls { .into(), Row::with_children(vec![ icon_button( - hammer_button, - *tool == STARTER_HAMMER, - Message::Tool(STARTER_HAMMER), - imgs.hammer, + sceptre_button, + *tool == STARTER_SCEPTRE, + Message::Tool(STARTER_SCEPTRE), + imgs.sceptre, ) .into(), icon_button( @@ -812,6 +817,18 @@ impl Controls { Some(Message::Back), ); + const DICE_SIZE: u16 = 35; + let randomize = Button::new( + randomize_button, + Space::new(Length::Units(DICE_SIZE), Length::Units(DICE_SIZE)), + ) + .style( + style::button::Style::new(imgs.dice) + .hover_image(imgs.dice_hover) + .press_image(imgs.dice_press), + ) + .on_press(Message::RandomizeCharacter); + let name_input = BackgroundContainer::new( Image::new(imgs.name_input) .height(Length::Units(40)) @@ -822,6 +839,18 @@ impl Controls { ) .padding(Padding::new().horizontal(7).top(5)); + let bottom_center = Container::new( + Row::with_children(vec![ + randomize.into(), + name_input.into(), + Space::new(Length::Units(DICE_SIZE), Length::Units(DICE_SIZE)).into(), + ]) + .align_items(Align::Center) + .spacing(5) + .padding(16), + ) + .style(style::container::Style::color(Rgba::new(0, 0, 0, 100))); + let create = neat_button( create_button, i18n.get("common.create"), @@ -835,7 +864,7 @@ impl Controls { .width(Length::Fill) .height(Length::Units(40)) .into(), - Container::new(name_input) + Container::new(bottom_center) .width(Length::Fill) .center_x() .into(), @@ -940,6 +969,13 @@ impl Controls { loadout.active_item = Some(LoadoutBuilder::default_item_config_from_str(*tool)); } }, + Message::RandomizeCharacter => { + if let Mode::Create { name, body, .. } = &mut self.mode { + use common::npc; + *body = comp::humanoid::Body::random(); + *name = npc::get_npc_name(npc::NpcKind::Humanoid).to_string(); + } + }, Message::ConfirmDeletion => { if let Mode::Select { info_content, .. } = &mut self.mode { if let Some(InfoContent::Deletion(idx)) = info_content { diff --git a/voxygen/src/menu/main/ui/connecting.rs b/voxygen/src/menu/main/ui/connecting.rs index 07a14fa221..0e770736a0 100644 --- a/voxygen/src/menu/main/ui/connecting.rs +++ b/voxygen/src/menu/main/ui/connecting.rs @@ -63,13 +63,6 @@ impl Screen { Space::new(Length::Fill, Length::Fill).into() }; - let gear = Container::new( - Image::new(gear_anim_image) - .width(Length::Units(74)) - .height(Length::Units(62)), - ) - .width(Length::Fill); - let cancel = Container::new(neat_button( &mut self.cancel_button, i18n.get("common.cancel"), @@ -78,19 +71,52 @@ impl Screen { Some(Message::CancelConnect), )) .width(Length::Fill) - .height(Length::Units(fonts.cyri.scale(50))) + .height(Length::Units(fonts.cyri.scale(40))) .center_x() .padding(3); - let gear_cancel = Row::with_children(vec![ - gear.into(), - cancel.into(), - Space::new(Length::Fill, Length::Shrink).into(), - ]) - .width(Length::Fill) - .align_items(Align::End); + let tip_cancel = Column::with_children(vec![tip, cancel.into()]) + .width(Length::FillPortion(3)) + .align_items(Align::Center) + .spacing(5) + .padding(5); - vec![tip, gear_cancel.into()] + let gear = Container::new( + Image::new(gear_anim_image) + .width(Length::Units(74)) + .height(Length::Units(62)), + ) + .width(Length::Fill) + .padding(10) + .align_x(Align::End); + + let bottom_content = Row::with_children(vec![ + Space::new(Length::Fill, Length::Shrink).into(), + tip_cancel.into(), + gear.into(), + ]) + .align_items(Align::Center) + .width(Length::Fill); + + let left_art = Image::new(imgs.loading_art_l) + .width(Length::Units(12)) + .height(Length::Units(12)); + let right_art = Image::new(imgs.loading_art_r) + .width(Length::Units(12)) + .height(Length::Units(12)); + + let bottom_bar = Container::new(Row::with_children(vec![ + left_art.into(), + bottom_content.into(), + right_art.into(), + ])) + .height(Length::Units(100)) + .style(style::container::Style::image(imgs.loading_art)); + + vec![ + Space::new(Length::Fill, Length::Fill).into(), + bottom_bar.into(), + ] }, ConnectionState::AuthTrustPrompt { msg, .. } => { let text = Text::new(msg).size(fonts.cyri.scale(25)); diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 7ef53d8c93..eabcd0d706 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -181,12 +181,8 @@ impl Controls { i18n: std::sync::Arc, settings: &Settings, ) -> Self { - let version = format!( - "{}-{}", - env!("CARGO_PKG_VERSION"), - common::util::GIT_VERSION.to_string() - ); - let alpha = format!("Veloren Pre-Alpha {}", env!("CARGO_PKG_VERSION"),); + let version = common::util::DISPLAY_VERSION_LONG.clone(); + let alpha = format!("Veloren {}", common::util::DISPLAY_VERSION.as_str()); let screen = /* if settings.show_disclaimer { Screen::Disclaimer { @@ -264,6 +260,7 @@ impl Controls { .into() }, ]) + .padding(3) .width(Length::Fill); let bg_img = if matches!(&self.screen, Screen::Connecting {..}) { @@ -319,7 +316,6 @@ impl Controls { .height(Length::Fill), ) .style(style::container::Style::image(bg_img)) - .padding(3) .into() } From a811148ee84335b4635d07ac881777eb35a0e34a Mon Sep 17 00:00:00 2001 From: Imbris Date: Sun, 18 Oct 2020 01:26:24 -0400 Subject: [PATCH 45/61] Create Tooltip widget in iced --- voxygen/src/ui/ice/widget/mod.rs | 2 + voxygen/src/ui/ice/widget/tooltip.rs | 299 +++++++++++++++++++++++++++ 2 files changed, 301 insertions(+) create mode 100644 voxygen/src/ui/ice/widget/tooltip.rs diff --git a/voxygen/src/ui/ice/widget/mod.rs b/voxygen/src/ui/ice/widget/mod.rs index d771b95138..5c777aad2a 100644 --- a/voxygen/src/ui/ice/widget/mod.rs +++ b/voxygen/src/ui/ice/widget/mod.rs @@ -6,6 +6,7 @@ pub mod image; pub mod mouse_detector; pub mod overlay; pub mod stack; +pub mod tooltip; pub use self::{ aspect_ratio_container::AspectRatioContainer, @@ -14,4 +15,5 @@ pub use self::{ image::Image, mouse_detector::MouseDetector, overlay::Overlay, + tooltip::{Tooltip, TooltipManager}, }; diff --git a/voxygen/src/ui/ice/widget/tooltip.rs b/voxygen/src/ui/ice/widget/tooltip.rs new file mode 100644 index 0000000000..d65241259d --- /dev/null +++ b/voxygen/src/ui/ice/widget/tooltip.rs @@ -0,0 +1,299 @@ +use iced::{ + layout, mouse, Align, Clipboard, Element, Event, Hasher, Layout, Length, Point, Rectangle, + Size, Widget, +}; +use std::time::Instant; +use std::time::Duration; +use std::hash::Hash; +use vek::*; + +#[derive(Copy, Clone)] +struct Hover { + start: Instant, + aabr: Aabr, +} + +impl Hover { + fn start(aabr: Aabr) -> Self { + Self { + start: Instant::now(), + aabr, + } + } +} + +#[derive(Copy, Clone)] +struct Show { + hover_pos: Vec2, + aabr: Aabr, +} + +#[derive(Copy, Clone)] +enum State { + Idle, + Start(Hover), + Showing(Show), + Fading(Instant, Show, Option) +} + +// Reports which widget the mouse is over +pub struct Update((Aabr, Vec2)); + +pub struct TooltipManager { + state: State, + hover_widget: Option>, + hover_pos: Vec2, + // How long before a tooltip is displayed when hovering + hover_dur: Duration, + // How long it takes a tooltip to disappear + fade_dur: Duration, +} + +impl TooltipManager { + pub fn new(hover_dur: Duration, fade_dur: Duration) -> Self { + Self { + state: State::Idle, + hover_widget: None, + hover_pos: Default::default(), + hover_dur, + fade_dur, + } + } + /// Call this at the top of your view function or for minimum latency at the end of message + /// handling + /// Only call this once per frame as it assumes no updates being received between calls means + /// that there is no tooltipped widget currently being hovered + pub fn maintain(&mut self) { + // Handle changes based on pointer moving + self.state = if let Some(aabr) = self.hover_widget.take() { + match self.state { + State::Idle => State::Start(Hover::start(aabr)), + State::Start(hover) if hover.aabr != aabr => State::Start(Hover::start(aabr)), + State::Start(hover) => State::Start(hover), + State::Showing(show) if show.aabr != aabr => State::Fading(Instant::now(), show, Some(Hover::start(aabr))), + State::Showing(show) => State::Showing(show), + State::Fading(start, show, Some(hover)) if hover.aabr == aabr => State::Fading(start, show, Some(hover)), + State::Fading(start, show, _) => State::Fading(start, show, Some(Hover::start(aabr))), + } + } else { + match self.state { + State::Idle | State::Start(_) => State::Idle, + State::Showing(show) => State::Fading(Instant::now(), show, None), + State::Fading(start, show, _) => State::Fading(start, show, None), + } + }; + + // Handle temporal changes + self.state = match self.state { + State::Start(Hover { start, aabr}) | State::Fading(_, _, Some(Hover { start, aabr })) if start.elapsed() >= self.hover_dur => + State::Showing(Show { aabr, hover_pos: self.hover_pos }), + State::Fading(start, _, hover) if start.elapsed() >= self.fade_dur => match hover { + Some(hover) => State::Start(hover), + None => State::Idle, + }, + state @ State::Idle | state @ State::Start(_) | state @ State::Showing(_) | state @ State::Fading(_, _, _) => state, + } + } + + pub fn update(&mut self, update: Update) { + self.hover_widget = Some(update.0.0); + self.hover_pos = update.0.1; + } + + pub fn showing(&self, aabr: Aabr) -> bool { + match self.state { + State::Idle | State::Start(_) => false, + State::Showing(show) | State::Fading(_, show, _) => show.aabr == aabr, + } + } +} + + +/// A widget used to display tooltips when the content element is hovered +pub struct Tooltip<'a, M, R: iced::Renderer> { + content: Element<'a, M, R>, + hover_content: Option Element<'a, M, R>>>, + on_update: Box M>, + manager: &'a TooltipManager, +} + +impl<'a, M, R> Tooltip<'a, M, R> +where + R: iced::Renderer, +{ + pub fn new(content: C, hover_content: H, on_update: F, manager: &'a TooltipManager) -> Self + where + C: Into>, + H: 'a + FnOnce() -> Element<'a, M, R>, + F: 'static + Fn(Update) -> M, + { + Self { + content: content.into(), + hover_content: Some(Box::new(hover_content)), + on_update: Box::new(on_update), + manager, + } + } +} + +impl<'a, M, R> Widget for Tooltip<'a, M, R> +where + R: iced::Renderer, +{ + fn width(&self) -> Length { self.content.width() } + + fn height(&self) -> Length { self.content.height() } + + fn layout(&self, renderer: &R, limits: &layout::Limits) -> layout::Node { + self.content.layout(renderer, limits) + } + + fn on_event( + &mut self, + event: Event, + layout: Layout<'_>, + cursor_position: Point, + messages: &mut Vec, + renderer: &R, + clipboard: Option<&dyn Clipboard>, + ) { + let bounds = layout.bounds(); + if bounds.contains(cursor_position) { + let aabr = aabr_from_bounds(bounds); + let m_pos = Vec2::new(cursor_position.x.trunc() as i32, cursor_position.y.trunc() as i32); + messages.push((self.on_update)(Update((aabr, m_pos)))); + } + + self.content.on_event( + event, + layout, + cursor_position, + messages, + renderer, + clipboard, + ); + } + + fn draw( + &self, + renderer: &mut R, + defaults: &R::Defaults, + layout: Layout<'_>, + cursor_position: Point, + ) -> R::Output { + self.content.draw( + renderer, + defaults, + layout, + cursor_position, + ) + } + + fn overlay( + &mut self, + layout: Layout<'_>, + ) -> Option> { + let bounds = layout.bounds(); + let aabr = aabr_from_bounds(bounds); + + self.manager.showing(aabr) + .then(|| self.hover_content.take()) + .flatten() + .map(|content| iced::overlay::Element::new( + Point { x: self.manager.hover_pos.x as f32, y: self.manager.hover_pos.y as f32 }, + Box::new(Overlay::new( + content(), + bounds, + )) + )) + + } + + fn hash_layout(&self, state: &mut Hasher) { + struct Marker; + std::any::TypeId::of::().hash(state); + self.content.hash_layout(state); + } +} + +impl<'a, M, R> From> for Element<'a, M, R> +where + R: 'a + iced::Renderer, + M: 'a, +{ + fn from(tooltip: Tooltip<'a, M, R>) -> Element<'a, M, R> { Element::new(tooltip) } +} + +fn aabr_from_bounds(bounds: iced::Rectangle) -> Aabr { + let min = Vec2::new(bounds.x.trunc() as i32, bounds.y.trunc() as i32); + let max = min + Vec2::new(bounds.width.trunc() as i32, bounds.height.trunc() as i32); + Aabr { min, max } +} + +struct Overlay<'a, M, R: iced::Renderer> { + content: Element<'a, M, R>, + /// Area to avoid overlapping with + avoid: Rectangle, +} + +impl<'a, M, R: iced::Renderer> Overlay<'a, M, R> +{ + pub fn new(content: Element<'a, M, R>, avoid: Rectangle) -> Self { + Self { content, avoid } + } +} + +impl<'a, M, R> iced::Overlay + for Overlay<'a, M, R> +where + R: iced::Renderer, +{ + fn layout( + &self, + renderer: &R, + bounds: Size, + position: Point, + ) -> layout::Node { + // TODO: Avoid avoid area + let space_below = bounds.height - position.y; + let space_above = position.y; + + let limits = layout::Limits::new( + Size::ZERO, + Size::new( + bounds.width - position.x, + if space_below > space_above { + space_below + } else { + space_above + }, + ), + ) + .width(self.content.width()); + + let mut node = self.content.layout(renderer, &limits); + + node.move_to(position - iced::Vector::new(0.0, node.size().height)); + + node + } + + fn hash_layout(&self, state: &mut Hasher, position: Point) { + struct Marker; + std::any::TypeId::of::().hash(state); + + (position.x as u32).hash(state); + (position.y as u32).hash(state); + self.content.hash_layout(state); + } + + fn draw( + &self, + renderer: &mut R, + defaults: &R::Defaults, + layout: Layout<'_>, + cursor_position: Point, + ) -> R::Output { + self.content.draw(renderer, defaults, layout, cursor_position) + } +} From acb7ba98f97b8fbdb76c1a1a9d73b2170c2d27b5 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sun, 18 Oct 2020 14:48:01 -0400 Subject: [PATCH 46/61] Add a tooltip to the char select screen to test --- voxygen/src/menu/char_selection/ui/mod.rs | 72 +++++--- voxygen/src/ui/graphic/mod.rs | 2 +- voxygen/src/ui/ice/component/mod.rs | 4 + voxygen/src/ui/ice/component/tooltip.rs | 40 +++++ voxygen/src/ui/ice/renderer/primitive.rs | 1 + .../src/ui/ice/renderer/style/container.rs | 2 + .../ui/ice/widget/aspect_ratio_container.rs | 4 + .../src/ui/ice/widget/background_container.rs | 4 + voxygen/src/ui/ice/widget/overlay.rs | 7 + voxygen/src/ui/ice/widget/stack.rs | 8 + voxygen/src/ui/ice/widget/tooltip.rs | 159 +++++++++--------- 11 files changed, 203 insertions(+), 100 deletions(-) create mode 100644 voxygen/src/ui/ice/component/tooltip.rs diff --git a/voxygen/src/menu/char_selection/ui/mod.rs b/voxygen/src/menu/char_selection/ui/mod.rs index 5f750c850f..2b0246efa9 100644 --- a/voxygen/src/menu/char_selection/ui/mod.rs +++ b/voxygen/src/menu/char_selection/ui/mod.rs @@ -5,11 +5,14 @@ use crate::{ self, fonts::IcedFonts as Fonts, ice::{ - component::neat_button, + component::{ + neat_button, + tooltip::{self, WithTooltip}, + }, style, widget::{ mouse_detector, AspectRatioContainer, BackgroundContainer, Image, MouseDetector, - Overlay, Padding, + Overlay, Padding, TooltipManager, }, Element, IcedRenderer, IcedUi as Ui, }, @@ -36,8 +39,11 @@ use vek::Rgba; pub const TEXT_COLOR: iced::Color = iced::Color::from_rgb(1.0, 1.0, 1.0); pub const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1.0, 0.2); +pub const TOOLTIP_BACK_COLOR: Rgba = Rgba::new(20, 18, 10, 255); const FILL_FRAC_ONE: f32 = 0.77; const FILL_FRAC_TWO: f32 = 0.60; +const TOOLTIP_HOVER_DUR: std::time::Duration = std::time::Duration::from_millis(500); +const TOOLTIP_FADE_DUR: std::time::Duration = std::time::Duration::from_millis(500); const STARTER_HAMMER: &str = "common.items.weapons.hammer.starter_hammer"; const STARTER_BOW: &str = "common.items.weapons.bow.starter_bow"; @@ -104,19 +110,12 @@ image_ids_ice! { button: "voxygen.element.buttons.button", button_hover: "voxygen.element.buttons.button_hover", button_press: "voxygen.element.buttons.button_press", - } -} -// TODO: do rotation in widget renderer -/*rotation_image_ids! { - pub struct ImgsRot { - - - // Tooltip Test - tt_side: "voxygen/element/frames/tt_test_edge", + // Tooltips + tt_edge: "voxygen/element/frames/tt_test_edge", tt_corner: "voxygen/element/frames/tt_test_corner_tr", } -}*/ +} pub enum Event { Logout, @@ -235,6 +234,7 @@ struct Controls { // Alpha disclaimer alpha: String, + tooltip_manager: TooltipManager, // Zone for rotating the character with the mouse mouse_detector: mouse_detector::State, // enter: bool, @@ -272,6 +272,7 @@ impl Controls { version, alpha, + tooltip_manager: TooltipManager::new(TOOLTIP_HOVER_DUR, TOOLTIP_FADE_DUR), mouse_detector: Default::default(), mode: Mode::select(), } @@ -279,12 +280,16 @@ impl Controls { fn view(&mut self, settings: &Settings, client: &Client) -> Element { // TODO: use font scale thing for text size (use on button size for buttons with - // text) TODO: if enter key pressed and character is selected then enter - // the world TODO: tooltip widget + // text) + // TODO: if enter key pressed and character is selected then enter the world + + // Maintain tooltip manager + self.tooltip_manager.maintain(); let imgs = &self.imgs; let fonts = &self.fonts; let i18n = &self.i18n; + let tooltip_manager = &self.tooltip_manager; let button_style = style::button::Style::new(imgs.button) .hover_image(imgs.button_hover) @@ -292,6 +297,16 @@ impl Controls { .text_color(TEXT_COLOR) .disabled_text_color(DISABLED_TEXT_COLOR); + let tooltip_style = tooltip::Style { + container: style::container::Style::color_with_image_border( + TOOLTIP_BACK_COLOR, + imgs.tt_corner, + imgs.tt_edge, + ), + text_color: TEXT_COLOR, + text_size: self.fonts.cyri.scale(15), + }; + let version = iced::Text::new(&self.version) .size(self.fonts.cyri.scale(15)) .width(Length::Fill) @@ -382,11 +397,21 @@ impl Controls { Button::new( delete_button, Column::with_children(vec![ - Text::new("Hi").into(), - Text::new("Hi").into(), - Text::new("Hi").into(), + Text::new(&character.character.alias).into(), + // TODO: only construct string once when characters are + // loaded + Text::new( + i18n.get("char_selection.level_fmt").replace( + "{level_nb}", + &character.level.to_string(), + ), + ) + .into(), + Text::new(i18n.get("char_selection.uncanny_valley")) + .into(), ]), ) + .padding(8) .style( style::button::Style::new(imgs.selection) .hover_image(imgs.selection_hover) @@ -394,7 +419,16 @@ impl Controls { ) .width(Length::Fill) .height(Length::Fill) - .on_press(Message::Select(i)), + .on_press(Message::Select(i)) + .with_tooltip( + tooltip_manager, + move || { + tooltip::text( + i18n.get("char_selection.delete_permanently"), + tooltip_style, + ) + }, + ), ) .ratio_of_image(imgs.selection), ) @@ -516,7 +550,7 @@ impl Controls { let over: Element<_> = match info_content { InfoContent::Deletion(_) => Container::new( Column::with_children(vec![ - Text::new(self.i18n.get("char_selection.delete_permanently")) + Text::new(i18n.get("char_selection.delete_permanently")) .size(fonts.cyri.scale(24)) .into(), Row::with_children(vec![ diff --git a/voxygen/src/ui/graphic/mod.rs b/voxygen/src/ui/graphic/mod.rs index 365ec2b7d3..0ff74b738c 100644 --- a/voxygen/src/ui/graphic/mod.rs +++ b/voxygen/src/ui/graphic/mod.rs @@ -28,7 +28,7 @@ pub enum Graphic { Blank, } -#[derive(Clone, Copy)] +#[derive(Clone, Copy, Debug)] pub enum Rotation { None, Cw90, diff --git a/voxygen/src/ui/ice/component/mod.rs b/voxygen/src/ui/ice/component/mod.rs index de0a8293eb..7e371a14e6 100644 --- a/voxygen/src/ui/ice/component/mod.rs +++ b/voxygen/src/ui/ice/component/mod.rs @@ -1,2 +1,6 @@ +/// Various composable helpers for making iced ui's pub mod neat_button; +pub mod tooltip; + pub use neat_button::neat_button; +//pub use tooltip::WithTooltip; diff --git a/voxygen/src/ui/ice/component/tooltip.rs b/voxygen/src/ui/ice/component/tooltip.rs new file mode 100644 index 0000000000..be27c83959 --- /dev/null +++ b/voxygen/src/ui/ice/component/tooltip.rs @@ -0,0 +1,40 @@ +use crate::ui::ice as ui; +use iced::{Container, Element, Text}; +use ui::{ + style, + widget::{Tooltip, TooltipManager}, +}; + +// :( all tooltips have to copy because this is needed outside the function +#[derive(Copy, Clone)] +pub struct Style { + pub container: style::container::Style, + pub text_color: iced::Color, + pub text_size: u16, +} + +/// Tooltip that is just text +pub fn text<'a, M: 'a>(text: &'a str, style: Style) -> Element<'a, M, ui::IcedRenderer> { + Container::new( + Text::new(text) + .color(style.text_color) + .size(style.text_size), + ) + .style(style.container) + .into() +} + +pub trait WithTooltip<'a, M, R: iced::Renderer> { + fn with_tooltip(self, manager: &'a TooltipManager, hover_content: H) -> Tooltip<'a, M, R> + where + H: 'a + FnMut() -> Element<'a, M, R>; +} + +impl<'a, M, R: iced::Renderer, E: Into>> WithTooltip<'a, M, R> for E { + fn with_tooltip(self, manager: &'a TooltipManager, hover_content: H) -> Tooltip<'a, M, R> + where + H: 'a + FnMut() -> Element<'a, M, R>, + { + Tooltip::new(self, hover_content, manager) + } +} diff --git a/voxygen/src/ui/ice/renderer/primitive.rs b/voxygen/src/ui/ice/renderer/primitive.rs index 05c08c9969..2ce358bc27 100644 --- a/voxygen/src/ui/ice/renderer/primitive.rs +++ b/voxygen/src/ui/ice/renderer/primitive.rs @@ -1,5 +1,6 @@ use crate::ui::{graphic, ice::widget::image}; +#[derive(Debug)] pub enum Primitive { // Allocation :( Group { diff --git a/voxygen/src/ui/ice/renderer/style/container.rs b/voxygen/src/ui/ice/renderer/style/container.rs index f30ca8cff9..5e446ac996 100644 --- a/voxygen/src/ui/ice/renderer/style/container.rs +++ b/voxygen/src/ui/ice/renderer/style/container.rs @@ -2,6 +2,7 @@ use super::super::super::widget::image; use vek::Rgba; /// Container Border +#[derive(Clone, Copy)] pub enum Border { DoubleCornerless { inner: Rgba, @@ -15,6 +16,7 @@ pub enum Border { } /// Background of the container +#[derive(Clone, Copy)] pub enum Style { Image(image::Handle, Rgba), Color(Rgba, Border), diff --git a/voxygen/src/ui/ice/widget/aspect_ratio_container.rs b/voxygen/src/ui/ice/widget/aspect_ratio_container.rs index e4ad81ea72..c178580e28 100644 --- a/voxygen/src/ui/ice/widget/aspect_ratio_container.rs +++ b/voxygen/src/ui/ice/widget/aspect_ratio_container.rs @@ -160,6 +160,10 @@ where self.content.hash_layout(state); } + + fn overlay(&mut self, layout: Layout<'_>) -> Option> { + self.content.overlay(layout.children().next().unwrap()) + } } pub trait Renderer: iced::Renderer { diff --git a/voxygen/src/ui/ice/widget/background_container.rs b/voxygen/src/ui/ice/widget/background_container.rs index aa8a10583d..1e16e66628 100644 --- a/voxygen/src/ui/ice/widget/background_container.rs +++ b/voxygen/src/ui/ice/widget/background_container.rs @@ -319,6 +319,10 @@ where self.content.hash_layout(state); } + + fn overlay(&mut self, layout: Layout<'_>) -> Option> { + self.content.overlay(layout.children().next().unwrap()) + } } pub trait Renderer: iced::Renderer { diff --git a/voxygen/src/ui/ice/widget/overlay.rs b/voxygen/src/ui/ice/widget/overlay.rs index 7a85d4e8ee..914ba0fc1f 100644 --- a/voxygen/src/ui/ice/widget/overlay.rs +++ b/voxygen/src/ui/ice/widget/overlay.rs @@ -195,6 +195,13 @@ where self.over.hash_layout(state); self.under.hash_layout(state); } + + fn overlay(&mut self, layout: Layout<'_>) -> Option> { + let mut children = layout.children(); + let (over, under) = (&mut self.over, &mut self.under); + over.overlay(children.next().unwrap()) + .or_else(move || under.overlay(children.next().unwrap())) + } } pub trait Renderer: iced::Renderer { diff --git a/voxygen/src/ui/ice/widget/stack.rs b/voxygen/src/ui/ice/widget/stack.rs index dd215dfae1..7fda1b5c2d 100644 --- a/voxygen/src/ui/ice/widget/stack.rs +++ b/voxygen/src/ui/ice/widget/stack.rs @@ -72,6 +72,14 @@ where .iter() .for_each(|child| child.hash_layout(state)); } + + fn overlay(&mut self, layout: Layout<'_>) -> Option> { + self.children + .iter_mut() + .zip(layout.children()) + .filter_map(|(child, layout)| child.overlay(layout)) + .next() + } } pub trait Renderer: iced::Renderer { diff --git a/voxygen/src/ui/ice/widget/tooltip.rs b/voxygen/src/ui/ice/widget/tooltip.rs index d65241259d..e686b2b4a1 100644 --- a/voxygen/src/ui/ice/widget/tooltip.rs +++ b/voxygen/src/ui/ice/widget/tooltip.rs @@ -1,13 +1,14 @@ use iced::{ - layout, mouse, Align, Clipboard, Element, Event, Hasher, Layout, Length, Point, Rectangle, - Size, Widget, + layout, Clipboard, Element, Event, Hasher, Layout, Length, Point, Rectangle, Size, Widget, +}; +use std::{ + hash::Hash, + sync::Mutex, + time::{Duration, Instant}, }; -use std::time::Instant; -use std::time::Duration; -use std::hash::Hash; use vek::*; -#[derive(Copy, Clone)] +#[derive(Copy, Clone, Debug)] struct Hover { start: Instant, aabr: Aabr, @@ -22,26 +23,29 @@ impl Hover { } } -#[derive(Copy, Clone)] +#[derive(Copy, Clone, Debug)] struct Show { hover_pos: Vec2, aabr: Aabr, } -#[derive(Copy, Clone)] +#[derive(Copy, Clone, Debug)] enum State { Idle, Start(Hover), Showing(Show), - Fading(Instant, Show, Option) + Fading(Instant, Show, Option), } // Reports which widget the mouse is over -pub struct Update((Aabr, Vec2)); +#[derive(Copy, Clone, Debug)] +struct Update((Aabr, Vec2)); +#[derive(Debug)] +// TODO: consider moving all this state into the Renderer pub struct TooltipManager { state: State, - hover_widget: Option>, + update: Mutex>, hover_pos: Vec2, // How long before a tooltip is displayed when hovering hover_dur: Duration, @@ -53,27 +57,35 @@ impl TooltipManager { pub fn new(hover_dur: Duration, fade_dur: Duration) -> Self { Self { state: State::Idle, - hover_widget: None, + update: Mutex::new(None), hover_pos: Default::default(), hover_dur, fade_dur, } } - /// Call this at the top of your view function or for minimum latency at the end of message - /// handling - /// Only call this once per frame as it assumes no updates being received between calls means + + /// Call this at the top of your view function or for minimum latency at the + /// end of message handling /// that there is no tooltipped widget currently being hovered pub fn maintain(&mut self) { + let update = self.update.get_mut().unwrap().take(); // Handle changes based on pointer moving - self.state = if let Some(aabr) = self.hover_widget.take() { + self.state = if let Some(Update((aabr, hover_pos))) = update { + self.hover_pos = hover_pos; match self.state { State::Idle => State::Start(Hover::start(aabr)), State::Start(hover) if hover.aabr != aabr => State::Start(Hover::start(aabr)), State::Start(hover) => State::Start(hover), - State::Showing(show) if show.aabr != aabr => State::Fading(Instant::now(), show, Some(Hover::start(aabr))), + State::Showing(show) if show.aabr != aabr => { + State::Fading(Instant::now(), show, Some(Hover::start(aabr))) + }, State::Showing(show) => State::Showing(show), - State::Fading(start, show, Some(hover)) if hover.aabr == aabr => State::Fading(start, show, Some(hover)), - State::Fading(start, show, _) => State::Fading(start, show, Some(Hover::start(aabr))), + State::Fading(start, show, Some(hover)) if hover.aabr == aabr => { + State::Fading(start, show, Some(hover)) + }, + State::Fading(start, show, _) => { + State::Fading(start, show, Some(Hover::start(aabr))) + }, } } else { match self.state { @@ -85,22 +97,29 @@ impl TooltipManager { // Handle temporal changes self.state = match self.state { - State::Start(Hover { start, aabr}) | State::Fading(_, _, Some(Hover { start, aabr })) if start.elapsed() >= self.hover_dur => - State::Showing(Show { aabr, hover_pos: self.hover_pos }), + State::Start(Hover { start, aabr }) + | State::Fading(_, _, Some(Hover { start, aabr })) + if start.elapsed() >= self.hover_dur => + { + State::Showing(Show { + aabr, + hover_pos: self.hover_pos, + }) + }, State::Fading(start, _, hover) if start.elapsed() >= self.fade_dur => match hover { Some(hover) => State::Start(hover), None => State::Idle, }, - state @ State::Idle | state @ State::Start(_) | state @ State::Showing(_) | state @ State::Fading(_, _, _) => state, - } + state @ State::Idle + | state @ State::Start(_) + | state @ State::Showing(_) + | state @ State::Fading(_, _, _) => state, + }; } - pub fn update(&mut self, update: Update) { - self.hover_widget = Some(update.0.0); - self.hover_pos = update.0.1; - } + fn update(&self, update: Update) { *self.update.lock().unwrap() = Some(update); } - pub fn showing(&self, aabr: Aabr) -> bool { + fn showing(&self, aabr: Aabr) -> bool { match self.state { State::Idle | State::Start(_) => false, State::Showing(show) | State::Fading(_, show, _) => show.aabr == aabr, @@ -108,12 +127,10 @@ impl TooltipManager { } } - /// A widget used to display tooltips when the content element is hovered pub struct Tooltip<'a, M, R: iced::Renderer> { content: Element<'a, M, R>, - hover_content: Option Element<'a, M, R>>>, - on_update: Box M>, + hover_content: Box Element<'a, M, R>>, manager: &'a TooltipManager, } @@ -121,16 +138,14 @@ impl<'a, M, R> Tooltip<'a, M, R> where R: iced::Renderer, { - pub fn new(content: C, hover_content: H, on_update: F, manager: &'a TooltipManager) -> Self + pub fn new(content: C, hover_content: H, manager: &'a TooltipManager) -> Self where C: Into>, - H: 'a + FnOnce() -> Element<'a, M, R>, - F: 'static + Fn(Update) -> M, + H: 'a + FnMut() -> Element<'a, M, R>, { Self { content: content.into(), - hover_content: Some(Box::new(hover_content)), - on_update: Box::new(on_update), + hover_content: Box::new(hover_content), manager, } } @@ -157,13 +172,6 @@ where renderer: &R, clipboard: Option<&dyn Clipboard>, ) { - let bounds = layout.bounds(); - if bounds.contains(cursor_position) { - let aabr = aabr_from_bounds(bounds); - let m_pos = Vec2::new(cursor_position.x.trunc() as i32, cursor_position.y.trunc() as i32); - messages.push((self.on_update)(Update((aabr, m_pos)))); - } - self.content.on_event( event, layout, @@ -181,32 +189,33 @@ where layout: Layout<'_>, cursor_position: Point, ) -> R::Output { - self.content.draw( - renderer, - defaults, - layout, - cursor_position, - ) + let bounds = layout.bounds(); + if bounds.contains(cursor_position) { + let aabr = aabr_from_bounds(bounds); + let m_pos = Vec2::new( + cursor_position.x.trunc() as i32, + cursor_position.y.trunc() as i32, + ); + self.manager.update(Update((aabr, m_pos))); + } + + self.content + .draw(renderer, defaults, layout, cursor_position) } - fn overlay( - &mut self, - layout: Layout<'_>, - ) -> Option> { + fn overlay(&mut self, layout: Layout<'_>) -> Option> { let bounds = layout.bounds(); let aabr = aabr_from_bounds(bounds); - self.manager.showing(aabr) - .then(|| self.hover_content.take()) - .flatten() - .map(|content| iced::overlay::Element::new( - Point { x: self.manager.hover_pos.x as f32, y: self.manager.hover_pos.y as f32 }, - Box::new(Overlay::new( - content(), - bounds, - )) - )) - + self.manager.showing(aabr).then(|| { + iced::overlay::Element::new( + Point { + x: self.manager.hover_pos.x as f32, + y: self.manager.hover_pos.y as f32, + }, + Box::new(Overlay::new((self.hover_content)(), bounds)), + ) + }) } fn hash_layout(&self, state: &mut Hasher) { @@ -236,24 +245,15 @@ struct Overlay<'a, M, R: iced::Renderer> { avoid: Rectangle, } -impl<'a, M, R: iced::Renderer> Overlay<'a, M, R> -{ - pub fn new(content: Element<'a, M, R>, avoid: Rectangle) -> Self { - Self { content, avoid } - } +impl<'a, M, R: iced::Renderer> Overlay<'a, M, R> { + pub fn new(content: Element<'a, M, R>, avoid: Rectangle) -> Self { Self { content, avoid } } } -impl<'a, M, R> iced::Overlay - for Overlay<'a, M, R> +impl<'a, M, R> iced::Overlay for Overlay<'a, M, R> where R: iced::Renderer, { - fn layout( - &self, - renderer: &R, - bounds: Size, - position: Point, - ) -> layout::Node { + fn layout(&self, renderer: &R, bounds: Size, position: Point) -> layout::Node { // TODO: Avoid avoid area let space_below = bounds.height - position.y; let space_above = position.y; @@ -274,8 +274,6 @@ where let mut node = self.content.layout(renderer, &limits); node.move_to(position - iced::Vector::new(0.0, node.size().height)); - - node } fn hash_layout(&self, state: &mut Hasher, position: Point) { @@ -294,6 +292,7 @@ where layout: Layout<'_>, cursor_position: Point, ) -> R::Output { - self.content.draw(renderer, defaults, layout, cursor_position) + self.content + .draw(renderer, defaults, layout, cursor_position) } } From 61a0aa2055f55782de05b00c0161fcbf618f8040 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sun, 18 Oct 2020 17:30:17 -0400 Subject: [PATCH 47/61] Fix tooltip border images, fade tooltips properly, improve tooltip positioning --- .../voxygen/element/frames/tooltip/corner.png | Bin 0 -> 6484 bytes .../voxygen/element/frames/tooltip/edge.png | Bin 0 -> 5966 bytes voxygen/src/menu/char_selection/ui/mod.rs | 39 ++++--- voxygen/src/ui/ice/component/tooltip.rs | 8 +- voxygen/src/ui/ice/renderer/mod.rs | 34 ++++-- voxygen/src/ui/ice/renderer/primitive.rs | 5 + voxygen/src/ui/ice/renderer/widget/mod.rs | 1 + voxygen/src/ui/ice/renderer/widget/tooltip.rs | 23 ++++ voxygen/src/ui/ice/widget/tooltip.rs | 109 ++++++++++++------ 9 files changed, 158 insertions(+), 61 deletions(-) create mode 100644 assets/voxygen/element/frames/tooltip/corner.png create mode 100644 assets/voxygen/element/frames/tooltip/edge.png create mode 100644 voxygen/src/ui/ice/renderer/widget/tooltip.rs diff --git a/assets/voxygen/element/frames/tooltip/corner.png b/assets/voxygen/element/frames/tooltip/corner.png new file mode 100644 index 0000000000000000000000000000000000000000..fff6903304fc6d10e42ec353347cd25e3143dd04 GIT binary patch literal 6484 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 a9T}y$CIN$kKT4ua7~+6X|8n2BtUm#q{+|y3 literal 0 HcmV?d00001 diff --git a/voxygen/src/menu/char_selection/ui/mod.rs b/voxygen/src/menu/char_selection/ui/mod.rs index 2b0246efa9..d1fd907d7c 100644 --- a/voxygen/src/menu/char_selection/ui/mod.rs +++ b/voxygen/src/menu/char_selection/ui/mod.rs @@ -42,8 +42,8 @@ pub const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1. pub const TOOLTIP_BACK_COLOR: Rgba = Rgba::new(20, 18, 10, 255); const FILL_FRAC_ONE: f32 = 0.77; const FILL_FRAC_TWO: f32 = 0.60; -const TOOLTIP_HOVER_DUR: std::time::Duration = std::time::Duration::from_millis(500); -const TOOLTIP_FADE_DUR: std::time::Duration = std::time::Duration::from_millis(500); +const TOOLTIP_HOVER_DUR: std::time::Duration = std::time::Duration::from_millis(300); +const TOOLTIP_FADE_DUR: std::time::Duration = std::time::Duration::from_millis(400); const STARTER_HAMMER: &str = "common.items.weapons.hammer.starter_hammer"; const STARTER_BOW: &str = "common.items.weapons.bow.starter_bow"; @@ -112,8 +112,8 @@ image_ids_ice! { button_press: "voxygen.element.buttons.button_press", // Tooltips - tt_edge: "voxygen/element/frames/tt_test_edge", - tt_corner: "voxygen/element/frames/tt_test_corner_tr", + tt_edge: "voxygen/element/frames/tooltip/edge", + tt_corner: "voxygen/element/frames/tooltip/corner", } } @@ -304,7 +304,8 @@ impl Controls { imgs.tt_edge, ), text_color: TEXT_COLOR, - text_size: self.fonts.cyri.scale(15), + text_size: self.fonts.cyri.scale(17), + padding: 10, }; let version = iced::Text::new(&self.version) @@ -384,7 +385,7 @@ impl Controls { .map(|(i, (character, (select_button, delete_button)))| { Overlay::new( Button::new( - select_button, + delete_button, Space::new(Length::Units(16), Length::Units(16)), ) .style( @@ -392,10 +393,19 @@ impl Controls { .hover_image(imgs.delete_button_hover) .press_image(imgs.delete_button_press), ) - .on_press(Message::Delete(i)), + .on_press(Message::Delete(i)) + .with_tooltip( + tooltip_manager, + move || { + tooltip::text( + i18n.get("char_selection.delete_permanently"), + tooltip_style, + ) + }, + ), AspectRatioContainer::new( Button::new( - delete_button, + select_button, Column::with_children(vec![ Text::new(&character.character.alias).into(), // TODO: only construct string once when characters are @@ -411,7 +421,7 @@ impl Controls { .into(), ]), ) - .padding(8) + .padding(10) .style( style::button::Style::new(imgs.selection) .hover_image(imgs.selection_hover) @@ -419,16 +429,7 @@ impl Controls { ) .width(Length::Fill) .height(Length::Fill) - .on_press(Message::Select(i)) - .with_tooltip( - tooltip_manager, - move || { - tooltip::text( - i18n.get("char_selection.delete_permanently"), - tooltip_style, - ) - }, - ), + .on_press(Message::Select(i)), ) .ratio_of_image(imgs.selection), ) diff --git a/voxygen/src/ui/ice/component/tooltip.rs b/voxygen/src/ui/ice/component/tooltip.rs index be27c83959..c53c693095 100644 --- a/voxygen/src/ui/ice/component/tooltip.rs +++ b/voxygen/src/ui/ice/component/tooltip.rs @@ -11,6 +11,7 @@ pub struct Style { pub container: style::container::Style, pub text_color: iced::Color, pub text_size: u16, + pub padding: u16, } /// Tooltip that is just text @@ -21,16 +22,19 @@ pub fn text<'a, M: 'a>(text: &'a str, style: Style) -> Element<'a, M, ui::IcedRe .size(style.text_size), ) .style(style.container) + .padding(style.padding) .into() } -pub trait WithTooltip<'a, M, R: iced::Renderer> { +pub trait WithTooltip<'a, M, R: ui::widget::tooltip::Renderer> { fn with_tooltip(self, manager: &'a TooltipManager, hover_content: H) -> Tooltip<'a, M, R> where H: 'a + FnMut() -> Element<'a, M, R>; } -impl<'a, M, R: iced::Renderer, E: Into>> WithTooltip<'a, M, R> for E { +impl<'a, M, R: ui::widget::tooltip::Renderer, E: Into>> WithTooltip<'a, M, R> + for E +{ fn with_tooltip(self, manager: &'a TooltipManager, hover_content: H) -> Tooltip<'a, M, R> where H: 'a + FnMut() -> Element<'a, M, R>, diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index b1f200aa45..c507b4c42f 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -181,7 +181,7 @@ impl IcedRenderer { //self.current_scissor = default_scissor(renderer); - self.draw_primitive(primitive, Vec2::zero(), renderer); + self.draw_primitive(primitive, Vec2::zero(), 1.0, renderer); // Enter the final command. self.draw_commands.push(match self.current_state { @@ -434,12 +434,18 @@ impl IcedRenderer { .collect() } - fn draw_primitive(&mut self, primitive: Primitive, offset: Vec2, renderer: &mut Renderer) { + fn draw_primitive( + &mut self, + primitive: Primitive, + offset: Vec2, + alpha: f32, + renderer: &mut Renderer, + ) { match primitive { Primitive::Group { primitives } => { primitives .into_iter() - .for_each(|p| self.draw_primitive(p, offset, renderer)); + .for_each(|p| self.draw_primitive(p, offset, alpha, renderer)); }, Primitive::Image { handle, @@ -447,8 +453,9 @@ impl IcedRenderer { color, } => { let color = srgba_to_linear(color.map(|e| e as f32 / 255.0)); + let color = apply_alpha(color, alpha); // Don't draw a transparent image. - if color[3] == 0.0 { + if color.a == 0.0 { return; } @@ -524,7 +531,9 @@ impl IcedRenderer { bottom_linear_color, } => { // Don't draw a transparent rectangle. - if top_linear_color[3] == 0.0 && bottom_linear_color[3] == 0.0 { + let top_linear_color = apply_alpha(top_linear_color, alpha); + let bottom_linear_color = apply_alpha(bottom_linear_color, alpha); + if top_linear_color.a == 0.0 && bottom_linear_color.a == 0.0 { return; } @@ -552,8 +561,9 @@ impl IcedRenderer { bounds, linear_color, } => { + let linear_color = apply_alpha(linear_color, alpha); // Don't draw a transparent rectangle. - if linear_color[3] == 0.0 { + if linear_color.a == 0.0 { return; } @@ -583,6 +593,7 @@ impl IcedRenderer { *horizontal_alignment, *vertical_alignment, */ } => { + let linear_color = apply_alpha(linear_color, alpha); self.switch_state(State::Plain); // TODO: makes sure we are not doing all this work for hidden text @@ -684,7 +695,7 @@ impl IcedRenderer { // TODO: cull primitives outside the current scissor // Renderer child - self.draw_primitive(*content, offset + clip_offset, renderer); + self.draw_primitive(*content, offset + clip_offset, alpha, renderer); // Reset scissor self.draw_commands.push(match self.current_state { @@ -698,6 +709,9 @@ impl IcedRenderer { self.draw_commands .push(DrawCommand::Scissor(self.window_scissor)); }, + Primitive::Opacity { alpha: a, content } => { + self.draw_primitive(*content, offset, alpha * a, renderer); + }, Primitive::Nothing => {}, } } @@ -806,4 +820,10 @@ impl iced::Renderer for IcedRenderer { } } +fn apply_alpha(color: Rgba, alpha: f32) -> Rgba { + Rgba { + a: alpha * color.a, + ..color + } +} // TODO: impl Debugger diff --git a/voxygen/src/ui/ice/renderer/primitive.rs b/voxygen/src/ui/ice/renderer/primitive.rs index 2ce358bc27..8230287130 100644 --- a/voxygen/src/ui/ice/renderer/primitive.rs +++ b/voxygen/src/ui/ice/renderer/primitive.rs @@ -36,5 +36,10 @@ pub enum Primitive { offset: vek::Vec2, content: Box, }, + // Make content translucent + Opacity { + alpha: f32, + content: Box, + }, Nothing, } diff --git a/voxygen/src/ui/ice/renderer/widget/mod.rs b/voxygen/src/ui/ice/renderer/widget/mod.rs index d165b6b2a8..8f9c84758e 100644 --- a/voxygen/src/ui/ice/renderer/widget/mod.rs +++ b/voxygen/src/ui/ice/renderer/widget/mod.rs @@ -13,3 +13,4 @@ mod slider; mod space; mod text; mod text_input; +mod tooltip; diff --git a/voxygen/src/ui/ice/renderer/widget/tooltip.rs b/voxygen/src/ui/ice/renderer/widget/tooltip.rs new file mode 100644 index 0000000000..626096dd0e --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/tooltip.rs @@ -0,0 +1,23 @@ +use super::super::{super::widget::tooltip, IcedRenderer, Primitive}; +use iced::{Element, Layout, Point}; + +impl tooltip::Renderer for IcedRenderer { + fn draw( + &mut self, + alpha: f32, + defaults: &Self::Defaults, + cursor_position: Point, + content: &Element<'_, M, Self>, + content_layout: Layout<'_>, + ) -> Self::Output { + let (primitive, cursor_interaction) = + content.draw(self, defaults, content_layout, cursor_position); + ( + Primitive::Opacity { + alpha, + content: Box::new(primitive), + }, + cursor_interaction, + ) + } +} diff --git a/voxygen/src/ui/ice/widget/tooltip.rs b/voxygen/src/ui/ice/widget/tooltip.rs index e686b2b4a1..0a7d496b2c 100644 --- a/voxygen/src/ui/ice/widget/tooltip.rs +++ b/voxygen/src/ui/ice/widget/tooltip.rs @@ -119,16 +119,37 @@ impl TooltipManager { fn update(&self, update: Update) { *self.update.lock().unwrap() = Some(update); } - fn showing(&self, aabr: Aabr) -> bool { + /// Returns an options with the position of the cursor when the tooltip + /// started being show and the transparency if it is fading + fn showing(&self, aabr: Aabr) -> Option<(Point, f32)> { match self.state { - State::Idle | State::Start(_) => false, - State::Showing(show) | State::Fading(_, show, _) => show.aabr == aabr, + State::Idle | State::Start(_) => None, + State::Showing(show) => (show.aabr == aabr).then(|| { + ( + Point { + x: show.hover_pos.x as f32, + y: show.hover_pos.y as f32, + }, + 1.0, + ) + }), + State::Fading(start, show, _) => (show.aabr == aabr) + .then(|| { + ( + Point { + x: show.hover_pos.x as f32, + y: show.hover_pos.y as f32, + }, + 1.0 - start.elapsed().as_secs_f32() / self.fade_dur.as_secs_f32(), + ) + }) + .filter(|(_, fade)| *fade > 0.0), } } } /// A widget used to display tooltips when the content element is hovered -pub struct Tooltip<'a, M, R: iced::Renderer> { +pub struct Tooltip<'a, M, R: self::Renderer> { content: Element<'a, M, R>, hover_content: Box Element<'a, M, R>>, manager: &'a TooltipManager, @@ -136,7 +157,7 @@ pub struct Tooltip<'a, M, R: iced::Renderer> { impl<'a, M, R> Tooltip<'a, M, R> where - R: iced::Renderer, + R: self::Renderer, { pub fn new(content: C, hover_content: H, manager: &'a TooltipManager) -> Self where @@ -153,7 +174,7 @@ where impl<'a, M, R> Widget for Tooltip<'a, M, R> where - R: iced::Renderer, + R: self::Renderer, { fn width(&self) -> Length { self.content.width() } @@ -207,13 +228,10 @@ where let bounds = layout.bounds(); let aabr = aabr_from_bounds(bounds); - self.manager.showing(aabr).then(|| { + self.manager.showing(aabr).map(|(pos, alpha)| { iced::overlay::Element::new( - Point { - x: self.manager.hover_pos.x as f32, - y: self.manager.hover_pos.y as f32, - }, - Box::new(Overlay::new((self.hover_content)(), bounds)), + pos, + Box::new(Overlay::new((self.hover_content)(), bounds, alpha)), ) }) } @@ -227,7 +245,7 @@ where impl<'a, M, R> From> for Element<'a, M, R> where - R: 'a + iced::Renderer, + R: 'a + self::Renderer, M: 'a, { fn from(tooltip: Tooltip<'a, M, R>) -> Element<'a, M, R> { Element::new(tooltip) } @@ -239,41 +257,56 @@ fn aabr_from_bounds(bounds: iced::Rectangle) -> Aabr { Aabr { min, max } } -struct Overlay<'a, M, R: iced::Renderer> { +struct Overlay<'a, M, R: self::Renderer> { content: Element<'a, M, R>, /// Area to avoid overlapping with avoid: Rectangle, + /// Alpha for fading out + alpha: f32, } -impl<'a, M, R: iced::Renderer> Overlay<'a, M, R> { - pub fn new(content: Element<'a, M, R>, avoid: Rectangle) -> Self { Self { content, avoid } } +impl<'a, M, R: self::Renderer> Overlay<'a, M, R> { + pub fn new(content: Element<'a, M, R>, avoid: Rectangle, alpha: f32) -> Self { + Self { + content, + avoid, + alpha, + } + } } impl<'a, M, R> iced::Overlay for Overlay<'a, M, R> where - R: iced::Renderer, + R: self::Renderer, { fn layout(&self, renderer: &R, bounds: Size, position: Point) -> layout::Node { - // TODO: Avoid avoid area - let space_below = bounds.height - position.y; - let space_above = position.y; + let avoid = self.avoid; + const PAD: f32 = 8.0; // TODO: allow configuration + let space_above = (avoid.y - PAD).max(0.0); + let space_below = (bounds.height - avoid.y - avoid.height - PAD).max(0.0); + //let space_left = avoid.x.min(0.0) + //let space_right = (bounds.width - avoid.x - avoid.width).min(0.0); let limits = layout::Limits::new( Size::ZERO, - Size::new( - bounds.width - position.x, - if space_below > space_above { - space_below - } else { - space_above - }, - ), - ) - .width(self.content.width()); + Size::new(bounds.width, space_above.max(space_below)), + ); + //.width(self.content.width()); let mut node = self.content.layout(renderer, &limits); - node.move_to(position - iced::Vector::new(0.0, node.size().height)); + let size = node.size(); + + node.move_to(Point { + x: (bounds.width - size.width).min(position.x), + y: if space_above >= space_below { + avoid.y - size.height - PAD + } else { + avoid.y + avoid.height + PAD + }, + }); + + node } fn hash_layout(&self, state: &mut Hasher, position: Point) { @@ -292,7 +325,17 @@ where layout: Layout<'_>, cursor_position: Point, ) -> R::Output { - self.content - .draw(renderer, defaults, layout, cursor_position) + renderer.draw(self.alpha, defaults, cursor_position, &self.content, layout) } } + +pub trait Renderer: iced::Renderer { + fn draw( + &mut self, + alpha: f32, + defaults: &Self::Defaults, + cursor_position: Point, + content: &Element<'_, M, Self>, + content_layout: Layout<'_>, + ) -> Self::Output; +} From 35a7c321805f733f4a45e3f0157d34bfc4ad146e Mon Sep 17 00:00:00 2001 From: Imbris Date: Sun, 18 Oct 2020 21:26:50 -0400 Subject: [PATCH 48/61] Highlight button of selected character, enter the world if character is selected when enter key is pressed --- voxygen/src/menu/char_selection/ui/mod.rs | 37 ++++++++++++++++++++--- 1 file changed, 32 insertions(+), 5 deletions(-) diff --git a/voxygen/src/menu/char_selection/ui/mod.rs b/voxygen/src/menu/char_selection/ui/mod.rs index d1fd907d7c..1df51ba042 100644 --- a/voxygen/src/menu/char_selection/ui/mod.rs +++ b/voxygen/src/menu/char_selection/ui/mod.rs @@ -371,6 +371,7 @@ impl Controls { // Ensure we have enough button states character_buttons.resize_with(num * 2, Default::default); + // Character Selection List let mut characters = characters .iter() .zip(character_buttons.chunks_exact_mut(2)) @@ -384,6 +385,7 @@ impl Controls { .enumerate() .map(|(i, (character, (select_button, delete_button)))| { Overlay::new( + // Delete button Button::new( delete_button, Space::new(Length::Units(16), Length::Units(16)), @@ -403,6 +405,7 @@ impl Controls { ) }, ), + // Select Button AspectRatioContainer::new( Button::new( select_button, @@ -423,9 +426,13 @@ impl Controls { ) .padding(10) .style( - style::button::Style::new(imgs.selection) - .hover_image(imgs.selection_hover) - .press_image(imgs.selection_press), + style::button::Style::new(if Some(i) == *selected { + imgs.selection_hover + } else { + imgs.selection + }) + .hover_image(imgs.selection_hover) + .press_image(imgs.selection_press), ) .width(Length::Fill) .height(Length::Fill) @@ -1048,6 +1055,7 @@ impl Controls { pub struct CharSelectionUi { ui: Ui, controls: Controls, + enter_pressed: bool, } impl CharSelectionUi { @@ -1080,7 +1088,11 @@ impl CharSelectionUi { i18n, ); - Self { ui, controls } + Self { + ui, + controls, + enter_pressed: false, + } } pub fn display_body_loadout<'a>( @@ -1093,6 +1105,16 @@ impl CharSelectionUi { pub fn handle_event(&mut self, event: window::Event) -> bool { match event { window::Event::IcedUi(event) => { + // Enter Key pressed + use iced::keyboard; + if let iced::Event::Keyboard(keyboard::Event::KeyPressed { + key_code: keyboard::KeyCode::Enter, + .. + }) = event + { + self.enter_pressed = true; + } + self.ui.handle_event(event); true }, @@ -1107,11 +1129,16 @@ impl CharSelectionUi { pub fn maintain(&mut self, global_state: &mut GlobalState, client: &mut Client) -> Vec { let mut events = Vec::new(); - let (messages, _) = self.ui.maintain( + let (mut messages, _) = self.ui.maintain( self.controls.view(&global_state.settings, &client), global_state.window.renderer_mut(), ); + if self.enter_pressed { + self.enter_pressed = false; + messages.push(Message::EnterWorld); + } + messages.into_iter().for_each(|message| { self.controls .update(message, &mut events, &client.character_list.characters) From 6aec322e99cde91a3cb6f059b443edeae8c8c6a7 Mon Sep 17 00:00:00 2001 From: Imbris Date: Mon, 19 Oct 2020 18:56:46 -0400 Subject: [PATCH 49/61] Add tooltips in iced char creation to species, tools, randomize button, and create button --- voxygen/src/menu/char_selection/ui/mod.rs | 64 +++++++++++++++++------ 1 file changed, 49 insertions(+), 15 deletions(-) diff --git a/voxygen/src/menu/char_selection/ui/mod.rs b/voxygen/src/menu/char_selection/ui/mod.rs index 1df51ba042..62129cedd7 100644 --- a/voxygen/src/menu/char_selection/ui/mod.rs +++ b/voxygen/src/menu/char_selection/ui/mod.rs @@ -42,8 +42,8 @@ pub const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1. pub const TOOLTIP_BACK_COLOR: Rgba = Rgba::new(20, 18, 10, 255); const FILL_FRAC_ONE: f32 = 0.77; const FILL_FRAC_TWO: f32 = 0.60; -const TOOLTIP_HOVER_DUR: std::time::Duration = std::time::Duration::from_millis(300); -const TOOLTIP_FADE_DUR: std::time::Duration = std::time::Duration::from_millis(400); +const TOOLTIP_HOVER_DUR: std::time::Duration = std::time::Duration::from_millis(150); +const TOOLTIP_FADE_DUR: std::time::Duration = std::time::Duration::from_millis(350); const STARTER_HAMMER: &str = "common.items.weapons.hammer.starter_hammer"; const STARTER_BOW: &str = "common.items.weapons.bow.starter_bow"; @@ -650,6 +650,12 @@ impl Controls { ) .style(style::container::Style::image(img)) }; + let icon_button_tooltip = |button, selected, msg, img, tooltip_i18n_key| { + icon_button(button, selected, msg, img) + .with_tooltip(tooltip_manager, move || { + tooltip::text(i18n.get(tooltip_i18n_key), tooltip_style) + }) + }; let (body_m_ico, body_f_ico) = match body.species { humanoid::Species::Human => (imgs.human_m, imgs.human_f), @@ -704,50 +710,56 @@ impl Controls { species_buttons; let species = Column::with_children(vec![ Row::with_children(vec![ - icon_button( + icon_button_tooltip( human_button, matches!(body.species, humanoid::Species::Human), Message::Species(humanoid::Species::Human), human_icon, + "common.species.human", ) .into(), - icon_button( + icon_button_tooltip( orc_button, matches!(body.species, humanoid::Species::Orc), Message::Species(humanoid::Species::Orc), orc_icon, + "common.species.orc", ) .into(), - icon_button( + icon_button_tooltip( dwarf_button, matches!(body.species, humanoid::Species::Dwarf), Message::Species(humanoid::Species::Dwarf), dwarf_icon, + "common.species.dwarf", ) .into(), ]) .spacing(1) .into(), Row::with_children(vec![ - icon_button( + icon_button_tooltip( elf_button, matches!(body.species, humanoid::Species::Elf), Message::Species(humanoid::Species::Elf), elf_icon, + "common.species.elf", ) .into(), - icon_button( + icon_button_tooltip( undead_button, matches!(body.species, humanoid::Species::Undead), Message::Species(humanoid::Species::Undead), undead_icon, + "common.species.undead", ) .into(), - icon_button( + icon_button_tooltip( danari_button, matches!(body.species, humanoid::Species::Danari), Message::Species(humanoid::Species::Danari), danari_icon, + "common.species.danari", ) .into(), ]) @@ -760,50 +772,56 @@ impl Controls { tool_buttons; let tool = Column::with_children(vec![ Row::with_children(vec![ - icon_button( + icon_button_tooltip( sword_button, *tool == STARTER_SWORD, Message::Tool(STARTER_SWORD), imgs.sword, + "common.weapons.sword", ) .into(), - icon_button( + icon_button_tooltip( hammer_button, *tool == STARTER_HAMMER, Message::Tool(STARTER_HAMMER), imgs.hammer, + "common.weapons.hammer", ) .into(), - icon_button( + icon_button_tooltip( axe_button, *tool == STARTER_AXE, Message::Tool(STARTER_AXE), imgs.axe, + "common.weapons.axe", ) .into(), ]) .spacing(1) .into(), Row::with_children(vec![ - icon_button( + icon_button_tooltip( sceptre_button, *tool == STARTER_SCEPTRE, Message::Tool(STARTER_SCEPTRE), imgs.sceptre, + "common.weapons.sceptre", ) .into(), - icon_button( + icon_button_tooltip( bow_button, *tool == STARTER_BOW, Message::Tool(STARTER_BOW), imgs.bow, + "common.weapons.bow", ) .into(), - icon_button( + icon_button_tooltip( staff_button, *tool == STARTER_STAFF, Message::Tool(STARTER_STAFF), imgs.staff, + "common.weapons.staff", ) .into(), ]) @@ -869,7 +887,10 @@ impl Controls { .hover_image(imgs.dice_hover) .press_image(imgs.dice_press), ) - .on_press(Message::RandomizeCharacter); + .on_press(Message::RandomizeCharacter) + .with_tooltip(tooltip_manager, move || { + tooltip::text(i18n.get("common.rand_appearance"), tooltip_style) + }); let name_input = BackgroundContainer::new( Image::new(imgs.name_input) @@ -901,6 +922,19 @@ impl Controls { (!name.is_empty()).then_some(Message::CreateCharacter), ); + let create: Element = if name.is_empty() { + create + .with_tooltip(tooltip_manager, move || { + tooltip::text( + i18n.get("char_selection.create_info_name"), + tooltip_style, + ) + }) + .into() + } else { + create.into() + }; + let bottom = Row::with_children(vec![ Container::new(back) .width(Length::Fill) From 8cf9121e6b698f2734d7816b1e4ea641b16a052c Mon Sep 17 00:00:00 2001 From: Imbris Date: Sat, 24 Oct 2020 01:59:25 -0400 Subject: [PATCH 50/61] Fix bug in body.validate(), add sliders to iced char creation screen, make char selection screen more closely resemble the master version in style --- common/src/comp/body/humanoid.rs | 2 +- voxygen/src/menu/char_selection/ui/mod.rs | 321 ++++++++++++++---- voxygen/src/menu/main/ui/servers.rs | 7 +- voxygen/src/ui/ice/renderer/style/slider.rs | 23 +- voxygen/src/ui/ice/renderer/widget/overlay.rs | 2 - .../src/ui/ice/renderer/widget/scrollable.rs | 4 +- voxygen/src/ui/ice/renderer/widget/slider.rs | 35 +- 7 files changed, 309 insertions(+), 85 deletions(-) diff --git a/common/src/comp/body/humanoid.rs b/common/src/comp/body/humanoid.rs index ccbb8f73c4..2658672127 100644 --- a/common/src/comp/body/humanoid.rs +++ b/common/src/comp/body/humanoid.rs @@ -58,7 +58,7 @@ impl Body { self.hair_color = self.hair_color.min(self.species.num_hair_colors() - 1); self.skin = self.skin.min(self.species.num_skin_colors() - 1); self.eyes = self.eyes.min(self.species.num_eyes(self.body_type) - 1); - self.eye_color = self.hair_style.min(self.species.num_eye_colors() - 1); + self.eye_color = self.eye_color.min(self.species.num_eye_colors() - 1); self.accessory = self .accessory .min(self.species.num_accessories(self.body_type) - 1); diff --git a/voxygen/src/menu/char_selection/ui/mod.rs b/voxygen/src/menu/char_selection/ui/mod.rs index 62129cedd7..c338207aab 100644 --- a/voxygen/src/menu/char_selection/ui/mod.rs +++ b/voxygen/src/menu/char_selection/ui/mod.rs @@ -32,8 +32,8 @@ use crate::settings::Settings; //use std::time::Duration; //use ui::ice::widget; use iced::{ - button, scrollable, text_input, Align, Button, Column, Container, HorizontalAlignment, Length, - Row, Scrollable, Space, Text, TextInput, + button, scrollable, slider, text_input, Align, Button, Column, Container, HorizontalAlignment, + Length, Row, Scrollable, Slider, Space, Text, TextInput, }; use vek::Rgba; @@ -44,6 +44,7 @@ const FILL_FRAC_ONE: f32 = 0.77; const FILL_FRAC_TWO: f32 = 0.60; const TOOLTIP_HOVER_DUR: std::time::Duration = std::time::Duration::from_millis(150); const TOOLTIP_FADE_DUR: std::time::Duration = std::time::Duration::from_millis(350); +const BANNER_ALPHA: u8 = 210; const STARTER_HAMMER: &str = "common.items.weapons.hammer.starter_hammer"; const STARTER_BOW: &str = "common.items.weapons.bow.starter_bow"; @@ -51,14 +52,17 @@ const STARTER_AXE: &str = "common.items.weapons.axe.starter_axe"; const STARTER_STAFF: &str = "common.items.weapons.staff.starter_staff"; const STARTER_SWORD: &str = "common.items.weapons.sword.starter_sword"; const STARTER_SCEPTRE: &str = "common.items.weapons.sceptre.starter_sceptre"; +// TODO: what does this comment mean? // // Use in future MR to make this a starter weapon -// TODO: look into what was using this in old ui +// TODO: use for info popup frame/background and for char list scrollbar const UI_MAIN: iced::Color = iced::Color::from_rgba(0.61, 0.70, 0.70, 1.0); // Greenish Blue image_ids_ice! { struct Imgs { + frame_bottom: "voxygen.element.frames.banner_bot", + slider_range: "voxygen.element.slider.track", slider_indicator: "voxygen.element.slider.indicator", @@ -139,7 +143,6 @@ enum Mode { new_character_button: button::State, logout_button: button::State, enter_world_button: button::State, - change_server_button: button::State, yes_button: button::State, no_button: button::State, }, @@ -152,6 +155,7 @@ enum Mode { body_type_buttons: [button::State; 2], species_buttons: [button::State; 6], tool_buttons: [button::State; 6], + sliders: Sliders, scroll: scrollable::State, name_input: text_input::State, back_button: button::State, @@ -170,7 +174,6 @@ impl Mode { new_character_button: Default::default(), logout_button: Default::default(), enter_world_button: Default::default(), - change_server_button: Default::default(), yes_button: Default::default(), no_button: Default::default(), } @@ -193,6 +196,7 @@ impl Mode { body_type_buttons: Default::default(), species_buttons: Default::default(), tool_buttons: Default::default(), + sliders: Default::default(), scroll: Default::default(), name_input: Default::default(), back_button: Default::default(), @@ -248,7 +252,6 @@ enum Message { EnterWorld, Select(usize), Delete(usize), - ChangeServer, NewCharacter, CreateCharacter, Name(String), @@ -258,6 +261,16 @@ enum Message { RandomizeCharacter, CancelDeletion, ConfirmDeletion, + HairStyle(u8), + HairColor(u8), + Skin(u8), + Eyes(u8), + EyeColor(u8), + Accessory(u8), + Beard(u8), + // Workaround for widgets that require a message but we don't want them to actually do + // anything + DoNothing, } impl Controls { @@ -334,7 +347,6 @@ impl Controls { ref mut new_character_button, ref mut logout_button, ref mut enter_world_button, - ref mut change_server_button, ref mut yes_button, ref mut no_button, } => { @@ -343,24 +355,11 @@ impl Controls { Text::new(&client.server_info.name) .size(fonts.cyri.scale(25)) .into(), - Container::new(neat_button( - change_server_button, - i18n.get("char_selection.change_server"), - FILL_FRAC_TWO, - button_style, - Some(Message::ChangeServer), - )) - .height(Length::Units(35)) - .into(), ]) .spacing(5) .align_items(Align::Center), ) - .style(style::container::Style::color_with_image_border( - Rgba::new(0, 0, 0, 217), - imgs.gray_corner, - imgs.gray_edge, - )) + .style(style::container::Style::color(Rgba::new(0, 0, 0, 217))) .padding(12) .center_x() .width(Length::Fill); @@ -490,23 +489,32 @@ impl Controls { // TODO: could replace column with scrollable completely if it had a with // children method - let characters = Container::new( - Scrollable::new(characters_scroll) - .push(Column::with_children(characters).spacing(4)), - ) - .style(style::container::Style::color_with_image_border( - Rgba::new(0, 0, 0, 217), - imgs.gray_corner, - imgs.gray_edge, - )) - .padding(9) - .width(Length::Fill) + let characters = Column::with_children(vec![ + Container::new( + Scrollable::new(characters_scroll) + .push(Column::with_children(characters).spacing(4)) + .width(Length::Fill), + ) + .padding(5) + .style(style::container::Style::color(Rgba::from_translucent( + 0, + BANNER_ALPHA, + ))) + .width(Length::Units(320)) + .height(Length::Fill) + .center_x() + .into(), + Image::new(imgs.frame_bottom) + .height(Length::Units(40)) + .width(Length::Units(320)) + .color(Rgba::from_translucent(0, BANNER_ALPHA)) + .into(), + ]) .height(Length::Fill); let right_column = Column::with_children(vec![server.into(), characters.into()]) - .padding(15) .spacing(10) - .width(Length::Units(360)) // TODO: see if we can get iced to work with settings below + .width(Length::Units(320)) // TODO: see if we can get iced to work with settings below //.max_width(360) //.width(Length::Fill) .height(Length::Fill); @@ -622,6 +630,7 @@ impl Controls { ref mut body_type_buttons, ref mut species_buttons, ref mut tool_buttons, + ref mut sliders, ref mut name_input, ref mut back_button, ref mut create_button, @@ -639,7 +648,7 @@ impl Controls { Container::new( Button::<_, IcedRenderer>::new( button, - Space::new(Length::Units(70), Length::Units(70)), + Space::new(Length::Units(60), Length::Units(60)), ) .style(if selected { selected_style @@ -830,42 +839,183 @@ impl Controls { ]) .spacing(1); - let column_content = vec![body_type.into(), species.into(), tool.into()]; + const SLIDER_TEXT_SIZE: u16 = 20; + const SLIDER_CURSOR_SIZE: (u16, u16) = (9, 21); + const SLIDER_BAR_HEIGHT: u16 = 9; + const SLIDER_BAR_PAD: u16 = 5; - let right_column = Container::new( + fn char_slider<'a>( + text: &str, + state: &'a mut slider::State, + max: u8, + selected_val: u8, + on_change: impl 'static + Fn(u8) -> Message, + (fonts, imgs): (&Fonts, &Imgs), + ) -> Element<'a, Message> { Column::with_children(vec![ - Text::new(i18n.get("char_selection.character_creation")) - .size(fonts.cyri.scale(26)) + Text::new(text) + .size(fonts.cyri.scale(SLIDER_TEXT_SIZE)) .into(), - Scrollable::new(scroll) - .push( - Column::with_children(column_content) - .align_items(Align::Center) - .spacing(16), - ) + Slider::new(state, 0..=max, selected_val, on_change) + .style(style::slider::Style::images( + imgs.slider_indicator, + imgs.slider_range, + SLIDER_BAR_PAD, + SLIDER_CURSOR_SIZE, + SLIDER_BAR_HEIGHT, + )) .into(), ]) - .spacing(20) - .padding(10) - .width(Length::Fill) - .align_items(Align::Center), + .align_items(Align::Center) + .into() + }; + fn char_slider_greyable<'a>( + active: bool, + text: &str, + state: &'a mut slider::State, + max: u8, + selected_val: u8, + on_change: impl 'static + Fn(u8) -> Message, + (fonts, imgs): (&Fonts, &Imgs), + ) -> Element<'a, Message> { + if active { + char_slider(text, state, max, selected_val, on_change, (fonts, imgs)) + } else { + Column::with_children(vec![ + Text::new(text) + .size(fonts.cyri.scale(SLIDER_TEXT_SIZE)) + .color(DISABLED_TEXT_COLOR) + .into(), + // "Disabled" slider + // TODO: add iced support for disabled sliders (like buttons) + Slider::new(state, 0..=max, selected_val, |_| Message::DoNothing) + .style(style::slider::Style { + cursor: style::slider::Cursor::Color(Rgba::zero()), + bar: style::slider::Bar::Image( + imgs.slider_range, + Rgba::from_translucent(255, 51), + SLIDER_BAR_PAD, + ), + labels: false, + ..Default::default() + }) + .into(), + ]) + .align_items(Align::Center) + .into() + } + }; + + let slider_options = Column::with_children(vec![ + char_slider( + i18n.get("char_selection.hair_style"), + &mut sliders.hair_style, + body.species.num_hair_styles(body.body_type) - 1, + body.hair_style, + Message::HairStyle, + (fonts, imgs), + ), + char_slider( + i18n.get("char_selection.hair_color"), + &mut sliders.hair_color, + body.species.num_hair_colors() - 1, + body.hair_color, + Message::HairColor, + (fonts, imgs), + ), + char_slider( + i18n.get("char_selection.skin"), + &mut sliders.skin, + body.species.num_skin_colors() - 1, + body.skin, + Message::Skin, + (fonts, imgs), + ), + char_slider( + i18n.get("char_selection.eyeshape"), + &mut sliders.eyes, + body.species.num_eyes(body.body_type) - 1, + body.eyes, + Message::Eyes, + (fonts, imgs), + ), + char_slider( + i18n.get("char_selection.eye_color"), + &mut sliders.eye_color, + body.species.num_eye_colors() - 1, + body.eye_color, + Message::EyeColor, + (fonts, imgs), + ), + char_slider_greyable( + body.species.num_accessories(body.body_type) > 1, + i18n.get("char_selection.accessories"), + &mut sliders.accessory, + body.species.num_accessories(body.body_type) - 1, + body.accessory, + Message::Accessory, + (fonts, imgs), + ), + char_slider_greyable( + body.species.num_beards(body.body_type) > 1, + i18n.get("char_selection.beard"), + &mut sliders.beard, + body.species.num_beards(body.body_type) - 1, + body.beard, + Message::Beard, + (fonts, imgs), + ), + ]) + .max_width(200) + .spacing(3) + .padding(5); + + let column_content = vec![ + body_type.into(), + species.into(), + tool.into(), + slider_options.into(), + ]; + + let right_column = Container::new( + Scrollable::new(scroll) + .push( + Column::with_children(column_content) + .align_items(Align::Center) + .width(Length::Fill) + .spacing(5), + ) + .padding(5) + .width(Length::Fill) + .align_items(Align::Center), ) - .style(style::container::Style::color_with_image_border( - Rgba::new(0, 0, 0, 217), - imgs.gray_corner, - imgs.gray_edge, - )) - .padding(10) - .width(Length::Units(360)) // TODO: see if we can get iced to work with settings below + .width(Length::Units(320)) // TODO: see if we can get iced to work with settings below //.max_width(360) //.width(Length::Fill) .height(Length::Fill); + let right_column = Column::with_children(vec![ + Container::new(right_column) + .style(style::container::Style::color(Rgba::from_translucent( + 0, + BANNER_ALPHA, + ))) + .width(Length::Units(320)) + .center_x() + .into(), + Image::new(imgs.frame_bottom) + .height(Length::Units(40)) + .width(Length::Units(320)) + .color(Rgba::from_translucent(0, BANNER_ALPHA)) + .into(), + ]) + .height(Length::Fill); + let top = Row::with_children(vec![ right_column.into(), MouseDetector::new(&mut self.mouse_detector, Length::Fill, Length::Fill).into(), ]) - .padding(15) + .padding(10) .width(Length::Fill) .height(Length::Fill); @@ -1001,9 +1151,6 @@ impl Controls { *info_content = Some(InfoContent::Deletion(idx)); } }, - Message::ChangeServer => { - events.push(Event::Logout); - }, Message::NewCharacter => { if matches!(&self.mode, Mode::Select { .. }) { self.mode = Mode::create(String::new()); @@ -1069,6 +1216,49 @@ impl Controls { } } }, + Message::HairStyle(value) => { + if let Mode::Create { body, .. } = &mut self.mode { + body.hair_style = value; + body.validate(); + } + }, + Message::HairColor(value) => { + if let Mode::Create { body, .. } = &mut self.mode { + body.hair_color = value; + body.validate(); + } + }, + Message::Skin(value) => { + if let Mode::Create { body, .. } = &mut self.mode { + body.skin = value; + body.validate(); + } + }, + Message::Eyes(value) => { + if let Mode::Create { body, .. } = &mut self.mode { + body.eyes = value; + body.validate(); + } + }, + Message::EyeColor(value) => { + if let Mode::Create { body, .. } = &mut self.mode { + body.eye_color = value; + body.validate(); + } + }, + Message::Accessory(value) => { + if let Mode::Create { body, .. } = &mut self.mode { + body.accessory = value; + body.validate(); + } + }, + Message::Beard(value) => { + if let Mode::Create { body, .. } = &mut self.mode { + body.beard = value; + body.validate(); + } + }, + Message::DoNothing => {}, } } @@ -1184,3 +1374,14 @@ impl CharSelectionUi { // TODO: do we need globals? pub fn render(&self, renderer: &mut Renderer) { self.ui.render(renderer); } } + +#[derive(Default)] +struct Sliders { + hair_style: slider::State, + hair_color: slider::State, + skin: slider::State, + eyes: slider::State, + eye_color: slider::State, + accessory: slider::State, + beard: slider::State, +} diff --git a/voxygen/src/menu/main/ui/servers.rs b/voxygen/src/menu/main/ui/servers.rs index e84ed5fe27..b07804ee63 100644 --- a/voxygen/src/menu/main/ui/servers.rs +++ b/voxygen/src/menu/main/ui/servers.rs @@ -3,12 +3,7 @@ use crate::{ i18n::Localization, ui::{ fonts::IcedFonts as Fonts, - ice::{ - component::neat_button, - style, - widget::background_container::{BackgroundContainer, Padding}, - Element, - }, + ice::{component::neat_button, style, Element}, }, }; use iced::{ diff --git a/voxygen/src/ui/ice/renderer/style/slider.rs b/voxygen/src/ui/ice/renderer/style/slider.rs index 6c6da652ac..08a12f8913 100644 --- a/voxygen/src/ui/ice/renderer/style/slider.rs +++ b/voxygen/src/ui/ice/renderer/style/slider.rs @@ -6,6 +6,8 @@ pub struct Style { pub cursor: Cursor, pub bar: Bar, pub labels: bool, + pub cursor_size: (u16, u16), + pub bar_height: u16, } impl Default for Style { @@ -14,6 +16,8 @@ impl Default for Style { cursor: Cursor::Color(Rgba::new(0.5, 0.5, 0.5, 1.0)), bar: Bar::Color(Rgba::new(0.5, 0.5, 0.5, 1.0)), labels: false, + cursor_size: (8, 16), + bar_height: 6, } } } @@ -27,6 +31,23 @@ pub enum Cursor { #[derive(Clone, Copy)] pub enum Bar { Color(Rgba), - Image(image::Handle, Rgba), + Image(image::Handle, Rgba, u16), } +impl Style { + pub fn images( + cursor: image::Handle, + bar: image::Handle, + bar_pad: u16, + cursor_size: (u16, u16), + bar_height: u16, + ) -> Self { + Self { + cursor: Cursor::Image(cursor, Rgba::white()), + bar: Bar::Image(bar, Rgba::white(), bar_pad), + labels: false, + cursor_size, + bar_height, + } + } +} diff --git a/voxygen/src/ui/ice/renderer/widget/overlay.rs b/voxygen/src/ui/ice/renderer/widget/overlay.rs index 8fac3ed675..7a602917ec 100644 --- a/voxygen/src/ui/ice/renderer/widget/overlay.rs +++ b/voxygen/src/ui/ice/renderer/widget/overlay.rs @@ -1,8 +1,6 @@ use super::super::{super::widget::overlay, IcedRenderer, Primitive}; use iced::{mouse::Interaction, Element, Layout, Point, Rectangle}; -const BORDER_SIZE: u16 = 8; - impl overlay::Renderer for IcedRenderer { fn draw( &mut self, diff --git a/voxygen/src/ui/ice/renderer/widget/scrollable.rs b/voxygen/src/ui/ice/renderer/widget/scrollable.rs index 014ccd12e3..6c176fb7d0 100644 --- a/voxygen/src/ui/ice/renderer/widget/scrollable.rs +++ b/voxygen/src/ui/ice/renderer/widget/scrollable.rs @@ -3,9 +3,9 @@ use common::util::srgba_to_linear; use iced::{mouse, scrollable, Rectangle}; use style::scrollable::{Scroller, Track}; -const SCROLLBAR_WIDTH: u16 = 10; +const SCROLLBAR_WIDTH: u16 = 6; const SCROLLBAR_MIN_HEIGHT: u16 = 6; -const SCROLLBAR_MARGIN: u16 = 2; +const SCROLLBAR_MARGIN: u16 = 1; impl scrollable::Renderer for IcedRenderer { type Style = style::scrollable::Style; diff --git a/voxygen/src/ui/ice/renderer/widget/slider.rs b/voxygen/src/ui/ice/renderer/widget/slider.rs index e58c6acc80..bdb8321f9b 100644 --- a/voxygen/src/ui/ice/renderer/widget/slider.rs +++ b/voxygen/src/ui/ice/renderer/widget/slider.rs @@ -4,14 +4,12 @@ use core::ops::RangeInclusive; use iced::{mouse, slider, Point, Rectangle}; use style::slider::{Bar, Cursor, Style}; -const CURSOR_WIDTH: f32 = 10.0; -const CURSOR_HEIGHT: f32 = 16.0; -const BAR_HEIGHT: f32 = 18.0; +const CURSOR_DRAG_SHIFT: f32 = 0.7; impl slider::Renderer for IcedRenderer { type Style = Style; - const DEFAULT_HEIGHT: u16 = 20; + const DEFAULT_HEIGHT: u16 = 25; fn draw( &mut self, @@ -23,7 +21,8 @@ impl slider::Renderer for IcedRenderer { style: &Self::Style, ) -> Self::Output { let bar_bounds = Rectangle { - height: BAR_HEIGHT, + height: style.bar_height as f32, + y: bounds.y + (bounds.height - style.bar_height as f32) / 2.0, ..bounds }; let bar = match style.bar { @@ -31,20 +30,30 @@ impl slider::Renderer for IcedRenderer { bounds: bar_bounds, linear_color: srgba_to_linear(color), }, - Bar::Image(handle, color) => Primitive::Image { + // Note: bar_pad adds to the size of the bar currently since the dragging logic wouldn't + // account for shrinking the area that the cursor is shown in + Bar::Image(handle, color, bar_pad) => Primitive::Image { handle: (handle, Rotation::None), - bounds: bar_bounds, + bounds: Rectangle { + x: bar_bounds.x - bar_pad as f32, + width: bar_bounds.width + bar_pad as f32 * 2.0, + ..bar_bounds + }, color, }, }; - let (max, min) = range.into_inner(); - let offset = bounds.width as f32 * (max - min) / (value - min); + let (cursor_width, cursor_height) = style.cursor_size; + let (cursor_width, cursor_height) = (f32::from(cursor_width), f32::from(cursor_height)); + let (min, max) = range.into_inner(); + let offset = bounds.width as f32 * (value - min) / (max - min); let cursor_bounds = Rectangle { - x: bounds.x + offset - CURSOR_WIDTH / 2.0, - y: bounds.y + if is_dragging { 2.0 } else { 0.0 }, - width: CURSOR_WIDTH, - height: CURSOR_HEIGHT, + x: bounds.x + offset - cursor_width / 2.0, + y: bounds.y + + if is_dragging { CURSOR_DRAG_SHIFT } else { 0.0 } + + (bounds.height - cursor_height) / 2.0, + width: cursor_width, + height: cursor_height, }; let cursor = match style.cursor { Cursor::Color(color) => Primitive::Rectangle { From 466793a4e8e8d1b4be076fb60a4a5e895e4ca219 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sat, 31 Oct 2020 02:25:52 -0400 Subject: [PATCH 51/61] Fix tooltip positioning for tooltipped elements inside Scrollable widgets --- voxygen/src/menu/char_selection/ui/mod.rs | 5 ++- voxygen/src/ui/ice/widget/tooltip.rs | 42 +++++++++++++++++++---- 2 files changed, 40 insertions(+), 7 deletions(-) diff --git a/voxygen/src/menu/char_selection/ui/mod.rs b/voxygen/src/menu/char_selection/ui/mod.rs index c338207aab..444a554dbc 100644 --- a/voxygen/src/menu/char_selection/ui/mod.rs +++ b/voxygen/src/menu/char_selection/ui/mod.rs @@ -843,6 +843,8 @@ impl Controls { const SLIDER_CURSOR_SIZE: (u16, u16) = (9, 21); const SLIDER_BAR_HEIGHT: u16 = 9; const SLIDER_BAR_PAD: u16 = 5; + // Height of interactable area + const SLIDER_HEIGHT: u16 = 30; fn char_slider<'a>( text: &str, @@ -857,6 +859,7 @@ impl Controls { .size(fonts.cyri.scale(SLIDER_TEXT_SIZE)) .into(), Slider::new(state, 0..=max, selected_val, on_change) + .height(SLIDER_HEIGHT) .style(style::slider::Style::images( imgs.slider_indicator, imgs.slider_range, @@ -889,6 +892,7 @@ impl Controls { // "Disabled" slider // TODO: add iced support for disabled sliders (like buttons) Slider::new(state, 0..=max, selected_val, |_| Message::DoNothing) + .height(SLIDER_HEIGHT) .style(style::slider::Style { cursor: style::slider::Cursor::Color(Rgba::zero()), bar: style::slider::Bar::Image( @@ -967,7 +971,6 @@ impl Controls { ), ]) .max_width(200) - .spacing(3) .padding(5); let column_content = vec![ diff --git a/voxygen/src/ui/ice/widget/tooltip.rs b/voxygen/src/ui/ice/widget/tooltip.rs index 0a7d496b2c..6d85836780 100644 --- a/voxygen/src/ui/ice/widget/tooltip.rs +++ b/voxygen/src/ui/ice/widget/tooltip.rs @@ -212,6 +212,9 @@ where ) -> R::Output { let bounds = layout.bounds(); if bounds.contains(cursor_position) { + // TODO: these bounds aren't actually global (for example see how the Scrollable + // widget handles its content) so it's not actually a good key to + // use here let aabr = aabr_from_bounds(bounds); let m_pos = Vec2::new( cursor_position.x.trunc() as i32, @@ -228,10 +231,15 @@ where let bounds = layout.bounds(); let aabr = aabr_from_bounds(bounds); - self.manager.showing(aabr).map(|(pos, alpha)| { + self.manager.showing(aabr).map(|(cursor_pos, alpha)| { iced::overlay::Element::new( - pos, - Box::new(Overlay::new((self.hover_content)(), bounds, alpha)), + Point::ORIGIN, + Box::new(Overlay::new( + (self.hover_content)(), + cursor_pos, + bounds, + alpha, + )), ) }) } @@ -259,6 +267,8 @@ fn aabr_from_bounds(bounds: iced::Rectangle) -> Aabr { struct Overlay<'a, M, R: self::Renderer> { content: Element<'a, M, R>, + /// Cursor position + cursor_position: Point, /// Area to avoid overlapping with avoid: Rectangle, /// Alpha for fading out @@ -266,9 +276,15 @@ struct Overlay<'a, M, R: self::Renderer> { } impl<'a, M, R: self::Renderer> Overlay<'a, M, R> { - pub fn new(content: Element<'a, M, R>, avoid: Rectangle, alpha: f32) -> Self { + pub fn new( + content: Element<'a, M, R>, + cursor_position: Point, + avoid: Rectangle, + alpha: f32, + ) -> Self { Self { content, + cursor_position, avoid, alpha, } @@ -280,7 +296,16 @@ where R: self::Renderer, { fn layout(&self, renderer: &R, bounds: Size, position: Point) -> layout::Node { - let avoid = self.avoid; + let avoid = Rectangle { + x: self.avoid.x + position.x, + y: self.avoid.y + position.y, + ..self.avoid + }; + let cursor_position = Point { + x: self.cursor_position.x + position.x, + y: self.cursor_position.y + position.y, + }; + const PAD: f32 = 8.0; // TODO: allow configuration let space_above = (avoid.y - PAD).max(0.0); let space_below = (bounds.height - avoid.y - avoid.height - PAD).max(0.0); @@ -298,7 +323,7 @@ where let size = node.size(); node.move_to(Point { - x: (bounds.width - size.width).min(position.x), + x: (bounds.width - size.width).min(cursor_position.x), y: if space_above >= space_below { avoid.y - size.height - PAD } else { @@ -315,6 +340,11 @@ where (position.x as u32).hash(state); (position.y as u32).hash(state); + (self.cursor_position.x as u32).hash(state); + (self.avoid.x as u32).hash(state); + (self.avoid.y as u32).hash(state); + (self.avoid.height as u32).hash(state); + (self.avoid.width as u32).hash(state); self.content.hash_layout(state); } From 1de4c950cd6811c42aac3d499cccdc4037664e35 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sat, 31 Oct 2020 04:52:38 -0400 Subject: [PATCH 52/61] Update iced --- Cargo.lock | 87 +++++++++---------- Cargo.toml | 2 +- voxygen/Cargo.toml | 10 +-- .../renderer/widget/aspect_ratio_container.rs | 3 +- .../renderer/widget/background_container.rs | 7 +- voxygen/src/ui/ice/renderer/widget/button.rs | 1 + voxygen/src/ui/ice/renderer/widget/column.rs | 5 +- .../src/ui/ice/renderer/widget/container.rs | 3 +- voxygen/src/ui/ice/renderer/widget/overlay.rs | 5 +- voxygen/src/ui/ice/renderer/widget/row.rs | 5 +- .../src/ui/ice/renderer/widget/scrollable.rs | 48 +++++----- voxygen/src/ui/ice/renderer/widget/stack.rs | 5 +- voxygen/src/ui/ice/renderer/widget/tooltip.rs | 5 +- .../ui/ice/widget/aspect_ratio_container.rs | 3 + .../src/ui/ice/widget/background_container.rs | 8 +- voxygen/src/ui/ice/widget/compound_graphic.rs | 8 +- voxygen/src/ui/ice/widget/fill_text.rs | 4 +- voxygen/src/ui/ice/widget/image.rs | 4 +- voxygen/src/ui/ice/widget/mouse_detector.rs | 1 + voxygen/src/ui/ice/widget/overlay.rs | 3 + voxygen/src/ui/ice/widget/stack.rs | 7 +- voxygen/src/ui/ice/widget/tooltip.rs | 14 ++- 22 files changed, 139 insertions(+), 99 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 2316f712f2..2405bc9387 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -612,21 +612,6 @@ dependencies = [ "bitflags", ] -[[package]] -name = "cocoa" -version = "0.20.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0c49e86fc36d5704151f5996b7b3795385f50ce09e3be0f47a0cfde869681cf8" -dependencies = [ - "bitflags", - "block", - "core-foundation 0.7.0", - "core-graphics 0.19.2", - "foreign-types", - "libc", - "objc", -] - [[package]] name = "cocoa" version = "0.23.0" @@ -1869,14 +1854,14 @@ dependencies = [ [[package]] name = "glutin" -version = "0.24.1" -source = "git+https://github.com/rust-windowing/glutin.git?rev=63a1ea7d6e64c5112418cab9f21cd409f0afd7c2#63a1ea7d6e64c5112418cab9f21cd409f0afd7c2" +version = "0.25.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d8bae26a39a728b003e9fad473ea89527de0de050143b4df866f18bb154bc86e" dependencies = [ "android_glue", "cgl", - "cocoa 0.20.2", - "core-foundation 0.7.0", - "core-graphics 0.19.2", + "cocoa", + "core-foundation 0.9.1", "glutin_egl_sys", "glutin_emscripten_sys", "glutin_gles2_sys", @@ -1887,8 +1872,8 @@ dependencies = [ "log", "objc", "osmesa-sys", - "parking_lot 0.10.2", - "wayland-client 0.27.0", + "parking_lot 0.11.0", + "wayland-client 0.28.1", "wayland-egl", "winapi 0.3.9", "winit", @@ -1897,7 +1882,8 @@ dependencies = [ [[package]] name = "glutin_egl_sys" version = "0.1.5" -source = "git+https://github.com/rust-windowing/glutin.git?rev=63a1ea7d6e64c5112418cab9f21cd409f0afd7c2#63a1ea7d6e64c5112418cab9f21cd409f0afd7c2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2abb6aa55523480c4adc5a56bbaa249992e2dddb2fc63dc96e04a3355364c211" dependencies = [ "gl_generator", "winapi 0.3.9", @@ -1906,12 +1892,14 @@ dependencies = [ [[package]] name = "glutin_emscripten_sys" version = "0.1.1" -source = "git+https://github.com/rust-windowing/glutin.git?rev=63a1ea7d6e64c5112418cab9f21cd409f0afd7c2#63a1ea7d6e64c5112418cab9f21cd409f0afd7c2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "80de4146df76e8a6c32b03007bc764ff3249dcaeb4f675d68a06caf1bac363f1" [[package]] name = "glutin_gles2_sys" version = "0.1.5" -source = "git+https://github.com/rust-windowing/glutin.git?rev=63a1ea7d6e64c5112418cab9f21cd409f0afd7c2#63a1ea7d6e64c5112418cab9f21cd409f0afd7c2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e8094e708b730a7c8a1954f4f8a31880af00eb8a1c5b5bf85d28a0a3c6d69103" dependencies = [ "gl_generator", "objc", @@ -1920,7 +1908,8 @@ dependencies = [ [[package]] name = "glutin_glx_sys" version = "0.1.7" -source = "git+https://github.com/rust-windowing/glutin.git?rev=63a1ea7d6e64c5112418cab9f21cd409f0afd7c2#63a1ea7d6e64c5112418cab9f21cd409f0afd7c2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7e393c8fc02b807459410429150e9c4faffdb312d59b8c038566173c81991351" dependencies = [ "gl_generator", "x11-dl", @@ -1929,7 +1918,8 @@ dependencies = [ [[package]] name = "glutin_wgl_sys" version = "0.1.5" -source = "git+https://github.com/rust-windowing/glutin.git?rev=63a1ea7d6e64c5112418cab9f21cd409f0afd7c2#63a1ea7d6e64c5112418cab9f21cd409f0afd7c2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3da5951a1569dbab865c6f2a863efafff193a93caf05538d193e9e3816d21696" dependencies = [ "gl_generator", ] @@ -2158,12 +2148,12 @@ dependencies = [ [[package]] name = "iced_core" version = "0.2.1" -source = "git+https://github.com/hecrj/iced?rev=4f2962d#4f2962d73f3bdeeca8a11817e404c45e91e2c2cc" +source = "git+https://github.com/hecrj/iced?rev=f464316#f46431600cb61d4e83e0ded1ca79525478436be3" [[package]] name = "iced_futures" version = "0.1.2" -source = "git+https://github.com/hecrj/iced?rev=4f2962d#4f2962d73f3bdeeca8a11817e404c45e91e2c2cc" +source = "git+https://github.com/hecrj/iced?rev=f464316#f46431600cb61d4e83e0ded1ca79525478436be3" dependencies = [ "futures 0.3.5", "log", @@ -2173,7 +2163,7 @@ dependencies = [ [[package]] name = "iced_graphics" version = "0.1.0" -source = "git+https://github.com/hecrj/iced?rev=4f2962d#4f2962d73f3bdeeca8a11817e404c45e91e2c2cc" +source = "git+https://github.com/hecrj/iced?rev=f464316#f46431600cb61d4e83e0ded1ca79525478436be3" dependencies = [ "bytemuck", "glam", @@ -2186,7 +2176,7 @@ dependencies = [ [[package]] name = "iced_native" version = "0.2.2" -source = "git+https://github.com/hecrj/iced?rev=4f2962d#4f2962d73f3bdeeca8a11817e404c45e91e2c2cc" +source = "git+https://github.com/hecrj/iced?rev=f464316#f46431600cb61d4e83e0ded1ca79525478436be3" dependencies = [ "iced_core", "iced_futures", @@ -2198,7 +2188,7 @@ dependencies = [ [[package]] name = "iced_style" version = "0.1.0" -source = "git+https://github.com/hecrj/iced?rev=4f2962d#4f2962d73f3bdeeca8a11817e404c45e91e2c2cc" +source = "git+https://github.com/hecrj/iced?rev=f464316#f46431600cb61d4e83e0ded1ca79525478436be3" dependencies = [ "iced_core", ] @@ -2206,7 +2196,7 @@ dependencies = [ [[package]] name = "iced_winit" version = "0.1.1" -source = "git+https://github.com/hecrj/iced?rev=4f2962d#4f2962d73f3bdeeca8a11817e404c45e91e2c2cc" +source = "git+https://github.com/hecrj/iced?rev=f464316#f46431600cb61d4e83e0ded1ca79525478436be3" dependencies = [ "iced_futures", "iced_graphics", @@ -3146,9 +3136,9 @@ dependencies = [ [[package]] name = "old_school_gfx_glutin_ext" -version = "0.24.0" +version = "0.25.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a0557cea37cc48d238c938ded2873a6cc772704ee1eb01e832b43c2dd99624bc" +checksum = "97d3bf7a77b32b947b6eaa3bc3671d50a74cd9aafdbbd4f9a4feb03ed3a0ee94" dependencies = [ "gfx_core", "gfx_device_gl", @@ -4246,10 +4236,8 @@ version = "0.11.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "562da6f2f0836e144f2e92118b35add58368280556af94f399666ebfd7d1e731" dependencies = [ - "andrew", "bitflags", "byteorder", - "calloop", "dlib", "lazy_static", "log", @@ -4266,8 +4254,10 @@ version = "0.12.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2ec5c077def8af49f9b5aeeb5fcf8079c638c6615c3a8f9305e2dea601de57f7" dependencies = [ + "andrew", "bitflags", "byteorder", + "calloop", "dlib", "lazy_static", "log", @@ -5560,12 +5550,12 @@ dependencies = [ [[package]] name = "wayland-egl" -version = "0.27.0" +version = "0.28.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "123b47be6f258fffd854f016e8e7397adb8c04d984fcf308dce13714ae2231ae" +checksum = "e7ca6190c84bcdc58beccc619bf4866709db32d653255e89da38867f97f90d61" dependencies = [ - "wayland-client 0.27.0", - "wayland-sys 0.27.0", + "wayland-client 0.28.1", + "wayland-sys 0.28.1", ] [[package]] @@ -5743,11 +5733,11 @@ dependencies = [ [[package]] name = "winit" -version = "0.22.2" -source = "git+https://gitlab.com/veloren/winit.git?branch=macos-test-rebased#5efbaa7e4644c627201a9c4d24217f448795ce0f" +version = "0.23.0" +source = "git+https://gitlab.com/veloren/winit.git?branch=macos-test-spiffed#7c8c5f21384c898f50d37298d229093549b08803" dependencies = [ "bitflags", - "cocoa 0.23.0", + "cocoa", "core-foundation 0.9.1", "core-graphics 0.22.1", "core-video-sys", @@ -5766,8 +5756,8 @@ dependencies = [ "percent-encoding 2.1.0", "raw-window-handle", "serde", - "smithay-client-toolkit 0.11.0", - "wayland-client 0.27.0", + "smithay-client-toolkit 0.12.0", + "wayland-client 0.28.1", "winapi 0.3.9", "x11-dl", ] @@ -5848,3 +5838,8 @@ name = "xml-rs" version = "0.8.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b07db065a5cf61a7e4ba64f29e67db906fb1787316516c4e6e5ff0fea1efcd8a" + +[[patch.unused]] +name = "glutin" +version = "0.24.1" +source = "git+https://github.com/rust-windowing/glutin.git?rev=63a1ea7d6e64c5112418cab9f21cd409f0afd7c2#63a1ea7d6e64c5112418cab9f21cd409f0afd7c2" diff --git a/Cargo.toml b/Cargo.toml index b3ba2382e9..b58a64ead1 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -79,6 +79,6 @@ debug = 1 [patch.crates-io] # cpal conflict fix isn't released yet -winit = { git = "https://gitlab.com/veloren/winit.git", branch = "macos-test-rebased" } +winit = { git = "https://gitlab.com/veloren/winit.git", branch = "macos-test-spiffed" } glutin = {git = "https://github.com/rust-windowing/glutin.git", rev="63a1ea7d6e64c5112418cab9f21cd409f0afd7c2"} vek = { git = "https://gitlab.com/veloren/vek.git", branch = "fix_intrinsics" } diff --git a/voxygen/Cargo.toml b/voxygen/Cargo.toml index 3a0bf5d634..57ac5addc4 100644 --- a/voxygen/Cargo.toml +++ b/voxygen/Cargo.toml @@ -27,16 +27,16 @@ anim = {package = "veloren-voxygen-anim", path = "src/anim", default-features = gfx = "0.18.2" gfx_device_gl = {version = "0.16.2", optional = true} gfx_gl = {version = "0.6.1", optional = true} -glutin = {git = "https://github.com/rust-windowing/glutin.git", rev="63a1ea7d6e64c5112418cab9f21cd409f0afd7c2"} -old_school_gfx_glutin_ext = "0.24" -winit = {version = "0.22.2", features = ["serde"]} +glutin = "0.25.1" +old_school_gfx_glutin_ext = "0.25" +winit = {version = "0.23.0", features = ["serde"]} # Ui conrod_core = {git = "https://gitlab.com/veloren/conrod.git", branch="copypasta_0.7"} conrod_winit = {git = "https://gitlab.com/veloren/conrod.git", branch="copypasta_0.7"} euc = {git = "https://github.com/zesterer/euc.git"} -iced = {package = "iced_native", git = "https://github.com/hecrj/iced", rev = "4f2962d"} -iced_winit = {git = "https://github.com/hecrj/iced", rev = "4f2962d"} +iced = {package = "iced_native", git = "https://github.com/hecrj/iced", rev = "f464316"} +iced_winit = {git = "https://github.com/hecrj/iced", rev = "f464316"} window_clipboard = "0.1.1" glyph_brush = "0.7.0" diff --git a/voxygen/src/ui/ice/renderer/widget/aspect_ratio_container.rs b/voxygen/src/ui/ice/renderer/widget/aspect_ratio_container.rs index 6732f73adc..3a07c909df 100644 --- a/voxygen/src/ui/ice/renderer/widget/aspect_ratio_container.rs +++ b/voxygen/src/ui/ice/renderer/widget/aspect_ratio_container.rs @@ -15,11 +15,12 @@ impl aspect_ratio_container::Renderer for IcedRenderer { defaults: &Self::Defaults, _bounds: Rectangle, cursor_position: Point, + viewport: &Rectangle, //style: &Self::Style, content: &Element<'_, M, Self>, content_layout: Layout<'_>, ) -> Self::Output { // TODO: stlying to add a background image and such - content.draw(self, defaults, content_layout, cursor_position) + content.draw(self, defaults, content_layout, cursor_position, viewport) } } diff --git a/voxygen/src/ui/ice/renderer/widget/background_container.rs b/voxygen/src/ui/ice/renderer/widget/background_container.rs index 8ddfa0e26b..eadda89bd8 100644 --- a/voxygen/src/ui/ice/renderer/widget/background_container.rs +++ b/voxygen/src/ui/ice/renderer/widget/background_container.rs @@ -1,5 +1,5 @@ use super::super::{super::widget::background_container, IcedRenderer, Primitive}; -use iced::{Element, Layout, Point}; +use iced::{Element, Layout, Point, Rectangle}; impl background_container::Renderer for IcedRenderer { fn draw( @@ -7,6 +7,7 @@ impl background_container::Renderer for IcedRenderer { defaults: &Self::Defaults, background: &B, background_layout: Layout<'_>, + viewport: &Rectangle, content: &Element<'_, M, Self>, content_layout: Layout<'_>, cursor_position: Point, @@ -15,10 +16,10 @@ impl background_container::Renderer for IcedRenderer { B: background_container::Background, { let back_primitive = background - .draw(self, defaults, background_layout, cursor_position) + .draw(self, defaults, background_layout, cursor_position, viewport) .0; let (content_primitive, mouse_interaction) = - content.draw(self, defaults, content_layout, cursor_position); + content.draw(self, defaults, content_layout, cursor_position, viewport); ( Primitive::Group { primitives: vec![back_primitive, content_primitive], diff --git a/voxygen/src/ui/ice/renderer/widget/button.rs b/voxygen/src/ui/ice/renderer/widget/button.rs index 5f9c17c783..a0e6399297 100644 --- a/voxygen/src/ui/ice/renderer/widget/button.rs +++ b/voxygen/src/ui/ice/renderer/widget/button.rs @@ -37,6 +37,7 @@ impl button::Renderer for IcedRenderer { &Defaults { text_color }, content_layout, cursor_position, + &bounds, ); let primitive = if let Some((handle, color)) = maybe_image { diff --git a/voxygen/src/ui/ice/renderer/widget/column.rs b/voxygen/src/ui/ice/renderer/widget/column.rs index 7dbbbde20f..b55840e9f0 100644 --- a/voxygen/src/ui/ice/renderer/widget/column.rs +++ b/voxygen/src/ui/ice/renderer/widget/column.rs @@ -1,5 +1,5 @@ use super::super::{IcedRenderer, Primitive}; -use iced::{column, mouse, Element, Layout, Point}; +use iced::{column, mouse, Element, Layout, Point, Rectangle}; impl column::Renderer for IcedRenderer { fn draw( @@ -8,6 +8,7 @@ impl column::Renderer for IcedRenderer { content: &[Element<'_, M, Self>], layout: Layout<'_>, cursor_position: Point, + viewport: &Rectangle, ) -> Self::Output { let mut mouse_interaction = mouse::Interaction::default(); @@ -18,7 +19,7 @@ impl column::Renderer for IcedRenderer { .zip(layout.children()) .map(|(child, layout)| { let (primitive, new_mouse_interaction) = - child.draw(self, defaults, layout, cursor_position); + child.draw(self, defaults, layout, cursor_position, viewport); if new_mouse_interaction > mouse_interaction { mouse_interaction = new_mouse_interaction; diff --git a/voxygen/src/ui/ice/renderer/widget/container.rs b/voxygen/src/ui/ice/renderer/widget/container.rs index a15436e160..9d8398421c 100644 --- a/voxygen/src/ui/ice/renderer/widget/container.rs +++ b/voxygen/src/ui/ice/renderer/widget/container.rs @@ -15,12 +15,13 @@ impl container::Renderer for IcedRenderer { defaults: &Self::Defaults, bounds: Rectangle, cursor_position: Point, + viewport: &Rectangle, style_sheet: &Self::Style, content: &Element<'_, M, Self>, content_layout: Layout<'_>, ) -> Self::Output { let (content, mouse_interaction) = - content.draw(self, defaults, content_layout, cursor_position); + content.draw(self, defaults, content_layout, cursor_position, viewport); let prim = match style_sheet { Self::Style::Image(handle, color) => { diff --git a/voxygen/src/ui/ice/renderer/widget/overlay.rs b/voxygen/src/ui/ice/renderer/widget/overlay.rs index 7a602917ec..9447396b28 100644 --- a/voxygen/src/ui/ice/renderer/widget/overlay.rs +++ b/voxygen/src/ui/ice/renderer/widget/overlay.rs @@ -7,16 +7,17 @@ impl overlay::Renderer for IcedRenderer { defaults: &Self::Defaults, _bounds: Rectangle, cursor_position: Point, + viewport: &Rectangle, over: &Element<'_, M, Self>, over_layout: Layout<'_>, under: &Element<'_, M, Self>, under_layout: Layout<'_>, ) -> Self::Output { let (under, under_mouse_interaction) = - under.draw(self, defaults, under_layout, cursor_position); + under.draw(self, defaults, under_layout, cursor_position, viewport); let (over, over_mouse_interaction) = - over.draw(self, defaults, over_layout, cursor_position); + over.draw(self, defaults, over_layout, cursor_position, viewport); // TODO: this isn't perfect but should be obselete when iced gets layer support let mouse_interaction = if over_mouse_interaction == Interaction::Idle { diff --git a/voxygen/src/ui/ice/renderer/widget/row.rs b/voxygen/src/ui/ice/renderer/widget/row.rs index 3de2f1ea26..634c81d0ae 100644 --- a/voxygen/src/ui/ice/renderer/widget/row.rs +++ b/voxygen/src/ui/ice/renderer/widget/row.rs @@ -1,5 +1,5 @@ use super::super::{IcedRenderer, Primitive}; -use iced::{mouse, row, Element, Layout, Point}; +use iced::{mouse, row, Element, Layout, Point, Rectangle}; impl row::Renderer for IcedRenderer { fn draw( @@ -8,6 +8,7 @@ impl row::Renderer for IcedRenderer { content: &[Element<'_, M, Self>], layout: Layout<'_>, cursor_position: Point, + viewport: &Rectangle, ) -> Self::Output { let mut mouse_interaction = mouse::Interaction::default(); @@ -18,7 +19,7 @@ impl row::Renderer for IcedRenderer { .zip(layout.children()) .map(|(child, layout)| { let (primitive, new_mouse_interaction) = - child.draw(self, defaults, layout, cursor_position); + child.draw(self, defaults, layout, cursor_position, viewport); if new_mouse_interaction > mouse_interaction { mouse_interaction = new_mouse_interaction; diff --git a/voxygen/src/ui/ice/renderer/widget/scrollable.rs b/voxygen/src/ui/ice/renderer/widget/scrollable.rs index 6c176fb7d0..edf7c1a4a7 100644 --- a/voxygen/src/ui/ice/renderer/widget/scrollable.rs +++ b/voxygen/src/ui/ice/renderer/widget/scrollable.rs @@ -3,9 +3,7 @@ use common::util::srgba_to_linear; use iced::{mouse, scrollable, Rectangle}; use style::scrollable::{Scroller, Track}; -const SCROLLBAR_WIDTH: u16 = 6; const SCROLLBAR_MIN_HEIGHT: u16 = 6; -const SCROLLBAR_MARGIN: u16 = 1; impl scrollable::Renderer for IcedRenderer { type Style = style::scrollable::Style; @@ -18,32 +16,42 @@ impl scrollable::Renderer for IcedRenderer { bounds: Rectangle, content_bounds: Rectangle, offset: u32, + scrollbar_width: u16, + scrollbar_margin: u16, + scroller_width: u16, ) -> Option { - // TODO: might actually want to divide by p_scale here (same in text&ext_input) - // (or just not use it) (or at least only account for dpi but not any - // additional scaling) - let width = (SCROLLBAR_WIDTH + 2 * SCROLLBAR_MARGIN) as f32 * self.p_scale; if content_bounds.height > bounds.height { - let scrollbar_bounds = Rectangle { - x: bounds.x + bounds.width - width, - width, + // Area containing both scrollbar and scroller + let outer_width = (scrollbar_width.max(scroller_width) + 2 * scrollbar_margin) as f32 /* * self.p_scale */; + let outer_bounds = Rectangle { + x: bounds.x + bounds.width - outer_width, + width: outer_width, ..bounds }; + // Background scrollbar (i.e. the track) + let scrollbar_bounds = Rectangle { + x: bounds.x + bounds.width - outer_width / 2.0 - (scrollbar_width / 2) as f32, + width: scrollbar_width as f32, + ..bounds + }; + + // Interactive scroller let visible_fraction = bounds.height / content_bounds.height; - let scrollbar_height = (bounds.height * visible_fraction) - .max((2 * SCROLLBAR_MIN_HEIGHT) as f32 * self.p_scale); + let scroller_height = (bounds.height * visible_fraction) + .max((2 * SCROLLBAR_MIN_HEIGHT) as f32/* * self.p_scale*/); let y_offset = offset as f32 * visible_fraction; let scroller_bounds = Rectangle { - x: scrollbar_bounds.x + SCROLLBAR_MARGIN as f32 * self.p_scale, - // TODO: check this behavior + x: bounds.x + bounds.width - outer_width / 2.0 - (scrollbar_width / 2) as f32, /* * self.p_scale*/ y: scrollbar_bounds.y + y_offset, - width: scrollbar_bounds.width - (2 * SCROLLBAR_MARGIN) as f32 * self.p_scale, - height: scrollbar_height, + width: scroller_width as f32, /* * self.p_scale*/ + height: scroller_height, }; Some(scrollable::Scrollbar { + outer_bounds, bounds: scrollbar_bounds, + margin: scrollbar_margin, scroller: scrollable::Scroller { bounds: scroller_bounds, }, @@ -145,20 +153,14 @@ impl scrollable::Renderer for IcedRenderer { } if let Some(track) = style.track { - let bounds = Rectangle { - x: scrollbar.bounds.x + SCROLLBAR_MARGIN as f32 * self.p_scale, - width: scrollbar.bounds.width - - (2 * SCROLLBAR_MARGIN) as f32 * self.p_scale, - ..scrollbar.bounds - }; primitives.push(match track { Track::Color(color) => Primitive::Rectangle { - bounds, + bounds: scrollbar.bounds, linear_color: srgba_to_linear(color.map(|e| e as f32 / 255.0)), }, Track::Image(handle, color) => Primitive::Image { handle: (handle, Rotation::None), - bounds, + bounds: scrollbar.bounds, color, }, }); diff --git a/voxygen/src/ui/ice/renderer/widget/stack.rs b/voxygen/src/ui/ice/renderer/widget/stack.rs index 0594a69ae3..081f516a98 100644 --- a/voxygen/src/ui/ice/renderer/widget/stack.rs +++ b/voxygen/src/ui/ice/renderer/widget/stack.rs @@ -1,5 +1,5 @@ use super::super::{super::widget::stack, IcedRenderer, Primitive}; -use iced::{mouse, Element, Layout, Point}; +use iced::{mouse, Element, Layout, Point, Rectangle}; impl stack::Renderer for IcedRenderer { fn draw( @@ -8,6 +8,7 @@ impl stack::Renderer for IcedRenderer { content: &[Element<'_, M, Self>], layout: Layout<'_>, cursor_position: Point, + viewport: &Rectangle, ) -> Self::Output { let mut mouse_interaction = mouse::Interaction::default(); @@ -18,7 +19,7 @@ impl stack::Renderer for IcedRenderer { .zip(layout.children()) .map(|(child, layout)| { let (primitive, new_mouse_interaction) = - child.draw(self, defaults, layout, cursor_position); + child.draw(self, defaults, layout, cursor_position, viewport); if new_mouse_interaction > mouse_interaction { mouse_interaction = new_mouse_interaction; diff --git a/voxygen/src/ui/ice/renderer/widget/tooltip.rs b/voxygen/src/ui/ice/renderer/widget/tooltip.rs index 626096dd0e..0707c6bfd7 100644 --- a/voxygen/src/ui/ice/renderer/widget/tooltip.rs +++ b/voxygen/src/ui/ice/renderer/widget/tooltip.rs @@ -1,5 +1,5 @@ use super::super::{super::widget::tooltip, IcedRenderer, Primitive}; -use iced::{Element, Layout, Point}; +use iced::{Element, Layout, Point, Rectangle}; impl tooltip::Renderer for IcedRenderer { fn draw( @@ -7,11 +7,12 @@ impl tooltip::Renderer for IcedRenderer { alpha: f32, defaults: &Self::Defaults, cursor_position: Point, + viewport: &Rectangle, content: &Element<'_, M, Self>, content_layout: Layout<'_>, ) -> Self::Output { let (primitive, cursor_interaction) = - content.draw(self, defaults, content_layout, cursor_position); + content.draw(self, defaults, content_layout, cursor_position, viewport); ( Primitive::Opacity { alpha, diff --git a/voxygen/src/ui/ice/widget/aspect_ratio_container.rs b/voxygen/src/ui/ice/widget/aspect_ratio_container.rs index c178580e28..4d118bf62f 100644 --- a/voxygen/src/ui/ice/widget/aspect_ratio_container.rs +++ b/voxygen/src/ui/ice/widget/aspect_ratio_container.rs @@ -139,11 +139,13 @@ where defaults: &R::Defaults, layout: Layout<'_>, cursor_position: Point, + viewport: &Rectangle, ) -> R::Output { renderer.draw( defaults, layout.bounds(), cursor_position, + viewport, &self.content, layout.children().next().unwrap(), ) @@ -179,6 +181,7 @@ pub trait Renderer: iced::Renderer { defaults: &Self::Defaults, bounds: Rectangle, cursor_position: Point, + viewport: &Rectangle, //style: &Self::Style, content: &Element<'_, M, Self>, content_layout: Layout<'_>, diff --git a/voxygen/src/ui/ice/widget/background_container.rs b/voxygen/src/ui/ice/widget/background_container.rs index 1e16e66628..7aad366bf4 100644 --- a/voxygen/src/ui/ice/widget/background_container.rs +++ b/voxygen/src/ui/ice/widget/background_container.rs @@ -1,4 +1,6 @@ -use iced::{layout, Clipboard, Element, Event, Hasher, Layout, Length, Point, Size, Widget}; +use iced::{ + layout, Clipboard, Element, Event, Hasher, Layout, Length, Point, Rectangle, Size, Widget, +}; use std::{hash::Hash, u32}; // TODO: decouple from image/compound graphic widgets (they could still use @@ -74,6 +76,7 @@ pub trait Background: Sized { defaults: &R::Defaults, layout: Layout<'_>, cursor_position: Point, + viewport: &Rectangle, ) -> R::Output; } @@ -294,11 +297,13 @@ where defaults: &R::Defaults, layout: Layout<'_>, cursor_position: Point, + viewport: &Rectangle, ) -> R::Output { renderer.draw( defaults, &self.background, layout, + viewport, &self.content, layout.children().next().unwrap(), cursor_position, @@ -331,6 +336,7 @@ pub trait Renderer: iced::Renderer { defaults: &Self::Defaults, background: &B, background_layout: Layout<'_>, + viewport: &Rectangle, content: &Element<'_, M, Self>, content_layout: Layout<'_>, cursor_position: Point, diff --git a/voxygen/src/ui/ice/widget/compound_graphic.rs b/voxygen/src/ui/ice/widget/compound_graphic.rs index 167e175734..dcd846df23 100644 --- a/voxygen/src/ui/ice/widget/compound_graphic.rs +++ b/voxygen/src/ui/ice/widget/compound_graphic.rs @@ -134,6 +134,8 @@ impl CompoundGraphic { _defaults: &R::Defaults, layout: Layout<'_>, _cursor_position: Point, + // Note: could use to skip elements outside the viewport + _viewport: &Rectangle, ) -> R::Output { let [pixel_w, pixel_h] = self.graphics_size; let bounds = layout.bounds(); @@ -195,8 +197,9 @@ where defaults: &R::Defaults, layout: Layout<'_>, cursor_position: Point, + viewport: &Rectangle, ) -> R::Output { - Self::draw(self, renderer, defaults, layout, cursor_position) + Self::draw(self, renderer, defaults, layout, cursor_position, viewport) } fn hash_layout(&self, state: &mut Hasher) { @@ -251,7 +254,8 @@ where defaults: &R::Defaults, layout: Layout<'_>, cursor_position: Point, + viewport: &Rectangle, ) -> R::Output { - Self::draw(self, renderer, defaults, layout, cursor_position) + Self::draw(self, renderer, defaults, layout, cursor_position, viewport) } } diff --git a/voxygen/src/ui/ice/widget/fill_text.rs b/voxygen/src/ui/ice/widget/fill_text.rs index 7cd5b8d103..0243d70656 100644 --- a/voxygen/src/ui/ice/widget/fill_text.rs +++ b/voxygen/src/ui/ice/widget/fill_text.rs @@ -1,4 +1,4 @@ -use iced::{layout, Element, Hasher, Layout, Length, Point, Size, Widget}; +use iced::{layout, Element, Hasher, Layout, Length, Point, Rectangle, Size, Widget}; use std::hash::Hash; const DEFAULT_FILL_FRACTION: f32 = 1.0; @@ -94,6 +94,7 @@ where defaults: &R::Defaults, layout: Layout<'_>, cursor_position: Point, + viewport: &Rectangle, ) -> R::Output { // Note: this breaks if the parent widget adjusts the bounds height let font_size = (layout.bounds().height * self.fill_fraction) as u16; @@ -103,6 +104,7 @@ where defaults, layout.children().next().unwrap(), cursor_position, + viewport, ) } diff --git a/voxygen/src/ui/ice/widget/image.rs b/voxygen/src/ui/ice/widget/image.rs index 4a74bf0c81..2d0aa564ca 100644 --- a/voxygen/src/ui/ice/widget/image.rs +++ b/voxygen/src/ui/ice/widget/image.rs @@ -1,5 +1,5 @@ use super::super::graphic; -use iced::{layout, Element, Hasher, Layout, Length, Point, Widget}; +use iced::{layout, Element, Hasher, Layout, Length, Point, Rectangle, Widget}; use std::hash::Hash; use vek::Rgba; @@ -86,6 +86,7 @@ where _defaults: &R::Defaults, layout: Layout<'_>, _cursor_position: Point, + _viewport: &Rectangle, ) -> R::Output { renderer.draw(self.handle, self.color, layout) } @@ -134,6 +135,7 @@ where _defaults: &R::Defaults, layout: Layout<'_>, _cursor_position: Point, + _viewport: &Rectangle, ) -> R::Output { renderer.draw(self.handle, self.color, layout) } diff --git a/voxygen/src/ui/ice/widget/mouse_detector.rs b/voxygen/src/ui/ice/widget/mouse_detector.rs index 742fbb6f34..eb25275213 100644 --- a/voxygen/src/ui/ice/widget/mouse_detector.rs +++ b/voxygen/src/ui/ice/widget/mouse_detector.rs @@ -86,6 +86,7 @@ where _defaults: &R::Defaults, layout: Layout<'_>, _cursor_position: Point, + _viewport: &Rectangle, ) -> R::Output { renderer.draw(layout.bounds()) } diff --git a/voxygen/src/ui/ice/widget/overlay.rs b/voxygen/src/ui/ice/widget/overlay.rs index 914ba0fc1f..33320325ea 100644 --- a/voxygen/src/ui/ice/widget/overlay.rs +++ b/voxygen/src/ui/ice/widget/overlay.rs @@ -169,12 +169,14 @@ where defaults: &R::Defaults, layout: Layout<'_>, cursor_position: Point, + viewport: &Rectangle, ) -> R::Output { let mut children = layout.children(); renderer.draw( defaults, layout.bounds(), cursor_position, + viewport, &self.over, children.next().unwrap(), &self.under, @@ -210,6 +212,7 @@ pub trait Renderer: iced::Renderer { defaults: &Self::Defaults, bounds: Rectangle, cursor_position: Point, + viewport: &Rectangle, //style: &self::Style, over: &Element<'_, M, Self>, over_layout: Layout<'_>, diff --git a/voxygen/src/ui/ice/widget/stack.rs b/voxygen/src/ui/ice/widget/stack.rs index 7fda1b5c2d..e325eed59b 100644 --- a/voxygen/src/ui/ice/widget/stack.rs +++ b/voxygen/src/ui/ice/widget/stack.rs @@ -1,4 +1,5 @@ -use iced::{layout, Element, Hasher, Layout, Length, Point, Size, Widget}; +// TODO: unused (I think?) consider slating for removal +use iced::{layout, Element, Hasher, Layout, Length, Point, Rectangle, Size, Widget}; use std::hash::Hash; /// Stack up some widgets @@ -60,8 +61,9 @@ where defaults: &R::Defaults, layout: Layout<'_>, cursor_position: Point, + viewport: &Rectangle, ) -> R::Output { - renderer.draw(defaults, &self.children, layout, cursor_position) + renderer.draw(defaults, &self.children, layout, cursor_position, viewport) } fn hash_layout(&self, state: &mut Hasher) { @@ -89,6 +91,7 @@ pub trait Renderer: iced::Renderer { children: &[Element<'_, M, Self>], layout: Layout<'_>, cursor_position: Point, + viewport: &Rectangle, ) -> Self::Output; } diff --git a/voxygen/src/ui/ice/widget/tooltip.rs b/voxygen/src/ui/ice/widget/tooltip.rs index 6d85836780..df1a788553 100644 --- a/voxygen/src/ui/ice/widget/tooltip.rs +++ b/voxygen/src/ui/ice/widget/tooltip.rs @@ -209,6 +209,7 @@ where defaults: &R::Defaults, layout: Layout<'_>, cursor_position: Point, + viewport: &Rectangle, ) -> R::Output { let bounds = layout.bounds(); if bounds.contains(cursor_position) { @@ -224,7 +225,7 @@ where } self.content - .draw(renderer, defaults, layout, cursor_position) + .draw(renderer, defaults, layout, cursor_position, viewport) } fn overlay(&mut self, layout: Layout<'_>) -> Option> { @@ -355,7 +356,15 @@ where layout: Layout<'_>, cursor_position: Point, ) -> R::Output { - renderer.draw(self.alpha, defaults, cursor_position, &self.content, layout) + renderer.draw( + self.alpha, + defaults, + cursor_position, + // TODO: hopefully this works + &layout.bounds(), + &self.content, + layout, + ) } } @@ -365,6 +374,7 @@ pub trait Renderer: iced::Renderer { alpha: f32, defaults: &Self::Defaults, cursor_position: Point, + viewport: &Rectangle, content: &Element<'_, M, Self>, content_layout: Layout<'_>, ) -> Self::Output; From 36a404a76478f32c990ae6269555cf2191f5534b Mon Sep 17 00:00:00 2001 From: Imbris Date: Sat, 31 Oct 2020 05:00:14 -0400 Subject: [PATCH 53/61] Change glyph cache position tolerance to match changes in master --- voxygen/src/ui/ice/cache.rs | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/voxygen/src/ui/ice/cache.rs b/voxygen/src/ui/ice/cache.rs index 778ab35bb2..fc38f5366a 100644 --- a/voxygen/src/ui/ice/cache.rs +++ b/voxygen/src/ui/ice/cache.rs @@ -10,8 +10,10 @@ use vek::*; // Multiplied by current window size const GLYPH_CACHE_SIZE: u16 = 1; // Glyph cache tolerances -const SCALE_TOLERANCE: f32 = 0.5; // Note: Changed from 0.1 change back if not decent -const POSITION_TOLERANCE: f32 = 0.1; +// TODO: consider scaling based on dpi as well as providing as an option to the +// user +const SCALE_TOLERANCE: f32 = 0.5; +const POSITION_TOLERANCE: f32 = 0.5; type GlyphBrush = glyph_brush::GlyphBrush<(Aabr, Aabr), ()>; From 343cdfa1ea092c4419b683520fbba295a14b119d Mon Sep 17 00:00:00 2001 From: Imbris Date: Sat, 31 Oct 2020 16:04:20 -0400 Subject: [PATCH 54/61] Make iced ui use scale from the settings --- Cargo.lock | 5 --- Cargo.toml | 1 - voxygen/src/menu/char_selection/ui/mod.rs | 7 +++- voxygen/src/menu/main/ui/mod.rs | 7 +++- voxygen/src/ui/ice/mod.rs | 51 ++++++++++++++++------- voxygen/src/ui/ice/renderer/mod.rs | 15 +++++-- voxygen/src/ui/mod.rs | 2 +- voxygen/src/ui/scale.rs | 35 ++++++++++------ 8 files changed, 83 insertions(+), 40 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 2405bc9387..abf6abc16f 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -5838,8 +5838,3 @@ name = "xml-rs" version = "0.8.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b07db065a5cf61a7e4ba64f29e67db906fb1787316516c4e6e5ff0fea1efcd8a" - -[[patch.unused]] -name = "glutin" -version = "0.24.1" -source = "git+https://github.com/rust-windowing/glutin.git?rev=63a1ea7d6e64c5112418cab9f21cd409f0afd7c2#63a1ea7d6e64c5112418cab9f21cd409f0afd7c2" diff --git a/Cargo.toml b/Cargo.toml index b58a64ead1..2c2721a2f2 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -80,5 +80,4 @@ debug = 1 [patch.crates-io] # cpal conflict fix isn't released yet winit = { git = "https://gitlab.com/veloren/winit.git", branch = "macos-test-spiffed" } -glutin = {git = "https://github.com/rust-windowing/glutin.git", rev="63a1ea7d6e64c5112418cab9f21cd409f0afd7c2"} vek = { git = "https://gitlab.com/veloren/vek.git", branch = "fix_intrinsics" } diff --git a/voxygen/src/menu/char_selection/ui/mod.rs b/voxygen/src/menu/char_selection/ui/mod.rs index 444a554dbc..3993ff82f5 100644 --- a/voxygen/src/menu/char_selection/ui/mod.rs +++ b/voxygen/src/menu/char_selection/ui/mod.rs @@ -1305,7 +1305,12 @@ impl CharSelectionUi { ui::ice::Font::try_from_vec(buf).unwrap() }; - let mut ui = Ui::new(&mut global_state.window, font).unwrap(); + let mut ui = Ui::new( + &mut global_state.window, + font, + global_state.settings.gameplay.ui_scale, + ) + .unwrap(); let fonts = Fonts::load(&i18n.fonts, &mut ui).expect("Impossible to load fonts"); diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index eabcd0d706..9ccbb7b7d7 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -505,7 +505,12 @@ impl<'a> MainMenuUi { Font::try_from_vec(buf).unwrap() }; - let mut ui = Ui::new(&mut global_state.window, font).unwrap(); + let mut ui = Ui::new( + &mut global_state.window, + font, + global_state.settings.gameplay.ui_scale, + ) + .unwrap(); let fonts = Fonts::load(&i18n.fonts, &mut ui).expect("Impossible to load fonts"); diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index 6538cef61a..aba416c448 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -30,10 +30,15 @@ pub struct IcedUi { // Scaling of the ui scale: Scale, window_resized: Option>, + scale_mode_changed: bool, } impl IcedUi { - pub fn new(window: &mut Window, default_font: Font) -> Result { - let scale = Scale::new(window, ScaleMode::Absolute(1.0)); + pub fn new( + window: &mut Window, + default_font: Font, + scale_mode: ScaleMode, + ) -> Result { + let scale = Scale::new(window, scale_mode, 1.2); let renderer = window.renderer_mut(); let scaled_dims = scale.scaled_window_size().map(|e| e as f32); @@ -48,6 +53,7 @@ impl IcedUi { cursor_position: Vec2::zero(), scale, window_resized: None, + scale_mode_changed: false, }) } @@ -59,6 +65,14 @@ impl IcedUi { self.renderer.add_graphic(graphic) } + pub fn scale(&self) -> Scale { self.scale } + + pub fn set_scaling_mode(&mut self, mode: ScaleMode) { + self.scale.set_scaling_mode(mode); + // Signal that change needs to be handled + self.scale_mode_changed = true; + } + pub fn handle_event(&mut self, event: Event) { use iced::window; match event { @@ -66,7 +80,9 @@ impl IcedUi { // TODO: examine if we are handling dpi properly here // ideally these values should be the logical ones Event::Window(window::Event::Resized { width, height }) => { - self.window_resized = Some(Vec2::new(width, height)); + if width != 0 && height != 0 { + self.window_resized = Some(Vec2::new(width, height)); + } }, // Scale cursor movement events // Note: in some cases the scaling could be off if a resized event occured in the same @@ -78,12 +94,12 @@ impl IcedUi { // may need to handle this in a different way to address // whatever issue iced was trying to address self.cursor_position = Vec2 { - x: x * scale, - y: y * scale, + x: x / scale, + y: y / scale, }; self.events.push(Event::Mouse(mouse::Event::CursorMoved { - x: x * scale, - y: y * scale, + x: x / scale, + y: y / scale, })); }, // Scale pixel scrolling events @@ -94,8 +110,8 @@ impl IcedUi { let scale = self.scale.scale_factor_logical() as f32; self.events.push(Event::Mouse(mouse::Event::WheelScrolled { delta: mouse::ScrollDelta::Pixels { - x: x * scale, - y: y * scale, + x: x / scale, + y: y / scale, }, })); }, @@ -111,26 +127,33 @@ impl IcedUi { root: E, renderer: &mut Renderer, ) -> (Vec, mouse::Interaction) { - // Handle window resizing - if let Some(new_dims) = self.window_resized.take() { + // Handle window resizing and scale mode changing + let scaled_dims = if let Some(new_dims) = self.window_resized.take() { let old_scaled_dims = self.scale.scaled_window_size(); // TODO maybe use u32 in Scale to be consistent with iced self.scale .window_resized(new_dims.map(|e| e as f64), renderer); let scaled_dims = self.scale.scaled_window_size(); + // Avoid resetting cache if window size didn't change + (scaled_dims != old_scaled_dims).then_some(scaled_dims) + } else if self.scale_mode_changed { + Some(self.scale.scaled_window_size()) + } else { + None + }; + if let Some(scaled_dims) = scaled_dims { + self.scale_mode_changed = false; self.events .push(Event::Window(iced::window::Event::Resized { width: scaled_dims.x as u32, height: scaled_dims.y as u32, })); - // Avoid panic in graphic cache when minimizing. - // Avoid resetting cache if window size didn't change // Somewhat inefficient for elements that won't change size after a window // resize let res = renderer.get_resolution(); - if res.x > 0 && res.y > 0 && scaled_dims != old_scaled_dims { + if res.x > 0 && res.y > 0 { self.renderer .resize(scaled_dims.map(|e| e as f32), renderer); } diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index c507b4c42f..45c72d5736 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -466,6 +466,17 @@ impl IcedRenderer { ..bounds }); + let resolution = Vec2::new( + (gl_aabr.size().w * self.half_res.x).round() as u16, + (gl_aabr.size().h * self.half_res.y).round() as u16, + ); + + // Don't do anything if resolution is zero + if resolution.map(|e| e == 0).reduce_or() { + return; + // TODO: consider logging uneeded elements + } + let graphic_cache = self.cache.graphic_cache_mut(); match graphic_cache.get_graphic(graphic_id) { @@ -473,10 +484,6 @@ impl IcedRenderer { _ => {}, } - let resolution = Vec2::new( - (gl_aabr.size().w * self.half_res.x).round() as u16, - (gl_aabr.size().h * self.half_res.y).round() as u16, - ); // Transform the source rectangle into uv coordinate. // TODO: Make sure this is right. let source_aabr = { diff --git a/voxygen/src/ui/mod.rs b/voxygen/src/ui/mod.rs index 6ff7d67c8c..e7fc961c17 100644 --- a/voxygen/src/ui/mod.rs +++ b/voxygen/src/ui/mod.rs @@ -125,7 +125,7 @@ pub struct Ui { impl Ui { pub fn new(window: &mut Window) -> Result { - let scale = Scale::new(window, ScaleMode::Absolute(1.0)); + let scale = Scale::new(window, ScaleMode::Absolute(1.0), 1.0); let win_dims = scale.scaled_window_size().into_array(); let renderer = window.renderer_mut(); diff --git a/voxygen/src/ui/scale.rs b/voxygen/src/ui/scale.rs index 2845585b77..6b7ecfde42 100644 --- a/voxygen/src/ui/scale.rs +++ b/voxygen/src/ui/scale.rs @@ -22,16 +22,19 @@ pub struct Scale { scale_factor: f64, // Current logical window size window_dims: Vec2, + // TEMP + extra_factor: f64, } impl Scale { - pub fn new(window: &Window, mode: ScaleMode) -> Self { + pub fn new(window: &Window, mode: ScaleMode, extra_factor: f64) -> Self { let window_dims = window.logical_size(); let scale_factor = window.renderer().get_resolution().x as f64 / window_dims.x; Scale { mode, scale_factor, window_dims, + extra_factor, } } @@ -50,20 +53,23 @@ impl Scale { ScaleMode::RelativeToWindow(self.window_dims.map(|e| e / scale)) } - // Calculate factor to transform between logical coordinates and our scaled - // coordinates. + /// Calculate factor to transform between logical coordinates and our scaled + /// coordinates. + /// Multiply by scaled coordinates to get the logical coordinates pub fn scale_factor_logical(&self) -> f64 { - match self.mode { - ScaleMode::Absolute(scale) => scale / self.scale_factor, - ScaleMode::DpiFactor => 1.0, - ScaleMode::RelativeToWindow(dims) => { - (self.window_dims.x / dims.x).min(self.window_dims.y / dims.y) - }, - } + self.extra_factor + * match self.mode { + ScaleMode::Absolute(scale) => scale / self.scale_factor, + ScaleMode::DpiFactor => 1.0, + ScaleMode::RelativeToWindow(dims) => { + (self.window_dims.x / dims.x).min(self.window_dims.y / dims.y) + }, + } } - // Calculate factor to transform between physical coordinates and our scaled - // coordinates. + /// Calculate factor to transform between physical coordinates and our + /// scaled coordinates. + /// Multiply by scaled coordinates to get the physical coordinates pub fn scale_factor_physical(&self) -> f64 { self.scale_factor_logical() * self.scale_factor } // Updates internal window size (and/or scale_factor). @@ -72,9 +78,12 @@ impl Scale { self.window_dims = new_dims; } - // Get scaled window size. + /// Get scaled window size. pub fn scaled_window_size(&self) -> Vec2 { self.window_dims / self.scale_factor_logical() } + /// Get logical window size + pub fn window_size(&self) -> Vec2 { self.window_dims } + // Transform point from logical to scaled coordinates. pub fn scale_point(&self, point: Vec2) -> Vec2 { point / self.scale_factor_logical() } } From 49d1709341aca11b453008606c3c186afdafd345 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sat, 31 Oct 2020 16:53:25 -0400 Subject: [PATCH 55/61] Add some tracing spans --- voxygen/src/ui/ice/mod.rs | 8 ++++++++ voxygen/src/ui/ice/renderer/mod.rs | 6 +++++- 2 files changed, 13 insertions(+), 1 deletion(-) diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index aba416c448..619839c731 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -15,6 +15,7 @@ use super::{ scale::{Scale, ScaleMode}, }; use crate::{render::Renderer, window::Window, Error}; +use common::span; use iced::{mouse, Cache, Size, UserInterface}; use iced_winit::Clipboard; use vek::*; @@ -127,6 +128,7 @@ impl IcedUi { root: E, renderer: &mut Renderer, ) -> (Vec, mouse::Interaction) { + span!(_guard, "maintain", "IcedUi::maintain"); // Handle window resizing and scale mode changing let scaled_dims = if let Some(new_dims) = self.window_resized.take() { let old_scaled_dims = self.scale.scaled_window_size(); @@ -167,24 +169,30 @@ impl IcedUi { // TODO: convert to f32 at source let window_size = self.scale.scaled_window_size().map(|e| e as f32); + span!(guard, "build user_interface"); let mut user_interface = UserInterface::build( root, Size::new(window_size.x, window_size.y), self.cache.take().unwrap(), &mut self.renderer, ); + drop(guard); + span!(guard, "update user_interface"); let messages = user_interface.update( &self.events, cursor_position, Some(&self.clipboard), &mut self.renderer, ); + drop(guard); // Clear events self.events.clear(); + span!(guard, "draw user_interface"); let (primitive, mouse_interaction) = user_interface.draw(&mut self.renderer, cursor_position); + drop(guard); self.cache = Some(user_interface.into_cache()); diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index 45c72d5736..2a64e8e64a 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -20,7 +20,7 @@ use crate::{ }, Error, }; -use common::util::srgba_to_linear; +use common::{span, util::srgba_to_linear}; use std::{convert::TryInto, ops::Range}; use vek::*; @@ -171,6 +171,7 @@ impl IcedRenderer { } pub fn draw(&mut self, primitive: Primitive, renderer: &mut Renderer) { + span!(_guard, "draw", "IcedRenderer::draw"); // Re-use memory self.draw_commands.clear(); self.mesh.clear(); @@ -739,6 +740,7 @@ impl IcedRenderer { } pub fn render(&self, renderer: &mut Renderer, maybe_globals: Option<&Consts>) { + span!(_guard, "render", "IcedRenderer::render"); let mut scissor = default_scissor(renderer); let globals = maybe_globals.unwrap_or(&self.default_globals); let mut locals = &self.interface_locals; @@ -796,6 +798,7 @@ impl iced::Renderer for IcedRenderer { element: &iced::Element<'a, M, Self>, limits: &iced::layout::Limits, ) -> iced::layout::Node { + span!(_guard, "layout", "IcedRenderer::layout"); let node = element.layout(self, limits); // Trim text measurements cache? @@ -809,6 +812,7 @@ impl iced::Renderer for IcedRenderer { (overlay_primitive, overlay_interaction): Self::Output, overlay_bounds: iced::Rectangle, ) -> Self::Output { + span!(_guard, "overlay", "IcedRenderer::overlay"); ( Primitive::Group { primitives: vec![base_primitive, Primitive::Clip { From 1b65a86772a2bc6dcafb5caecf5d9a389c615862 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sat, 31 Oct 2020 20:34:38 -0400 Subject: [PATCH 56/61] Tweak scrollbar in char select screen to look nicer, fix warnings, remove old ui code, add character loading/deleting/creating/error info popups --- assets/voxygen/element/frames/gray/corner.png | Bin 142 -> 0 bytes assets/voxygen/element/frames/gray/edge.png | Bin 133 -> 0 bytes voxygen/src/menu/char_selection/mod.rs | 3 + voxygen/src/menu/char_selection/old_ui.rs | 1609 ----------------- voxygen/src/menu/char_selection/ui/mod.rs | 179 +- voxygen/src/ui/ice/mod.rs | 4 + voxygen/src/ui/ice/renderer/mod.rs | 6 + 7 files changed, 125 insertions(+), 1676 deletions(-) delete mode 100644 assets/voxygen/element/frames/gray/corner.png delete mode 100644 assets/voxygen/element/frames/gray/edge.png delete mode 100644 voxygen/src/menu/char_selection/old_ui.rs diff --git a/assets/voxygen/element/frames/gray/corner.png b/assets/voxygen/element/frames/gray/corner.png deleted file mode 100644 index 2a44e690e0cd90512c6357a1d642558ee6cc12a9..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 142 zcmeAS@N?(olHy`uVBq!ia0vp^Od!m`1|*BN@u~nRwj^(N7l!{JxM1({$v_d#0*}aI z1_o|n5N2eUHAey{$X?><>&pIwO@`S@xcvTkU!ah*r;B3<$MxiY|Nq+`?iSboXS~1S e<0E@!AYdpu!&S6N?uIB(8H1;*pUXO@geCyUekGLv diff --git a/assets/voxygen/element/frames/gray/edge.png b/assets/voxygen/element/frames/gray/edge.png deleted file mode 100644 index 4dad2e753f595c9fb67a0c0015233865aef90c0c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 133 zcmeAS@N?(olHy`uVBq!ia0vp^j6lr9!3HE}Hf~b~Qfx`y?k)@*44MpFa#bnkfg+p* z9+AZi4BWyX%*Zfnjs#GUy~NYkmHi2u3=4-~;k!3efkHx_E{-7_*ONW`{r{goaNt0J ZFoQrVGoPIZyB<)M!PC{xWt~$(697UW9IgNW diff --git a/voxygen/src/menu/char_selection/mod.rs b/voxygen/src/menu/char_selection/mod.rs index f312da2705..7fc03783de 100644 --- a/voxygen/src/menu/char_selection/mod.rs +++ b/voxygen/src/menu/char_selection/mod.rs @@ -111,6 +111,9 @@ impl PlayState for CharSelectionState { Rc::clone(&self.client), ))); }, + ui::Event::ClearCharacterListError => { + self.client.borrow_mut().character_list.error = None; + }, } } diff --git a/voxygen/src/menu/char_selection/old_ui.rs b/voxygen/src/menu/char_selection/old_ui.rs deleted file mode 100644 index bf3ae697af..0000000000 --- a/voxygen/src/menu/char_selection/old_ui.rs +++ /dev/null @@ -1,1609 +0,0 @@ -use crate::{ - i18n::{i18n_asset_key, Localization}, - render::{Consts, Globals, Renderer}, - ui::{ - fonts::Fonts, - img_ids::{BlankGraphic, ImageGraphic, VoxelGraphic, VoxelSs9Graphic}, - ImageFrame, ImageSlider, Tooltip, Tooltipable, Ui, - }, - window::{Event as WinEvent, PressState}, - GlobalState, -}; -use client::Client; -use common::{ - assets::Asset, - character::{Character, CharacterId, CharacterItem, MAX_CHARACTERS_PER_PLAYER}, - comp::{self, humanoid}, - npc, LoadoutBuilder, -}; -use conrod_core::{ - color, - color::TRANSPARENT, - event::{Event as WorldEvent, Input}, - input::{Button as ButtonType, Key}, - position::Relative, - widget::{text_box::Event as TextBoxEvent, Button, Image, Rectangle, Scrollbar, Text, TextBox}, - widget_ids, Borderable, Color, Colorable, Labelable, Positionable, Sizeable, UiCell, Widget, -}; -use rand::{thread_rng, Rng}; - -const STARTER_HAMMER: &str = "common.items.weapons.hammer.starter_hammer"; -const STARTER_BOW: &str = "common.items.weapons.bow.starter_bow"; -const STARTER_AXE: &str = "common.items.weapons.axe.starter_axe"; -const STARTER_STAFF: &str = "common.items.weapons.staff.starter_staff"; -const STARTER_SWORD: &str = "common.items.weapons.sword.starter_sword"; -const STARTER_SCEPTRE: &str = "common.items.weapons.sceptre.starter_sceptre"; -// // Use in future MR to make this a starter weapon - -// UI Color-Theme -const UI_MAIN: Color = Color::Rgba(0.61, 0.70, 0.70, 1.0); // Greenish Blue -//const UI_HIGHLIGHT_0: Color = Color::Rgba(0.79, 1.09, 1.09, 1.0); - -widget_ids! { - struct Ids { - // Background and logo - charlist_bg, - charlist_frame, - charlist_bottom, - selection_bot, - charlist_alignment, - selection_scrollbar, - creation_bot, - creation_frame, - creation_alignment, - server_name_text, - change_server, - server_frame_bg, - server_frame, - v_logo, - version, - divider, - bodyspecies_text, - facialfeatures_text, - info_bg, - info_frame, - info_button_align, - info_ok, - info_no, - delete_text, - space, - loading_characters_text, - creating_character_text, - deleting_character_text, - character_error_message, - - //Alpha Disclaimer - alpha_text, - - - // Characters - character_boxes[], - character_deletes[], - character_names[], - character_locations[], - character_levels[], - - character_box_2, - character_name_2, - character_location_2, - character_level_2, - - - // Windows - selection_window, - char_name, - char_level, - creation_window, - select_window_title, - creation_buttons_alignment_1, - creation_buttons_alignment_2, - weapon_heading, - weapon_description, - human_skin_bg, - orc_skin_bg, - dwarf_skin_bg, - undead_skin_bg, - elf_skin_bg, - danari_skin_bg, - name_input_bg, - info, - - // Sliders - hairstyle_slider, - hairstyle_text, - haircolor_slider, - haircolor_text, - skin_slider, - skin_text, - eyecolor_slider, - eyecolor_text, - eyebrows_slider, - eyebrows_text, - beard_slider, - beard_text, - accessories_slider, - accessories_text, - chest_slider, - chest_text, - pants_slider, - pants_text, - - // Buttons - enter_world_button, - back_button, - logout_button, - create_character_button, - delete_button, - create_button, - name_input, - name_field, - species_1, - species_2, - species_3, - species_4, - species_5, - species_6, - body_type_1, - body_type_2, - random_button, - - // Tools - sword, - sword_button, - sceptre, - sceptre_button, - axe, - axe_button, - hammer, - hammer_button, - bow, - bow_button, - staff, - staff_button, - // Char Creation - // Species Icons - male, - female, - human, - orc, - dwarf, - undead, - elf, - danari, - } -} - -image_ids! { - struct Imgs { - - - // Info Window - info_frame: "voxygen.element.frames.info_frame", - - - delete_button: "voxygen.element.buttons.x_red", - delete_button_hover: "voxygen.element.buttons.x_red_hover", - delete_button_press: "voxygen.element.buttons.x_red_press", - - - frame_bot: "voxygen.element.frames.banner_bot", - selection: "voxygen.element.frames.selection", - selection_hover: "voxygen.element.frames.selection_hover", - selection_press: "voxygen.element.frames.selection_press", - - name_input: "voxygen.element.misc_bg.textbox", - - slider_range: "voxygen.element.slider.track", - slider_indicator: "voxygen.element.slider.indicator", - - // Tool Icons - sceptre: "voxygen.element.icons.sceptre", - sword: "voxygen.element.icons.sword", - axe: "voxygen.element.icons.axe", - hammer: "voxygen.element.icons.hammer", - bow: "voxygen.element.icons.bow", - staff: "voxygen.element.icons.staff", - - // Dice icons - dice: "voxygen.element.icons.dice", - dice_hover: "voxygen.element.icons.dice_hover", - dice_press: "voxygen.element.icons.dice_press", - - // Species Icons - human_m: "voxygen.element.icons.human_m", - human_f: "voxygen.element.icons.human_f", - orc_m: "voxygen.element.icons.orc_m", - orc_f: "voxygen.element.icons.orc_f", - dwarf_m: "voxygen.element.icons.dwarf_m", - dwarf_f: "voxygen.element.icons.dwarf_f", - undead_m: "voxygen.element.icons.ud_m", - undead_f: "voxygen.element.icons.ud_f", - elf_m: "voxygen.element.icons.elf_m", - elf_f: "voxygen.element.icons.elf_f", - danari_m: "voxygen.element.icons.danari_m", - danari_f: "voxygen.element.icons.danari_f", - //unknown: "voxygen.element.icons.missing_icon_grey", - // Icon Borders - icon_border: "voxygen.element.buttons.border", - icon_border_mo: "voxygen.element.buttons.border_mo", - icon_border_press: "voxygen.element.buttons.border_press", - icon_border_pressed: "voxygen.element.buttons.border_pressed", - - - button: "voxygen.element.buttons.button", - button_hover: "voxygen.element.buttons.button_hover", - button_press: "voxygen.element.buttons.button_press", - - - nothing: (), - } -} -rotation_image_ids! { - pub struct ImgsRot { - - // Tooltip Test - tt_side: "voxygen/element/frames/tt_test_edge", - tt_corner: "voxygen/element/frames/tt_test_corner_tr", - } -} - -pub enum Event { - Logout, - Play, - AddCharacter { - alias: String, - tool: Option, - body: comp::Body, - }, - DeleteCharacter(CharacterId), -} - -const TEXT_COLOR: Color = Color::Rgba(1.0, 1.0, 1.0, 1.0); -const TEXT_COLOR_2: Color = Color::Rgba(1.0, 1.0, 1.0, 0.2); - -#[derive(PartialEq)] -enum InfoContent { - None, - Deletion(usize), - LoadingCharacters, - CreatingCharacter, - DeletingCharacter, - CharacterError, -} - -impl InfoContent { - pub fn has_content(&self, character_list_loading: &bool) -> bool { - match self { - Self::None => false, - Self::CreatingCharacter | Self::DeletingCharacter | Self::LoadingCharacters => { - *character_list_loading - }, - _ => true, - } - } -} - -#[allow(clippy::large_enum_variant)] // TODO: Pending review in #587 -pub enum Mode { - Select(Option>), - Create { - name: String, - body: humanoid::Body, - loadout: comp::Loadout, - tool: Option<&'static str>, - }, -} - -pub struct CharSelectionUi { - ui: Ui, - ids: Ids, - imgs: Imgs, - rot_imgs: ImgsRot, - fonts: Fonts, - info_content: InfoContent, - i18n: std::sync::Arc, - enter: bool, - pub mode: Mode, - pub selected_character: usize, -} - -impl CharSelectionUi { - pub fn new(global_state: &mut GlobalState) -> Self { - let window = &mut global_state.window; - let settings = &global_state.settings; - - let mut ui = Ui::new(window).unwrap(); - ui.set_scaling_mode(settings.gameplay.ui_scale); - // Generate ids - let ids = Ids::new(ui.id_generator()); - // Load images - let imgs = Imgs::load(&mut ui).expect("Failed to load images!"); - let rot_imgs = ImgsRot::load(&mut ui).expect("Failed to load images!"); - // Load language - let i18n = Localization::load_expect(&i18n_asset_key( - &global_state.settings.language.selected_language, - )); - // Load fonts. - let fonts = Fonts::load(&i18n.fonts, &mut ui).expect("Impossible to load fonts!"); - - Self { - ui, - ids, - imgs, - rot_imgs, - fonts, - info_content: InfoContent::LoadingCharacters, - selected_character: 0, - i18n, - mode: Mode::Select(None), - enter: false, - } - } - - pub fn get_character_list(&self) -> Option> { - match &self.mode { - Mode::Select(data) => data.clone(), - Mode::Create { - name, body, tool, .. - } => { - let body = comp::Body::Humanoid(*body); - - Some(vec![CharacterItem { - character: Character { - id: None, - alias: name.clone(), - }, - body, - level: 1, - loadout: LoadoutBuilder::new() - .defaults() - .active_item(Some(LoadoutBuilder::default_item_config_from_str( - (*tool).expect( - "Attempted to create character with non-existent \ - item_definition_id for tool", - ), - ))) - .build(), - }]) - }, - } - } - - pub fn get_loadout(&mut self) -> Option { - match &mut self.mode { - Mode::Select(character_list) => { - if let Some(data) = character_list { - data.get(self.selected_character).map(|c| c.loadout.clone()) - } else { - None - } - }, - Mode::Create { loadout, tool, .. } => { - loadout.active_item = tool.map(|tool| comp::ItemConfig { - // FIXME: Error gracefully. - item: comp::Item::new_from_asset_expect(tool), - ability1: None, - ability2: None, - ability3: None, - block_ability: None, - dodge_ability: None, - }); - // FIXME: Error gracefully - loadout.chest = Some(comp::Item::new_from_asset_expect( - "common.items.armor.starter.rugged_chest", - )); - // FIXME: Error gracefully - loadout.pants = Some(comp::Item::new_from_asset_expect( - "common.items.armor.starter.rugged_pants", - )); - // FIXME: Error gracefully - loadout.foot = Some(comp::Item::new_from_asset_expect( - "common.items.armor.starter.sandals_0", - )); - loadout.glider = Some(comp::Item::new_from_asset_expect( - "common.items.armor.starter.glider", - )); - Some(loadout.clone()) - }, - } - } - - // TODO: Split this into multiple modules or functions. - #[allow(clippy::useless_let_if_seq)] // TODO: Pending review in #587 - #[allow(clippy::unnecessary_operation)] // TODO: Pending review in #587 - #[allow(clippy::unnested_or_patterns)] // TODO: Pending review in #587 - fn update_layout(&mut self, client: &mut Client) -> Vec { - let mut events = Vec::new(); - - let can_enter_world = match &self.mode { - Mode::Select(opt) => opt.is_some(), - Mode::Create { .. } => false, - }; - - // Handle enter keypress to enter world - if can_enter_world { - for event in self.ui.ui.global_input().events() { - match event { - // TODO allow this to be rebound - WorldEvent::Raw(Input::Press(ButtonType::Keyboard(Key::Return))) - | WorldEvent::Raw(Input::Press(ButtonType::Keyboard(Key::Return2))) - | WorldEvent::Raw(Input::Press(ButtonType::Keyboard(Key::NumPadEnter))) => { - events.push(Event::Play) - }, - _ => {}, - } - } - } - let (ref mut ui_widgets, ref mut tooltip_manager) = self.ui.set_widgets(); - let version = common::util::DISPLAY_VERSION_LONG.clone(); - - // Tooltip - let tooltip_human = Tooltip::new({ - // Edge images [t, b, r, l] - // Corner images [tr, tl, br, bl] - let edge = &self.rot_imgs.tt_side; - let corner = &self.rot_imgs.tt_corner; - ImageFrame::new( - [edge.cw180, edge.none, edge.cw270, edge.cw90], - [corner.none, corner.cw270, corner.cw90, corner.cw180], - Color::Rgba(0.08, 0.07, 0.04, 1.0), - 5.0, - ) - }) - .title_font_size(self.fonts.cyri.scale(15)) - .desc_font_size(self.fonts.cyri.scale(10)) - .parent(ui_widgets.window) - .font_id(self.fonts.cyri.conrod_id) - .desc_text_color(TEXT_COLOR_2); - - // Set the info content if we encountered an error related to characters - if client.character_list.error.is_some() { - self.info_content = InfoContent::CharacterError; - } - - // Information Window - if self - .info_content - .has_content(&client.character_list.loading) - { - Rectangle::fill_with([520.0, 150.0], color::rgba(0.0, 0.0, 0.0, 0.9)) - .mid_top_with_margin_on(ui_widgets.window, 300.0) - .set(self.ids.info_bg, ui_widgets); - Image::new(self.imgs.info_frame) - .w_h(550.0, 150.0) - .middle_of(self.ids.info_bg) - .color(Some(UI_MAIN)) - .set(self.ids.info_frame, ui_widgets); - Rectangle::fill_with([275.0, 150.0], color::TRANSPARENT) - .bottom_left_with_margins_on(self.ids.info_frame, 0.0, 0.0) - .set(self.ids.info_button_align, ui_widgets); - - match self.info_content { - InfoContent::None => unreachable!(), - InfoContent::Deletion(character_index) => { - Text::new(&self.i18n.get("char_selection.delete_permanently")) - .mid_top_with_margin_on(self.ids.info_frame, 40.0) - .font_size(self.fonts.cyri.scale(24)) - .font_id(self.fonts.cyri.conrod_id) - .color(TEXT_COLOR) - .set(self.ids.delete_text, ui_widgets); - if Button::image(self.imgs.button) - .w_h(150.0, 40.0) - .bottom_right_with_margins_on(self.ids.info_button_align, 20.0, 50.0) - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .label_y(Relative::Scalar(2.0)) - .label(&self.i18n.get("common.no")) - .label_font_id(self.fonts.cyri.conrod_id) - .label_font_size(self.fonts.cyri.scale(18)) - .label_color(TEXT_COLOR) - .set(self.ids.info_no, ui_widgets) - .was_clicked() - { - self.info_content = InfoContent::None; - }; - if Button::image(self.imgs.button) - .w_h(150.0, 40.0) - .right_from(self.ids.info_no, 100.0) - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .label_y(Relative::Scalar(2.0)) - .label(&self.i18n.get("common.yes")) - .label_font_id(self.fonts.cyri.conrod_id) - .label_font_size(self.fonts.cyri.scale(18)) - .label_color(TEXT_COLOR) - .set(self.ids.info_ok, ui_widgets) - .was_clicked() - { - self.info_content = InfoContent::None; - - if let Some(character_item) = - client.character_list.characters.get(character_index) - { - // Unsaved characters have no id, this should never be the case here - if let Some(character_id) = character_item.character.id { - self.info_content = InfoContent::DeletingCharacter; - - events.push(Event::DeleteCharacter(character_id)); - } - } - }; - }, - InfoContent::LoadingCharacters => { - Text::new(&self.i18n.get("char_selection.loading_characters")) - .mid_top_with_margin_on(self.ids.info_frame, 40.0) - .font_size(self.fonts.cyri.scale(24)) - .font_id(self.fonts.cyri.conrod_id) - .color(TEXT_COLOR) - .set(self.ids.loading_characters_text, ui_widgets); - }, - InfoContent::CreatingCharacter => { - Text::new(&self.i18n.get("char_selection.creating_character")) - .mid_top_with_margin_on(self.ids.info_frame, 40.0) - .font_size(self.fonts.cyri.scale(24)) - .font_id(self.fonts.cyri.conrod_id) - .color(TEXT_COLOR) - .set(self.ids.creating_character_text, ui_widgets); - }, - InfoContent::DeletingCharacter => { - Text::new(&self.i18n.get("char_selection.deleting_character")) - .mid_top_with_margin_on(self.ids.info_frame, 40.0) - .font_size(self.fonts.cyri.scale(24)) - .font_id(self.fonts.cyri.conrod_id) - .color(TEXT_COLOR) - .set(self.ids.deleting_character_text, ui_widgets); - }, - InfoContent::CharacterError => { - if let Some(error_message) = &client.character_list.error { - Text::new(&format!( - "{}: {}", - &self.i18n.get("common.error"), - error_message - )) - .mid_top_with_margin_on(self.ids.info_frame, 40.0) - .font_size(self.fonts.cyri.scale(24)) - .font_id(self.fonts.cyri.conrod_id) - .color(TEXT_COLOR) - .set(self.ids.character_error_message, ui_widgets); - - if Button::image(self.imgs.button) - .w_h(150.0, 40.0) - .bottom_right_with_margins_on(self.ids.info_button_align, 20.0, 20.0) - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .label_y(Relative::Scalar(2.0)) - .label(&self.i18n.get("common.close")) - .label_font_id(self.fonts.cyri.conrod_id) - .label_font_size(self.fonts.cyri.scale(18)) - .label_color(TEXT_COLOR) - .set(self.ids.info_ok, ui_widgets) - .was_clicked() - { - self.info_content = InfoContent::None; - client.character_list.error = None; - } - } else { - self.info_content = InfoContent::None; - } - }, - } - } - - // Character Selection ///////////////// - match &mut self.mode { - Mode::Select(data) => { - // Set active body - *data = if client - .character_list - .characters - .get(self.selected_character) - .is_some() - { - Some(client.character_list.characters.clone()) - } else { - None - }; - - // Background for Server Frame - Rectangle::fill_with([400.0, 95.0], color::rgba(0.0, 0.0, 0.0, 0.8)) - .top_left_with_margins_on(ui_widgets.window, 30.0, 30.0) - .set(self.ids.server_frame_bg, ui_widgets); - - // Background for Char List - Rectangle::fill_with([400.0, 800.0], color::rgba(0.0, 0.0, 0.0, 0.8)) - .down_from(self.ids.server_frame_bg, 5.0) - .set(self.ids.charlist_frame, ui_widgets); - Image::new(self.imgs.frame_bot) - .w_h(400.0, 48.0) - .down_from(self.ids.charlist_frame, 0.0) - .color(Some(Color::Rgba(1.0, 1.0, 1.0, 0.8))) - .set(self.ids.selection_bot, ui_widgets); - Rectangle::fill_with([386.0, 800.0], color::TRANSPARENT) - .mid_top_with_margin_on(self.ids.charlist_frame, 2.0) - .scroll_kids() - .scroll_kids_vertically() - .set(self.ids.charlist_alignment, ui_widgets); - Scrollbar::y_axis(self.ids.charlist_alignment) - .thickness(5.0) - .auto_hide(true) - .color(UI_MAIN) - .set(self.ids.selection_scrollbar, ui_widgets); - // Server Name - Text::new(&client.server_info.name) - .mid_top_with_margin_on(self.ids.server_frame_bg, 5.0) - .font_size(self.fonts.cyri.scale(26)) - .font_id(self.fonts.cyri.conrod_id) - .color(TEXT_COLOR) - .set(self.ids.server_name_text, ui_widgets); - //Change Server - if Button::image(self.imgs.button) - .mid_top_with_margin_on(self.ids.server_frame_bg, 45.0) - .w_h(200.0, 40.0) - .parent(self.ids.charlist_bg) - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .label(&self.i18n.get("char_selection.change_server")) - .label_color(TEXT_COLOR) - .label_font_id(self.fonts.cyri.conrod_id) - .label_font_size(self.fonts.cyri.scale(18)) - .label_y(conrod_core::position::Relative::Scalar(3.0)) - .set(self.ids.change_server, ui_widgets) - .was_clicked() - { - events.push(Event::Logout); - } - - // Enter World Button - let character_count = client.character_list.characters.len(); - let enter_world_str = &self.i18n.get("char_selection.enter_world"); - let enter_button = Button::image(self.imgs.button) - .mid_bottom_with_margin_on(ui_widgets.window, 10.0) - .w_h(250.0, 60.0) - .label(enter_world_str) - .label_font_size(self.fonts.cyri.scale(26)) - .label_font_id(self.fonts.cyri.conrod_id) - .label_y(conrod_core::position::Relative::Scalar(3.0)); - - if can_enter_world { - if enter_button - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .label_color(TEXT_COLOR) - .set(self.ids.enter_world_button, ui_widgets) - .was_clicked() - { - self.enter = !self.enter; - if self.enter { - events.push(Event::Play) - }; - } - } else { - &enter_button - .label_color(TEXT_COLOR_2) - .set(self.ids.enter_world_button, ui_widgets); - } - - // Logout_Button - if Button::image(self.imgs.button) - .bottom_left_with_margins_on(ui_widgets.window, 10.0, 10.0) - .w_h(150.0, 40.0) - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .label(&self.i18n.get("char_selection.logout")) - .label_font_id(self.fonts.cyri.conrod_id) - .label_color(TEXT_COLOR) - .label_font_size(self.fonts.cyri.scale(20)) - .label_y(conrod_core::position::Relative::Scalar(3.0)) - .set(self.ids.logout_button, ui_widgets) - .was_clicked() - { - events.push(Event::Logout); - } - - // Alpha Version - Text::new(&version) - .top_right_with_margins_on(ui_widgets.window, 5.0, 5.0) - .font_size(self.fonts.cyri.scale(14)) - .font_id(self.fonts.cyri.conrod_id) - .color(TEXT_COLOR) - .set(self.ids.version, ui_widgets); - // Alpha Disclaimer - Text::new(&format!( - "Veloren {}", - common::util::DISPLAY_VERSION.as_str() - )) - .font_id(self.fonts.cyri.conrod_id) - .font_size(self.fonts.cyri.scale(10)) - .color(TEXT_COLOR) - .mid_top_with_margin_on(ui_widgets.window, 2.0) - .set(self.ids.alpha_text, ui_widgets); - - // Resize character selection widgets - self.ids - .character_boxes - .resize(character_count, &mut ui_widgets.widget_id_generator()); - self.ids - .character_deletes - .resize(character_count, &mut ui_widgets.widget_id_generator()); - self.ids - .character_names - .resize(character_count, &mut ui_widgets.widget_id_generator()); - self.ids - .character_levels - .resize(character_count, &mut ui_widgets.widget_id_generator()); - self.ids - .character_locations - .resize(character_count, &mut ui_widgets.widget_id_generator()); - - // Character selection - for (i, character_item) in client.character_list.characters.iter().enumerate() { - let character_box = Button::image(if self.selected_character == i { - self.imgs.selection_hover - } else { - self.imgs.selection - }); - let character_box = if i == 0 { - character_box.top_left_with_margins_on( - self.ids.charlist_alignment, - 0.0, - 2.0, - ) - } else { - character_box.down_from(self.ids.character_boxes[i - 1], 5.0) - }; - if character_box - .w_h(386.0, 80.0) - .image_color(Color::Rgba(1.0, 1.0, 1.0, 0.8)) - .hover_image(self.imgs.selection_hover) - .press_image(self.imgs.selection_press) - .label_font_id(self.fonts.cyri.conrod_id) - .label_y(conrod_core::position::Relative::Scalar(20.0)) - .set(self.ids.character_boxes[i], ui_widgets) - .was_clicked() - { - self.selected_character = i; - } - if Button::image(self.imgs.delete_button) - .w_h(30.0 * 0.5, 30.0 * 0.5) - .top_right_with_margins_on(self.ids.character_boxes[i], 15.0, 15.0) - .hover_image(self.imgs.delete_button_hover) - .press_image(self.imgs.delete_button_press) - .with_tooltip( - tooltip_manager, - &self.i18n.get("char_selection.delete_permanently"), - "", - &tooltip_human, - TEXT_COLOR, - ) - .set(self.ids.character_deletes[i], ui_widgets) - .was_clicked() - { - self.info_content = InfoContent::Deletion(i); - } - Text::new(&character_item.character.alias) - .top_left_with_margins_on(self.ids.character_boxes[i], 6.0, 9.0) - .font_size(self.fonts.cyri.scale(19)) - .font_id(self.fonts.cyri.conrod_id) - .color(TEXT_COLOR) - .set(self.ids.character_names[i], ui_widgets); - - Text::new( - &self - .i18n - .get("char_selection.level_fmt") - .replace("{level_nb}", &character_item.level.to_string()), - ) - .down_from(self.ids.character_names[i], 4.0) - .font_size(self.fonts.cyri.scale(17)) - .font_id(self.fonts.cyri.conrod_id) - .color(TEXT_COLOR) - .set(self.ids.character_levels[i], ui_widgets); - - Text::new(&self.i18n.get("char_selection.uncanny_valley")) - .down_from(self.ids.character_levels[i], 4.0) - .font_size(self.fonts.cyri.scale(17)) - .font_id(self.fonts.cyri.conrod_id) - .color(TEXT_COLOR) - .set(self.ids.character_locations[i], ui_widgets); - } - - // Create Character Button - let create_char_button = Button::image(self.imgs.selection); - - let create_char_button = if character_count > 0 { - create_char_button.down_from(self.ids.character_boxes[character_count - 1], 5.0) - } else { - create_char_button.top_left_with_margins_on( - self.ids.charlist_alignment, - 0.0, - 2.0, - ) - }; - - let character_limit_reached = character_count >= MAX_CHARACTERS_PER_PLAYER; - - let color = if character_limit_reached { - Color::Rgba(0.38, 0.38, 0.10, 1.0) - } else { - Color::Rgba(0.38, 1.0, 0.07, 1.0) - }; - - if create_char_button - .w_h(386.0, 80.0) - .hover_image(self.imgs.selection_hover) - .press_image(self.imgs.selection_press) - .label(&self.i18n.get("char_selection.create_new_charater")) - .label_color(color) - .label_font_id(self.fonts.cyri.conrod_id) - .image_color(color) - .set(self.ids.character_box_2, ui_widgets) - .was_clicked() - && !character_limit_reached - { - self.mode = Mode::Create { - name: "Character Name".to_string(), - body: humanoid::Body::random(), - loadout: comp::Loadout::default(), - tool: Some(STARTER_SWORD), - }; - } - - // LOADING SCREEN HERE - if self.enter { /*stuff*/ }; - }, - // Character_Creation - // ////////////////////////////////////////////////////////////////////// - Mode::Create { - name, - body, - loadout: _, - tool, - } => { - let mut rng = thread_rng(); - let mut to_select = false; - // Back Button - if Button::image(self.imgs.button) - .bottom_left_with_margins_on(ui_widgets.window, 10.0, 10.0) - .w_h(150.0, 40.0) - .hover_image(self.imgs.button_hover) - .press_image(self.imgs.button_press) - .label(&self.i18n.get("common.back")) - .label_font_id(self.fonts.cyri.conrod_id) - .label_color(TEXT_COLOR) - .label_font_size(self.fonts.cyri.scale(20)) - .label_y(conrod_core::position::Relative::Scalar(3.0)) - .set(self.ids.back_button, ui_widgets) - .was_clicked() - { - to_select = true; - } - // Create Button - let create_button = Button::image(self.imgs.button) - .bottom_right_with_margins_on(ui_widgets.window, 10.0, 10.0) - .w_h(150.0, 40.0) - .hover_image(if *name != "Character Name" && *name != "" { - self.imgs.button_hover - } else { - self.imgs.button - }) - .press_image(if *name != "Character Name" && *name != "" { - self.imgs.button_press - } else { - self.imgs.button - }) - .label(&self.i18n.get("common.create")) - .label_font_id(self.fonts.cyri.conrod_id) - .label_color(if *name != "Character Name" && *name != "" { - TEXT_COLOR - } else { - TEXT_COLOR_2 - }) - .label_font_size(self.fonts.cyri.scale(20)) - .label_y(conrod_core::position::Relative::Scalar(3.0)); - - if *name == "Character Name" || *name == "" { - //TODO: We need a server side list of disallowed names and certain naming rules - if create_button - .with_tooltip( - tooltip_manager, - &self.i18n.get("char_selection.create_info_name"), - "", - &tooltip_human, - TEXT_COLOR, - ) - .set(self.ids.create_button, ui_widgets) - .was_clicked() - {} - } else if create_button - .set(self.ids.create_button, ui_widgets) - .was_clicked() - { - self.info_content = InfoContent::CreatingCharacter; - - events.push(Event::AddCharacter { - alias: name.clone(), - tool: tool.map(|tool| tool.to_string()), - body: comp::Body::Humanoid(*body), - }); - - to_select = true; - } - // Character Name Input - Rectangle::fill_with([320.0, 50.0], color::rgba(0.0, 0.0, 0.0, 0.97)) - .mid_bottom_with_margin_on(ui_widgets.window, 20.0) - .set(self.ids.name_input_bg, ui_widgets); - Button::image(self.imgs.name_input) - .image_color(Color::Rgba(1.0, 1.0, 1.0, 0.9)) - .w_h(337.0, 67.0) - .middle_of(self.ids.name_input_bg) - .set(self.ids.name_input, ui_widgets); - for event in TextBox::new(name) - .w_h(300.0, 60.0) - .mid_top_with_margin_on(self.ids.name_input, 2.0) - .font_size(self.fonts.cyri.scale(26)) - .font_id(self.fonts.cyri.conrod_id) - .center_justify() - .text_color(TEXT_COLOR) - .font_id(self.fonts.cyri.conrod_id) - .color(TRANSPARENT) - .border_color(TRANSPARENT) - .set(self.ids.name_field, ui_widgets) - { - match event { - TextBoxEvent::Update(new_name) => *name = new_name, - TextBoxEvent::Enter => {}, - } - } - - // Window - Rectangle::fill_with( - [400.0, ui_widgets.win_h - ui_widgets.win_h * 0.15], - color::rgba(0.0, 0.0, 0.0, 0.8), - ) - .top_left_with_margins_on(ui_widgets.window, 30.0, 30.0) - .set(self.ids.creation_frame, ui_widgets); - Image::new(self.imgs.frame_bot) - .w_h(400.0, 48.0) - .down_from(self.ids.creation_frame, 0.0) - .color(Some(Color::Rgba(1.0, 1.0, 1.0, 0.8))) - .set(self.ids.creation_bot, ui_widgets); - Rectangle::fill_with( - [386.0, ui_widgets.win_h - ui_widgets.win_h * 0.15], - color::TRANSPARENT, - ) - .mid_top_with_margin_on(self.ids.creation_frame, 10.0) - .scroll_kids_vertically() - .set(self.ids.creation_alignment, ui_widgets); - Scrollbar::y_axis(self.ids.creation_alignment) - .thickness(5.0) - .auto_hide(true) - .rgba(0.33, 0.33, 0.33, 1.0) - .set(self.ids.selection_scrollbar, ui_widgets); - - // BodyType/Species Icons - let body_m_ico = match body.species { - humanoid::Species::Human => self.imgs.human_m, - humanoid::Species::Orc => self.imgs.orc_m, - humanoid::Species::Dwarf => self.imgs.dwarf_m, - humanoid::Species::Elf => self.imgs.elf_m, - humanoid::Species::Undead => self.imgs.undead_m, - humanoid::Species::Danari => self.imgs.danari_m, - }; - let body_f_ico = match body.species { - humanoid::Species::Human => self.imgs.human_f, - humanoid::Species::Orc => self.imgs.orc_f, - humanoid::Species::Dwarf => self.imgs.dwarf_f, - humanoid::Species::Elf => self.imgs.elf_f, - humanoid::Species::Undead => self.imgs.undead_f, - humanoid::Species::Danari => self.imgs.danari_f, - }; - // Alignment - Rectangle::fill_with([140.0, 72.0], color::TRANSPARENT) - .mid_top_with_margin_on(self.ids.creation_alignment, 60.0) - .set(self.ids.creation_buttons_alignment_1, ui_widgets); - // Bodytype M - Image::new(body_m_ico) - .w_h(70.0, 70.0) - .top_left_with_margins_on(self.ids.creation_buttons_alignment_1, 0.0, 0.0) - .set(self.ids.male, ui_widgets); - if Button::image(if let humanoid::BodyType::Male = body.body_type { - self.imgs.icon_border_pressed - } else { - self.imgs.icon_border - }) - .middle_of(self.ids.male) - .hover_image(self.imgs.icon_border_mo) - .press_image(self.imgs.icon_border_press) - .set(self.ids.body_type_1, ui_widgets) - .was_clicked() - { - body.body_type = humanoid::BodyType::Male; - body.validate(); - } - // Bodytype F - Image::new(body_f_ico) - .w_h(70.0, 70.0) - .top_right_with_margins_on(self.ids.creation_buttons_alignment_1, 0.0, 0.0) - .set(self.ids.female, ui_widgets); - if Button::image(if let humanoid::BodyType::Female = body.body_type { - self.imgs.icon_border_pressed - } else { - self.imgs.icon_border - }) - .middle_of(self.ids.female) - .hover_image(self.imgs.icon_border_mo) - .press_image(self.imgs.icon_border_press) - .set(self.ids.body_type_2, ui_widgets) - .was_clicked() - { - body.body_type = humanoid::BodyType::Female; - body.validate(); - } - - // Alignment for Species and Tools - Rectangle::fill_with([214.0, 304.0], color::TRANSPARENT) - .mid_bottom_with_margin_on(self.ids.creation_buttons_alignment_1, -324.0) - .set(self.ids.creation_buttons_alignment_2, ui_widgets); - - let (human_icon, orc_icon, dwarf_icon, elf_icon, undead_icon, danari_icon) = - match body.body_type { - humanoid::BodyType::Male => ( - self.imgs.human_m, - self.imgs.orc_m, - self.imgs.dwarf_m, - self.imgs.elf_m, - self.imgs.undead_m, - self.imgs.danari_m, - ), - humanoid::BodyType::Female => ( - self.imgs.human_f, - self.imgs.orc_f, - self.imgs.dwarf_f, - self.imgs.elf_f, - self.imgs.undead_f, - self.imgs.danari_f, - ), - }; - // Human - Image::new(human_icon) - .w_h(70.0, 70.0) - .top_left_with_margins_on(self.ids.creation_buttons_alignment_2, 0.0, 0.0) - .set(self.ids.human, ui_widgets); - if Button::image(if let humanoid::Species::Human = body.species { - self.imgs.icon_border_pressed - } else { - self.imgs.icon_border - }) - .middle_of(self.ids.human) - .hover_image(self.imgs.icon_border_mo) - .press_image(self.imgs.icon_border_press) - .with_tooltip( - tooltip_manager, - &self.i18n.get("common.species.human"), - "", - &tooltip_human, - TEXT_COLOR, - ) - .set(self.ids.species_1, ui_widgets) - .was_clicked() - { - body.species = humanoid::Species::Human; - body.validate(); - } - - // Orc - Image::new(orc_icon) - .w_h(70.0, 70.0) - .right_from(self.ids.human, 2.0) - .set(self.ids.orc, ui_widgets); - if Button::image(if let humanoid::Species::Orc = body.species { - self.imgs.icon_border_pressed - } else { - self.imgs.icon_border - }) - .middle_of(self.ids.orc) - .hover_image(self.imgs.icon_border_mo) - .press_image(self.imgs.icon_border_press) - .with_tooltip( - tooltip_manager, - &self.i18n.get("common.species.orc"), - "", - &tooltip_human, - TEXT_COLOR, - ) - .set(self.ids.species_2, ui_widgets) - .was_clicked() - { - body.species = humanoid::Species::Orc; - body.validate(); - } - // Dwarf - Image::new(dwarf_icon) - .w_h(70.0, 70.0) - .right_from(self.ids.orc, 2.0) - .set(self.ids.dwarf, ui_widgets); - if Button::image(if let humanoid::Species::Dwarf = body.species { - self.imgs.icon_border_pressed - } else { - self.imgs.icon_border - }) - .middle_of(self.ids.dwarf) - .hover_image(self.imgs.icon_border_mo) - .press_image(self.imgs.icon_border_press) - .with_tooltip( - tooltip_manager, - &self.i18n.get("common.species.dwarf"), - "", - &tooltip_human, - TEXT_COLOR, - ) - .set(self.ids.species_3, ui_widgets) - .was_clicked() - { - body.species = humanoid::Species::Dwarf; - body.validate(); - } - // Elf - Image::new(elf_icon) - .w_h(70.0, 70.0) - .down_from(self.ids.human, 2.0) - .set(self.ids.elf, ui_widgets); - if Button::image(if let humanoid::Species::Elf = body.species { - self.imgs.icon_border_pressed - } else { - self.imgs.icon_border - }) - .middle_of(self.ids.elf) - .hover_image(self.imgs.icon_border_mo) - .press_image(self.imgs.icon_border_press) - .with_tooltip( - tooltip_manager, - &self.i18n.get("common.species.elf"), - "", - &tooltip_human, - TEXT_COLOR, - ) - .set(self.ids.species_4, ui_widgets) - .was_clicked() - { - body.species = humanoid::Species::Elf; - body.validate(); - } - - // Undead - Image::new(undead_icon) - .w_h(70.0, 70.0) - .right_from(self.ids.elf, 2.0) - .set(self.ids.undead, ui_widgets); - if Button::image(if let humanoid::Species::Undead = body.species { - self.imgs.icon_border_pressed - } else { - self.imgs.icon_border - }) - .middle_of(self.ids.undead) - .hover_image(self.imgs.icon_border_mo) - .press_image(self.imgs.icon_border_press) - .with_tooltip( - tooltip_manager, - &self.i18n.get("common.species.undead"), - "", - &tooltip_human, - TEXT_COLOR, - ) - .set(self.ids.species_5, ui_widgets) - .was_clicked() - { - body.species = humanoid::Species::Undead; - body.validate(); - } - // Danari - Image::new(danari_icon) - .w_h(70.0, 70.0) - .right_from(self.ids.undead, 2.0) - .set(self.ids.danari, ui_widgets); - if Button::image(if let humanoid::Species::Danari = body.species { - self.imgs.icon_border_pressed - } else { - self.imgs.icon_border - }) - .middle_of(self.ids.danari) - .hover_image(self.imgs.icon_border_mo) - .press_image(self.imgs.icon_border_press) - .with_tooltip( - tooltip_manager, - &self.i18n.get("common.species.danari"), - "", - &tooltip_human, - TEXT_COLOR, - ) - .set(self.ids.species_6, ui_widgets) - .was_clicked() - { - body.species = humanoid::Species::Danari; - body.validate(); - } - // Healing Sceptre - Image::new(self.imgs.sceptre) - .w_h(70.0, 70.0) - .bottom_left_with_margins_on(self.ids.creation_buttons_alignment_2, 0.0, 0.0) - .set(self.ids.sceptre, ui_widgets); - if Button::image(if let Some(STARTER_SCEPTRE) = tool { - self.imgs.icon_border_pressed - } else { - self.imgs.icon_border - }) - .middle_of(self.ids.sceptre) - .hover_image(self.imgs.icon_border_mo) - .press_image(self.imgs.icon_border_press) - .with_tooltip( - tooltip_manager, - &self.i18n.get("common.weapons.sceptre"), - "", - &tooltip_human, - TEXT_COLOR, - ) - .set(self.ids.sceptre_button, ui_widgets) - .was_clicked() - { - *tool = Some(STARTER_SCEPTRE); - } - - // Bow - Image::new(self.imgs.bow) - .w_h(70.0, 70.0) - .right_from(self.ids.sceptre, 2.0) - .set(self.ids.bow, ui_widgets); - if Button::image(if let Some(STARTER_BOW) = tool { - self.imgs.icon_border_pressed - } else { - self.imgs.icon_border - }) - .middle_of(self.ids.bow) - .hover_image(self.imgs.icon_border_mo) - .press_image(self.imgs.icon_border_press) - .with_tooltip( - tooltip_manager, - &self.i18n.get("common.weapons.bow"), - "", - &tooltip_human, - TEXT_COLOR, - ) - .set(self.ids.bow_button, ui_widgets) - .was_clicked() - { - *tool = Some(STARTER_BOW); - } - // Staff - Image::new(self.imgs.staff) - .w_h(70.0, 70.0) - .right_from(self.ids.bow, 2.0) - .set(self.ids.staff, ui_widgets); - if Button::image(if let Some(STARTER_STAFF) = tool { - self.imgs.icon_border_pressed - } else { - self.imgs.icon_border - }) - .middle_of(self.ids.staff) - .hover_image(self.imgs.icon_border_mo) - .press_image(self.imgs.icon_border_press) - .with_tooltip( - tooltip_manager, - &self.i18n.get("common.weapons.staff"), - "", - &tooltip_human, - TEXT_COLOR, - ) - .set(self.ids.staff_button, ui_widgets) - .was_clicked() - { - *tool = Some(STARTER_STAFF); - } - // Sword - Image::new(self.imgs.sword) - .w_h(70.0, 70.0) - .up_from(self.ids.sceptre, 2.0) - .set(self.ids.sword, ui_widgets); - if Button::image(if let Some(STARTER_SWORD) = tool { - self.imgs.icon_border_pressed - } else { - self.imgs.icon_border - }) - .middle_of(self.ids.sword) - .hover_image(self.imgs.icon_border_mo) - .press_image(self.imgs.icon_border_press) - .with_tooltip( - tooltip_manager, - &self.i18n.get("common.weapons.sword"), - "", - &tooltip_human, - TEXT_COLOR, - ) - .set(self.ids.sword_button, ui_widgets) - .was_clicked() - { - *tool = Some(STARTER_SWORD); - } - - // Hammer - Image::new(self.imgs.hammer) - .w_h(70.0, 70.0) - .right_from(self.ids.sword, 2.0) - .set(self.ids.hammer, ui_widgets); - if Button::image(if let Some(STARTER_HAMMER) = tool { - self.imgs.icon_border_pressed - } else { - self.imgs.icon_border - }) - .middle_of(self.ids.hammer) - .hover_image(self.imgs.icon_border_mo) - .press_image(self.imgs.icon_border_press) - .with_tooltip( - tooltip_manager, - &self.i18n.get("common.weapons.hammer"), - "", - &tooltip_human, - TEXT_COLOR, - ) - .set(self.ids.hammer_button, ui_widgets) - .was_clicked() - { - *tool = Some(STARTER_HAMMER); - } - - // Axe - Image::new(self.imgs.axe) - .w_h(70.0, 70.0) - .right_from(self.ids.hammer, 2.0) - .set(self.ids.axe, ui_widgets); - if Button::image(if let Some(STARTER_AXE) = tool { - self.imgs.icon_border_pressed - } else { - self.imgs.icon_border - }) - .middle_of(self.ids.axe) - .hover_image(self.imgs.icon_border_mo) - .press_image(self.imgs.icon_border_press) - .with_tooltip( - tooltip_manager, - &self.i18n.get("common.weapons.axe"), - "", - &tooltip_human, - TEXT_COLOR, - ) - .set(self.ids.axe_button, ui_widgets) - .was_clicked() - { - *tool = Some(STARTER_AXE); - } - // Random button - if Button::image(self.imgs.dice) - .wh([35.0; 2]) - .bottom_left_with_margins_on(self.ids.name_input, 15.0, -45.0) - .hover_image(self.imgs.dice_hover) - .press_image(self.imgs.dice_press) - .with_tooltip( - tooltip_manager, - &self.i18n.get("common.rand_appearance"), - "", - &tooltip_human, - TEXT_COLOR, - ) - .set(self.ids.random_button, ui_widgets) - .was_clicked() - { - body.hair_style = - rng.gen_range(0, body.species.num_hair_styles(body.body_type)); - body.beard = rng.gen_range(0, body.species.num_beards(body.body_type)); - body.accessory = rng.gen_range(0, body.species.num_accessories(body.body_type)); - body.hair_color = rng.gen_range(0, body.species.num_hair_colors()); - body.skin = rng.gen_range(0, body.species.num_skin_colors()); - body.eye_color = rng.gen_range(0, body.species.num_eye_colors()); - body.eyes = rng.gen_range(0, body.species.num_eyes(body.body_type)); - *name = npc::get_npc_name(npc::NpcKind::Humanoid).to_string(); - } - // Sliders - let (cyri, cyri_size, slider_indicator, slider_range) = ( - self.fonts.cyri.conrod_id, - self.fonts.cyri.scale(18), - self.imgs.slider_indicator, - self.imgs.slider_range, - ); - let char_slider = move |prev_id, - text: &str, - text_id, - max, - selected_val, - slider_id, - ui_widgets: &mut UiCell| { - Text::new(text) - .down_from(prev_id, 22.0) - .align_middle_x_of(prev_id) - .font_size(cyri_size) - .font_id(cyri) - .color(TEXT_COLOR) - .set(text_id, ui_widgets); - ImageSlider::discrete(selected_val, 0, max, slider_indicator, slider_range) - .w_h(208.0, 22.0) - .down_from(text_id, 8.0) - .align_middle_x() - .track_breadth(12.0) - .slider_length(10.0) - .pad_track((5.0, 5.0)) - .set(slider_id, ui_widgets) - }; - // Hair Style - if let Some(new_val) = char_slider( - self.ids.creation_buttons_alignment_2, - self.i18n.get("char_selection.hair_style"), - self.ids.hairstyle_text, - body.species.num_hair_styles(body.body_type) as usize - 1, - body.hair_style as usize, - self.ids.hairstyle_slider, - ui_widgets, - ) { - body.hair_style = new_val as u8; - } - // Hair Color - if let Some(new_val) = char_slider( - self.ids.hairstyle_slider, - self.i18n.get("char_selection.hair_color"), - self.ids.haircolor_text, - body.species.num_hair_colors() as usize - 1, - body.hair_color as usize, - self.ids.haircolor_slider, - ui_widgets, - ) { - body.hair_color = new_val as u8; - } - // Skin - if let Some(new_val) = char_slider( - self.ids.haircolor_slider, - self.i18n.get("char_selection.skin"), - self.ids.skin_text, - body.species.num_skin_colors() as usize - 1, - body.skin as usize, - self.ids.skin_slider, - ui_widgets, - ) { - body.skin = new_val as u8; - } - // Eyebrows - if let Some(new_val) = char_slider( - self.ids.skin_slider, - self.i18n.get("char_selection.eyeshape"), - self.ids.eyebrows_text, - body.species.num_eyes(body.body_type) as usize - 1, - body.eyes as usize, - self.ids.eyebrows_slider, - ui_widgets, - ) { - body.eyes = new_val as u8; - } - // EyeColor - if let Some(new_val) = char_slider( - self.ids.eyebrows_slider, - self.i18n.get("char_selection.eye_color"), - self.ids.eyecolor_text, - body.species.num_eye_colors() as usize - 1, - body.eye_color as usize, - self.ids.eyecolor_slider, - ui_widgets, - ) { - body.eye_color = new_val as u8; - } - // Accessories - let _current_accessory = body.accessory; - if let Some(new_val) = char_slider( - self.ids.eyecolor_slider, - self.i18n.get("char_selection.accessories"), - self.ids.accessories_text, - body.species.num_accessories(body.body_type) as usize - 1, - body.accessory as usize, - self.ids.accessories_slider, - ui_widgets, - ) { - body.accessory = new_val as u8; - } - // Beard - if body.species.num_beards(body.body_type) > 1 { - if let Some(new_val) = char_slider( - self.ids.accessories_slider, - self.i18n.get("char_selection.beard"), - self.ids.beard_text, - body.species.num_beards(body.body_type) as usize - 1, - body.beard as usize, - self.ids.beard_slider, - ui_widgets, - ) { - body.beard = new_val as u8; - } - } else { - Text::new(&self.i18n.get("char_selection.beard")) - .mid_bottom_with_margin_on(self.ids.accessories_slider, -40.0) - .font_size(self.fonts.cyri.scale(18)) - .font_id(self.fonts.cyri.conrod_id) - .color(TEXT_COLOR_2) - .set(self.ids.beard_text, ui_widgets); - ImageSlider::discrete(5, 0, 10, self.imgs.nothing, self.imgs.slider_range) - .w_h(208.0, 22.0) - .mid_bottom_with_margin_on(self.ids.beard_text, -30.0) - .track_breadth(12.0) - .slider_length(10.0) - .track_color(Color::Rgba(1.0, 1.0, 1.0, 0.2)) - .slider_color(Color::Rgba(1.0, 1.0, 1.0, 0.2)) - .pad_track((5.0, 5.0)) - .set(self.ids.beard_slider, ui_widgets); - } - // Chest - /*let armor = load_glob::("common.items.armor.chest.*") - .expect("Unable to load armor!"); - if let Some(new_val) = char_slider( - self.ids.beard_slider, - self.i18n.get("char_selection.chest_color"), - self.ids.chest_text, - armor.len() - 1, - armor - .iter() - .position(|c| { - loadout - .chest - .as_ref() - .map(|lc| lc == c.borrow()) - .unwrap_or_default() - }) - .unwrap_or(0), - self.ids.chest_slider, - ui_widgets, - ) { - loadout.chest = Some((*armor[new_val]).clone()); - }*/ - // Pants - /*let current_pants = body.pants; - if let Some(new_val) = char_slider( - self.ids.chest_slider, - "Pants", - self.ids.pants_text, - humanoid::ALL_PANTS.len() - 1, - humanoid::ALL_PANTS - .iter() - .position(|&c| c == current_pants) - .unwrap_or(0), - self.ids.pants_slider, - ui_widgets, - ) { - body.pants = humanoid::ALL_PANTS[new_val]; - }*/ - Rectangle::fill_with([20.0, 20.0], color::TRANSPARENT) - .down_from(self.ids.beard_slider, 15.0) - .set(self.ids.space, ui_widgets); - - if to_select { - self.mode = Mode::Select(None); - } - }, // Char Creation fin - } - - events - } - - pub fn handle_event(&mut self, event: WinEvent) -> bool { - match event { - WinEvent::Ui(event) => { - self.ui.handle_event(event); - true - }, - WinEvent::MouseButton(_, PressState::Pressed) => !self.ui.no_widget_capturing_mouse(), - _ => false, - } - } - - pub fn maintain(&mut self, global_state: &mut GlobalState, client: &mut Client) -> Vec { - let events = self.update_layout(client); - self.ui.maintain(global_state.window.renderer_mut(), None); - events - } - - pub fn render(&self, renderer: &mut Renderer, globals: &Consts) { - self.ui.render(renderer, Some(globals)); - } -} diff --git a/voxygen/src/menu/char_selection/ui/mod.rs b/voxygen/src/menu/char_selection/ui/mod.rs index 3993ff82f5..3281c2ac53 100644 --- a/voxygen/src/menu/char_selection/ui/mod.rs +++ b/voxygen/src/menu/char_selection/ui/mod.rs @@ -55,8 +55,8 @@ const STARTER_SCEPTRE: &str = "common.items.weapons.sceptre.starter_sceptre"; // TODO: what does this comment mean? // // Use in future MR to make this a starter weapon -// TODO: use for info popup frame/background and for char list scrollbar -const UI_MAIN: iced::Color = iced::Color::from_rgba(0.61, 0.70, 0.70, 1.0); // Greenish Blue +// TODO: use for info popup frame/background +const UI_MAIN: Rgba = Rgba::new(156, 179, 179, 255); // Greenish Blue image_ids_ice! { struct Imgs { @@ -66,9 +66,6 @@ image_ids_ice! { slider_range: "voxygen.element.slider.track", slider_indicator: "voxygen.element.slider.indicator", - gray_corner: "voxygen.element.frames.gray.corner", - gray_edge: "voxygen.element.frames.gray.edge", - selection: "voxygen.element.frames.selection", selection_hover: "voxygen.element.frames.selection_hover", selection_press: "voxygen.element.frames.selection_press", @@ -130,6 +127,7 @@ pub enum Event { body: comp::Body, }, DeleteCharacter(CharacterId), + ClearCharacterListError, } enum Mode { @@ -165,9 +163,9 @@ enum Mode { } impl Mode { - pub fn select() -> Self { + pub fn select(info_content: Option) -> Self { Self::Select { - info_content: None, + info_content, selected: None, characters_scroll: Default::default(), character_buttons: Vec::new(), @@ -212,23 +210,9 @@ enum InfoContent { LoadingCharacters, CreatingCharacter, DeletingCharacter, - CharacterError, + CharacterError(String), } -/* -impl InfoContent { - pub fn has_content(&self, character_list_loading: &bool) -> bool { - match self { - Self::None => false, - Self::CreatingCharacter | Self::DeletingCharacter | Self::LoadingCharacters => { - *character_list_loading - }, - _ => true, - } - } -} -*/ - struct Controls { fonts: Fonts, imgs: Imgs, @@ -261,6 +245,7 @@ enum Message { RandomizeCharacter, CancelDeletion, ConfirmDeletion, + ClearCharacterListError, HairStyle(u8), HairColor(u8), Skin(u8), @@ -287,14 +272,13 @@ impl Controls { tooltip_manager: TooltipManager::new(TOOLTIP_HOVER_DUR, TOOLTIP_FADE_DUR), mouse_detector: Default::default(), - mode: Mode::select(), + mode: Mode::select(Some(InfoContent::LoadingCharacters)), } } - fn view(&mut self, settings: &Settings, client: &Client) -> Element { + fn view(&mut self, _settings: &Settings, client: &Client) -> Element { // TODO: use font scale thing for text size (use on button size for buttons with // text) - // TODO: if enter key pressed and character is selected then enter the world // Maintain tooltip manager self.tooltip_manager.maintain(); @@ -340,7 +324,7 @@ impl Controls { let content = match &mut self.mode { Mode::Select { - info_content, + ref mut info_content, selected, ref mut characters_scroll, ref mut character_buttons, @@ -350,6 +334,26 @@ impl Controls { ref mut yes_button, ref mut no_button, } => { + if let Some(error) = &client.character_list.error { + // TODO: use more user friendly errors with suggestions on potential solutions + // instead of directly showing error message here + *info_content = Some(InfoContent::CharacterError(format!( + "{}: {}", + i18n.get("common.error"), + error + ))) + } else if let Some(InfoContent::CharacterError(_)) = info_content { + *info_content = None; + } else if matches!( + info_content, + Some(InfoContent::LoadingCharacters) + | Some(InfoContent::CreatingCharacter) + | Some(InfoContent::DeletingCharacter) + ) && !client.character_list.loading + { + *info_content = None; + } + let server = Container::new( Column::with_children(vec![ Text::new(&client.server_info.name) @@ -493,9 +497,15 @@ impl Controls { Container::new( Scrollable::new(characters_scroll) .push(Column::with_children(characters).spacing(4)) - .width(Length::Fill), + .padding(5) + .scrollbar_width(5) + .scroller_width(5) + .width(Length::Fill) + .style(style::scrollable::Style { + track: None, + scroller: style::scrollable::Scroller::Color(UI_MAIN), + }), ) - .padding(5) .style(style::container::Style::color(Rgba::from_translucent( 0, BANNER_ALPHA, @@ -538,7 +548,7 @@ impl Controls { let enter_world = neat_button( enter_world_button, i18n.get("char_selection.enter_world"), - FILL_FRAC_ONE, + FILL_FRAC_TWO, button_style, selected.map(|_| Message::EnterWorld), ); @@ -563,35 +573,67 @@ impl Controls { // Overlay delete prompt if let Some(info_content) = info_content { - let over: Element<_> = match info_content { - InfoContent::Deletion(_) => Container::new( - Column::with_children(vec![ - Text::new(i18n.get("char_selection.delete_permanently")) - .size(fonts.cyri.scale(24)) - .into(), - Row::with_children(vec![ - neat_button( - no_button, - i18n.get("common.no"), - FILL_FRAC_ONE, - button_style, - Some(Message::CancelDeletion), - ), - neat_button( - yes_button, - i18n.get("common.yes"), - FILL_FRAC_ONE, - button_style, - Some(Message::ConfirmDeletion), - ), - ]) - .height(Length::Units(28)) - .spacing(30) + let over_content: Element<_> = match &info_content { + InfoContent::Deletion(_) => Column::with_children(vec![ + Text::new(i18n.get("char_selection.delete_permanently")) + .size(fonts.cyri.scale(24)) .into(), + Row::with_children(vec![ + neat_button( + no_button, + i18n.get("common.no"), + FILL_FRAC_ONE, + button_style, + Some(Message::CancelDeletion), + ), + neat_button( + yes_button, + i18n.get("common.yes"), + FILL_FRAC_ONE, + button_style, + Some(Message::ConfirmDeletion), + ), ]) - .align_items(Align::Center) - .spacing(10), - ) + .height(Length::Units(28)) + .spacing(30) + .into(), + ]) + .align_items(Align::Center) + .spacing(10) + .into(), + InfoContent::LoadingCharacters => { + Text::new(i18n.get("char_selection.loading_characters")) + .size(fonts.cyri.scale(24)) + .into() + }, + InfoContent::CreatingCharacter => { + Text::new(i18n.get("char_selection.creating_character")) + .size(fonts.cyri.scale(24)) + .into() + }, + InfoContent::DeletingCharacter => { + Text::new(i18n.get("char_selection.deleting_character")) + .size(fonts.cyri.scale(24)) + .into() + }, + InfoContent::CharacterError(error) => Column::with_children(vec![ + Text::new(error).size(fonts.cyri.scale(24)).into(), + Container::new(neat_button( + no_button, + i18n.get("common.close"), + FILL_FRAC_ONE, + button_style, + Some(Message::ClearCharacterListError), + )) + .height(Length::Units(28)) + .into(), + ]) + .align_items(Align::Center) + .spacing(10) + .into(), + }; + + let over = Container::new(over_content) .style( style::container::Style::color_with_double_cornerless_border( (0, 0, 0, 200).into(), @@ -605,11 +647,7 @@ impl Controls { .max_height(130) .padding(16) .center_x() - .center_y() - .into(), - // TODO - _ => Space::new(Length::Shrink, Length::Shrink).into(), - }; + .center_y(); Overlay::new(over, content) .width(Length::Fill) @@ -624,7 +662,7 @@ impl Controls { Mode::Create { name, body, - loadout, + loadout: _, tool, ref mut scroll, ref mut body_type_buttons, @@ -990,7 +1028,11 @@ impl Controls { ) .padding(5) .width(Length::Fill) - .align_items(Align::Center), + .align_items(Align::Center) + .style(style::scrollable::Style { + track: None, + scroller: style::scrollable::Scroller::Color(UI_MAIN), + }), ) .width(Length::Units(320)) // TODO: see if we can get iced to work with settings below //.max_width(360) @@ -1126,7 +1168,7 @@ impl Controls { match message { Message::Back => { if matches!(&self.mode, Mode::Create { .. }) { - self.mode = Mode::select(); + self.mode = Mode::select(None); } }, Message::Logout => { @@ -1169,7 +1211,7 @@ impl Controls { tool: String::from(*tool), body: comp::Body::Humanoid(*body), }); - self.mode = Mode::select(); + self.mode = Mode::select(Some(InfoContent::CreatingCharacter)); } }, Message::Name(value) => { @@ -1208,17 +1250,20 @@ impl Controls { if let Some(id) = characters.get(*idx).and_then(|i| i.character.id) { events.push(Event::DeleteCharacter(id)); } - *info_content = None; + *info_content = Some(InfoContent::DeletingCharacter); } } }, Message::CancelDeletion => { if let Mode::Select { info_content, .. } = &mut self.mode { - if let Some(InfoContent::Deletion(idx)) = info_content { + if let Some(InfoContent::Deletion(_)) = info_content { *info_content = None; } } }, + Message::ClearCharacterListError => { + events.push(Event::ClearCharacterListError); + }, Message::HairStyle(value) => { if let Mode::Create { body, .. } = &mut self.mode { body.hair_style = value; diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index 619839c731..88eb96481a 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -66,6 +66,10 @@ impl IcedUi { self.renderer.add_graphic(graphic) } + pub fn replace_graphic(&mut self, id: graphic::Id, graphic: Graphic) { + self.renderer.replace_graphic(id, graphic); + } + pub fn scale(&self) -> Scale { self.scale } pub fn set_scaling_mode(&mut self, mode: ScaleMode) { diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index 2a64e8e64a..284146e215 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -29,6 +29,8 @@ enum DrawKind { // Text and non-textured geometry Plain, } + +#[allow(dead_code)] // TODO: remove once WorldPos is used enum DrawCommand { Draw { kind: DrawKind, verts: Range }, Scissor(Aabr), @@ -149,6 +151,10 @@ impl IcedRenderer { self.cache.add_graphic(graphic) } + pub fn replace_graphic(&mut self, id: graphic::Id, graphic: Graphic) { + self.cache.replace_graphic(id, graphic); + } + fn image_dims(&self, handle: image::Handle) -> (u32, u32) { self .cache From 16a29342617d406e671ebcb253012c6720e9a596 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sat, 31 Oct 2020 20:43:22 -0400 Subject: [PATCH 57/61] Update changelog --- CHANGELOG.md | 1 + assets/voxygen/i18n/PL.ron | 2 +- assets/voxygen/i18n/en.ron | 2 +- assets/voxygen/i18n/es_ES.ron | 2 +- assets/voxygen/i18n/es_la.ron | 2 +- assets/voxygen/i18n/pt_BR.ron | 2 +- assets/voxygen/i18n/sv.ron | 2 +- assets/voxygen/i18n/zh_CN.ron | 2 +- assets/voxygen/i18n/zh_TW.ron | 2 +- 9 files changed, 9 insertions(+), 8 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index b406757b78..d8905696a9 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -67,6 +67,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Switched to procedural snow cover on trees - Significantly improved terrain generation performance - Significantly stabilized the game clock, to produce more "constant" TPS +- Transitioned main menu and character selection screen to a using iced for the ui (fixes paste keybinding on macos, removes password field limits, adds tabbing between input fields in the main menu, adds language selection in the main menu) ### Removed diff --git a/assets/voxygen/i18n/PL.ron b/assets/voxygen/i18n/PL.ron index 5069802cdc..545eae7ce1 100644 --- a/assets/voxygen/i18n/PL.ron +++ b/assets/voxygen/i18n/PL.ron @@ -1,5 +1,5 @@ /// Localization for Polish / Tłumaczenia dla języka polskiego -Localization( +( metadata: ( language_name: "Polish", language_identifier: "PL", diff --git a/assets/voxygen/i18n/en.ron b/assets/voxygen/i18n/en.ron index 8a298e3850..16cc9752a6 100644 --- a/assets/voxygen/i18n/en.ron +++ b/assets/voxygen/i18n/en.ron @@ -160,7 +160,7 @@ https://veloren.net/account/."#, "main.login.kicked": "You have been kicked with the following reason", "main.login.select_language": "Select a language", - "main.servers.select_server": "Select a server", + "main.servers.select_server": "Select a server", /// End Main screen section diff --git a/assets/voxygen/i18n/es_ES.ron b/assets/voxygen/i18n/es_ES.ron index 8b688abe05..a78e33a4b9 100644 --- a/assets/voxygen/i18n/es_ES.ron +++ b/assets/voxygen/i18n/es_ES.ron @@ -12,7 +12,7 @@ /// /// Localization for Spanish (Spain) -Localization( +( metadata: ( language_name: "Español de España", language_identifier: "es_ES", diff --git a/assets/voxygen/i18n/es_la.ron b/assets/voxygen/i18n/es_la.ron index b15f416386..241238e114 100644 --- a/assets/voxygen/i18n/es_la.ron +++ b/assets/voxygen/i18n/es_la.ron @@ -13,7 +13,7 @@ /// WARNING: Localization files shall be saved in UTF-8 format without BOM /// Localization for "latinoamericano" Latin-American -Localization( +( metadata: ( language_name: "Español Latino", language_identifier: "es_la", diff --git a/assets/voxygen/i18n/pt_BR.ron b/assets/voxygen/i18n/pt_BR.ron index 6918614aca..9c9226722b 100644 --- a/assets/voxygen/i18n/pt_BR.ron +++ b/assets/voxygen/i18n/pt_BR.ron @@ -1,5 +1,5 @@ /// Localization for Portuguese (Brazil) -Localization( +( metadata: ( language_name: "Português Brasileiro", language_identifier: "pt_BR", diff --git a/assets/voxygen/i18n/sv.ron b/assets/voxygen/i18n/sv.ron index f1198c0e93..5a914e423a 100644 --- a/assets/voxygen/i18n/sv.ron +++ b/assets/voxygen/i18n/sv.ron @@ -11,7 +11,7 @@ /// `assets/voxygen/i18n` and that's it! /// Localization for Swedish -Localization( +( metadata: ( language_name: "Svenska", language_identifier: "sv", diff --git a/assets/voxygen/i18n/zh_CN.ron b/assets/voxygen/i18n/zh_CN.ron index 40622d9e0c..9273541188 100644 --- a/assets/voxygen/i18n/zh_CN.ron +++ b/assets/voxygen/i18n/zh_CN.ron @@ -13,7 +13,7 @@ /// 注意: 本地化文件应以 UTF-8无BOM 格式保存 /// "全局"本地化 Simplified Chinese-简体中文 -Localization( +( metadata: ( language_name: "Simplified Chinese", language_identifier: "zh_CN", diff --git a/assets/voxygen/i18n/zh_TW.ron b/assets/voxygen/i18n/zh_TW.ron index 4d507c1a71..493e6bf423 100644 --- a/assets/voxygen/i18n/zh_TW.ron +++ b/assets/voxygen/i18n/zh_TW.ron @@ -1,5 +1,5 @@ /// Localization for Traditional Chinese -Localization( +( metadata: ( language_name: "繁體中文", language_identifier: "zh_TW", From af8fd6a88d3d2e973d208ff9f94a00d54e3eba07 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sun, 1 Nov 2020 01:51:09 -0400 Subject: [PATCH 58/61] Shink file size of some image assets --- assets/voxygen/element/buttons/button.png | Bin 4971 -> 1345 bytes assets/voxygen/element/buttons/x_red.png | Bin 9714 -> 236 bytes .../voxygen/element/buttons/x_red_hover.png | Bin 10485 -> 236 bytes .../voxygen/element/buttons/x_red_press.png | Bin 8668 -> 226 bytes assets/voxygen/element/frames/banner_top.png | Bin 5125 -> 411 bytes .../frames/loading_screen/loading_bg.png | Bin 2109 -> 116 bytes .../frames/loading_screen/loading_bg_l.png | Bin 7802 -> 122 bytes .../frames/loading_screen/loading_bg_r.png | Bin 7814 -> 118 bytes .../voxygen/element/frames/tooltip/corner.png | Bin 6484 -> 110 bytes .../voxygen/element/frames/tooltip/edge.png | Bin 5966 -> 96 bytes assets/voxygen/element/misc_bg/textbox.png | Bin 7799 -> 1014 bytes 11 files changed, 0 insertions(+), 0 deletions(-) diff --git a/assets/voxygen/element/buttons/button.png b/assets/voxygen/element/buttons/button.png index ec740a7d4bcb1cc8a7f1d50eba6ec9b2cab2c831..ff2fa731c4daea38286dd0bc535a96bace0d272f 100644 GIT binary patch delta 11 ScmaE@c93g=@@8K?IaUB1nFGN9 delta 3663 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 A=>Px# diff --git a/assets/voxygen/element/buttons/x_red_hover.png b/assets/voxygen/element/buttons/x_red_hover.png index 667b189086889604ec641e9ee9f203b752251824..088686d55952d32905eacd1411d009120e270f93 100644 GIT binary patch delta 209 zcmV;?051RaQS1SbBYyz1NklZl z{%K1+A>lj>u|vT-9^?4LmvNn@0s!S#V1$c=xIq~>19#CQm;nLc()wl&`$qV621+(7ZB@*GDf(&rA6;(@ z+r!e++$=0T&CSBnJ1q<>z0<5j?Matv}gI00000 LNkvXXu0mjfm}+7X literal 10485 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 diff --git a/assets/voxygen/element/buttons/x_red_press.png b/assets/voxygen/element/buttons/x_red_press.png index 7044cb22ba1168090948f1c7785f162ac2ffb5f0..1f3b9c458cd4f0e53c38c9be3973b86091e7098e 100644 GIT binary patch delta 199 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

$! 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 diff --git a/assets/voxygen/element/frames/loading_screen/loading_bg_l.png b/assets/voxygen/element/frames/loading_screen/loading_bg_l.png index 41124a5daa79daf33725fde0cb602b48cb34695f..6394ab6a67c5d71d895dc9f9b27b08c2775685fa 100644 GIT binary patch delta 9 QcmexmQ#C<(va(z_02N>Zj{pDw literal 7802 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 St5)(IO$(P#gfhd diff --git a/assets/voxygen/element/frames/tooltip/corner.png b/assets/voxygen/element/frames/tooltip/corner.png index fff6903304fc6d10e42ec353347cd25e3143dd04..7bb83552360fd84634377344c7328a82a5e095ce 100644 GIT binary patch delta 9 Qcmca&ls7?nGN+_B022ZOC;$Ke literal 6484 zcmeHLcT^ME8V`yhO)RLWtj1uWC7Bcg2}A{g5R{HISzwY(Ad+N4GLT?f7f=xpl~vb* z1=fmE1YJ-;Z~zSq%dC~gs@1g1j0fJ0Uv`= zT>Fpzl9=+W(fxmz){U9(FlueNESnN<`=;QW`_5n4yH4~M$8dmcYfco-1U9_uw|y|( zIK9Tpb0O(Mha|z6JPAp&O)bv7wPqzVRr>4Oew**718X(2jSFK^U&lqQTz;3lF1P!F z*=`S;-AM$>ULB~mrlV(x390Tvk-?!%yHxiYNzeJFYUk_j=S0;nCtc6Vb!}nZ9IZLr z_4^sKY8;Dgf&<-k+xmXCI-9F44Z*##Tol2&A<)Le-oJGqx@+RT-aC7luM*#$?A2I7 zxL>rQj2se>+&f)SV(@ltAD?q7sG@uN4OaD>P4_zmbx{g`2hGeUDW=YokQUo9$5scw zh#Y(9Zf?i*xs5QF{k$GMAoeX7Jhlj z{%yadRNgr4pp(!S`}E+{I~QXcyTz?q89F1Q%hT_KMLqZQYA&tZCfqYZn)ISd$D_q# z#OhHpi)6SGE!X7S@ZY!Ihr&!ztq{X}N$P22&#KqiI7xSfURw>ss zafVlUcV;it2%KTsdcna#Z^8uqtSQaZ&3wEUCvgh*L{U;O>3G=EOEqbb1>4(YCntZ4 zxBD#F#HRVRq$Te3``+7iW}7!G#)bmkr(E?HM;`O@sXa+e@X&QzB5d)#n2mHbdd*(e z+2T`QqUSMuD<>htWb{-)o7ZfPs9Mh_Je_vArXV=bHn1N#Tefpv;Jw;u;ny|SziWAAE*DW^cz1tm(#|)icQ&uk9v6J3={#+Yxo5pBN8S=W zay^tpq@+A}yzhYdSg9w;i{onMEqfT{ty2_FBZydq%T2R#3ws)M8x2D&ABBl@z}9sk zBPVKdFP@$}PDtY|`*EF!ziasGj7Qtq$5{c2uoY6m46XA}W2Pmoz5y;UC%Y z0oVGf8fG1wdNiOMsFw2v9C4r0V zni~t^-ud_VC%NK})UPdG%-av$4T_KL%-`f8+_C$r<#)~68T>IzGTP1K4v$W8i@;8+ z)@FBYt_fMP+|qnJ7;t*u$^GU0y=PW)7H3snK9;6`-MYdczk5`_S9$6cz3&n|Ba6ph ziR^34oXISFb!Db+Cw_z3w4G~O4K5asn0Kw=Y?7C`%b|7d$AYVw{448A^s-vEYww^I zd1h7CM%Ybh949b$zO}ot>de&G{d3appMBlaUK$fV9do8?Z)4xIfYzh96>-UP!yYqv z1_Ar!qwKUNCM7@A8r4A=**pJ?!Q<>_$c5LvVKX$O=l$2V-k(;8#Na%R85}KcH;Kuc zbKZRX(U2sI!e5FAKedG9^)()+z8I6mUcUv?oOW^=Y?U(#;gq&2sRuNiv0XGGVglAN}aw&?-zWEL3zOAhFio2a*CSf@=; zZ*uLaZHtMvsf}Y_k#d|^d}hr%^!c_ z{3R0XajC^M+w$g-Q>Trq^QP}57W=g#TL6jdLhP~;mvi^+-}R%e-5eXEtB;R$=VT?; zZ3P6r6{Czp@7=t1+;vWu=CNHdJIr2#{o4(Wc3jk$U;4OxH>GD&XZ=~@J(lyjitOh- zy=U99=f%CzW<8b`w>LZdaHlzDZOF5B(!dy9Po1h>s?UFV-7uC>(Glo()@049txsY?#`!RAV_yrzEIs)h?Vri~$e6!xUPt zSVBqvaTW%nksxq%^kh3azVG&Ek56B@ndwqK-?%c=$H{EMxtK9tRi3*o6DLMBnHy#} z9G(yq4V>!H-ASEkMo5i|Ja=wmWmbH0LC)x$IBcIrYg@93m7 zD=~Xr>@MnEo_!0u=hg}Ry4Mw;-4wqmfdZx(lWa1-jIwsn1q+ z+H4qK_D(POf#`<=qia^Ag+xYfHre;X)S@DBWc4)3so{EaC+C};N!xpHQ~mJn@E*57 zETXH~va{smi>Oza{YQ?r=}RP|QYM%#zj^Ow$?lT=<7jD%VMYP7Wwg zj#czBghynf11jnd3LugJzy=SH@nkx2s64vr<~ArTmJLJ^jVDn7N{A!^Kop5Sl8_-z zF(34OQbOj7UV@0;kPMENav`S}NQ{^aRVooi%ZBQVmO;uXmEA%fpNJZz(mYhgneFB| zD5DHxxIiRPNhr}nkv#4oP7*B@sxUk*5fVZo)DRiUO!^2fiGUGU76D5J#XfRsaQQDL zp^85(@;-7Zt5gwYuwZVqQazi+#Gx|;o`A=&rju#b)>IG=@u>hFutC3UC;$mh2dp@J z3YkNvg4`h#Y_SXh#a!q;g#e|Xacy{14xNjq+kiHBDuhySP&Pc3#--87bP}CHr3|6) zkP6UR0EPdPHOfpd>^<2`9GL*9$W=3*LJ;A@QV|okNFa_@d|2=mh#(&XR0fMg11NwM z*~$u_QfPDv?E}aUlFHC+uH+;E1hSf!$7L))89{Wr2}EExM3jibRU(5qVK~AA7JeUDxI+i=7P$Opf;SRT2!|S2QZwd#uC2-{tJ^&1S}W-GoBC7K^A){ zB8R1s9#Rj^QizLundejBK_+jsQOgi%EcP5jPYV9AF>mZ8^<|A@Y3fYIFmCwZ4D-S5F=lgbp zC6fqbtf{|1%!B1JvyTmsM?lJA`0R#MCwkzV_>5;VFb4XQU)%^#91fv};6MlblD}vN zfoN7V3f{^J;^V0R2;xB!WR2%@_%!9b6-ZLIr_cH8jCN>MePB&hE#Z?c;sL6%zy|!W zBP!3i_ldJpo&;`gjG?KuQ=SwIwz3`gL`x(>0VMr6j3JqdJes2ptly{;I(C7!=owrH=lt* z`ZrZzvEP(@ReZmN>sz?KDgs{x{*+fJtu0uoxWs=E|5E6;(OO!xZL`A7gCIdv0Oq>~3wJl50r=mg? zkb<^-YF#R};DSX!3*v?X6}3`>))i~r*Q#hI2_jmx^?R@X311*{&hPxra_>3!e)At` zXpoip5OW5DVI>askD&kSzMGqw(tq>d<{Ad0&uIO`Gx1W)amQyAJuLtQO4dQ3R5dP&h)u!_Oz9}(dsq# z#-7W>!;8ZUZ#Bm3{@6Ri>Z$21Z37YAu&n=2o(@kLw@XDw1ow}xw8>2Lw7aOaD&v`E z4qATL-SumS(10^L3R`Wdlir~Zj!fMaxN2}`SWu>LPzl%lmknPv?8&w&-1WV()~RvT z7p|*+oXtJuG`Yy6Vo3k7#{4zgTio%EOGzsra%>Ll=m3^@C0*aVpqa~0m|>G2Ha2?0 z;vyovQKpU^(API{#O^DmH@6JG`*=c^_og^0zj&biw0YXdsC9hR!Ny7;6nPNqbM#R@ z!v;|YaGc`y1lg_&?kLVP^B7|jB^69QYi=66a>AkufbA&<^l+Sd38!5)U2vN?uQg%X zRUh0VDKxI|m3Hp;&B|T0$aG5H?2^)WFkH592)eE6BVu1e*2Vz0$%R{-#EyzZ2O}m4 zRyIw)Y27xg^5P1T^;zs*|GvA-&slh`+}tMJ-Z-0geCFirc`=FYvdY33^*XX>=}G@( z+Kk_J75S*oN;p5Z3|NrIz3n_O<_oyK&EcCP+&ivgllpBQ**@^w*1bdUkS!h)re~jA zrHyFGDC<0DxnvFisTUiWm3){mG2Co-@r=?xntrxBT>%w>7sGqRdShN>xhZo%iD{^C$(n_%2Fy?bMKGdH*zgG zqc&>sn$|R~#Xzs@iojvF9<+|9j@kuUsg8Oc*pZWKvs0J@p01g4Ft={FbM2*VPj(J) zmW`|oJgb&{dZaP8>x0FMhT2YPh_nA)o4@vQ^mt@K!H)YA?OpE2g@hlPV0L<;agr{){B7H3SWt&VxtH?D{=@}b(MDva~cO+NMq3&w*FjL(={mBd~jZ@D_W z=rM6GY{TK$!&lRjcO|8Zwz5*et(T8qEWdbf_YbGH@2>r-91%J9jh&V7q;psD`L5ef znrdxpx_`T3ne#Xqjh7$Jy1jk1x!QD7fBx*uGhj z0WO2~rr3lZke0bFw4c*9&h9JU4TEBmp|Z!))B4ET<%)_!>5D6)%PUTZ{;-<@t%^VF8u3s8kRb1`&Ec=W$ z=8ugB<0Bwzs?Ou+e&?1-3-0Wtcjo`@kvDM9Gs~G*RG+M~Zkku{QBqR=s4qVmT2-Y^ zIy6GFtB>VGhYFYKf;H=xH}|=ba91)Npe#&Il+DSDFSvlQ0U9 zVF?Pgh9D5;3Pqq<|kzU;vlJWkFy-nj!^ePBaHR33Sadh{>e&KsPz9MFY}kVYdU_v!0%$??aJ}P~Mh5k62taC5+7cprGORkU$5jj^b=R+vS zM^QeA!CV+b_hfWZ_=|SLO)!I6B1mGiI{G$sOUK@HC)a_Ofn+K7Z3;dxpM^W?jY?$kEItv=tp1( zh56GT+BXD4AjAh@_b8YngatyFKOTaGkRh5LG)_d}xKhN_r48uf?IDxU6GDzqNN+k1 z5avcfkkE}IbmP53M+dHmPyai#ZXN)h-JLEB!BIpv5e$LL^-9CI@I?~X1}yt+;J+|M zCgQ2;f5!6y+QZ^QP^mbP6h?%}W@9MzcAnRPdzd2VK}}Lby7+IBdIjfc=&)eg7AMks z`H#fr7@iE}Q7ZIv69DuROo*VmDvFq>egnw$Ye?7Gpj5o3*Zy(4_OOt6Je7(Nm`?)T z`68x|?tlWM&&Le_b67CoJXN8V(0673+Jgj2_zC>@1^zBofJZ`ZX84=Ycxs)M!dY_&s*RPxL)@9 z4#NM|yf@5X?T2g9=}DSMg{7+B>i#uA14D=c#nj|mefNeK%F^pPK&Nbw(bo+6*23<& zwsdcJdP)AxpYBHbH(db0dnfNy-}mKuU#@qmz&nB8@2>addZ!A!6Zrk^`oERS{Kbn8 zOif?(Qt8(kk^Q+O{Yu1CHaW+Sw1CNA|I&MPtSOK2d{@p_Z=W9yNQL#K{0#n=Q-pGcz-tH%}itc<@u}B}4m3 a9T}y$CIN$kKT4ua7~+6X|8n2BtUm#q{+|y3 diff --git a/assets/voxygen/element/misc_bg/textbox.png b/assets/voxygen/element/misc_bg/textbox.png index 222091e16c2405c8d064521bb11c99b55105ba0c..528077c2a1c5d601f2f3ada7163ec7326efaa817 100644 GIT binary patch delta 11 Scmexv^NoFi^5!K{mze<~rUfei literal 7799 zcmeHLc|4SB`=3M#g-W7g3?<8~1~ZdM7;6SeQY2$G6K2LN3?)TVT2!bkWh=5JZHk0? zZ7tT2qL4_ov?xnQ-e)LB=k1*Hp3m?1{?~jy&y4%Nzt{KL?(4d5&tV5U%Y{Z`V)r;#FQScoG|BNFD0|T59MR3p)H)(wGiOn<_he6d&$wcUww62c zbi8ZL{*tGj4K?Nqb4&a~HinOkOfD?azE}_({|BFRcGX}>e11m{*N1U zlW#1(Zp%OG(&0LI^{#u;+gBsE5Ve)J>KqKYLa4~{led?rPH<&~O_l_&IYYqELtruU2nFDtYWNPcKG5XN@YP_L3gyYs@d&p{~Ez!{V zX_`tw?8~Y#!|Px|O{wecIeS%QCthy$2+Xj!h^IUuGH^52a1lHRh}B`YWQ=?A(roYhL& zy55C6tleF7Ape|3k4yzx>X;&Z-#+IaU8lpF60kkG=+#*D z!U{?{2W>v(3K}NMpLaJ-7{?YSzunqaO!5s(IBQ7G)#cfSMFv~aS6{O5bxw3?ijV z#VY&e{zr~bt;-S{qE(X>6;I?X_BkJ}?(Y5Yt;5xL_^p4v(dm`~`%!FZ_RF4}yRuw4 zUsmYt+$0Y=rQloI8Sk_ma*23@aqA4Z-10%*SsUmh^sP0!!TGVv{a0+O4OdyXL^H>O z*?Tli$UN|{E^3&%+e>G0Z2q+_wX#0QebcL(&%UhJZPQBTC|96HJX8;q)oKjN**4-M znk>pR*Cad*Nqp*xJ>V1GGkkKxz@>&kpS#{j>w;@->!Tc%OCvVR1Utl49PX~NRa0Lq zwA3!Rn{^<(HcQ8Q@rYzRG^xL6A+#|jV$bQJ7?ty?uwj!id3Ky4*5X{3l4bk;yE#|G z=(!(d(y@_BnMqBr`#N+USG|e}D$vHnKr@TnqRkH)h>h%XShAw*O9mvBpR131yxZjQ z?VL57P)~h)c~?bP@~~3wWxf#;m(hN2tn^W(<#2Cy4CC|Bi4f^@*2p^P=lkTD=Ue

r4PFx0;`Tc9F})<_saRq79}C`zSjQ0u^NYG zBb^h<84-m#i!dCwMatEo`->&tm!h@jjH9`x^9_|3*TuSbrfjtiyeVX!JgPWspuIAe<-k*mc695nA>K4Ph56k_GZ)W z`geXpcctvr7q$mM_Tp>sk*wIpHF(6nIyH@~!52gZ>(` z`uCwPvm3mY2YYCC_J+Bs}XM8%!xT zH}Zyz%KOSf#>)xx<5-@FLS~9x`JJt~H_}ZNHkNnaATWZPS2mS`r5+xo^;l4QGxXkD zJiw?y4h3!3l#JUXpvA&!zbfy*EI^svZTCmX~Vg5#ycRW{Ya_ZWpk%bw}UsEeg z_uJk`88pCme+j5mcY@?L;dbdf5@NrYQ_VfGCfk1PF z3^OwaYcsR&=P+>ko(qn|Ta|8DS-#UX&rxU90Ju}`3T~U6f?uQd(iGDSh3$Mq!LXUv z2iZJt+A(NP79(i z7c-Q*zg&muoV$g|?$-8hi!MQ&Yd#Z`Z^xCh*7cBwh~LlfoJ(~HZ^(-Dgb=?hke%H3 zO7X$!Xz#hlR({ObAnY2wH`x(zi$?N%PL$HzSEJbab@{x|!VMQ_t?iXJUts2l)!se* z0A_EU4mnfvRqT3xxS z>9QLR-nMxGl00t$)AudD*uo;U(1*2hT z7y@c8VECgT>!rX(910ERNU-=u0gUhvFCLGLgTn&?0$>4pFcyanM`E#9I06Mnp`ZW) z$_-@lNCGI6t1hCL<{(hHWDbMPW3ZTD5hux$#pmH65MUkrjT~4dn+t%+Zwf@?DS9rC z0=EVRXuux;07oGZ7$^b-MPcEy<$+ZqaYmZS{T4+ao^Sz)4M)Nda9`gaB)B|t|KIif zRD$aS+)v?-R4$9pAydu$sZ5^wY^7`;K6kcGK9?$*nzGx6LW2WFO=+GjV`)uvn2`~M zkI)dc1(=aP;Mra*9*gV6V$X>E;MC0Wf0zU) z{<+Ba$W2)_6=58KMdpjtTNCgQUIo>5)7Q z5D1i~p1~{%YbKXRVv?!fDHs3+5=}!QFcf_#35!8N(O3gAltiYGp=h!JMjwGS#Gp`= zSrqmh22cwmpZ}9JqD99~F#tET^ z(?jA=$Uo2lhBIh^e}@+J1F+GzO2^r<$Rtrmm^Pd;wK&}_Oi6V3G#36>;QwIS?8ORT z{&zgTLuXh_IJ^KB$J?G`@41Ug=KVF#&%iTGjzFX4@;HIkf8*32a7NP=W(CNyIDxbD zZ>IW9kEX-p!aHsSPk)B|w~~7a2dPwg0+YXIQu_8ZUsvp_I1xXswMo?MJNI-DzI<<7{Nv7-1o#8iB19{5jQ)vMxv7fXDw<6oB3?E4};42 zG3DP&;7@R~-Jf2jft{-xv}#rIdZeue8FMc^NSf9Ne7Q~9xGrHXLP@e}qzR}YvE zCWosk?2x?stoDLzD-`LXV`QSL%s9OMt%*F8lt&<4So>U7(HL zD>yHtBRl{8CvPUxlj2ifn_8ZJr;_w3(4nSjVF+lbs?pYAr)tSpXCoCE zFT2!+d{F``t4d9bZaK^D(c|Tq*Cj_EH2>s~QMkpa`pq-zb^JQJ1oTSby;N6xA>YyX kb6;vPOUnIdGH6ol Date: Sun, 1 Nov 2020 01:13:36 -0500 Subject: [PATCH 59/61] Clippy fixes --- voxygen/src/menu/char_selection/ui/mod.rs | 6 +++-- voxygen/src/menu/main/mod.rs | 3 ++- voxygen/src/menu/main/ui/connecting.rs | 2 +- voxygen/src/menu/main/ui/login.rs | 25 +++++++++---------- voxygen/src/menu/main/ui/mod.rs | 2 +- voxygen/src/ui/ice/mod.rs | 2 +- voxygen/src/ui/ice/renderer/mod.rs | 1 + voxygen/src/ui/ice/renderer/widget/slider.rs | 1 + .../src/ui/ice/renderer/widget/text_input.rs | 7 +++--- .../src/ui/ice/widget/background_container.rs | 4 +++ 10 files changed, 31 insertions(+), 22 deletions(-) diff --git a/voxygen/src/menu/char_selection/ui/mod.rs b/voxygen/src/menu/char_selection/ui/mod.rs index 3281c2ac53..a57a82b841 100644 --- a/voxygen/src/menu/char_selection/ui/mod.rs +++ b/voxygen/src/menu/char_selection/ui/mod.rs @@ -147,7 +147,7 @@ enum Mode { Create { name: String, // TODO: default to username body: humanoid::Body, - loadout: comp::Loadout, + loadout: Box, tool: &'static str, body_type_buttons: [button::State; 2], @@ -185,6 +185,8 @@ impl Mode { .active_item(Some(LoadoutBuilder::default_item_config_from_str(tool))) .build(); + let loadout = Box::new(loadout); + Self::Create { name, body: humanoid::Body::random(), @@ -1127,7 +1129,7 @@ impl Controls { }) .into() } else { - create.into() + create }; let bottom = Row::with_children(vec![ diff --git a/voxygen/src/menu/main/mod.rs b/voxygen/src/menu/main/mod.rs index b26213a3cb..d9ae300079 100644 --- a/voxygen/src/menu/main/mod.rs +++ b/voxygen/src/menu/main/mod.rs @@ -258,7 +258,8 @@ impl PlayState for MainMenuState { &global_state.settings.language.selected_language, )); localized_strings.log_missing_entries(); - self.main_menu_ui.update_language(localized_strings.clone()); + self.main_menu_ui + .update_language(std::sync::Arc::clone(&localized_strings)); }, #[cfg(feature = "singleplayer")] MainMenuEvent::StartSingleplayer => { diff --git a/voxygen/src/menu/main/ui/connecting.rs b/voxygen/src/menu/main/ui/connecting.rs index 0e770736a0..0636486e7a 100644 --- a/voxygen/src/menu/main/ui/connecting.rs +++ b/voxygen/src/menu/main/ui/connecting.rs @@ -139,7 +139,7 @@ impl Screen { let content = Column::with_children(vec![ text.into(), Container::new( - Row::with_children(vec![cancel.into(), add.into()]) + Row::with_children(vec![cancel, add]) .spacing(20) .height(Length::Units(25)), ) diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index d9ca8bc8e4..140f1aac7d 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -51,6 +51,7 @@ impl Screen { } } + #[allow(clippy::too_many_arguments)] pub(super) fn view( &mut self, fonts: &Fonts, @@ -155,20 +156,18 @@ impl Screen { .height(Length::Units(180)) .padding(20) .into() + } else if is_selecting_language { + self.language_selection.view( + fonts, + imgs, + i18n, + language_metadatas, + selected_language_index, + button_style, + ) } else { - if is_selecting_language { - self.language_selection.view( - fonts, - imgs, - i18n, - language_metadatas, - selected_language_index, - button_style, - ) - } else { - self.banner - .view(fonts, imgs, login_info, i18n, button_style) - } + self.banner + .view(fonts, imgs, login_info, i18n, button_style) }; let central_column = Container::new(central_content) diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 9ccbb7b7d7..f42407aec3 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -232,7 +232,7 @@ impl Controls { } fn view(&mut self, settings: &Settings, dt: f32) -> Element { - self.time = self.time + dt as f64; + self.time += dt as f64; // TODO: consider setting this as the default in the renderer let button_style = style::button::Style::new(self.imgs.button) diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs index 88eb96481a..7ea39b11ec 100644 --- a/voxygen/src/ui/ice/mod.rs +++ b/voxygen/src/ui/ice/mod.rs @@ -187,7 +187,7 @@ impl IcedUi { &self.events, cursor_position, Some(&self.clipboard), - &mut self.renderer, + &self.renderer, ); drop(guard); // Clear events diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index 284146e215..10bb9eab18 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -799,6 +799,7 @@ impl iced::Renderer for IcedRenderer { // TODO: use graph of primitives to enable diffing??? type Output = (Primitive, iced::mouse::Interaction); + #[allow(clippy::let_and_return)] fn layout<'a, M>( &mut self, element: &iced::Element<'a, M, Self>, diff --git a/voxygen/src/ui/ice/renderer/widget/slider.rs b/voxygen/src/ui/ice/renderer/widget/slider.rs index bdb8321f9b..63dbbaa3fc 100644 --- a/voxygen/src/ui/ice/renderer/widget/slider.rs +++ b/voxygen/src/ui/ice/renderer/widget/slider.rs @@ -77,6 +77,7 @@ impl slider::Renderer for IcedRenderer { mouse::Interaction::Idle }; + #[allow(clippy::if_same_then_else)] // TODO: remove let primitives = if style.labels { // TODO text label on left and right ends vec![bar, cursor] diff --git a/voxygen/src/ui/ice/renderer/widget/text_input.rs b/voxygen/src/ui/ice/renderer/widget/text_input.rs index ec833214b3..82a86c059a 100644 --- a/voxygen/src/ui/ice/renderer/widget/text_input.rs +++ b/voxygen/src/ui/ice/renderer/widget/text_input.rs @@ -31,9 +31,10 @@ impl text_input::Renderer for IcedRenderer { }; let mut glyph_calculator = self.cache.glyph_calculator(); - let width = glyph_calculator + /* let width = */ + glyph_calculator .glyph_bounds(section) - .map_or(0.0, |rect| rect.width() / p_scale); + .map_or(0.0, |rect| rect.width() / p_scale) // glyph_brush ignores the exterior spaces // or does it!!! @@ -75,7 +76,7 @@ impl text_input::Renderer for IcedRenderer { width += exterior_spaces as f32 * space_width; }*/ - width + //width } fn offset( diff --git a/voxygen/src/ui/ice/widget/background_container.rs b/voxygen/src/ui/ice/widget/background_container.rs index 7aad366bf4..4ee21ae042 100644 --- a/voxygen/src/ui/ice/widget/background_container.rs +++ b/voxygen/src/ui/ice/widget/background_container.rs @@ -63,6 +63,10 @@ impl Padding { } } +impl Default for Padding { + fn default() -> Self { Self::new() } +} + pub trait Background: Sized { // The intended implementors already store the state accessed in the three // functions below From ed821fd80ff9b36228dd9c82ab70e1593a2be417 Mon Sep 17 00:00:00 2001 From: Imbris Date: Sun, 1 Nov 2020 16:42:07 -0500 Subject: [PATCH 60/61] Remove uneeded commented code, add notes to remaining commented code, use source rect calculations from conrod backend code --- voxygen/src/menu/main/mod.rs | 1 + voxygen/src/menu/main/ui/login.rs | 3 - voxygen/src/menu/main/ui/mod.rs | 22 +-- voxygen/src/ui/graphic/pixel_art.rs | 6 +- voxygen/src/ui/ice/renderer/mod.rs | 147 ++++++++++-------- voxygen/src/ui/ice/renderer/primitive.rs | 5 +- voxygen/src/ui/ice/renderer/style/button.rs | 2 - .../renderer/widget/aspect_ratio_container.rs | 3 - voxygen/src/ui/ice/renderer/widget/button.rs | 1 + .../ice/renderer/widget/compound_graphic.rs | 8 +- .../src/ui/ice/renderer/widget/container.rs | 9 ++ voxygen/src/ui/ice/renderer/widget/image.rs | 1 + .../src/ui/ice/renderer/widget/scrollable.rs | 15 +- voxygen/src/ui/ice/renderer/widget/slider.rs | 2 + voxygen/src/ui/ice/renderer/widget/text.rs | 4 - .../src/ui/ice/renderer/widget/text_input.rs | 26 +--- .../ui/ice/widget/aspect_ratio_container.rs | 3 - .../src/ui/ice/widget/background_container.rs | 26 ---- voxygen/src/ui/ice/widget/compound_graphic.rs | 36 ++--- voxygen/src/ui/ice/widget/mouse_detector.rs | 18 +-- voxygen/src/ui/ice/widget/overlay.rs | 1 - voxygen/src/ui/ice/widget/stack.rs | 11 -- voxygen/src/ui/ice/widget/tooltip.rs | 4 - 23 files changed, 135 insertions(+), 219 deletions(-) diff --git a/voxygen/src/menu/main/mod.rs b/voxygen/src/menu/main/mod.rs index d9ae300079..f3b57d1acc 100644 --- a/voxygen/src/menu/main/mod.rs +++ b/voxygen/src/menu/main/mod.rs @@ -268,6 +268,7 @@ impl PlayState for MainMenuState { global_state.singleplayer = Some(singleplayer); }, MainMenuEvent::Quit => return PlayStateResult::Shutdown, + // Note: Keeping in case we re-add the disclaimer /*MainMenuEvent::DisclaimerAccepted => { global_state.settings.show_disclaimer = false },*/ diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index 140f1aac7d..c40dfab8fa 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -398,7 +398,6 @@ impl LoginBanner { .into(), ]) .spacing(5) - //.height(Length::Units(200)) .into(), Space::new(Length::Fill, Length::Units(8)).into(), Column::with_children(vec![ @@ -430,7 +429,5 @@ impl LoginBanner { .height(Length::Fill) .center_y() .into() - //.padding(Padding::new().horizontal(8).vertical(15)) - //.max_width(350); } } diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index f42407aec3..075f6f8ded 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -1,4 +1,5 @@ mod connecting; +// Note: Keeping in case we re-add the disclaimer //mod disclaimer; mod login; mod servers; @@ -26,10 +27,6 @@ use std::time::Duration; // TODO: what is this? (showed up in rebase) //const COL1: Color = Color::Rgba(0.07, 0.1, 0.1, 0.9); -// UI Color-Theme -/*const UI_MAIN: Color = Color::Rgba(0.61, 0.70, 0.70, 1.0); // Greenish Blue -const UI_HIGHLIGHT_0: Color = Color::Rgba(0.79, 1.09, 1.09, 1.0);*/ - pub const TEXT_COLOR: iced::Color = iced::Color::from_rgb(1.0, 1.0, 1.0); pub const DISABLED_TEXT_COLOR: iced::Color = iced::Color::from_rgba(1.0, 1.0, 1.0, 0.2); @@ -95,7 +92,7 @@ pub enum Event { #[cfg(feature = "singleplayer")] StartSingleplayer, Quit, - //DisclaimerClosed, TODO: remove all traces? + // Note: Keeping in case we re-add the disclaimer //DisclaimerAccepted, AuthServerTrust(String, bool), } @@ -112,6 +109,7 @@ enum ConnectionState { } enum Screen { + // Note: Keeping in case we re-add the disclaimer /*Disclaimer { screen: disclaimer::Screen, },*/ @@ -169,7 +167,7 @@ enum Message { TrustPromptAdd, TrustPromptCancel, CloseError, - /*CloseDisclaimer, + /* Note: Keeping in case we re-add the disclaimer *AcceptDisclaimer, */ } @@ -184,6 +182,7 @@ impl Controls { let version = common::util::DISPLAY_VERSION_LONG.clone(); let alpha = format!("Veloren {}", common::util::DISPLAY_VERSION.as_str()); + // Note: Keeping in case we re-add the disclaimer let screen = /* if settings.show_disclaimer { Screen::Disclaimer { screen: disclaimer::Screen::new(), @@ -274,6 +273,7 @@ impl Controls { // TODO: make any large text blocks scrollable so that if the area is to // small they can still be read let content = match &mut self.screen { + // Note: Keeping in case we re-add the disclaimer //Screen::Disclaimer { screen } => screen.view(&self.fonts, &self.i18n, button_style), Screen::Login { screen, error } => screen.view( &self.fonts, @@ -389,11 +389,7 @@ impl Controls { connection_state, .. } = &mut self.screen { - if let ConnectionState::AuthTrustPrompt { - auth_server, - .. - } = connection_state - { + if let ConnectionState::AuthTrustPrompt { auth_server, .. } = connection_state { let auth_server = std::mem::take(auth_server); let added = matches!(msg, Message::TrustPromptAdd); @@ -407,9 +403,7 @@ impl Controls { *error = None; } }, - //Message::CloseDisclaimer => { - // events.push(Event::DisclaimerClosed); - //}, + /* Note: Keeping in case we re-add the disclaimer */ /*Message::AcceptDisclaimer => { if let Screen::Disclaimer { .. } = &self.screen { events.push(Event::DisclaimerAccepted); diff --git a/voxygen/src/ui/graphic/pixel_art.rs b/voxygen/src/ui/graphic/pixel_art.rs index a269cd6e53..104caafe14 100644 --- a/voxygen/src/ui/graphic/pixel_art.rs +++ b/voxygen/src/ui/graphic/pixel_art.rs @@ -1,4 +1,7 @@ -use common::util::{linear_to_srgba, srgba_to_linear}; +use common::{ + span, + util::{linear_to_srgba, srgba_to_linear}, +}; /// Pixel art scaling /// Note: The current ui is locked to the pixel grid with little animation, if /// we want smoothly moving pixel art this should be done in the shaders @@ -35,6 +38,7 @@ const EPSILON: f32 = 0.0001; // E9: c3 = (A1 * c1 * a1 + A2 * c2 * a2) / a3 #[allow(clippy::manual_saturating_arithmetic)] // TODO: Pending review in #587 pub fn resize_pixel_art(image: &RgbaImage, new_width: u32, new_height: u32) -> RgbaImage { + span!(_guard, "resize_pixel_art"); let (width, height) = image.dimensions(); let mut new_image = RgbaImage::new(new_width, new_height); diff --git a/voxygen/src/ui/ice/renderer/mod.rs b/voxygen/src/ui/ice/renderer/mod.rs index 10bb9eab18..03721eff98 100644 --- a/voxygen/src/ui/ice/renderer/mod.rs +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -113,7 +113,6 @@ pub struct IcedRenderer { start: usize, // Draw commands for the next render draw_commands: Vec, - //current_scissor: Aabr, } impl IcedRenderer { pub fn new( @@ -141,7 +140,6 @@ impl IcedRenderer { win_dims: scaled_dims, window_scissor: default_scissor(renderer), start: 0, - //current_scissor: default_scissor(renderer), }) } @@ -186,8 +184,6 @@ impl IcedRenderer { self.current_state = State::Plain; self.start = 0; - //self.current_scissor = default_scissor(renderer); - self.draw_primitive(primitive, Vec2::zero(), 1.0, renderer); // Enter the final command. @@ -309,20 +305,6 @@ impl IcedRenderer { } // Update model with new mesh. renderer.update_model(&self.model, &self.mesh, 0).unwrap(); - - // Handle window resizing. - /*if let Some(new_dims) = self.window_resized.take() { - let (old_w, old_h) = self.scale.scaled_window_size().into_tuple(); - self.scale.window_resized(new_dims, renderer); - let (w, h) = self.scale.scaled_window_size().into_tuple(); - self.ui.handle_event(Input::Resize(w, h)); - - // Avoid panic in graphic cache when minimizing. - // Avoid resetting cache if window size didn't change - // Somewhat inefficient for elements that won't change size after a window resize - let res = renderer.get_resolution(); - self.need_cache_resize = res.x > 0 && res.y > 0 && !(old_w == w && old_h == h); - }*/ } // Returns (half_res, align) @@ -346,19 +328,6 @@ impl IcedRenderer { } fn gl_aabr(&self, bounds: iced::Rectangle) -> Aabr { - /*let (ui_win_w, ui_win_h) = self.win_dims.into_tuple(); - let (l, b) = aabr.min.into_tuple(); - let (r, t) = aabr.max.into_tuple(); - let vx = |x: f64| (x / ui_win_w * 2.0) as f32; - let vy = |y: f64| (y / ui_win_h * 2.0) as f32; - let min = Vec2::new( - ((vx(l) * half_res.x + x_align).round() - x_align) / half_res.x, - ((vy(b) * half_res.y + y_align).round() - y_align) / half_res.y, - ); - let max = Vec2::new( - ((vx(r) * half_res.x + x_align).round() - x_align) / half_res.x, - ((vy(t) * half_res.y + y_align).round() - y_align) / half_res.y, - );*/ let flipped_y = self.win_dims.y - bounds.y; let half_win_dims = self.win_dims.map(|e| e / 2.0); let half_res = self.half_res; @@ -458,6 +427,7 @@ impl IcedRenderer { handle, bounds, color, + source_rect, } => { let color = srgba_to_linear(color.map(|e| e as f32 / 255.0)); let color = apply_alpha(color, alpha); @@ -473,9 +443,78 @@ impl IcedRenderer { ..bounds }); + let graphic_cache = self.cache.graphic_cache_mut(); + let half_res = self.half_res; // Make borrow checker happy by avoiding self in closure + let (source_aabr, gl_size) = { + // Transform the source rectangle into uv coordinate. + // TODO: Make sure this is right. Especially the conversions. + let ((uv_l, uv_r, uv_b, uv_t), gl_size) = match graphic_cache + .get_graphic(graphic_id) + { + Some(Graphic::Blank) | None => return, + Some(Graphic::Image(image, ..)) => { + source_rect.and_then(|src_rect| { + #[rustfmt::skip] use ::image::GenericImageView; + let (image_w, image_h) = image.dimensions(); + let (source_w, source_h) = src_rect.size().into_tuple(); + let gl_size = gl_aabr.size(); + if image_w == 0 + || image_h == 0 + || source_w < 1.0 + || source_h < 1.0 + || gl_size.reduce_partial_min() < f32::EPSILON + { + None + } else { + // TODO: do this earlier + // Multiply drawn image size by ratio of original image + // size to + // source rectangle size (since as the proportion of the + // image gets + // smaller, the drawn size should get bigger), up to the + // actual + // size of the original image. + let ratio_x = (image_w as f32 / source_w) + .min((image_w as f32 / (gl_size.w * half_res.x)).max(1.0)); + let ratio_y = (image_h as f32 / source_h) + .min((image_h as f32 / (gl_size.h * half_res.y)).max(1.0)); + let (l, b) = src_rect.min.into_tuple(); + let (r, t) = src_rect.max.into_tuple(); + Some(( + ( + l / image_w as f32, /* * ratio_x*/ + r / image_w as f32, /* * ratio_x*/ + b / image_h as f32, /* * ratio_y*/ + t / image_h as f32, /* * ratio_y*/ + ), + Extent2::new( + gl_size.w as f32 * ratio_x, + gl_size.h as f32 * ratio_y, + ), + )) + /* ((l / image_w as f32), + (r / image_w as f32), + (b / image_h as f32), + (t / image_h as f32)) */ + } + }) + }, + // No easy way to interpret source_rect for voxels... + Some(Graphic::Voxel(..)) => None, + } + .unwrap_or_else(|| ((0.0, 1.0, 0.0, 1.0), gl_aabr.size())); + ( + Aabr { + min: Vec2::new(uv_l, uv_b), + max: Vec2::new(uv_r, uv_t), + }, + gl_size, + ) + }; + let resolution = Vec2::new( - (gl_aabr.size().w * self.half_res.x).round() as u16, - (gl_aabr.size().h * self.half_res.y).round() as u16, + (gl_size.w * self.half_res.x).round() as u16, + (gl_size.h * self.half_res.y).round() as u16, ); // Don't do anything if resolution is zero @@ -484,39 +523,13 @@ impl IcedRenderer { // TODO: consider logging uneeded elements } - let graphic_cache = self.cache.graphic_cache_mut(); - - match graphic_cache.get_graphic(graphic_id) { - Some(Graphic::Blank) | None => return, - _ => {}, - } - - // Transform the source rectangle into uv coordinate. - // TODO: Make sure this is right. - let source_aabr = { - let (uv_l, uv_r, uv_b, uv_t) = (0.0, 1.0, 0.0, 1.0); - /*match source_rect { - Some(src_rect) => { - let (l, r, b, t) = src_rect.l_r_b_t(); - ((l / image_w) as f32, - (r / image_w) as f32, - (b / image_h) as f32, - (t / image_h) as f32) - } - None => (0.0, 1.0, 0.0, 1.0), - };*/ - Aabr { - min: Vec2::new(uv_l, uv_b), - max: Vec2::new(uv_r, uv_t), - } - }; - // Cache graphic at particular resolution. let (uv_aabr, tex_id) = match graphic_cache.cache_res( renderer, graphic_id, resolution, - source_aabr, + // TODO: take f32 here + source_aabr.map(|e| e as f64), rotation, ) { // TODO: get dims from graphic_cache (or have it return floats directly) @@ -601,11 +614,8 @@ impl IcedRenderer { }, Primitive::Text { glyphs, - bounds: _bounds, // iced::Rectangle + bounds: _, // iced::Rectangle linear_color, - /*font, - *horizontal_alignment, - *vertical_alignment, */ } => { let linear_color = apply_alpha(linear_color, alpha); self.switch_state(State::Plain); @@ -627,6 +637,7 @@ impl IcedRenderer { // Note: we can't actually use this because dropping glyphs messeses up the // counting and there is not a method provided to drop out of bounds // glyphs while positioning them + // Note: keeping commented code in case how we handle text changes glyph_brush::ab_glyph::Rect { min: glyph_brush::ab_glyph::point( -10000.0, //bounds.x * self.p_scale, @@ -688,11 +699,11 @@ impl IcedRenderer { Aabr::new_empty(Vec2::zero()) } }; - // Not expecting this case: new_cursor == current_scissor + // Not expecting this case: new_scissor == current_scissor + // So not optimizing for it // Finish the current command. - // TODO: ensure we never push empty commands (make fields private & debug assert - // in constructors?) + // TODO: ensure we never push empty commands self.draw_commands.push(match self.current_state { State::Plain => DrawCommand::plain(self.start..self.mesh.vertices().len()), State::Image(id) => { diff --git a/voxygen/src/ui/ice/renderer/primitive.rs b/voxygen/src/ui/ice/renderer/primitive.rs index 8230287130..a0ceec22c7 100644 --- a/voxygen/src/ui/ice/renderer/primitive.rs +++ b/voxygen/src/ui/ice/renderer/primitive.rs @@ -10,6 +10,7 @@ pub enum Primitive { handle: (image::Handle, graphic::Rotation), bounds: iced::Rectangle, color: vek::Rgba, + source_rect: Option>, }, // A vertical gradient // TODO: could be combined with rectangle @@ -24,12 +25,8 @@ pub enum Primitive { }, Text { glyphs: Vec, - //size: f32, bounds: iced::Rectangle, linear_color: vek::Rgba, - /*font: iced::Font, - *horizontal_alignment: iced::HorizontalAlignment, - *vertical_alignment: iced::VerticalAlignment, */ }, Clip { bounds: iced::Rectangle, diff --git a/voxygen/src/ui/ice/renderer/style/button.rs b/voxygen/src/ui/ice/renderer/style/button.rs index beea30781b..32e8ab0074 100644 --- a/voxygen/src/ui/ice/renderer/style/button.rs +++ b/voxygen/src/ui/ice/renderer/style/button.rs @@ -27,8 +27,6 @@ pub struct Style { background: Option, enabled_text: Color, disabled_text: Color, - /* greying out / changing text color - *disabled: , */ } impl Style { diff --git a/voxygen/src/ui/ice/renderer/widget/aspect_ratio_container.rs b/voxygen/src/ui/ice/renderer/widget/aspect_ratio_container.rs index 3a07c909df..8ae4ab2a36 100644 --- a/voxygen/src/ui/ice/renderer/widget/aspect_ratio_container.rs +++ b/voxygen/src/ui/ice/renderer/widget/aspect_ratio_container.rs @@ -5,7 +5,6 @@ use super::super::{ use iced::{Element, Layout, Point, Rectangle}; impl aspect_ratio_container::Renderer for IcedRenderer { - //type Style type ImageHandle = image::Handle; fn dimensions(&self, handle: &Self::ImageHandle) -> (u32, u32) { self.image_dims(*handle) } @@ -16,11 +15,9 @@ impl aspect_ratio_container::Renderer for IcedRenderer { _bounds: Rectangle, cursor_position: Point, viewport: &Rectangle, - //style: &Self::Style, content: &Element<'_, M, Self>, content_layout: Layout<'_>, ) -> Self::Output { - // TODO: stlying to add a background image and such content.draw(self, defaults, content_layout, cursor_position, viewport) } } diff --git a/voxygen/src/ui/ice/renderer/widget/button.rs b/voxygen/src/ui/ice/renderer/widget/button.rs index a0e6399297..cf05a448bf 100644 --- a/voxygen/src/ui/ice/renderer/widget/button.rs +++ b/voxygen/src/ui/ice/renderer/widget/button.rs @@ -45,6 +45,7 @@ impl button::Renderer for IcedRenderer { handle: (handle, Rotation::None), bounds, color, + source_rect: None, }; Primitive::Group { diff --git a/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs b/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs index 797fd916d7..f8616d4e8a 100644 --- a/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs +++ b/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs @@ -7,12 +7,7 @@ use compound_graphic::GraphicKind; use iced::{mouse, Rectangle}; impl compound_graphic::Renderer for IcedRenderer { - fn draw( - &mut self, - graphics: I, - //color: Rgba, - _layout: iced::Layout<'_>, - ) -> Self::Output + fn draw(&mut self, graphics: I) -> Self::Output where I: Iterator, { @@ -24,6 +19,7 @@ impl compound_graphic::Renderer for IcedRenderer { handle: (handle, Rotation::None), bounds, color, + source_rect: None, }, GraphicKind::Color(color) => Primitive::Rectangle { bounds, diff --git a/voxygen/src/ui/ice/renderer/widget/container.rs b/voxygen/src/ui/ice/renderer/widget/container.rs index 9d8398421c..246cf678ef 100644 --- a/voxygen/src/ui/ice/renderer/widget/container.rs +++ b/voxygen/src/ui/ice/renderer/widget/container.rs @@ -29,6 +29,7 @@ impl container::Renderer for IcedRenderer { handle: (*handle, Rotation::None), bounds, color: *color, + source_rect: None, }; Primitive::Group { @@ -177,6 +178,7 @@ impl container::Renderer for IcedRenderer { height: border_size, }, color, + source_rect: None, }; let tr_corner = Primitive::Image { @@ -188,6 +190,7 @@ impl container::Renderer for IcedRenderer { height: border_size, }, color, + source_rect: None, }; let bl_corner = Primitive::Image { @@ -199,6 +202,7 @@ impl container::Renderer for IcedRenderer { height: border_size, }, color, + source_rect: None, }; let br_corner = Primitive::Image { @@ -210,6 +214,7 @@ impl container::Renderer for IcedRenderer { height: border_size, }, color, + source_rect: None, }; let top_edge = Primitive::Image { @@ -221,6 +226,7 @@ impl container::Renderer for IcedRenderer { height: border_size, }, color, + source_rect: None, }; let bottom_edge = Primitive::Image { @@ -232,6 +238,7 @@ impl container::Renderer for IcedRenderer { height: border_size, }, color, + source_rect: None, }; let left_edge = Primitive::Image { @@ -243,6 +250,7 @@ impl container::Renderer for IcedRenderer { height: bounds.height - 2.0 * border_size, }, color, + source_rect: None, }; let right_edge = Primitive::Image { @@ -254,6 +262,7 @@ impl container::Renderer for IcedRenderer { height: bounds.height - 2.0 * border_size, }, color, + source_rect: None, }; // Is this worth it as opposed to using a giant image? (Probably) diff --git a/voxygen/src/ui/ice/renderer/widget/image.rs b/voxygen/src/ui/ice/renderer/widget/image.rs index 8f7c9deaf2..245d01167b 100644 --- a/voxygen/src/ui/ice/renderer/widget/image.rs +++ b/voxygen/src/ui/ice/renderer/widget/image.rs @@ -19,6 +19,7 @@ impl image::Renderer for IcedRenderer { handle: (handle, Rotation::None), bounds: layout.bounds(), color, + source_rect: None, }, mouse::Interaction::default(), ) diff --git a/voxygen/src/ui/ice/renderer/widget/scrollable.rs b/voxygen/src/ui/ice/renderer/widget/scrollable.rs index edf7c1a4a7..3d138d38a2 100644 --- a/voxygen/src/ui/ice/renderer/widget/scrollable.rs +++ b/voxygen/src/ui/ice/renderer/widget/scrollable.rs @@ -22,7 +22,7 @@ impl scrollable::Renderer for IcedRenderer { ) -> Option { if content_bounds.height > bounds.height { // Area containing both scrollbar and scroller - let outer_width = (scrollbar_width.max(scroller_width) + 2 * scrollbar_margin) as f32 /* * self.p_scale */; + let outer_width = (scrollbar_width.max(scroller_width) + 2 * scrollbar_margin) as f32; let outer_bounds = Rectangle { x: bounds.x + bounds.width - outer_width, width: outer_width, @@ -38,14 +38,14 @@ impl scrollable::Renderer for IcedRenderer { // Interactive scroller let visible_fraction = bounds.height / content_bounds.height; - let scroller_height = (bounds.height * visible_fraction) - .max((2 * SCROLLBAR_MIN_HEIGHT) as f32/* * self.p_scale*/); + let scroller_height = + (bounds.height * visible_fraction).max((2 * SCROLLBAR_MIN_HEIGHT) as f32); let y_offset = offset as f32 * visible_fraction; let scroller_bounds = Rectangle { - x: bounds.x + bounds.width - outer_width / 2.0 - (scrollbar_width / 2) as f32, /* * self.p_scale*/ + x: bounds.x + bounds.width - outer_width / 2.0 - (scrollbar_width / 2) as f32, y: scrollbar_bounds.y + y_offset, - width: scroller_width as f32, /* * self.p_scale*/ + width: scroller_width as f32, height: scroller_height, }; Some(scrollable::Scrollbar { @@ -85,6 +85,7 @@ impl scrollable::Renderer for IcedRenderer { }); let style = style_sheet; + // Note: for future use if we vary style with the state of the scrollable //let style = if state.is_scroller_grabbed() { // style_sheet.dragging() //} else if is_mouse_over_scrollbar { @@ -127,6 +128,7 @@ impl scrollable::Renderer for IcedRenderer { ..bounds }, color, + source_rect: None, }); // Middle primitives.push(Primitive::Image { @@ -137,6 +139,7 @@ impl scrollable::Renderer for IcedRenderer { ..bounds }, color, + source_rect: None, }); // Bottom primitives.push(Primitive::Image { @@ -147,6 +150,7 @@ impl scrollable::Renderer for IcedRenderer { ..bounds }, color, + source_rect: None, }); }, } @@ -162,6 +166,7 @@ impl scrollable::Renderer for IcedRenderer { handle: (handle, Rotation::None), bounds: scrollbar.bounds, color, + source_rect: None, }, }); } diff --git a/voxygen/src/ui/ice/renderer/widget/slider.rs b/voxygen/src/ui/ice/renderer/widget/slider.rs index 63dbbaa3fc..ec624105bf 100644 --- a/voxygen/src/ui/ice/renderer/widget/slider.rs +++ b/voxygen/src/ui/ice/renderer/widget/slider.rs @@ -40,6 +40,7 @@ impl slider::Renderer for IcedRenderer { ..bar_bounds }, color, + source_rect: None, }, }; @@ -64,6 +65,7 @@ impl slider::Renderer for IcedRenderer { handle: (handle, Rotation::None), bounds: cursor_bounds, color, + source_rect: None, }, }; diff --git a/voxygen/src/ui/ice/renderer/widget/text.rs b/voxygen/src/ui/ice/renderer/widget/text.rs index 99af86aa5e..79a3716547 100644 --- a/voxygen/src/ui/ice/renderer/widget/text.rs +++ b/voxygen/src/ui/ice/renderer/widget/text.rs @@ -54,12 +54,8 @@ impl text::Renderer for IcedRenderer { ( Primitive::Text { glyphs, - //size: size as f32, bounds, linear_color: color.unwrap_or(defaults.text_color).into_linear().into(), - /*font, - *horizontal_alignment, - *vertical_alignment, */ }, mouse::Interaction::default(), ) diff --git a/voxygen/src/ui/ice/renderer/widget/text_input.rs b/voxygen/src/ui/ice/renderer/widget/text_input.rs index 82a86c059a..12ca612e17 100644 --- a/voxygen/src/ui/ice/renderer/widget/text_input.rs +++ b/voxygen/src/ui/ice/renderer/widget/text_input.rs @@ -31,6 +31,7 @@ impl text_input::Renderer for IcedRenderer { }; let mut glyph_calculator = self.cache.glyph_calculator(); + // Note: keeping comments below for now in case this needs to be debugged again /* let width = */ glyph_calculator .glyph_bounds(section) @@ -126,6 +127,8 @@ impl text_input::Renderer for IcedRenderer { ) -> Self::Output { let is_mouse_over = bounds.contains(cursor_position); + // Note: will be useful in the future if we vary the style with the state of the + // text input /* let style = if state.is_focused() { style.focused() @@ -224,34 +227,11 @@ impl text_input::Renderer for IcedRenderer { size, font, ); - // TODO: delete if new arrangment behaves nicely - /*let section = glyph_brush::Section { - screen_position: (text_bounds.x * p_scale, text_bounds.center_y() * p_scale), - bounds: ( - 10000.0, /* text_bounds.width * p_scale */ - text_bounds.height * p_scale, - ), - layout: glyph_brush::Layout::SingleLine { - line_breaker: Default::default(), - h_align: glyph_brush::HorizontalAlign::Left, - v_align: glyph_brush::VerticalAlign::Center, - }, - text: vec![glyph_brush::Text { - text: display_text, - scale: (size as f32 * p_scale).into(), - font_id: font.0, - extra: (), - }], - };*/ let text_primitive = Primitive::Text { glyphs, - //size: size as f32, bounds, linear_color, - /*font, - *horizontal_alignment, - *vertical_alignment, */ }; let primitive = match cursor_primitive { diff --git a/voxygen/src/ui/ice/widget/aspect_ratio_container.rs b/voxygen/src/ui/ice/widget/aspect_ratio_container.rs index 4d118bf62f..6790984c16 100644 --- a/voxygen/src/ui/ice/widget/aspect_ratio_container.rs +++ b/voxygen/src/ui/ice/widget/aspect_ratio_container.rs @@ -169,8 +169,6 @@ where } pub trait Renderer: iced::Renderer { - /// The style supported by this renderer. - //type Style: Default; /// The handle used by this renderer for images. type ImageHandle: Hash; @@ -182,7 +180,6 @@ pub trait Renderer: iced::Renderer { bounds: Rectangle, cursor_position: Point, viewport: &Rectangle, - //style: &Self::Style, content: &Element<'_, M, Self>, content_layout: Layout<'_>, ) -> Self::Output; diff --git a/voxygen/src/ui/ice/widget/background_container.rs b/voxygen/src/ui/ice/widget/background_container.rs index 4ee21ae042..4d8305e335 100644 --- a/voxygen/src/ui/ice/widget/background_container.rs +++ b/voxygen/src/ui/ice/widget/background_container.rs @@ -86,8 +86,6 @@ pub trait Background: Sized { /// This widget is displays a background image behind it's content pub struct BackgroundContainer<'a, M, R: self::Renderer, B: Background> { - //width: Length, - //height: Length, max_width: u32, max_height: u32, background: B, @@ -104,8 +102,6 @@ where { pub fn new(background: B, content: impl Into>) -> Self { Self { - //width: Length::Shrink, - //height: Length::Shrink, max_width: u32::MAX, max_height: u32::MAX, background, @@ -119,16 +115,6 @@ where self } - /*pub fn width(mut self, width: Length) -> Self { - self.width = width; - self - } - - pub fn height(mut self, height: Length) -> Self { - self.height = height; - self - }*/ - pub fn max_width(mut self, max_width: u32) -> Self { self.max_width = max_width; self @@ -138,17 +124,6 @@ where self.max_height = max_height; self } - - // Consider having these wire into underlying background - /*pub fn fix_aspect_ratio(mut self) -> Self { - self.fix_aspect_ratio = true; - self - } - - pub fn color(mut self, color: Rgba) -> Self { - self.color = color; - self - }*/ } impl<'a, M, R, B> Widget for BackgroundContainer<'a, M, R, B> @@ -221,7 +196,6 @@ where // TODO: handle cases where self and/or children are not Length::Fill // If fill use max_size - //if match self.width(), self.height() // This time we need to adjust up to meet the aspect ratio // so that the container is larger than the contents diff --git a/voxygen/src/ui/ice/widget/compound_graphic.rs b/voxygen/src/ui/ice/widget/compound_graphic.rs index dcd846df23..054bc041dc 100644 --- a/voxygen/src/ui/ice/widget/compound_graphic.rs +++ b/voxygen/src/ui/ice/widget/compound_graphic.rs @@ -128,15 +128,7 @@ impl CompoundGraphic { // self //} - fn draw( - &self, - renderer: &mut R, - _defaults: &R::Defaults, - layout: Layout<'_>, - _cursor_position: Point, - // Note: could use to skip elements outside the viewport - _viewport: &Rectangle, - ) -> R::Output { + fn draw(&self, renderer: &mut R, layout: Layout<'_>) -> R::Output { let [pixel_w, pixel_h] = self.graphics_size; let bounds = layout.bounds(); let scale = Vec2::new( @@ -158,7 +150,7 @@ impl CompoundGraphic { (bounds, graphic.kind) }); - renderer.draw(graphics, /* self.color, */ layout) + renderer.draw(graphics) } } @@ -194,12 +186,13 @@ where fn draw( &self, renderer: &mut R, - defaults: &R::Defaults, + _defaults: &R::Defaults, layout: Layout<'_>, - cursor_position: Point, - viewport: &Rectangle, + _cursor_position: Point, + // Note: could use to skip elements outside the viewport + _viewport: &Rectangle, ) -> R::Output { - Self::draw(self, renderer, defaults, layout, cursor_position, viewport) + Self::draw(self, renderer, layout) } fn hash_layout(&self, state: &mut Hasher) { @@ -215,12 +208,7 @@ where } pub trait Renderer: iced::Renderer { - fn draw( - &mut self, - graphics: I, - //color: Rgba, - layout: Layout<'_>, - ) -> Self::Output + fn draw(&mut self, graphics: I) -> Self::Output where I: Iterator; } @@ -251,11 +239,11 @@ where fn draw( &self, renderer: &mut R, - defaults: &R::Defaults, + _defaults: &R::Defaults, layout: Layout<'_>, - cursor_position: Point, - viewport: &Rectangle, + _cursor_position: Point, + _viewport: &Rectangle, ) -> R::Output { - Self::draw(self, renderer, defaults, layout, cursor_position, viewport) + Self::draw(self, renderer, layout) } } diff --git a/voxygen/src/ui/ice/widget/mouse_detector.rs b/voxygen/src/ui/ice/widget/mouse_detector.rs index eb25275213..59f095688a 100644 --- a/voxygen/src/ui/ice/widget/mouse_detector.rs +++ b/voxygen/src/ui/ice/widget/mouse_detector.rs @@ -16,23 +16,13 @@ impl State { pub struct MouseDetector<'a> { width: Length, height: Length, - //on_enter: M, - //on_exit: M, state: &'a mut State, } impl<'a> MouseDetector<'a> { - pub fn new( - state: &'a mut State, - //on_enter: M, - //on_exit: M, - width: Length, - height: Length, - ) -> Self { + pub fn new(state: &'a mut State, width: Length, height: Length) -> Self { Self { state, - //on_enter, - //on_exit, width, height, } @@ -70,12 +60,6 @@ where && y < bounds.y + bounds.height; if mouse_over != self.state.mouse_over { self.state.mouse_over = mouse_over; - - /*messages.push(if mouse_over { - self.on_enter.clone() - } else { - self.on_exit.clone() - });*/ } } } diff --git a/voxygen/src/ui/ice/widget/overlay.rs b/voxygen/src/ui/ice/widget/overlay.rs index 33320325ea..6b63b75b1a 100644 --- a/voxygen/src/ui/ice/widget/overlay.rs +++ b/voxygen/src/ui/ice/widget/overlay.rs @@ -213,7 +213,6 @@ pub trait Renderer: iced::Renderer { bounds: Rectangle, cursor_position: Point, viewport: &Rectangle, - //style: &self::Style, over: &Element<'_, M, Self>, over_layout: Layout<'_>, under: &Element<'_, M, Self>, diff --git a/voxygen/src/ui/ice/widget/stack.rs b/voxygen/src/ui/ice/widget/stack.rs index e325eed59b..86102a4dd7 100644 --- a/voxygen/src/ui/ice/widget/stack.rs +++ b/voxygen/src/ui/ice/widget/stack.rs @@ -4,17 +4,6 @@ use std::hash::Hash; /// Stack up some widgets pub struct Stack<'a, M, R> { - // Add these if it is useful - /* - padding: u16, - width: Length, - height: Length, - max_width: u32, - max_height: u32, - horizontal_alignment: Align, - vertical_alignment: Align - align_items: Align, - */ children: Vec>, } diff --git a/voxygen/src/ui/ice/widget/tooltip.rs b/voxygen/src/ui/ice/widget/tooltip.rs index df1a788553..bb63380f7a 100644 --- a/voxygen/src/ui/ice/widget/tooltip.rs +++ b/voxygen/src/ui/ice/widget/tooltip.rs @@ -310,14 +310,11 @@ where const PAD: f32 = 8.0; // TODO: allow configuration let space_above = (avoid.y - PAD).max(0.0); let space_below = (bounds.height - avoid.y - avoid.height - PAD).max(0.0); - //let space_left = avoid.x.min(0.0) - //let space_right = (bounds.width - avoid.x - avoid.width).min(0.0); let limits = layout::Limits::new( Size::ZERO, Size::new(bounds.width, space_above.max(space_below)), ); - //.width(self.content.width()); let mut node = self.content.layout(renderer, &limits); @@ -360,7 +357,6 @@ where self.alpha, defaults, cursor_position, - // TODO: hopefully this works &layout.bounds(), &self.content, layout, From 6bbb4a243fc16b4991157f8c9ae6e8ae4200e40d Mon Sep 17 00:00:00 2001 From: Imbris Date: Sun, 8 Nov 2020 18:49:54 -0500 Subject: [PATCH 61/61] ui: Improve login info banner gradient, smallify things, put login menu quit button at the bottom --- .../element/frames/banner_gradient_bottom.png | Bin 0 -> 138 bytes voxygen/src/menu/char_selection/ui/mod.rs | 22 ++++++++----- voxygen/src/menu/main/ui/connecting.rs | 4 +-- voxygen/src/menu/main/ui/login.rs | 29 +++++++++--------- voxygen/src/menu/main/ui/mod.rs | 1 + 5 files changed, 32 insertions(+), 24 deletions(-) create mode 100644 assets/voxygen/element/frames/banner_gradient_bottom.png diff --git a/assets/voxygen/element/frames/banner_gradient_bottom.png b/assets/voxygen/element/frames/banner_gradient_bottom.png new file mode 100644 index 0000000000000000000000000000000000000000..bb2204fe3d008a1853b36d0e03414476d7fe3f44 GIT binary patch literal 138 zcmeAS@N?(olHy`uVBq!ia0vp^j6j^s!3HFwG_!61sX$K`#}J9jO9MUm8XS0(YMYM# z{I8!Xk?$sD5~$xY?cGtcHm(M)14R=KuQjk;@>42WeYT_TGwlGz1;$l@Z6;NXf1`{4 m$;R|M-S&OB;bp3L8AJV&WS++twdMj1XYh3Ob6Mw<&;$V7crO$H literal 0 HcmV?d00001 diff --git a/voxygen/src/menu/char_selection/ui/mod.rs b/voxygen/src/menu/char_selection/ui/mod.rs index a57a82b841..8fa52d72a0 100644 --- a/voxygen/src/menu/char_selection/ui/mod.rs +++ b/voxygen/src/menu/char_selection/ui/mod.rs @@ -45,6 +45,8 @@ const FILL_FRAC_TWO: f32 = 0.60; const TOOLTIP_HOVER_DUR: std::time::Duration = std::time::Duration::from_millis(150); const TOOLTIP_FADE_DUR: std::time::Duration = std::time::Duration::from_millis(350); const BANNER_ALPHA: u8 = 210; +// Buttons in the bottom corners +const SMALL_BUTTON_HEIGHT: u16 = 31; const STARTER_HAMMER: &str = "common.items.weapons.hammer.starter_hammer"; const STARTER_BOW: &str = "common.items.weapons.bow.starter_bow"; @@ -361,6 +363,8 @@ impl Controls { Text::new(&client.server_info.name) .size(fonts.cyri.scale(25)) .into(), + // TODO: show additional server info here + Space::new(Length::Fill, Length::Units(25)).into(), ]) .spacing(5) .align_items(Align::Center), @@ -499,7 +503,7 @@ impl Controls { Container::new( Scrollable::new(characters_scroll) .push(Column::with_children(characters).spacing(4)) - .padding(5) + .padding(6) .scrollbar_width(5) .scroller_width(5) .width(Length::Fill) @@ -512,13 +516,13 @@ impl Controls { 0, BANNER_ALPHA, ))) - .width(Length::Units(320)) + .width(Length::Units(322)) .height(Length::Fill) .center_x() .into(), Image::new(imgs.frame_bottom) .height(Length::Units(40)) - .width(Length::Units(320)) + .width(Length::Units(322)) .color(Rgba::from_translucent(0, BANNER_ALPHA)) .into(), ]) @@ -526,7 +530,7 @@ impl Controls { let right_column = Column::with_children(vec![server.into(), characters.into()]) .spacing(10) - .width(Length::Units(320)) // TODO: see if we can get iced to work with settings below + .width(Length::Units(322)) // TODO: see if we can get iced to work with settings below //.max_width(360) //.width(Length::Fill) .height(Length::Fill); @@ -558,11 +562,11 @@ impl Controls { let bottom = Row::with_children(vec![ Container::new(logout) .width(Length::Fill) - .height(Length::Units(40)) + .height(Length::Units(SMALL_BUTTON_HEIGHT)) .into(), Container::new(enter_world) .width(Length::Fill) - .height(Length::Units(60)) + .height(Length::Units(52)) .center_x() .into(), Space::new(Length::Fill, Length::Shrink).into(), @@ -571,6 +575,7 @@ impl Controls { let content = Column::with_children(vec![top.into(), bottom.into()]) .width(Length::Fill) + .padding(5) .height(Length::Fill); // Overlay delete prompt @@ -1135,7 +1140,7 @@ impl Controls { let bottom = Row::with_children(vec![ Container::new(back) .width(Length::Fill) - .height(Length::Units(40)) + .height(Length::Units(SMALL_BUTTON_HEIGHT)) .into(), Container::new(bottom_center) .width(Length::Fill) @@ -1143,7 +1148,7 @@ impl Controls { .into(), Container::new(create) .width(Length::Fill) - .height(Length::Units(40)) + .height(Length::Units(SMALL_BUTTON_HEIGHT)) .align_x(Align::End) .into(), ]) @@ -1152,6 +1157,7 @@ impl Controls { Column::with_children(vec![top.into(), bottom.into()]) .width(Length::Fill) .height(Length::Fill) + .padding(5) .into() }, }; diff --git a/voxygen/src/menu/main/ui/connecting.rs b/voxygen/src/menu/main/ui/connecting.rs index 0636486e7a..1ca9c3d5bf 100644 --- a/voxygen/src/menu/main/ui/connecting.rs +++ b/voxygen/src/menu/main/ui/connecting.rs @@ -71,7 +71,7 @@ impl Screen { Some(Message::CancelConnect), )) .width(Length::Fill) - .height(Length::Units(fonts.cyri.scale(40))) + .height(Length::Units(fonts.cyri.scale(30))) .center_x() .padding(3); @@ -110,7 +110,7 @@ impl Screen { bottom_content.into(), right_art.into(), ])) - .height(Length::Units(100)) + .height(Length::Units(85)) .style(style::container::Style::image(imgs.loading_art)); vec![ diff --git a/voxygen/src/menu/main/ui/login.rs b/voxygen/src/menu/main/ui/login.rs index c40dfab8fa..cb8303b951 100644 --- a/voxygen/src/menu/main/ui/login.rs +++ b/voxygen/src/menu/main/ui/login.rs @@ -20,8 +20,8 @@ use iced::{ }; use vek::*; -const INPUT_WIDTH: u16 = 280; -const INPUT_TEXT_SIZE: u16 = 24; +const INPUT_WIDTH: u16 = 230; +const INPUT_TEXT_SIZE: u16 = 20; /// Login screen for the main menu pub struct Screen { @@ -80,13 +80,6 @@ impl Screen { button_style, None, ), - neat_button( - &mut self.quit_button, - i18n.get("common.quit"), - FILL_FRAC_ONE, - button_style, - Some(Message::Quit), - ), neat_button( &mut self.language_select_button, i18n.get("common.languages"), @@ -94,9 +87,16 @@ impl Screen { button_style, Some(Message::OpenLanguageMenu), ), + neat_button( + &mut self.quit_button, + i18n.get("common.quit"), + FILL_FRAC_ONE, + button_style, + Some(Message::Quit), + ), ]) .width(Length::Fill) - .max_width(160) + .max_width(100) .spacing(5); let buttons = Container::new(buttons) @@ -112,12 +112,13 @@ impl Screen { // Note: a way to tell it to keep the height of this one piece constant and // unstreched would be nice, I suppose we could just break this out into a // column and use Length::Units - Graphic::gradient(Rgba::new(0, 0, 0, 240), Rgba::zero(), [500, 50], [0, 300]), + Graphic::image(imgs.banner_gradient_bottom, [500, 50], [0, 300]) + .color(Rgba::new(0, 0, 0, 240)), ]) .height(Length::Shrink), - Text::new(intro_text).size(fonts.cyri.scale(21)), + Text::new(intro_text).size(fonts.cyri.scale(18)), ) - .max_width(450) + .max_width(360) .padding(Padding::new().horizontal(20).top(10).bottom(60)); let left_column = Column::with_children(vec![info_window.into(), buttons.into()]) @@ -417,7 +418,7 @@ impl LoginBanner { Some(Message::Singleplayer), ), ]) - .max_width(200) + .max_width(170) .height(Length::Units(200)) .spacing(8) .into(), diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs index 075f6f8ded..4515d61c4b 100644 --- a/voxygen/src/menu/main/ui/mod.rs +++ b/voxygen/src/menu/main/ui/mod.rs @@ -41,6 +41,7 @@ image_ids_ice! { bg: "voxygen.background.bg_main", banner_top: "voxygen.element.frames.banner_top", + banner_gradient_bottom: "voxygen.element.frames.banner_gradient_bottom", button: "voxygen.element.buttons.button", button_hover: "voxygen.element.buttons.button_hover", button_press: "voxygen.element.buttons.button_press",