use core::marker::PhantomData; use dyn_any::{DynAny, StaticType}; use graphene_core::generic::FnNode; use graphene_core::ops::{FlatMapResultNode, MapResultNode}; use graphene_core::raster::{Color, Image}; use graphene_core::structural::{ComposeNode, ConsNode, Then}; use graphene_core::value::ValueNode; use graphene_core::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) } } 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.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 GrayscaleNode; #[node_macro::node_fn(GrayscaleNode)] fn grayscale_image(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 InvertRGBNode; #[node_macro::node_fn(InvertRGBNode)] fn invert_image(mut image: Image) -> Image { for pixel in &mut image.data { *pixel = Color::from_rgbaf32_unchecked(1. - pixel.r(), 1. - pixel.g(), 1. - pixel.b(), pixel.a()); } image } #[derive(Debug, Clone, Copy)] pub struct HueSaturationNode { hue_shift: Hue, saturation_shift: Sat, lightness_shift: Lit, } #[node_macro::node_fn(HueSaturationNode)] fn shift_image_hsl(mut image: Image, hue_shift: f64, saturation_shift: f64, lightness_shift: f64) -> Image { let (hue_shift, saturation_shift, lightness_shift) = (hue_shift as f32, saturation_shift as f32, lightness_shift as f32); for pixel in &mut image.data { let [hue, saturation, lightness, alpha] = pixel.to_hsla(); *pixel = Color::from_hsla( (hue + hue_shift / 360.) % 1., (saturation + saturation_shift / 100.).clamp(0., 1.), (lightness + lightness_shift / 100.).clamp(0., 1.), alpha, ); } image } #[derive(Debug, Clone, Copy)] pub struct BrightnessContrastNode { brightness: Brightness, contrast: Contrast, } // From https://stackoverflow.com/questions/2976274/adjust-bitmap-image-brightness-contrast-using-c #[node_macro::node_fn(BrightnessContrastNode)] fn adjust_image_brightness_and_contrast(mut image: Image, brightness: f64, contrast: f64) -> Image { let (brightness, contrast) = (brightness as f32, contrast as f32); let factor = (259. * (contrast + 255.)) / (255. * (259. - contrast)); let channel = |channel: f32| ((factor * (channel * 255. + brightness - 128.) + 128.) / 255.).clamp(0., 1.); for pixel in &mut image.data { *pixel = Color::from_rgbaf32_unchecked(channel(pixel.r()), channel(pixel.g()), channel(pixel.b()), pixel.a()) } image } #[derive(Debug, Clone, Copy)] pub struct GammaNode { gamma: G, } // https://www.dfstudios.co.uk/articles/programming/image-programming-algorithms/image-processing-algorithms-part-6-gamma-correction/ #[node_macro::node_fn(GammaNode)] fn image_gamma(mut image: Image, gamma: f64) -> Image { let inverse_gamma = 1. / gamma; let channel = |channel: f32| channel.powf(inverse_gamma as f32); for pixel in &mut image.data { *pixel = Color::from_rgbaf32_unchecked(channel(pixel.r()), channel(pixel.g()), channel(pixel.b()), pixel.a()) } image } #[derive(Debug, Clone, Copy)] pub struct OpacityNode { opacity_multiplier: O, } #[node_macro::node_fn(OpacityNode)] fn image_opacity(mut image: Image, opacity_multiplier: f64) -> Image { let opacity_multiplier = opacity_multiplier as f32; for pixel in &mut image.data { *pixel = Color::from_rgbaf32_unchecked(pixel.r(), pixel.g(), pixel.b(), pixel.a() * opacity_multiplier) } image } #[derive(Debug, Clone, Copy)] pub struct PosterizeNode

{ posterize_value: P, } // Based on http://www.axiomx.com/posterize.htm #[node_macro::node_fn(PosterizeNode)] fn posterize(mut image: Image, posterize_value: f64) -> Image { let posterize_value = posterize_value as f32; let number_of_areas = posterize_value.recip(); let size_of_areas = (posterize_value - 1.).recip(); let channel = |channel: f32| (channel / number_of_areas).floor() * size_of_areas; for pixel in &mut image.data { *pixel = Color::from_rgbaf32_unchecked(channel(pixel.r()), channel(pixel.g()), channel(pixel.b()), pixel.a()) } image } #[derive(Debug, Clone, Copy)] pub struct ExposureNode { exposure: E, } // Based on https://stackoverflow.com/questions/12166117/what-is-the-math-behind-exposure-adjustment-on-photoshop #[node_macro::node_fn(ExposureNode)] fn exposure(mut image: Image, exposure: f64) -> Image { let multiplier = 2f32.powf(exposure as f32); let channel = |channel: f32| channel * multiplier; for pixel in &mut image.data { *pixel = Color::from_rgbaf32_unchecked(channel(pixel.r()), channel(pixel.g()), channel(pixel.b()), pixel.a()) } image } #[derive(Debug, Clone, Copy)] pub struct ImaginateNode { cached: E, } // Based on https://stackoverflow.com/questions/12166117/what-is-the-math-behind-exposure-adjustment-on-photoshop #[node_macro::node_fn(ImaginateNode)] fn imaginate(image: Image, cached: Option>) -> Image { info!("Imaginating image with {} pixels", image.data.len()); cached.map(|mut x| std::sync::Arc::make_mut(&mut x).clone()).unwrap_or(image) } #[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(); } }