From d9030dc732bddec121523415784f6a1931a9ec7c Mon Sep 17 00:00:00 2001 From: mo8it Date: Tue, 30 Jul 2024 22:49:21 +0200 Subject: [PATCH 1/5] Add `PlotPoints::Borrowed` --- demo/src/plot_demo.rs | 26 +++++++++++----------- egui_plot/src/items/mod.rs | 42 +++++++++++++++++------------------ egui_plot/src/items/values.rs | 31 ++++++++++++++++++++------ egui_plot/src/legend.rs | 4 ++-- egui_plot/src/lib.rs | 10 ++++----- egui_plot/src/plot_ui.rs | 16 ++++++------- 6 files changed, 73 insertions(+), 56 deletions(-) diff --git a/demo/src/plot_demo.rs b/demo/src/plot_demo.rs index a669648..f601acc 100644 --- a/demo/src/plot_demo.rs +++ b/demo/src/plot_demo.rs @@ -209,9 +209,9 @@ impl LineDemo { }); } - fn circle(&self) -> Line { + fn circle<'a>(&self) -> Line<'a> { let n = 512; - let circle_points: PlotPoints = (0..=n) + let circle_points: PlotPoints<'_> = (0..=n) .map(|i| { let t = remap(i as f64, 0.0..=(n as f64), 0.0..=TAU); let r = self.circle_radius; @@ -227,7 +227,7 @@ impl LineDemo { .name("circle") } - fn sin(&self) -> Line { + fn sin<'a>(&self) -> Line<'a> { let time = self.time; Line::new(PlotPoints::from_explicit_callback( move |x| 0.5 * (2.0 * x).sin() * time.sin(), @@ -239,7 +239,7 @@ impl LineDemo { .name("wave") } - fn thingy(&self) -> Line { + fn thingy<'a>(&self) -> Line<'a> { let time = self.time; Line::new(PlotPoints::from_parametric_callback( move |t| ((2.0 * t + time).sin(), (3.0 * t).sin()), @@ -304,7 +304,7 @@ impl Default for MarkerDemo { } impl MarkerDemo { - fn markers(&self) -> Vec { + fn markers<'a>(&self) -> Vec> { MarkerShape::all() .enumerate() .map(|(i, marker)| { @@ -367,7 +367,7 @@ struct LegendDemo { } impl LegendDemo { - fn line_with_slope(slope: f64) -> Line { + fn line_with_slope<'a>(slope: f64) -> Line<'a> { Line::new(PlotPoints::from_explicit_callback( move |x| slope * x, .., @@ -375,7 +375,7 @@ impl LegendDemo { )) } - fn sin() -> Line { + fn sin<'a>() -> Line<'a> { Line::new(PlotPoints::from_explicit_callback( move |x| x.sin(), .., @@ -383,7 +383,7 @@ impl LegendDemo { )) } - fn cos() -> Line { + fn cos<'a>() -> Line<'a> { Line::new(PlotPoints::from_explicit_callback( move |x| x.cos(), .., @@ -444,7 +444,7 @@ impl CustomAxesDemo { const MINS_PER_DAY: f64 = 24.0 * 60.0; const MINS_PER_H: f64 = 60.0; - fn logistic_fn() -> Line { + fn logistic_fn<'a>() -> Line<'a> { fn days(min: f64) -> f64 { CustomAxesDemo::MINS_PER_DAY * min } @@ -598,7 +598,7 @@ impl Default for LinkedAxesDemo { } impl LinkedAxesDemo { - fn line_with_slope(slope: f64) -> Line { + fn line_with_slope<'a>(slope: f64) -> Line<'a> { Line::new(PlotPoints::from_explicit_callback( move |x| slope * x, .., @@ -606,7 +606,7 @@ impl LinkedAxesDemo { )) } - fn sin() -> Line { + fn sin<'a>() -> Line<'a> { Line::new(PlotPoints::from_explicit_callback( move |x| x.sin(), .., @@ -614,7 +614,7 @@ impl LinkedAxesDemo { )) } - fn cos() -> Line { + fn cos<'a>() -> Line<'a> { Line::new(PlotPoints::from_explicit_callback( move |x| x.cos(), .., @@ -622,7 +622,7 @@ impl LinkedAxesDemo { )) } - fn configure_plot(plot_ui: &mut egui_plot::PlotUi) { + fn configure_plot(plot_ui: &mut egui_plot::PlotUi<'_>) { plot_ui.line(Self::line_with_slope(0.5)); plot_ui.line(Self::line_with_slope(1.0)); plot_ui.line(Self::line_with_slope(2.0)); diff --git a/egui_plot/src/items/mod.rs b/egui_plot/src/items/mod.rs index 6d5f7cf..59d8915 100644 --- a/egui_plot/src/items/mod.rs +++ b/egui_plot/src/items/mod.rs @@ -420,8 +420,8 @@ impl PlotItem for VLine { } /// A series of values forming a path. -pub struct Line { - pub(super) series: PlotPoints, +pub struct Line<'a> { + pub(super) series: PlotPoints<'a>, pub(super) stroke: Stroke, pub(super) name: String, pub(super) highlight: bool, @@ -431,8 +431,8 @@ pub struct Line { id: Option, } -impl Line { - pub fn new(series: impl Into) -> Self { +impl<'a> Line<'a> { + pub fn new(series: impl Into>) -> Self { Self { series: series.into(), stroke: Stroke::new(1.5, Color32::TRANSPARENT), // Note: a stroke of 1.0 (or less) can look bad on low-dpi-screens @@ -522,7 +522,7 @@ fn y_intersection(p1: &Pos2, p2: &Pos2, y: f32) -> Option { .then_some(((y * (p1.x - p2.x)) - (p1.x * p2.y - p1.y * p2.x)) / (p1.y - p2.y)) } -impl PlotItem for Line { +impl<'a> PlotItem for Line<'a> { fn shapes(&self, _ui: &Ui, transform: &PlotTransform, shapes: &mut Vec) { let Self { series, @@ -620,8 +620,8 @@ impl PlotItem for Line { } /// A convex polygon. -pub struct Polygon { - pub(super) series: PlotPoints, +pub struct Polygon<'a> { + pub(super) series: PlotPoints<'a>, pub(super) stroke: Stroke, pub(super) name: String, pub(super) highlight: bool, @@ -631,8 +631,8 @@ pub struct Polygon { id: Option, } -impl Polygon { - pub fn new(series: impl Into) -> Self { +impl<'a> Polygon<'a> { + pub fn new(series: impl Into>) -> Self { Self { series: series.into(), stroke: Stroke::new(1.0, Color32::TRANSPARENT), @@ -709,7 +709,7 @@ impl Polygon { } } -impl PlotItem for Polygon { +impl<'a> PlotItem for Polygon<'a> { fn shapes(&self, _ui: &Ui, transform: &PlotTransform, shapes: &mut Vec) { let Self { series, @@ -918,8 +918,8 @@ impl PlotItem for Text { } /// A set of points. -pub struct Points { - pub(super) series: PlotPoints, +pub struct Points<'a> { + pub(super) series: PlotPoints<'a>, pub(super) shape: MarkerShape, @@ -942,8 +942,8 @@ pub struct Points { id: Option, } -impl Points { - pub fn new(series: impl Into) -> Self { +impl<'a> Points<'a> { + pub fn new(series: impl Into>) -> Self { Self { series: series.into(), shape: MarkerShape::Circle, @@ -1028,7 +1028,7 @@ impl Points { } } -impl PlotItem for Points { +impl<'a> PlotItem for Points<'a> { #[allow(clippy::too_many_lines)] // TODO(emilk): shorten this function fn shapes(&self, _ui: &Ui, transform: &PlotTransform, shapes: &mut Vec) { let sqrt_3 = 3_f32.sqrt(); @@ -1196,9 +1196,9 @@ impl PlotItem for Points { } /// A set of arrows. -pub struct Arrows { - pub(super) origins: PlotPoints, - pub(super) tips: PlotPoints, +pub struct Arrows<'a> { + pub(super) origins: PlotPoints<'a>, + pub(super) tips: PlotPoints<'a>, pub(super) tip_length: Option, pub(super) color: Color32, pub(super) name: String, @@ -1207,8 +1207,8 @@ pub struct Arrows { id: Option, } -impl Arrows { - pub fn new(origins: impl Into, tips: impl Into) -> Self { +impl<'a> Arrows<'a> { + pub fn new(origins: impl Into>, tips: impl Into>) -> Self { Self { origins: origins.into(), tips: tips.into(), @@ -1270,7 +1270,7 @@ impl Arrows { } } -impl PlotItem for Arrows { +impl<'a> PlotItem for Arrows<'a> { fn shapes(&self, _ui: &Ui, transform: &PlotTransform, shapes: &mut Vec) { let Self { origins, diff --git a/egui_plot/src/items/values.rs b/egui_plot/src/items/values.rs index 1251396..f9ed3c4 100644 --- a/egui_plot/src/items/values.rs +++ b/egui_plot/src/items/values.rs @@ -154,37 +154,45 @@ impl Default for Orientation { /// Represents many [`PlotPoint`]s. /// /// These can be an owned `Vec` or generated with a function. -pub enum PlotPoints { +pub enum PlotPoints<'a> { Owned(Vec), Generator(ExplicitGenerator), - // Borrowed(&[PlotPoint]), // TODO(EmbersArc): Lifetimes are tricky in this case. + Borrowed(&'a [PlotPoint]), } -impl Default for PlotPoints { +impl<'a> Default for PlotPoints<'a> { fn default() -> Self { Self::Owned(Vec::new()) } } -impl From<[f64; 2]> for PlotPoints { +impl<'a> From<[f64; 2]> for PlotPoints<'a> { fn from(coordinate: [f64; 2]) -> Self { Self::new(vec![coordinate]) } } -impl From> for PlotPoints { +impl<'a> From> for PlotPoints<'a> { + #[inline] fn from(coordinates: Vec<[f64; 2]>) -> Self { Self::new(coordinates) } } -impl FromIterator<[f64; 2]> for PlotPoints { +impl<'a> From<&'a [PlotPoint]> for PlotPoints<'a> { + #[inline] + fn from(points: &'a [PlotPoint]) -> Self { + Self::Borrowed(points) + } +} + +impl<'a> FromIterator<[f64; 2]> for PlotPoints<'a> { fn from_iter>(iter: T) -> Self { Self::Owned(iter.into_iter().map(|point| point.into()).collect()) } } -impl PlotPoints { +impl<'a> PlotPoints<'a> { pub fn new(points: Vec<[f64; 2]>) -> Self { Self::from_iter(points) } @@ -193,6 +201,7 @@ impl PlotPoints { match self { Self::Owned(points) => points.as_slice(), Self::Generator(_) => &[], + Self::Borrowed(points) => points, } } @@ -271,6 +280,7 @@ impl PlotPoints { match self { Self::Owned(points) => points.is_empty(), Self::Generator(_) => false, + Self::Borrowed(points) => points.is_empty(), } } @@ -314,6 +324,13 @@ impl PlotPoints { bounds } Self::Generator(generator) => generator.estimate_bounds(), + Self::Borrowed(points) => { + let mut bounds = PlotBounds::NOTHING; + for point in *points { + bounds.extend_with(point); + } + bounds + } } } } diff --git a/egui_plot/src/legend.rs b/egui_plot/src/legend.rs index 8e81752..b72b737 100644 --- a/egui_plot/src/legend.rs +++ b/egui_plot/src/legend.rs @@ -187,10 +187,10 @@ pub(super) struct LegendWidget { impl LegendWidget { /// Create a new legend from items, the names of items that are hidden and the style of the /// text. Returns `None` if the legend has no entries. - pub(super) fn try_new( + pub(super) fn try_new<'a>( rect: Rect, config: Legend, - items: &[Box], + items: &[Box], hidden_items: &ahash::HashSet, // Existing hidden items in the plot memory. ) -> Option { // If `config.hidden_items` is not `None`, it is used. diff --git a/egui_plot/src/lib.rs b/egui_plot/src/lib.rs index 50b12cf..52aa108 100644 --- a/egui_plot/src/lib.rs +++ b/egui_plot/src/lib.rs @@ -728,20 +728,20 @@ impl<'a> Plot<'a> { } /// Interact with and add items to the plot and finally draw it. - pub fn show( + pub fn show<'b, R>( self, ui: &mut Ui, - build_fn: impl FnOnce(&mut PlotUi) -> R + 'a, + build_fn: impl FnOnce(&mut PlotUi<'b>) -> R + 'a, ) -> PlotResponse { self.show_dyn(ui, Box::new(build_fn)) } #[allow(clippy::too_many_lines)] // TODO(emilk): shorten this function #[allow(clippy::type_complexity)] // build_fn - fn show_dyn( + fn show_dyn<'b, R>( self, ui: &mut Ui, - build_fn: Box R + 'a>, + build_fn: Box) -> R + 'a>, ) -> PlotResponse { let Self { id_source, @@ -1467,7 +1467,7 @@ pub fn uniform_grid_spacer<'a>(spacer: impl Fn(GridInput) -> [f64; 3] + 'a) -> G // ---------------------------------------------------------------------------- struct PreparedPlot<'a> { - items: Vec>, + items: Vec>, show_x: bool, show_y: bool, label_formatter: LabelFormatter<'a>, diff --git a/egui_plot/src/plot_ui.rs b/egui_plot/src/plot_ui.rs index 8b24650..973c2b1 100644 --- a/egui_plot/src/plot_ui.rs +++ b/egui_plot/src/plot_ui.rs @@ -7,9 +7,9 @@ use crate::Plot; /// Provides methods to interact with a plot while building it. It is the single argument of the closure /// provided to [`Plot::show`]. See [`Plot`] for an example of how to use it. -pub struct PlotUi { +pub struct PlotUi<'a> { pub(crate) ctx: egui::Context, - pub(crate) items: Vec>, + pub(crate) items: Vec>, pub(crate) next_auto_color_idx: usize, pub(crate) last_plot_transform: PlotTransform, pub(crate) last_auto_bounds: Vec2b, @@ -17,7 +17,7 @@ pub struct PlotUi { pub(crate) bounds_modifications: Vec, } -impl PlotUi { +impl<'a> PlotUi<'a> { fn auto_color(&mut self) -> Color32 { let i = self.next_auto_color_idx; self.next_auto_color_idx += 1; @@ -122,12 +122,12 @@ impl PlotUi { } /// Add an arbitrary item. - pub fn add(&mut self, item: impl PlotItem + 'static) { + pub fn add(&mut self, item: impl PlotItem + 'a) { self.items.push(Box::new(item)); } /// Add a data line. - pub fn line(&mut self, mut line: crate::Line) { + pub fn line(&mut self, mut line: crate::Line<'a>) { if line.series.is_empty() { return; }; @@ -140,7 +140,7 @@ impl PlotUi { } /// Add a polygon. The polygon has to be convex. - pub fn polygon(&mut self, mut polygon: crate::Polygon) { + pub fn polygon(&mut self, mut polygon: crate::Polygon<'a>) { if polygon.series.is_empty() { return; }; @@ -162,7 +162,7 @@ impl PlotUi { } /// Add data points. - pub fn points(&mut self, mut points: crate::Points) { + pub fn points(&mut self, mut points: crate::Points<'a>) { if points.series.is_empty() { return; }; @@ -175,7 +175,7 @@ impl PlotUi { } /// Add arrows. - pub fn arrows(&mut self, mut arrows: crate::Arrows) { + pub fn arrows(&mut self, mut arrows: crate::Arrows<'a>) { if arrows.origins.is_empty() || arrows.tips.is_empty() { return; }; From c39a4d5655ad33a02a5539a5e8437cd4913b5f42 Mon Sep 17 00:00:00 2001 From: mo8it Date: Fri, 13 Sep 2024 13:10:59 +0200 Subject: [PATCH 2/5] Allow borrowing when using from_explicit_callback --- egui_plot/src/items/values.rs | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/egui_plot/src/items/values.rs b/egui_plot/src/items/values.rs index f9ed3c4..fe2be68 100644 --- a/egui_plot/src/items/values.rs +++ b/egui_plot/src/items/values.rs @@ -156,7 +156,7 @@ impl Default for Orientation { /// These can be an owned `Vec` or generated with a function. pub enum PlotPoints<'a> { Owned(Vec), - Generator(ExplicitGenerator), + Generator(ExplicitGenerator<'a>), Borrowed(&'a [PlotPoint]), } @@ -207,7 +207,7 @@ impl<'a> PlotPoints<'a> { /// Draw a line based on a function `y=f(x)`, a range (which can be infinite) for x and the number of points. pub fn from_explicit_callback( - function: impl Fn(f64) -> f64 + 'static, + function: impl Fn(f64) -> f64 + 'a, x_range: impl RangeBounds, points: usize, ) -> Self { @@ -391,13 +391,13 @@ pub enum PlotGeometry<'a> { // ---------------------------------------------------------------------------- /// Describes a function y = f(x) with an optional range for x and a number of points. -pub struct ExplicitGenerator { - function: Box f64>, +pub struct ExplicitGenerator<'a> { + function: Box f64 + 'a>, x_range: RangeInclusive, points: usize, } -impl ExplicitGenerator { +impl ExplicitGenerator<'_> { fn estimate_bounds(&self) -> PlotBounds { let mut bounds = PlotBounds::NOTHING; From 34c663fa72986be6f782bacd417934b5ba30eb12 Mon Sep 17 00:00:00 2001 From: Nicolas Date: Tue, 7 Jan 2025 20:48:27 +0100 Subject: [PATCH 3/5] fiix docstring --- egui_plot/src/items/values.rs | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/egui_plot/src/items/values.rs b/egui_plot/src/items/values.rs index fe2be68..51f66da 100644 --- a/egui_plot/src/items/values.rs +++ b/egui_plot/src/items/values.rs @@ -153,7 +153,9 @@ impl Default for Orientation { /// Represents many [`PlotPoint`]s. /// -/// These can be an owned `Vec` or generated with a function. +/// These can be an owned `Vec` +/// or generated on-the-fly by a function +/// or borrowed from a slice. pub enum PlotPoints<'a> { Owned(Vec), Generator(ExplicitGenerator<'a>), From e4417cac3f860f578134f43bf5113c1eafebe692 Mon Sep 17 00:00:00 2001 From: Nicolas Date: Tue, 7 Jan 2025 20:59:30 +0100 Subject: [PATCH 4/5] add example --- examples/borrow_points/Cargo.toml | 19 ++++++++++++++ examples/borrow_points/README.md | 5 ++++ examples/borrow_points/src/main.rs | 40 ++++++++++++++++++++++++++++++ 3 files changed, 64 insertions(+) create mode 100644 examples/borrow_points/Cargo.toml create mode 100644 examples/borrow_points/README.md create mode 100644 examples/borrow_points/src/main.rs diff --git a/examples/borrow_points/Cargo.toml b/examples/borrow_points/Cargo.toml new file mode 100644 index 0000000..fd05d36 --- /dev/null +++ b/examples/borrow_points/Cargo.toml @@ -0,0 +1,19 @@ +[package] +name = "borrow_points" +version = "0.1.0" +authors = ["Nicolas "] +license = "MIT OR Apache-2.0" +edition = "2021" +rust-version = "1.80" +publish = false + +[lints] +workspace = true + +[dependencies] +eframe = { workspace = true, features = ["default"] } +egui_plot.workspace = true +env_logger = { workspace = true, default-features = false, features = [ + "auto-color", + "humantime", +] } diff --git a/examples/borrow_points/README.md b/examples/borrow_points/README.md new file mode 100644 index 0000000..d5fc595 --- /dev/null +++ b/examples/borrow_points/README.md @@ -0,0 +1,5 @@ +This example shows how to borrow points. + +```sh +cargo run -p borrow_points +``` diff --git a/examples/borrow_points/src/main.rs b/examples/borrow_points/src/main.rs new file mode 100644 index 0000000..831c37e --- /dev/null +++ b/examples/borrow_points/src/main.rs @@ -0,0 +1,40 @@ +#![cfg_attr(not(debug_assertions), windows_subsystem = "windows")] // hide console window on Windows in release +#![allow(rustdoc::missing_crate_level_docs)] // it's an example + +use eframe::egui; +use egui_plot::{Legend, Line, Plot, PlotPoint, PlotPoints}; + +fn main() -> eframe::Result { + env_logger::init(); // Log to stderr (if you run with `RUST_LOG=debug`). + + let options = eframe::NativeOptions { + viewport: egui::ViewportBuilder::default().with_inner_size([350.0, 200.0]), + ..Default::default() + }; + + let points: Vec<[f64; 2]> = vec![[0.0, 1.0], [2.0, 3.0], [3.0, 2.0]]; + + let points: Vec = points.iter().map(|p| PlotPoint::new(p[0], p[1])).collect(); + eframe::run_native( + "My egui App with a plot", + options, + Box::new(|_cc| Ok(Box::new(MyApp { points }))), + ) +} + +#[derive(Default)] +struct MyApp { + points: Vec, +} + +impl eframe::App for MyApp { + fn update(&mut self, ctx: &egui::Context, _frame: &mut eframe::Frame) { + egui::CentralPanel::default().show(ctx, |ui| { + Plot::new("My Plot") + .legend(Legend::default()) + .show(ui, |plot_ui| { + plot_ui.line(Line::new(PlotPoints::Borrowed(&self.points)).name("curve")); + }); + }); + } +} From bf92aadade316e76b2ddde148fe52f684af0503d Mon Sep 17 00:00:00 2001 From: Nicolas Date: Tue, 7 Jan 2025 20:59:42 +0100 Subject: [PATCH 5/5] update env_logger --- Cargo.lock | 15 ++++++++++++--- Cargo.toml | 2 +- 2 files changed, 13 insertions(+), 4 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index ec0d82c..dc7cf8f 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -582,6 +582,15 @@ dependencies = [ "piper", ] +[[package]] +name = "borrow_points" +version = "0.1.0" +dependencies = [ + "eframe", + "egui_plot", + "env_logger", +] + [[package]] name = "bumpalo" version = "3.16.0" @@ -1099,9 +1108,9 @@ dependencies = [ [[package]] name = "env_logger" -version = "0.11.5" +version = "0.11.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e13fa619b91fb2381732789fc5de83b45675e882f66623b7d8cb4f643017018d" +checksum = "dcaee3d8e3cfc3fd92428d477bc97fc29ec8716d180c0d74c643bb26166660e0" dependencies = [ "anstream", "anstyle", @@ -3372,7 +3381,7 @@ version = "0.1.9" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "cf221c93e13a30d793f7645a0e7762c55d169dbb0a49671918a2319d289b10bb" dependencies = [ - "windows-sys 0.48.0", + "windows-sys 0.59.0", ] [[package]] diff --git a/Cargo.toml b/Cargo.toml index 1e2d2e2..49d857b 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -29,7 +29,7 @@ document-features = "0.2.10" eframe = { version = "0.30.0", default-features = false } egui = { version = "0.30.0", default-features = false } emath = { version = "0.30.0", default-features = false } -env_logger = { version = "0.11.5", default-features = false, features = [ +env_logger = { version = "0.11.6", default-features = false, features = [ "auto-color", "humantime", ] }