use std::rc::Rc;
use gpui::{App, Bounds, Hsla, Pixels, Window};
use gpui_component_macros::IntoPlot;
use num_traits::Zero;
use crate::{
plot::{
shape::{Arc, ArcData, Pie},
Plot,
},
ActiveTheme, PixelsExt,
};
#[derive(IntoPlot)]
pub struct PieChart<T: 'static> {
data: Vec<T>,
inner_radius: f32,
inner_radius_fn: Option<Rc<dyn Fn(&ArcData<T>) -> f32 + 'static>>,
outer_radius: f32,
outer_radius_fn: Option<Rc<dyn Fn(&ArcData<T>) -> f32 + 'static>>,
pad_angle: f32,
value: Option<Rc<dyn Fn(&T) -> f32>>,
color: Option<Rc<dyn Fn(&T) -> Hsla>>,
}
impl<T> PieChart<T> {
pub fn new<I>(data: I) -> Self
where
I: IntoIterator<Item = T>,
{
Self {
data: data.into_iter().collect(),
inner_radius: 0.,
inner_radius_fn: None,
outer_radius: 0.,
outer_radius_fn: None,
pad_angle: 0.,
value: None,
color: None,
}
}
pub fn inner_radius(mut self, inner_radius: f32) -> Self {
self.inner_radius = inner_radius;
self
}
pub fn inner_radius_fn(
mut self,
inner_radius_fn: impl Fn(&ArcData<T>) -> f32 + 'static,
) -> Self {
self.inner_radius_fn = Some(Rc::new(inner_radius_fn));
self
}
fn get_inner_radius(&self, arc: &ArcData<T>) -> f32 {
if let Some(inner_radius_fn) = self.inner_radius_fn.as_ref() {
inner_radius_fn(arc)
} else {
self.inner_radius
}
}
pub fn outer_radius(mut self, outer_radius: f32) -> Self {
self.outer_radius = outer_radius;
self
}
pub fn outer_radius_fn(
mut self,
outer_radius_fn: impl Fn(&ArcData<T>) -> f32 + 'static,
) -> Self {
self.outer_radius_fn = Some(Rc::new(outer_radius_fn));
self
}
fn get_outer_radius(&self, arc: &ArcData<T>) -> f32 {
if let Some(outer_radius_fn) = self.outer_radius_fn.as_ref() {
outer_radius_fn(arc)
} else {
self.outer_radius
}
}
pub fn pad_angle(mut self, pad_angle: f32) -> Self {
self.pad_angle = pad_angle;
self
}
pub fn value(mut self, value: impl Fn(&T) -> f32 + 'static) -> Self {
self.value = Some(Rc::new(value));
self
}
pub fn color<H>(mut self, color: impl Fn(&T) -> H + 'static) -> Self
where
H: Into<Hsla> + 'static,
{
self.color = Some(Rc::new(move |t| color(t).into()));
self
}
}
impl<T> Plot for PieChart<T> {
fn paint(&mut self, bounds: Bounds<Pixels>, window: &mut Window, cx: &mut App) {
let Some(value_fn) = self.value.as_ref() else {
return;
};
let outer_radius = if self.outer_radius.is_zero() {
bounds.size.height.as_f32() * 0.4
} else {
self.outer_radius
};
let arc = Arc::new()
.inner_radius(self.inner_radius)
.outer_radius(outer_radius);
let value_fn = value_fn.clone();
let mut pie = Pie::<T>::new().value(move |d| Some(value_fn(d)));
pie = pie.pad_angle(self.pad_angle);
let arcs = pie.arcs(&self.data);
for a in &arcs {
let inner_radius = self.get_inner_radius(a);
let outer_radius = self.get_outer_radius(a);
arc.paint(
a,
if let Some(color_fn) = self.color.as_ref() {
color_fn(a.data)
} else {
cx.theme().chart_2
},
Some(inner_radius),
Some(outer_radius),
&bounds,
window,
);
}
}
}