Animated thumbnails

This commit is contained in:
Adam 2025-07-17 00:31:09 -07:00
parent c2815ea0e3
commit 08c2d0543c
12 changed files with 239 additions and 60 deletions

View file

@ -6,9 +6,7 @@ use crate::messages::prelude::*;
#[derive(Debug, Default)]
pub struct Dispatcher {
evaluation_queue: Vec<Message>,
introspection_queue: Vec<Message>,
queueing_evaluation_messages: bool,
queueing_introspection_messages: bool,
message_queues: Vec<VecDeque<Message>>,
pub responses: Vec<FrontendMessage>,
pub message_handlers: DispatcherMessageHandlers,

View file

@ -17,7 +17,6 @@ use crate::messages::portfolio::document::utility_types::document_metadata::{Doc
use crate::messages::portfolio::document::utility_types::misc::{AlignAggregate, AlignAxis, DocumentMode, FlipAxis, PTZ};
use crate::messages::portfolio::document::utility_types::network_interface::{FlowType, NodeTemplate};
use crate::messages::portfolio::document::utility_types::nodes::RawBuffer;
use crate::messages::portfolio::utility_types::PersistentData;
use crate::messages::prelude::*;
use crate::messages::tool::common_functionality::graph_modification_utils::{self, get_blend_mode, get_fill, get_opacity};
use crate::messages::tool::tool_messages::select_tool::SelectToolPointerKeys;

View file

@ -1096,7 +1096,7 @@ fn static_nodes() -> Vec<DocumentNodeDefinition> {
implementation: DocumentNodeImplementation::ProtoNode(text::text::IDENTIFIER),
manual_composition: Some(concrete!(Context)),
inputs: vec![
NodeInput::scope("editor-api"),
NodeInput::scope("font-cache"),
NodeInput::value(TaggedValue::String("Lorem ipsum".to_string()), false),
NodeInput::value(
TaggedValue::Font(Font::new(graphene_std::consts::DEFAULT_FONT_FAMILY.into(), graphene_std::consts::DEFAULT_FONT_STYLE.into())),

View file

@ -796,7 +796,7 @@ impl MessageHandler<PortfolioMessage, PortfolioMessageContext<'_>> for Portfolio
// Remove all thumbnails
cleared_thumbnails.push(sni);
}
document.node_graph_handler.node_graph_errors = Vec::new();
self.thumbnails_to_clear.extend(cleared_thumbnails);
}
PortfolioMessage::EvaluateActiveDocumentWithThumbnails => {
@ -869,7 +869,9 @@ impl MessageHandler<PortfolioMessage, PortfolioMessageContext<'_>> for Portfolio
let evaluated_data = match monitor_result {
MonitorIntrospectResult::Error => continue,
MonitorIntrospectResult::Disabled => continue,
MonitorIntrospectResult::NotEvaluated => continue,
MonitorIntrospectResult::NotEvaluated => {
continue;
}
MonitorIntrospectResult::Evaluated((data, changed)) => {
// If the evaluated value is the same as the previous, then just remap the ID
if !changed {
@ -1253,30 +1255,30 @@ impl PortfolioMessageHandler {
.network_interface
.viewport_loaded_thumbnail_position(&input_connector, graph_wire_style, &document.breadcrumb_network_path)
{
let in_view = viewport_position.x > 0.0 && viewport_position.y > 0.0 && viewport_position.x < ipp.viewport_bounds()[1].x && viewport_position.y < ipp.viewport_bounds()[1].y;
if in_view {
let Some(protonode) = document.network_interface.protonode_from_input(&input_connector, &document.breadcrumb_network_path) else {
// The input is not connected to the export, which occurs if inside a disconnected node
wire_stack = Vec::new();
nodes_to_render.clear();
continue;
};
nodes_to_render.insert(protonode);
}
// let in_view = viewport_position.x > 0.0 && viewport_position.y > 0.0 && viewport_position.x < ipp.viewport_bounds()[1].x && viewport_position.y < ipp.viewport_bounds()[1].y;
// if in_view {
let Some(protonode) = document.network_interface.protonode_from_input(&input_connector, &document.breadcrumb_network_path) else {
// The input is not connected to the export, which occurs if inside a disconnected node
wire_stack = Vec::new();
nodes_to_render.clear();
continue;
};
nodes_to_render.insert(protonode);
// }
}
}
};
// Get thumbnails for all visible layer
for visible_node in &document.node_graph_handler.visible_nodes(&mut document.network_interface, &document.breadcrumb_network_path, ipp) {
if document.network_interface.is_layer(&visible_node, &document.breadcrumb_network_path) {
let Some(protonode) = document
// Get thumbnails for all visible layer ouputs
// for visible_node in &document.node_graph_handler.visible_nodes(&mut document.network_interface, &document.breadcrumb_network_path, ipp) {
for visible_node in document.network_interface.nested_network(&document.breadcrumb_network_path).unwrap().nodes.keys() {
if document.network_interface.is_layer(visible_node, &document.breadcrumb_network_path) {
if let Some(protonode) = document
.network_interface
.protonode_from_output(&OutputConnector::node(*visible_node, 0), &document.breadcrumb_network_path)
else {
continue;
{
nodes_to_render.insert(protonode);
};
nodes_to_render.insert(protonode);
}
}

View file

@ -1,5 +1,5 @@
use crate::Color;
use glam::{DAffine2, DVec2};
use glam::{DAffine2, DVec2, IVec2, UVec2};
pub trait BoundingBox {
fn bounding_box(&self, transform: DAffine2, include_stroke: bool) -> Option<[DVec2; 2]>;
@ -15,10 +15,80 @@ macro_rules! none_impl {
};
}
none_impl!(String);
none_impl!(bool);
none_impl!(f32);
none_impl!(f64);
none_impl!(DVec2);
none_impl!(Option<Color>);
none_impl!(Vec<Color>);
impl BoundingBox for u32 {
fn bounding_box(&self, _transform: DAffine2, _include_stroke: bool) -> Option<[DVec2; 2]> {
text_bbox(i32_width(*self as i32))
}
}
impl BoundingBox for f64 {
fn bounding_box(&self, _transform: DAffine2, _include_stroke: bool) -> Option<[DVec2; 2]> {
text_bbox(f64_width(*self))
}
}
impl BoundingBox for DVec2 {
fn bounding_box(&self, _transform: DAffine2, _include_stroke: bool) -> Option<[DVec2; 2]> {
let width_x = f64_width(self.x);
let width_y = f64_width(self.y);
let total_width = width_x + width_y + 50.;
text_bbox(total_width)
}
}
impl BoundingBox for IVec2 {
fn bounding_box(&self, _transform: DAffine2, _include_stroke: bool) -> Option<[DVec2; 2]> {
let width_x = i32_width(self.x);
let width_y = i32_width(self.y);
let total_width = width_x + width_y + 50.;
text_bbox(total_width)
}
}
impl BoundingBox for UVec2 {
fn bounding_box(&self, _transform: DAffine2, _include_stroke: bool) -> Option<[DVec2; 2]> {
let width_x = i32_width(self.x as i32);
let width_y = i32_width(self.y as i32);
let total_width = width_x + width_y + 50.;
text_bbox(total_width)
}
}
impl BoundingBox for bool {
fn bounding_box(&self, _transform: DAffine2, _include_stroke: bool) -> Option<[DVec2; 2]> {
text_bbox(60.)
}
}
impl BoundingBox for String {
fn bounding_box(&self, _transform: DAffine2, _include_stroke: bool) -> Option<[DVec2; 2]> {
let width = self.len() * 16;
text_bbox(width as f64)
}
}
impl BoundingBox for Option<Color> {
fn bounding_box(&self, _transform: DAffine2, _include_stroke: bool) -> Option<[DVec2; 2]> {
Some([(0., -5.).into(), (150., 110.).into()])
}
}
fn f64_width(f64: f64) -> f64 {
let left_of_decimal_width = i32_width(f64 as i32);
left_of_decimal_width + 5. + 2. * 16.
}
fn i32_width(i32: i32) -> f64 {
let number_of_digits = (i32.abs()).checked_ilog10().unwrap_or(0) + 1;
let mut width = number_of_digits * 16;
if i32 < 0 {
width += 20;
}
width.into()
}
fn text_bbox(width: f64) -> Option<[DVec2; 2]> {
Some([(-width / 2., 0.).into(), (width / 2., 30.).into()])
}

View file

@ -630,8 +630,8 @@ fn get_animation_time(ctx: impl Ctx + ExtractAnimationTime) -> Option<f64> {
}
#[node_macro::node(category("Context Getter"))]
fn get_index(ctx: impl Ctx + ExtractIndex) -> Option<usize> {
ctx.try_index()
fn get_index(ctx: impl Ctx + ExtractIndex) -> Option<u32> {
ctx.try_index().map(|index| index as u32)
}
// #[node_macro::node(category("Loop"))]

View file

@ -1,4 +1,4 @@
use crate::Color;
use crate::{Color, bounds::BoundingBox, vector::VectorDataTable};
use dyn_any::DynAny;
use glam::{DAffine2, DVec2};
@ -32,6 +32,12 @@ impl Default for GradientStops {
}
}
impl BoundingBox for GradientStops {
fn bounding_box(&self, _transform: DAffine2, _include_stroke: bool) -> Option<[DVec2; 2]> {
Into::<VectorDataTable>::into(Into::<Gradient>::into(self.clone())).bounding_box(DAffine2::default(), false)
}
}
impl IntoIterator for GradientStops {
type Item = (f64, Color);
type IntoIter = std::vec::IntoIter<(f64, Color)>;
@ -169,6 +175,24 @@ impl std::fmt::Display for Gradient {
}
}
impl From<GradientStops> for Gradient {
fn from(gradient_stops: GradientStops) -> Gradient {
Gradient {
stops: gradient_stops.clone(),
gradient_type: GradientType::Linear,
start: (0., 0.).into(),
end: (1., 0.).into(),
transform: DAffine2::IDENTITY,
}
}
}
impl BoundingBox for Gradient {
fn bounding_box(&self, _transform: DAffine2, _include_stroke: bool) -> Option<[DVec2; 2]> {
Into::<VectorDataTable>::into(self.clone()).bounding_box(DAffine2::default(), false)
}
}
impl Gradient {
/// Constructs a new gradient with the colors at 0 and 1 specified.
pub fn new(start: DVec2, start_color: Color, end: DVec2, end_color: Color, transform: DAffine2, gradient_type: GradientType) -> Self {

View file

@ -1,3 +1,4 @@
use crate::gradient::{Gradient, GradientStops};
use crate::instances::Instances;
use crate::raster_types::{CPU, GPU, Raster};
use crate::vector::VectorData;
@ -54,8 +55,10 @@ impl RenderComplexity for Raster<GPU> {
impl RenderComplexity for String {}
impl RenderComplexity for bool {}
impl RenderComplexity for f32 {}
impl RenderComplexity for u32 {}
impl RenderComplexity for f64 {}
impl RenderComplexity for DVec2 {}
impl RenderComplexity for Option<Color> {}
impl RenderComplexity for Vec<Color> {}
impl RenderComplexity for GradientStops {}
impl RenderComplexity for Gradient {}

View file

@ -5,13 +5,15 @@ mod modification;
use super::misc::{dvec2_to_point, point_to_dvec2};
use super::style::{PathStyle, Stroke};
use crate::bounds::BoundingBox;
use crate::gradient::{Gradient, GradientType};
use crate::instances::Instances;
use crate::math::quad::Quad;
use crate::transform::Transform;
use crate::vector::click_target::{ClickTargetType, FreePoint};
use crate::vector::style::Fill;
use crate::{AlphaBlending, Color, GraphicGroupTable};
pub use attributes::*;
use bezier_rs::{BezierHandles, ManipulatorGroup};
use bezier_rs::{BezierHandles, ManipulatorGroup, Subpath};
use core::borrow::Borrow;
use core::hash::Hash;
use dyn_any::DynAny;
@ -511,6 +513,36 @@ impl BoundingBox for VectorDataTable {
}
}
/// Convert a Gradient/GradientStops into VectorDataTable for rendering thumbnails
impl From<Gradient> for VectorDataTable {
fn from(mut gradient: Gradient) -> VectorDataTable {
match gradient.gradient_type {
GradientType::Linear => {
let mut rectangle = VectorData::from_subpath(Subpath::new_rect((0., 0.).into(), (150., 100.).into()));
// Handle vertical gradients
let intersection = if gradient.start.x == gradient.end.x {
DVec2::new(0., 100.)
} else {
let slope = (gradient.start.y - gradient.end.y) / (gradient.start.x - gradient.end.x);
if slope > 100. / 150. { DVec2::new(100. / slope, 100.) } else { DVec2::new(150., slope * 150.) }
};
gradient.start = (0., 0.).into();
gradient.end = intersection;
rectangle.style.fill = Fill::Gradient(gradient);
Instances::new(rectangle)
}
GradientType::Radial => {
let mut circle = VectorData::from_subpath(Subpath::new_ellipse((-100., -100.).into(), (100., 100.).into()));
gradient.start = (0., 0.).into();
gradient.end = (100., 0.).into();
gradient.transform = DAffine2::IDENTITY;
circle.style.fill = Fill::Gradient(gradient);
Instances::new(circle)
}
}
}
}
/// A selectable part of a curve, either an anchor (start or end of a bézier) or a handle (doesn't necessarily go through the bézier but influences curvature).
#[derive(Clone, Copy, PartialEq, Eq, Hash, Debug, DynAny, serde::Serialize, serde::Deserialize)]
pub enum ManipulatorPointId {

View file

@ -7,6 +7,7 @@ pub use glam::{DAffine2, DVec2, IVec2, UVec2};
use graphene_application_io::SurfaceFrame;
use graphene_brush::brush_cache::BrushCache;
use graphene_brush::brush_stroke::BrushStroke;
use graphene_core::gradient::GradientStops;
use graphene_core::raster_types::{CPU, GPU};
use graphene_core::transform::ReferencePoint;
use graphene_core::uuid::NodeId;
@ -562,8 +563,13 @@ thumbnail_render! {
graphene_core::raster_types::RasterDataTable<GPU>,
graphene_core::GraphicElement,
Option<Color>,
GradientStops,
Vec<Color>,
u32,
f64,
DVec2,
bool,
String,
}
pub enum ThumbnailRenderResult {

View file

@ -1,7 +1,9 @@
use glam::DVec2;
pub use graph_craft::document::value::RenderOutputType;
use graph_craft::document::value::{EditorMetadata, RenderOutput};
pub use graph_craft::wasm_application_io::*;
use graphene_application_io::ApplicationIo;
use graphene_core::gradient::GradientStops;
#[cfg(target_arch = "wasm32")]
use graphene_core::instances::Instances;
#[cfg(target_arch = "wasm32")]
@ -240,10 +242,11 @@ async fn render<'a: 'n, T: 'n + GraphicElementRendered + WasmNotSend>(
graphene_core::Artboard,
graphene_core::ArtboardGroupTable,
Option<Color>,
GradientStops,
Vec<Color>,
bool,
f32,
f64,
DVec2,
String,
)]
data: T,

View file

@ -6,6 +6,7 @@ use glam::{DAffine2, DVec2};
use graphene_core::blending::BlendMode;
use graphene_core::bounds::BoundingBox;
use graphene_core::color::Color;
use graphene_core::gradient::{Gradient, GradientStops};
use graphene_core::instances::Instance;
use graphene_core::math::quad::Quad;
use graphene_core::raster::Image;
@ -1185,23 +1186,36 @@ impl GraphicElementRendered for GraphicElement {
}
}
/// Used to stop rust complaining about upstream traits adding display implementations to `Option<Color>`. This would not be an issue as we control that crate.
trait Primitive: std::fmt::Display + BoundingBox + RenderComplexity {}
impl Primitive for String {}
trait Primitive: std::fmt::Display + BoundingBox + RenderComplexity {
fn precision() -> bool {
false
}
}
impl Primitive for u32 {}
impl Primitive for f64 {
fn precision() -> bool {
true
}
}
impl Primitive for DVec2 {
fn precision() -> bool {
true
}
}
impl Primitive for bool {}
impl Primitive for f32 {}
impl Primitive for f64 {}
impl Primitive for DVec2 {}
impl Primitive for String {}
fn text_attributes(attributes: &mut SvgRenderAttrs) {
attributes.push("fill", "black");
attributes.push("font-size", "30");
attributes.push("dominant-baseline", "hanging");
attributes.push("text-anchor", "middle");
}
impl<P: Primitive> GraphicElementRendered for P {
fn render_svg(&self, render: &mut SvgRender, _render_params: &RenderParams) {
log::debug!("Rendering svg for primative: {}", self);
render.parent_tag("text", text_attributes, |render| render.leaf_node(format!("{self}")));
let text = if P::precision() { format!("{:.2}", self) } else { format!("{self}") };
render.parent_tag("text", text_attributes, |render| render.leaf_node(text));
}
#[cfg(feature = "vello")]
@ -1210,22 +1224,33 @@ impl<P: Primitive> GraphicElementRendered for P {
impl GraphicElementRendered for Option<Color> {
fn render_svg(&self, render: &mut SvgRender, _render_params: &RenderParams) {
let Some(color) = self else {
render.parent_tag("text", |_| {}, |render| render.leaf_node("Empty color"));
return;
};
let color_info = format!("{:?} #{} {:?}", color, color.to_rgba_hex_srgb(), color.to_rgba8_srgb());
render.leaf_tag("rect", |attributes| {
attributes.push("width", "100");
attributes.push("height", "100");
attributes.push("y", "40");
attributes.push("fill", format!("#{}", color.to_rgb_hex_srgb_from_gamma()));
if color.a() < 1. {
attributes.push("fill-opacity", ((color.a() * 1000.).round() / 1000.).to_string());
match self {
Some(color) => {
render.leaf_tag("rect", |attributes| {
attributes.push("width", "150");
attributes.push("height", "100");
attributes.push("fill", format!("#{}", color.to_rgb_hex_srgb_from_gamma()));
if color.a() < 1. {
attributes.push("fill-opacity", ((color.a() * 1000.).round() / 1000.).to_string());
}
});
}
});
render.parent_tag("text", text_attributes, |render| render.leaf_node(color_info))
None => {
render.leaf_tag("rect", |attributes| {
attributes.push("width", "150");
attributes.push("height", "100");
attributes.push("fill", format!("#ffffff"));
});
render.leaf_tag("line", |attributes| {
attributes.push("x1", "0");
attributes.push("y1", "100");
attributes.push("x2", "150");
attributes.push("y2", "0");
attributes.push("stroke", "red");
attributes.push("stroke-width", "5");
});
}
}
}
#[cfg(feature = "vello")]
@ -1236,7 +1261,7 @@ impl GraphicElementRendered for Vec<Color> {
fn render_svg(&self, render: &mut SvgRender, _render_params: &RenderParams) {
for (index, &color) in self.iter().enumerate() {
render.leaf_tag("rect", |attributes| {
attributes.push("width", "100");
attributes.push("width", "150");
attributes.push("height", "100");
attributes.push("x", (index * 120).to_string());
attributes.push("y", "40");
@ -1252,6 +1277,23 @@ impl GraphicElementRendered for Vec<Color> {
fn render_to_vello(&self, _scene: &mut Scene, _transform: DAffine2, _context: &mut RenderContext, _render_params: &RenderParams) {}
}
impl GraphicElementRendered for GradientStops {
fn render_svg(&self, render: &mut SvgRender, render_params: &RenderParams) {
// Gradient stops -> Gradient -> Vector data table
Into::<VectorDataTable>::into(Into::<Gradient>::into(self.clone())).render_svg(render, render_params);
}
#[cfg(feature = "vello")]
fn render_to_vello(&self, _scene: &mut Scene, _transform: DAffine2, _context: &mut RenderContext, _render_params: &RenderParams) {}
}
impl GraphicElementRendered for Gradient {
fn render_svg(&self, render: &mut SvgRender, render_params: &RenderParams) {
Into::<VectorDataTable>::into(self.clone()).render_svg(render, render_params);
}
#[cfg(feature = "vello")]
fn render_to_vello(&self, _scene: &mut Scene, _transform: DAffine2, _context: &mut RenderContext, _render_params: &RenderParams) {}
}
#[derive(Debug, Clone, PartialEq, Eq)]
pub enum SvgSegment {
Slice(&'static str),