1 /*****************************************************************************
2 * svg.c : Put SVG on the video
3 *****************************************************************************
4 * Copyright (C) 2002, 2003 the VideoLAN team
7 * Authors: Olivier Aubert <oaubert@lisi.univ-lyon1.fr>
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option ) any later version.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22 *****************************************************************************/
24 /*****************************************************************************
26 *****************************************************************************/
32 #include <vlc_common.h>
33 #include <vlc_plugin.h>
37 #include <vlc_block.h>
38 #include <vlc_filter.h>
40 #include <sys/types.h>
44 #elif defined( WIN32 ) && !defined( UNDER_CE )
49 #include <glib/gstdio.h>
50 #include <glib-object.h> /* g_object_unref( ) */
51 #include <librsvg-2/librsvg/rsvg.h>
53 typedef struct svg_rendition_t svg_rendition_t
;
55 /*****************************************************************************
57 *****************************************************************************/
58 static int Create ( vlc_object_t
* );
59 static void Destroy ( vlc_object_t
* );
60 static int RenderText( filter_t
*p_filter
, subpicture_region_t
*p_region_out
,
61 subpicture_region_t
*p_region_in
);
62 static char *svg_GetTemplate( vlc_object_t
*p_this
);
64 /*****************************************************************************
66 *****************************************************************************/
68 #define TEMPLATE_TEXT N_( "SVG template file" )
69 #define TEMPLATE_LONGTEXT N_( "Location of a file holding a SVG template "\
70 "for automatic string conversion" )
73 set_category( CAT_INPUT
)
74 set_category( SUBCAT_INPUT_SCODEC
)
75 set_capability( "text renderer", 99 )
77 add_string( "svg-template-file", "", NULL
, TEMPLATE_TEXT
, TEMPLATE_LONGTEXT
, true )
78 set_callbacks( Create
, Destroy
)
82 Describes a SVG string to be displayed on the video
84 struct svg_rendition_t
89 /** The SVG source associated with this subpicture */
91 /* The rendered SVG, as a GdkPixbuf */
92 GdkPixbuf
*p_rendition
;
95 static int Render( filter_t
*, subpicture_region_t
*, svg_rendition_t
*, int, int);
96 static char *svg_GetTemplate ();
97 static void svg_set_size( filter_t
*p_filter
, int width
, int height
);
98 static void svg_SizeCallback ( int *width
, int *height
, gpointer data
);
99 static void svg_RenderPicture ( filter_t
*p_filter
,
100 svg_rendition_t
*p_svg
);
101 static void FreeString( svg_rendition_t
* );
103 /*****************************************************************************
104 * filter_sys_t: svg local data
105 *****************************************************************************
106 * This structure is part of the filter thread descriptor.
107 * It describes the svg specific properties of an output thread.
108 *****************************************************************************/
111 /* The SVG template used to convert strings */
113 /* Default size for rendering. Initialized to the output size. */
118 /*****************************************************************************
119 * Create: allocates svg video thread output method
120 *****************************************************************************
121 * This function allocates and initializes a vout method.
122 *****************************************************************************/
123 static int Create( vlc_object_t
*p_this
)
125 filter_t
*p_filter
= ( filter_t
* )p_this
;
128 /* Allocate structure */
129 p_sys
= malloc( sizeof( filter_sys_t
) );
133 /* Initialize psz_template */
134 p_sys
->psz_template
= svg_GetTemplate( p_this
);
135 if( !p_sys
->psz_template
)
141 p_sys
->i_width
= p_filter
->fmt_out
.video
.i_width
;
142 p_sys
->i_height
= p_filter
->fmt_out
.video
.i_height
;
144 p_filter
->pf_render_text
= RenderText
;
145 p_filter
->pf_render_html
= NULL
;
146 p_filter
->p_sys
= p_sys
;
148 /* MUST call this before any RSVG funcs */
154 static char *svg_GetTemplate( vlc_object_t
*p_this
)
156 filter_t
*p_filter
= ( filter_t
* )p_this
;
161 psz_filename
= var_InheritString( p_filter
, "svg-template-file" );
162 if( !psz_filename
|| (psz_filename
[0] == 0) )
164 /* No filename. Use a default value. */
169 /* Read the template */
170 file
= vlc_fopen( psz_filename
, "rt" );
173 msg_Warn( p_this
, "SVG template file %s does not exist.",
181 if( fstat( fileno( file
), &s
) )
183 /* Problem accessing file information. Should not
184 happen as we could open it. */
188 if( ((signed)s
.st_size
) < 0 )
190 msg_Err( p_this
, "SVG template too big" );
195 msg_Dbg( p_this
, "reading %ld bytes from template %s",
196 (unsigned long)s
.st_size
, psz_filename
);
198 psz_template
= malloc( s
.st_size
+ 42 );
202 free( psz_filename
);
205 memset( psz_template
, 0, s
.st_size
+ 1 );
206 if(! fread( psz_template
, s
.st_size
, 1, file
) )
208 msg_Dbg( p_this
, "No data read from template." );
214 free( psz_filename
);
217 /* Either there was no file, or there was an error.
218 Use the default value */
219 psz_template
= strdup( "<?xml version='1.0' encoding='UTF-8' standalone='no'?> \
220 <svg version='1' preserveAspectRatio='xMinYMin meet' viewBox='0 0 800 600'> \
221 <text x='10' y='560' fill='white' font-size='32' \
222 font-family='sans-serif'>%s</text></svg>" );
228 /*****************************************************************************
229 * Destroy: destroy Clone video thread output method
230 *****************************************************************************
231 * Clean up all data and library connections
232 *****************************************************************************/
233 static void Destroy( vlc_object_t
*p_this
)
235 filter_t
*p_filter
= ( filter_t
* )p_this
;
236 filter_sys_t
*p_sys
= p_filter
->p_sys
;
238 free( p_sys
->psz_template
);
243 /*****************************************************************************
244 * Render: render SVG in picture
245 *****************************************************************************/
246 static int Render( filter_t
*p_filter
, subpicture_region_t
*p_region
,
247 svg_rendition_t
*p_svg
, int i_width
, int i_height
)
250 uint8_t *p_y
, *p_u
, *p_v
, *p_a
;
251 int x
, y
, i_pitch
, i_u_pitch
;
252 guchar
*pixels_in
= NULL
;
258 if ( p_filter
->p_sys
->i_width
!= i_width
||
259 p_filter
->p_sys
->i_height
!= i_height
)
261 svg_set_size( p_filter
, i_width
, i_height
);
262 p_svg
->p_rendition
= NULL
;
265 if( p_svg
->p_rendition
== NULL
) {
266 svg_RenderPicture( p_filter
, p_svg
);
267 if( ! p_svg
->p_rendition
)
269 msg_Err( p_filter
, "Cannot render SVG" );
273 i_width
= gdk_pixbuf_get_width( p_svg
->p_rendition
);
274 i_height
= gdk_pixbuf_get_height( p_svg
->p_rendition
);
276 /* Create a new subpicture region */
277 memset( &fmt
, 0, sizeof( video_format_t
) );
278 fmt
.i_chroma
= VLC_CODEC_YUVA
;
279 fmt
.i_width
= fmt
.i_visible_width
= i_width
;
280 fmt
.i_height
= fmt
.i_visible_height
= i_height
;
281 fmt
.i_x_offset
= fmt
.i_y_offset
= 0;
285 p_region
->p_picture
= picture_NewFromFormat( &fmt
);
286 if( !p_region
->p_picture
)
290 p_region
->i_x
= p_region
->i_y
= 0;
291 p_y
= p_region
->p_picture
->Y_PIXELS
;
292 p_u
= p_region
->p_picture
->U_PIXELS
;
293 p_v
= p_region
->p_picture
->V_PIXELS
;
294 p_a
= p_region
->p_picture
->A_PIXELS
;
296 i_pitch
= p_region
->p_picture
->Y_PITCH
;
297 i_u_pitch
= p_region
->p_picture
->U_PITCH
;
299 /* Initialize the region pixels (only the alpha will be changed later) */
300 memset( p_y
, 0x00, i_pitch
* p_region
->fmt
.i_height
);
301 memset( p_u
, 0x80, i_u_pitch
* p_region
->fmt
.i_height
);
302 memset( p_v
, 0x80, i_u_pitch
* p_region
->fmt
.i_height
);
304 p_pic
= p_region
->p_picture
;
308 /* This rendering code is in no way optimized. If someone has some time to
309 make it work faster or better, please do.
313 p_pixbuf->get_rowstride() is the number of bytes in a line.
314 p_pixbuf->get_height() is the number of lines.
316 The number of bytes of p_pixbuf->p_pixels is get_rowstride * get_height
319 alpha = pixels [ n_channels * ( y*rowstride + x ) + 3 ];
321 red = pixels [ n_channels * ( y*rowstride ) + x ) ];
322 green = pixels [ n_channels * ( y*rowstride ) + x ) + 1 ];
323 blue = pixels [ n_channels * ( y*rowstride ) + x ) + 2 ];
326 pixels_in
= gdk_pixbuf_get_pixels( p_svg
->p_rendition
);
327 rowstride_in
= gdk_pixbuf_get_rowstride( p_svg
->p_rendition
);
328 channels_in
= gdk_pixbuf_get_n_channels( p_svg
->p_rendition
);
329 alpha
= gdk_pixbuf_get_has_alpha( p_svg
->p_rendition
);
332 This crashes the plugin (if !alpha). As there is always an alpha value,
333 it does not matter for the moment :
336 memset( p_a, 0xFF, i_pitch * p_region->fmt.i_height );
339 #define INDEX_IN( x, y ) ( y * rowstride_in + x * channels_in )
340 #define INDEX_OUT( x, y ) ( y * i_pitch + x * p_pic->p[Y_PLANE].i_pixel_pitch )
342 for( y
= 0; y
< i_height
; y
++ )
344 for( x
= 0; x
< i_width
; x
++ )
349 p_in
= &pixels_in
[INDEX_IN( x
, y
)];
351 #define R( pixel ) *pixel
352 #define G( pixel ) *( pixel+1 )
353 #define B( pixel ) *( pixel+2 )
354 #define ALPHA( pixel ) *( pixel+3 )
356 /* From http://www.geocrawler.com/archives/3/8263/2001/6/0/6020594/ :
357 Y = 0.29900 * R + 0.58700 * G + 0.11400 * B
358 U = -0.1687 * r - 0.3313 * g + 0.5 * b + 128
359 V = 0.5 * r - 0.4187 * g - 0.0813 * b + 128
362 i_out
= INDEX_OUT( x
, y
);
364 p_pic
->Y_PIXELS
[i_out
] = .299 * R( p_in
) + .587 * G( p_in
) + .114 * B( p_in
);
366 p_pic
->U_PIXELS
[i_out
] = -.1687 * R( p_in
) - .3313 * G( p_in
) + .5 * B( p_in
) + 128;
367 p_pic
->V_PIXELS
[i_out
] = .5 * R( p_in
) - .4187 * G( p_in
) - .0813 * B( p_in
) + 128;
369 p_pic
->A_PIXELS
[i_out
] = ALPHA( p_in
);
377 static void svg_set_size( filter_t
*p_filter
, int width
, int height
)
379 p_filter
->p_sys
->i_width
= width
;
380 p_filter
->p_sys
->i_height
= height
;
383 static void svg_SizeCallback( int *width
, int *height
, gpointer data
)
385 filter_t
*p_filter
= data
;
387 *width
= p_filter
->p_sys
->i_width
;
388 *height
= p_filter
->p_sys
->i_height
;
392 static void svg_RenderPicture( filter_t
*p_filter
,
393 svg_rendition_t
*p_svg
)
395 /* Render the SVG string p_string->psz_text into a new picture_t
396 p_string->p_rendition with dimensions ( ->i_width, ->i_height ) */
397 RsvgHandle
*p_handle
;
398 GError
*error
= NULL
;
400 p_svg
->p_rendition
= NULL
;
402 p_handle
= rsvg_handle_new();
406 msg_Err( p_filter
, "Error creating SVG reader" );
410 rsvg_handle_set_size_callback( p_handle
, svg_SizeCallback
, p_filter
, NULL
);
412 if( ! rsvg_handle_write( p_handle
,
413 ( guchar
* )p_svg
->psz_text
, strlen( p_svg
->psz_text
),
416 msg_Err( p_filter
, "error while rendering SVG: %s", error
->message
);
417 g_object_unref( G_OBJECT( p_handle
) );
421 if( ! rsvg_handle_close( p_handle
, &error
) )
423 msg_Err( p_filter
, "error while rendering SVG (close): %s", error
->message
);
424 g_object_unref( G_OBJECT( p_handle
) );
428 p_svg
->p_rendition
= rsvg_handle_get_pixbuf( p_handle
);
430 g_object_unref( G_OBJECT( p_handle
) );
434 static int RenderText( filter_t
*p_filter
, subpicture_region_t
*p_region_out
,
435 subpicture_region_t
*p_region_in
)
437 filter_sys_t
*p_sys
= p_filter
->p_sys
;
438 svg_rendition_t
*p_svg
= NULL
;
442 if( !p_region_in
|| !p_region_out
) return VLC_EGENERIC
;
443 psz_string
= p_region_in
->psz_text
;
444 if( !psz_string
|| !*psz_string
) return VLC_EGENERIC
;
446 p_svg
= malloc( sizeof( svg_rendition_t
) );
450 p_region_out
->i_x
= p_region_in
->i_x
;
451 p_region_out
->i_y
= p_region_in
->i_y
;
453 /* Check if the data is SVG or pure text. In the latter case,
454 convert the text to SVG. FIXME: find a better test */
455 if( strstr( psz_string
, "<svg" ))
457 /* Data is SVG: duplicate */
458 p_svg
->psz_text
= strdup( psz_string
);
459 if( !p_svg
->psz_text
)
467 /* Data is text. Convert to SVG */
468 /* FIXME: handle p_style attributes */
470 char* psz_template
= p_sys
->psz_template
;
471 length
= strlen( psz_string
) + strlen( psz_template
) + 42;
472 p_svg
->psz_text
= malloc( length
+ 1 );
473 if( !p_svg
->psz_text
)
478 memset( p_svg
->psz_text
, 0, length
+ 1 );
479 snprintf( p_svg
->psz_text
, length
, psz_template
, psz_string
);
481 p_svg
->i_width
= p_sys
->i_width
;
482 p_svg
->i_height
= p_sys
->i_height
;
483 p_svg
->i_chroma
= VLC_CODEC_YUVA
;
486 The input data is stored in the p_string structure,
487 and the function updates the p_rendition attribute. */
488 svg_RenderPicture( p_filter
, p_svg
);
490 Render( p_filter
, p_region_out
, p_svg
, p_svg
->i_width
, p_svg
->i_height
);
496 static void FreeString( svg_rendition_t
*p_svg
)
498 free( p_svg
->psz_text
);
499 /* p_svg->p_rendition is a GdkPixbuf, and its allocation is
500 managed through ref. counting */
501 if( p_svg
->p_rendition
)
502 g_object_unref( p_svg
->p_rendition
);