| |
|
|
| use crate::curve::{Curve, CurveManipulatorGroup, ValueMapperNode}; |
| use bezier_rs::{Bezier, TValue}; |
| use graphene_core::color::{Channel, Linear}; |
| use graphene_core::context::Ctx; |
|
|
| const WINDOW_SIZE: usize = 1024; |
|
|
| #[node_macro::node(category(""))] |
| fn generate_curves<C: Channel + Linear>(_: impl Ctx, curve: Curve, #[implementations(f32, f64)] _target_format: C) -> ValueMapperNode<C> { |
| let [mut pos, mut param]: [[f32; 2]; 2] = [[0.; 2], curve.first_handle]; |
| let mut lut = vec![C::from_f64(0.); WINDOW_SIZE]; |
| let end = CurveManipulatorGroup { |
| anchor: [1.; 2], |
| handles: [curve.last_handle, [0.; 2]], |
| }; |
| for sample in curve.manipulator_groups.iter().chain(std::iter::once(&end)) { |
| let [x0, y0, x1, y1, x2, y2, x3, y3] = [pos[0], pos[1], param[0], param[1], sample.handles[0][0], sample.handles[0][1], sample.anchor[0], sample.anchor[1]].map(f64::from); |
|
|
| let bezier = Bezier::from_cubic_coordinates(x0, y0, x1, y1, x2, y2, x3, y3); |
|
|
| let [left, right] = [pos[0], sample.anchor[0]].map(|c| c.clamp(0., 1.)); |
| let lut_index_left: usize = (left * (lut.len() - 1) as f32).floor() as _; |
| let lut_index_right: usize = (right * (lut.len() - 1) as f32).ceil() as _; |
| for index in lut_index_left..=lut_index_right { |
| let x = index as f64 / (lut.len() - 1) as f64; |
| let y = if x <= x0 { |
| y0 |
| } else if x >= x3 { |
| y3 |
| } else { |
| bezier.find_tvalues_for_x(x) |
| .next() |
| .map(|t| bezier.evaluate(TValue::Parametric(t.clamp(0., 1.))).y) |
| |
| .unwrap_or_else(|| (x - x0) / (x3 - x0) * (y3 - y0) + y0) |
| }; |
| lut[index] = C::from_f64(y); |
| } |
|
|
| pos = sample.anchor; |
| param = sample.handles[1]; |
| } |
| ValueMapperNode::new(lut) |
| } |
|
|