use graph_craft::document::value::RenderOutput; pub use graph_craft::document::value::RenderOutputType; pub use graph_craft::wasm_application_io::*; use graphene_application_io::{ApplicationIo, ExportFormat, ImageTexture, RenderConfig, SurfaceFrame}; use graphene_core::gradient::GradientStops; use graphene_core::raster::image::Image; use graphene_core::raster_types::{CPU, Raster}; use graphene_core::table::Table; use graphene_core::transform::Footprint; use graphene_core::vector::Vector; use graphene_core::{Artboard, CloneVarArgs, ExtractAll, ExtractVarArgs}; use graphene_core::{Color, Context, Ctx, ExtractFootprint, Graphic, OwnedContextImpl, WasmNotSend}; use graphene_svg_renderer::{Render, RenderOutputType as RenderOutputTypeRequest, RenderParams, RenderSvgSegmentList, SvgRender, format_transform_matrix}; use graphene_svg_renderer::{RenderMetadata, SvgSegment}; use std::sync::Arc; use wgpu_executor::RenderContext; /// List of (canvas id, image data) pairs for embedding images as canvases in the final SVG string. type ImageData = Vec<(u64, Image)>; #[derive(Clone, dyn_any::DynAny)] pub enum RenderIntermediateType { Vello(Arc<(vello::Scene, RenderContext)>), Svg(Arc<(String, ImageData, String)>), } #[derive(Clone, dyn_any::DynAny)] pub struct RenderIntermediate { ty: RenderIntermediateType, metadata: RenderMetadata, contains_artboard: bool, } #[node_macro::node(category(""))] async fn render_intermediate<'a: 'n, T: 'static + Render + WasmNotSend + Send + Sync>( ctx: impl Ctx + ExtractVarArgs + ExtractAll + CloneVarArgs, #[implementations( Context -> Table, Context -> Table, Context -> Table, Context -> Table>, Context -> Table, Context -> Table, )] data: impl Node, Output = T>, ) -> RenderIntermediate { let render_params = ctx .vararg(0) .expect("Did not find var args") .downcast_ref::() .expect("Downcasting render params yielded invalid type"); let ctx = OwnedContextImpl::from(ctx.clone()).into_context(); let data = data.eval(ctx).await; let footprint = Footprint::default(); let mut metadata = RenderMetadata::default(); data.collect_metadata(&mut metadata, footprint, None); let contains_artboard = data.contains_artboard(); match &render_params.render_output_type { RenderOutputTypeRequest::Vello => { let mut scene = vello::Scene::new(); let mut context = wgpu_executor::RenderContext::default(); data.render_to_vello(&mut scene, Default::default(), &mut context, render_params); RenderIntermediate { ty: RenderIntermediateType::Vello(Arc::new((scene, context))), metadata, contains_artboard, } } RenderOutputTypeRequest::Svg => { let mut render = SvgRender::new(); data.render_svg(&mut render, render_params); RenderIntermediate { ty: RenderIntermediateType::Svg(Arc::new((render.svg.to_svg_string(), render.image_data, render.svg_defs.clone()))), metadata, contains_artboard, } } } } #[node_macro::node(category(""))] async fn create_context<'a: 'n>( // Context injections are defined in the wrap_network_in_scope function render_config: RenderConfig, data: impl Node, Output = RenderOutput>, ) -> RenderOutput { let footprint = render_config.viewport; let render_output_type = match render_config.export_format { ExportFormat::Svg => RenderOutputTypeRequest::Svg, ExportFormat::Raster => RenderOutputTypeRequest::Vello, }; let render_params = RenderParams { render_mode: render_config.render_mode, hide_artboards: render_config.hide_artboards, for_export: render_config.for_export, render_output_type, footprint: Footprint::default(), ..Default::default() }; let ctx = OwnedContextImpl::default() .with_footprint(footprint) .with_real_time(render_config.time.time) .with_animation_time(render_config.time.animation_time.as_secs_f64()) .with_vararg(Box::new(render_params)) .into_context(); data.eval(ctx).await } #[node_macro::node(category(""))] async fn render<'a: 'n>( ctx: impl Ctx + ExtractFootprint + ExtractVarArgs, editor_api: &'a WasmEditorApi, data: RenderIntermediate, _surface_handle: impl Node, Output = Option>, ) -> RenderOutput { let footprint = ctx.footprint(); let render_params = ctx .vararg(0) .expect("Did not find var args") .downcast_ref::() .expect("Downcasting render params yielded invalid type"); let mut render_params = render_params.clone(); render_params.footprint = *footprint; let render_params = &render_params; let RenderIntermediate { ty, mut metadata, contains_artboard } = data; metadata.apply_transform(footprint.transform); let surface_handle = if cfg!(all(feature = "vello", target_family = "wasm")) { _surface_handle.eval(None).await } else { None }; let data = match (render_params.render_output_type, &ty) { (RenderOutputTypeRequest::Svg, RenderIntermediateType::Svg(svg_data)) => { let mut svg_renderer = SvgRender::new(); if !contains_artboard && !render_params.hide_artboards { svg_renderer.leaf_tag("rect", |attributes| { attributes.push("x", "0"); attributes.push("y", "0"); attributes.push("width", footprint.resolution.x.to_string()); attributes.push("height", footprint.resolution.y.to_string()); let matrix = format_transform_matrix(footprint.transform.inverse()); if !matrix.is_empty() { attributes.push("transform", matrix); } attributes.push("fill", "white"); }); } svg_renderer.svg.push(SvgSegment::from(svg_data.0.clone())); svg_renderer.image_data = svg_data.1.clone(); svg_renderer.svg_defs = svg_data.2.clone(); svg_renderer.wrap_with_transform(footprint.transform, Some(footprint.resolution.as_dvec2())); RenderOutputType::Svg { svg: svg_renderer.svg.to_svg_string(), image_data: svg_renderer.image_data, } } (RenderOutputTypeRequest::Vello, RenderIntermediateType::Vello(vello_data)) => { let Some(exec) = editor_api.application_io.as_ref().unwrap().gpu_executor() else { unreachable!("Attempted to render with Vello when no GPU executor is available"); }; let (child, context) = Arc::as_ref(vello_data); let footprint_transform = vello::kurbo::Affine::new(footprint.transform.to_cols_array()); let mut scene = vello::Scene::new(); scene.append(child, Some(footprint_transform)); let encoding = scene.encoding_mut(); // We now replace all transforms which are supposed to be infinite with a transform which covers the entire viewport // See for more detail for transform in encoding.transforms.iter_mut() { if transform.matrix[0] == f32::INFINITY { *transform = vello_encoding::Transform::from_kurbo(&(vello::kurbo::Affine::scale_non_uniform(footprint.resolution.x as f64, footprint.resolution.y as f64))) } } let mut background = Color::from_rgb8_srgb(0x22, 0x22, 0x22); if !contains_artboard && !render_params.hide_artboards { background = Color::WHITE; } if let Some(surface_handle) = surface_handle { exec.render_vello_scene(&scene, &surface_handle, footprint.resolution, context, background) .await .expect("Failed to render Vello scene"); let frame = SurfaceFrame { surface_id: surface_handle.window_id, resolution: footprint.resolution, transform: glam::DAffine2::IDENTITY, }; RenderOutputType::CanvasFrame(frame) } else { let texture = exec .render_vello_scene_to_texture(&scene, footprint.resolution, context, background) .await .expect("Failed to render Vello scene"); RenderOutputType::Texture(ImageTexture { texture }) } } _ => unreachable!("Render node did not receive its requested data type"), }; RenderOutput { data, metadata } }