#![allow(clippy::too_many_arguments)] #[cfg(feature = "alloc")] use crate::raster::curve::{Curve, CurveManipulatorGroup, ValueMapperNode}; #[cfg(feature = "alloc")] use crate::raster::image::{Image, ImageFrameTable}; use crate::raster::{Channel, Color, Pixel}; use crate::registry::types::{Angle, Percentage, SignedPercentage}; use crate::vector::VectorDataTable; use crate::vector::style::GradientStops; use crate::{Ctx, Node}; use crate::{GraphicElement, GraphicGroupTable}; use core::cmp::Ordering; use core::fmt::Debug; use dyn_any::DynAny; #[cfg(feature = "serde")] #[cfg(target_arch = "spirv")] use spirv_std::num_traits::float::Float; // TODO: Implement the following: // Color Balance // Aims for interoperable compatibility with: // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=%27blnc%27%20%3D%20Color%20Balance // // Photo Filter // Aims for interoperable compatibility with: // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=%27phfl%27%20%3D%20Photo%20Filter // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=of%20the%20file.-,Photo%20Filter,-Key%20is%20%27phfl // // Color Lookup // Aims for interoperable compatibility with: // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=%27clrL%27%20%3D%20Color%20Lookup // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=Color%20Lookup%20(Photoshop%20CS6 #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] #[cfg_attr(feature = "std", derive(specta::Type))] #[derive(Debug, Default, Clone, Copy, Eq, PartialEq, DynAny, Hash)] pub enum LuminanceCalculation { #[default] SRGB, Perceptual, AverageChannels, MinimumChannels, MaximumChannels, } impl LuminanceCalculation { pub fn list() -> [LuminanceCalculation; 5] { [ LuminanceCalculation::SRGB, LuminanceCalculation::Perceptual, LuminanceCalculation::AverageChannels, LuminanceCalculation::MinimumChannels, LuminanceCalculation::MaximumChannels, ] } } impl core::fmt::Display for LuminanceCalculation { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { match self { LuminanceCalculation::SRGB => write!(f, "sRGB"), LuminanceCalculation::Perceptual => write!(f, "Perceptual"), LuminanceCalculation::AverageChannels => write!(f, "Average Channels"), LuminanceCalculation::MinimumChannels => write!(f, "Minimum Channels"), LuminanceCalculation::MaximumChannels => write!(f, "Maximum Channels"), } } } #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] #[cfg_attr(feature = "std", derive(specta::Type))] #[derive(Debug, Default, Clone, Copy, Eq, PartialEq, DynAny, Hash)] #[repr(i32)] // TODO: Enable Int8 capability for SPIR-V so that we don't need this? pub enum BlendMode { // Basic group #[default] Normal, // Darken group Darken, Multiply, ColorBurn, LinearBurn, DarkerColor, // Lighten group Lighten, Screen, ColorDodge, LinearDodge, LighterColor, // Contrast group Overlay, SoftLight, HardLight, VividLight, LinearLight, PinLight, HardMix, // Inversion group Difference, Exclusion, Subtract, Divide, // Component group Hue, Saturation, Color, Luminosity, // Other stuff Erase, Restore, MultiplyAlpha, } impl BlendMode { /// All standard blend modes ordered by group. pub fn list() -> [&'static [BlendMode]; 6] { use BlendMode::*; [ // Normal group &[Normal], // Darken group &[Darken, Multiply, ColorBurn, LinearBurn, DarkerColor], // Lighten group &[Lighten, Screen, ColorDodge, LinearDodge, LighterColor], // Contrast group &[Overlay, SoftLight, HardLight, VividLight, LinearLight, PinLight, HardMix], // Inversion group &[Difference, Exclusion, Subtract, Divide], // Component group &[Hue, Saturation, Color, Luminosity], ] } /// The subset of [`BlendMode::list()`] that is supported by SVG. pub fn list_svg_subset() -> [&'static [BlendMode]; 6] { use BlendMode::*; [ // Normal group &[Normal], // Darken group &[Darken, Multiply, ColorBurn], // Lighten group &[Lighten, Screen, ColorDodge], // Contrast group &[Overlay, SoftLight, HardLight], // Inversion group &[Difference, Exclusion], // Component group &[Hue, Saturation, Color, Luminosity], ] } pub fn index_in_list(&self) -> Option { Self::list().iter().flat_map(|x| x.iter()).position(|&blend_mode| blend_mode == *self) } pub fn index_in_list_svg_subset(&self) -> Option { Self::list_svg_subset().iter().flat_map(|x| x.iter()).position(|&blend_mode| blend_mode == *self) } /// Convert the enum to the CSS string for the blend mode. /// [Read more](https://developer.mozilla.org/en-US/docs/Web/CSS/blend-mode#values) pub fn to_svg_style_name(&self) -> Option<&'static str> { match self { // Normal group BlendMode::Normal => Some("normal"), // Darken group BlendMode::Darken => Some("darken"), BlendMode::Multiply => Some("multiply"), BlendMode::ColorBurn => Some("color-burn"), // Lighten group BlendMode::Lighten => Some("lighten"), BlendMode::Screen => Some("screen"), BlendMode::ColorDodge => Some("color-dodge"), // Contrast group BlendMode::Overlay => Some("overlay"), BlendMode::SoftLight => Some("soft-light"), BlendMode::HardLight => Some("hard-light"), // Inversion group BlendMode::Difference => Some("difference"), BlendMode::Exclusion => Some("exclusion"), // Component group BlendMode::Hue => Some("hue"), BlendMode::Saturation => Some("saturation"), BlendMode::Color => Some("color"), BlendMode::Luminosity => Some("luminosity"), _ => None, } } /// Renders the blend mode CSS style declaration. pub fn render(&self) -> String { format!( r#" mix-blend-mode: {};"#, self.to_svg_style_name().unwrap_or_else(|| { warn!("Unsupported blend mode {self:?}"); "normal" }) ) } } impl core::fmt::Display for BlendMode { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { match self { // Normal group BlendMode::Normal => write!(f, "Normal"), // Darken group BlendMode::Darken => write!(f, "Darken"), BlendMode::Multiply => write!(f, "Multiply"), BlendMode::ColorBurn => write!(f, "Color Burn"), BlendMode::LinearBurn => write!(f, "Linear Burn"), BlendMode::DarkerColor => write!(f, "Darker Color"), // Lighten group BlendMode::Lighten => write!(f, "Lighten"), BlendMode::Screen => write!(f, "Screen"), BlendMode::ColorDodge => write!(f, "Color Dodge"), BlendMode::LinearDodge => write!(f, "Linear Dodge"), BlendMode::LighterColor => write!(f, "Lighter Color"), // Contrast group BlendMode::Overlay => write!(f, "Overlay"), BlendMode::SoftLight => write!(f, "Soft Light"), BlendMode::HardLight => write!(f, "Hard Light"), BlendMode::VividLight => write!(f, "Vivid Light"), BlendMode::LinearLight => write!(f, "Linear Light"), BlendMode::PinLight => write!(f, "Pin Light"), BlendMode::HardMix => write!(f, "Hard Mix"), // Inversion group BlendMode::Difference => write!(f, "Difference"), BlendMode::Exclusion => write!(f, "Exclusion"), BlendMode::Subtract => write!(f, "Subtract"), BlendMode::Divide => write!(f, "Divide"), // Component group BlendMode::Hue => write!(f, "Hue"), BlendMode::Saturation => write!(f, "Saturation"), BlendMode::Color => write!(f, "Color"), BlendMode::Luminosity => write!(f, "Luminosity"), // Other utility blend modes (hidden from the normal list) BlendMode::Erase => write!(f, "Erase"), BlendMode::Restore => write!(f, "Restore"), BlendMode::MultiplyAlpha => write!(f, "Multiply Alpha"), } } } #[cfg(feature = "vello")] impl From for vello::peniko::Mix { fn from(val: BlendMode) -> Self { match val { // Normal group BlendMode::Normal => vello::peniko::Mix::Normal, // Darken group BlendMode::Darken => vello::peniko::Mix::Darken, BlendMode::Multiply => vello::peniko::Mix::Multiply, BlendMode::ColorBurn => vello::peniko::Mix::ColorBurn, // Lighten group BlendMode::Lighten => vello::peniko::Mix::Lighten, BlendMode::Screen => vello::peniko::Mix::Screen, BlendMode::ColorDodge => vello::peniko::Mix::ColorDodge, // Contrast group BlendMode::Overlay => vello::peniko::Mix::Overlay, BlendMode::SoftLight => vello::peniko::Mix::SoftLight, BlendMode::HardLight => vello::peniko::Mix::HardLight, // Inversion group BlendMode::Difference => vello::peniko::Mix::Difference, BlendMode::Exclusion => vello::peniko::Mix::Exclusion, // Component group BlendMode::Hue => vello::peniko::Mix::Hue, BlendMode::Saturation => vello::peniko::Mix::Saturation, BlendMode::Color => vello::peniko::Mix::Color, BlendMode::Luminosity => vello::peniko::Mix::Luminosity, _ => todo!(), } } } #[node_macro::node(category("Raster: Adjustment"))] fn luminance>( _: impl Ctx, #[implementations( Color, ImageFrameTable, GradientStops, )] mut input: T, luminance_calc: LuminanceCalculation, ) -> T { input.adjust(|color| { let luminance = match luminance_calc { LuminanceCalculation::SRGB => color.luminance_srgb(), LuminanceCalculation::Perceptual => color.luminance_perceptual(), LuminanceCalculation::AverageChannels => color.average_rgb_channels(), LuminanceCalculation::MinimumChannels => color.minimum_rgb_channels(), LuminanceCalculation::MaximumChannels => color.maximum_rgb_channels(), }; color.map_rgb(|_| luminance) }); input } #[node_macro::node(category("Raster"))] fn extract_channel>( _: impl Ctx, #[implementations( Color, ImageFrameTable, GradientStops, )] mut input: T, channel: RedGreenBlueAlpha, ) -> T { input.adjust(|color| { let extracted_value = match channel { RedGreenBlueAlpha::Red => color.r(), RedGreenBlueAlpha::Green => color.g(), RedGreenBlueAlpha::Blue => color.b(), RedGreenBlueAlpha::Alpha => color.a(), }; color.map_rgb(|_| extracted_value).with_alpha(1.) }); input } #[node_macro::node(category("Raster"))] fn make_opaque>( _: impl Ctx, #[implementations( Color, ImageFrameTable, GradientStops, )] mut input: T, ) -> T { input.adjust(|color| { if color.a() == 0. { return color.with_alpha(1.); } Color::from_rgbaf32(color.r() / color.a(), color.g() / color.a(), color.b() / color.a(), 1.).unwrap() }); input } // Aims for interoperable compatibility with: // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=%27%20%3D%20Brightness/Contrast-,%27levl%27%20%3D%20Levels,-%27curv%27%20%3D%20Curves // // Algorithm from: // https://stackoverflow.com/questions/39510072/algorithm-for-adjustment-of-image-levels #[node_macro::node(category("Raster: Adjustment"))] fn levels>( _: impl Ctx, #[implementations( Color, ImageFrameTable, GradientStops, )] mut image: T, #[default(0.)] shadows: Percentage, #[default(50.)] midtones: Percentage, #[default(100.)] highlights: Percentage, #[default(0.)] output_minimums: Percentage, #[default(100.)] output_maximums: Percentage, ) -> T { image.adjust(|color| { let color = color.to_gamma_srgb(); // Input Range (Range: 0-1) let input_shadows = (shadows / 100.) as f32; let input_midtones = (midtones / 100.) as f32; let input_highlights = (highlights / 100.) as f32; // Output Range (Range: 0-1) let output_minimums = (output_minimums / 100.) as f32; let output_maximums = (output_maximums / 100.) as f32; // Midtones interpolation factor between minimums and maximums (Range: 0-1) let midtones = output_minimums + (output_maximums - output_minimums) * input_midtones; // Gamma correction (Range: 0.01-10) let gamma = if midtones < 0.5 { // Range: 0-1 let x = 1. - midtones * 2.; // Range: 1-10 1. + 9. * x } else { // Range: 0-0.5 let x = 1. - midtones; // Range: 0-1 let x = x * 2.; // Range: 0.01-1 x.max(0.01) }; // Input levels (Range: 0-1) let highlights_minus_shadows = (input_highlights - input_shadows).clamp(f32::EPSILON, 1.); let color = color.map_rgb(|c| ((c - input_shadows).max(0.) / highlights_minus_shadows).min(1.)); // Midtones (Range: 0-1) let color = color.gamma(gamma); // Output levels (Range: 0-1) let color = color.map_rgb(|c| c * (output_maximums - output_minimums) + output_minimums); color.to_linear_srgb() }); image } // Aims for interoperable compatibility with: // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=%27blwh%27%20%3D%20Black%20and%20White // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=Black%20White%20(Photoshop%20CS3) // // Algorithm from: // https://stackoverflow.com/a/55233732/775283 // Works the same for gamma and linear color #[node_macro::node(name("Black & White"), category("Raster: Adjustment"))] async fn black_and_white>( _: impl Ctx, #[implementations( Color, ImageFrameTable, GradientStops, )] mut image: T, #[default(Color::BLACK)] tint: Color, #[default(40.)] #[range((-200., 300.))] reds: Percentage, #[default(60.)] #[range((-200., 300.))] yellows: Percentage, #[default(40.)] #[range((-200., 300.))] greens: Percentage, #[default(60.)] #[range((-200., 300.))] cyans: Percentage, #[default(20.)] #[range((-200., 300.))] blues: Percentage, #[default(80.)] #[range((-200., 300.))] magentas: Percentage, ) -> T { image.adjust(|color| { let color = color.to_gamma_srgb(); let reds = reds as f32 / 100.; let yellows = yellows as f32 / 100.; let greens = greens as f32 / 100.; let cyans = cyans as f32 / 100.; let blues = blues as f32 / 100.; let magentas = magentas as f32 / 100.; let gray_base = color.r().min(color.g()).min(color.b()); let red_part = color.r() - gray_base; let green_part = color.g() - gray_base; let blue_part = color.b() - gray_base; let alpha_part = color.a(); let additional = if red_part == 0. { let cyan_part = green_part.min(blue_part); cyan_part * cyans + (green_part - cyan_part) * greens + (blue_part - cyan_part) * blues } else if green_part == 0. { let magenta_part = red_part.min(blue_part); magenta_part * magentas + (red_part - magenta_part) * reds + (blue_part - magenta_part) * blues } else { let yellow_part = red_part.min(green_part); yellow_part * yellows + (red_part - yellow_part) * reds + (green_part - yellow_part) * greens }; let luminance = gray_base + additional; // TODO: Fix "Color" blend mode implementation so it matches the expected behavior perfectly (it's currently close) let color = tint.with_luminance(luminance); let color = Color::from_rgbaf32(color.r(), color.g(), color.b(), alpha_part).unwrap(); color.to_linear_srgb() }); image } // Aims for interoperable compatibility with: // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=%27hue%20%27%20%3D%20Old,saturation%2C%20Photoshop%205.0 // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=0%20%3D%20Use%20other.-,Hue/Saturation,-Hue/Saturation%20settings #[node_macro::node(name("Hue/Saturation"), category("Raster: Adjustment"))] async fn hue_saturation>( _: impl Ctx, #[implementations( Color, ImageFrameTable, GradientStops, )] mut input: T, hue_shift: Angle, saturation_shift: SignedPercentage, lightness_shift: SignedPercentage, ) -> T { input.adjust(|color| { let color = color.to_gamma_srgb(); let [hue, saturation, lightness, alpha] = color.to_hsla(); let color = Color::from_hsla( (hue + hue_shift as f32 / 360.) % 1., // TODO: Improve the way saturation works (it's slightly off) (saturation + saturation_shift as f32 / 100.).clamp(0., 1.), // TODO: Fix the way lightness works (it's very off) (lightness + lightness_shift as f32 / 100.).clamp(0., 1.), alpha, ); color.to_linear_srgb() }); input } // Aims for interoperable compatibility with: // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=%27%20%3D%20Color%20Lookup-,%27nvrt%27%20%3D%20Invert,-%27post%27%20%3D%20Posterize #[node_macro::node(category("Raster: Adjustment"))] async fn invert>( _: impl Ctx, #[implementations( Color, ImageFrameTable, GradientStops, )] mut input: T, ) -> T { input.adjust(|color| { let color = color.to_gamma_srgb(); let color = color.map_rgb(|c| color.a() - c); color.to_linear_srgb() }); input } // Aims for interoperable compatibility with: // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=post%27%20%3D%20Posterize-,%27thrs%27%20%3D%20Threshold,-%27grdm%27%20%3D%20Gradient #[node_macro::node(category("Raster: Adjustment"))] async fn threshold>( _: impl Ctx, #[implementations( Color, ImageFrameTable, GradientStops, )] mut image: T, #[default(50.)] min_luminance: Percentage, #[default(100.)] max_luminance: Percentage, luminance_calc: LuminanceCalculation, ) -> T { image.adjust(|color| { let min_luminance = Color::srgb_to_linear(min_luminance as f32 / 100.); let max_luminance = Color::srgb_to_linear(max_luminance as f32 / 100.); let luminance = match luminance_calc { LuminanceCalculation::SRGB => color.luminance_srgb(), LuminanceCalculation::Perceptual => color.luminance_perceptual(), LuminanceCalculation::AverageChannels => color.average_rgb_channels(), LuminanceCalculation::MinimumChannels => color.minimum_rgb_channels(), LuminanceCalculation::MaximumChannels => color.maximum_rgb_channels(), }; if luminance >= min_luminance && luminance <= max_luminance { Color::WHITE } else { Color::BLACK } }); image } trait Blend { fn blend(&self, under: &Self, blend_fn: impl Fn(P, P) -> P) -> Self; } impl Blend for Color { fn blend(&self, under: &Self, blend_fn: impl Fn(Color, Color) -> Color) -> Self { blend_fn(*self, *under) } } impl Blend for Option { fn blend(&self, under: &Self, blend_fn: impl Fn(Color, Color) -> Color) -> Self { match (self, under) { (Some(a), Some(b)) => Some(blend_fn(*a, *b)), (a, None) => *a, (None, b) => *b, } } } impl Blend for ImageFrameTable { fn blend(&self, under: &Self, blend_fn: impl Fn(Color, Color) -> Color) -> Self { let mut result = self.clone(); for (over, under) in result.instance_mut_iter().zip(under.instance_ref_iter()) { let data = over.instance.data.iter().zip(under.instance.data.iter()).map(|(a, b)| blend_fn(*a, *b)).collect(); *over.instance = Image { data, width: over.instance.width, height: over.instance.height, base64_string: None, }; } result } } impl Blend for GradientStops { fn blend(&self, under: &Self, blend_fn: impl Fn(Color, Color) -> Color) -> Self { let mut combined_stops = self.iter().map(|(position, _)| position).chain(under.iter().map(|(position, _)| position)).collect::>(); combined_stops.dedup_by(|&mut a, &mut b| (a - b).abs() < 1e-6); combined_stops.sort_by(|a, b| a.partial_cmp(b).unwrap_or(Ordering::Equal)); let stops = combined_stops .into_iter() .map(|&position| { let over_color = self.evaluate(position); let under_color = under.evaluate(position); let color = blend_fn(over_color, under_color); (position, color) }) .collect::>(); GradientStops::new(stops) } } #[node_macro::node(category("Raster"))] async fn blend + Send>( _: impl Ctx, #[implementations( Color, ImageFrameTable, GradientStops, )] over: T, #[expose] #[implementations( Color, ImageFrameTable, GradientStops, )] under: T, blend_mode: BlendMode, #[default(100.)] opacity: Percentage, ) -> T { over.blend(&under, |a, b| blend_colors(a, b, blend_mode, opacity / 100.)) } #[node_macro::node(category(""), skip_impl)] fn blend_color_pair(input: (Color, Color), blend_mode: &'n BlendModeNode, opacity: &'n OpacityNode) -> Color where BlendModeNode: Node<'n, (), Output = BlendMode> + 'n, OpacityNode: Node<'n, (), Output = Percentage> + 'n, { let blend_mode = blend_mode.eval(()); let opacity = opacity.eval(()); blend_colors(input.0, input.1, blend_mode, opacity / 100.) } pub fn apply_blend_mode(foreground: Color, background: Color, blend_mode: BlendMode) -> Color { match blend_mode { // Normal group BlendMode::Normal => background.blend_rgb(foreground, Color::blend_normal), // Darken group BlendMode::Darken => background.blend_rgb(foreground, Color::blend_darken), BlendMode::Multiply => background.blend_rgb(foreground, Color::blend_multiply), BlendMode::ColorBurn => background.blend_rgb(foreground, Color::blend_color_burn), BlendMode::LinearBurn => background.blend_rgb(foreground, Color::blend_linear_burn), BlendMode::DarkerColor => background.blend_darker_color(foreground), // Lighten group BlendMode::Lighten => background.blend_rgb(foreground, Color::blend_lighten), BlendMode::Screen => background.blend_rgb(foreground, Color::blend_screen), BlendMode::ColorDodge => background.blend_rgb(foreground, Color::blend_color_dodge), BlendMode::LinearDodge => background.blend_rgb(foreground, Color::blend_linear_dodge), BlendMode::LighterColor => background.blend_lighter_color(foreground), // Contrast group BlendMode::Overlay => foreground.blend_rgb(background, Color::blend_hardlight), BlendMode::SoftLight => background.blend_rgb(foreground, Color::blend_softlight), BlendMode::HardLight => background.blend_rgb(foreground, Color::blend_hardlight), BlendMode::VividLight => background.blend_rgb(foreground, Color::blend_vivid_light), BlendMode::LinearLight => background.blend_rgb(foreground, Color::blend_linear_light), BlendMode::PinLight => background.blend_rgb(foreground, Color::blend_pin_light), BlendMode::HardMix => background.blend_rgb(foreground, Color::blend_hard_mix), // Inversion group BlendMode::Difference => background.blend_rgb(foreground, Color::blend_difference), BlendMode::Exclusion => background.blend_rgb(foreground, Color::blend_exclusion), BlendMode::Subtract => background.blend_rgb(foreground, Color::blend_subtract), BlendMode::Divide => background.blend_rgb(foreground, Color::blend_divide), // Component group BlendMode::Hue => background.blend_hue(foreground), BlendMode::Saturation => background.blend_saturation(foreground), BlendMode::Color => background.blend_color(foreground), BlendMode::Luminosity => background.blend_luminosity(foreground), // Other utility blend modes (hidden from the normal list) - do not have alpha blend _ => panic!("Used blend mode without alpha blend"), } } trait Adjust

