/* GSK - The GTK Scene Kit * * Copyright 2016 Endless * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see . */ /** * SECTION:GskRenderNode * @Title: GskRenderNode * @Short_description: Simple scene graph element * * #GskRenderNode is the basic block in a scene graph to be * rendered using #GskRenderer. * * Each node has a parent, except the top-level node; each node may have * children nodes. * * Each node has an associated drawing surface, which has the size of * the rectangle set using gsk_render_node_set_bounds(). Nodes have an * associated transformation matrix, which is used to position and * transform the node on the scene graph; additionally, they also have * a child transformation matrix, which will be applied to each child. * * Render nodes are meant to be transient; once they have been associated * to a #GskRenderer it's safe to release any reference you have on them. * All #GskRenderNodes are immutable, you can only specify their properties * during construction. */ #include "config.h" #include "gskrendernodeprivate.h" #include "gskdebugprivate.h" #include "gskrendererprivate.h" #include "gsktexture.h" #include #include #include /** * GskRenderNode: (ref-func gsk_render_node_ref) (unref-func gsk_render_node_unref) * * The `GskRenderNode` structure contains only private data. * * Since: 3.90 */ G_DEFINE_BOXED_TYPE (GskRenderNode, gsk_render_node, gsk_render_node_ref, gsk_render_node_unref) static void gsk_render_node_finalize (GskRenderNode *self) { self->node_class->finalize (self); g_clear_pointer (&self->name, g_free); g_free (self); } /*< private > * gsk_render_node_new: * @node_class: class structure for this node * * Returns: (transfer full): the newly created #GskRenderNode */ GskRenderNode * gsk_render_node_new (const GskRenderNodeClass *node_class, gsize extra_size) { GskRenderNode *self; g_return_val_if_fail (node_class != NULL, NULL); g_return_val_if_fail (node_class->node_type != GSK_NOT_A_RENDER_NODE, NULL); self = g_malloc0 (node_class->struct_size + extra_size); self->node_class = node_class; self->ref_count = 1; self->min_filter = GSK_SCALING_FILTER_NEAREST; self->mag_filter = GSK_SCALING_FILTER_NEAREST; return self; } /** * gsk_render_node_ref: * @node: a #GskRenderNode * * Acquires a reference on the given #GskRenderNode. * * Returns: (transfer none): the #GskRenderNode with an additional reference * * Since: 3.90 */ GskRenderNode * gsk_render_node_ref (GskRenderNode *node) { g_return_val_if_fail (GSK_IS_RENDER_NODE (node), NULL); g_atomic_int_inc (&node->ref_count); return node; } /** * gsk_render_node_unref: * @node: a #GskRenderNode * * Releases a reference on the given #GskRenderNode. * * If the reference was the last, the resources associated to the @node are * freed. * * Since: 3.90 */ void gsk_render_node_unref (GskRenderNode *node) { g_return_if_fail (GSK_IS_RENDER_NODE (node)); if (g_atomic_int_dec_and_test (&node->ref_count)) gsk_render_node_finalize (node); } /** * gsk_render_node_get_node_type: * @node: a #GskRenderNode * * Returns the type of the @node. * * Returns: the type of the #GskRenderNode * * Since: 3.90 */ GskRenderNodeType gsk_render_node_get_node_type (GskRenderNode *node) { g_return_val_if_fail (GSK_IS_RENDER_NODE (node), GSK_NOT_A_RENDER_NODE); return node->node_class->node_type; } /** * gsk_render_node_get_bounds: * @node: a #GskRenderNode * @bounds: (out caller-allocates): return location for the boundaries * * Retrieves the boundaries of the @node. The node will not draw outside * of its boundaries. * * Since: 3.90 */ void gsk_render_node_get_bounds (GskRenderNode *node, graphene_rect_t *bounds) { g_return_if_fail (GSK_IS_RENDER_NODE (node)); g_return_if_fail (bounds != NULL); graphene_rect_init_from_rect (bounds, &node->bounds); } void gsk_render_node_set_scaling_filters (GskRenderNode *node, GskScalingFilter min_filter, GskScalingFilter mag_filter) { g_return_if_fail (GSK_IS_RENDER_NODE (node)); if (node->min_filter != min_filter) node->min_filter = min_filter; if (node->mag_filter != mag_filter) node->mag_filter = mag_filter; } /** * gsk_render_node_set_name: * @node: a #GskRenderNode * @name: (nullable): a name for the node * * Sets the name of the node. * * A name is generally useful for debugging purposes. * * Since: 3.90 */ void gsk_render_node_set_name (GskRenderNode *node, const char *name) { g_return_if_fail (GSK_IS_RENDER_NODE (node)); g_free (node->name); node->name = g_strdup (name); } /** * gsk_render_node_get_name: * @node: a #GskRenderNode * * Retrieves the name previously set via gsk_render_node_set_name(). * If no name has been set, %NULL is returned. * * Returns: (nullable): The name previously set via * gsk_render_node_set_name() or %NULL * * Since: 3.90 **/ const char * gsk_render_node_get_name (GskRenderNode *node) { g_return_val_if_fail (GSK_IS_RENDER_NODE (node), NULL); return node->name; } /** * gsk_render_node_draw: * @node: a #GskRenderNode * @cr: cairo context to draw to * * Draw the contents of @node to the given cairo context. * * Typically, you'll use this function to implement fallback rendering * of #GskRenderNodes on an intermediate Cairo context, instead of using * the drawing context associated to a #GdkWindow's rendering buffer. * * For advanced nodes that cannot be supported using Cairo, in particular * for nodes doing 3D operations, this function may fail. **/ void gsk_render_node_draw (GskRenderNode *node, cairo_t *cr) { g_return_if_fail (GSK_IS_RENDER_NODE (node)); g_return_if_fail (cr != NULL); cairo_save (cr); if (!GSK_RENDER_MODE_CHECK (GEOMETRY)) { GSK_NOTE (CAIRO, g_print ("CLIP = { .x = %g, .y = %g, .width = %g, .height = %g }\n", node->bounds.origin.x, node->bounds.origin.y, node->bounds.size.width, node->bounds.size.height)); cairo_rectangle (cr, node->bounds.origin.x, node->bounds.origin.y, node->bounds.size.width, node->bounds.size.height); cairo_clip (cr); } GSK_NOTE (CAIRO, g_print ("Rendering node %s[%p]\n", node->name, node)); node->node_class->draw (node, cr); if (GSK_RENDER_MODE_CHECK (GEOMETRY)) { cairo_set_operator (cr, CAIRO_OPERATOR_OVER); cairo_rectangle (cr, node->bounds.origin.x - 1, node->bounds.origin.y - 1, node->bounds.size.width + 2, node->bounds.size.height + 2); cairo_set_line_width (cr, 2); cairo_set_source_rgba (cr, 0, 0, 0, 0.5); cairo_stroke (cr); } cairo_restore (cr); } #define GSK_RENDER_NODE_SERIALIZATION_VERSION 0 #define GSK_RENDER_NODE_SERIALIZATION_ID "GskRenderNode" GBytes * gsk_render_node_serialize (GskRenderNode *node) { GVariant *node_variant, *variant; GBytes *result; node_variant = gsk_render_node_serialize_node (node); variant = g_variant_new ("(suuv)", GSK_RENDER_NODE_SERIALIZATION_ID, (guint32) GSK_RENDER_NODE_SERIALIZATION_VERSION, (guint32) gsk_render_node_get_node_type (node), node_variant); result = g_variant_get_data_as_bytes (variant); g_variant_unref (variant); return result; } GskRenderNode * gsk_render_node_deserialize (GBytes *bytes) { char *id_string; guint32 version, node_type; GVariant *variant, *node_variant; GskRenderNode *node = NULL; variant = g_variant_new_from_bytes (G_VARIANT_TYPE ("(suuv)"), bytes, FALSE); g_variant_get (variant, "(suuv)", &id_string, &version, &node_type, &node_variant); if (!g_str_equal (id_string, GSK_RENDER_NODE_SERIALIZATION_ID)) goto out; if (version != GSK_RENDER_NODE_SERIALIZATION_VERSION) goto out; node = gsk_render_node_deserialize_node (node_type, node_variant); out: g_free (id_string); g_variant_unref (node_variant); g_variant_unref (variant); return node; }