svg.c
上传用户:kjfoods
上传日期:2020-07-06
资源大小:29949k
文件大小:16k
源码类别:

midi

开发平台:

Unix_Linux

  1. /*****************************************************************************
  2.  * svg.c : Put SVG on the video
  3.  *****************************************************************************
  4.  * Copyright (C) 2002, 2003 the VideoLAN team
  5.  * $Id: 46a5457620ec37a1396170a8ab350eee32f5d71c $
  6.  *
  7.  * Authors: Olivier Aubert <oaubert@lisi.univ-lyon1.fr>
  8.  *
  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.
  13.  *
  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.
  18.  *
  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.  *****************************************************************************/
  23. /*****************************************************************************
  24.  * Preamble
  25.  *****************************************************************************/
  26. #ifdef HAVE_CONFIG_H
  27. # include "config.h"
  28. #endif
  29. #include <vlc_common.h>
  30. #include <vlc_plugin.h>
  31. #include <vlc_charset.h>
  32. #include <vlc_vout.h>
  33. #include <vlc_osd.h>
  34. #include <vlc_block.h>
  35. #include <vlc_filter.h>
  36. #ifdef HAVE_SYS_TYPES_H
  37. #   include <sys/types.h>
  38. #endif
  39. #ifdef HAVE_UNISTD_H
  40. #    include <unistd.h>
  41. #elif defined( WIN32 ) && !defined( UNDER_CE )
  42. #   include <io.h>
  43. #endif
  44. #include <glib.h>
  45. #include <glib/gstdio.h>
  46. #include <glib-object.h>                                  /* g_object_unref( ) */
  47. #include <librsvg-2/librsvg/rsvg.h>
  48. typedef struct svg_rendition_t svg_rendition_t;
  49. /*****************************************************************************
  50.  * Local prototypes
  51.  *****************************************************************************/
  52. static int  Create    ( vlc_object_t * );
  53. static void Destroy   ( vlc_object_t * );
  54. static int  RenderText( filter_t *p_filter, subpicture_region_t *p_region_out,
  55.                         subpicture_region_t *p_region_in );
  56. static char *svg_GetTemplate( vlc_object_t *p_this );
  57. /*****************************************************************************
  58.  * Module descriptor
  59.  *****************************************************************************/
  60. #define TEMPLATE_TEXT N_( "SVG template file" )
  61. #define TEMPLATE_LONGTEXT N_( "Location of a file holding a SVG template "
  62.         "for automatic string conversion" )
  63. vlc_module_begin ()
  64.     set_category( CAT_INPUT )
  65.     set_category( SUBCAT_INPUT_SCODEC )
  66.     set_capability( "text renderer", 99 )
  67.     add_shortcut( "svg" )
  68.     add_string( "svg-template-file", "", NULL, TEMPLATE_TEXT, TEMPLATE_LONGTEXT, true )
  69.     set_callbacks( Create, Destroy )
  70. vlc_module_end ()
  71. /**
  72.    Describes a SVG string to be displayed on the video
  73. */
  74. struct svg_rendition_t
  75. {
  76.     int            i_width;
  77.     int            i_height;
  78.     int            i_chroma;
  79.     /** The SVG source associated with this subpicture */
  80.     char           *psz_text;
  81.     /* The rendered SVG, as a GdkPixbuf */
  82.     GdkPixbuf      *p_rendition;
  83. };
  84. static int Render( filter_t *, subpicture_region_t *, svg_rendition_t *, int, int);
  85. static char *svg_GetTemplate ();
  86. static void svg_set_size( filter_t *p_filter, int width, int height );
  87. static void svg_SizeCallback  ( int *width, int *height, gpointer data );
  88. static void svg_RenderPicture ( filter_t *p_filter,
  89.                                 svg_rendition_t *p_svg );
  90. static void FreeString( svg_rendition_t * );
  91. /*****************************************************************************
  92.  * filter_sys_t: svg local data
  93.  *****************************************************************************
  94.  * This structure is part of the filter thread descriptor.
  95.  * It describes the svg specific properties of an output thread.
  96.  *****************************************************************************/
  97. struct filter_sys_t
  98. {
  99.     /* The SVG template used to convert strings */
  100.     char          *psz_template;
  101.     /* Default size for rendering. Initialized to the output size. */
  102.     int            i_width;
  103.     int            i_height;
  104. };
  105. /*****************************************************************************
  106.  * Create: allocates svg video thread output method
  107.  *****************************************************************************
  108.  * This function allocates and initializes a  vout method.
  109.  *****************************************************************************/
  110. static int Create( vlc_object_t *p_this )
  111. {
  112.     filter_t *p_filter = ( filter_t * )p_this;
  113.     filter_sys_t *p_sys;
  114.     /* Allocate structure */
  115.     p_sys = malloc( sizeof( filter_sys_t ) );
  116.     if( !p_sys )
  117.         return VLC_ENOMEM;
  118.     /* Initialize psz_template */
  119.     p_sys->psz_template = svg_GetTemplate( p_this );
  120.     if( !p_sys->psz_template )
  121.     {
  122.         free( p_sys );
  123.         return VLC_ENOMEM;
  124.     }
  125.     p_sys->i_width = p_filter->fmt_out.video.i_width;
  126.     p_sys->i_height = p_filter->fmt_out.video.i_height;
  127.     p_filter->pf_render_text = RenderText;
  128.     p_filter->pf_render_html = NULL;
  129.     p_filter->p_sys = p_sys;
  130.     /* MUST call this before any RSVG funcs */
  131.     rsvg_init( );
  132.     return VLC_SUCCESS;
  133. }
  134. static char *svg_GetTemplate( vlc_object_t *p_this )
  135. {
  136.     filter_t *p_filter = ( filter_t * )p_this;
  137.     char *psz_filename;
  138.     char *psz_template;
  139.     FILE *file;
  140.     psz_filename = config_GetPsz( p_filter, "svg-template-file" );
  141.     if( !psz_filename || (psz_filename[0] == 0) )
  142.     {
  143.         /* No filename. Use a default value. */
  144.         psz_template = NULL;
  145.     }
  146.     else
  147.     {
  148.         /* Read the template */
  149.         file = utf8_fopen( psz_filename, "rt" );
  150.         if( !file )
  151.         {
  152.             msg_Warn( p_this, "SVG template file %s does not exist.",
  153.                                          psz_filename );
  154.             psz_template = NULL;
  155.         }
  156.         else
  157.         {
  158.             struct stat s;
  159.             if( fstat( fileno( file ), &s ) )
  160.             {
  161.                 /* Problem accessing file information. Should not
  162.                    happen as we could open it. */
  163.                 psz_template = NULL;
  164.             }
  165.             else
  166.             if( ((signed)s.st_size) < 0 )
  167.             {
  168.                 msg_Err( p_this, "SVG template too big" );
  169.                 psz_template = NULL;
  170.             }
  171.             else
  172.             {
  173.                 msg_Dbg( p_this, "reading %ld bytes from template %s",
  174.                          (unsigned long)s.st_size, psz_filename );
  175.                 psz_template = malloc( s.st_size + 42 );
  176.                 if( !psz_template )
  177.                 {
  178.                     fclose( file );
  179.                     free( psz_filename );
  180.                     return NULL;
  181.                 }
  182.                 memset( psz_template, 0, s.st_size + 1 );
  183.                 if(! fread( psz_template, s.st_size, 1, file ) )
  184.                 {
  185.                     msg_Dbg( p_this, "No data read from template." );
  186.                 }
  187.             }
  188.             fclose( file );
  189.         }
  190.     }
  191.     free( psz_filename );
  192.     if( !psz_template )
  193.     {
  194.         /* Either there was no file, or there was an error.
  195.            Use the default value */
  196.         psz_template = strdup( "<?xml version='1.0' encoding='UTF-8' standalone='no'?> 
  197. <svg version='1' preserveAspectRatio='xMinYMin meet' viewBox='0 0 800 600'> 
  198.   <text x='10' y='560' fill='white' font-size='32'  
  199.         font-family='sans-serif'>%s</text></svg>" );
  200.     }
  201.     return psz_template;
  202. }
  203. /*****************************************************************************
  204.  * Destroy: destroy Clone video thread output method
  205.  *****************************************************************************
  206.  * Clean up all data and library connections
  207.  *****************************************************************************/
  208. static void Destroy( vlc_object_t *p_this )
  209. {
  210.     filter_t *p_filter = ( filter_t * )p_this;
  211.     filter_sys_t *p_sys = p_filter->p_sys;
  212.     free( p_sys->psz_template );
  213.     free( p_sys );
  214.     rsvg_term( );
  215. }
  216. /*****************************************************************************
  217.  * Render: render SVG in picture
  218.  *****************************************************************************/
  219. static int Render( filter_t *p_filter, subpicture_region_t *p_region,
  220.                    svg_rendition_t *p_svg, int i_width, int i_height )
  221. {
  222.     video_format_t fmt;
  223.     uint8_t *p_y, *p_u, *p_v, *p_a;
  224.     int x, y, i_pitch, i_u_pitch;
  225.     guchar *pixels_in = NULL;
  226.     int rowstride_in;
  227.     int channels_in;
  228.     int alpha;
  229.     picture_t *p_pic;
  230.     if ( p_filter->p_sys->i_width != i_width ||
  231.          p_filter->p_sys->i_height != i_height )
  232.     {
  233.         svg_set_size( p_filter, i_width, i_height );
  234.         p_svg->p_rendition = NULL;
  235.     }
  236.     if( p_svg->p_rendition == NULL ) {
  237.         svg_RenderPicture( p_filter, p_svg );
  238.         if( ! p_svg->p_rendition )
  239.         {
  240.             msg_Err( p_filter, "Cannot render SVG" );
  241.             return VLC_EGENERIC;
  242.         }
  243.     }
  244.     i_width = gdk_pixbuf_get_width( p_svg->p_rendition );
  245.     i_height = gdk_pixbuf_get_height( p_svg->p_rendition );
  246.     /* Create a new subpicture region */
  247.     memset( &fmt, 0, sizeof( video_format_t ) );
  248.     fmt.i_chroma = VLC_FOURCC( 'Y','U','V','A' );
  249.     fmt.i_aspect = VOUT_ASPECT_FACTOR;
  250.     fmt.i_width = fmt.i_visible_width = i_width;
  251.     fmt.i_height = fmt.i_visible_height = i_height;
  252.     fmt.i_x_offset = fmt.i_y_offset = 0;
  253.     p_region->p_picture = picture_New( fmt.i_chroma, fmt.i_width, fmt.i_height, fmt.i_aspect );
  254.     if( !p_region->p_picture )
  255.         return VLC_EGENERIC;
  256.     p_region->fmt = fmt;
  257.     p_region->i_x = p_region->i_y = 0;
  258.     p_y = p_region->p_picture->Y_PIXELS;
  259.     p_u = p_region->p_picture->U_PIXELS;
  260.     p_v = p_region->p_picture->V_PIXELS;
  261.     p_a = p_region->p_picture->A_PIXELS;
  262.     i_pitch = p_region->p_picture->Y_PITCH;
  263.     i_u_pitch = p_region->p_picture->U_PITCH;
  264.     /* Initialize the region pixels (only the alpha will be changed later) */
  265.     memset( p_y, 0x00, i_pitch * p_region->fmt.i_height );
  266.     memset( p_u, 0x80, i_u_pitch * p_region->fmt.i_height );
  267.     memset( p_v, 0x80, i_u_pitch * p_region->fmt.i_height );
  268.     p_pic = p_region->p_picture;
  269.     /* Copy the data */
  270.     /* This rendering code is in no way optimized. If someone has some time to
  271.        make it work faster or better, please do.
  272.     */
  273.     /*
  274.       p_pixbuf->get_rowstride() is the number of bytes in a line.
  275.       p_pixbuf->get_height() is the number of lines.
  276.       The number of bytes of p_pixbuf->p_pixels is get_rowstride * get_height
  277.       if( has_alpha() ) {
  278.       alpha = pixels [ n_channels * ( y*rowstride + x ) + 3 ];
  279.       }
  280.       red   = pixels [ n_channels * ( y*rowstride ) + x ) ];
  281.       green = pixels [ n_channels * ( y*rowstride ) + x ) + 1 ];
  282.       blue  = pixels [ n_channels * ( y*rowstride ) + x ) + 2 ];
  283.     */
  284.     pixels_in = gdk_pixbuf_get_pixels( p_svg->p_rendition );
  285.     rowstride_in = gdk_pixbuf_get_rowstride( p_svg->p_rendition );
  286.     channels_in = gdk_pixbuf_get_n_channels( p_svg->p_rendition );
  287.     alpha = gdk_pixbuf_get_has_alpha( p_svg->p_rendition );
  288.     /*
  289.       This crashes the plugin (if !alpha). As there is always an alpha value,
  290.       it does not matter for the moment :
  291.     if( !alpha )
  292.       memset( p_a, 0xFF, i_pitch * p_region->fmt.i_height );
  293.     */
  294. #define INDEX_IN( x, y ) ( y * rowstride_in + x * channels_in )
  295. #define INDEX_OUT( x, y ) ( y * i_pitch + x * p_pic->p[Y_PLANE].i_pixel_pitch )
  296.     for( y = 0; y < i_height; y++ )
  297.     {
  298.         for( x = 0; x < i_width; x++ )
  299.         {
  300.             guchar *p_in;
  301.             int i_out;
  302.             p_in = &pixels_in[INDEX_IN( x, y )];
  303. #define R( pixel ) *pixel
  304. #define G( pixel ) *( pixel+1 )
  305. #define B( pixel ) *( pixel+2 )
  306. #define ALPHA( pixel ) *( pixel+3 )
  307.             /* From http://www.geocrawler.com/archives/3/8263/2001/6/0/6020594/ :
  308.                Y = 0.29900 * R + 0.58700 * G + 0.11400 * B
  309.                U = -0.1687 * r  - 0.3313 * g + 0.5 * b + 128
  310.                V = 0.5   * r - 0.4187 * g - 0.0813 * b + 128
  311.             */
  312.             if ( alpha ) {
  313.                 i_out = INDEX_OUT( x, y );
  314.                 p_pic->Y_PIXELS[i_out] = .299 * R( p_in ) + .587 * G( p_in ) + .114 * B( p_in );
  315.                 p_pic->U_PIXELS[i_out] = -.1687 * R( p_in ) - .3313 * G( p_in ) + .5 * B( p_in ) + 128;
  316.                 p_pic->V_PIXELS[i_out] = .5 * R( p_in ) - .4187 * G( p_in ) - .0813 * B( p_in ) + 128;
  317.                 p_pic->A_PIXELS[i_out] = ALPHA( p_in );
  318.             }
  319.         }
  320.     }
  321.     return VLC_SUCCESS;
  322. }
  323. static void svg_set_size( filter_t *p_filter, int width, int height )
  324. {
  325.   p_filter->p_sys->i_width = width;
  326.   p_filter->p_sys->i_height = height;
  327. }
  328. static void svg_SizeCallback( int *width, int *height, gpointer data )
  329. {
  330.     filter_t *p_filter = data;
  331.     *width = p_filter->p_sys->i_width;
  332.     *height = p_filter->p_sys->i_height;
  333.     return;
  334. }
  335. static void svg_RenderPicture( filter_t *p_filter,
  336.                                svg_rendition_t *p_svg )
  337. {
  338.     /* Render the SVG string p_string->psz_text into a new picture_t
  339.        p_string->p_rendition with dimensions ( ->i_width, ->i_height ) */
  340.     RsvgHandle *p_handle;
  341.     GError *error = NULL;
  342.     p_svg->p_rendition = NULL;
  343.     p_handle = rsvg_handle_new();
  344.     if( !p_handle )
  345.     {
  346.         msg_Err( p_filter, "Error creating SVG reader" );
  347.         return;
  348.     }
  349.     rsvg_handle_set_size_callback( p_handle, svg_SizeCallback, p_filter, NULL );
  350.     if( ! rsvg_handle_write( p_handle,
  351.                  ( guchar* )p_svg->psz_text, strlen( p_svg->psz_text ),
  352.                  &error ) )
  353.     {
  354.         msg_Err( p_filter, "error while rendering SVG: %s", error->message );
  355.         g_object_unref( G_OBJECT( p_handle ) );
  356.         return;
  357.     }
  358.     if( ! rsvg_handle_close( p_handle, &error ) )
  359.     {
  360.         msg_Err( p_filter, "error while rendering SVG (close): %s", error->message );
  361.         g_object_unref( G_OBJECT( p_handle ) );
  362.         return;
  363.     }
  364.     p_svg->p_rendition = rsvg_handle_get_pixbuf( p_handle );
  365.     g_object_unref( G_OBJECT( p_handle ) );
  366. }
  367. static int RenderText( filter_t *p_filter, subpicture_region_t *p_region_out,
  368.                        subpicture_region_t *p_region_in )
  369. {
  370.     filter_sys_t *p_sys = p_filter->p_sys;
  371.     svg_rendition_t *p_svg = NULL;
  372.     char *psz_string;
  373.     /* Sanity check */
  374.     if( !p_region_in || !p_region_out ) return VLC_EGENERIC;
  375.     psz_string = p_region_in->psz_text;
  376.     if( !psz_string || !*psz_string ) return VLC_EGENERIC;
  377.     p_svg = malloc( sizeof( svg_rendition_t ) );
  378.     if( !p_svg )
  379.         return VLC_ENOMEM;
  380.     p_region_out->i_x = p_region_in->i_x;
  381.     p_region_out->i_y = p_region_in->i_y;
  382.     /* Check if the data is SVG or pure text. In the latter case,
  383.        convert the text to SVG. FIXME: find a better test */
  384.     if( strstr( psz_string, "<svg" ))
  385.     {
  386.         /* Data is SVG: duplicate */
  387.         p_svg->psz_text = strdup( psz_string );
  388.         if( !p_svg->psz_text )
  389.         {
  390.             free( p_svg );
  391.             return VLC_ENOMEM;
  392.         }
  393.     }
  394.     else
  395.     {
  396.         /* Data is text. Convert to SVG */
  397.         /* FIXME: handle p_style attributes */
  398.         int length;
  399.         char* psz_template = p_sys->psz_template;
  400.         length = strlen( psz_string ) + strlen( psz_template ) + 42;
  401.         p_svg->psz_text = malloc( length + 1 );
  402.         if( !p_svg->psz_text )
  403.         {
  404.             free( p_svg );
  405.             return VLC_ENOMEM;
  406.         }
  407.         memset( p_svg->psz_text, 0, length + 1 );
  408.         snprintf( p_svg->psz_text, length, psz_template, psz_string );
  409.     }
  410.     p_svg->i_width = p_sys->i_width;
  411.     p_svg->i_height = p_sys->i_height;
  412.     p_svg->i_chroma = VLC_FOURCC( 'Y','U','V','A' );
  413.     /* Render the SVG.
  414.        The input data is stored in the p_string structure,
  415.        and the function updates the p_rendition attribute. */
  416.     svg_RenderPicture( p_filter, p_svg );
  417.     Render( p_filter, p_region_out, p_svg, p_svg->i_width, p_svg->i_height );
  418.     FreeString( p_svg );
  419.     return VLC_SUCCESS;
  420. }
  421. static void FreeString( svg_rendition_t *p_svg )
  422. {
  423.     free( p_svg->psz_text );
  424.     /* p_svg->p_rendition is a GdkPixbuf, and its allocation is
  425.        managed through ref. counting */
  426.     if( p_svg->p_rendition )
  427.         g_object_unref( p_svg->p_rendition );
  428.     free( p_svg );
  429. }