{ fn adjust(&mut self, map_fn: impl Fn(&P) -> P); } impl Adjust for Color { fn adjust(&mut self, map_fn: impl Fn(&Color) -> Color) { *self = map_fn(self); } } impl Adjust for Option { fn adjust(&mut self, map_fn: impl Fn(&Color) -> Color) { if let Some(v) = self { *v = map_fn(v) } } } impl Adjust for GradientStops { fn adjust(&mut self, map_fn: impl Fn(&Color) -> Color) { for (_pos, c) in self.iter_mut() { *c = map_fn(c); } } } impl Adjust

for ImageFrameTable

where GraphicElement: From>, { fn adjust(&mut self, map_fn: impl Fn(&P) -> P) { for instance in self.instance_mut_iter() { for c in instance.instance.data.iter_mut() { *c = map_fn(c); } } } } #[inline(always)] pub fn blend_colors(foreground: Color, background: Color, blend_mode: BlendMode, opacity: f64) -> Color { let target_color = match blend_mode { // Other utility blend modes (hidden from the normal list) - do not have alpha blend BlendMode::Erase => return background.alpha_subtract(foreground), BlendMode::Restore => return background.alpha_add(foreground), BlendMode::MultiplyAlpha => return background.alpha_multiply(foreground), blend_mode => apply_blend_mode(foreground, background, blend_mode), }; background.alpha_blend(target_color.to_associated_alpha(opacity as f32)) } // Aims for interoperable compatibility with: // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=%27grdm%27%20%3D%20Gradient%20Map // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=Gradient%20settings%20(Photoshop%206.0) #[node_macro::node(category("Raster: Adjustment"))] async fn gradient_map>( _: impl Ctx, #[implementations( Color, ImageFrameTable, GradientStops, )] mut image: T, gradient: GradientStops, reverse: bool, ) -> T { image.adjust(|color| { let intensity = color.luminance_srgb(); let intensity = if reverse { 1. - intensity } else { intensity }; gradient.evaluate(intensity as f64) }); image } // Aims for interoperable compatibility with: // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=%27-,vibA%27%20%3D%20Vibrance,-%27hue%20%27%20%3D%20Old // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=Vibrance%20(Photoshop%20CS3) // // Algorithm based on: // https://stackoverflow.com/questions/33966121/what-is-the-algorithm-for-vibrance-filters // The results of this implementation are very close to correct, but not quite perfect #[node_macro::node(category("Raster: Adjustment"))] async fn vibrance>( _: impl Ctx, #[implementations( Color, ImageFrameTable, GradientStops, )] mut image: T, vibrance: SignedPercentage, ) -> T { image.adjust(|color| { let vibrance = vibrance as f32 / 100.; // Slow the effect down by half when it's negative, since artifacts begin appearing past -50%. // So this scales the 0% to -50% range to 0% to -100%. let slowed_vibrance = if vibrance >= 0. { vibrance } else { vibrance * 0.5 }; let channel_max = color.r().max(color.g()).max(color.b()); let channel_min = color.r().min(color.g()).min(color.b()); let channel_difference = channel_max - channel_min; let scale_multiplier = if channel_max == color.r() { let green_blue_difference = (color.g() - color.b()).abs(); let t = (green_blue_difference / channel_difference).min(1.); t * 0.5 + 0.5 } else { 1. }; let scale = slowed_vibrance * scale_multiplier * (2. - channel_difference); let channel_reduction = channel_min * scale; let scale = 1. + scale * (1. - channel_difference); let luminance_initial = color.to_linear_srgb().luminance_srgb(); let altered_color = color.map_rgb(|c| c * scale - channel_reduction).to_linear_srgb(); let luminance = altered_color.luminance_srgb(); let altered_color = altered_color.map_rgb(|c| c * luminance_initial / luminance); let channel_max = altered_color.r().max(altered_color.g()).max(altered_color.b()); let altered_color = if Color::linear_to_srgb(channel_max) > 1. { let scale = (1. - luminance) / (channel_max - luminance); altered_color.map_rgb(|c| (c - luminance) * scale + luminance) } else { altered_color }; let altered_color = altered_color.to_gamma_srgb(); if vibrance >= 0. { altered_color } else { // TODO: The result ends up a bit darker than it should be, further investigation is needed let luminance = color.luminance_rec_601(); // Near -0% vibrance we mostly use `altered_color`. // Near -100% vibrance, we mostly use half the desaturated luminance color and half `altered_color`. let factor = -slowed_vibrance; altered_color.map_rgb(|c| c * (1. - factor) + luminance * factor) } }); image } #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] #[cfg_attr(feature = "std", derive(specta::Type))] #[derive(Debug, Clone, Copy, Default, PartialEq, Eq, Hash, DynAny)] pub enum RedGreenBlue { #[default] Red, Green, Blue, } impl core::fmt::Display for RedGreenBlue { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { match self { RedGreenBlue::Red => write!(f, "Red"), RedGreenBlue::Green => write!(f, "Green"), RedGreenBlue::Blue => write!(f, "Blue"), } } } #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] #[cfg_attr(feature = "std", derive(specta::Type))] #[derive(Debug, Clone, Copy, Default, PartialEq, Eq, Hash, DynAny)] pub enum RedGreenBlueAlpha { #[default] Red, Green, Blue, Alpha, } impl core::fmt::Display for RedGreenBlueAlpha { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { match self { RedGreenBlueAlpha::Red => write!(f, "Red"), RedGreenBlueAlpha::Green => write!(f, "Green"), RedGreenBlueAlpha::Blue => write!(f, "Blue"), RedGreenBlueAlpha::Alpha => write!(f, "Alpha"), } } } #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] #[cfg_attr(feature = "std", derive(specta::Type))] #[derive(Debug, Clone, Copy, Default, PartialEq, Eq, Hash, DynAny)] pub enum NoiseType { #[default] Perlin, OpenSimplex2, OpenSimplex2S, Cellular, ValueCubic, Value, WhiteNoise, } impl core::fmt::Display for NoiseType { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { match self { NoiseType::Perlin => write!(f, "Perlin"), NoiseType::OpenSimplex2 => write!(f, "OpenSimplex2"), NoiseType::OpenSimplex2S => write!(f, "OpenSimplex2S"), NoiseType::Cellular => write!(f, "Cellular"), NoiseType::ValueCubic => write!(f, "Value Cubic"), NoiseType::Value => write!(f, "Value"), NoiseType::WhiteNoise => write!(f, "White Noise"), } } } impl NoiseType { pub fn list() -> &'static [NoiseType; 7] { &[ NoiseType::Perlin, NoiseType::OpenSimplex2, NoiseType::OpenSimplex2S, NoiseType::Cellular, NoiseType::ValueCubic, NoiseType::Value, NoiseType::WhiteNoise, ] } } #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] #[cfg_attr(feature = "std", derive(specta::Type))] #[derive(Debug, Clone, Copy, Default, PartialEq, Eq, Hash, DynAny)] pub enum FractalType { #[default] None, FBm, Ridged, PingPong, DomainWarpProgressive, DomainWarpIndependent, } impl core::fmt::Display for FractalType { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { match self { FractalType::None => write!(f, "None"), FractalType::FBm => write!(f, "Fractional Brownian Motion"), FractalType::Ridged => write!(f, "Ridged"), FractalType::PingPong => write!(f, "Ping Pong"), FractalType::DomainWarpProgressive => write!(f, "Progressive (Domain Warp Only)"), FractalType::DomainWarpIndependent => write!(f, "Independent (Domain Warp Only)"), } } } impl FractalType { pub fn list() -> &'static [FractalType; 6] { &[ FractalType::None, FractalType::FBm, FractalType::Ridged, FractalType::PingPong, FractalType::DomainWarpProgressive, FractalType::DomainWarpIndependent, ] } } #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] #[cfg_attr(feature = "std", derive(specta::Type))] #[derive(Debug, Clone, Copy, Default, PartialEq, Eq, Hash, DynAny)] pub enum CellularDistanceFunction { #[default] Euclidean, EuclideanSq, Manhattan, Hybrid, } impl core::fmt::Display for CellularDistanceFunction { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { match self { CellularDistanceFunction::Euclidean => write!(f, "Euclidean"), CellularDistanceFunction::EuclideanSq => write!(f, "Euclidean Squared (Faster)"), CellularDistanceFunction::Manhattan => write!(f, "Manhattan"), CellularDistanceFunction::Hybrid => write!(f, "Hybrid"), } } } impl CellularDistanceFunction { pub fn list() -> &'static [CellularDistanceFunction; 4] { &[ CellularDistanceFunction::Euclidean, CellularDistanceFunction::EuclideanSq, CellularDistanceFunction::Manhattan, CellularDistanceFunction::Hybrid, ] } } #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] #[cfg_attr(feature = "std", derive(specta::Type))] #[derive(Debug, Clone, Copy, Default, PartialEq, Eq, Hash, DynAny)] pub enum CellularReturnType { CellValue, #[default] Nearest, NextNearest, Average, Difference, Product, Division, } impl core::fmt::Display for CellularReturnType { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { match self { CellularReturnType::CellValue => write!(f, "Cell Value"), CellularReturnType::Nearest => write!(f, "Nearest (F1)"), CellularReturnType::NextNearest => write!(f, "Next Nearest (F2)"), CellularReturnType::Average => write!(f, "Average (F1 / 2 + F2 / 2)"), CellularReturnType::Difference => write!(f, "Difference (F2 - F1)"), CellularReturnType::Product => write!(f, "Product (F2 * F1 / 2)"), CellularReturnType::Division => write!(f, "Division (F1 / F2)"), } } } impl CellularReturnType { pub fn list() -> &'static [CellularReturnType; 7] { &[ CellularReturnType::CellValue, CellularReturnType::Nearest, CellularReturnType::NextNearest, CellularReturnType::Average, CellularReturnType::Difference, CellularReturnType::Product, CellularReturnType::Division, ] } } #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] #[cfg_attr(feature = "std", derive(specta::Type))] #[derive(Debug, Clone, Copy, Default, PartialEq, Eq, Hash, DynAny)] pub enum DomainWarpType { #[default] None, OpenSimplex2, OpenSimplex2Reduced, BasicGrid, } impl core::fmt::Display for DomainWarpType { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { match self { DomainWarpType::None => write!(f, "None"), DomainWarpType::OpenSimplex2 => write!(f, "OpenSimplex2"), DomainWarpType::OpenSimplex2Reduced => write!(f, "OpenSimplex2 Reduced"), DomainWarpType::BasicGrid => write!(f, "Basic Grid"), } } } impl DomainWarpType { pub fn list() -> &'static [DomainWarpType; 4] { &[DomainWarpType::None, DomainWarpType::OpenSimplex2, DomainWarpType::OpenSimplex2Reduced, DomainWarpType::BasicGrid] } } // Aims for interoperable compatibility with: // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=%27mixr%27%20%3D%20Channel%20Mixer // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=Lab%20color%20only-,Channel%20Mixer,-Key%20is%20%27mixr #[node_macro::node(category("Raster: Adjustment"), properties("channel_mixer_properties"))] async fn channel_mixer>( _: impl Ctx, #[implementations( Color, ImageFrameTable, GradientStops, )] mut image: T, monochrome: bool, #[default(40.)] #[name("Red")] monochrome_r: f64, #[default(40.)] #[name("Green")] monochrome_g: f64, #[default(20.)] #[name("Blue")] monochrome_b: f64, #[default(0.)] #[name("Constant")] monochrome_c: f64, #[default(100.)] #[name("(Red) Red")] red_r: f64, #[default(0.)] #[name("(Red) Green")] red_g: f64, #[default(0.)] #[name("(Red) Blue")] red_b: f64, #[default(0.)] #[name("(Red) Constant")] red_c: f64, #[default(0.)] #[name("(Green) Red")] green_r: f64, #[default(100.)] #[name("(Green) Green")] green_g: f64, #[default(0.)] #[name("(Green) Blue")] green_b: f64, #[default(0.)] #[name("(Green) Constant")] green_c: f64, #[default(0.)] #[name("(Blue) Red")] blue_r: f64, #[default(0.)] #[name("(Blue) Green")] blue_g: f64, #[default(100.)] #[name("(Blue) Blue")] blue_b: f64, #[default(0.)] #[name("(Blue) Constant")] blue_c: f64, // Display-only properties (not used within the node) _output_channel: RedGreenBlue, ) -> T { image.adjust(|color| { let color = color.to_gamma_srgb(); let (r, g, b, a) = color.components(); let color = if monochrome { let (monochrome_r, monochrome_g, monochrome_b, monochrome_c) = (monochrome_r as f32 / 100., monochrome_g as f32 / 100., monochrome_b as f32 / 100., monochrome_c as f32 / 100.); let gray = (r * monochrome_r + g * monochrome_g + b * monochrome_b + monochrome_c).clamp(0., 1.); Color::from_rgbaf32_unchecked(gray, gray, gray, a) } else { let (red_r, red_g, red_b, red_c) = (red_r as f32 / 100., red_g as f32 / 100., red_b as f32 / 100., red_c as f32 / 100.); let (green_r, green_g, green_b, green_c) = (green_r as f32 / 100., green_g as f32 / 100., green_b as f32 / 100., green_c as f32 / 100.); let (blue_r, blue_g, blue_b, blue_c) = (blue_r as f32 / 100., blue_g as f32 / 100., blue_b as f32 / 100., blue_c as f32 / 100.); let red = (r * red_r + g * red_g + b * red_b + red_c).clamp(0., 1.); let green = (r * green_r + g * green_g + b * green_b + green_c).clamp(0., 1.); let blue = (r * blue_r + g * blue_g + b * blue_b + blue_c).clamp(0., 1.); Color::from_rgbaf32_unchecked(red, green, blue, a) }; color.to_linear_srgb() }); image } #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] #[cfg_attr(feature = "std", derive(specta::Type))] #[derive(Debug, Clone, Copy, Default, PartialEq, Eq, Hash, DynAny)] pub enum RelativeAbsolute { #[default] Relative, Absolute, } impl core::fmt::Display for RelativeAbsolute { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { match self { RelativeAbsolute::Relative => write!(f, "Relative"), RelativeAbsolute::Absolute => write!(f, "Absolute"), } } } #[repr(C)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] #[cfg_attr(feature = "std", derive(specta::Type))] #[derive(Debug, Clone, Copy, Default, PartialEq, Eq, Hash, DynAny)] pub enum SelectiveColorChoice { #[default] Reds, Yellows, Greens, Cyans, Blues, Magentas, Whites, Neutrals, Blacks, } impl core::fmt::Display for SelectiveColorChoice { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { match self { SelectiveColorChoice::Reds => write!(f, "Reds"), SelectiveColorChoice::Yellows => write!(f, "Yellows"), SelectiveColorChoice::Greens => write!(f, "Greens"), SelectiveColorChoice::Cyans => write!(f, "Cyans"), SelectiveColorChoice::Blues => write!(f, "Blues"), SelectiveColorChoice::Magentas => write!(f, "Magentas"), SelectiveColorChoice::Whites => write!(f, "Whites"), SelectiveColorChoice::Neutrals => write!(f, "Neutrals"), SelectiveColorChoice::Blacks => write!(f, "Blacks"), } } } // Aims for interoperable compatibility with: // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=%27selc%27%20%3D%20Selective%20color // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=from%20%2D100...100.%20.-,Selective%20Color,-Selective%20Color%20settings // // Algorithm based on: // https://blog.pkh.me/p/22-understanding-selective-coloring-in-adobe-photoshop.html #[node_macro::node(category("Raster: Adjustment"), properties("selective_color_properties"))] async fn selective_color>( _: impl Ctx, #[implementations( Color, ImageFrameTable, GradientStops, )] mut image: T, mode: RelativeAbsolute, #[name("(Reds) Cyan")] r_c: f64, #[name("(Reds) Magenta")] r_m: f64, #[name("(Reds) Yellow")] r_y: f64, #[name("(Reds) Black")] r_k: f64, #[name("(Yellows) Cyan")] y_c: f64, #[name("(Yellows) Magenta")] y_m: f64, #[name("(Yellows) Yellow")] y_y: f64, #[name("(Yellows) Black")] y_k: f64, #[name("(Greens) Cyan")] g_c: f64, #[name("(Greens) Magenta")] g_m: f64, #[name("(Greens) Yellow")] g_y: f64, #[name("(Greens) Black")] g_k: f64, #[name("(Cyans) Cyan")] c_c: f64, #[name("(Cyans) Magenta")] c_m: f64, #[name("(Cyans) Yellow")] c_y: f64, #[name("(Cyans) Black")] c_k: f64, #[name("(Blues) Cyan")] b_c: f64, #[name("(Blues) Magenta")] b_m: f64, #[name("(Blues) Yellow")] b_y: f64, #[name("(Blues) Black")] b_k: f64, #[name("(Magentas) Cyan")] m_c: f64, #[name("(Magentas) Magenta")] m_m: f64, #[name("(Magentas) Yellow")] m_y: f64, #[name("(Magentas) Black")] m_k: f64, #[name("(Whites) Cyan")] w_c: f64, #[name("(Whites) Magenta")] w_m: f64, #[name("(Whites) Yellow")] w_y: f64, #[name("(Whites) Black")] w_k: f64, #[name("(Neutrals) Cyan")] n_c: f64, #[name("(Neutrals) Magenta")] n_m: f64, #[name("(Neutrals) Yellow")] n_y: f64, #[name("(Neutrals) Black")] n_k: f64, #[name("(Blacks) Cyan")] k_c: f64, #[name("(Blacks) Magenta")] k_m: f64, #[name("(Blacks) Yellow")] k_y: f64, #[name("(Blacks) Black")] k_k: f64, _colors: SelectiveColorChoice, ) -> T { image.adjust(|color| { let color = color.to_gamma_srgb(); let (r, g, b, a) = color.components(); let min = |a: f32, b: f32, c: f32| a.min(b).min(c); let max = |a: f32, b: f32, c: f32| a.max(b).max(c); let med = |a: f32, b: f32, c: f32| a + b + c - min(a, b, c) - max(a, b, c); let max_channel = max(r, g, b); let min_channel = min(r, g, b); let pixel_color_range = |choice| match choice { SelectiveColorChoice::Reds => max_channel == r, SelectiveColorChoice::Yellows => min_channel == b, SelectiveColorChoice::Greens => max_channel == g, SelectiveColorChoice::Cyans => min_channel == r, SelectiveColorChoice::Blues => max_channel == b, SelectiveColorChoice::Magentas => min_channel == g, SelectiveColorChoice::Whites => r > 0.5 && g > 0.5 && b > 0.5, SelectiveColorChoice::Neutrals => r > 0. && g > 0. && b > 0. && r < 1. && g < 1. && b < 1., SelectiveColorChoice::Blacks => r < 0.5 && g < 0.5 && b < 0.5, }; let color_parameter_group_scale_factor_rgb = max(r, g, b) - med(r, g, b); let color_parameter_group_scale_factor_cmy = med(r, g, b) - min(r, g, b); // Used to apply the r, g, or b channel slope (by multiplying it by 1) in relative mode, or no slope (by multiplying it by 0) in absolute mode let (slope_r, slope_g, slope_b) = match mode { RelativeAbsolute::Relative => (r - 1., g - 1., b - 1.), RelativeAbsolute::Absolute => (-1., -1., -1.), }; let (sum_r, sum_g, sum_b) = [ (SelectiveColorChoice::Reds, (r_c as f32, r_m as f32, r_y as f32, r_k as f32)), (SelectiveColorChoice::Yellows, (y_c as f32, y_m as f32, y_y as f32, y_k as f32)), (SelectiveColorChoice::Greens, (g_c as f32, g_m as f32, g_y as f32, g_k as f32)), (SelectiveColorChoice::Cyans, (c_c as f32, c_m as f32, c_y as f32, c_k as f32)), (SelectiveColorChoice::Blues, (b_c as f32, b_m as f32, b_y as f32, b_k as f32)), (SelectiveColorChoice::Magentas, (m_c as f32, m_m as f32, m_y as f32, m_k as f32)), (SelectiveColorChoice::Whites, (w_c as f32, w_m as f32, w_y as f32, w_k as f32)), (SelectiveColorChoice::Neutrals, (n_c as f32, n_m as f32, n_y as f32, n_k as f32)), (SelectiveColorChoice::Blacks, (k_c as f32, k_m as f32, k_y as f32, k_k as f32)), ] .into_iter() .fold((0., 0., 0.), |acc, (color_parameter_group, (c, m, y, k))| { // Skip this color parameter group... // ...if it's unchanged from the default of zero offset on all CMYK parameters, or... // ...if this pixel's color isn't in the range affected by this color parameter group if (c < f32::EPSILON && m < f32::EPSILON && y < f32::EPSILON && k < f32::EPSILON) || (!pixel_color_range(color_parameter_group)) { return acc; } let (c, m, y, k) = (c / 100., m / 100., y / 100., k / 100.); let color_parameter_group_scale_factor = match color_parameter_group { SelectiveColorChoice::Reds | SelectiveColorChoice::Greens | SelectiveColorChoice::Blues => color_parameter_group_scale_factor_rgb, SelectiveColorChoice::Cyans | SelectiveColorChoice::Magentas | SelectiveColorChoice::Yellows => color_parameter_group_scale_factor_cmy, SelectiveColorChoice::Whites => min(r, g, b) * 2. - 1., SelectiveColorChoice::Neutrals => 1. - ((max(r, g, b) - 0.5).abs() + (min(r, g, b) - 0.5).abs()), SelectiveColorChoice::Blacks => 1. - max(r, g, b) * 2., }; let offset_r = ((c + k * (c + 1.)) * slope_r).clamp(-r, -r + 1.) * color_parameter_group_scale_factor; let offset_g = ((m + k * (m + 1.)) * slope_g).clamp(-g, -g + 1.) * color_parameter_group_scale_factor; let offset_b = ((y + k * (y + 1.)) * slope_b).clamp(-b, -b + 1.) * color_parameter_group_scale_factor; (acc.0 + offset_r, acc.1 + offset_g, acc.2 + offset_b) }); let color = Color::from_rgbaf32_unchecked((r + sum_r).clamp(0., 1.), (g + sum_g).clamp(0., 1.), (b + sum_b).clamp(0., 1.), a); color.to_linear_srgb() }); image } pub(super) trait MultiplyAlpha { fn multiply_alpha(&mut self, factor: f64); } impl MultiplyAlpha for Color { fn multiply_alpha(&mut self, factor: f64) { *self = Color::from_rgbaf32_unchecked(self.r(), self.g(), self.b(), (self.a() * factor as f32).clamp(0., 1.)) } } impl MultiplyAlpha for VectorDataTable { fn multiply_alpha(&mut self, factor: f64) { for instance in self.instance_mut_iter() { instance.alpha_blending.opacity *= factor as f32; } } } impl MultiplyAlpha for GraphicGroupTable { fn multiply_alpha(&mut self, factor: f64) { for instance in self.instance_mut_iter() { instance.alpha_blending.opacity *= factor as f32; } } } impl MultiplyAlpha for ImageFrameTable

