GNOME Bugzilla – Bug 555682
Feature request: rsvg_handle_get_dimensions_sub()
Last modified: 2011-04-16 20:02:18 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);
Created attachment 120275 [details] [review] The patch
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.
Created attachment 120277 [details] [review] Patch Now rsvg_handle_get_dimensions_sub returns the proper gboolean value. Main issue still exists.
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.
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
>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.
> 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.
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.
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.
Created attachment 120700 [details] Very simple test app A very very simple and basic test app.
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.
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.
Created attachment 120934 [details] [review] rsvg_handle_get_dimensions() uses rsvg_handle_get_dimensions_sub() ADDED gtk-doc comment Added a gtk-doc comment
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].
Thanks again, and sorry that it's taken me so long to get around to this. Nice work, guys.
I wonder why this was made to return the dimension in ints instread of using doubles? Same question for rsvg_handle_get_position_sub().
(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.
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.
Let me know if you run into trouble. If so, I can change all of them to use doubles and bump the soname.