#![allow(clippy::comparison_chain)]
use color::Hsva;
use crate::*;
#[derive(Clone, Copy, Debug, PartialEq)]
pub struct Value {
pub x: f64,
pub y: f64,
}
impl Value {
pub fn new(x: impl Into<f64>, y: impl Into<f64>) -> Self {
Self {
x: x.into(),
y: y.into(),
}
}
}
#[derive(Clone, Copy, PartialEq)]
struct Bounds {
min: [f64; 2],
max: [f64; 2],
}
impl Bounds {
pub const NOTHING: Self = Self {
min: [f64::INFINITY; 2],
max: [-f64::INFINITY; 2],
};
pub fn width(&self) -> f64 {
self.max[0] - self.min[0]
}
pub fn height(&self) -> f64 {
self.max[1] - self.min[1]
}
pub fn is_finite(&self) -> bool {
self.min[0].is_finite()
&& self.min[1].is_finite()
&& self.max[0].is_finite()
&& self.max[1].is_finite()
}
pub fn extend_with(&mut self, value: &Value) {
self.extend_with_x(value.x);
self.extend_with_y(value.y);
}
pub fn extend_with_x(&mut self, x: f64) {
self.min[0] = self.min[0].min(x);
self.max[0] = self.max[0].max(x);
}
pub fn extend_with_y(&mut self, y: f64) {
self.min[1] = self.min[1].min(y);
self.max[1] = self.max[1].max(y);
}
pub fn expand_x(&mut self, pad: f64) {
self.min[0] -= pad;
self.max[0] += pad;
}
pub fn expand_y(&mut self, pad: f64) {
self.min[1] -= pad;
self.max[1] += pad;
}
pub fn union_mut(&mut self, other: &Bounds) {
self.min[0] = self.min[0].min(other.min[0]);
self.min[1] = self.min[1].min(other.min[1]);
self.max[0] = self.max[0].max(other.max[0]);
self.max[1] = self.max[1].max(other.max[1]);
}
}
#[derive(Clone, Copy, Debug, PartialEq)]
pub struct HLine {
y: f64,
stroke: Stroke,
}
impl HLine {
pub fn new(y: impl Into<f64>, stroke: impl Into<Stroke>) -> Self {
Self {
y: y.into(),
stroke: stroke.into(),
}
}
}
#[derive(Clone, Copy, Debug, PartialEq)]
pub struct VLine {
x: f64,
stroke: Stroke,
}
impl VLine {
pub fn new(x: impl Into<f64>, stroke: impl Into<Stroke>) -> Self {
Self {
x: x.into(),
stroke: stroke.into(),
}
}
}
#[derive(Clone, PartialEq)]
pub struct Curve {
values: Vec<Value>,
bounds: Bounds,
stroke: Stroke,
name: String,
}
impl Curve {
pub fn from_values(values: Vec<Value>) -> Self {
let mut bounds = Bounds::NOTHING;
for value in &values {
bounds.extend_with(value);
}
Self {
values,
bounds,
stroke: Stroke::new(2.0, Color32::TRANSPARENT),
name: Default::default(),
}
}
pub fn from_values_iter(iter: impl Iterator<Item = Value>) -> Self {
Self::from_values(iter.collect())
}
pub fn from_ys_f32(ys: &[f32]) -> Self {
let values: Vec<Value> = ys
.iter()
.enumerate()
.map(|(i, &y)| Value {
x: i as f64,
y: y as f64,
})
.collect();
Self::from_values(values)
}
pub fn stroke(mut self, stroke: impl Into<Stroke>) -> Self {
self.stroke = stroke.into();
self
}
pub fn width(mut self, width: f32) -> Self {
self.stroke.width = width;
self
}
pub fn color(mut self, color: impl Into<Color32>) -> Self {
self.stroke.color = color.into();
self
}
pub fn name(mut self, name: impl Into<String>) -> Self {
self.name = name.into();
self
}
}
#[derive(Clone, PartialEq)]
pub struct Plot {
next_auto_color_idx: usize,
curves: Vec<Curve>,
hlines: Vec<HLine>,
vlines: Vec<VLine>,
bounds: Bounds,
symmetrical_x_bounds: bool,
symmetrical_y_bounds: bool,
margin_fraction: Vec2,
min_size: Vec2,
width: Option<f32>,
height: Option<f32>,
data_aspect: Option<f32>,
view_aspect: Option<f32>,
show_x: bool,
show_y: bool,
}
impl Default for Plot {
fn default() -> Self {
Self {
next_auto_color_idx: 0,
curves: Default::default(),
hlines: Default::default(),
vlines: Default::default(),
bounds: Bounds::NOTHING,
symmetrical_x_bounds: false,
symmetrical_y_bounds: false,
margin_fraction: Vec2::splat(0.05),
min_size: Vec2::splat(64.0),
width: None,
height: None,
data_aspect: None,
view_aspect: None,
show_x: true,
show_y: true,
}
}
}
impl Plot {
fn auto_color(&mut self, color: &mut Color32) {
if *color == Color32::TRANSPARENT {
let i = self.next_auto_color_idx;
self.next_auto_color_idx += 1;
let golden_ratio = (5.0_f32.sqrt() - 1.0) / 2.0;
let h = i as f32 * golden_ratio;
*color = Hsva::new(h, 0.85, 0.5, 1.0).into();
}
}
pub fn curve(mut self, mut curve: Curve) -> Self {
self.auto_color(&mut curve.stroke.color);
self.bounds.union_mut(&curve.bounds);
self.curves.push(curve);
self
}
pub fn hline(mut self, mut hline: HLine) -> Self {
self.auto_color(&mut hline.stroke.color);
self = self.include_y(hline.y);
self.hlines.push(hline);
self
}
pub fn vline(mut self, mut vline: VLine) -> Self {
self.auto_color(&mut vline.stroke.color);
self = self.include_x(vline.x);
self.vlines.push(vline);
self
}
pub fn include_x(mut self, x: impl Into<f64>) -> Self {
self.bounds.extend_with_x(x.into());
self
}
pub fn include_y(mut self, y: impl Into<f64>) -> Self {
self.bounds.extend_with_y(y.into());
self
}
pub fn symmetrical_x_bounds(mut self, symmetrical_x_bounds: bool) -> Self {
self.symmetrical_x_bounds = symmetrical_x_bounds;
self
}
pub fn symmetrical_y_bounds(mut self, symmetrical_y_bounds: bool) -> Self {
self.symmetrical_y_bounds = symmetrical_y_bounds;
self
}
pub fn data_aspect(mut self, data_aspect: f32) -> Self {
self.data_aspect = Some(data_aspect);
self
}
pub fn view_aspect(mut self, view_aspect: f32) -> Self {
self.view_aspect = Some(view_aspect);
self
}
pub fn width(mut self, width: f32) -> Self {
self.min_size.x = width;
self.width = Some(width);
self
}
pub fn height(mut self, height: f32) -> Self {
self.min_size.y = height;
self.height = Some(height);
self
}
pub fn min_size(mut self, min_size: Vec2) -> Self {
self.min_size = min_size;
self
}
pub fn show_x(mut self, show_x: bool) -> Self {
self.show_x = show_x;
self
}
pub fn show_y(mut self, show_y: bool) -> Self {
self.show_y = show_y;
self
}
}
impl Widget for Plot {
fn ui(self, ui: &mut Ui) -> Response {
let Self {
next_auto_color_idx: _,
curves,
hlines,
vlines,
bounds,
symmetrical_x_bounds,
symmetrical_y_bounds,
margin_fraction,
width,
height,
min_size,
data_aspect,
view_aspect,
show_x,
show_y,
} = self;
let size = {
let width = width.map(|w| w.at_least(min_size.x));
let height = height.map(|w| w.at_least(min_size.y));
let width = width.unwrap_or_else(|| {
if let (Some(height), Some(aspect)) = (height, view_aspect) {
height * aspect
} else {
ui.available_size_before_wrap_finite().x
}
});
let width = width.at_least(min_size.x);
let height = height.unwrap_or_else(|| {
if let Some(aspect) = view_aspect {
width / aspect
} else {
ui.available_size_before_wrap_finite().y
}
});
let height = height.at_least(min_size.y);
vec2(width, height)
};
let (rect, response) = ui.allocate_exact_size(size, Sense::hover());
let mut bounds = bounds;
if symmetrical_x_bounds {
let x_abs = bounds.min[0].abs().max(bounds.max[0].abs());
bounds.min[0] = -x_abs;
bounds.max[0] = x_abs;
};
if symmetrical_y_bounds {
let y_abs = bounds.min[1].abs().max(bounds.max[1].abs());
bounds.min[1] = -y_abs;
bounds.max[1] = y_abs;
};
bounds.expand_x(margin_fraction.x as f64 * bounds.width());
bounds.expand_y(margin_fraction.y as f64 * bounds.height());
if let Some(data_aspect) = data_aspect {
let data_aspect = data_aspect as f64;
let rw = rect.width() as f64;
let rh = rect.height() as f64;
let current_data_aspect = (bounds.width() / rw) / (bounds.height() / rh);
if current_data_aspect < data_aspect {
bounds.expand_x((data_aspect / current_data_aspect - 1.0) * bounds.width() * 0.5);
} else {
bounds.expand_y((current_data_aspect / data_aspect - 1.0) * bounds.height() * 0.5);
}
}
ui.painter().add(Shape::Rect {
rect,
corner_radius: 2.0,
fill: ui.visuals().extreme_bg_color,
stroke: ui.visuals().window_stroke(),
});
if bounds.is_finite() && bounds.width() > 0.0 && bounds.height() > 0.0 {
let prepared = Prepared {
curves,
hlines,
vlines,
rect,
bounds,
show_x,
show_y,
};
prepared.ui(ui, &response);
}
response
}
}
struct Prepared {
curves: Vec<Curve>,
hlines: Vec<HLine>,
vlines: Vec<VLine>,
rect: Rect,
bounds: Bounds,
show_x: bool,
show_y: bool,
}
impl Prepared {
fn position_from_value(&self, value: &Value) -> Pos2 {
let x = remap(
value.x,
self.bounds.min[0]..=self.bounds.max[0],
(self.rect.left() as f64)..=(self.rect.right() as f64),
);
let y = remap(
value.y,
self.bounds.min[1]..=self.bounds.max[1],
(self.rect.bottom() as f64)..=(self.rect.top() as f64),
);
pos2(x as f32, y as f32)
}
fn dpos_dvalue_x(&self) -> f64 {
self.rect.width() as f64 / self.bounds.width()
}
fn dpos_dvalue_y(&self) -> f64 {
-self.rect.height() as f64 / self.bounds.height()
}
fn dpos_dvalue(&self) -> [f64; 2] {
[self.dpos_dvalue_x(), self.dpos_dvalue_y()]
}
fn dvalue_dpos(&self) -> [f64; 2] {
[1.0 / self.dpos_dvalue_x(), 1.0 / self.dpos_dvalue_y()]
}
fn value_from_position(&self, pos: Pos2) -> Value {
let x = remap(
pos.x as f64,
(self.rect.left() as f64)..=(self.rect.right() as f64),
self.bounds.min[0]..=self.bounds.max[0],
);
let y = remap(
pos.y as f64,
(self.rect.bottom() as f64)..=(self.rect.top() as f64),
self.bounds.min[1]..=self.bounds.max[1],
);
Value::new(x, y)
}
fn ui(&self, ui: &mut Ui, response: &Response) {
let mut shapes = Vec::with_capacity(self.hlines.len() + self.curves.len() + 2);
for d in 0..2 {
self.paint_axis(ui, d, &mut shapes);
}
for &hline in &self.hlines {
let HLine { y, stroke } = hline;
let points = [
self.position_from_value(&Value::new(self.bounds.min[0], y)),
self.position_from_value(&Value::new(self.bounds.max[0], y)),
];
shapes.push(Shape::line_segment(points, stroke));
}
for &vline in &self.vlines {
let VLine { x, stroke } = vline;
let points = [
self.position_from_value(&Value::new(x, self.bounds.min[1])),
self.position_from_value(&Value::new(x, self.bounds.max[1])),
];
shapes.push(Shape::line_segment(points, stroke));
}
for curve in &self.curves {
let stroke = curve.stroke;
let values = &curve.values;
if values.len() == 1 {
let point = self.position_from_value(&values[0]);
shapes.push(Shape::circle_filled(
point,
stroke.width / 2.0,
stroke.color,
));
} else if values.len() > 1 {
shapes.push(Shape::line(
values.iter().map(|v| self.position_from_value(v)).collect(),
stroke,
));
}
}
if response.hovered() {
if let Some(pointer) = ui.input().pointer.tooltip_pos() {
self.hover(ui, pointer, &mut shapes);
}
}
ui.painter().sub_region(self.rect).extend(shapes);
}
fn paint_axis(&self, ui: &Ui, axis: usize, shapes: &mut Vec<Shape>) {
let bounds = self.bounds;
let text_style = TextStyle::Body;
let base: f64 = 10.0;
let min_label_spacing_in_points = 60.0;
let step_size = self.dvalue_dpos()[axis] * min_label_spacing_in_points;
let step_size = base.powi(step_size.abs().log(base).ceil() as i32);
let step_size_in_points = (self.dpos_dvalue()[axis] * step_size) as f32;
let value_cross = clamp(0.0, bounds.min[1 - axis]..=bounds.max[1 - axis]);
for i in 0.. {
let value_main = step_size * (bounds.min[axis] / step_size + i as f64).floor();
if value_main > bounds.max[axis] {
break;
}
let value = if axis == 0 {
Value::new(value_main, value_cross)
} else {
Value::new(value_cross, value_main)
};
let pos_in_gui = self.position_from_value(&value);
{
let n = if step_size_in_points.abs() < 40.0 {
2
} else if step_size_in_points.abs() < 100.0 {
5
} else {
10
};
for i in 0..n {
let strength = if i == 0 && value_main == 0.0 {
Strength::Strong
} else if i == 0 {
Strength::Middle
} else {
Strength::Weak
};
let color = line_color(ui, strength);
let mut pos_in_gui = pos_in_gui;
pos_in_gui[axis] += step_size_in_points * (i as f32) / (n as f32);
let mut p0 = pos_in_gui;
let mut p1 = pos_in_gui;
p0[1 - axis] = self.rect.min[1 - axis];
p1[1 - axis] = self.rect.max[1 - axis];
shapes.push(Shape::line_segment([p0, p1], Stroke::new(1.0, color)));
}
}
let text = emath::round_to_decimals(value_main, 5).to_string();
let font = &ui.fonts()[text_style];
let galley = font.layout_multiline(text, f32::INFINITY);
let mut text_pos = pos_in_gui + vec2(1.0, -galley.size.y);
text_pos[1 - axis] = text_pos[1 - axis]
.at_most(self.rect.max[1 - axis] - galley.size[1 - axis] - 2.0)
.at_least(self.rect.min[1 - axis] + 1.0);
shapes.push(Shape::Text {
pos: text_pos,
galley,
text_style,
color: ui.visuals().text_color(),
fake_italics: false,
});
}
}
fn hover(&self, ui: &Ui, pointer: Pos2, shapes: &mut Vec<Shape>) {
if !self.show_x && !self.show_y {
return;
}
let interact_radius: f32 = 16.0;
let mut closest_value = None;
let mut closest_curve = None;
let mut closest_dist_sq = interact_radius.powi(2);
for curve in &self.curves {
for value in &curve.values {
let pos = self.position_from_value(value);
let dist_sq = pointer.distance_sq(pos);
if dist_sq < closest_dist_sq {
closest_dist_sq = dist_sq;
closest_value = Some(value);
closest_curve = Some(curve);
}
}
}
let mut prefix = String::new();
if let Some(curve) = closest_curve {
if !curve.name.is_empty() {
prefix = format!("{}\n", curve.name);
}
}
let line_color = line_color(ui, Strength::Strong);
let value = if let Some(value) = closest_value {
let position = self.position_from_value(value);
shapes.push(Shape::circle_filled(position, 3.0, line_color));
*value
} else {
self.value_from_position(pointer)
};
let pointer = self.position_from_value(&value);
let rect = self.rect;
if self.show_x {
shapes.push(Shape::line_segment(
[pos2(pointer.x, rect.top()), pos2(pointer.x, rect.bottom())],
(1.0, line_color),
));
}
if self.show_y {
shapes.push(Shape::line_segment(
[pos2(rect.left(), pointer.y), pos2(rect.right(), pointer.y)],
(1.0, line_color),
));
}
let text = {
let scale = self.dvalue_dpos();
let x_decimals = ((-scale[0].abs().log10()).ceil().at_least(0.0) as usize).at_most(6);
let y_decimals = ((-scale[1].abs().log10()).ceil().at_least(0.0) as usize).at_most(6);
if self.show_x && self.show_y {
format!(
"{}x = {:.*}\ny = {:.*}",
prefix, x_decimals, value.x, y_decimals, value.y
)
} else if self.show_x {
format!("{}x = {:.*}", prefix, x_decimals, value.x)
} else if self.show_y {
format!("{}y = {:.*}", prefix, y_decimals, value.y)
} else {
unreachable!()
}
};
shapes.push(Shape::text(
ui.fonts(),
pointer + vec2(3.0, -2.0),
Align2::LEFT_BOTTOM,
text,
TextStyle::Body,
ui.visuals().text_color(),
));
}
}
#[derive(Clone, Copy)]
enum Strength {
Strong,
Middle,
Weak,
}
fn line_color(ui: &Ui, strength: Strength) -> Color32 {
if ui.visuals().dark_mode {
match strength {
Strength::Strong => Color32::from_gray(130).additive(),
Strength::Middle => Color32::from_gray(55).additive(),
Strength::Weak => Color32::from_gray(25).additive(),
}
} else {
match strength {
Strength::Strong => Color32::from_black_alpha(220),
Strength::Middle => Color32::from_black_alpha(120),
Strength::Weak => Color32::from_black_alpha(35),
}
}
}