where GraphicElement: From>, { fn multiply_alpha(&mut self, factor: f64) { for instance in self.instance_mut_iter() { instance.alpha_blending.opacity *= factor as f32; } } } // Aims for interoperable compatibility with: // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=nvrt%27%20%3D%20Invert-,%27post%27%20%3D%20Posterize,-%27thrs%27%20%3D%20Threshold // // Algorithm based on: // https://www.axiomx.com/posterize.htm // This algorithm produces fully accurate output in relation to the industry standard. #[node_macro::node(category("Raster: Adjustment"))] async fn posterize>( _: impl Ctx, #[implementations( Color, ImageFrameTable, GradientStops, )] mut input: T, #[default(4)] #[min(2.)] levels: u32, ) -> T { input.adjust(|color| { let color = color.to_gamma_srgb(); let levels = levels as f32; let number_of_areas = levels.recip(); let size_of_areas = (levels - 1.).recip(); let channel = |channel: f32| (channel / number_of_areas).floor() * size_of_areas; let color = color.map_rgb(channel); color.to_linear_srgb() }); input } // Aims for interoperable compatibility with: // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=curv%27%20%3D%20Curves-,%27expA%27%20%3D%20Exposure,-%27vibA%27%20%3D%20Vibrance // https://www.adobe.com/devnet-apps/photoshop/fileformatashtml/#:~:text=Flag%20(%20%3D%20128%20)-,Exposure,-Key%20is%20%27expA // // Algorithm based on: // https://geraldbakker.nl/psnumbers/exposure.html #[node_macro::node(category("Raster: Adjustment"), properties("exposure_properties"))] async fn exposure>( _: impl Ctx, #[implementations( Color, ImageFrameTable, GradientStops, )] mut input: T, exposure: f64, offset: f64, #[default(1.)] #[range((0.01, 10.))] gamma_correction: f64, ) -> T { input.adjust(|color| { let adjusted = color // Exposure .map_rgb(|c: f32| c * 2_f32.powf(exposure as f32)) // Offset .map_rgb(|c: f32| c + offset as f32) // Gamma correction .gamma(gamma_correction as f32); adjusted.map_rgb(|c: f32| c.clamp(0., 1.)) }); input } const WINDOW_SIZE: usize = 1024; #[cfg(feature = "alloc")] #[node_macro::node(category(""))] fn generate_curves(_: impl Ctx, curve: Curve, #[implementations(f32, f64)] _target_format: C) -> ValueMapperNode { use bezier_rs::{Bezier, TValue}; 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(core::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) // Fall back to a very bad approximation if Bezier-rs fails .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) } #[cfg(feature = "alloc")] #[node_macro::node(category("Raster: Adjustment"))] fn color_overlay>( _: impl Ctx, #[implementations( Color, ImageFrameTable, GradientStops, )] mut image: T, #[default(Color::BLACK)] color: Color, blend_mode: BlendMode, #[default(100.)] opacity: Percentage, ) -> T { let opacity = (opacity as f32 / 100.).clamp(0., 1.); image.adjust(|pixel| { let image = pixel.map_rgb(|channel| channel * (1. - opacity)); // The apply blend mode function divides rgb by the alpha channel for the background. This undoes that. let associated_pixel = Color::from_rgbaf32_unchecked(pixel.r() * pixel.a(), pixel.g() * pixel.a(), pixel.b() * pixel.a(), pixel.a()); let overlay = apply_blend_mode(color, associated_pixel, blend_mode).map_rgb(|channel| channel * opacity); Color::from_rgbaf32_unchecked(image.r() + overlay.r(), image.g() + overlay.g(), image.b() + overlay.b(), pixel.a()) }); image } // #[cfg(feature = "alloc")] // pub use index_node::IndexNode; // #[cfg(feature = "alloc")] // mod index_node { // use crate::raster::{Color, Image}; // use crate::Ctx; // #[node_macro::node(category(""))] // pub fn index( // _: impl Ctx, // #[implementations(Vec>, Vec)] // #[widget(ParsedWidgetOverride::Hidden)] // input: Vec, // index: u32, // ) -> T { // if (index as usize) < input.len() { // input[index as usize].clone() // } else { // warn!("The number of segments is {} but the requested segment is {}!", input.len(), index); // Default::default() // } // } // } #[cfg(test)] mod test { use crate::raster::adjustments::BlendMode; use crate::raster::image::{Image, ImageFrameTable}; use crate::{Color, Node}; use std::pin::Pin; #[derive(Clone)] pub struct FutureWrapperNode(T); impl<'i, T: 'i + Clone + Send> Node<'i, ()> for FutureWrapperNode { type Output = Pin + 'i + Send>>; fn eval(&'i self, _input: ()) -> Self::Output { let value = self.0.clone(); Box::pin(async move { value }) } } #[tokio::test] async fn color_overlay_multiply() { let image_color = Color::from_rgbaf32_unchecked(0.7, 0.6, 0.5, 0.4); let image = Image::new(1, 1, image_color); // Color { red: 0., green: 1., blue: 0., alpha: 1. } let overlay_color = Color::GREEN; // 100% of the output should come from the multiplied value let opacity = 100_f64; let result = super::color_overlay((), ImageFrameTable::new(image.clone()), overlay_color, BlendMode::Multiply, opacity); let result = result.instance_ref_iter().next().unwrap().instance; // The output should just be the original green and alpha channels (as we multiply them by 1 and other channels by 0) assert_eq!(result.data[0], Color::from_rgbaf32_unchecked(0., image_color.g(), 0., image_color.a())); } }