use dioxus::prelude::*;
use freya_common::EventMessage;
use freya_elements::elements as dioxus_elements;
use freya_hooks::{use_canvas, use_platform};
use freya_node_state::Parse;
use skia_safe::{
textlayout::{ParagraphBuilder, ParagraphStyle, TextAlign, TextStyle},
Color, Paint, PaintStyle,
};
#[derive(Debug, PartialEq, Clone)]
pub struct GraphLine {
color: String,
points: Vec<Option<i32>>,
}
impl GraphLine {
pub fn new(color: &str, points: Vec<Option<i32>>) -> Self {
Self {
color: color.to_string(),
points,
}
}
}
#[derive(Debug, Props, PartialEq, Clone)]
pub struct GraphProps {
labels: Vec<String>,
data: Vec<GraphLine>,
#[props(default = "100%".to_string(), into)]
width: String,
#[props(default = "100%".to_string(), into)]
height: String,
}
#[allow(non_snake_case)]
pub fn Graph(cx: Scope<GraphProps>) -> Element {
let platform = use_platform(cx);
use_effect(cx, (cx.props,), move |_| async move {
platform.send(EventMessage::RequestRerender)
});
let canvas = use_canvas(cx, cx.props, |state| {
Box::new(move |canvas, font_collection, region| {
canvas.translate((region.min_x(), region.min_y()));
let mut paragraph_style = ParagraphStyle::default();
paragraph_style.set_text_align(TextAlign::Center);
let mut text_style = TextStyle::new();
text_style.set_color(Color::BLACK);
paragraph_style.set_text_style(&text_style);
let x_labels = &state.labels;
let x_height: f32 = 50.0;
let start_x = region.min_x();
let start_y = region.height() - x_height;
let height = region.height() - x_height;
let space_x = region.width() / x_labels.len() as f32;
let (smallest_y, biggest_y) = {
let mut smallest_y = 0;
let mut biggest_y = 0;
for line in state.data.iter() {
let max = line.points.iter().max().unwrap();
let min = line.points.iter().min().unwrap();
if let Some(max) = *max {
if max > biggest_y {
biggest_y = max;
}
}
if let Some(min) = *min {
if min < smallest_y {
smallest_y = min;
}
}
}
(smallest_y, biggest_y)
};
let y_axis_len = biggest_y - smallest_y;
let space_y = height / y_axis_len as f32;
for line in &state.data {
let mut paint = Paint::default();
paint.set_anti_alias(true);
paint.set_style(PaintStyle::Fill);
paint.set_color(Color::parse(&line.color).unwrap());
paint.set_stroke_width(3.0);
let mut previous_x = None;
let mut previous_y = None;
for (i, y_point) in line.points.iter().enumerate() {
let line_x = (space_x * i as f32) + start_x + (space_x / 2.0);
let new_previous_x = previous_x.unwrap_or(line_x);
if let Some(y_point) = y_point {
let line_y = start_y - (space_y * ((y_point - smallest_y) as f32));
let new_previous_y = previous_y.unwrap_or(line_y);
canvas.draw_circle((line_x, line_y), 5.0, &paint);
canvas.draw_line(
(new_previous_x, new_previous_y),
(line_x, line_y),
&paint,
);
previous_y = Some(line_y);
previous_x = Some(line_x);
} else {
previous_y = None;
previous_x = None;
}
}
}
let space_x = region.width() / x_labels.len() as f32;
for (i, point) in x_labels.iter().enumerate() {
let x = (space_x * i as f32) + start_x;
let mut paragrap_builder = ParagraphBuilder::new(¶graph_style, font_collection);
paragrap_builder.add_text(point);
let mut text = paragrap_builder.build();
text.layout(space_x);
text.paint(canvas, (x, start_y + x_height - 30.0));
}
canvas.restore();
})
});
let width = &cx.props.width;
let height = &cx.props.height;
render!(
rect {
width: "{width}",
height: "{height}",
padding: "15 5",
background: "white",
rect {
canvas_reference: canvas.attribute(cx),
width: "100%",
height: "100%",
}
}
)
}