After an evaluation, GNOME has moved from Bugzilla to GitLab. Learn more about GitLab.
No new issues can be reported in GNOME Bugzilla anymore.
To report an issue in a GNOME project, go to GNOME GitLab.
Do not go to GNOME Gitlab for: Bluefish, Doxygen, GnuCash, GStreamer, java-gnome, LDTP, NetworkManager, Tomboy.
Bug 555682 - Feature request: rsvg_handle_get_dimensions_sub()
Feature request: rsvg_handle_get_dimensions_sub()
Status: RESOLVED FIXED
Product: librsvg
Classification: Core
Component: general
unspecified
Other Linux
: Normal enhancement
: ---
Assigned To: librsvg maintainers
librsvg maintainers
Depends on:
Blocks: 580998
 
 
Reported: 2008-10-09 14:36 UTC by Hagen Schink
Modified: 2011-04-16 20:02 UTC
See Also:
GNOME target: ---
GNOME version: ---


Attachments
The patch (5.65 KB, patch)
2008-10-09 14:38 UTC, Hagen Schink
none Details | Review
The test file (21.34 KB, image/svg)
2008-10-09 14:43 UTC, Hagen Schink
  Details
Patch (5.65 KB, patch)
2008-10-09 15:10 UTC, Hagen Schink
none Details | Review
New Patch with handling of quadratic and cubic Bézier curves (12.64 KB, patch)
2008-10-15 10:09 UTC, Hagen Schink
none Details | Review
Determination of the bounding box by using cairo (3.13 KB, patch)
2008-10-16 09:55 UTC, Hagen Schink
none Details | Review
The extended test file (SVG) (35.04 KB, image/svg)
2008-10-16 09:58 UTC, Hagen Schink
  Details
Very simple test app (950 bytes, application/octet-stream)
2008-10-16 10:02 UTC, Hagen Schink
  Details
Determination of the bounding box by using cairo (2) (14.47 KB, patch)
2008-10-16 10:10 UTC, Hagen Schink
none Details | Review
rsvg_handle_get_dimensions() uses rsvg_handle_get_dimensions_sub() (12.00 KB, patch)
2008-10-18 12:57 UTC, Hagen Schink
none Details | Review
rsvg_handle_get_dimensions() uses rsvg_handle_get_dimensions_sub() ADDED gtk-doc comment (12.48 KB, patch)
2008-10-20 14:55 UTC, Hagen Schink
none Details | Review
rsvg-dimensions.c (2.71 KB, text/x-csrc)
2008-10-30 15:51 UTC, Rob Staudinger
  Details

Description Hagen Schink 2008-10-09 14:36:03 UTC
Here is a patch which provides features requested by the team gtk-css-engine team. The purpose of the new API call 

gboolean
rsvg_handle_get_dimensions_sub (RsvgHandle * handle, RsvgDimensionData * dimension_data, const char *id)

is to provide RsvgDimensionData of subelements of the SVG file.

But I discovered a problem. Some elements do not return the right size in comparison with inkscape. --> I'll attach the SVG file which I used for the test right after this post.

IMPORTANT:
I didn't remove the debug statements for the test of the stated issue. If desired I can provide a new patch without debug statements and the missing comments.

Patch:

=== modified file 'librsvg.def'
--- librsvg.def	2006-11-03 01:32:16 +0000
+++ librsvg.def	2008-10-07 12:25:27 +0000
@@ -15,6 +15,7 @@
 rsvg_handle_get_base_uri
 rsvg_handle_set_base_uri
 rsvg_handle_get_dimensions
+rsvg_handle_get_dimensions_sub
 rsvg_handle_get_title
 rsvg_handle_get_desc
 rsvg_handle_get_metadata

=== modified file 'rsvg-base.c'
--- rsvg-base.c	2008-08-26 16:51:17 +0000
+++ rsvg-base.c	2008-10-09 14:10:06 +0000
@@ -1329,6 +1329,43 @@
     return output;
 }
 
