egui/egui_demo_lib/src/apps/demo/plot_demo.rs

277 lines
7.7 KiB
Rust
Raw Normal View History

use egui::plot::{Line, MarkerShape, Plot, Points, Value, Values};
2021-02-14 20:39:04 +00:00
use egui::*;
use std::f64::consts::TAU;
#[derive(PartialEq)]
struct LineDemo {
2021-02-14 20:39:04 +00:00
animate: bool,
time: f64,
circle_radius: f64,
2021-02-14 20:39:04 +00:00
circle_center: Pos2,
square: bool,
legend: bool,
2021-02-14 20:39:04 +00:00
proportional: bool,
}
impl Default for LineDemo {
2021-02-14 20:39:04 +00:00
fn default() -> Self {
Self {
animate: true,
time: 0.0,
circle_radius: 1.5,
2021-02-14 20:39:04 +00:00
circle_center: Pos2::new(0.0, 0.0),
square: false,
legend: true,
2021-02-14 20:39:04 +00:00
proportional: true,
}
}
}
impl LineDemo {
2021-02-14 20:39:04 +00:00
fn options_ui(&mut self, ui: &mut Ui) {
let Self {
animate,
time: _,
circle_radius,
circle_center,
square,
legend,
2021-02-14 20:39:04 +00:00
proportional,
..
2021-02-14 20:39:04 +00:00
} = self;
ui.horizontal(|ui| {
ui.group(|ui| {
ui.vertical(|ui| {
ui.label("Circle:");
ui.add(
egui::DragValue::new(circle_radius)
.speed(0.1)
.clamp_range(0.0..=f32::INFINITY)
.prefix("r: "),
2021-02-14 20:39:04 +00:00
);
ui.horizontal(|ui| {
ui.add(
egui::DragValue::new(&mut circle_center.x)
.speed(0.1)
.prefix("x: "),
);
ui.add(
egui::DragValue::new(&mut circle_center.y)
.speed(1.0)
.prefix("y: "),
);
});
2021-02-14 20:39:04 +00:00
});
});
ui.vertical(|ui| {
ui.style_mut().wrap = Some(false);
ui.checkbox(animate, "animate");
ui.checkbox(square, "square view");
ui.checkbox(legend, "legend");
2021-02-14 20:39:04 +00:00
ui.checkbox(proportional, "proportional data axes");
});
});
}
fn circle(&self) -> Line {
2021-02-14 20:39:04 +00:00
let n = 512;
let circle = (0..=n).map(|i| {
let t = remap(i as f64, 0.0..=(n as f64), 0.0..=TAU);
let r = self.circle_radius;
2021-02-14 20:39:04 +00:00
Value::new(
r * t.cos() + self.circle_center.x as f64,
r * t.sin() + self.circle_center.y as f64,
)
});
Line::new(Values::from_values_iter(circle))
2021-02-14 20:39:04 +00:00
.color(Color32::from_rgb(100, 200, 100))
.name("circle")
}
fn sin(&self) -> Line {
let time = self.time;
Line::new(Values::from_explicit_callback(
move |x| 0.5 * (2.0 * x).sin() * time.sin(),
f64::NEG_INFINITY..=f64::INFINITY,
512,
))
.color(Color32::from_rgb(200, 100, 100))
.name("wave")
2021-02-14 20:39:04 +00:00
}
fn thingy(&self) -> Line {
let time = self.time;
Line::new(Values::from_parametric_callback(
move |t| ((2.0 * t + time).sin(), (3.0 * t).sin()),
0.0..=TAU,
256,
))
.color(Color32::from_rgb(100, 150, 250))
.name("x = sin(2t), y = sin(3t)")
2021-02-14 20:39:04 +00:00
}
}
impl Widget for &mut LineDemo {
fn ui(self, ui: &mut Ui) -> Response {
2021-02-14 20:39:04 +00:00
self.options_ui(ui);
if self.animate {
ui.ctx().request_repaint();
self.time += ui.input().unstable_dt.at_most(1.0 / 30.0) as f64;
};
let mut plot = Plot::new("Lines Demo")
.line(self.circle())
.line(self.sin())
.line(self.thingy())
.show_legend(self.legend);
2021-02-14 20:39:04 +00:00
if self.square {
plot = plot.view_aspect(1.0);
}
if self.proportional {
plot = plot.data_aspect(1.0);
}
ui.add(plot)
}
}
#[derive(PartialEq)]
struct MarkerDemo {
fill_markers: bool,
marker_radius: f32,
custom_marker_color: bool,
marker_color: Color32,
}
impl Default for MarkerDemo {
fn default() -> Self {
Self {
fill_markers: true,
marker_radius: 5.0,
custom_marker_color: false,
marker_color: Color32::GRAY,
}
}
}
impl MarkerDemo {
fn markers(&self) -> Vec<Points> {
MarkerShape::all()
.into_iter()
.enumerate()
.map(|(i, marker)| {
let y_offset = i as f32 * 0.5 + 1.0;
let mut points = Points::new(Values::from_values(vec![
Value::new(1.0, 0.0 + y_offset),
Value::new(2.0, 0.5 + y_offset),
Value::new(3.0, 0.0 + y_offset),
Value::new(4.0, 0.5 + y_offset),
Value::new(5.0, 0.0 + y_offset),
Value::new(6.0, 0.5 + y_offset),
]))
.name(format!("{:?}", marker))
.filled(self.fill_markers)
.radius(self.marker_radius)
.shape(marker);
if self.custom_marker_color {
points = points.color(self.marker_color);
}
points
})
.collect()
}
}
impl Widget for &mut MarkerDemo {
fn ui(self, ui: &mut Ui) -> Response {
ui.horizontal(|ui| {
ui.checkbox(&mut self.fill_markers, "fill markers");
ui.add(
egui::DragValue::new(&mut self.marker_radius)
.speed(0.1)
.clamp_range(0.0..=f32::INFINITY)
.prefix("marker radius: "),
);
ui.checkbox(&mut self.custom_marker_color, "custom marker color");
if self.custom_marker_color {
ui.color_edit_button_srgba(&mut self.marker_color);
}
});
let mut markers_plot = Plot::new("Markers Demo").data_aspect(1.0);
for marker in self.markers() {
markers_plot = markers_plot.points(marker);
}
ui.add(markers_plot)
}
}
#[derive(PartialEq, Eq)]
enum Panel {
Lines,
Markers,
}
impl Default for Panel {
fn default() -> Self {
Self::Lines
}
}
#[derive(PartialEq, Default)]
pub struct PlotDemo {
line_demo: LineDemo,
marker_demo: MarkerDemo,
open_panel: Panel,
}
impl super::Demo for PlotDemo {
fn name(&self) -> &'static str {
"🗠 Plot"
}
fn show(&mut self, ctx: &CtxRef, open: &mut bool) {
use super::View;
Window::new(self.name())
.open(open)
.default_size(vec2(400.0, 400.0))
.scroll(false)
.show(ctx, |ui| self.ui(ui));
}
}
impl super::View for PlotDemo {
fn ui(&mut self, ui: &mut Ui) {
ui.vertical_centered(|ui| {
egui::reset_button(ui, self);
ui.add(crate::__egui_github_link_file!());
ui.label("Pan by dragging, or scroll (+ shift = horizontal).");
if cfg!(target_arch = "wasm32") {
ui.label("Zoom with ctrl / ⌘ + mouse wheel, or with pinch gesture.");
} else if cfg!(target_os = "macos") {
ui.label("Zoom with ctrl / ⌘ + scroll.");
} else {
ui.label("Zoom with ctrl + scroll.");
}
ui.label("Reset view with double-click.");
});
ui.separator();
ui.horizontal(|ui| {
ui.selectable_value(&mut self.open_panel, Panel::Lines, "Lines");
ui.selectable_value(&mut self.open_panel, Panel::Markers, "Markers");
});
ui.separator();
match self.open_panel {
Panel::Lines => {
ui.add(&mut self.line_demo);
}
Panel::Markers => {
ui.add(&mut self.marker_demo);
}
}
2021-02-14 20:39:04 +00:00
}
}