use core::marker::PhantomData; use graphene_core::ops::FlatMapResultNode; use graphene_core::raster::color::Color; use graphene_core::structural::{ComposeNode, ConsNode}; use graphene_core::{generic::FnNode, ops::MapResultNode, structural::After, value::ValueNode, Node}; use image::Pixel; use std::path::Path; pub struct MapNode, I: IntoIterator, S>(pub MN, PhantomData<(S, I)>); impl, MN: Node + Copy, S> Node for MapNode { type Output = Vec; fn eval(self, input: I) -> Self::Output { input.into_iter().map(|x| self.0.eval(x)).collect() } } impl, MN: Node, S> MapNode { pub const fn new(mn: MN) -> Self { MapNode(mn, PhantomData) } } pub struct MapImageNode + Copy>(pub MN); impl<'n, MN: Node + Copy> Node for &'n MapImageNode { type Output = Image; fn eval(self, input: Image) -> Self::Output { Image { width: input.width, height: input.height, data: input.data.iter().map(|x| self.0.eval(*x)).collect(), } } } impl + Copy> MapImageNode { pub const fn new(mn: MN) -> Self { MapImageNode(mn) } } #[derive(Debug)] 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>(PhantomData<(P, FS)>); impl, FS: FileSystem> Node<(P, FS)> for FileNode { type Output = Result; fn eval(self, input: (P, FS)) -> Self::Output { let (path, fs) = input; fs.open(path) } } pub struct BufferNode; impl Node for BufferNode { type Output = Result, Error>; fn eval(self, mut reader: Reader) -> Self::Output { let mut buffer = Vec::new(); reader.read_to_end(&mut buffer)?; Ok(buffer) } } #[derive(Clone)] pub struct Image { pub width: u32, pub height: u32, pub data: Vec, } impl IntoIterator for Image { type Item = Color; type IntoIter = std::vec::IntoIter; fn into_iter(self) -> Self::IntoIter { self.data.into_iter() } } impl<'a> IntoIterator for &'a Image { type Item = &'a Color; type IntoIter = std::slice::Iter<'a, Color>; fn into_iter(self) -> Self::IntoIter { self.data.iter() } } pub fn file_node<'n, P: AsRef + 'n>() -> impl Node, Error>> { let fs = ValueNode(StdFs).clone(); let fs = ConsNode(fs); let file: ComposeNode<_, _, P> = FileNode(PhantomData).after(fs); FlatMapResultNode::new(BufferNode).after(file) } pub fn image_node<'n, P: AsRef + 'n>() -> impl Node> { 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: ComposeNode<_, _, P> = FlatMapResultNode::new(image_loader).after(file); 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, } }); MapResultNode::new(convert_image).after(image) } pub fn export_image_node<'n>() -> impl Node<(Image, &'n 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).into_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) }) } #[cfg(test)] mod test { use super::*; use graphene_core::raster::color::Color; use graphene_core::raster::GrayscaleNode; #[test] fn map_node() { let array = [Color::from_rgbaf32(1.0, 0.0, 0.0, 1.0).unwrap()]; let map = MapNode(GrayscaleNode, PhantomData); let values = map.eval(array.into_iter()); assert_eq!(values[0], Color::from_rgbaf32(0.33333334, 0.33333334, 0.33333334, 1.0).unwrap()); } #[test] fn load_image() { let image = image_node::<&str>(); let gray = MapImageNode::new(GrayscaleNode); let grayscale_picture = MapResultNode::new(&gray).after(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(); } }