Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow borrowing plot points #64

Open
wants to merge 6 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
15 changes: 12 additions & 3 deletions Cargo.lock
Original file line number Diff line number Diff line change
Expand Up @@ -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"
Expand Down Expand Up @@ -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",
Expand Down Expand Up @@ -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]]
Expand Down
2 changes: 1 addition & 1 deletion Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -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",
] }
Expand Down
26 changes: 13 additions & 13 deletions demo/src/plot_demo.rs
Original file line number Diff line number Diff line change
Expand Up @@ -211,9 +211,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;
Expand All @@ -229,7 +229,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(),
Expand All @@ -241,7 +241,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()),
Expand Down Expand Up @@ -308,7 +308,7 @@ impl Default for MarkerDemo {
}

impl MarkerDemo {
fn markers(&self) -> Vec<Points> {
fn markers<'a>(&self) -> Vec<Points<'a>> {
MarkerShape::all()
.enumerate()
.map(|(i, marker)| {
Expand Down Expand Up @@ -371,23 +371,23 @@ 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,
..,
100,
))
}

fn sin() -> Line {
fn sin<'a>() -> Line<'a> {
Line::new(PlotPoints::from_explicit_callback(
move |x| x.sin(),
..,
100,
))
}

fn cos() -> Line {
fn cos<'a>() -> Line<'a> {
Line::new(PlotPoints::from_explicit_callback(
move |x| x.cos(),
..,
Expand Down Expand Up @@ -455,7 +455,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
}
Expand Down Expand Up @@ -609,31 +609,31 @@ 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,
..,
100,
))
}

fn sin() -> Line {
fn sin<'a>() -> Line<'a> {
Line::new(PlotPoints::from_explicit_callback(
move |x| x.sin(),
..,
100,
))
}

fn cos() -> Line {
fn cos<'a>() -> Line<'a> {
Line::new(PlotPoints::from_explicit_callback(
move |x| x.cos(),
..,
100,
))
}

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));
Expand Down
42 changes: 21 additions & 21 deletions egui_plot/src/items/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -410,8 +410,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,
Expand All @@ -422,8 +422,8 @@ pub struct Line {
id: Option<Id>,
}

impl Line {
pub fn new(series: impl Into<PlotPoints>) -> Self {
impl<'a> Line<'a> {
pub fn new(series: impl Into<PlotPoints<'a>>) -> 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
Expand Down Expand Up @@ -521,7 +521,7 @@ fn y_intersection(p1: &Pos2, p2: &Pos2, y: f32) -> Option<f32> {
.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<Shape>) {
let Self {
series,
Expand Down Expand Up @@ -619,8 +619,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,
Expand All @@ -630,8 +630,8 @@ pub struct Polygon {
id: Option<Id>,
}

impl Polygon {
pub fn new(series: impl Into<PlotPoints>) -> Self {
impl<'a> Polygon<'a> {
pub fn new(series: impl Into<PlotPoints<'a>>) -> Self {
Self {
series: series.into(),
stroke: Stroke::new(1.0, Color32::TRANSPARENT),
Expand Down Expand Up @@ -708,7 +708,7 @@ impl Polygon {
}
}

impl PlotItem for Polygon {
impl<'a> PlotItem for Polygon<'a> {
fn shapes(&self, _ui: &Ui, transform: &PlotTransform, shapes: &mut Vec<Shape>) {
let Self {
series,
Expand Down Expand Up @@ -917,8 +917,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,

Expand All @@ -941,8 +941,8 @@ pub struct Points {
id: Option<Id>,
}

impl Points {
pub fn new(series: impl Into<PlotPoints>) -> Self {
impl<'a> Points<'a> {
pub fn new(series: impl Into<PlotPoints<'a>>) -> Self {
Self {
series: series.into(),
shape: MarkerShape::Circle,
Expand Down Expand Up @@ -1027,7 +1027,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<Shape>) {
let sqrt_3 = 3_f32.sqrt();
Expand Down Expand Up @@ -1195,9 +1195,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<f32>,
pub(super) color: Color32,
pub(super) name: String,
Expand All @@ -1206,8 +1206,8 @@ pub struct Arrows {
id: Option<Id>,
}

impl Arrows {
pub fn new(origins: impl Into<PlotPoints>, tips: impl Into<PlotPoints>) -> Self {
impl<'a> Arrows<'a> {
pub fn new(origins: impl Into<PlotPoints<'a>>, tips: impl Into<PlotPoints<'a>>) -> Self {
Self {
origins: origins.into(),
tips: tips.into(),
Expand Down Expand Up @@ -1269,7 +1269,7 @@ impl Arrows {
}
}

impl PlotItem for Arrows {
impl<'a> PlotItem for Arrows<'a> {
fn shapes(&self, _ui: &Ui, transform: &PlotTransform, shapes: &mut Vec<Shape>) {
let Self {
origins,
Expand Down
45 changes: 32 additions & 13 deletions egui_plot/src/items/values.rs
Original file line number Diff line number Diff line change
Expand Up @@ -153,38 +153,48 @@ impl Default for Orientation {

/// Represents many [`PlotPoint`]s.
///
/// These can be an owned `Vec` or generated with a function.
pub enum PlotPoints {
/// 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<PlotPoint>),
Generator(ExplicitGenerator),
// Borrowed(&[PlotPoint]), // TODO(EmbersArc): Lifetimes are tricky in this case.
Generator(ExplicitGenerator<'a>),
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<Vec<[f64; 2]>> for PlotPoints {
impl<'a> From<Vec<[f64; 2]>> 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<T: IntoIterator<Item = [f64; 2]>>(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)
}
Expand All @@ -193,12 +203,13 @@ impl PlotPoints {
match self {
Self::Owned(points) => points.as_slice(),
Self::Generator(_) => &[],
Self::Borrowed(points) => points,
}
}

/// 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<f64>,
points: usize,
) -> Self {
Expand Down Expand Up @@ -271,6 +282,7 @@ impl PlotPoints {
match self {
Self::Owned(points) => points.is_empty(),
Self::Generator(_) => false,
Self::Borrowed(points) => points.is_empty(),
}
}

Expand Down Expand Up @@ -314,6 +326,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
}
}
}
}
Expand Down Expand Up @@ -374,13 +393,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<dyn Fn(f64) -> f64>,
pub struct ExplicitGenerator<'a> {
function: Box<dyn Fn(f64) -> f64 + 'a>,
x_range: RangeInclusive<f64>,
points: usize,
}

impl ExplicitGenerator {
impl ExplicitGenerator<'_> {
fn estimate_bounds(&self) -> PlotBounds {
let mut bounds = PlotBounds::NOTHING;

Expand Down
Loading
Loading