+static RsvgBbox
+_rsvg_find_bbox_sub (RsvgHandle * handle, RsvgNode * node)
+{
+    RsvgDrawingCtx *ctx = g_new (RsvgDrawingCtx, 1);
+    RsvgBbox output;
+    RsvgBboxRender *render = rsvg_bbox_render_new ();
+    ctx->drawsub_stack = NULL;
+    ctx->render = (RsvgRender *) render;
+
+    ctx->state = NULL;
+
+    ctx->defs = handle->priv->defs;
+    ctx->base_uri = g_strdup (handle->priv->base_uri);
+    ctx->dpi_x = handle->priv->dpi_x;
+    ctx->dpi_y = handle->priv->dpi_y;
+    ctx->vb.w = 512;
+    ctx->vb.h = 512;
+    ctx->pango_context = NULL;
+
+    rsvg_state_push (ctx);
+    _rsvg_affine_identity (rsvg_state_current (ctx)->affine);
+    
+    if (node->children->len > 0 && g_strcmp0 (node->type->str, "text")) {
+	printf ("Children in <%s>! %d\n", node->type->str, node->children->len);
+	_rsvg_node_draw_children (node, ctx, 0);
+    }
+    else
+	rsvg_node_draw (node, ctx, 0);
+
+    rsvg_state_pop (ctx);
+
+    output = render->bbox;
+    rsvg_render_free (ctx->render);
+    g_free (ctx);
+    return output;
+}
+
 /**
  * rsvg_handle_get_dimensions
  * @handle: A #RsvgHandle
@@ -1364,6 +1401,8 @@
             bbox = _rsvg_find_bbox (handle);
     }
 
+    printf("Bbox %f %f\n", bbox.w, bbox.h);
+
     dimension_data->width = (int) (_rsvg_css_hand_normalize_length (&sself->w, handle->priv->dpi_x,
                                                                     bbox.w + bbox.x * 2, 12) + 0.5);
     dimension_data->height = (int) (_rsvg_css_hand_normalize_length (&sself->h, handle->priv->dpi_y,
@@ -1378,6 +1417,57 @@
                                     handle->priv->user_data);
 }
 
+gboolean
+rsvg_handle_get_dimensions_sub (RsvgHandle * handle, RsvgDimensionData * dimension_data, const char *id)
+{
+    RsvgNodeSvg *root = NULL;
+    RsvgNode *sself = NULL;
+    RsvgBbox bbox;
+
+    g_return_val_if_fail (handle, FALSE);
+    g_return_val_if_fail (dimension_data, FALSE);
+
+    memset (dimension_data, 0, sizeof (RsvgDimensionData));
+
+    if (id && *id)
+	sself = (RsvgNode *) rsvg_defs_lookup (handle->priv->defs, id);
+
+    if (!sself)
+        return FALSE;
+
+    root = (RsvgNodeSvg *) handle->priv->treebase;
+
+    if (!root)
+	return FALSE;
+
+    printf("Type %s\n", sself->type->str);
+
+    bbox.x = bbox.y = 0;
+    bbox.w = bbox.h = 1;
+
+    bbox = _rsvg_find_bbox_sub (handle, sself);
+
+    printf("Bbox New x: %f y: %f \n \tw: %f h: %f\n\n", bbox.x, bbox.y, bbox.w, bbox.h);
+
+    dimension_data->width = (int) (_rsvg_css_hand_normalize_length_sub (&root->w, bbox.w, handle->priv->dpi_x,
+                                                                    bbox.w + bbox.x * 2, 12) + 0.5);
+    dimension_data->height = (int) (_rsvg_css_hand_normalize_length_sub (&root->h, bbox.h, handle->priv->dpi_y,
+                                                                     bbox.h + bbox.y * 2,
+                                                                     12) + 0.5);
+
+    printf("Dimension width: %d height: %d\n\n", dimension_data->width, dimension_data->height);
+
+    dimension_data->em = dimension_data->width;
+    dimension_data->ex = dimension_data->height;
+
+    if (handle->priv->size_func)
+        (*handle->priv->size_func) (&dimension_data->width, &dimension_data->height,
+                                    handle->priv->user_data);
+
+    return FALSE;
+}
+
+
 /** 
  * rsvg_set_default_dpi
  * @dpi: Dots Per Inch (aka Pixels Per Inch)

=== modified file 'rsvg-css.c'
--- rsvg-css.c	2008-04-25 14:27:44 +0000
+++ rsvg-css.c	2008-10-09 09:10:58 +0000
@@ -252,6 +252,28 @@
     return 0;
 }
 
+double
+_rsvg_css_hand_normalize_length_sub (const RsvgLength * in, gdouble length, 
+				     gdouble pixels_per_inch, gdouble width_or_height, 
+				     gdouble font_size)
+{
+
+    printf("Bbox length: %f\n\n", length);
+
+    if (in->factor == '\0')
+        return length;
+    else if (in->factor == 'p')
+        return length * width_or_height;
+    else if (in->factor == 'm')
+        return length * font_size;
+    else if (in->factor == 'x')
+        return length * font_size / 2.;
+    else if (in->factor == 'i')
+        return length * pixels_per_inch;
+
+    return 0;
+}
+
 gboolean
 rsvg_css_param_match (const char *str, const char *param_name)
 {

=== modified file 'rsvg-private.h'
--- rsvg-private.h	2008-08-26 16:51:17 +0000
+++ rsvg-private.h	2008-10-09 09:21:21 +0000
@@ -344,6 +344,9 @@
 double _rsvg_css_normalize_length	(const RsvgLength * in, RsvgDrawingCtx * ctx, char dir);
 double _rsvg_css_hand_normalize_length	(const RsvgLength * in, gdouble pixels_per_inch,
 					 gdouble width_or_height, gdouble font_size);
+double _rsvg_css_hand_normalize_length_sub	(const RsvgLength * in, gdouble length,
+						 gdouble pixels_per_inch, gdouble width_or_height, 
+						 gdouble font_size);
 
 RsvgLength _rsvg_css_parse_length (const char *str);
 

=== modified file 'rsvg.h'
--- rsvg.h	2007-01-13 15:22:25 +0000
+++ rsvg.h	2008-10-09 14:05:11 +0000
@@ -118,6 +118,8 @@
 
 void rsvg_handle_get_dimensions (RsvgHandle * handle, RsvgDimensionData * dimension_data);
 
+gboolean rsvg_handle_get_dimensions_sub (RsvgHandle * handle, RsvgDimensionData * dimension_data, const char *id);
+
 /* Accessibility API */
 
 G_CONST_RETURN char *rsvg_handle_get_title	(RsvgHandle * handle);
