use core::marker::PhantomData; use dyn_any::{DynAny, StaticType}; 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::Then, 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 + Copy> Node for 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<'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, 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>(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, Debug, PartialEq, DynAny)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] 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::new(fs); let file = fs.then(FileNode(PhantomData)); file.then(FlatMapResultNode::new(BufferNode)) } 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> = file.then(FlatMapResultNode::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<'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) }) } #[derive(Debug, Clone, Copy)] pub struct GrayscaleImageNode; impl Node for GrayscaleImageNode { type Output = Image; fn eval(self, mut image: Image) -> Image { for pixel in &mut image.data { let avg = (pixel.r() + pixel.g() + pixel.b()) / 3.; *pixel = Color::from_rgbaf32_unchecked(avg, avg, avg, pixel.a()); } image } } impl Node for &GrayscaleImageNode { type Output = Image; fn eval(self, mut image: Image) -> Image { for pixel in &mut image.data { let avg = (pixel.r() + pixel.g() + pixel.b()) / 3.; *pixel = Color::from_rgbaf32_unchecked(avg, avg, avg, pixel.a()); } image } } #[derive(Debug, Clone, Copy)] pub struct BrightenImageNode>(N); impl> Node for BrightenImageNode { type Output = Image; fn eval(self, mut image: Image) -> Image { let brightness = self.0.eval(()); let per_channel = |col: f32| (col + brightness / 255.).clamp(0., 1.); for pixel in &mut image.data { *pixel = Color::from_rgbaf32_unchecked(per_channel(pixel.r()), per_channel(pixel.g()), per_channel(pixel.b()), pixel.a()); } image } } impl + Copy> Node for &BrightenImageNode { type Output = Image; fn eval(self, mut image: Image) -> Image { let brightness = self.0.eval(()); let per_channel = |col: f32| (col + brightness / 255.).clamp(0., 1.); for pixel in &mut image.data { *pixel = Color::from_rgbaf32_unchecked(per_channel(pixel.r()), per_channel(pixel.g()), per_channel(pixel.b()), pixel.a()); } image } } impl + Copy> BrightenImageNode { pub fn new(node: N) -> Self { Self(node) } } #[derive(Debug, Clone, Copy)] pub struct HueShiftImage>(N); impl> Node for HueShiftImage { type Output = Image; fn eval(self, mut image: Image) -> Image { let hue_shift = self.0.eval(()); for pixel in &mut image.data { let [hue, saturation, luminance, alpha] = pixel.to_hsla(); *pixel = Color::from_hsla(hue + hue_shift / 360., saturation, luminance, alpha); } image } } impl + Copy> Node for &HueShiftImage { type Output = Image; fn eval(self, mut image: Image) -> Image { let hue_shift = self.0.eval(()); for pixel in &mut image.data { let [hue, saturation, luminance, alpha] = pixel.to_hsla(); *pixel = Color::from_hsla(hue + hue_shift / 360., saturation, luminance, alpha); } image } } impl + Copy> HueShiftImage { pub fn new(node: N) -> Self { Self(node) } } #[cfg(test)] mod test { use super::*; use graphene_core::raster::color::Color; use graphene_core::raster::GrayscaleColorNode; #[test] fn map_node() { let array = [Color::from_rgbaf32(1.0, 0.0, 0.0, 1.0).unwrap()]; let map = MapNode(GrayscaleColorNode, 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(GrayscaleColorNode); let grayscale_picture = image.then(MapResultNode::new(&gray)); 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(); } }