diff --git a/rustfmt.toml b/rustfmt.toml index 32a758e12d..57be816e73 100644 --- a/rustfmt.toml +++ b/rustfmt.toml @@ -1,19 +1,19 @@ -max_width = 80 +max_width = 100 hard_tabs = false tab_spaces = 4 -newline_style = "Auto" +newline_style = "Unix" indent_style = "Block" -use_small_heuristics = "Default" -fn_call_width = 60 -attr_fn_like_width = 70 -struct_lit_width = 18 -struct_variant_width = 35 -array_width = 60 -chain_width = 60 -single_line_if_else_max_width = 50 -wrap_comments = false -format_code_in_doc_comments = false -comment_width = 80 +use_small_heuristics = "Max" +fn_call_width = 100 +attr_fn_like_width = 100 +struct_lit_width = 100 +struct_variant_width = 100 +array_width = 100 +chain_width = 100 +single_line_if_else_max_width = 100 +wrap_comments = true +format_code_in_doc_comments = true +comment_width = 100 normalize_comments = false normalize_doc_attributes = false license_template_path = "" @@ -23,10 +23,10 @@ format_macro_bodies = true empty_item_single_line = true struct_lit_single_line = true fn_single_line = false -where_single_line = false +where_single_line = true imports_indent = "Block" imports_layout = "Mixed" -imports_granularity = "Preserve" +imports_granularity = "Item" group_imports = "Preserve" reorder_imports = true reorder_modules = true @@ -38,26 +38,26 @@ spaces_around_ranges = false binop_separator = "Front" remove_nested_parens = true combine_control_expr = true -overflow_delimited_expr = false -struct_field_align_threshold = 0 -enum_discrim_align_threshold = 0 -match_arm_blocks = true +overflow_delimited_expr = true +struct_field_align_threshold = 20 +enum_discrim_align_threshold = 20 +match_arm_blocks = false match_arm_leading_pipes = "Never" force_multiline_blocks = false fn_args_layout = "Tall" -brace_style = "SameLineWhere" +brace_style = "PreferSameLine" control_brace_style = "AlwaysSameLine" trailing_semicolon = true trailing_comma = "Vertical" match_block_trailing_comma = false -blank_lines_upper_bound = 1 +blank_lines_upper_bound = 3 blank_lines_lower_bound = 0 -edition = "2015" +edition = "2018" version = "One" inline_attribute_width = 0 merge_derives = true use_try_shorthand = false -use_field_init_shorthand = false +use_field_init_shorthand = true force_explicit_abi = true condense_wildcard_suffixes = false color = "Auto" diff --git a/src/rust/build/src/lib.rs b/src/rust/build/src/lib.rs index 88c6b2db5c..421a7218ed 100644 --- a/src/rust/build/src/lib.rs +++ b/src/rust/build/src/lib.rs @@ -1,36 +1,37 @@ #![feature(trait_alias)] -use std::path; -use std::io::ErrorKind; use std::fmt::Display; +use std::io::ErrorKind; +use std::path; /// Types that can yield a reference to std::path::Path. pub trait PathRef = AsRef; /// A structure describing a concrete release package on GitHub. pub struct GithubRelease { - pub project_url : T, - pub version : T, - pub filename : T, + pub project_url: T, + pub version: T, + pub filename: T, } -impl+Display> GithubRelease { +impl + Display> GithubRelease { /// Download the release package from GitHub if the target file was missing. Returns true if /// the file was downloaded or false if it already existed. /// /// The project_url should be a project's main page on GitHub. - pub fn download(&self, destination_dir:&path::Path) { - let url = format!("{}/releases/download/{}/{}",self.project_url,self.version,self.filename); + pub fn download(&self, destination_dir: &path::Path) { + let url = + format!("{}/releases/download/{}/{}", self.project_url, self.version, self.filename); let destination_file = destination_dir.join(self.filename.as_ref()); Self::remove_old_file(&destination_file); let mut resp = reqwest::blocking::get(&url).expect("Download failed."); - let mut out = std::fs::File::create(destination_file).expect("Failed to create file."); + let mut out = std::fs::File::create(destination_file).expect("Failed to create file."); std::io::copy(&mut resp, &mut out).expect("Failed to copy file content."); } - fn remove_old_file(file:&path::Path) { - let result = std::fs::remove_file(&file); - let error = result.err(); + fn remove_old_file(file: &path::Path) { + let result = std::fs::remove_file(&file); + let error = result.err(); let fatal_error = error.filter(|err| err.kind() != ErrorKind::NotFound); assert!(fatal_error.is_none()); } @@ -48,10 +49,10 @@ pub fn absolute_path(path: impl PathRef) -> std::io::Result { } /// Get the environment variable or panic if not available. -pub fn env_var_or_panic(var_name:&str) -> String { +pub fn env_var_or_panic(var_name: &str) -> String { match std::env::var(var_name) { Ok(var) => var, - Err(e) => panic!("Failed to read environment variable {}: {}.", var_name, e), + Err(e) => panic!("Failed to read environment variable {}: {}.", var_name, e), } } diff --git a/src/rust/ensogl/build.rs b/src/rust/ensogl/build.rs index 3f0b255703..a890d8afc2 100644 --- a/src/rust/ensogl/build.rs +++ b/src/rust/ensogl/build.rs @@ -1,18 +1,20 @@ -use std::{path, env}; +use std::env; +use std::path; /// A module with functions generating huge chunk of texts for text component benchmarks. mod huge_text_generator { use std::collections::hash_map::DefaultHasher; use std::fs::File; - use std::hash::{Hash,Hasher}; + use std::hash::Hash; + use std::hash::Hasher; use std::io::Write; use std::path::Path; - const MANY : usize = 100000; - const NOT_SO_MANY : usize = 100; + const MANY: usize = 100000; + const NOT_SO_MANY: usize = 100; /// Create a file with many lines. - pub fn make_long_text_file(name:&Path) { + pub fn make_long_text_file(name: &Path) { let mut file = File::create(name).unwrap(); for i in (1..MANY).rev() { write_verse(&mut file, i); @@ -21,40 +23,42 @@ mod huge_text_generator { } /// Create a file with not so many long lines - pub fn make_wide_text_file(name:&Path) { - let mut file = File::create(name).unwrap(); - let verses_in_line = MANY/NOT_SO_MANY; + pub fn make_wide_text_file(name: &Path) { + let mut file = File::create(name).unwrap(); + let verses_in_line = MANY / NOT_SO_MANY; for i in (1..MANY).rev() { write_verse(&mut file, i); if i % verses_in_line == 0 { let line_index = i / verses_in_line; - let offset = hash_from(line_index) % 32; - let prefix = (0..offset).map(|_| '|').collect::(); + let offset = hash_from(line_index) % 32; + let prefix = (0..offset).map(|_| '|').collect::(); writeln!(file).unwrap(); - write!(file,"{}",prefix).unwrap(); + write!(file, "{}", prefix).unwrap(); } } } - fn hash_from(i:usize) -> u64 { + fn hash_from(i: usize) -> u64 { let mut hasher = DefaultHasher::new(); i.hash(&mut hasher); hasher.finish() } - fn write_verse(file:&mut File, i:usize) { - write!(file, + fn write_verse(file: &mut File, i: usize) { + write!( + file, "{i} bottles of beer on the wall, {i} bottles of beer.\ Take one down and pass it around, {j} bottles of beer on the wall. ", i = i, - j = i-1 - ).unwrap(); + j = i - 1 + ) + .unwrap(); } } fn main() { - let out = env::var("OUT_DIR").unwrap(); + let out = env::var("OUT_DIR").unwrap(); let out_dir = path::Path::new(&out); huge_text_generator::make_long_text_file(out_dir.join("long.txt").as_path()); huge_text_generator::make_wide_text_file(out_dir.join("wide.txt").as_path()); diff --git a/src/rust/ensogl/example/src/animation.rs b/src/rust/ensogl/example/src/animation.rs index 75f4c5791e..d03b9d7042 100644 --- a/src/rust/ensogl/example/src/animation.rs +++ b/src/rust/ensogl/example/src/animation.rs @@ -2,8 +2,8 @@ use crate::prelude::*; -use ensogl_core::system::web; use ensogl_core::application::Application; +use ensogl_core::system::web; use ensogl_core::DEPRECATED_Animation; use ensogl_text_msdf_sys::run_once_initialized; use logger::TraceLogger as Logger; @@ -34,8 +34,7 @@ pub fn entry_point_animation() { // ======================== fn init() { - - let logger : Logger = Logger::new("AnimationTest"); + let logger: Logger = Logger::new("AnimationTest"); let network = enso_frp::Network::new("test"); let animation = DEPRECATED_Animation::::new(&network); animation.set_target_value(-259_830.0); diff --git a/src/rust/ensogl/example/src/complex_shape_system.rs b/src/rust/ensogl/example/src/complex_shape_system.rs index a66646ce2c..469ef67ca6 100644 --- a/src/rust/ensogl/example/src/complex_shape_system.rs +++ b/src/rust/ensogl/example/src/complex_shape_system.rs @@ -2,15 +2,15 @@ use ensogl_core::prelude::*; +use ensogl_core::data::color; use ensogl_core::display::navigation::navigator::Navigator; -use ensogl_core::system::web; -use wasm_bindgen::prelude::*; use ensogl_core::display::object::ObjectOps; -use ensogl_core::display::world::*; use ensogl_core::display::scene; use ensogl_core::display::shape::*; -use ensogl_core::data::color; use ensogl_core::display::style::theme; +use ensogl_core::display::world::*; +use ensogl_core::system::web; +use wasm_bindgen::prelude::*; @@ -58,30 +58,30 @@ pub fn entry_point_complex_shape_system() { web::forward_panic_hook_to_console(); web::set_stack_trace_limit(); - let world = World::new(&web::get_html_element_by_id("root").unwrap()); - let scene = world.scene(); - let camera = scene.camera().clone_ref(); - let navigator = Navigator::new(scene,&camera); - let logger = Logger::new("ShapeView"); + let world = World::new(&web::get_html_element_by_id("root").unwrap()); + let scene = world.scene(); + let camera = scene.camera().clone_ref(); + let navigator = Navigator::new(scene, &camera); + let logger = Logger::new("ShapeView"); let theme_manager = theme::Manager::from(&scene.style_sheet); let theme1 = theme::Theme::new(); - theme1.set("base_color", color::Rgba::new(0.0,0.0,1.0,1.0)); + theme1.set("base_color", color::Rgba::new(0.0, 0.0, 1.0, 1.0)); theme1.set("animation.duration", 0.5); theme1.set("graph.node.shadow.color", 5.0); theme1.set("graph.node.shadow.size", 5.0); - theme1.set("mouse.pointer.color", color::Rgba::new(0.3,0.3,0.3,1.0)); + theme1.set("mouse.pointer.color", color::Rgba::new(0.3, 0.3, 0.3, 1.0)); let theme2 = theme::Theme::new(); - theme2.set("base_color", color::Rgba::new(0.0,1.0,0.0,1.0)); + theme2.set("base_color", color::Rgba::new(0.0, 1.0, 0.0, 1.0)); theme2.set("animation.duration", 0.7); theme2.set("graph.node.shadow.color", 5.0); theme2.set("graph.node.shadow.size", 5.0); - theme2.set("mouse.pointer.color", color::Rgba::new(0.3,0.3,0.3,1.0)); + theme2.set("mouse.pointer.color", color::Rgba::new(0.3, 0.3, 0.3, 1.0)); - theme_manager.register("theme1",theme1); - theme_manager.register("theme2",theme2); + theme_manager.register("theme1", theme1); + theme_manager.register("theme2", theme2); theme_manager.set_enabled(&["theme1".to_string()]); @@ -97,7 +97,8 @@ pub fn entry_point_complex_shape_system() { mask.size.set(Vector2::new(300.0, 300.0)); mask.mod_position(|t| *t = Vector3::new(-50.0, 0.0, 0.0)); - let scissor_box = scene::layer::ScissorBox::new_with_position_and_size(default(),Vector2(600,600)); + let scissor_box = + scene::layer::ScissorBox::new_with_position_and_size(default(), Vector2(600, 600)); scene.layers.main.set_scissor_box(Some(&scissor_box)); let view2 = shape::View::new(&logger); @@ -111,51 +112,52 @@ pub fn entry_point_complex_shape_system() { let scene = world.scene().clone_ref(); let mut frame = 0; - world.on_frame(move |_time| { - mask.set_position_x(((frame as f32)/30.0).sin()*100.0); - - let _keep_alive = &navigator; - let _keep_alive = &style_watch; - let _keep_alive = &theme_manager; - if frame == 50 { - // These comments are left for easy debugging in the future. - // DEBUG!("---------------"); - // DEBUG!("{scene.layers.node_searcher:#?}"); - // DEBUG!("{scene.layers.main:#?}"); - // DEBUG!("{scene.layers.mask:#?}"); - // DEBUG!("{scene.layers.node_searcher_mask:#?}"); - // DEBUG!("{scene.layers.viz:#?}"); - } - if frame == 100 { - DEBUG!("Adding previously hidden element."); - scene.add_child(&view2); - // These comments are left for easy debugging in the future. - // DEBUG!("---------------"); - // DEBUG!("{scene.layers.node_searcher:#?}"); - // DEBUG!("{scene.layers.main:#?}"); - // DEBUG!("{scene.layers.mask:#?}"); - // DEBUG!("{scene.layers.node_searcher_mask:#?}"); - // DEBUG!("{scene.layers.viz:#?}"); - } - if frame == 150 { - DEBUG!("Enabling masking."); - // These comments are left for easy debugging in the future. - // DEBUG!("---------------"); - // DEBUG!("{scene.layers.node_searcher:#?}"); - // DEBUG!("{scene.layers.main:#?}"); - // DEBUG!("{scene.layers.mask:#?}"); - // DEBUG!("{scene.layers.node_searcher_mask:#?}"); - // DEBUG!("{scene.layers.viz:#?}"); - - scene.layers.node_searcher.add_exclusive(&view1); - scene.layers.node_searcher.add_exclusive(&view2); - scene.layers.node_searcher_mask.add_exclusive(&mask); - } - if frame == 200 { - DEBUG!("Changing the theme."); - theme_manager.set_enabled(&["theme2".to_string()]); - } - frame += 1; - }).forget(); - + world + .on_frame(move |_time| { + mask.set_position_x(((frame as f32) / 30.0).sin() * 100.0); + + let _keep_alive = &navigator; + let _keep_alive = &style_watch; + let _keep_alive = &theme_manager; + if frame == 50 { + // These comments are left for easy debugging in the future. + // DEBUG!("---------------"); + // DEBUG!("{scene.layers.node_searcher:#?}"); + // DEBUG!("{scene.layers.main:#?}"); + // DEBUG!("{scene.layers.mask:#?}"); + // DEBUG!("{scene.layers.node_searcher_mask:#?}"); + // DEBUG!("{scene.layers.viz:#?}"); + } + if frame == 100 { + DEBUG!("Adding previously hidden element."); + scene.add_child(&view2); + // These comments are left for easy debugging in the future. + // DEBUG!("---------------"); + // DEBUG!("{scene.layers.node_searcher:#?}"); + // DEBUG!("{scene.layers.main:#?}"); + // DEBUG!("{scene.layers.mask:#?}"); + // DEBUG!("{scene.layers.node_searcher_mask:#?}"); + // DEBUG!("{scene.layers.viz:#?}"); + } + if frame == 150 { + DEBUG!("Enabling masking."); + // These comments are left for easy debugging in the future. + // DEBUG!("---------------"); + // DEBUG!("{scene.layers.node_searcher:#?}"); + // DEBUG!("{scene.layers.main:#?}"); + // DEBUG!("{scene.layers.mask:#?}"); + // DEBUG!("{scene.layers.node_searcher_mask:#?}"); + // DEBUG!("{scene.layers.viz:#?}"); + + scene.layers.node_searcher.add_exclusive(&view1); + scene.layers.node_searcher.add_exclusive(&view2); + scene.layers.node_searcher_mask.add_exclusive(&mask); + } + if frame == 200 { + DEBUG!("Changing the theme."); + theme_manager.set_enabled(&["theme2".to_string()]); + } + frame += 1; + }) + .forget(); } diff --git a/src/rust/ensogl/example/src/dom_symbols.rs b/src/rust/ensogl/example/src/dom_symbols.rs index fec452aee4..e9d7ce68a3 100644 --- a/src/rust/ensogl/example/src/dom_symbols.rs +++ b/src/rust/ensogl/example/src/dom_symbols.rs @@ -1,15 +1,15 @@ #![allow(missing_docs)] -use ensogl_core::traits::*; use ensogl_core::prelude::*; +use ensogl_core::traits::*; -use ensogl_core::system::web; -use ensogl_core::system::web::NodeInserter; -use ensogl_core::display::symbol::DomSymbol; +use ensogl_core::display::navigation::navigator::Navigator; use ensogl_core::display::symbol::geometry::Sprite; use ensogl_core::display::symbol::geometry::SpriteSystem; +use ensogl_core::display::symbol::DomSymbol; use ensogl_core::display::world::*; -use ensogl_core::display::navigation::navigator::Navigator; +use ensogl_core::system::web; +use ensogl_core::system::web::NodeInserter; use web::StyleSetter; use nalgebra::Vector2; @@ -21,49 +21,49 @@ use wasm_bindgen::prelude::*; #[allow(clippy::many_single_char_names)] pub fn entry_point_dom_symbols() { web::forward_panic_hook_to_console(); - let world = World::new(&web::get_html_element_by_id("root").unwrap()); - let scene = world.scene(); - let camera = scene.camera(); - let screen = camera.screen(); - let navigator = Navigator::new(scene,&camera); + let world = World::new(&web::get_html_element_by_id("root").unwrap()); + let scene = world.scene(); + let camera = scene.camera(); + let screen = camera.screen(); + let navigator = Navigator::new(scene, &camera); let sprite_system = SpriteSystem::new(&world); world.add_child(&sprite_system); let dom_front_layer = &scene.dom.layers.front; - let dom_back_layer = &scene.dom.layers.back; + let dom_back_layer = &scene.dom.layers.back; - let mut sprites: Vec = default(); + let mut sprites: Vec = default(); let mut css3d_objects: Vec = default(); let count = 10; - for i in 0 .. count { - let x = i as f32; - let width = screen.width * 1.5 / count as f32; + for i in 0..count { + let x = i as f32; + let width = screen.width * 1.5 / count as f32; let height = screen.height; - let y = height / 2.0; + let y = height / 2.0; if i % 2 == 0 { - let height = height * 0.75; - let size = Vector2::new(width, height); + let height = height * 0.75; + let size = Vector2::new(width, height); let position = Vector3::new(width / 1.5 * x + width / 2.0, y, 0.0); - let sprite = sprite_system.new_instance(); + let sprite = sprite_system.new_instance(); sprite.size.set(size); sprite.mod_position(|t| *t = position); sprites.push(sprite); } else { let div = web::create_div(); - div.set_style_or_panic("width" , "100%"); - div.set_style_or_panic("height" , "100%"); + div.set_style_or_panic("width", "100%"); + div.set_style_or_panic("height", "100%"); div.set_inner_html("top-left"); - let size = Vector2::new(width, height); + let size = Vector2::new(width, height); let position = Vector3::new(width / 1.5 * x + width / 2.0, y, 0.0); - let object = DomSymbol::new(&div); + let object = DomSymbol::new(&div); dom_front_layer.manage(&object); world.add_child(&object); - let r = ((x + 0.0) * 16.0) as u8; - let g = ((x + 2.0) * 32.0) as u8; - let b = ((x + 4.0) * 64.0) as u8; + let r = ((x + 0.0) * 16.0) as u8; + let g = ((x + 2.0) * 32.0) as u8; + let b = ((x + 4.0) * 64.0) as u8; let color = iformat!("rgb({r},{g},{b})"); - div.set_style_or_panic("background-color",color); + div.set_style_or_panic("background-color", color); object.dom().append_or_panic(&div); object.set_size(size); @@ -71,23 +71,25 @@ pub fn entry_point_dom_symbols() { css3d_objects.push(object); } } - let layers = vec![dom_front_layer.clone_ref(),dom_back_layer.clone_ref()]; + let layers = vec![dom_front_layer.clone_ref(), dom_back_layer.clone_ref()]; let mut iter_to_change = 0; let mut i = 0; world.keep_alive_forever(); - world.on_frame(move |_| { - let _keep_alive = &navigator; - let _keep_alive = &sprites; - let _keep_alive = &sprite_system; + world + .on_frame(move |_| { + let _keep_alive = &navigator; + let _keep_alive = &sprites; + let _keep_alive = &sprite_system; - if iter_to_change == 0 { - iter_to_change = 50; - i = (i + 1) % 2; - for (j, object) in css3d_objects.iter_mut().enumerate() { - layers[(i + j) % 2].manage(object); + if iter_to_change == 0 { + iter_to_change = 50; + i = (i + 1) % 2; + for (j, object) in css3d_objects.iter_mut().enumerate() { + layers[(i + j) % 2].manage(object); + } } - } - iter_to_change -= 1; - }).forget(); + iter_to_change -= 1; + }) + .forget(); } diff --git a/src/rust/ensogl/example/src/drop_manager.rs b/src/rust/ensogl/example/src/drop_manager.rs index b62c2f1014..1832979340 100644 --- a/src/rust/ensogl/example/src/drop_manager.rs +++ b/src/rust/ensogl/example/src/drop_manager.rs @@ -3,24 +3,24 @@ use enso_prelude::*; -use ensogl_core::frp::web; use ensogl_core::display::world::World; +use ensogl_core::frp::web; +use ensogl_web::drop; use wasm_bindgen::prelude::*; use wasm_bindgen_futures::spawn_local; -use ensogl_web::drop; -fn download_file(file:drop::File) { +fn download_file(file: drop::File) { spawn_local(async move { INFO!("Received file: {file:?}"); loop { match file.read_chunk().await { Ok(Some(chunk)) => { INFO!("Received chunk: {chunk:?}"); - }, + } Ok(None) => { INFO!("All chunks received successfully"); - break - }, + break; + } Err(err) => { ERROR!("Error in receiving chunk promise: {err:?}"); break; @@ -36,25 +36,25 @@ fn download_file(file:drop::File) { pub fn entry_point_drop_manager() { web::forward_panic_hook_to_console(); - let world = World::new(&web::get_html_element_by_id("root").unwrap()); + let world = World::new(&web::get_html_element_by_id("root").unwrap()); let drop_manager = drop::Manager::new(world.scene().dom.root.as_ref()); - let network = enso_frp::Network::new("Debug Scene"); + let network = enso_frp::Network::new("Debug Scene"); enso_frp::extend! { network let file_received = drop_manager.files_received().clone_ref(); eval file_received ([](files) for file in files { download_file(file.clone_ref())}); } let mut loader_hidden = false; - world.on_frame(move |_| { - if !loader_hidden { - web::get_element_by_id("loader").map(|t| { - t.parent_node().map(|p| { - p.remove_child(&t).unwrap() - }) - }).ok(); - loader_hidden = true; - } - }).forget(); + world + .on_frame(move |_| { + if !loader_hidden { + web::get_element_by_id("loader") + .map(|t| t.parent_node().map(|p| p.remove_child(&t).unwrap())) + .ok(); + loader_hidden = true; + } + }) + .forget(); std::mem::forget(world); std::mem::forget(network); diff --git a/src/rust/ensogl/example/src/easing_animator.rs b/src/rust/ensogl/example/src/easing_animator.rs index 84839db74d..d373fb64d0 100644 --- a/src/rust/ensogl/example/src/easing_animator.rs +++ b/src/rust/ensogl/example/src/easing_animator.rs @@ -2,21 +2,21 @@ use crate::prelude::*; -use ensogl_core::animation::easing::*; use ensogl_core::animation; -use ensogl_core::system::web::AttributeSetter; +use ensogl_core::animation::easing::*; +use ensogl_core::system::web; use ensogl_core::system::web::create_element; use ensogl_core::system::web::get_element_by_id; +use ensogl_core::system::web::AttributeSetter; use ensogl_core::system::web::NodeInserter; use ensogl_core::system::web::StyleSetter; -use ensogl_core::system::web; use js_sys::Math; use nalgebra::Vector2; use std::ops::Add; use std::ops::Mul; use std::rc::Rc; -use wasm_bindgen::JsCast; use wasm_bindgen::prelude::*; +use wasm_bindgen::JsCast; use web_sys::CanvasRenderingContext2d; use web_sys::HtmlCanvasElement; use web_sys::HtmlElement; @@ -28,44 +28,44 @@ use web_sys::HtmlElement; // ================== /// Look and feel properties of sprite objects. -#[derive(Clone,Copy,Debug,PartialEq)] +#[derive(Clone, Copy, Debug, PartialEq)] #[allow(missing_docs)] pub struct SpriteData { - pub position : Vector2, - pub size : f64 + pub position: Vector2, + pub size: f64, } impl SpriteData { /// Constructor. - pub fn new(position:Vector2, size:f64) -> Self { - Self {position,size} + pub fn new(position: Vector2, size: f64) -> Self { + Self { position, size } } /// Creates random SpriteData. pub fn random() -> Self { let x = ((Math::random() - 0.5) * 2.0) as f32; let y = ((Math::random() - 0.5) * 2.0) as f32; - let position = Vector2::new(x,y); - let size = Math::random() * 100.0; - Self::new(position,size) + let position = Vector2::new(x, y); + let size = Math::random() * 100.0; + Self::new(position, size) } } impl Mul for SpriteData { type Output = Self; - fn mul(self, rhs:f32) -> Self { + fn mul(self, rhs: f32) -> Self { let position = self.position * rhs; - let size = self.size * rhs as f64; - SpriteData {position,size} + let size = self.size * rhs as f64; + SpriteData { position, size } } } impl Add for SpriteData { type Output = Self; - fn add(self, rhs:Self) -> Self { + fn add(self, rhs: Self) -> Self { let position = self.position + rhs.position; - let size = self.size + rhs.size; - SpriteData {position,size} + let size = self.size + rhs.size; + SpriteData { position, size } } } @@ -76,32 +76,32 @@ impl Add for SpriteData { // ============== /// A simplified Canvas object used in the EasingAnimator example. -#[derive(Clone,Debug)] +#[derive(Clone, Debug)] pub struct Canvas { - canvas : HtmlCanvasElement, - context : CanvasRenderingContext2d + canvas: HtmlCanvasElement, + context: CanvasRenderingContext2d, } impl Canvas { /// Constructor. - pub fn new(container_id:&str) -> Self { + pub fn new(container_id: &str) -> Self { let canvas = web::create_canvas(); canvas.set_style_or_panic("border", "1px solid black"); - canvas.set_width (256); + canvas.set_width(256); canvas.set_height(256); let context = canvas.get_context("2d").unwrap().unwrap(); - let context : CanvasRenderingContext2d = context.dyn_into().unwrap(); + let context: CanvasRenderingContext2d = context.dyn_into().unwrap(); - let app : HtmlElement = get_element_by_id(container_id).unwrap().dyn_into().unwrap(); + let app: HtmlElement = get_element_by_id(container_id).unwrap().dyn_into().unwrap(); app.append_or_panic(&canvas); - Self {canvas,context} + Self { canvas, context } } /// Clears the canvas. pub fn clear(&self) { - self.context.clear_rect(0.0,0.0,self.width(),self.height()) + self.context.clear_rect(0.0, 0.0, self.width(), self.height()) } /// Gets Canvas' width. @@ -115,7 +115,7 @@ impl Canvas { } /// Draw sprite of the provided properties. - pub fn draw_sprite(&self, data:SpriteData, color:&str) { + pub fn draw_sprite(&self, data: SpriteData, color: &str) { let size = (20.0 + data.size) / self.height(); let point = data.position; self.context.save(); @@ -123,15 +123,20 @@ impl Canvas { self.context.scale(self.width() / 2.0, self.height() / 2.0).ok(); self.context.set_line_width(2.0 / self.height()); self.context.translate(1.0, 1.0).ok(); - self.context.fill_rect(point.x as f64 - size / 2.0,point.y as f64 - size / 2.0,size,size); + self.context.fill_rect( + point.x as f64 - size / 2.0, + point.y as f64 - size / 2.0, + size, + size, + ); self.context.restore(); } /// Draw a 2D graph of the provided easing function. - pub fn draw_graphf32>(&self, f:F, color:&str, time_ms:f32) { + pub fn draw_graph f32>(&self, f: F, color: &str, time_ms: f32) { let time_ms = time_ms as f64; - let width = self.width() - 1.0; - let height = self.height(); + let width = self.width() - 1.0; + let height = self.height(); self.context.set_stroke_style(&color.into()); self.context.begin_path(); @@ -148,11 +153,11 @@ impl Canvas { self.context.stroke(); self.context.set_fill_style(&color.into()); - let width = 8.0 / width; + let width = 8.0 / width; let height = 16.0 / height; let time_s = time_ms / 1000.0; - let x = time_s / 2.0; - let y = f(x as f32) as f64; + let x = time_s / 2.0; + let y = f(x as f32) as f64; self.context.fill_rect(x - width / 2.0, y - height / 2.0, width, height); self.context.restore(); } @@ -160,35 +165,43 @@ impl Canvas { #[allow(clippy::type_complexity)] struct Sampler { - color : &'static str, - time : f32, - left_canvas : Canvas, - right_canvas : Canvas, - easing_animator : Animatorf32>, Box>, - properties : Rc>, - easing_function : Boxf32>, + color: &'static str, + time: f32, + left_canvas: Canvas, + right_canvas: Canvas, + easing_animator: Animator f32>, Box>, + properties: Rc>, + easing_function: Box f32>, } impl Sampler { #[allow(trivial_casts)] - fn new(color:&'static str, left_canvas:&Canvas, right_canvas:&Canvas, f:F) -> Self - where F:CloneableFnEasing { - let left_canvas = left_canvas.clone(); - let right_canvas = right_canvas.clone(); - let properties = Rc::new(Cell::new(SpriteData::new(Vector2::new(0.0,0.0),1.0))); - let start = SpriteData::random(); - let end = SpriteData::random(); - let prop = properties.clone(); - let easing_function = Box::new(f.clone()) as Boxf32>; - let easing_function2 = Box::new(f) as Boxf32>; - let animation_cb = Box::new(move |t| prop.set(t)) as Box; - let easing_animator = Animator::new(start,end,easing_function2,animation_cb,()); - let time = 0.0; + fn new(color: &'static str, left_canvas: &Canvas, right_canvas: &Canvas, f: F) -> Self + where F: CloneableFnEasing { + let left_canvas = left_canvas.clone(); + let right_canvas = right_canvas.clone(); + let properties = Rc::new(Cell::new(SpriteData::new(Vector2::new(0.0, 0.0), 1.0))); + let start = SpriteData::random(); + let end = SpriteData::random(); + let prop = properties.clone(); + let easing_function = Box::new(f.clone()) as Box f32>; + let easing_function2 = Box::new(f) as Box f32>; + let animation_cb = Box::new(move |t| prop.set(t)) as Box; + let easing_animator = Animator::new(start, end, easing_function2, animation_cb, ()); + let time = 0.0; easing_animator.set_duration(2000.0); - Self {color,time,left_canvas,right_canvas,easing_animator,properties,easing_function} + Self { + color, + time, + left_canvas, + right_canvas, + easing_animator, + properties, + easing_function, + } } - fn render(&mut self, time_diff:f32) { + fn render(&mut self, time_diff: f32) { self.time += time_diff; if self.time > 3000.0 { self.time = 0.0; @@ -197,8 +210,8 @@ impl Sampler { animator.set_target_value_no_restart(SpriteData::random()); animator.reset(); } - self.left_canvas.draw_graph(&self.easing_function,self.color,self.time); - self.right_canvas.draw_sprite(self.properties.get(),self.color); + self.left_canvas.draw_graph(&self.easing_function, self.color, self.time); + self.right_canvas.draw_sprite(self.properties.get(), self.color); } } @@ -209,41 +222,41 @@ impl Sampler { // =============== struct Example { - _animator : animation::Loop> + _animator: animation::Loop>, } impl Example { #[allow(trivial_casts)] - pub fn new - ( name : &str - , ease_in : impl CloneableFnEasing - , ease_out : impl CloneableFnEasing - , ease_in_out : impl CloneableFnEasing + pub fn new( + name: &str, + ease_in: impl CloneableFnEasing, + ease_out: impl CloneableFnEasing, + ease_in_out: impl CloneableFnEasing, ) -> Self { let example = web::create_div(); example.set_attribute_or_panic("id", name); example.set_style_or_panic("margin", "10px"); - let container : HtmlElement = get_element_by_id("examples").unwrap().dyn_into().unwrap(); - let header : HtmlElement = create_element("center").dyn_into().unwrap(); + let container: HtmlElement = get_element_by_id("examples").unwrap().dyn_into().unwrap(); + let header: HtmlElement = create_element("center").dyn_into().unwrap(); header.set_style_or_panic("background-color", "black"); header.set_style_or_panic("color", "white"); header.set_inner_html(name); example.append_or_panic(&header); container.append_or_panic(&example); - let left_canvas = Canvas::new(name); + let left_canvas = Canvas::new(name); let right_canvas = Canvas::new(name); - let mut sampler1 = Sampler::new("green",&left_canvas,&right_canvas,ease_in); - let mut sampler2 = Sampler::new("blue" ,&left_canvas,&right_canvas,ease_out); - let mut sampler3 = Sampler::new("red" ,&left_canvas,&right_canvas,ease_in_out); + let mut sampler1 = Sampler::new("green", &left_canvas, &right_canvas, ease_in); + let mut sampler2 = Sampler::new("blue", &left_canvas, &right_canvas, ease_out); + let mut sampler3 = Sampler::new("red", &left_canvas, &right_canvas, ease_in_out); - let _animator = animation::Loop::new(Box::new(move |time_info:animation::TimeInfo| { + let _animator = animation::Loop::new(Box::new(move |time_info: animation::TimeInfo| { left_canvas.clear(); right_canvas.clear(); sampler1.render(time_info.frame); sampler2.render(time_info.frame); sampler3.render(time_info.frame); }) as Box); - Self {_animator} + Self { _animator } } } @@ -271,5 +284,5 @@ pub fn entry_point_easing_animator() { container.set_style_or_panic("position", "absolute"); container.set_style_or_panic("top", "0px"); web::body().append_or_panic(&container); - examples![expo,bounce,circ,quad,cubic,quart,quint,sine,back,elastic]; + examples![expo, bounce, circ, quad, cubic, quart, quint, sine, back, elastic]; } diff --git a/src/rust/ensogl/example/src/glyph_system.rs b/src/rust/ensogl/example/src/glyph_system.rs index e0c0591744..ef6ec36ce3 100644 --- a/src/rust/ensogl/example/src/glyph_system.rs +++ b/src/rust/ensogl/example/src/glyph_system.rs @@ -5,8 +5,8 @@ use ensogl_core::prelude::*; use ensogl_core::data::color; use ensogl_core::display::world::*; use ensogl_core::system::web; -use ensogl_text_msdf_sys::run_once_initialized; use ensogl_text::typeface::*; +use ensogl_text_msdf_sys::run_once_initialized; use wasm_bindgen::prelude::*; @@ -19,16 +19,16 @@ pub fn entry_point_glyph_system() { run_once_initialized(|| init(&World::new(&web::get_html_element_by_id("root").unwrap()))); } -fn init(world:&World) { - let fonts = world.scene().extension::(); - let font = fonts.load("DejaVuSans"); - let glyph_system = glyph::System::new(world.scene(),font); - let height = 32.0; - let color = color::Rgba::new(0.5,0.0,0.0,1.0); - let glyph = glyph_system.new_glyph(); +fn init(world: &World) { + let fonts = world.scene().extension::(); + let font = fonts.load("DejaVuSans"); + let glyph_system = glyph::System::new(world.scene(), font); + let height = 32.0; + let color = color::Rgba::new(0.5, 0.0, 0.0, 1.0); + let glyph = glyph_system.new_glyph(); glyph.set_char('Q'); glyph.set_color(color); - glyph.size.set(Vector2(height,height)); + glyph.size.set(Vector2(height, height)); world.add_child(&glyph_system); world.add_child(&glyph); diff --git a/src/rust/ensogl/example/src/leak.rs b/src/rust/ensogl/example/src/leak.rs index 73e7b3986d..f50c7e5970 100644 --- a/src/rust/ensogl/example/src/leak.rs +++ b/src/rust/ensogl/example/src/leak.rs @@ -13,14 +13,14 @@ /// leaked when the `Leak` is dropped. #[derive(Debug)] pub struct Leak { - value: Option + value: Option, } impl Leak { /// Constructor. The passed value will be prevented from being dropped. This will cause memory /// leaks. - pub fn new(value:T) -> Self { - Self{value:Some(value)} + pub fn new(value: T) -> Self { + Self { value: Some(value) } } /// Return a reference to the wrapped value. diff --git a/src/rust/ensogl/example/src/lib.rs b/src/rust/ensogl/example/src/lib.rs index b31e365e02..4c1f42d292 100644 --- a/src/rust/ensogl/example/src/lib.rs +++ b/src/rust/ensogl/example/src/lib.rs @@ -20,18 +20,16 @@ #![warn(unsafe_code)] #![warn(unused_import_braces)] #![warn(unused_qualifications)] +#![recursion_limit = "1024"] -#![recursion_limit="1024"] - -#[allow(clippy::option_map_unit_fn)] - -mod leak; pub mod animation; pub mod complex_shape_system; pub mod dom_symbols; pub mod drop_manager; pub mod easing_animator; pub mod glyph_system; +#[allow(clippy::option_map_unit_fn)] +mod leak; pub mod list_view; pub mod mouse_events; pub mod scroll_area; @@ -43,6 +41,6 @@ pub mod text_area; /// Common types that should be visible across the whole crate. pub mod prelude { - pub use ensogl_core::prelude::*; pub use super::leak::*; + pub use ensogl_core::prelude::*; } diff --git a/src/rust/ensogl/example/src/list_view.rs b/src/rust/ensogl/example/src/list_view.rs index e0552242cb..b510f8e7b7 100644 --- a/src/rust/ensogl/example/src/list_view.rs +++ b/src/rust/ensogl/example/src/list_view.rs @@ -2,16 +2,16 @@ use crate::prelude::*; -use ensogl_core::system::web; use ensogl_core::application::Application; use ensogl_core::display::object::ObjectOps; -use ensogl_text_msdf_sys::run_once_initialized; -use ensogl_gui_components::list_view; -use logger::TraceLogger as Logger; -use wasm_bindgen::prelude::*; use ensogl_core::display::Scene; +use ensogl_core::system::web; +use ensogl_gui_components::list_view; use ensogl_text::buffer::data::unit::Bytes; +use ensogl_text_msdf_sys::run_once_initialized; use ensogl_theme as theme; +use logger::TraceLogger as Logger; +use wasm_bindgen::prelude::*; @@ -38,31 +38,33 @@ pub fn entry_point_list_view() { // === Mock Entries === // ==================== -#[derive(Clone,Debug)] +#[derive(Clone, Debug)] struct MockEntries { - logger : Logger, - scene : Scene, - entries_count : usize, + logger: Logger, + scene: Scene, + entries_count: usize, } impl MockEntries { - fn new(app:&Application, entries_count:usize) -> Self { - let logger = Logger::new("MockEntries"); - let scene = app.display.scene().clone_ref(); - Self {logger,scene,entries_count} + fn new(app: &Application, entries_count: usize) -> Self { + let logger = Logger::new("MockEntries"); + let scene = app.display.scene().clone_ref(); + Self { logger, scene, entries_count } } } impl list_view::entry::ModelProvider for MockEntries { - fn entry_count(&self) -> usize { self.entries_count } + fn entry_count(&self) -> usize { + self.entries_count + } - fn get(&self, id:usize) -> Option { + fn get(&self, id: usize) -> Option { if id >= self.entries_count { None } else { let label = iformat!("Entry {id}"); let highlighted = if id == 10 { vec![(Bytes(1)..Bytes(3)).into()] } else { vec![] }; - Some(list_view::entry::GlyphHighlightedLabelModel {label,highlighted}) + Some(list_view::entry::GlyphHighlightedLabelModel { label, highlighted }) } } } @@ -73,20 +75,20 @@ impl list_view::entry::ModelProvider fo // === Init Application === // ======================== -fn init(app:&Application) { +fn init(app: &Application) { theme::builtin::dark::register(&app); theme::builtin::light::register(&app); theme::builtin::light::enable(&app); let list_view = app.new_view::>(); - let provider = list_view::entry::AnyModelProvider::new(MockEntries::new(app,1000)); - list_view.frp.resize(Vector2(100.0,160.0)); + let provider = list_view::entry::AnyModelProvider::new(MockEntries::new(app, 1000)); + list_view.frp.resize(Vector2(100.0, 160.0)); list_view.frp.set_entries(provider); app.display.add_child(&list_view); // FIXME[WD]: This should not be needed after text gets proper depth-handling. app.display.scene().layers.below_main.add_exclusive(&list_view); - let logger : Logger = Logger::new("SelectDebugScene"); + let logger: Logger = Logger::new("SelectDebugScene"); let network = enso_frp::Network::new("test"); enso_frp::extend! {network eval list_view.chosen_entry([logger](entry) { diff --git a/src/rust/ensogl/example/src/mouse_events.rs b/src/rust/ensogl/example/src/mouse_events.rs index 691cf28b97..bbd6a5ebc5 100644 --- a/src/rust/ensogl/example/src/mouse_events.rs +++ b/src/rust/ensogl/example/src/mouse_events.rs @@ -3,7 +3,6 @@ use ensogl_core::prelude::*; use wasm_bindgen::prelude::*; -use ensogl_core::system::web; use ensogl_core::application; use ensogl_core::application::Application; use ensogl_core::data::color; @@ -12,6 +11,7 @@ use ensogl_core::display; use ensogl_core::display::navigation::navigator::Navigator; use ensogl_core::display::object::ObjectOps; use ensogl_core::display::shape::*; +use ensogl_core::system::web; use enso_frp as frp; use ensogl_text_msdf_sys::run_once_initialized; @@ -36,23 +36,23 @@ mod shape { // === Model === // ============= -#[derive(Clone,CloneRef,Debug)] +#[derive(Clone, CloneRef, Debug)] struct Model { - app : Application, - logger : DefaultTraceLogger, - display_object : display::object::Instance, - shape : shape::View, + app: Application, + logger: DefaultTraceLogger, + display_object: display::object::Instance, + shape: shape::View, } impl Model { - fn new(app:&Application) -> Self { - let app = app.clone_ref(); - let logger = DefaultTraceLogger::new("Button"); + fn new(app: &Application) -> Self { + let app = app.clone_ref(); + let logger = DefaultTraceLogger::new("Button"); let display_object = display::object::Instance::new(&logger); - let shape = shape::View::new(&logger); + let shape = shape::View::new(&logger); shape.size.set(Vector2::new(100.0, 100.0)); display_object.add_child(&shape); - Self{app,logger,display_object,shape} + Self { app, logger, display_object, shape } } } @@ -75,18 +75,18 @@ ensogl_core::define_endpoints! { [TRACE_ALL] // === View === // ============ -#[derive(Clone,CloneRef,Debug)] +#[derive(Clone, CloneRef, Debug)] struct View { - frp : Frp, - model : Model, + frp: Frp, + model: Model, } impl View { /// Constructor. pub fn new(app: &Application) -> Self { - let frp = Frp::new(); - let model = Model::new(app); - let events = &model.shape.events; + let frp = Frp::new(); + let model = Model::new(app); + let events = &model.shape.events; let network = &events.network; frp::extend! { network // FIXME [mwu] Currently only `mouse_over` and `mouse_out` events are delivered. @@ -98,27 +98,39 @@ impl View { trace model.shape.events.on_drop; } - Self {frp,model } + Self { frp, model } } } impl display::Object for View { - fn display_object(&self) -> &display::object::Instance { &self.model.display_object } + fn display_object(&self) -> &display::object::Instance { + &self.model.display_object + } } impl Deref for View { type Target = Frp; - fn deref(&self) -> &Self::Target { &self.frp } + fn deref(&self) -> &Self::Target { + &self.frp + } } impl application::command::FrpNetworkProvider for View { - fn network(&self) -> &frp::Network { &self.frp.network } + fn network(&self) -> &frp::Network { + &self.frp.network + } } impl application::View for View { - fn label() -> &'static str { "Circul" } - fn new(app:&Application) -> Self { View::new(app) } - fn app(&self) -> &Application { &self.model.app } + fn label() -> &'static str { + "Circul" + } + fn new(app: &Application) -> Self { + View::new(app) + } + fn app(&self) -> &Application { + &self.model.app + } } @@ -134,13 +146,13 @@ pub fn entry_point_mouse_events() { run_once_initialized(|| { let app = Application::new(&web::get_html_element_by_id("root").unwrap()); - let shape:View = app.new_view(); + let shape: View = app.new_view(); shape.model.shape.size.set(Vector2::new(300.0, 300.0)); app.display.add_child(&shape); - let scene = app.display.scene(); - let camera = scene.camera().clone_ref(); - let navigator = Navigator::new(scene,&camera); + let scene = app.display.scene(); + let camera = scene.camera().clone_ref(); + let navigator = Navigator::new(scene, &camera); std::mem::forget(shape); std::mem::forget(navigator); diff --git a/src/rust/ensogl/example/src/scroll_area.rs b/src/rust/ensogl/example/src/scroll_area.rs index e4d7e4fea6..286135cf6a 100644 --- a/src/rust/ensogl/example/src/scroll_area.rs +++ b/src/rust/ensogl/example/src/scroll_area.rs @@ -6,14 +6,16 @@ use wasm_bindgen::prelude::*; use ensogl_core::application::Application; use ensogl_core::data::color; use ensogl_core::display::object::ObjectOps; -use ensogl_core::system::web; -use ensogl_text_msdf_sys::run_once_initialized; -use ensogl_theme as theme; -use ensogl_gui_components::scroll_area::ScrollArea; -use ensogl_core::display::shape::{Circle, Rect, ShapeSystem}; +use ensogl_core::display::shape::Circle; use ensogl_core::display::shape::PixelDistance; +use ensogl_core::display::shape::Rect; use ensogl_core::display::shape::ShapeOps; +use ensogl_core::display::shape::ShapeSystem; use ensogl_core::display::Sprite; +use ensogl_core::system::web; +use ensogl_gui_components::scroll_area::ScrollArea; +use ensogl_text_msdf_sys::run_once_initialized; +use ensogl_theme as theme; @@ -39,24 +41,24 @@ pub fn entry_point_scroll_area() { // === Init Application === // ======================== -fn init(app:&Application) { +fn init(app: &Application) { theme::builtin::dark::register(&app); theme::builtin::light::register(&app); theme::builtin::light::enable(&app); let scene = app.display.scene(); - scene.camera().set_position_xy(Vector2(100.0,-100.0)); + scene.camera().set_position_xy(Vector2(100.0, -100.0)); // === Background === - let background_color = color::Rgba::new(0.9,0.9,0.9,1.0); - let background_size = (200.px(), 200.px()); - let background_shape = Rect(background_size).corners_radius(5.5.px()).fill(background_color); - let background_system = ShapeSystem::new(scene,background_shape); + let background_color = color::Rgba::new(0.9, 0.9, 0.9, 1.0); + let background_size = (200.px(), 200.px()); + let background_shape = Rect(background_size).corners_radius(5.5.px()).fill(background_color); + let background_system = ShapeSystem::new(scene, background_shape); let background: Sprite = background_system.new_instance(); scene.add_child(&background); - background.size.set(Vector2::new(200.0,200.0)); + background.size.set(Vector2::new(200.0, 200.0)); background.set_position_x(100.0); background.set_position_y(-100.0); std::mem::forget(background); @@ -66,17 +68,17 @@ fn init(app:&Application) { let scroll_area = ScrollArea::new(app); app.display.add_child(&scroll_area); - scroll_area.resize(Vector2(200.0,200.0)); + scroll_area.resize(Vector2(200.0, 200.0)); scroll_area.set_content_width(300.0); scroll_area.set_content_height(1000.0); // === Content === - let sprite_system = ShapeSystem::new(scene,&Circle(50.px())); + let sprite_system = ShapeSystem::new(scene, &Circle(50.px())); let sprite: Sprite = sprite_system.new_instance(); scroll_area.content.add_child(&sprite); - sprite.size.set(Vector2::new(100.0,100.0)); + sprite.size.set(Vector2::new(100.0, 100.0)); sprite.set_position_x(100.0); sprite.set_position_y(-100.0); std::mem::forget(sprite); diff --git a/src/rust/ensogl/example/src/shape_system.rs b/src/rust/ensogl/example/src/shape_system.rs index d8a804590b..a112538fad 100644 --- a/src/rust/ensogl/example/src/shape_system.rs +++ b/src/rust/ensogl/example/src/shape_system.rs @@ -2,14 +2,14 @@ use ensogl_core::prelude::*; +use ensogl_core::data::color; use ensogl_core::display::navigation::navigator::Navigator; -use ensogl_core::system::web; -use wasm_bindgen::prelude::*; use ensogl_core::display::object::ObjectOps; use ensogl_core::display::shape::ShapeSystem; -use ensogl_core::display::world::*; use ensogl_core::display::shape::*; -use ensogl_core::data::color; +use ensogl_core::display::world::*; +use ensogl_core::system::web; +use wasm_bindgen::prelude::*; @@ -19,12 +19,12 @@ use ensogl_core::data::color; /// The shape definition. pub fn shape() -> AnyShape { - let circle1 = Circle(50.px()); - let circle_bg = circle1.translate_x(-(50.0.px())); + let circle1 = Circle(50.px()); + let circle_bg = circle1.translate_x(-(50.0.px())); let circle_sub = circle1.translate_y(-(50.0.px())); - let rect = Rect((100.0.px(),100.0.px())); - let shape = circle_bg + rect - circle_sub; - let shape = shape.fill(color::Rgb::new(1.0,0.0,0.0)); + let rect = Rect((100.0.px(), 100.0.px())); + let shape = circle_bg + rect - circle_sub; + let shape = shape.fill(color::Rgb::new(1.0, 0.0, 0.0)); shape.into() } @@ -40,12 +40,12 @@ pub fn shape() -> AnyShape { pub fn entry_point_shape_system() { web::forward_panic_hook_to_console(); - let world = World::new(&web::get_html_element_by_id("root").unwrap()); - let scene = world.scene(); - let camera = scene.camera().clone_ref(); - let navigator = Navigator::new(scene,&camera); - let sprite_system = ShapeSystem::new(&world,&shape()); - let sprite = sprite_system.new_instance(); + let world = World::new(&web::get_html_element_by_id("root").unwrap()); + let scene = world.scene(); + let camera = scene.camera().clone_ref(); + let navigator = Navigator::new(scene, &camera); + let sprite_system = ShapeSystem::new(&world, &shape()); + let sprite = sprite_system.new_instance(); sprite.size.set(Vector2::new(300.0, 300.0)); sprite.mod_position(|t| *t = Vector3::new(50.0, 50.0, 0.0)); @@ -53,8 +53,10 @@ pub fn entry_point_shape_system() { world.add_child(&sprite_system); world.keep_alive_forever(); - world.on_frame(move |_time| { - let _keep_alive = &sprite; - let _keep_alive = &navigator; - }).forget(); + world + .on_frame(move |_time| { + let _keep_alive = &sprite; + let _keep_alive = &navigator; + }) + .forget(); } diff --git a/src/rust/ensogl/example/src/slider.rs b/src/rust/ensogl/example/src/slider.rs index 72eac2af88..1a07dd09ab 100644 --- a/src/rust/ensogl/example/src/slider.rs +++ b/src/rust/ensogl/example/src/slider.rs @@ -7,8 +7,8 @@ use ensogl_core::application::Application; use ensogl_core::data::color; use ensogl_core::display::object::ObjectOps; use ensogl_core::system::web; -use ensogl_gui_components::selector::Bounds; use ensogl_gui_components::selector; +use ensogl_gui_components::selector::Bounds; use ensogl_text_msdf_sys::run_once_initialized; use ensogl_theme as theme; @@ -31,16 +31,16 @@ pub fn entry_point_slider() { }); } -fn make_number_picker(app:&Application) -> Leak { +fn make_number_picker(app: &Application) -> Leak { let slider = app.new_view::(); - slider.frp.resize(Vector2(200.0,50.0)); + slider.frp.resize(Vector2(200.0, 50.0)); app.display.add_child(&slider); Leak::new(slider) } -fn make_range_picker(app:&Application) -> Leak { +fn make_range_picker(app: &Application) -> Leak { let slider = app.new_view::(); - slider.frp.resize(Vector2(400.0,50.0)); + slider.frp.resize(Vector2(400.0, 50.0)); app.display.add_child(&slider); Leak::new(slider) } @@ -51,7 +51,7 @@ fn make_range_picker(app:&Application) -> Leak { // === Init Application === // ======================== -fn init(app:&Application) { +fn init(app: &Application) { theme::builtin::dark::register(&app); theme::builtin::light::register(&app); theme::builtin::light::enable(&app); @@ -60,19 +60,19 @@ fn init(app:&Application) { slider1.inner().frp.allow_click_selection(true); let slider2 = make_number_picker(app); - slider2.inner().frp.resize(Vector2(400.0,50.0)); - slider2.inner().frp.set_bounds.emit(Bounds::new(-100.0,100.0)); + slider2.inner().frp.resize(Vector2(400.0, 50.0)); + slider2.inner().frp.set_bounds.emit(Bounds::new(-100.0, 100.0)); slider2.inner().set_position_y(50.0); - slider2.inner().frp.use_overflow_bounds(Bounds::new(-150.0,200.0)); + slider2.inner().frp.use_overflow_bounds(Bounds::new(-150.0, 200.0)); slider2.inner().frp.set_caption(Some("Value:".to_string())); let slider3 = make_range_picker(app); slider3.inner().set_position_y(-100.0); - slider3.inner().set_track_color(color::Rgba::new(0.0,0.80,0.80,1.0)); + slider3.inner().set_track_color(color::Rgba::new(0.0, 0.80, 0.80, 1.0)); let slider4 = make_range_picker(app); slider4.inner().set_position_y(-200.0); - slider4.inner().frp.use_overflow_bounds(Bounds::new(-2.0,3.0)); + slider4.inner().frp.use_overflow_bounds(Bounds::new(-2.0, 3.0)); slider4.inner().frp.set_caption(Some("Caption".to_string())); - slider4.inner().set_track_color(color::Rgba::new(0.5,0.70,0.70,1.0)); + slider4.inner().set_track_color(color::Rgba::new(0.5, 0.70, 0.70, 1.0)); } diff --git a/src/rust/ensogl/example/src/sprite_system.rs b/src/rust/ensogl/example/src/sprite_system.rs index a824a69d78..024eca26e4 100644 --- a/src/rust/ensogl/example/src/sprite_system.rs +++ b/src/rust/ensogl/example/src/sprite_system.rs @@ -6,8 +6,8 @@ use wasm_bindgen::prelude::*; use ensogl_core::display::navigation::navigator::Navigator; use ensogl_core::display::symbol::geometry::SpriteSystem; use ensogl_core::display::world::*; -use ensogl_core::system::web::forward_panic_hook_to_console; use ensogl_core::system::web; +use ensogl_core::system::web::forward_panic_hook_to_console; #[wasm_bindgen] @@ -15,30 +15,32 @@ use ensogl_core::system::web; pub fn entry_point_sprite_system() { forward_panic_hook_to_console(); - let world = World::new(&web::get_html_element_by_id("root").unwrap()); - let scene = world.scene(); - let camera = scene.camera().clone_ref(); - let navigator = Navigator::new(scene,&camera); + let world = World::new(&web::get_html_element_by_id("root").unwrap()); + let scene = world.scene(); + let camera = scene.camera().clone_ref(); + let navigator = Navigator::new(scene, &camera); let sprite_system = SpriteSystem::new(&world); - let sprite2 = sprite_system.new_instance(); - let sprite1 = sprite_system.new_instance(); - sprite1.size.set(Vector2::new(15.0,15.0)); - sprite2.size.set(Vector2::new(15.0,15.0)); + let sprite2 = sprite_system.new_instance(); + let sprite1 = sprite_system.new_instance(); + sprite1.size.set(Vector2::new(15.0, 15.0)); + sprite2.size.set(Vector2::new(15.0, 15.0)); scene.add_child(&sprite_system); world.keep_alive_forever(); let mut i = 0; - world.on_frame(move |_| { - i += 1; - let _keep_alive = &navigator; - let _keep_alive = &sprite1; - let _keep_alive = &sprite2; - let _keep_alive = &sprite_system; - if i <= 100 { - sprite1.mod_position(|p| p.x += 1.0); - sprite2.mod_position(|p| p.y += 1.0); - } - }).forget(); + world + .on_frame(move |_| { + i += 1; + let _keep_alive = &navigator; + let _keep_alive = &sprite1; + let _keep_alive = &sprite2; + let _keep_alive = &sprite_system; + if i <= 100 { + sprite1.mod_position(|p| p.x += 1.0); + sprite2.mod_position(|p| p.y += 1.0); + } + }) + .forget(); } diff --git a/src/rust/ensogl/example/src/sprite_system_benchmark.rs b/src/rust/ensogl/example/src/sprite_system_benchmark.rs index 58a664b668..5defa11ccd 100644 --- a/src/rust/ensogl/example/src/sprite_system_benchmark.rs +++ b/src/rust/ensogl/example/src/sprite_system_benchmark.rs @@ -2,15 +2,15 @@ use ensogl_core::traits::*; +use ensogl_core::animation; use ensogl_core::display::camera::Camera2d; use ensogl_core::display::navigation::navigator::Navigator; use ensogl_core::display::symbol::geometry::Sprite; use ensogl_core::display::symbol::geometry::SpriteSystem; use ensogl_core::display::world::*; use ensogl_core::prelude::*; -use ensogl_core::system::web::forward_panic_hook_to_console; use ensogl_core::system::web; -use ensogl_core::animation; +use ensogl_core::system::web::forward_panic_hook_to_console; use nalgebra::Vector2; use nalgebra::Vector3; use wasm_bindgen::prelude::*; @@ -21,65 +21,67 @@ use wasm_bindgen::prelude::*; pub fn entry_point_sprite_system_benchmark() { forward_panic_hook_to_console(); - let world = World::new(&web::get_html_element_by_id("root").unwrap()); - let scene = world.scene(); - let camera = scene.camera().clone_ref(); - let navigator = Navigator::new(scene,&camera); + let world = World::new(&web::get_html_element_by_id("root").unwrap()); + let scene = world.scene(); + let camera = scene.camera().clone_ref(); + let navigator = Navigator::new(scene, &camera); let sprite_system = SpriteSystem::new(&world); - let sprite1 = sprite_system.new_instance(); - sprite1.size.set(Vector2::new(10.0,10.0)); - sprite1.mod_position(|t| *t = Vector3::new(5.0,5.0,0.0)); + let sprite1 = sprite_system.new_instance(); + sprite1.size.set(Vector2::new(10.0, 10.0)); + sprite1.mod_position(|t| *t = Vector3::new(5.0, 5.0, 0.0)); scene.add_child(&sprite_system); let mut sprites: Vec = default(); let count = 100; - for _ in 0 .. count { + for _ in 0..count { let sprite = sprite_system.new_instance(); - sprite.size.set(Vector2::new(1.0,1.0)); + sprite.size.set(Vector2::new(1.0, 1.0)); sprites.push(sprite); } world.keep_alive_forever(); - let mut iter:i32 = 0; + let mut iter: i32 = 0; let mut i = 0; - world.on_frame(move |time| { - i += 1; - if i <= 100 { - sprite1.mod_position(|p| p.x += 1.0); - } - let _keep_alive = &camera; - let _keep_alive = &iter; - let _keep_alive = &sprite1; - let _keep_alive = &sprites; - let _keep_alive = &sprite_system; - let _keep_alive = &navigator; - // FIXME: these logs crash gui after some time! - - // println!("sprite count: {:?}",sprites.len()); - // println!("sprite_system is visible? {:?}",sprite_system.is_visible()); - // println!("sprite[5] is visible? {:?}",sprites[5].is_visible()); - - on_frame(&camera,time,&mut iter,&sprite1,&mut sprites,&sprite_system) - }).forget(); + world + .on_frame(move |time| { + i += 1; + if i <= 100 { + sprite1.mod_position(|p| p.x += 1.0); + } + let _keep_alive = &camera; + let _keep_alive = &iter; + let _keep_alive = &sprite1; + let _keep_alive = &sprites; + let _keep_alive = &sprite_system; + let _keep_alive = &navigator; + // FIXME: these logs crash gui after some time! + + // println!("sprite count: {:?}",sprites.len()); + // println!("sprite_system is visible? {:?}",sprite_system.is_visible()); + // println!("sprite[5] is visible? {:?}",sprites[5].is_visible()); + + on_frame(&camera, time, &mut iter, &sprite1, &mut sprites, &sprite_system) + }) + .forget(); } #[allow(clippy::too_many_arguments)] #[allow(clippy::many_single_char_names)] -pub fn on_frame -( camera : &Camera2d -, time : animation::TimeInfo -, iter : &mut i32 -, sprite1 : &Sprite -, sprites : &mut Vec -, sprite_system : &SpriteSystem +pub fn on_frame( + camera: &Camera2d, + time: animation::TimeInfo, + iter: &mut i32, + sprite1: &Sprite, + sprites: &mut Vec, + sprite_system: &SpriteSystem, ) { *iter += 1; - let cycle_duration = 300; - let pause_duration = 100; + let cycle_duration = 300; + let pause_duration = 100; let sprite_diff_per_cycle = 100; let mut frozen = false; @@ -87,7 +89,7 @@ pub fn on_frame if *iter < cycle_duration { for _ in 0..sprite_diff_per_cycle { let sprite = sprite_system.new_instance(); - sprite.size.set(Vector2::new(1.0,1.0)); + sprite.size.set(Vector2::new(1.0, 1.0)); sprites.push(sprite); } } else if *iter < pause_duration + cycle_duration { @@ -122,7 +124,11 @@ pub fn on_frame y += (z * 1.25 + t * 2.00).cos() * 0.5; z += (x * 1.25 + t * 3.25).cos() * 0.5; - let position = Vector3::new(x * 150.0 + half_width - 75.0, y * 150.0 + half_height - 75.0, z * 150.0); + let position = Vector3::new( + x * 150.0 + half_width - 75.0, + y * 150.0 + half_height - 75.0, + z * 150.0, + ); sprite.set_position(position); } } diff --git a/src/rust/ensogl/example/src/text_area.rs b/src/rust/ensogl/example/src/text_area.rs index 10ca2d1689..66de64784f 100644 --- a/src/rust/ensogl/example/src/text_area.rs +++ b/src/rust/ensogl/example/src/text_area.rs @@ -2,12 +2,12 @@ use ensogl_core::prelude::*; +use ensogl_core::application::Application; +use ensogl_core::display::navigation::navigator::Navigator; use ensogl_core::system::web; +use ensogl_text::Area; use ensogl_text_msdf_sys::run_once_initialized; use wasm_bindgen::prelude::*; -use ensogl_core::application::Application; -use ensogl_text::Area; -use ensogl_core::display::navigation::navigator::Navigator; /// Main example runner. @@ -23,21 +23,25 @@ pub fn entry_point_text_area() { }); } -fn init(app:&Application) { +fn init(app: &Application) { let area = app.new_view::(); area.set_position_x(-100.0); - area.set_content("Et Eärello Endorenna utúlien.\nSinome maruvan ar Hildinyar tenn' Ambar-metta"); + area.set_content( + "Et Eärello Endorenna utúlien.\nSinome maruvan ar Hildinyar tenn' Ambar-metta", + ); area.focus(); area.hover(); area.set_cursor_at_end(); - let scene = app.display.scene(); - let navigator = Navigator::new(scene,&scene.camera()); + let scene = app.display.scene(); + let navigator = Navigator::new(scene, &scene.camera()); app.display.scene().add_child(&area); let keep = Some(area); - app.display.on_frame(move |_frame| { - let _ = &keep; - }).forget(); + app.display + .on_frame(move |_frame| { + let _ = &keep; + }) + .forget(); std::mem::forget(navigator); } diff --git a/src/rust/ensogl/lib/components/src/component.rs b/src/rust/ensogl/lib/components/src/component.rs index fde2e3d4f0..752d0c447a 100644 --- a/src/rust/ensogl/lib/components/src/component.rs +++ b/src/rust/ensogl/lib/components/src/component.rs @@ -9,11 +9,11 @@ use crate::prelude::*; use enso_frp as frp; -use ensogl_core::application::Application; -use ensogl_core::application::command::CommandApi; use ensogl_core::application; -use ensogl_core::display::shape::*; +use ensogl_core::application::command::CommandApi; +use ensogl_core::application::Application; use ensogl_core::display; +use ensogl_core::display::shape::*; @@ -26,7 +26,7 @@ use ensogl_core::display; /// `Component`. pub trait Model { /// Constructor. - fn new(app:&Application) -> Self; + fn new(app: &Application) -> Self; } @@ -38,9 +38,9 @@ pub trait Model { /// Frp that can be used in a Component. The FRP requires an initializer that will be called during /// the construction of the component. `Default` + `CommandApi` are usually implemented when using /// the `ensogl_core::define_endpoints!` macro to create an FRP API. -pub trait Frp : Default + CommandApi { +pub trait Frp: Default + CommandApi { /// Frp initializer. - fn init(&self, app:&Application, model:&Model, style:&StyleWatchFrp); + fn init(&self, app: &Application, model: &Model, style: &StyleWatchFrp); } @@ -51,43 +51,48 @@ pub trait Frp : Default + CommandApi { /// Base struct for UI components in EnsoGL. Contains the Data/Shape model and the FPR exposing its /// behaviour. -#[derive(CloneRef,Debug,Derivative)] -#[derivative(Clone(bound=""))] -pub struct Component { +#[derive(CloneRef, Debug, Derivative)] +#[derivative(Clone(bound = ""))] +pub struct Component { /// Public FRP api of the Component. - pub frp : Rc, - model : Rc, + pub frp: Rc, + model: Rc, /// Reference to the application the Component belongs to. Generally required for implementing /// `application::View` and initialising the `Mode`l and `Frp` and thus provided by the /// `Component`. - pub app : Application, + pub app: Application, } -impl> Component { +impl> Component { /// Constructor. - pub fn new(app:&Application) -> Self { - let app = app.clone_ref(); + pub fn new(app: &Application) -> Self { + let app = app.clone_ref(); let model = Rc::new(M::new(&app)); - let frp = F::default(); + let frp = F::default(); let style = StyleWatchFrp::new(&app.display.scene().style_sheet); - frp.init(&app,&model,&style); - let frp = Rc::new(frp); - Self{frp,model,app} + frp.init(&app, &model, &style); + let frp = Rc::new(frp); + Self { frp, model, app } } } -impl display::Object for Component { +impl display::Object for Component { fn display_object(&self) -> &display::object::Instance { self.model.display_object() } } -impl> Deref for Component { +impl> Deref for Component { type Target = F; - fn deref(&self) -> &Self::Target { &self.frp } + fn deref(&self) -> &Self::Target { + &self.frp + } } -impl application::command::FrpNetworkProvider -for Component { - fn network(&self) -> &frp::Network { self.frp.network() } +impl application::command::FrpNetworkProvider + for Component +{ + fn network(&self) -> &frp::Network { + self.frp.network() + } } diff --git a/src/rust/ensogl/lib/components/src/drop_down_menu.rs b/src/rust/ensogl/lib/components/src/drop_down_menu.rs index 10e24f152d..b6072f3955 100644 --- a/src/rust/ensogl/lib/components/src/drop_down_menu.rs +++ b/src/rust/ensogl/lib/components/src/drop_down_menu.rs @@ -6,12 +6,12 @@ use crate::list_view::entry::ModelProvider; use enso_frp as frp; use enso_frp; -use ensogl_core::DEPRECATED_Animation; use ensogl_core::application::Application; use ensogl_core::data::color; -use ensogl_core::display::shape::*; -use ensogl_core::display::shape::primitive::StyleWatch; use ensogl_core::display; +use ensogl_core::display::shape::primitive::StyleWatch; +use ensogl_core::display::shape::*; +use ensogl_core::DEPRECATED_Animation; use ensogl_text as text; use ensogl_theme as theme; @@ -21,9 +21,9 @@ use ensogl_theme as theme; // ================= /// Invisible dummy color to catch hover events. -const HOVER_COLOR : color::Rgba = color::Rgba::new(1.0,0.0,0.0,0.000_001); +const HOVER_COLOR: color::Rgba = color::Rgba::new(1.0, 0.0, 0.0, 0.000_001); /// The width of the visualisation selection menu. -const MENU_WIDTH : f32 = 180.0; +const MENU_WIDTH: f32 = 180.0; @@ -99,34 +99,35 @@ ensogl_core::define_endpoints! { /// A type of Entry used in DropDownMenu's ListView. pub type Entry = list_view::entry::Label; -#[derive(Clone,Debug)] +#[derive(Clone, Debug)] struct Model { - logger : Logger, - app : Application, - display_object : display::object::Instance, + logger: Logger, + app: Application, + display_object: display::object::Instance, - icon : arrow::View, - icon_overlay : chooser_hover_area::View, + icon: arrow::View, + icon_overlay: chooser_hover_area::View, - label : text::Area, - selection_menu : list_view::ListView, + label: text::Area, + selection_menu: list_view::ListView, // `SingleMaskedProvider` allows us to hide the selected element. - content : RefCell>>, + content: RefCell>>, } impl Model { - fn new(app:&Application) -> Self { - let logger = Logger::new("drop_down_menu"); - let app = app.clone_ref(); + fn new(app: &Application) -> Self { + let logger = Logger::new("drop_down_menu"); + let app = app.clone_ref(); let display_object = display::object::Instance::new(&logger); - let icon = arrow::View::new(&logger); - let icon_overlay = chooser_hover_area::View::new(&logger); + let icon = arrow::View::new(&logger); + let icon_overlay = chooser_hover_area::View::new(&logger); let selection_menu = list_view::ListView::new(&app); - let label = app.new_view::(); - let content = default(); + let label = app.new_view::(); + let content = default(); - Self{logger,app,display_object,icon,icon_overlay,label,selection_menu,content}.init() + Self { logger, app, display_object, icon, icon_overlay, label, selection_menu, content } + .init() } fn init(self) -> Self { @@ -138,7 +139,7 @@ impl Model { self } - fn set_label(&self, label:&str) { + fn set_label(&self, label: &str) { self.label.set_cursor(&default()); self.label.select_all(); self.label.insert(label); @@ -153,8 +154,10 @@ impl Model { self.selection_menu.unset_parent() } - fn get_content_item - (&self, id:Option) -> Option<::Model> { + fn get_content_item( + &self, + id: Option, + ) -> Option<::Model> { self.content.borrow().as_ref()?.get(id?) } @@ -166,7 +169,7 @@ impl Model { /// Item list [A, B, C] /// Unmasked index [0, 1, 2] /// Masked indices [0, na, 1] - fn get_unmasked_index(&self, ix:Option) -> Option { + fn get_unmasked_index(&self, ix: Option) -> Option { Some(self.content.borrow().as_ref()?.unmasked_index(ix?)) } } @@ -185,32 +188,34 @@ impl display::Object for Model { /// UI entity that shows a button that opens a list of visualisations that can be selected from. #[allow(missing_docs)] -#[derive(Clone,CloneRef,Debug)] +#[derive(Clone, CloneRef, Debug)] pub struct DropDownMenu { - model : Rc, - pub frp : Frp, + model: Rc, + pub frp: Frp, } impl Deref for DropDownMenu { type Target = Frp; - fn deref(&self) -> &Self::Target { &self.frp } + fn deref(&self) -> &Self::Target { + &self.frp + } } impl DropDownMenu { /// Constructor. - pub fn new(app:&Application) -> Self { - let frp = Frp::new(); + pub fn new(app: &Application) -> Self { + let frp = Frp::new(); let model = Rc::new(Model::new(app)); - Self {model,frp}.init(app) + Self { model, frp }.init(app) } - fn init(self, app:&Application) -> Self { + fn init(self, app: &Application) -> Self { let network = &self.frp.network; - let frp = &self.frp; - let model = &self.model; + let frp = &self.frp; + let model = &self.model; - let scene = app.display.scene(); - let mouse = &scene.mouse.frp; + let scene = app.display.scene(); + let mouse = &scene.mouse.frp; frp::extend! { network @@ -357,7 +362,7 @@ impl DropDownMenu { // FIXME : StyleWatch is unsuitable here, as it was designed as an internal tool for // shape system (#795) - let styles = StyleWatch::new(&app.display.scene().style_sheet); + let styles = StyleWatch::new(&app.display.scene().style_sheet); let text_color = styles.get_color(theme::widget::list_view::text); model.label.set_default_color(text_color); diff --git a/src/rust/ensogl/lib/components/src/file_browser.rs b/src/rust/ensogl/lib/components/src/file_browser.rs index fe59090a24..d0fafd2e28 100644 --- a/src/rust/ensogl/lib/components/src/file_browser.rs +++ b/src/rust/ensogl/lib/components/src/file_browser.rs @@ -7,10 +7,10 @@ use crate::prelude::*; use crate::file_browser::model::*; -use ensogl_core::display::shape::*; -use std::path::PathBuf; use ensogl_core::display; +use ensogl_core::display::shape::*; use ensogl_core::display::Scene; +use std::path::PathBuf; // =========== @@ -47,33 +47,39 @@ ensogl_core::define_endpoints! { /// A file browser component. It allows to browse the content of a folder and it's subfolders and /// emits an event when an entry is chosen. -#[derive(Clone,CloneRef,Debug)] +#[derive(Clone, CloneRef, Debug)] pub struct FileBrowser { - logger : Logger, - frp : Frp, - display_object : display::object::Instance, + logger: Logger, + frp: Frp, + display_object: display::object::Instance, } impl Deref for FileBrowser { type Target = Frp; - fn deref(&self) -> &Self::Target { &self.frp } + fn deref(&self) -> &Self::Target { + &self.frp + } } impl FileBrowser { /// Constructore pub fn new() -> Self { - let logger = Logger::new("FileBrowser"); - let frp = Frp::new(); + let logger = Logger::new("FileBrowser"); + let frp = Frp::new(); let display_object = display::object::Instance::new(&logger); - Self {logger,frp,display_object} + Self { logger, frp, display_object } } } impl Default for FileBrowser { - fn default() -> Self { Self::new() } + fn default() -> Self { + Self::new() + } } impl display::Object for FileBrowser { - fn display_object(&self) -> &display::object::Instance {&self.display_object } + fn display_object(&self) -> &display::object::Instance { + &self.display_object + } } diff --git a/src/rust/ensogl/lib/components/src/file_browser/model.rs b/src/rust/ensogl/lib/components/src/file_browser/model.rs index 721994e27f..ce8fc4447c 100644 --- a/src/rust/ensogl/lib/components/src/file_browser/model.rs +++ b/src/rust/ensogl/lib/components/src/file_browser/model.rs @@ -4,8 +4,8 @@ use crate::prelude::*; use enso_frp as frp; -use std::path::PathBuf; use std::cmp::Ordering; +use std::path::PathBuf; // ============= @@ -16,7 +16,7 @@ use std::cmp::Ordering; /// The type of a folder. This is used to distinguish standard folders from the different kinds of /// content roots. -#[derive(Debug,Copy,Clone,Eq,Ord,PartialEq,PartialOrd)] +#[derive(Debug, Copy, Clone, Eq, Ord, PartialEq, PartialOrd)] pub enum FolderType { /// A normal sufolder in the file system. Standard, @@ -34,49 +34,54 @@ pub enum FolderType { /// The type of a file system entry. Distinguishes files from the different kind of folders. The /// `EntryType` of a folder also caries the folder's content. -#[derive(Clone,Debug)] +#[derive(Clone, Debug)] pub enum EntryType { /// A file. File, /// A folder. This can also mean a content root. Folder { /// The folder type. - type_ : FolderType, + type_: FolderType, /// The folder's content. - content : AnyFolderContent, + content: AnyFolderContent, }, } impl Ord for EntryType { - fn cmp(&self, other:&Self) -> Ordering { - match (self,other) { - (Self::File,Self::File) => Ordering::Equal, - (Self::File,Self::Folder {..}) => Ordering::Greater, - (Self::Folder {..},Self::File) => Ordering::Less, - (Self::Folder {type_:type1,..},Self::Folder {type_:type2,..}) => type1.cmp(type2), + fn cmp(&self, other: &Self) -> Ordering { + match (self, other) { + (Self::File, Self::File) => Ordering::Equal, + (Self::File, Self::Folder { .. }) => Ordering::Greater, + (Self::Folder { .. }, Self::File) => Ordering::Less, + (Self::Folder { type_: type1, .. }, Self::Folder { type_: type2, .. }) => + type1.cmp(type2), } } } impl PartialOrd for EntryType { - fn partial_cmp(&self, other: &Self) -> Option { Some(self.cmp(other)) } + fn partial_cmp(&self, other: &Self) -> Option { + Some(self.cmp(other)) + } } impl PartialEq for EntryType { - fn eq(&self, other: &Self) -> bool { self.cmp(other) == Ordering::Equal } + fn eq(&self, other: &Self) -> bool { + self.cmp(other) == Ordering::Equal + } } impl Eq for EntryType {} /// A file system entry. Either a file or a folder. -#[derive(Debug,Clone,Eq,Ord,PartialEq,PartialOrd)] +#[derive(Debug, Clone, Eq, Ord, PartialEq, PartialOrd)] pub struct Entry { /// The entry type. - pub type_ : EntryType, + pub type_: EntryType, /// The entrie's name. - pub name : String, + pub name: String, /// The entrie's global path in the file system. - pub path : PathBuf, + pub path: PathBuf, } @@ -87,13 +92,16 @@ pub struct Entry { pub trait FolderContent: Debug { /// Request the list of entries inside the folder. When the list is ready, it is emitted at /// `entries_loaded`. If an error occurs then the error message is emitted at `error_occurred`. - fn request_entries - (&self, entries_loaded:frp::Any>>, error_occurred:frp::Any); + fn request_entries( + &self, + entries_loaded: frp::Any>>, + error_occurred: frp::Any, + ); } /// A wrapper around `Rc`. Necessary to implement the `Default` trait on this /// type, which we need to pass it through FRP networks. -#[derive(Debug,Clone)] +#[derive(Debug, Clone)] pub struct AnyFolderContent(Rc); impl Deref for AnyFolderContent { @@ -104,7 +112,7 @@ impl Deref for AnyFolderContent { } } -impl From for AnyFolderContent { +impl From for AnyFolderContent { fn from(dir: D) -> Self { AnyFolderContent(Rc::new(dir)) } @@ -114,12 +122,15 @@ impl From for AnyFolderContent { // === EmptyFolder === /// `FolderContent` that immediately provides an empty content list on request. -#[derive(Debug,Copy,Clone)] +#[derive(Debug, Copy, Clone)] pub struct EmptyFolderContent; impl FolderContent for EmptyFolderContent { - fn request_entries - (&self, entries_loaded:frp::Any>>, _error_occured:frp::Any) { + fn request_entries( + &self, + entries_loaded: frp::Any>>, + _error_occured: frp::Any, + ) { entries_loaded.emit(Rc::new(vec![])); } } diff --git a/src/rust/ensogl/lib/components/src/label.rs b/src/rust/ensogl/lib/components/src/label.rs index 8e31f140bf..92fee471aa 100644 --- a/src/rust/ensogl/lib/components/src/label.rs +++ b/src/rust/ensogl/lib/components/src/label.rs @@ -7,9 +7,9 @@ use enso_frp as frp; use enso_frp; use ensogl_core::application::Application; use ensogl_core::data::color; +use ensogl_core::display; use ensogl_core::display::shape::*; use ensogl_core::display::traits::*; -use ensogl_core::display; use ensogl_text as text; use ensogl_theme::component::label as theme; @@ -64,23 +64,23 @@ ensogl_core::define_endpoints! { // === Model === // ============= -#[derive(Clone,Debug)] +#[derive(Clone, Debug)] struct Model { - background : background::View, - label : text::Area, - display_object : display::object::Instance, - app : Application, - style : StyleWatch, + background: background::View, + label: text::Area, + display_object: display::object::Instance, + app: Application, + style: StyleWatch, } impl Model { fn new(app: Application) -> Self { - let app = app.clone_ref(); - let scene = app.display.scene(); - let logger = Logger::new("TextLabel"); + let app = app.clone_ref(); + let scene = app.display.scene(); + let logger = Logger::new("TextLabel"); let display_object = display::object::Instance::new(&logger); - let label = app.new_view::(); - let background = background::View::new(&logger); + let label = app.new_view::(); + let background = background::View::new(&logger); // FIXME[MM/WD]: Depth sorting of labels to in front of everything else in the scene. // Temporary solution. The depth management needs to allow defining relative position of @@ -94,44 +94,44 @@ impl Model { let style = StyleWatch::new(&app.display.scene().style_sheet); - Model {background,label,display_object,app,style} + Model { background, label, display_object, app, style } } pub fn height(&self) -> f32 { self.style.get_number(theme::height) } - fn set_width(&self, width:f32) -> Vector2 { - let padding_outer = self.style.get_number(theme::padding_outer); + fn set_width(&self, width: f32) -> Vector2 { + let padding_outer = self.style.get_number(theme::padding_outer); let padding_inner_x = self.style.get_number(theme::padding_inner_x); let padding_inner_y = self.style.get_number(theme::padding_inner_y); - let padding_x = padding_outer + padding_inner_x; - let padding_y = padding_outer + padding_inner_y; - let padding = Vector2(padding_x,padding_y); - let text_size = self.style.get_number(theme::text::size); - let text_offset = self.style.get_number(theme::text::offset); - let height = self.height(); - let size = Vector2(width,height); - let padded_size = size + padding * 2.0; + let padding_x = padding_outer + padding_inner_x; + let padding_y = padding_outer + padding_inner_y; + let padding = Vector2(padding_x, padding_y); + let text_size = self.style.get_number(theme::text::size); + let text_offset = self.style.get_number(theme::text::offset); + let height = self.height(); + let size = Vector2(width, height); + let padded_size = size + padding * 2.0; self.background.size.set(padded_size); - let text_origin = Vector2(text_offset - size.x/2.0, text_size /2.0); + let text_origin = Vector2(text_offset - size.x / 2.0, text_size / 2.0); self.label.set_position_xy(text_origin); padded_size } - fn set_content(&self, t:&str) -> Vector2 { + fn set_content(&self, t: &str) -> Vector2 { self.label.set_content(t); self.set_width(self.label.width.value()) } - fn set_opacity(&self, value:f32) { + fn set_opacity(&self, value: f32) { let text_color_path = theme::text; - let text_color = self.style.get_color(text_color_path).multiply_alpha(value); + let text_color = self.style.get_color(text_color_path).multiply_alpha(value); self.label.frp.set_color_all.emit(text_color); self.label.frp.set_default_color.emit(text_color); let bg_color_path = theme::background; - let bg_color = self.style.get_color(bg_color_path).multiply_alpha(value); + let bg_color = self.style.get_color(bg_color_path).multiply_alpha(value); self.background.bg_color.set(bg_color.into()) } } @@ -143,24 +143,24 @@ impl Model { // ======================= #[allow(missing_docs)] -#[derive(Clone,CloneRef,Debug)] +#[derive(Clone, CloneRef, Debug)] pub struct Label { - model : Rc, - pub frp : Rc, + model: Rc, + pub frp: Rc, } impl Label { /// Constructor. - pub fn new(app:&Application) -> Self { - let frp = Rc::new(Frp::new()); + pub fn new(app: &Application) -> Self { + let frp = Rc::new(Frp::new()); let model = Rc::new(Model::new(app.clone_ref())); - Label {model,frp}.init() + Label { model, frp }.init() } fn init(self) -> Self { - let frp = &self.frp; + let frp = &self.frp; let network = &frp.network; - let model = &self.model; + let model = &self.model; frp::extend! { network frp.source.size <+ frp.set_content.map(f!((t) @@ -175,5 +175,7 @@ impl Label { } impl display::Object for Label { - fn display_object(&self) -> &display::object::Instance { &self.model.display_object } + fn display_object(&self) -> &display::object::Instance { + &self.model.display_object + } } diff --git a/src/rust/ensogl/lib/components/src/lib.rs b/src/rust/ensogl/lib/components/src/lib.rs index a20ccded88..427b2eafde 100644 --- a/src/rust/ensogl/lib/components/src/lib.rs +++ b/src/rust/ensogl/lib/components/src/lib.rs @@ -4,7 +4,6 @@ #![feature(option_result_contains)] #![feature(trait_alias)] - #![warn(missing_copy_implementations)] #![warn(missing_debug_implementations)] #![warn(missing_docs)] @@ -13,8 +12,7 @@ #![warn(unsafe_code)] #![warn(unused_import_braces)] #![warn(unused_qualifications)] - -#![recursion_limit="512"] +#![recursion_limit = "512"] pub mod component; pub mod drop_down_menu; diff --git a/src/rust/ensogl/lib/components/src/list_view.rs b/src/rust/ensogl/lib/components/src/list_view.rs index 691eb97fa5..e199425225 100644 --- a/src/rust/ensogl/lib/components/src/list_view.rs +++ b/src/rust/ensogl/lib/components/src/list_view.rs @@ -10,8 +10,8 @@ use crate::shadow; use enso_frp as frp; use ensogl_core::application; -use ensogl_core::application::Application; use ensogl_core::application::shortcut; +use ensogl_core::application::Application; use ensogl_core::display; use ensogl_core::display::scene::layer::LayerId; use ensogl_core::display::shape::*; @@ -29,8 +29,8 @@ pub use entry::Entry; // === Constants === /// The size of shadow under element. It is not counted in the component width and height. -pub const SHADOW_PX:f32 = 10.0; -const SHAPE_PADDING:f32 = 5.0; +pub const SHADOW_PX: f32 = 10.0; +const SHAPE_PADDING: f32 = 5.0; // === Selection === @@ -40,7 +40,7 @@ pub mod selection { use super::*; /// The corner radius in pixels. - pub const CORNER_RADIUS_PX:f32 = 12.0; + pub const CORNER_RADIUS_PX: f32 = 12.0; ensogl_core::define_shape_system! { (style:Style) { @@ -66,7 +66,7 @@ pub mod background { use super::*; /// The corner radius in pixels. - pub const CORNER_RADIUS_PX:f32 = selection::CORNER_RADIUS_PX; + pub const CORNER_RADIUS_PX: f32 = selection::CORNER_RADIUS_PX; ensogl_core::define_shape_system! { below = [selection]; @@ -93,100 +93,105 @@ pub mod background { // ============= /// Information about displayed fragment of entries list. -#[derive(Copy,Clone,Debug,Default)] +#[derive(Copy, Clone, Debug, Default)] struct View { - position_y : f32, - size : Vector2, + position_y: f32, + size: Vector2, } /// The Model of Select Component. -#[derive(Clone,CloneRef,Debug)] -struct Model { - app : Application, - entries : entry::List, - selection : selection::View, - background : background::View, - scrolled_area : display::object::Instance, - display_object : display::object::Instance, +#[derive(Clone, CloneRef, Debug)] +struct Model { + app: Application, + entries: entry::List, + selection: selection::View, + background: background::View, + scrolled_area: display::object::Instance, + display_object: display::object::Instance, } -impl Model { - - fn new(app:&Application) -> Self { - let app = app.clone_ref(); - let logger = Logger::new("SelectionContainer"); +impl Model { + fn new(app: &Application) -> Self { + let app = app.clone_ref(); + let logger = Logger::new("SelectionContainer"); let display_object = display::object::Instance::new(&logger); - let scrolled_area = display::object::Instance::new(&logger); - let entries = entry::List::new(&logger,&app); - let background = background::View::new(&logger); - let selection = selection::View::new(&logger); + let scrolled_area = display::object::Instance::new(&logger); + let entries = entry::List::new(&logger, &app); + let background = background::View::new(&logger); + let selection = selection::View::new(&logger); display_object.add_child(&background); display_object.add_child(&scrolled_area); scrolled_area.add_child(&entries); scrolled_area.add_child(&selection); - Model{app,entries,selection,background,scrolled_area,display_object} + Model { app, entries, selection, background, scrolled_area, display_object } } fn padding(&self) -> f32 { // FIXME : StyleWatch is unsuitable here, as it was designed as an internal tool for shape // system (#795) - let styles = StyleWatch::new(&self.app.display.scene().style_sheet); + let styles = StyleWatch::new(&self.app.display.scene().style_sheet); styles.get_number(ensogl_theme::application::searcher::padding) } /// Update the displayed entries list when _view_ has changed - the list was scrolled or /// resized. - fn update_after_view_change(&self, view:&View) { - let visible_entries = Self::visible_entries(view,self.entries.entry_count()); - let padding_px = self.padding(); - let padding = 2.0 * padding_px + SHAPE_PADDING; - let padding = Vector2(padding, padding); - let shadow = Vector2(2.0 * SHADOW_PX, 2.0 * SHADOW_PX); + fn update_after_view_change(&self, view: &View) { + let visible_entries = Self::visible_entries(view, self.entries.entry_count()); + let padding_px = self.padding(); + let padding = 2.0 * padding_px + SHAPE_PADDING; + let padding = Vector2(padding, padding); + let shadow = Vector2(2.0 * SHADOW_PX, 2.0 * SHADOW_PX); self.entries.set_position_x(-view.size.x / 2.0); self.background.size.set(view.size + padding + shadow); self.scrolled_area.set_position_y(view.size.y / 2.0 - view.position_y); self.entries.update_entries(visible_entries); } - fn set_entries(&self, provider:entry::AnyModelProvider, view:&View) { - let visible_entries = Self::visible_entries(view,provider.entry_count()); - self.entries.update_entries_new_provider(provider,visible_entries); + fn set_entries(&self, provider: entry::AnyModelProvider, view: &View) { + let visible_entries = Self::visible_entries(view, provider.entry_count()); + self.entries.update_entries_new_provider(provider, visible_entries); } - fn visible_entries(View {position_y,size}:&View, entry_count:usize) -> Range { + fn visible_entries(View { position_y, size }: &View, entry_count: usize) -> Range { if entry_count == 0 { 0..0 } else { - let entry_at_y_saturating = |y:f32| { - match entry::List::::entry_at_y_position(y,entry_count) { + let entry_at_y_saturating = + |y: f32| match entry::List::::entry_at_y_position(y, entry_count) { entry::list::IdAtYPosition::AboveFirst => 0, - entry::list::IdAtYPosition::UnderLast => entry_count - 1, - entry::list::IdAtYPosition::Entry(id) => id, - } - }; + entry::list::IdAtYPosition::UnderLast => entry_count - 1, + entry::list::IdAtYPosition::Entry(id) => id, + }; let first = entry_at_y_saturating(*position_y); - let last = entry_at_y_saturating(position_y - size.y) + 1; + let last = entry_at_y_saturating(position_y - size.y) + 1; first..last } } /// Check if the `point` is inside component assuming that it have given `size`. - fn is_inside(&self, point:Vector2, size:Vector2) -> bool { - let pos_obj_space = self.app.display.scene().screen_to_object_space(&self.background,point); - let x_range = (-size.x / 2.0)..=(size.x / 2.0); - let y_range = (-size.y / 2.0)..=(size.y / 2.0); + fn is_inside(&self, point: Vector2, size: Vector2) -> bool { + let pos_obj_space = + self.app.display.scene().screen_to_object_space(&self.background, point); + let x_range = (-size.x / 2.0)..=(size.x / 2.0); + let y_range = (-size.y / 2.0)..=(size.y / 2.0); x_range.contains(&pos_obj_space.x) && y_range.contains(&pos_obj_space.y) } - fn selected_entry_after_jump - (&self, current_entry:Option, jump:isize) -> Option { + fn selected_entry_after_jump( + &self, + current_entry: Option, + jump: isize, + ) -> Option { if jump < 0 { let current_entry = current_entry?; - if current_entry == 0 { None } - else { Some(current_entry.saturating_sub(-jump as usize)) } + if current_entry == 0 { + None + } else { + Some(current_entry.saturating_sub(-jump as usize)) + } } else { let max_entry = self.entries.entry_count().checked_sub(1)?; - Some(current_entry.map_or(0, |id| id+(jump as usize)).min(max_entry)) + Some(current_entry.map_or(0, |id| id + (jump as usize)).min(max_entry)) } } } @@ -243,40 +248,43 @@ ensogl_core::define_endpoints! { /// This is a displayed list of entries (of any type `E`) with possibility of selecting one and /// "choosing" by clicking or pressing enter. The basic entry types are defined in [`entry`] module. #[allow(missing_docs)] -#[derive(Clone,CloneRef,Debug)] -pub struct ListView { - model : Model, - pub frp : Frp, +#[derive(Clone, CloneRef, Debug)] +pub struct ListView { + model: Model, + pub frp: Frp, } -impl Deref for ListView { +impl Deref for ListView { type Target = Frp; - fn deref(&self) -> &Self::Target { &self.frp } + fn deref(&self) -> &Self::Target { + &self.frp + } } -impl ListView -where E::Model : Default { +impl ListView +where E::Model: Default +{ /// Constructor. - pub fn new(app:&Application) -> Self { - let frp = Frp::new(); + pub fn new(app: &Application) -> Self { + let frp = Frp::new(); let model = Model::new(app); - ListView {model,frp}.init(app) + ListView { model, frp }.init(app) } - fn init(self, app:&Application) -> Self { - const MAX_SCROLL:f32 = entry::HEIGHT/2.0; - const MOUSE_MOVE_THRESHOLD:f32 = std::f32::EPSILON; - - let frp = &self.frp; - let network = &frp.network; - let model = &self.model; - let scene = app.display.scene(); - let mouse = &scene.mouse.frp; - let view_y = DEPRECATED_Animation::::new(network); - let selection_y = DEPRECATED_Animation::::new(network); + fn init(self, app: &Application) -> Self { + const MAX_SCROLL: f32 = entry::HEIGHT / 2.0; + const MOUSE_MOVE_THRESHOLD: f32 = std::f32::EPSILON; + + let frp = &self.frp; + let network = &frp.network; + let model = &self.model; + let scene = app.display.scene(); + let mouse = &scene.mouse.frp; + let view_y = DEPRECATED_Animation::::new(network); + let selection_y = DEPRECATED_Animation::::new(network); let selection_height = DEPRECATED_Animation::::new(network); - frp::extend!{ network + frp::extend! { network // === Mouse Position === @@ -424,32 +432,46 @@ where E::Model : Default { } /// Sets the scene layer where the labels will be placed. - pub fn set_label_layer(&self, layer:LayerId) { + pub fn set_label_layer(&self, layer: LayerId) { self.model.entries.set_label_layer(layer); } } -impl display::Object for ListView { - fn display_object(&self) -> &display::object::Instance { &self.model.display_object } +impl display::Object for ListView { + fn display_object(&self) -> &display::object::Instance { + &self.model.display_object + } } -impl application::command::FrpNetworkProvider for ListView { - fn network(&self) -> &frp::Network { &self.frp.network } +impl application::command::FrpNetworkProvider for ListView { + fn network(&self) -> &frp::Network { + &self.frp.network + } } -impl application::View for ListView { - fn label() -> &'static str { "ListView" } - fn new(app:&Application) -> Self { ListView::new(app) } - fn app(&self) -> &Application { &self.model.app } +impl application::View for ListView { + fn label() -> &'static str { + "ListView" + } + fn new(app: &Application) -> Self { + ListView::new(app) + } + fn app(&self) -> &Application { + &self.model.app + } fn default_shortcuts() -> Vec { use shortcut::ActionType::*; - (&[ (PressAndRepeat , "up" , "move_selection_up") - , (PressAndRepeat , "down" , "move_selection_down") - , (Press , "page-up" , "move_selection_page_up") - , (Press , "page-down" , "move_selection_page_down") - , (Press , "home" , "move_selection_to_first") - , (Press , "end" , "move_selection_to_last") - , (Press , "enter" , "chose_selected_entry") - ]).iter().map(|(a,b,c)|Self::self_shortcut(*a,*b,*c)).collect() + (&[ + (PressAndRepeat, "up", "move_selection_up"), + (PressAndRepeat, "down", "move_selection_down"), + (Press, "page-up", "move_selection_page_up"), + (Press, "page-down", "move_selection_page_down"), + (Press, "home", "move_selection_to_first"), + (Press, "end", "move_selection_to_last"), + (Press, "enter", "chose_selected_entry"), + ]) + .iter() + .map(|(a, b, c)| Self::self_shortcut(*a, *b, *c)) + .collect() } } diff --git a/src/rust/ensogl/lib/components/src/list_view/entry.rs b/src/rust/ensogl/lib/components/src/list_view/entry.rs index 1f5ef0bdfc..fb29cf413f 100644 --- a/src/rust/ensogl/lib/components/src/list_view/entry.rs +++ b/src/rust/ensogl/lib/components/src/list_view/entry.rs @@ -17,9 +17,9 @@ use ensogl_theme as theme; // ================= /// Padding inside entry in pixels. -pub const PADDING:f32 = 14.0; +pub const PADDING: f32 = 14.0; /// The overall entry's height (including padding). -pub const HEIGHT:f32 = 30.0; +pub const HEIGHT: f32 = 30.0; @@ -52,18 +52,18 @@ pub trait Entry: CloneRef + Debug + display::Object + 'static { /// The model of this entry. The entry should be a representation of data from the Model. /// For example, the entry being just a caption can have [`String`] as its model - the text to /// be displayed. - type Model : Debug + Default; + type Model: Debug + Default; /// An Object constructor. - fn new(app:&Application) -> Self; + fn new(app: &Application) -> Self; /// Update content with new model. - fn update(&self, model:&Self::Model); + fn update(&self, model: &Self::Model); /// Set the layer of all [`text::Area`] components inside. The [`text::Area`] component is /// handled in a special way, and is often in different layer than shapes. See TODO comment /// in [`text::Area::add_to_scene_layer`] method. - fn set_label_layer(&self, label_layer:&display::scene::Layer); + fn set_label_layer(&self, label_layer: &display::scene::Layer); } @@ -74,25 +74,25 @@ pub trait Entry: CloneRef + Debug + display::Object + 'static { // === Label === /// The [`Entry`] being a single text field displaying String. -#[derive(Clone,CloneRef,Debug)] +#[derive(Clone, CloneRef, Debug)] pub struct Label { - display_object : display::object::Instance, - label : text::Area, - network : enso_frp::Network, - style_watch : StyleWatchFrp, + display_object: display::object::Instance, + label: text::Area, + network: enso_frp::Network, + style_watch: StyleWatchFrp, } impl Entry for Label { type Model = String; fn new(app: &Application) -> Self { - let logger = Logger::new("list_view::entry::Label"); + let logger = Logger::new("list_view::entry::Label"); let display_object = display::object::Instance::new(logger); - let label = app.new_view::(); - let network = frp::Network::new("list_view::entry::Label"); - let style_watch = StyleWatchFrp::new(&app.display.scene().style_sheet); - let color = style_watch.get_color(theme::widget::list_view::text); - let size = style_watch.get_number(theme::widget::list_view::text::size); + let label = app.new_view::(); + let network = frp::Network::new("list_view::entry::Label"); + let style_watch = StyleWatchFrp::new(&app.display.scene().style_sheet); + let color = style_watch.get_color(theme::widget::list_view::text); + let size = style_watch.get_number(theme::widget::list_view::text::size); display_object.add_child(&label); frp::extend! { network @@ -105,20 +105,22 @@ impl Entry for Label { eval size ((size) label.set_position_y(size/2.0)); } init.emit(()); - Self {display_object,label,network,style_watch} + Self { display_object, label, network, style_watch } } fn update(&self, model: &Self::Model) { self.label.set_content(model); } - fn set_label_layer(&self, label_layer:&display::scene::Layer) { + fn set_label_layer(&self, label_layer: &display::scene::Layer) { self.label.add_to_scene_layer(label_layer); } } impl display::Object for Label { - fn display_object(&self) -> &display::object::Instance { &self.display_object } + fn display_object(&self) -> &display::object::Instance { + &self.display_object + } } @@ -126,29 +128,30 @@ impl display::Object for Label { /// The model for [`HighlightedLabel`], being an entry displayed as a single label with highlighted /// some parts of text. -#[derive(Clone,Debug,Default)] +#[derive(Clone, Debug, Default)] pub struct GlyphHighlightedLabelModel { /// Displayed text. - pub label:String, + pub label: String, /// A list of ranges of highlighted bytes. - pub highlighted:Vec>, + pub highlighted: Vec>, } /// The [`Entry`] similar to the [`Label`], but allows highlighting some parts of text. -#[derive(Clone,CloneRef,Debug)] +#[derive(Clone, CloneRef, Debug)] pub struct GlyphHighlightedLabel { - inner : Label, - highlight : frp::Source>>, + inner: Label, + highlight: frp::Source>>, } impl Entry for GlyphHighlightedLabel { type Model = GlyphHighlightedLabelModel; fn new(app: &Application) -> Self { - let inner = Label::new(app); - let network = &inner.network; - let highlight_color = inner.style_watch.get_color(theme::widget::list_view::text::highlight); - let label = &inner.label; + let inner = Label::new(app); + let network = &inner.network; + let highlight_color = + inner.style_watch.get_color(theme::widget::list_view::text::highlight); + let label = &inner.label; frp::extend! { network highlight <- source::>>(); @@ -159,7 +162,7 @@ impl Entry for GlyphHighlightedLabel { } }); } - Self {inner,highlight} + Self { inner, highlight } } fn update(&self, model: &Self::Model) { @@ -167,13 +170,15 @@ impl Entry for GlyphHighlightedLabel { self.highlight.emit(&model.highlighted); } - fn set_label_layer(&self, layer:&display::scene::Layer) { + fn set_label_layer(&self, layer: &display::scene::Layer) { self.inner.set_label_layer(layer); } } impl display::Object for GlyphHighlightedLabel { - fn display_object(&self) -> &display::object::Instance { self.inner.display_object() } + fn display_object(&self) -> &display::object::Instance { + self.inner.display_object() + } } @@ -189,37 +194,51 @@ impl display::Object for GlyphHighlightedLabel { /// The [`crate::ListView`] component does not display all entries at once, instead it lazily ask /// for models of entries when they're about to be displayed. So setting the select content is /// essentially providing an implementor of this trait. -pub trait ModelProvider : Debug { +pub trait ModelProvider: Debug { /// Number of all entries. fn entry_count(&self) -> usize; /// Get the model of entry with given id. The implementors should return `None` only when /// requested id greater or equal to entries count. - fn get(&self, id:Id) -> Option - where E : Entry; + fn get(&self, id: Id) -> Option + where E: Entry; } // === AnyModelProvider === /// A wrapper for shared instance of some Provider of models for `E` entries. -#[derive(Debug,Shrinkwrap)] +#[derive(Debug, Shrinkwrap)] pub struct AnyModelProvider(Rc>); -impl Clone for AnyModelProvider { fn clone (&self) -> Self { Self(self.0.clone()) }} -impl CloneRef for AnyModelProvider { fn clone_ref(&self) -> Self { Self(self.0.clone_ref()) }} +impl Clone for AnyModelProvider { + fn clone(&self) -> Self { + Self(self.0.clone()) + } +} +impl CloneRef for AnyModelProvider { + fn clone_ref(&self) -> Self { + Self(self.0.clone_ref()) + } +} impl AnyModelProvider { /// Create from typed provider. - pub fn new+'static>(provider:T) -> Self { Self(Rc::new(provider)) } + pub fn new + 'static>(provider: T) -> Self { + Self(Rc::new(provider)) + } } -impl+'static> From> for AnyModelProvider { - fn from(provider:Rc) -> Self { Self(provider) } +impl + 'static> From> for AnyModelProvider { + fn from(provider: Rc) -> Self { + Self(provider) + } } impl Default for AnyModelProvider { - fn default() -> Self { Self::new(EmptyProvider) } + fn default() -> Self { + Self::new(EmptyProvider) + } } @@ -228,26 +247,33 @@ impl Default for AnyModelProvider { /// An Entry Model Provider giving no entries. /// /// This is the default provider for new select components. -#[derive(Clone,CloneRef,Copy,Debug)] +#[derive(Clone, CloneRef, Copy, Debug)] pub struct EmptyProvider; impl ModelProvider for EmptyProvider { - fn entry_count(&self) -> usize { 0 } - fn get (&self, _:usize) -> Option where E : Entry { None } + fn entry_count(&self) -> usize { + 0 + } + fn get(&self, _: usize) -> Option + where E: Entry { + None + } } // === ModelProvider for Vectors === -impl ModelProvider for Vec -where E : Entry, - T : Debug + Clone + Into { +impl ModelProvider for Vec +where + E: Entry, + T: Debug + Clone + Into, +{ fn entry_count(&self) -> usize { self.len() } - fn get(&self, id:usize) -> Option { - Some(<[T]>::get(self, id)?.clone().into()) + fn get(&self, id: usize) -> Option { + Some(<[T]>::get(self, id)?.clone().into()) } } @@ -255,29 +281,28 @@ where E : Entry, // === SingleMaskedProvider === /// An Entry Model Provider that wraps a `AnyModelProvider` and allows the masking of a single item. -#[derive(Clone,Debug)] +#[derive(Clone, Debug)] pub struct SingleMaskedProvider { - content : AnyModelProvider, - mask : Cell>, + content: AnyModelProvider, + mask: Cell>, } -impl ModelProvider for SingleMaskedProvider { +impl ModelProvider for SingleMaskedProvider { fn entry_count(&self) -> usize { match self.mask.get() { - None => self.content.entry_count(), + None => self.content.entry_count(), Some(_) => self.content.entry_count().saturating_sub(1), } } - fn get(&self, ix:usize) -> Option - where E : Entry { + fn get(&self, ix: usize) -> Option + where E: Entry { let internal_ix = self.unmasked_index(ix); self.content.get(internal_ix) } } impl SingleMaskedProvider { - /// Return the index to the unmasked underlying data. Will only be valid to use after /// calling `clear_mask`. /// @@ -294,11 +319,11 @@ impl SingleMaskedProvider { /// Masked indices [0, 1, 2, 3] /// Unmasked Index [0, 1, 2, 3] /// ``` - pub fn unmasked_index(&self, ix:Id) -> Id { + pub fn unmasked_index(&self, ix: Id) -> Id { match self.mask.get() { - None => ix, - Some(id) if ix < id => ix, - Some(_) => ix+1, + None => ix, + Some(id) if ix < id => ix, + Some(_) => ix + 1, } } @@ -306,7 +331,7 @@ impl SingleMaskedProvider { /// will behave as if it was not there. /// /// *Important:* The index is interpreted according to the _masked_ position of elements. - pub fn set_mask(&self, ix:Id) { + pub fn set_mask(&self, ix: Id) { let internal_ix = self.unmasked_index(ix); self.mask.set(Some(internal_ix)); } @@ -315,7 +340,7 @@ impl SingleMaskedProvider { /// will behave as if it was not there. /// /// *Important:* The index is interpreted according to the _unmasked_ position of elements. - pub fn set_mask_raw(&self, ix:Id) { + pub fn set_mask_raw(&self, ix: Id) { self.mask.set(Some(ix)); } @@ -326,9 +351,9 @@ impl SingleMaskedProvider { } impl From> for SingleMaskedProvider { - fn from(content:AnyModelProvider) -> Self { + fn from(content: AnyModelProvider) -> Self { let mask = default(); - SingleMaskedProvider{content,mask} + SingleMaskedProvider { content, mask } } } @@ -344,10 +369,10 @@ mod tests { #[test] fn test_masked_provider() { - let test_data = vec!["A", "B", "C", "D"]; + let test_data = vec!["A", "B", "C", "D"]; let test_models = test_data.into_iter().map(|label| label.to_owned()).collect_vec(); - let provider = AnyModelProvider::