Comment 1 Hagen Schink 2008-10-09 14:38:22 UTC
Created attachment 120275 [details] [review]
The patch
Comment 2 Hagen Schink 2008-10-09 14:43:31 UTC
Created attachment 120276 [details]
The test file

The values of the path elements (id=pathFree, id=pathBezier, id=pathSpiral) match not with the sizes returned by inkscape.
Comment 3 Hagen Schink 2008-10-09 15:10:27 UTC
Created attachment 120277 [details] [review]
Patch

Now rsvg_handle_get_dimensions_sub returns the proper gboolean value. Main issue still exists.
Comment 4 Hagen Schink 2008-10-10 15:36:39 UTC
The function rsvg_bbox_render_path in rsvg-base.c (line 1212) doesn't handle curves. I guess that is why the patch fails to provide correct sizes for the bounding box when curved pathes are involved.
Comment 5 Hagen Schink 2008-10-15 10:09:39 UTC
Created attachment 120633 [details] [review]
New Patch with handling of quadratic and cubic Bézier curves

This patch provides proper handling for cubic and quadratic Bézier curves. Furthermore, it eliminates a small logical bug (line 244 and the following in the patch).

Please be beware of the fact that this patch doesn't take the stroke width into account! This patch provides only the mathematical foundation!

There are commented debug statements in the patch on line:
262, 296, 351
Comment 6 Emmanuel Pacaud 2008-10-15 12:02:37 UTC
>Please be beware of the fact that this patch doesn't take the stroke width into
>account! This patch provides only the mathematical foundation!

I'm not sure trying to reimplement cairo_stroke_extents in librsvg is a good idea. This calculation is more complex than it appears. In addition to the stroke width, you have to also take into account the dash and end cap properties.

If you get a fast analytical solution to this issue, I'm pretty sure the cairo developpers will welcome such a patch to cairo_stroke_extents.
Comment 7 Hagen Schink 2008-10-15 13:14:35 UTC
> I'm not sure trying to reimplement cairo_stroke_extents in librsvg is a good
> idea. 

