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/Cargo.lock b/Cargo.lock index 42287b4056..abf6abc16f 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", @@ -535,6 +554,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" @@ -553,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" @@ -677,6 +721,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" @@ -708,11 +758,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", ] @@ -903,7 +953,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", @@ -928,6 +978,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" @@ -939,6 +999,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" @@ -968,6 +1039,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" @@ -1009,6 +1094,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" @@ -1324,6 +1421,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" @@ -1723,6 +1830,12 @@ dependencies = [ "xml-rs", ] +[[package]] +name = "glam" +version = "0.9.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8637c7ec4fd0776c51eeab3e0d5d1aa7e440ece3fc2ee7d674e13c957287bfc1" + [[package]] name = "glob" version = "0.3.0" @@ -1741,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", @@ -1759,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", @@ -1769,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", @@ -1778,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", @@ -1792,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", @@ -1801,11 +1918,51 @@ 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", ] +[[package]] +name = "glyph_brush" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "afd3e2cfd503a5218dd56172a8bf7c8655a4a7cf745737c606a6edfeea1b343f" +dependencies = [ + "glyph_brush_draw_cache", + "glyph_brush_layout", + "log", + "ordered-float 1.1.0", + "rustc-hash", + "twox-hash", +] + +[[package]] +name = "glyph_brush_draw_cache" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8cef969a091be5565c2c10b31fd2f115cbeed9f783a27c96ae240ff8ceee067c" +dependencies = [ + "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", +] + [[package]] name = "guillotiere" version = "0.5.2" @@ -1988,6 +2145,69 @@ dependencies = [ "want", ] +[[package]] +name = "iced_core" +version = "0.2.1" +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=f464316#f46431600cb61d4e83e0ded1ca79525478436be3" +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=f464316#f46431600cb61d4e83e0ded1ca79525478436be3" +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=f464316#f46431600cb61d4e83e0ded1ca79525478436be3" +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=f464316#f46431600cb61d4e83e0ded1ca79525478436be3" +dependencies = [ + "iced_core", +] + +[[package]] +name = "iced_winit" +version = "0.1.1" +source = "git+https://github.com/hecrj/iced?rev=f464316#f46431600cb61d4e83e0ded1ca79525478436be3" +dependencies = [ + "iced_futures", + "iced_graphics", + "iced_native", + "log", + "thiserror", + "winapi 0.3.9", + "window_clipboard", + "winit", +] + [[package]] name = "ident_case" version = "1.0.1" @@ -2916,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", @@ -3010,7 +3230,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]] @@ -3525,7 +3754,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", ] @@ -3537,7 +3766,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", @@ -3708,8 +3937,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", @@ -3725,7 +3954,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]] @@ -4007,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", @@ -4027,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", @@ -4039,6 +4268,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 +4412,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" @@ -4472,7 +4717,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", @@ -4646,6 +4891,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" @@ -4669,6 +4920,9 @@ name = "twox-hash" version = "1.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3bfd5b7557925ce778ff9b9ef90e3ade34c524b5ff10e239c69a42d546d2af56" +dependencies = [ + "rand 0.7.3", +] [[package]] name = "tynm" @@ -4815,7 +5069,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", @@ -4828,7 +5082,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", @@ -4987,8 +5241,11 @@ dependencies = [ "git2", "glsl-include", "glutin", + "glyph_brush", "guillotiere", "hashbrown 0.7.2", + "iced_native", + "iced_winit", "image", "inline_tweak", "itertools", @@ -5014,6 +5271,7 @@ dependencies = [ "veloren-server", "veloren-voxygen-anim", "veloren-world", + "window_clipboard", "winit", "winres", ] @@ -5171,6 +5429,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 0.1.10", + "js-sys", + "wasm-bindgen", + "web-sys", +] + [[package]] name = "wasm-bindgen-macro" version = "0.2.68" @@ -5280,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]] @@ -5448,13 +5718,26 @@ 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" -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", @@ -5473,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", ] @@ -5544,6 +5827,12 @@ version = "2.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d089681aa106a86fade1b0128fb5daf07d5867a509ab036d99988dec80429a57" +[[package]] +name = "xi-unicode" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a67300977d3dc3f8034dae89778f502b6ba20b269527b3223ba59c0cf393bb8a" + [[package]] name = "xml-rs" version = "0.8.3" diff --git a/Cargo.toml b/Cargo.toml index b3ba2382e9..2c2721a2f2 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -79,6 +79,5 @@ debug = 1 [patch.crates-io] # cpal conflict fix isn't released yet -winit = { git = "https://gitlab.com/veloren/winit.git", branch = "macos-test-rebased" } -glutin = {git = "https://github.com/rust-windowing/glutin.git", rev="63a1ea7d6e64c5112418cab9f21cd409f0afd7c2"} +winit = { git = "https://gitlab.com/veloren/winit.git", branch = "macos-test-spiffed" } vek = { git = "https://gitlab.com/veloren/vek.git", branch = "fix_intrinsics" } diff --git a/assets/voxygen/element/buttons/button.png b/assets/voxygen/element/buttons/button.png index 0f29f105f4..ff2fa731c4 100644 Binary files a/assets/voxygen/element/buttons/button.png and b/assets/voxygen/element/buttons/button.png differ diff --git a/assets/voxygen/element/buttons/x_red.png b/assets/voxygen/element/buttons/x_red.png new file mode 100644 index 0000000000..86996cca2b Binary files /dev/null and b/assets/voxygen/element/buttons/x_red.png differ diff --git a/assets/voxygen/element/buttons/x_red.vox b/assets/voxygen/element/buttons/x_red.vox deleted file mode 100644 index 19ec8922f7..0000000000 Binary files a/assets/voxygen/element/buttons/x_red.vox and /dev/null differ diff --git a/assets/voxygen/element/buttons/x_red_hover.png b/assets/voxygen/element/buttons/x_red_hover.png new file mode 100644 index 0000000000..088686d559 Binary files /dev/null and b/assets/voxygen/element/buttons/x_red_hover.png differ 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 59608cc7eb..0000000000 Binary files a/assets/voxygen/element/buttons/x_red_hover.vox and /dev/null differ 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 0000000000..1f3b9c458c Binary files /dev/null and b/assets/voxygen/element/buttons/x_red_press.png differ diff --git a/assets/voxygen/element/buttons/x_red_press.vox b/assets/voxygen/element/buttons/x_red_press.vox deleted file mode 100644 index 609ba072c5..0000000000 Binary files a/assets/voxygen/element/buttons/x_red_press.vox and /dev/null differ diff --git a/assets/voxygen/element/frames/banner.png b/assets/voxygen/element/frames/banner.png deleted file mode 100644 index 748f0d7f0c..0000000000 Binary files a/assets/voxygen/element/frames/banner.png and /dev/null differ diff --git a/assets/voxygen/element/frames/banner_bottom.png b/assets/voxygen/element/frames/banner_bottom.png deleted file mode 100644 index c82073520a..0000000000 Binary files a/assets/voxygen/element/frames/banner_bottom.png and /dev/null differ 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 0000000000..bb2204fe3d Binary files /dev/null and b/assets/voxygen/element/frames/banner_gradient_bottom.png differ diff --git a/assets/voxygen/element/frames/banner_top.png b/assets/voxygen/element/frames/banner_top.png index e75a275660..42098fca1c 100644 Binary files a/assets/voxygen/element/frames/banner_top.png and b/assets/voxygen/element/frames/banner_top.png differ 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 0682f36efe..0000000000 Binary files a/assets/voxygen/element/frames/info_frame_2.vox and /dev/null differ diff --git a/assets/voxygen/element/frames/loading_screen/loading_bg.png b/assets/voxygen/element/frames/loading_screen/loading_bg.png index e3d15844dc..9a0b2a1b06 100644 Binary files a/assets/voxygen/element/frames/loading_screen/loading_bg.png and b/assets/voxygen/element/frames/loading_screen/loading_bg.png differ 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 d91a841eaf..6394ab6a67 100644 Binary files a/assets/voxygen/element/frames/loading_screen/loading_bg_l.png and b/assets/voxygen/element/frames/loading_screen/loading_bg_l.png differ 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 6431e7bb85..6f37632582 100644 Binary files a/assets/voxygen/element/frames/loading_screen/loading_bg_r.png and b/assets/voxygen/element/frames/loading_screen/loading_bg_r.png differ diff --git a/assets/voxygen/element/frames/server_frame.vox b/assets/voxygen/element/frames/server_frame.vox deleted file mode 100644 index 2478bcce6c..0000000000 Binary files a/assets/voxygen/element/frames/server_frame.vox and /dev/null differ diff --git a/assets/voxygen/element/frames/tooltip/corner.png b/assets/voxygen/element/frames/tooltip/corner.png new file mode 100644 index 0000000000..7bb8355236 Binary files /dev/null and b/assets/voxygen/element/frames/tooltip/corner.png differ diff --git a/assets/voxygen/element/frames/tooltip/edge.png b/assets/voxygen/element/frames/tooltip/edge.png new file mode 100644 index 0000000000..587e99c81f Binary files /dev/null and b/assets/voxygen/element/frames/tooltip/edge.png differ diff --git a/assets/voxygen/element/frames/window_4.vox b/assets/voxygen/element/frames/window_4.vox deleted file mode 100644 index 0b3c769fe0..0000000000 Binary files a/assets/voxygen/element/frames/window_4.vox and /dev/null differ diff --git a/assets/voxygen/element/misc_bg/textbox.png b/assets/voxygen/element/misc_bg/textbox.png new file mode 100644 index 0000000000..528077c2a1 Binary files /dev/null and b/assets/voxygen/element/misc_bg/textbox.png differ diff --git a/assets/voxygen/element/misc_bg/textbox_bot.png b/assets/voxygen/element/misc_bg/textbox_bot.png deleted file mode 100644 index 330ab52d80..0000000000 Binary files a/assets/voxygen/element/misc_bg/textbox_bot.png and /dev/null differ 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 88bef50fec..0000000000 Binary files a/assets/voxygen/element/misc_bg/textbox_mid.png and /dev/null differ diff --git a/assets/voxygen/element/misc_bg/textbox_top.png b/assets/voxygen/element/misc_bg/textbox_top.png deleted file mode 100644 index 8666ed9d1f..0000000000 Binary files a/assets/voxygen/element/misc_bg/textbox_top.png and /dev/null differ diff --git a/assets/voxygen/font/haxrcorp_4089_cyrillic_altgr_extended.ttf b/assets/voxygen/font/haxrcorp_4089_cyrillic_altgr_extended.ttf index 7d3477d3a1..6a63f5fb04 100644 Binary files a/assets/voxygen/font/haxrcorp_4089_cyrillic_altgr_extended.ttf and b/assets/voxygen/font/haxrcorp_4089_cyrillic_altgr_extended.ttf differ diff --git a/assets/voxygen/i18n/PL.ron b/assets/voxygen/i18n/PL.ron index 49b4d3173d..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 -VoxygenLocalization( +( 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..2c95575941 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", @@ -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 daab2aa0ea..16cc9752a6 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", @@ -65,6 +65,7 @@ VoxygenLocalization( "common.back": "Back", "common.create": "Create", "common.okay": "Okay", + "common.add": "Add", "common.accept": "Accept", "common.decline": "Decline", "common.disclaimer": "Disclaimer", @@ -107,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:", @@ -154,6 +158,9 @@ 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", /// End Main screen section @@ -476,7 +483,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", @@ -493,7 +500,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/assets/voxygen/i18n/es_ES.ron b/assets/voxygen/i18n/es_ES.ron index 5c608525df..a78e33a4b9 100644 --- a/assets/voxygen/i18n/es_ES.ron +++ b/assets/voxygen/i18n/es_ES.ron @@ -12,7 +12,7 @@ /// /// Localization for Spanish (Spain) -VoxygenLocalization( +( metadata: ( language_name: "Español de España", language_identifier: "es_ES", @@ -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/es_la.ron b/assets/voxygen/i18n/es_la.ron index ffdb904665..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 -VoxygenLocalization( +( 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..2649daefaf 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", @@ -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 74a92c6d8b..5a37d39d36 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", @@ -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/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..9c9226722b 100644 --- a/assets/voxygen/i18n/pt_BR.ron +++ b/assets/voxygen/i18n/pt_BR.ron @@ -1,5 +1,5 @@ /// Localization for Portuguese (Brazil) -VoxygenLocalization( +( 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..87bd22bcc7 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", @@ -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 8bc8479551..0b26362297 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", @@ -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/assets/voxygen/i18n/sv.ron b/assets/voxygen/i18n/sv.ron index f074769680..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 -VoxygenLocalization( +( 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..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-简体中文 -VoxygenLocalization( +( 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..493e6bf423 100644 --- a/assets/voxygen/i18n/zh_TW.ron +++ b/assets/voxygen/i18n/zh_TW.ron @@ -1,5 +1,5 @@ /// Localization for Traditional Chinese -VoxygenLocalization( +( metadata: ( language_name: "繁體中文", language_identifier: "zh_TW", 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/Cargo.toml b/voxygen/Cargo.toml index e7a4d8a30d..57ac5addc4 100644 --- a/voxygen/Cargo.toml +++ b/voxygen/Cargo.toml @@ -24,15 +24,21 @@ 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} -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 = "f464316"} +iced_winit = {git = "https://github.com/hecrj/iced", rev = "f464316"} +window_clipboard = "0.1.1" +glyph_brush = "0.7.0" # ECS specs = {git = "https://github.com/amethyst/specs.git", rev = "7a2e348ab2223818bad487695c66c43db88050a5"} 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..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::{i18n::VoxygenLocalization, ui::fonts::ConrodVoxygenFonts, GlobalState}; +use crate::{i18n::Localization, ui::fonts::Fonts, GlobalState}; 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 59f5a6f14b..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 { @@ -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/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..7fc03783de 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, @@ -37,19 +37,22 @@ 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 { + 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, - } - } else { - None - } + }, + Some(loadout), + ) }) + .unwrap_or_default() } } @@ -93,39 +96,34 @@ 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); }, - 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_id) => { + self.client.borrow_mut().request_character(character_id); return PlayStateResult::Switch(Box::new(SessionState::new( global_state, Rc::clone(&self.client), ))); }, + ui::Event::ClearCharacterListError => { + self.client.borrow_mut().character_list.error = None; + }, } } - 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, loadout) = + Self::get_humanoid_body_loadout(&self.char_selection_ui, &client); + + // Maintain the scene. let scene_data = scene::SceneData { time: client.state().get_time(), delta_time: client.state().ecs().read_resource::().0, @@ -141,15 +139,13 @@ impl PlayState for CharSelectionState { .figure_lod_render_distance 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). - let localized_strings = VoxygenLocalization::load_expect(&i18n_asset_key( + let localized_strings = Localization::load_expect(&i18n_asset_key( &global_state.settings.language.selected_language, )); @@ -199,19 +195,15 @@ 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 loadout = self.char_selection_ui.get_loadout(); + let client = self.client.borrow(); + let (humanoid_body, loadout) = + Self::get_humanoid_body_loadout(&self.char_selection_ui, &client); // 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); // Draw the UI to the screen. - self.char_selection_ui - .render(renderer, self.scene.globals()); + self.char_selection_ui.render(renderer); } } diff --git a/voxygen/src/menu/char_selection/ui.rs b/voxygen/src/menu/char_selection/ui.rs deleted file mode 100644 index 25fc83a9c7..0000000000 --- a/voxygen/src/menu/char_selection/ui.rs +++ /dev/null @@ -1,1611 +0,0 @@ -use crate::{ - i18n::{i18n_asset_key, VoxygenLocalization}, - render::{Consts, Globals, Renderer}, - ui::{ - fonts::ConrodVoxygenFonts, - 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}; -use std::sync::Arc; - -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_mid", - - 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: ConrodVoxygenFonts, - info_content: InfoContent, - voxygen_i18n: 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 voxygen_i18n = VoxygenLocalization::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!"); - - Self { - ui, - ids, - imgs, - rot_imgs, - fonts, - info_content: InfoContent::LoadingCharacters, - selected_character: 0, - voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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 - .voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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 new file mode 100644 index 0000000000..8fa52d72a0 --- /dev/null +++ b/voxygen/src/menu/char_selection/ui/mod.rs @@ -0,0 +1,1448 @@ +use crate::{ + i18n::{i18n_asset_key, Localization}, + render::Renderer, + ui::{ + self, + fonts::IcedFonts as Fonts, + ice::{ + component::{ + neat_button, + tooltip::{self, WithTooltip}, + }, + style, + widget::{ + mouse_detector, AspectRatioContainer, BackgroundContainer, Image, MouseDetector, + Overlay, Padding, TooltipManager, + }, + Element, IcedRenderer, IcedUi as Ui, + }, + img_ids::ImageGraphic, + }, + window, GlobalState, +}; +use client::Client; +use common::{ + assets::Asset, + character::{CharacterId, CharacterItem, MAX_CHARACTERS_PER_PLAYER}, + comp::{self, humanoid}, + LoadoutBuilder, +}; +//ImageFrame, Tooltip, +use crate::settings::Settings; +//use std::time::Duration; +//use ui::ice::widget; +use iced::{ + button, scrollable, slider, text_input, Align, Button, Column, Container, HorizontalAlignment, + Length, Row, Scrollable, Slider, Space, Text, TextInput, +}; +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(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"; +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: use for info popup frame/background +const UI_MAIN: Rgba = Rgba::new(156, 179, 179, 255); // 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", + + 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", + + // 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", + // 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", + + // Tooltips + tt_edge: "voxygen/element/frames/tooltip/edge", + tt_corner: "voxygen/element/frames/tooltip/corner", + } +} + +pub enum Event { + Logout, + Play(CharacterId), + AddCharacter { + alias: String, + tool: String, + body: comp::Body, + }, + DeleteCharacter(CharacterId), + ClearCharacterListError, +} + +enum Mode { + Select { + info_content: 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, + yes_button: button::State, + no_button: button::State, + }, + Create { + name: String, // TODO: default to username + body: humanoid::Body, + loadout: Box, + tool: &'static str, + + 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, + create_button: button::State, + randomize_button: button::State, + }, +} + +impl Mode { + pub fn select(info_content: Option) -> Self { + Self::Select { + info_content, + selected: None, + characters_scroll: Default::default(), + character_buttons: Vec::new(), + new_character_button: Default::default(), + logout_button: Default::default(), + enter_world_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(); + + let loadout = Box::new(loadout); + + Self::Create { + name, + body: humanoid::Body::random(), + loadout, + tool, + + 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(), + create_button: Default::default(), + randomize_button: Default::default(), + } + } +} + +#[derive(PartialEq)] +enum InfoContent { + Deletion(usize), + LoadingCharacters, + CreatingCharacter, + DeletingCharacter, + CharacterError(String), +} + +struct Controls { + fonts: Fonts, + imgs: Imgs, + i18n: std::sync::Arc, + // Voxygen version + version: String, + // Alpha disclaimer + alpha: String, + + tooltip_manager: TooltipManager, + // Zone for rotating the character with the mouse + mouse_detector: mouse_detector::State, + // enter: bool, + mode: Mode, +} + +#[derive(Clone)] +enum Message { + Back, + Logout, + EnterWorld, + Select(usize), + Delete(usize), + NewCharacter, + CreateCharacter, + Name(String), + BodyType(humanoid::BodyType), + Species(humanoid::Species), + Tool(&'static str), + RandomizeCharacter, + CancelDeletion, + ConfirmDeletion, + ClearCharacterListError, + 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 { + fn new(fonts: Fonts, imgs: Imgs, i18n: std::sync::Arc) -> Self { + let version = common::util::DISPLAY_VERSION_LONG.clone(); + let alpha = format!("Veloren {}", common::util::DISPLAY_VERSION.as_str()); + + Self { + fonts, + imgs, + i18n, + version, + alpha, + + tooltip_manager: TooltipManager::new(TOOLTIP_HOVER_DUR, TOOLTIP_FADE_DUR), + mouse_detector: Default::default(), + mode: Mode::select(Some(InfoContent::LoadingCharacters)), + } + } + + 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) + + // 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) + .press_image(imgs.button_press) + .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(17), + padding: 10, + }; + + 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) + .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 { + ref mut info_content, + selected, + ref mut characters_scroll, + ref mut character_buttons, + ref mut new_character_button, + ref mut logout_button, + ref mut enter_world_button, + 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) + .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), + ) + .style(style::container::Style::color(Rgba::new(0, 0, 0, 217))) + .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); + + // Character Selection List + let mut characters = 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( + // Delete button + Button::new( + delete_button, + 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)) + .with_tooltip( + tooltip_manager, + move || { + tooltip::text( + i18n.get("char_selection.delete_permanently"), + tooltip_style, + ) + }, + ), + // Select Button + AspectRatioContainer::new( + Button::new( + select_button, + Column::with_children(vec![ + 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(10) + .style( + 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) + .on_press(Message::Select(i)), + ) + .ratio_of_image(imgs.selection), + ) + .padding(12) + .align_x(Align::End) + .into() + }) + .collect::>(); + + // Add create new character button + let color = if num >= MAX_CHARACTERS_PER_PLAYER { + (97, 97, 25) + } else { + (97, 255, 18) + }; + characters.push( + 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); + if num < MAX_CHARACTERS_PER_PLAYER { + button.on_press(Message::NewCharacter) + } else { + button + } + }) + .ratio_of_image(imgs.selection) + .into(), + ); + characters + }; + + // TODO: could replace column with scrollable completely if it had a with + // children method + let characters = Column::with_children(vec![ + Container::new( + Scrollable::new(characters_scroll) + .push(Column::with_children(characters).spacing(4)) + .padding(6) + .scrollbar_width(5) + .scroller_width(5) + .width(Length::Fill) + .style(style::scrollable::Style { + track: None, + scroller: style::scrollable::Scroller::Color(UI_MAIN), + }), + ) + .style(style::container::Style::color(Rgba::from_translucent( + 0, + BANNER_ALPHA, + ))) + .width(Length::Units(322)) + .height(Length::Fill) + .center_x() + .into(), + Image::new(imgs.frame_bottom) + .height(Length::Units(40)) + .width(Length::Units(322)) + .color(Rgba::from_translucent(0, BANNER_ALPHA)) + .into(), + ]) + .height(Length::Fill); + + let right_column = Column::with_children(vec![server.into(), characters.into()]) + .spacing(10) + .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); + + 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"), + FILL_FRAC_ONE, + button_style, + Some(Message::Logout), + ); + + let enter_world = neat_button( + enter_world_button, + i18n.get("char_selection.enter_world"), + FILL_FRAC_TWO, + button_style, + selected.map(|_| Message::EnterWorld), + ); + + let bottom = Row::with_children(vec![ + Container::new(logout) + .width(Length::Fill) + .height(Length::Units(SMALL_BUTTON_HEIGHT)) + .into(), + Container::new(enter_world) + .width(Length::Fill) + .height(Length::Units(52)) + .center_x() + .into(), + Space::new(Length::Fill, Length::Shrink).into(), + ]) + .align_items(Align::End); + + let content = Column::with_children(vec![top.into(), bottom.into()]) + .width(Length::Fill) + .padding(5) + .height(Length::Fill); + + // Overlay delete prompt + if let Some(info_content) = info_content { + 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), + ), + ]) + .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(), + (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(); + + Overlay::new(over, content) + .width(Length::Fill) + .height(Length::Fill) + .center_x() + .center_y() + .into() + } else { + content.into() + } + }, + Mode::Create { + name, + body, + loadout: _, + tool, + ref mut scroll, + 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, + ref mut randomize_button, + } => { + let unselected_style = style::button::Style::new(imgs.icon_border) + .hover_image(imgs.icon_border_mo) + .press_image(imgs.icon_border_press); + + 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(60), Length::Units(60)), + ) + .style(if selected { + selected_style + } else { + unselected_style + }) + .on_press(msg), + ) + .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), + 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_tooltip( + human_button, + matches!(body.species, humanoid::Species::Human), + Message::Species(humanoid::Species::Human), + human_icon, + "common.species.human", + ) + .into(), + icon_button_tooltip( + orc_button, + matches!(body.species, humanoid::Species::Orc), + Message::Species(humanoid::Species::Orc), + orc_icon, + "common.species.orc", + ) + .into(), + 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_tooltip( + elf_button, + matches!(body.species, humanoid::Species::Elf), + Message::Species(humanoid::Species::Elf), + elf_icon, + "common.species.elf", + ) + .into(), + icon_button_tooltip( + undead_button, + matches!(body.species, humanoid::Species::Undead), + Message::Species(humanoid::Species::Undead), + undead_icon, + "common.species.undead", + ) + .into(), + icon_button_tooltip( + danari_button, + matches!(body.species, humanoid::Species::Danari), + Message::Species(humanoid::Species::Danari), + danari_icon, + "common.species.danari", + ) + .into(), + ]) + .spacing(1) + .into(), + ]) + .spacing(1); + + 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![ + icon_button_tooltip( + sword_button, + *tool == STARTER_SWORD, + Message::Tool(STARTER_SWORD), + imgs.sword, + "common.weapons.sword", + ) + .into(), + icon_button_tooltip( + hammer_button, + *tool == STARTER_HAMMER, + Message::Tool(STARTER_HAMMER), + imgs.hammer, + "common.weapons.hammer", + ) + .into(), + 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_tooltip( + sceptre_button, + *tool == STARTER_SCEPTRE, + Message::Tool(STARTER_SCEPTRE), + imgs.sceptre, + "common.weapons.sceptre", + ) + .into(), + icon_button_tooltip( + bow_button, + *tool == STARTER_BOW, + Message::Tool(STARTER_BOW), + imgs.bow, + "common.weapons.bow", + ) + .into(), + icon_button_tooltip( + staff_button, + *tool == STARTER_STAFF, + Message::Tool(STARTER_STAFF), + imgs.staff, + "common.weapons.staff", + ) + .into(), + ]) + .spacing(1) + .into(), + ]) + .spacing(1); + + 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; + // Height of interactable area + const SLIDER_HEIGHT: u16 = 30; + + 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(text) + .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, + SLIDER_BAR_PAD, + SLIDER_CURSOR_SIZE, + SLIDER_BAR_HEIGHT, + )) + .into(), + ]) + .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) + .height(SLIDER_HEIGHT) + .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) + .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::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) + //.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(10) + .width(Length::Fill) + .height(Length::Fill); + + let back = neat_button( + back_button, + i18n.get("common.back"), + FILL_FRAC_ONE, + button_style, + 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) + .with_tooltip(tooltip_manager, move || { + tooltip::text(i18n.get("common.rand_appearance"), tooltip_style) + }); + + 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 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"), + FILL_FRAC_ONE, + button_style, + (!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 + }; + + let bottom = Row::with_children(vec![ + Container::new(back) + .width(Length::Fill) + .height(Length::Units(SMALL_BUTTON_HEIGHT)) + .into(), + Container::new(bottom_center) + .width(Length::Fill) + .center_x() + .into(), + Container::new(create) + .width(Length::Fill) + .height(Length::Units(SMALL_BUTTON_HEIGHT)) + .align_x(Align::End) + .into(), + ]) + .align_items(Align::End); + + Column::with_children(vec![top.into(), bottom.into()]) + .width(Length::Fill) + .height(Length::Fill) + .padding(5) + .into() + }, + }; + + Container::new( + Column::with_children(vec![top_text.into(), content]) + .spacing(3) + .width(Length::Fill) + .height(Length::Fill), + ) + .padding(3) + .into() + } + + fn update(&mut self, message: Message, events: &mut Vec, characters: &[CharacterItem]) { + match message { + Message::Back => { + if matches!(&self.mode, Mode::Create { .. }) { + self.mode = Mode::select(None); + } + }, + 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::Select(idx) => { + if let Mode::Select { selected, .. } = &mut self.mode { + *selected = Some(idx); + } + }, + Message::Delete(idx) => { + if let Mode::Select { info_content, .. } = &mut self.mode { + *info_content = Some(InfoContent::Deletion(idx)); + } + }, + 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: String::from(*tool), + body: comp::Body::Humanoid(*body), + }); + self.mode = Mode::select(Some(InfoContent::CreatingCharacter)); + } + }, + Message::Name(value) => { + if let Mode::Create { name, .. } = &mut self.mode { + *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::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 { + if let Some(id) = characters.get(*idx).and_then(|i| i.character.id) { + events.push(Event::DeleteCharacter(id)); + } + *info_content = Some(InfoContent::DeletingCharacter); + } + } + }, + Message::CancelDeletion => { + if let Mode::Select { info_content, .. } = &mut self.mode { + 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; + 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 => {}, + } + } + + /// Get the character to display + 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)), + } + } +} + +pub struct CharSelectionUi { + ui: Ui, + controls: Controls, + enter_pressed: bool, +} + +impl CharSelectionUi { + pub fn new(global_state: &mut GlobalState) -> Self { + // Load language + let i18n = Localization::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, + global_state.settings.gameplay.ui_scale, + ) + .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, + enter_pressed: false, + } + } + + 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 { + 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 + }, + window::Event::MouseButton(_, window::PressState::Pressed) => { + !self.controls.mouse_detector.mouse_over() + }, + _ => false, + } + } + + // 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 (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) + }); + + events + } + + // 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/mod.rs b/voxygen/src/menu/main/mod.rs index 7f4bb43bd0..f3b57d1acc 100644 --- a/voxygen/src/menu/main/mod.rs +++ b/voxygen/src/menu/main/mod.rs @@ -5,12 +5,15 @@ 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}; -use tracing::{error, warn}; +use tracing::error; use ui::{Event as MainMenuEvent, MainMenuUi}; pub struct MainMenuState { @@ -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::VoxygenLocalization::load_expect( + let mut localized_strings = crate::i18n::Localization::load_expect( &crate::i18n::i18n_asset_key(&global_state.settings.language.selected_language), ); @@ -84,7 +87,7 @@ impl PlayState for MainMenuState { match event { Event::Close => return PlayStateResult::Shutdown, // Pass events to ui. - Event::Ui(event) => { + Event::IcedUi(event) => { self.main_menu_ui.handle_event(event); }, // Ignore all other events. @@ -219,6 +222,14 @@ impl PlayState for MainMenuState { password, server_address, } => { + 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()); + } + global_state.settings.save_to_file_warn(); + attempt_login( &mut global_state.settings, &mut global_state.info_message, @@ -240,15 +251,25 @@ 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(std::sync::Arc::clone(&localized_strings)); + }, #[cfg(feature = "singleplayer")] MainMenuEvent::StartSingleplayer => { let singleplayer = Singleplayer::new(None); // TODO: Make client and server use the same thread pool global_state.singleplayer = Some(singleplayer); }, - MainMenuEvent::Settings => {}, // TODO MainMenuEvent::Quit => return PlayStateResult::Shutdown, - /*MainMenuEvent::DisclaimerClosed => { + // Note: Keeping in case we re-add the disclaimer + /*MainMenuEvent::DisclaimerAccepted => { global_state.settings.show_disclaimer = false },*/ MainMenuEvent::AuthServerTrust(auth_server, trust) => { @@ -291,15 +312,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.rs b/voxygen/src/menu/main/ui.rs deleted file mode 100644 index bcb5a4a74b..0000000000 --- a/voxygen/src/menu/main/ui.rs +++ /dev/null @@ -1,907 +0,0 @@ -use crate::{ - i18n::{i18n_asset_key, VoxygenLocalization}, - render::Renderer, - ui::{ - self, - fonts::ConrodVoxygenFonts, - img_ids::{BlankGraphic, ImageGraphic, VoxelGraphic}, - Graphic, ImageFrame, Tooltip, Ui, - }, - GlobalState, -}; -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; - -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);*/ - -widget_ids! { - struct Ids { - // Background and logo - bg, - v_logo, - alpha_version, - alpha_text, - banner, - banner_top, - gears, - // 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, - // Loading Screen Tips - tip_txt_bg, - tip_txt, - // Loading Screen Artwork - mid, - left, - right, - } -} - -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_mid", - //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: (), - } -} - -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 { - 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 MainMenuUi { - ui: Ui, - ids: Ids, - imgs: Imgs, - rot_imgs: ImgsRot, - 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, - voxygen_i18n: std::sync::Arc, - fonts: ConrodVoxygenFonts, - tip_no: u16, -} - -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; - // 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); - // 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!"); - 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(); - // Load language - let voxygen_i18n = VoxygenLocalization::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!"); - - Self { - ui, - ids, - imgs, - rot_imgs, - username: networking.username.clone(), - password: "".to_owned(), - server_address: networking - .servers - .get(networking.default_server) - .cloned() - .unwrap_or_default(), - 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, - voxygen_i18n, - fonts, - tip_no: 0, - } - } - - #[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.voxygen_i18n.get("main.tip"), - &self.voxygen_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.voxygen_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 - } 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.voxygen_i18n.get("common.okay"), - PopupType::ConnectionInfo => self.voxygen_i18n.get("common.cancel"), - PopupType::AuthTrustPrompt(_) => self.voxygen_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.voxygen_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.voxygen_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")) - .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.voxygen_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::DisclaimerClosed); - } - } 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.voxygen_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.voxygen_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 = item.i; - } - } - - 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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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.voxygen_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 - } - - pub fn auth_trust_prompt(&mut self, auth_server: String) { - 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), - }) - } - - pub fn show_info(&mut self, msg: String) { - self.popup = Some(PopupData { - msg, - popup_type: PopupType::Error, - }); - self.connecting = None; - self.connect = false; - } - - pub fn connected(&mut self) { - self.popup = None; - self.connecting = None; - self.connect = false; - } - - pub fn cancel_connection(&mut self) { - self.popup = None; - self.connecting = None; - self.connect = false; - } - - pub fn handle_event(&mut self, event: ui::Event) { self.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); - events - } - - pub fn render(&self, renderer: &mut Renderer) { self.ui.render(renderer, None); } -} diff --git a/voxygen/src/menu/main/ui/connecting.rs b/voxygen/src/menu/main/ui/connecting.rs new file mode 100644 index 0000000000..1ca9c3d5bf --- /dev/null +++ b/voxygen/src/menu/main/ui/connecting.rs @@ -0,0 +1,183 @@ +use super::{ConnectionState, Imgs, Message}; +use crate::{ + i18n::Localization, + ui::{ + fonts::IcedFonts as Fonts, + ice::{component::neat_button, style, widget::Image, Element}, + }, +}; +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 { + pub fn new() -> Self { + 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: f64, + i18n: &Localization, + button_style: style::button::Style, + show_tip: bool, + ) -> Element { + 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 => { + 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 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(30))) + .center_x() + .padding(3); + + let tip_cancel = Column::with_children(vec![tip, cancel.into()]) + .width(Length::FillPortion(3)) + .align_items(Align::Center) + .spacing(5) + .padding(5); + + 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(85)) + .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)); + + 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, add]) + .spacing(20) + .height(Length::Units(25)), + ) + .align_x(Align::End) + .width(Length::Fill) + .into(), + ]) + .spacing(4) + .max_width(520) + .width(Length::Fill) + .height(Length::Fill); + + let prompt_window = Container::new(content) + .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) + .width(Length::Fill) + .height(Length::Fill) + .center_x() + .center_y(); + + vec![ + container.into(), + Space::new(Length::Fill, Length::Units(fonts.cyri.scale(15))).into(), + ] + }, + }; + + Column::with_children(children) + .width(Length::Fill) + .height(Length::Fill) + .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..b7225a0edf --- /dev/null +++ b/voxygen/src/menu/main/ui/disclaimer.rs @@ -0,0 +1,82 @@ +use super::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}; + +/// 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, + 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_with_double_cornerless_border( + (22, 19, 17, 255).into(), + (11, 11, 11, 255).into(), + (54, 46, 38, 255).into(), + ), + ), + ) + .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 new file mode 100644 index 0000000000..cb8303b951 --- /dev/null +++ b/voxygen/src/menu/main/ui/login.rs @@ -0,0 +1,434 @@ +use super::{Imgs, LoginInfo, Message, FILL_FRAC_ONE, FILL_FRAC_TWO}; +use crate::{ + i18n::Localization, + ui::{ + fonts::IcedFonts as Fonts, + ice::{ + component::neat_button, + style, + widget::{ + compound_graphic::{CompoundGraphic, Graphic}, + BackgroundContainer, Image, Padding, + }, + Element, + }, + }, +}; +use iced::{ + button, scrollable, text_input, Align, Button, Column, Container, Length, Row, Scrollable, + Space, Text, TextInput, +}; +use vek::*; + +const INPUT_WIDTH: u16 = 230; +const INPUT_TEXT_SIZE: u16 = 20; + +/// Login screen for the main menu +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: LoginBanner, + language_selection: LanguageSelectBanner, +} + +impl Screen { + pub fn new() -> Self { + Self { + servers_button: Default::default(), + settings_button: Default::default(), + quit_button: Default::default(), + language_select_button: Default::default(), + + error_okay_button: Default::default(), + + banner: LoginBanner::new(), + language_selection: LanguageSelectBanner::new(), + } + } + + #[allow(clippy::too_many_arguments)] + pub(super) fn view( + &mut self, + fonts: &Fonts, + imgs: &Imgs, + 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, + version: &str, + ) -> Element { + 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.language_select_button, + i18n.get("common.languages"), + FILL_FRAC_ONE, + 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(100) + .spacing(5); + + let buttons = Container::new(buttons) + .width(Length::Fill) + .height(Length::Fill) + .align_y(Align::End); + + 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_gradient_bottom, [500, 50], [0, 300]) + .color(Rgba::new(0, 0, 0, 240)), + ]) + .height(Length::Shrink), + Text::new(intro_text).size(fonts.cyri.scale(18)), + ) + .max_width(360) + .padding(Padding::new().horizontal(20).top(10).bottom(60)); + + let left_column = Column::with_children(vec![info_window.into(), buttons.into()]) + .width(Length::Fill) + .height(Length::Fill) + .padding(27) + .into(); + + 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_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) + .into() + } 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) + }; + + let central_column = Container::new(central_content) + .width(Length::Fill) + .height(Length::Fill) + .center_x() + .center_y(); + + 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, + central_column.into(), + right_column.into(), + ]) + .width(Length::Fill) + .height(Length::Fill) + .spacing(10) + .into() + } +} + +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 { + // 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) + .align_items(Align::Start); + + let list_items = self + .language_buttons + .iter_mut() + .zip(language_metadatas) + .enumerate() + .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( + &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(5).top(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, + + multiplayer_button: button::State, + #[cfg(feature = "singleplayer")] + singleplayer_button: button::State, +} + +impl LoginBanner { + 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, + fonts: &Fonts, + imgs: &Imgs, + login_info: &LoginInfo, + i18n: &Localization, + button_style: style::button::Style, + ) -> Element { + let input_text_size = fonts.cyri.scale(INPUT_TEXT_SIZE); + + let banner_content = Column::with_children(vec![ + Column::with_children(vec![ + BackgroundContainer::new( + Image::new(imgs.input_bg) + .width(Length::Units(INPUT_WIDTH)) + .fix_aspect_ratio(), + TextInput::new( + &mut self.username, + i18n.get("main.username"), + &login_info.username, + Message::Username, + ) + .size(input_text_size) + .on_submit(Message::FocusPassword), + ) + .padding(Padding::new().horizontal(7).top(5)) + .into(), + BackgroundContainer::new( + Image::new(imgs.input_bg) + .width(Length::Units(INPUT_WIDTH)) + .fix_aspect_ratio(), + TextInput::new( + &mut self.password, + i18n.get("main.password"), + &login_info.password, + Message::Password, + ) + .size(input_text_size) + .password() + .on_submit(Message::Multiplayer), + ) + .padding(Padding::new().horizontal(7).top(5)) + .into(), + BackgroundContainer::new( + Image::new(imgs.input_bg) + .width(Length::Units(INPUT_WIDTH)) + .fix_aspect_ratio(), + TextInput::new( + &mut self.server, + i18n.get("main.server"), + &login_info.server, + Message::Server, + ) + .size(input_text_size) + .on_submit(Message::Multiplayer), + ) + .padding(Padding::new().horizontal(7).top(5)) + .into(), + ]) + .spacing(5) + .into(), + Space::new(Length::Fill, Length::Units(8)).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(170) + .height(Length::Units(200)) + .spacing(8) + .into(), + ]) + .width(Length::Fill) + .align_items(Align::Center); + + Container::new(banner_content) + .height(Length::Fill) + .center_y() + .into() + } +} diff --git a/voxygen/src/menu/main/ui/mod.rs b/voxygen/src/menu/main/ui/mod.rs new file mode 100644 index 0000000000..4515d61c4b --- /dev/null +++ b/voxygen/src/menu/main/ui/mod.rs @@ -0,0 +1,574 @@ +mod connecting; +// Note: Keeping in case we re-add the disclaimer +//mod disclaimer; +mod login; +mod servers; + +use crate::{ + i18n::{i18n_asset_key, LanguageMetadata, Localization}, + render::Renderer, + ui::{ + self, + fonts::IcedFonts as Fonts, + ice::{style, widget, Element, Font, IcedUi as Ui}, + img_ids::{ImageGraphic, VoxelGraphic}, + Graphic, + }, + GlobalState, +}; +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}; +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); + +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 { + + v_logo: "voxygen.element.v_logo", + + + 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", + input_bg: "voxygen.element.misc_bg.textbox", + 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", + + // 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", + } +} + +// Randomly loaded background images +const BG_IMGS: [&str; 13] = [ + "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, + ChangeLanguage(LanguageMetadata), + #[cfg(feature = "singleplayer")] + StartSingleplayer, + Quit, + // Note: Keeping in case we re-add the disclaimer + //DisclaimerAccepted, + AuthServerTrust(String, bool), +} + +pub struct LoginInfo { + pub username: String, + pub password: String, + pub server: String, +} + +enum ConnectionState { + InProgress, + AuthTrustPrompt { auth_server: String, msg: String }, +} + +enum Screen { + // Note: Keeping in case we re-add the disclaimer + /*Disclaimer { + screen: disclaimer::Screen, + },*/ + Login { + screen: login::Screen, + // Error to display in a box + error: Option, + }, + Servers { + screen: servers::Screen, + }, + Connecting { + screen: connecting::Screen, + connection_state: ConnectionState, + }, +} + +struct Controls { + fonts: Fonts, + imgs: Imgs, + bg_img: widget::image::Handle, + i18n: std::sync::Arc, + // Voxygen version + version: String, + // Alpha disclaimer + alpha: String, + + selected_server_index: Option, + login_info: LoginInfo, + + is_selecting_language: bool, + selected_language_index: Option, + + time: f64, + + screen: Screen, +} + +#[derive(Clone)] +enum Message { + Quit, + Back, + ShowServers, + #[cfg(feature = "singleplayer")] + Singleplayer, + Multiplayer, + LanguageChanged(usize), + OpenLanguageMenu, + Username(String), + Password(String), + Server(String), + ServerChanged(usize), + FocusPassword, + CancelConnect, + TrustPromptAdd, + TrustPromptCancel, + CloseError, + /* Note: Keeping in case we re-add the disclaimer + *AcceptDisclaimer, */ +} + +impl Controls { + fn new( + fonts: Fonts, + imgs: Imgs, + bg_img: widget::image::Handle, + i18n: std::sync::Arc, + settings: &Settings, + ) -> Self { + 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(), + } + } else { */ + Screen::Login { + screen: login::Screen::new(), + error: None, + }; + //}; + + let login_info = LoginInfo { + username: settings.networking.username.clone(), + password: String::new(), + server: settings.networking.default_server.clone(), + }; + let selected_server_index = settings + .networking + .servers + .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, + bg_img, + i18n, + version, + alpha, + + selected_server_index, + login_info, + + is_selecting_language: false, + selected_language_index, + + time: 0.0, + + screen, + } + } + + fn view(&mut self, settings: &Settings, dt: f32) -> Element { + 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) + .hover_image(self.imgs.button_hover) + .press_image(self.imgs.button_press) + .text_color(TEXT_COLOR) + .disabled_text_color(DISABLED_TEXT_COLOR); + + let alpha = iced::Text::new(&self.alpha) + .size(self.fonts.cyri.scale(12)) + .width(Length::Fill) + .horizontal_alignment(HorizontalAlignment::Center); + + let top_text = Row::with_children(vec![ + Space::new(Length::Fill, Length::Shrink).into(), + alpha.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() + }, + ]) + .padding(3) + .width(Length::Fill); + + let bg_img = if matches!(&self.screen, Screen::Connecting {..}) { + self.bg_img + } else { + 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 { + // 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, + &self.imgs, + &self.login_info, + error.as_deref(), + &self.i18n, + self.is_selecting_language, + self.selected_language_index, + &language_metadatas, + button_style, + &self.version, + ), + Screen::Servers { screen } => screen.view( + &self.fonts, + &self.imgs, + &settings.networking.servers, + self.selected_server_index, + &self.i18n, + button_style, + ), + Screen::Connecting { + screen, + connection_state, + } => screen.view( + &self.fonts, + &self.imgs, + &connection_state, + self.time, + &self.i18n, + button_style, + settings.gameplay.loading_tips, + ), + }; + + Container::new( + Column::with_children(vec![top_text.into(), content]) + .spacing(3) + .width(Length::Fill) + .height(Length::Fill), + ) + .style(style::container::Style::image(bg_img)) + .into() + } + + 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), + Message::Back => { + self.screen = Screen::Login { + screen: login::Screen::new(), + error: None, + }; + }, + Message::ShowServers => { + 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 => { + self.screen = Screen::Connecting { + screen: connecting::Screen::new(), + connection_state: ConnectionState::InProgress, + }; + events.push(Event::StartSingleplayer); + }, + Message::Multiplayer => { + self.screen = Screen::Connecting { + screen: connecting::Screen::new(), + connection_state: ConnectionState::InProgress, + }; + + 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::LanguageChanged(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; + }, + Message::ServerChanged(new_value) => { + 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 { + screen.banner.password = text_input::State::focused(); + screen.banner.username = text_input::State::new(); + } + }, + Message::CancelConnect => { + self.exit_connect_screen(); + 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, .. } = connection_state { + let auth_server = std::mem::take(auth_server); + let added = matches!(msg, Message::TrustPromptAdd); + + *connection_state = ConnectionState::InProgress; + events.push(Event::AuthServerTrust(auth_server, added)); + } + } + }, + Message::CloseError => { + if let Screen::Login { error, .. } = &mut self.screen { + *error = None; + } + }, + /* Note: Keeping in case we re-add the disclaimer */ + /*Message::AcceptDisclaimer => { + if let Screen::Disclaimer { .. } = &self.screen { + events.push(Event::DisclaimerAccepted); + self.screen = Screen::Login { + screen: login::Screen::new(), + error: None, + }; + } + },*/ + } + } + + // Connection successful of failed + fn exit_connect_screen(&mut self) { + 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 }; + } + } + + fn connection_error(&mut self, error: String) { + if matches!(&self.screen, Screen::Connecting {..}) { + self.screen = Screen::Login { + screen: login::Screen::new(), + error: Some(error), + } + } + } + + 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 { + ui: Ui, + // TODO: re add this + // tip_no: u16, + controls: Controls, +} + +impl<'a> MainMenuUi { + pub fn new(global_state: &mut GlobalState) -> Self { + // Load language + let i18n = Localization::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(); + Font::try_from_vec(buf).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"); + + 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), None)), + i18n, + &global_state.settings, + ); + + 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); + } + + pub fn show_info(&mut self, msg: String) { self.controls.connection_error(msg); } + + pub fn connected(&mut self) { self.controls.exit_connect_screen(); } + + pub fn cancel_connection(&mut self) { self.controls.exit_connect_screen(); } + + 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(); + + 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); } +} diff --git a/voxygen/src/menu/main/ui/servers.rs b/voxygen/src/menu/main/ui/servers.rs new file mode 100644 index 0000000000..b07804ee63 --- /dev/null +++ b/voxygen/src/menu/main/ui/servers.rs @@ -0,0 +1,129 @@ +use super::{Imgs, Message, FILL_FRAC_ONE}; +use crate::{ + i18n::Localization, + ui::{ + fonts::IcedFonts as Fonts, + ice::{component::neat_button, style, Element}, + }, +}; +use iced::{ + button, scrollable, Align, Button, Column, Container, Length, Row, Scrollable, Space, 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: &[impl AsRef], + selected_server_index: Option, + i18n: &Localization, + button_style: style::button::Style, + ) -> Element { + let title = Text::new(i18n.get("main.servers.select_server")) + .size(fonts.cyri.scale(35)) + .width(Length::Fill) + .horizontal_alignment(iced::HorizontalAlignment::Center); + + let back_button = Container::new( + Container::new(neat_button( + &mut self.back_button, + i18n.get("common.back"), + FILL_FRAC_ONE, + 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) + .align_items(Align::Start) + .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()]; + } + + 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( + Container::new( + Column::with_children(vec![title.into(), list.into(), back_button.into()]) + .width(Length::Fill) + .height(Length::Fill) + .spacing(10) + .padding(20), + ) + .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) + .align_x(Align::Center) + .padding(80) + .into() + } +} 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/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/run.rs b/voxygen/src/run.rs index a8eac50d09..1c75aced18 100644 --- a/voxygen/src/run.rs +++ b/voxygen/src/run.rs @@ -34,6 +34,16 @@ 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::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)); + } + } match event { winit::event::Event::NewEvents(_) => { 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/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()) 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 9c726eadba..583bee4282 100644 --- a/voxygen/src/ui/fonts.rs +++ b/voxygen/src/ui/fonts.rs @@ -1,18 +1,18 @@ -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 { - 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)), - }; + conrod_id: ui.new_font(crate::ui::ice::RawFont::load_expect(&font.asset_key)), + } } /// Scale input size to final UI size @@ -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),)* }) } } @@ -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/graphic/mod.rs b/voxygen/src/ui/graphic/mod.rs index b89a9ac9ae..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, @@ -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 { @@ -187,6 +188,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/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/cache.rs b/voxygen/src/ui/ice/cache.rs new file mode 100644 index 0000000000..fc38f5366a --- /dev/null +++ b/voxygen/src/ui/ice/cache.rs @@ -0,0 +1,121 @@ +use super::graphic::{Graphic, GraphicCache, Id as GraphicId}; +use crate::{ + render::{Renderer, Texture}, + Error, +}; +use glyph_brush::GlyphBrushBuilder; +use std::cell::{RefCell, RefMut}; +use vek::*; + +// Multiplied by current window size +const GLYPH_CACHE_SIZE: u16 = 1; +// Glyph cache tolerances +// 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), ()>; + +// 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, + 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)); + + let glyph_brush = GlyphBrushBuilder::using_font(default_font) + .initial_cache_size((glyph_cache_dims.x as u32, glyph_cache_dims.y as u32)) + .draw_cache_scale_tolerance(SCALE_TOLERANCE) + .draw_cache_position_tolerance(POSITION_TOLERANCE) + .build(); + + Ok(Self { + 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), + }) + } + + pub fn glyph_cache_tex(&self) -> &Texture { &self.glyph_cache_tex } + + pub fn glyph_cache_mut_and_tex(&mut self) -> (&mut GlyphBrush, &Texture) { + (self.glyph_brush.get_mut(), &self.glyph_cache_tex) + } + + 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: 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 } + + 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)); + 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(()) + } +} + +// 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, + _specifier: &str, + ) -> 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/component/mod.rs b/voxygen/src/ui/ice/component/mod.rs new file mode 100644 index 0000000000..7e371a14e6 --- /dev/null +++ b/voxygen/src/ui/ice/component/mod.rs @@ -0,0 +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/neat_button.rs b/voxygen/src/ui/ice/component/neat_button.rs new file mode 100644 index 0000000000..7b5508c4fd --- /dev/null +++ b/voxygen/src/ui/ice/component/neat_button.rs @@ -0,0 +1,32 @@ +use crate::ui::ice as ui; +use iced::{button::State, Button, Element, Length}; +use ui::{ + style::button::Style, + widget::{AspectRatioContainer, FillText}, +}; + +pub fn neat_button( + state: &mut State, + label: impl Into, + fill_fraction: f32, + button_style: Style, + 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 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/component/tooltip.rs b/voxygen/src/ui/ice/component/tooltip.rs new file mode 100644 index 0000000000..c53c693095 --- /dev/null +++ b/voxygen/src/ui/ice/component/tooltip.rs @@ -0,0 +1,44 @@ +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, + pub padding: 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) + .padding(style.padding) + .into() +} + +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: 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>, + { + Tooltip::new(self, hover_content, manager) + } +} diff --git a/voxygen/src/ui/ice/mod.rs b/voxygen/src/ui/ice/mod.rs new file mode 100644 index 0000000000..7ea39b11ec --- /dev/null +++ b/voxygen/src/ui/ice/mod.rs @@ -0,0 +1,209 @@ +// tooltip_manager: TooltipManager, +mod cache; +pub mod component; +mod renderer; +pub mod widget; + +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}; + +use super::{ + graphic::{self, Graphic}, + 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::*; + +pub type Element<'a, M> = iced::Element<'a, M, IcedRenderer>; + +pub struct IcedUi { + renderer: IcedRenderer, + cache: Option, + events: Vec, + clipboard: Clipboard, + cursor_position: Vec2, + // Scaling of the ui + scale: Scale, + window_resized: Option>, + scale_mode_changed: bool, +} +impl IcedUi { + 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); + + // TODO: examine how much mem fonts take up and reduce clones if significant + Ok(Self { + renderer: IcedRenderer::new(renderer, scaled_dims, default_font)?, + cache: Some(Cache::new()), + events: Vec::new(), + // TODO: handle None + clipboard: Clipboard::new(window.window()).unwrap(), + cursor_position: Vec2::zero(), + scale, + window_resized: None, + scale_mode_changed: false, + }) + } + + /// 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) + } + + 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) { + 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 { + // 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 }) => { + 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 + // 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; + // 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, + })); + }, + // 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??? + // 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, + ) -> (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(); + // 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. + // 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 { + self.renderer + .resize(scaled_dims.map(|e| e as f32), renderer); + } + } + + 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); + + 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), + &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()); + + 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/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 new file mode 100644 index 0000000000..03721eff98 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/mod.rs @@ -0,0 +1,858 @@ +mod defaults; +mod primitive; +pub mod style; +mod widget; + +pub use defaults::Defaults; + +pub(self) use primitive::Primitive; + +use super::{ + super::graphic::{self, Graphic, TexId}, + cache::Cache, + widget::image, + Font, FontId, RawFont, Rotation, +}; +use crate::{ + render::{ + create_ui_quad, create_ui_quad_vert_gradient, Consts, DynamicModel, Globals, Mesh, + Renderer, UiLocals, UiMode, UiPipeline, + }, + Error, +}; +use common::{span, util::srgba_to_linear}; +use std::{convert::TryInto, ops::Range}; +use vek::*; + +enum DrawKind { + Image(TexId), + // Text and non-textured geometry + Plain, +} + +#[allow(dead_code)] // TODO: remove once WorldPos is used +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), + // 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 + .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!"), + } + } +} + +#[derive(PartialEq)] +enum State { + Image(TexId), + Plain, +} + +// 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, + // 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, + + // 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, + // Scissor for the whole window + window_scissor: Aabr, + + // Per-frame/update + current_state: State, + mesh: Mesh, + 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) + last_glyph_verts: Vec<(Aabr, Aabr)>, + start: usize, + // Draw commands for the next render + draw_commands: Vec, +} +impl IcedRenderer { + 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, 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, + window_scissor: default_scissor(renderer), + start: 0, + }) + } + + 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) + } + + 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 + .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; + self.window_scissor = default_scissor(renderer); + + 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) { + span!(_guard, "draw", "IcedRenderer::draw"); + // Re-use memory + self.draw_commands.clear(); + self.mesh.clear(); + self.glyphs.clear(); + + self.current_state = State::Plain; + self.start = 0; + + self.draw_primitive(primitive, Vec2::zero(), 1.0, 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))); + 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(), + }, + Aabr { + min: (0.0, 1.0).into(), + max: (1.0, 0.0).into(), + }, + Rgba::new(1.0, 1.0, 1.0, 0.3), + UiMode::Text, + )); + self.draw_commands + .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(); + let half_res = self.half_res; + + let brush_result = glyph_cache.process_queued( + |rect, tex_data| { + 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 + .iter() + .map(|x| [255, 255, 255, *x]) + .collect::>(); + + if let Err(err) = renderer.update_texture(cache_tex, offset, size, &new_data) { + tracing::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, + 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, + 1.0 - pixel_coords.min.y as f32 / half_res.y, + ), + }; + (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; + let p_scale = self.p_scale; + let half_res = self.half_res; + + glyphs + .iter() + .flat_map(|(mesh_index, glyph_count, linear_color, offset)| { + 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 * 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()) + .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), + ) + }); + }, + Err(glyph_brush::BrushError::TextureTooSmall { suggested: (x, y) }) => { + tracing::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() { + 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(); + } + + // Returns (half_res, align) + 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, p_scale) + } + + fn update_resolution_dependents(&mut self, res: Vec2) { + 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 { + 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 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, + alpha: f32, + renderer: &mut Renderer, + ) { + match primitive { + Primitive::Group { primitives } => { + primitives + .into_iter() + .for_each(|p| self.draw_primitive(p, offset, alpha, renderer)); + }, + Primitive::Image { + handle, + bounds, + color, + source_rect, + } => { + 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.a == 0.0 { + return; + } + + 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, + ..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_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 + if resolution.map(|e| e == 0).reduce_or() { + return; + // TODO: consider logging uneeded elements + } + + // Cache graphic at particular resolution. + let (uv_aabr, tex_id) = match graphic_cache.cache_res( + renderer, + graphic_id, + resolution, + // TODO: take f32 here + source_aabr.map(|e| e as f64), + 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, + }; + + // 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::Gradient { + bounds, + top_linear_color, + bottom_linear_color, + } => { + // Don't draw a transparent rectangle. + 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; + } + + 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, + } => { + let linear_color = apply_alpha(linear_color, alpha); + // Don't draw a transparent rectangle. + if linear_color.a == 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( + gl_aabr, + Aabr { + min: Vec2::zero(), + max: Vec2::zero(), + }, + linear_color, + UiMode::Geometry, + )); + }, + Primitive::Text { + glyphs, + bounds: _, // iced::Rectangle + linear_color, + } => { + 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 + // 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, + // 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... + // 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, + -10000.0, //bounds.y * self.p_scale, + ), + max: glyph_brush::ab_glyph::point( + 10000.0, //(bounds.x + bounds.width) * self.p_scale, + 10000.0, //(bounds.y + bounds.height) * self.p_scale, + ), + }, + ); + + // 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, + offset, + )); + 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::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, + y: (bounds.y * self.p_scale) as u16, + }, + max: Vec2 { + x: ((bounds.x + bounds.width) * self.p_scale) as u16, + y: ((bounds.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_scissor == current_scissor + // So not optimizing for it + + // Finish the current command. + // 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) => { + 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 + // TODO: cull primitives outside the current scissor + + // Renderer child + self.draw_primitive(*content, offset + clip_offset, alpha, 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::Opacity { alpha: a, content } => { + self.draw_primitive(*content, offset, alpha * a, renderer); + }, + Primitive::Nothing => {}, + } + } + + // 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; + } + } + + 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; + 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 +#[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 + 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 = Defaults; + // 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>, + limits: &iced::layout::Limits, + ) -> iced::layout::Node { + span!(_guard, "layout", "IcedRenderer::layout"); + let node = element.layout(self, limits); + + // Trim text measurements cache? + + node + } + + fn overlay( + &mut self, + (base_primitive, base_interaction): Self::Output, + (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 { + 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), + ) + } +} + +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 new file mode 100644 index 0000000000..a0ceec22c7 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/primitive.rs @@ -0,0 +1,42 @@ +use crate::ui::{graphic, ice::widget::image}; + +#[derive(Debug)] +pub enum Primitive { + // Allocation :( + Group { + primitives: Vec, + }, + Image { + handle: (image::Handle, graphic::Rotation), + bounds: iced::Rectangle, + color: vek::Rgba, + source_rect: Option>, + }, + // 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, + }, + Text { + glyphs: Vec, + bounds: iced::Rectangle, + linear_color: vek::Rgba, + }, + Clip { + bounds: iced::Rectangle, + offset: vek::Vec2, + content: Box, + }, + // Make content translucent + Opacity { + alpha: f32, + content: Box, + }, + Nothing, +} diff --git a/voxygen/src/ui/ice/renderer/style/button.rs b/voxygen/src/ui/ice/renderer/style/button.rs new file mode 100644 index 0000000000..32e8ab0074 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/style/button.rs @@ -0,0 +1,118 @@ +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: Rgba, +} + +impl Background { + fn new(image: image::Handle) -> Self { + Self { + default: image, + hover: image, + press: image, + color: Rgba::white(), + } + } +} +// 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, +} + +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 + } + + // TODO: this needs to be refactored since the color isn't used if there is no + // background + pub fn image_color(mut self, color: Rgba) -> 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 + } + + pub fn disabled_text_color(mut self, color: Color) -> Self { + self.disabled_text = color; + self + } + + pub fn disabled(&self) -> (Option<(image::Handle, Rgba)>, Color) { + ( + self.background.as_ref().map(|b| (b.default, b.color)), + self.disabled_text, + ) + } + + pub fn pressed(&self) -> (Option<(image::Handle, Rgba)>, Color) { + ( + 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, + ) + } +} + +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), + } + } +} 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..5e446ac996 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/style/container.rs @@ -0,0 +1,55 @@ +use super::super::super::widget::image; +use vek::Rgba; + +/// Container Border +#[derive(Clone, Copy)] +pub enum Border { + DoubleCornerless { + inner: Rgba, + outer: Rgba, + }, + Image { + corner: image::Handle, + edge: image::Handle, + }, + None, +} + +/// Background of the container +#[derive(Clone, Copy)] +pub enum Style { + Image(image::Handle, 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_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 { + 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 new file mode 100644 index 0000000000..04297da836 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/style/mod.rs @@ -0,0 +1,4 @@ +pub mod button; +pub mod container; +pub mod scrollable; +pub mod slider; 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/style/slider.rs b/voxygen/src/ui/ice/renderer/style/slider.rs new file mode 100644 index 0000000000..08a12f8913 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/style/slider.rs @@ -0,0 +1,53 @@ +use super::super::super::widget::image; +use vek::Rgba; + +#[derive(Clone, Copy)] +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 { + 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, + cursor_size: (8, 16), + bar_height: 6, + } + } +} + +#[derive(Clone, Copy)] +pub enum Cursor { + Color(Rgba), + Image(image::Handle, Rgba), +} + +#[derive(Clone, Copy)] +pub enum Bar { + Color(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/aspect_ratio_container.rs b/voxygen/src/ui/ice/renderer/widget/aspect_ratio_container.rs new file mode 100644 index 0000000000..8ae4ab2a36 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/aspect_ratio_container.rs @@ -0,0 +1,23 @@ +use super::super::{ + super::widget::{aspect_ratio_container, image}, + IcedRenderer, +}; +use iced::{Element, Layout, Point, Rectangle}; + +impl aspect_ratio_container::Renderer for IcedRenderer { + 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, + viewport: &Rectangle, + content: &Element<'_, M, Self>, + content_layout: Layout<'_>, + ) -> Self::Output { + 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 new file mode 100644 index 0000000000..eadda89bd8 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/background_container.rs @@ -0,0 +1,30 @@ +use super::super::{super::widget::background_container, IcedRenderer, Primitive}; +use iced::{Element, Layout, Point, Rectangle}; + +impl background_container::Renderer for IcedRenderer { + fn draw( + &mut self, + defaults: &Self::Defaults, + background: &B, + background_layout: Layout<'_>, + viewport: &Rectangle, + content: &Element<'_, M, Self>, + content_layout: Layout<'_>, + cursor_position: Point, + ) -> Self::Output + where + B: background_container::Background, + { + let back_primitive = background + .draw(self, defaults, background_layout, cursor_position, viewport) + .0; + let (content_primitive, mouse_interaction) = + content.draw(self, defaults, content_layout, cursor_position, viewport); + ( + Primitive::Group { + primitives: vec![back_primitive, content_primitive], + }, + mouse_interaction, + ) + } +} diff --git a/voxygen/src/ui/ice/renderer/widget/button.rs b/voxygen/src/ui/ice/renderer/widget/button.rs new file mode 100644 index 0000000000..cf05a448bf --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/button.rs @@ -0,0 +1,66 @@ +use super::super::{super::Rotation, style, Defaults, IcedRenderer, Primitive}; +use iced::{button, mouse, Element, Layout, Point, Rectangle}; + +impl button::Renderer for IcedRenderer { + // TODO: what if this gets large enough to not be copied around? + type Style = style::button::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, + &bounds, + ); + + let primitive = if let Some((handle, color)) = maybe_image { + let background = Primitive::Image { + handle: (handle, Rotation::None), + bounds, + color, + source_rect: None, + }; + + 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/widget/column.rs b/voxygen/src/ui/ice/renderer/widget/column.rs new file mode 100644 index 0000000000..b55840e9f0 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/column.rs @@ -0,0 +1,35 @@ +use super::super::{IcedRenderer, Primitive}; +use iced::{column, mouse, Element, Layout, Point, Rectangle}; + +impl column::Renderer for IcedRenderer { + fn draw( + &mut self, + defaults: &Self::Defaults, + content: &[Element<'_, M, Self>], + layout: Layout<'_>, + cursor_position: Point, + viewport: &Rectangle, + ) -> Self::Output { + let mut mouse_interaction = mouse::Interaction::default(); + + ( + Primitive::Group { + primitives: content + .iter() + .zip(layout.children()) + .map(|(child, layout)| { + let (primitive, new_mouse_interaction) = + child.draw(self, defaults, layout, cursor_position, viewport); + + if new_mouse_interaction > mouse_interaction { + mouse_interaction = new_mouse_interaction; + } + + primitive + }) + .collect(), + }, + mouse_interaction, + ) + } +} diff --git a/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs b/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs new file mode 100644 index 0000000000..f8616d4e8a --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/compound_graphic.rs @@ -0,0 +1,41 @@ +use super::super::{ + super::{widget::compound_graphic, Rotation}, + IcedRenderer, Primitive, +}; +use common::util::srgba_to_linear; +use compound_graphic::GraphicKind; +use iced::{mouse, Rectangle}; + +impl compound_graphic::Renderer for IcedRenderer { + fn draw(&mut self, graphics: I) -> Self::Output + where + I: Iterator, + { + ( + Primitive::Group { + primitives: graphics + .map(|(bounds, kind)| match kind { + GraphicKind::Image(handle, color) => Primitive::Image { + handle: (handle, Rotation::None), + bounds, + color, + source_rect: None, + }, + GraphicKind::Color(color) => Primitive::Rectangle { + 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(), + }, + mouse::Interaction::default(), + ) + } +} diff --git a/voxygen/src/ui/ice/renderer/widget/container.rs b/voxygen/src/ui/ice/renderer/widget/container.rs new file mode 100644 index 0000000000..246cf678ef --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/container.rs @@ -0,0 +1,291 @@ +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 { + type Style = style::container::Style; + + fn draw( + &mut self, + 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, viewport); + + let prim = match style_sheet { + Self::Style::Image(handle, color) => { + let background = Primitive::Image { + handle: (*handle, Rotation::None), + bounds, + color: *color, + source_rect: None, + }; + + Primitive::Group { + primitives: vec![background, content], + } + }, + 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, + ] + }, + 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, + source_rect: None, + }; + + 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, + source_rect: None, + }; + + 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, + source_rect: None, + }; + + 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, + source_rect: None, + }; + + 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, + source_rect: None, + }; + + 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, + source_rect: None, + }; + + 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, + source_rect: None, + }; + + 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, + source_rect: None, + }; + + // 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 } + }, + Self::Style::None => content, + }; + + (prim, mouse_interaction) + } +} diff --git a/voxygen/src/ui/ice/renderer/widget/image.rs b/voxygen/src/ui/ice/renderer/widget/image.rs new file mode 100644 index 0000000000..245d01167b --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/image.rs @@ -0,0 +1,27 @@ +use super::super::{ + super::{widget::image, Rotation}, + IcedRenderer, Primitive, +}; +use iced::mouse; +use vek::Rgba; + +impl image::Renderer for IcedRenderer { + fn dimensions(&self, handle: image::Handle) -> (u32, u32) { self.image_dims(handle) } + + fn draw( + &mut self, + handle: image::Handle, + color: Rgba, + layout: iced::Layout<'_>, + ) -> Self::Output { + ( + Primitive::Image { + handle: (handle, Rotation::None), + bounds: layout.bounds(), + color, + source_rect: None, + }, + mouse::Interaction::default(), + ) + } +} diff --git a/voxygen/src/ui/ice/renderer/widget/mod.rs b/voxygen/src/ui/ice/renderer/widget/mod.rs new file mode 100644 index 0000000000..8f9c84758e --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/mod.rs @@ -0,0 +1,16 @@ +mod aspect_ratio_container; +mod background_container; +mod button; +mod column; +mod compound_graphic; +mod container; +mod image; +mod mouse_detector; +mod overlay; +mod row; +mod scrollable; +mod slider; +mod space; +mod text; +mod text_input; +mod tooltip; 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/overlay.rs b/voxygen/src/ui/ice/renderer/widget/overlay.rs new file mode 100644 index 0000000000..9447396b28 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/overlay.rs @@ -0,0 +1,35 @@ +use super::super::{super::widget::overlay, IcedRenderer, Primitive}; +use iced::{mouse::Interaction, Element, Layout, Point, Rectangle}; + +impl overlay::Renderer for IcedRenderer { + fn draw( + &mut self, + 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, viewport); + + let (over, over_mouse_interaction) = + 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 { + 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/renderer/widget/row.rs b/voxygen/src/ui/ice/renderer/widget/row.rs new file mode 100644 index 0000000000..634c81d0ae --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/row.rs @@ -0,0 +1,35 @@ +use super::super::{IcedRenderer, Primitive}; +use iced::{mouse, row, Element, Layout, Point, Rectangle}; + +impl row::Renderer for IcedRenderer { + fn draw( + &mut self, + defaults: &Self::Defaults, + content: &[Element<'_, M, Self>], + layout: Layout<'_>, + cursor_position: Point, + viewport: &Rectangle, + ) -> Self::Output { + let mut mouse_interaction = mouse::Interaction::default(); + + ( + Primitive::Group { + primitives: content + .iter() + .zip(layout.children()) + .map(|(child, layout)| { + let (primitive, new_mouse_interaction) = + child.draw(self, defaults, layout, cursor_position, viewport); + + if new_mouse_interaction > mouse_interaction { + mouse_interaction = new_mouse_interaction; + } + + primitive + }) + .collect(), + }, + mouse_interaction, + ) + } +} 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..3d138d38a2 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/scrollable.rs @@ -0,0 +1,185 @@ +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_MIN_HEIGHT: u16 = 6; + +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, + scrollbar_width: u16, + scrollbar_margin: u16, + scroller_width: u16, + ) -> 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; + 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 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, + y: scrollbar_bounds.y + y_offset, + width: scroller_width as f32, + height: scroller_height, + }; + Some(scrollable::Scrollbar { + outer_bounds, + bounds: scrollbar_bounds, + margin: scrollbar_margin, + 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; + // 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 { + // 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, + source_rect: None, + }); + // Middle + primitives.push(Primitive::Image { + handle: (mid, Rotation::None), + bounds: Rectangle { + y: bounds.y + end_height, + height: middle_height, + ..bounds + }, + color, + source_rect: None, + }); + // Bottom + primitives.push(Primitive::Image { + handle: (ends, Rotation::Cw180), + bounds: Rectangle { + y: bounds.y + end_height + middle_height, + height: end_height, + ..bounds + }, + color, + source_rect: None, + }); + }, + } + } + + if let Some(track) = style.track { + primitives.push(match track { + Track::Color(color) => Primitive::Rectangle { + 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: scrollbar.bounds, + color, + source_rect: None, + }, + }); + } + + 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/slider.rs b/voxygen/src/ui/ice/renderer/widget/slider.rs new file mode 100644 index 0000000000..ec624105bf --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/slider.rs @@ -0,0 +1,92 @@ +use super::super::{super::Rotation, style, IcedRenderer, Primitive}; +use common::util::srgba_to_linear; +use core::ops::RangeInclusive; +use iced::{mouse, slider, Point, Rectangle}; +use style::slider::{Bar, Cursor, Style}; + +const CURSOR_DRAG_SHIFT: f32 = 0.7; + +impl slider::Renderer for IcedRenderer { + type Style = Style; + + const DEFAULT_HEIGHT: u16 = 25; + + 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: style.bar_height as f32, + y: bounds.y + (bounds.height - style.bar_height as f32) / 2.0, + ..bounds + }; + let bar = match style.bar { + Bar::Color(color) => Primitive::Rectangle { + bounds: bar_bounds, + linear_color: srgba_to_linear(color), + }, + // 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: Rectangle { + x: bar_bounds.x - bar_pad as f32, + width: bar_bounds.width + bar_pad as f32 * 2.0, + ..bar_bounds + }, + color, + source_rect: None, + }, + }; + + 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 { 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 { + bounds: cursor_bounds, + linear_color: srgba_to_linear(color), + }, + Cursor::Image(handle, color) => Primitive::Image { + handle: (handle, Rotation::None), + bounds: cursor_bounds, + color, + source_rect: None, + }, + }; + + 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 + }; + + #[allow(clippy::if_same_then_else)] // TODO: remove + 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) + } +} diff --git a/voxygen/src/ui/ice/renderer/widget/space.rs b/voxygen/src/ui/ice/renderer/widget/space.rs new file mode 100644 index 0000000000..61abfcb1b2 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/space.rs @@ -0,0 +1,8 @@ +use super::super::{IcedRenderer, Primitive}; +use iced::{mouse, space, Rectangle}; + +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/widget/stack.rs b/voxygen/src/ui/ice/renderer/widget/stack.rs new file mode 100644 index 0000000000..081f516a98 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/stack.rs @@ -0,0 +1,35 @@ +use super::super::{super::widget::stack, IcedRenderer, Primitive}; +use iced::{mouse, Element, Layout, Point, Rectangle}; + +impl stack::Renderer for IcedRenderer { + fn draw( + &mut self, + defaults: &Self::Defaults, + content: &[Element<'_, M, Self>], + layout: Layout<'_>, + cursor_position: Point, + viewport: &Rectangle, + ) -> Self::Output { + let mut mouse_interaction = mouse::Interaction::default(); + + ( + Primitive::Group { + primitives: content + .iter() + .zip(layout.children()) + .map(|(child, layout)| { + let (primitive, new_mouse_interaction) = + child.draw(self, defaults, layout, cursor_position, viewport); + + if new_mouse_interaction > mouse_interaction { + mouse_interaction = new_mouse_interaction; + } + + primitive + }) + .collect(), + }, + mouse_interaction, + ) + } +} diff --git a/voxygen/src/ui/ice/renderer/widget/text.rs b/voxygen/src/ui/ice/renderer/widget/text.rs new file mode 100644 index 0000000000..79a3716547 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/text.rs @@ -0,0 +1,63 @@ +use super::super::{super::FontId, IcedRenderer, Primitive}; +use glyph_brush::GlyphCruncher; +use iced::{mouse, text, Color, HorizontalAlignment, Rectangle, Size, VerticalAlignment}; + +impl text::Renderer for IcedRenderer { + type Font = FontId; + + // 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? + // 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 { + screen_position: (0.0, 0.0), + bounds: (bounds.width * p_scale, bounds.height * p_scale), + 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); + 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 { + let glyphs = self.position_glyphs( + bounds, + horizontal_alignment, + vertical_alignment, + content, + size, + font, + ); + + ( + Primitive::Text { + glyphs, + bounds, + linear_color: color.unwrap_or(defaults.text_color).into_linear().into(), + }, + 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 new file mode 100644 index 0000000000..12ca612e17 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/text_input.rs @@ -0,0 +1,285 @@ +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 = 2.0; +// Extra scroll offset past the cursor +const EXTRA_OFFSET: f32 = 10.0; + +impl text_input::Renderer for IcedRenderer { + type Style = (); + + 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(); + // Note: keeping comments below for now in case this needs to be debugged again + /* let width = */ + glyph_calculator + .glyph_bounds(section) + .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(); + + 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: (), + }], + 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'); + 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 + } + + 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); + + // 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() + } 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.3) + }; + 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 - CURSOR_WIDTH / p_scale / 2.0, + 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 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, + 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(), + }, + offset, + ) + }, + }; + + (Some(cursor_primitive), offset) + } else { + (None, 0.0) + }; + + let display_text = text.unwrap_or(if state.is_focused() { "" } else { placeholder }); + // 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, + ); + + let text_primitive = Primitive::Text { + glyphs, + bounds, + linear_color, + }; + + 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(display_text, size, font); + + let primitive = if text_width > text_bounds.width { + Primitive::Clip { + bounds: text_bounds, + offset: (scroll_offset as u32, 0).into(), + content: Box::new(primitive), + } + } 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) +} 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..0707c6bfd7 --- /dev/null +++ b/voxygen/src/ui/ice/renderer/widget/tooltip.rs @@ -0,0 +1,24 @@ +use super::super::{super::widget::tooltip, IcedRenderer, Primitive}; +use iced::{Element, Layout, Point, Rectangle}; + +impl tooltip::Renderer for IcedRenderer { + fn draw( + &mut self, + 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, viewport); + ( + Primitive::Opacity { + alpha, + content: Box::new(primitive), + }, + cursor_interaction, + ) + } +} 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..6790984c16 --- /dev/null +++ b/voxygen/src/ui/ice/widget/aspect_ratio_container.rs @@ -0,0 +1,195 @@ +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? + +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::Shrink } + + 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); + + 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) + }; + + // 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]) + } + + 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, + viewport: &Rectangle, + ) -> R::Output { + renderer.draw( + defaults, + layout.bounds(), + cursor_position, + viewport, + &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); + } + + fn overlay(&mut self, layout: Layout<'_>) -> Option> { + self.content.overlay(layout.children().next().unwrap()) + } +} + +pub trait Renderer: iced::Renderer { + /// 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, + viewport: &Rectangle, + 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 new file mode 100644 index 0000000000..4d8305e335 --- /dev/null +++ b/voxygen/src/ui/ice/widget/background_container.rs @@ -0,0 +1,335 @@ +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 +// 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 +// the aspect ratio stuff) + +#[derive(Copy, Clone, Hash)] +pub struct Padding { + pub top: u16, + pub bottom: u16, + pub right: u16, + pub left: u16, +} + +impl Padding { + pub fn new() -> Self { + Padding { + top: 0, + bottom: 0, + right: 0, + left: 0, + } + } + + 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 + } +} + +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 + fn width(&self) -> Length; + fn height(&self) -> Length; + fn aspect_ratio_fixed(&self) -> bool; + fn pixel_dims(&self, renderer: &R) -> (u16, u16); + fn draw( + &self, + renderer: &mut R, + defaults: &R::Defaults, + layout: Layout<'_>, + cursor_position: Point, + viewport: &Rectangle, + ) -> R::Output; +} + +/// This widget is displays a background image behind it's content +pub struct BackgroundContainer<'a, M, R: self::Renderer, B: Background> { + 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 { + 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 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, B> Widget for BackgroundContainer<'a, M, R, B> +where + R: self::Renderer, + B: Background, +{ + // Uses the width and height from the background + fn width(&self) -> Length { self.background.width() } + + 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()); + + 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 = 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(); + 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) + }; + // 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, + ) + }); + + // Get content size + // 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 + + // 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(); + // 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) + } else { + Size::new(content_size.height * aspect_ratio, content_size.height) + }; + + // Move content to account for padding + content.move_to(Point::new( + left_pad_frac * size.width, + top_pad_frac * size.height, + )); + + (size, content) + } else { + // 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) + }; + + 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, + 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, + ) + } + + 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.background.aspect_ratio_fixed().hash(state); + self.padding.hash(state); + // TODO: add pixel dims (need renderer) + + self.content.hash_layout(state); + } + + fn overlay(&mut self, layout: Layout<'_>) -> Option> { + self.content.overlay(layout.children().next().unwrap()) + } +} + +pub trait Renderer: iced::Renderer { + fn draw( + &mut self, + defaults: &Self::Defaults, + background: &B, + background_layout: Layout<'_>, + viewport: &Rectangle, + content: &Element<'_, M, Self>, + content_layout: Layout<'_>, + cursor_position: Point, + ) -> Self::Output + where + B: Background; +} + +// They got to live ¯\_(ツ)_/¯ +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, 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 new file mode 100644 index 0000000000..054bc041dc --- /dev/null +++ b/voxygen/src/ui/ice/widget/compound_graphic.rs @@ -0,0 +1,249 @@ +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 { + Image(Handle, Rgba), + Color(Rgba), + /// Vertical gradient + Gradient(Rgba, 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, 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, + // Not relevant here + GraphicKind::Gradient(_, _) => (), + } + self + } + + 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) + } +} + +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 from_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 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 + } + + 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 + //} + + 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( + 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) + } +} + +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, + // Note: could use to skip elements outside the viewport + _viewport: &Rectangle, + ) -> R::Output { + Self::draw(self, renderer, 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) -> 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) + } +} + +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, u16) { + (self.graphics_size[0], self.graphics_size[1]) + } + + fn draw( + &self, + renderer: &mut R, + _defaults: &R::Defaults, + layout: Layout<'_>, + _cursor_position: Point, + _viewport: &Rectangle, + ) -> R::Output { + Self::draw(self, renderer, layout) + } +} 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..0243d70656 --- /dev/null +++ b/voxygen/src/ui/ice/widget/fill_text.rs @@ -0,0 +1,126 @@ +use iced::{layout, Element, Hasher, Layout, Length, Point, Rectangle, 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, + 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; + Widget::::draw( + &self.text.clone().size(font_size), + renderer, + defaults, + layout.children().next().unwrap(), + cursor_position, + viewport, + ) + } + + 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/image.rs b/voxygen/src/ui/ice/widget/image.rs new file mode 100644 index 0000000000..2d0aa564ca --- /dev/null +++ b/voxygen/src/ui/ice/widget/image.rs @@ -0,0 +1,142 @@ +use super::super::graphic; +use iced::{layout, Element, Hasher, Layout, Length, Point, Rectangle, 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; + +pub struct Image { + handle: Handle, + width: Length, + height: Length, + fix_aspect_ratio: bool, + color: Rgba, +} + +impl Image { + pub fn new(handle: Handle) -> Self { + let width = Length::Fill; + let height = Length::Fill; + Self { + handle, + 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 Image +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) = 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) + } + + fn draw( + &self, + renderer: &mut R, + _defaults: &R::Defaults, + layout: Layout<'_>, + _cursor_position: Point, + _viewport: &Rectangle, + ) -> R::Output { + renderer.draw(self.handle, 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); + self.fix_aspect_ratio.hash(state); + // TODO: also depends on dims but we have no way to access + } +} + +pub trait Renderer: iced::Renderer { + 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> +where + R: self::Renderer, +{ + 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, u16) { + 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, + _viewport: &Rectangle, + ) -> R::Output { + renderer.draw(self.handle, self.color, layout) + } +} diff --git a/voxygen/src/ui/ice/widget/mod.rs b/voxygen/src/ui/ice/widget/mod.rs new file mode 100644 index 0000000000..5c777aad2a --- /dev/null +++ b/voxygen/src/ui/ice/widget/mod.rs @@ -0,0 +1,19 @@ +pub mod aspect_ratio_container; +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; +pub mod tooltip; + +pub use self::{ + aspect_ratio_container::AspectRatioContainer, + background_container::{BackgroundContainer, Padding}, + fill_text::FillText, + image::Image, + mouse_detector::MouseDetector, + overlay::Overlay, + tooltip::{Tooltip, TooltipManager}, +}; 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..59f095688a --- /dev/null +++ b/voxygen/src/ui/ice/widget/mouse_detector.rs @@ -0,0 +1,97 @@ +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, + state: &'a mut State, +} + +impl<'a> MouseDetector<'a> { + pub fn new(state: &'a mut State, width: Length, height: Length) -> Self { + Self { + state, + 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; + } + } + } + + fn draw( + &self, + renderer: &mut R, + _defaults: &R::Defaults, + layout: Layout<'_>, + _cursor_position: Point, + _viewport: &Rectangle, + ) -> 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 new file mode 100644 index 0000000000..6b63b75b1a --- /dev/null +++ b/voxygen/src/ui/ice/widget/overlay.rs @@ -0,0 +1,229 @@ +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.over.layout(renderer, &limits.loose()); + let over_size = over.size(); + + let size = limits.resolve(Size { + 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, vec![over, under]) + } + + fn on_event( + &mut self, + event: Event, + layout: Layout<'_>, + cursor_position: Point, + messages: &mut Vec, + renderer: &R, + clipboard: Option<&dyn Clipboard>, + ) { + let mut children = layout.children(); + let over_layout = children.next().unwrap(); + + self.over.on_event( + event.clone(), + over_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(_))) + || !over_layout.bounds().contains(cursor_position) + { + self.under.on_event( + event, + children.next().unwrap(), + cursor_position, + messages, + renderer, + clipboard, + ); + } + } + + fn draw( + &self, + renderer: &mut R, + 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, + 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); + } + + 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 { + fn draw( + &mut self, + 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; +} + +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) } +} diff --git a/voxygen/src/ui/ice/widget/stack.rs b/voxygen/src/ui/ice/widget/stack.rs new file mode 100644 index 0000000000..86102a4dd7 --- /dev/null +++ b/voxygen/src/ui/ice/widget/stack.rs @@ -0,0 +1,93 @@ +// 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 +pub struct Stack<'a, M, R> { + 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, + viewport: &Rectangle, + ) -> R::Output { + renderer.draw(defaults, &self.children, layout, cursor_position, viewport) + } + + 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)); + } + + 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 { + fn draw( + &mut self, + defaults: &Self::Defaults, + children: &[Element<'_, M, Self>], + layout: Layout<'_>, + cursor_position: Point, + viewport: &Rectangle, + ) -> 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/ice/widget/tooltip.rs b/voxygen/src/ui/ice/widget/tooltip.rs new file mode 100644 index 0000000000..bb63380f7a --- /dev/null +++ b/voxygen/src/ui/ice/widget/tooltip.rs @@ -0,0 +1,377 @@ +use iced::{ + layout, Clipboard, Element, Event, Hasher, Layout, Length, Point, Rectangle, Size, Widget, +}; +use std::{ + hash::Hash, + sync::Mutex, + time::{Duration, Instant}, +}; +use vek::*; + +#[derive(Copy, Clone, Debug)] +struct Hover { + start: Instant, + aabr: Aabr, +} + +impl Hover { + fn start(aabr: Aabr) -> Self { + Self { + start: Instant::now(), + aabr, + } + } +} + +#[derive(Copy, Clone, Debug)] +struct Show { + hover_pos: Vec2, + aabr: Aabr, +} + +#[derive(Copy, Clone, Debug)] +enum State { + Idle, + Start(Hover), + Showing(Show), + Fading(Instant, Show, Option), +} + +// Reports which widget the mouse is over +#[derive(Copy, Clone, Debug)] +struct Update((Aabr, Vec2)); + +#[derive(Debug)] +// TODO: consider moving all this state into the Renderer +pub struct TooltipManager { + state: State, + update: Mutex>, + 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, + 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 + /// 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(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) => 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, + }; + } + + fn update(&self, update: Update) { *self.update.lock().unwrap() = Some(update); } + + /// 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(_) => 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: self::Renderer> { + content: Element<'a, M, R>, + hover_content: Box Element<'a, M, R>>, + manager: &'a TooltipManager, +} + +impl<'a, M, R> Tooltip<'a, M, R> +where + R: self::Renderer, +{ + pub fn new(content: C, hover_content: H, manager: &'a TooltipManager) -> Self + where + C: Into>, + H: 'a + FnMut() -> Element<'a, M, R>, + { + Self { + content: content.into(), + hover_content: Box::new(hover_content), + manager, + } + } +} + +impl<'a, M, R> Widget for Tooltip<'a, M, R> +where + R: self::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>, + ) { + 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, + viewport: &Rectangle, + ) -> 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, + cursor_position.y.trunc() as i32, + ); + self.manager.update(Update((aabr, m_pos))); + } + + self.content + .draw(renderer, defaults, layout, cursor_position, viewport) + } + + fn overlay(&mut self, layout: Layout<'_>) -> Option> { + let bounds = layout.bounds(); + let aabr = aabr_from_bounds(bounds); + + self.manager.showing(aabr).map(|(cursor_pos, alpha)| { + iced::overlay::Element::new( + Point::ORIGIN, + Box::new(Overlay::new( + (self.hover_content)(), + cursor_pos, + bounds, + alpha, + )), + ) + }) + } + + 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 + self::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: self::Renderer> { + content: Element<'a, M, R>, + /// Cursor position + cursor_position: Point, + /// Area to avoid overlapping with + avoid: Rectangle, + /// Alpha for fading out + alpha: f32, +} + +impl<'a, M, R: self::Renderer> Overlay<'a, M, R> { + pub fn new( + content: Element<'a, M, R>, + cursor_position: Point, + avoid: Rectangle, + alpha: f32, + ) -> Self { + Self { + content, + cursor_position, + avoid, + alpha, + } + } +} + +impl<'a, M, R> iced::Overlay for Overlay<'a, M, R> +where + R: self::Renderer, +{ + fn layout(&self, renderer: &R, bounds: Size, position: Point) -> layout::Node { + 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); + + let limits = layout::Limits::new( + Size::ZERO, + Size::new(bounds.width, space_above.max(space_below)), + ); + + let mut node = self.content.layout(renderer, &limits); + + let size = node.size(); + + node.move_to(Point { + x: (bounds.width - size.width).min(cursor_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) { + struct Marker; + std::any::TypeId::of::().hash(state); + + (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); + } + + fn draw( + &self, + renderer: &mut R, + defaults: &R::Defaults, + layout: Layout<'_>, + cursor_position: Point, + ) -> R::Output { + renderer.draw( + self.alpha, + defaults, + cursor_position, + &layout.bounds(), + &self.content, + layout, + ) + } +} + +pub trait Renderer: iced::Renderer { + fn draw( + &mut self, + alpha: f32, + defaults: &Self::Defaults, + cursor_position: Point, + viewport: &Rectangle, + content: &Element<'_, M, Self>, + content_layout: Layout<'_>, + ) -> Self::Output; +} 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..e7fc961c17 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, @@ -32,11 +33,11 @@ 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}, - image::{self, Map}, + image::{Id as ImageId, Map}, input::{touch::Touch, Motion, Widget}, render::{Primitive, PrimitiveKind}, text::{self, font}, @@ -44,14 +45,9 @@ 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, - io::{BufReader, Read}, - sync::Arc, - time::Duration, -}; +use std::{sync::Arc, time::Duration}; use tracing::{error, warn}; use vek::*; @@ -101,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)>, @@ -140,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(); @@ -187,7 +172,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 +197,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 { @@ -223,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() } 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() } } diff --git a/voxygen/src/window.rs b/voxygen/src/window.rs index 79eb08a429..eba4675f8c 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. @@ -497,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, @@ -585,6 +588,8 @@ impl Window { channel::Receiver, ) = channel::unbounded::(); + let scale_factor = window.window().scale_factor(); + let mut this = Self { renderer: Renderer::new( device, @@ -601,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, @@ -622,12 +628,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 } @@ -926,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, .. } => { @@ -1377,6 +1377,12 @@ impl Window { pub fn set_keybinding_mode(&mut self, game_input: GameInput) { self.remapping_keybindings = Some(game_input); } + + 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)]