use dyn_any::{DynAny, StaticType}; use glam::DAffine2; use graphene_core::raster::{Color, Image, ImageFrame}; use graphene_core::Node; use std::path::Path; #[derive(Debug, DynAny)] pub enum Error { IO(std::io::Error), Image(image::ImageError), } impl From for Error { fn from(e: std::io::Error) -> Self { Error::IO(e) } } pub trait FileSystem { fn open>(&self, path: P) -> Result, Error>; } #[derive(Clone)] pub struct StdFs; impl FileSystem for StdFs { fn open>(&self, path: P) -> Result { Ok(Box::new(std::fs::File::open(path)?)) } } type Reader = Box; pub struct FileNode { fs: FileSystem, } #[node_macro::node_fn(FileNode)] fn file_node, FS: FileSystem>(path: P, fs: FS) -> Result { fs.open(path) } pub struct BufferNode; #[node_macro::node_fn(BufferNode)] fn buffer_node(reader: R) -> Result, Error> { Ok(std::io::Read::bytes(reader).collect::, _>>()?) } /* pub fn file_node<'i, 's: 'i, P: AsRef + 'i>() -> impl Node<'i, 's, P, Output = Result, Error>> { let fs = ValueNode(StdFs).then(CloneNode::new()); let file = FileNode::new(fs); file.then(FlatMapResultNode::new(ValueNode::new(BufferNode))) } pub fn image_node<'i, 's: 'i, P: AsRef + 'i>() -> impl Node<'i, 's, P, Output = Result> { let file = file_node(); let image_loader = FnNode::new(|data: Vec| image::load_from_memory(&data).map_err(Error::Image).map(|image| image.into_rgba32f())); let image = file.then(FlatMapResultNode::new(ValueNode::new(image_loader))); let convert_image = FnNode::new(|image: image::ImageBuffer<_, _>| { let data = image .enumerate_pixels() .map(|(_, _, pixel): (_, _, &image::Rgba)| { let c = pixel.channels(); Color::from_rgbaf32(c[0], c[1], c[2], c[3]).unwrap() }) .collect(); Image { width: image.width(), height: image.height(), data, } }); image.then(MapResultNode::new(convert_image)) } pub fn export_image_node<'i, 's: 'i>() -> impl Node<'i, 's, (Image, &'i str), Output = Result<(), Error>> { FnNode::new(|input: (Image, &str)| { let (image, path) = input; let mut new_image = image::ImageBuffer::new(image.width, image.height); for ((x, y, pixel), color) in new_image.enumerate_pixels_mut().zip(image.data.iter()) { let color: Color = *color; assert!(x < image.width); assert!(y < image.height); *pixel = image::Rgba(color.to_rgba8()) } new_image.save(path).map_err(Error::Image) }) } */ #[derive(Debug, Clone, Copy)] pub struct MapImageNode { map_fn: MapFn, } #[node_macro::node_fn(MapImageNode)] fn map_image(image: Image, map_fn: &'any_input MapFn) -> Image where MapFn: for<'any_input> Node<'any_input, Color, Output = Color> + 'input, { let mut image = image; for pixel in &mut image.data { *pixel = map_fn.eval(*pixel); } image } #[derive(Debug, Clone, Copy)] pub struct MapImageFrameNode { map_fn: MapFn, } #[node_macro::node_fn(MapImageFrameNode)] fn map_image(mut image_frame: ImageFrame, map_fn: &'any_input MapFn) -> ImageFrame where MapFn: for<'any_input> Node<'any_input, Color, Output = Color> + 'input, { for pixel in &mut image_frame.image.data { *pixel = map_fn.eval(*pixel); } image_frame } #[derive(Debug, Clone, Copy)] pub struct BlendImageNode { second: Second, map_fn: MapFn, } // TODO: Implement proper blending #[node_macro::node_fn(BlendImageNode)] fn blend_image(image: ImageFrame, second: ImageFrame, map_fn: &'any_input MapFn) -> ImageFrame where MapFn: for<'any_input> Node<'any_input, (Color, Color), Output = Color> + 'input, { let mut image = image; for (pixel, sec_pixel) in &mut image.image.data.iter_mut().zip(second.image.data.iter()) { *pixel = map_fn.eval((*pixel, *sec_pixel)); } image } #[derive(Debug, Clone, Copy)] pub struct ImaginateNode { cached: E, } #[node_macro::node_fn(ImaginateNode)] fn imaginate(image_frame: ImageFrame, cached: Option>) -> ImageFrame { info!("Imaginating image with {} pixels", image_frame.image.data.len()); let cached_image = cached.map(|mut x| std::sync::Arc::make_mut(&mut x).clone()).unwrap_or(image_frame.image); ImageFrame { image: cached_image, transform: image_frame.transform, } } #[derive(Debug, Clone, Copy)] pub struct ImageFrameNode { transform: Transform, } #[node_macro::node_fn(ImageFrameNode)] fn image_frame(image: Image, transform: DAffine2) -> graphene_core::raster::ImageFrame { graphene_core::raster::ImageFrame { image, transform } } #[cfg(test)] mod test { #[test] fn load_image() { // TODO: reenable this test /* let image = image_node::<&str>(); let grayscale_picture = image.then(MapResultNode::new(&image)); let export = export_image_node(); let picture = grayscale_picture.eval("test-image-1.png").expect("Failed to load image"); export.eval((picture, "test-image-1-result.png")).unwrap(); */ } }