To use cairo_stroke_extents was one of the first ideas I had in mind. (http://sourceforge.net/mailarchive/forum.php?thread_name=20081011191610.f1497fa7.hagen.schink%40st.ovgu.de&forum_name=librsvg-devel)

But Dominic Lachowicz prefers the analytical approach and that is the only opinion I was aware of so far.

> This calculation is more complex than it appears. In addition to the
> stroke width, you have to also take into account the dash and end cap
> properties.

After an entire week of math I fully agree. librsvg itself doesn't seem to take the stroke width into account when calculating the bounding box. But I have to have a deeper look on the sources first.

> If you get a fast analytical solution to this issue, I'm pretty sure the cairo
> developpers will welcome such a patch to cairo_stroke_extents.

I don't know if the algorithm is fast but it does its job at least. Although it needs further testing. BUT as I mentioned one has to make several assumptions in regard to the analytical approach for example an infinitesimal small stroke width.

I think it's possible to extend the algorithm so it can also handle stroke widths which are not as small as the mentioned one but I cannot promise anything. First I have to make further investigations on that issue.
Comment 8 Hagen Schink 2008-10-16 09:55:02 UTC
Created attachment 120697 [details] [review]
Determination of the bounding box by using cairo

This patch uses the functionality which is already implemented in librsvg. For that purpose it creates a dummy cairo_surface and a dummy cairo_context.

I recommend further testing!

My test SVG and the test app will follow.
Comment 9 Hagen Schink 2008-10-16 09:58:53 UTC
Created attachment 120698 [details]
The extended test file (SVG)

There are several pathes with different stroke widths in the file. That are the test cases I used. Other test cases are appreciated.
Comment 10 Hagen Schink 2008-10-16 10:02:11 UTC
Created attachment 120700 [details]
Very simple test app

A very very simple and basic test app.
Comment 11 Hagen Schink 2008-10-16 10:10:13 UTC
Created attachment 120703 [details] [review]
Determination of the bounding box by using cairo (2)

This patch should work, it includes all changes so far... excuse the inconvenience.
Comment 12 Hagen Schink 2008-10-18 12:57:43 UTC
Created attachment 120827 [details] [review]
rsvg_handle_get_dimensions() uses rsvg_handle_get_dimensions_sub()

rsvg_handle_get_dimensions() uses rsvg_handle_get_dimensions_sub() to determine the bounding box of the svg element, now. The patch works but is slightly tested.
Comment 13 Hagen Schink 2008-10-20 14:55:49 UTC
Created attachment 120934 [details] [review]
rsvg_handle_get_dimensions() uses rsvg_handle_get_dimensions_sub() ADDED gtk-doc comment

Added a gtk-doc comment
Comment 14 Rob Staudinger 2008-10-30 15:51:24 UTC
Created attachment 121659 [details]
rsvg-dimensions.c

Test app for reading dimension data from CSS files. Check help output for usage. Not tested with fragments yet.

Build using  "cc $(pkg-config --cflags --libs librsvg-2.0) -std=c99 -o rsvg-dimensions rsvg-dimensions.c"

Potentially obsoletes attachment #120700 [details].
Comment 15 Dominic Lachowicz 2008-11-22 16:44:55 UTC
Thanks again, and sorry that it's taken me so long to get around to this. Nice work, guys.
Comment 16 Christian Persch 2008-12-05 20:03:55 UTC
I wonder why this was made to return the dimension in ints instread of using doubles? Same question for rsvg_handle_get_position_sub().
Comment 17 Dominic Lachowicz 2008-12-05 20:08:10 UTC
(In reply to comment #16)
> I wonder why this was made to return the dimension in ints instread of using
> doubles? Same question for rsvg_handle_get_position_sub().
> 

Because of pre-existing API, and consistency's sake. rsvg_handle_get_dimensions() introduced the RsvgDimensionData struct, so it maked sense for the "_sub()" version to match the API of the parent version. 

I might consider changing it, but I'll get b-slapped if I change the API, even though there's no API stability guarantee in desktop modules.
Comment 18 Christian Persch 2008-12-05 20:21:07 UTC
I see, thanks.

For the get_dimensions_sub that shouldn't be too big a problem. For the get_position_sub however, consider the case that one has 2 elements in the SVG that one wants to render_cairo_sub, aligned with respect to each other (ex: aisleriot cards). Using get_position_sub to compute the offsets, the int precision loss could translate into ugly non-alignments...
I'll try to use this as-is for now. I guess we can always later add another variant of these _sub functions that return doubles.
Comment 19 Dominic Lachowicz 2008-12-05 20:23:04 UTC
Let me know if you run into trouble. If so, I can change all of them to use doubles and bump the soname.