video_widgets.c
上传用户:riyaled888
上传日期:2009-03-27
资源大小:7338k
文件大小:12k
源码类别:

多媒体

开发平台:

MultiPlatform

  1. /*****************************************************************************
  2.  * video_widgets.c : OSD widgets manipulation functions
  3.  *****************************************************************************
  4.  * Copyright (C) 2004 VideoLAN
  5.  * $Id: video_widgets.c 9274 2004-11-10 15:16:51Z gbazin $
  6.  *
  7.  * Author: Yoann Peronneau <yoann@videolan.org>
  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., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
  22.  *****************************************************************************/
  23. /*****************************************************************************
  24.  * Preamble
  25.  *****************************************************************************/
  26. #include <stdlib.h>                                                /* free() */
  27. #include <vlc/vout.h>
  28. #include <osd.h>
  29. #include "vlc_video.h"
  30. #include "vlc_filter.h"
  31. #define STYLE_EMPTY 0
  32. #define STYLE_FILLED 1
  33. /*****************************************************************************
  34.  * Local prototypes
  35.  *****************************************************************************/
  36. static void DrawRect( subpicture_t *, int, int, int, int, short );
  37. static void DrawTriangle( subpicture_t *, int, int, int, int, short );
  38. static void CreatePicture( spu_t *, subpicture_t *, int, int, int, int );
  39. static subpicture_t *vout_CreateWidget( spu_t *, int );
  40. /*****************************************************************************
  41.  * Draws a rectangle at the given position in the subpic.
  42.  * It may be filled (fill == STYLE_FILLED) or empty (fill == STYLE_EMPTY).
  43.  *****************************************************************************/
  44. static void DrawRect( subpicture_t *p_subpic, int i_x1, int i_y1,
  45.                       int i_x2, int i_y2, short fill )
  46. {
  47.     int x, y;
  48.     uint8_t *p_a = p_subpic->p_region->picture.A_PIXELS;
  49.     int i_pitch = p_subpic->p_region->picture.Y_PITCH;
  50.     if( fill == STYLE_FILLED )
  51.     {
  52.         for( y = i_y1; y <= i_y2; y++ )
  53.         {
  54.             for( x = i_x1; x <= i_x2; x++ )
  55.             {
  56.                 p_a[ x + i_pitch * y ] = 0xff;
  57.             }
  58.         }
  59.     }
  60.     else
  61.     {
  62.         for( y = i_y1; y <= i_y2; y++ )
  63.         {
  64.             p_a[ i_x1 + i_pitch * y ] = 0xff;
  65.             p_a[ i_x2 + i_pitch * y ] = 0xff;
  66.         }
  67.         for( x = i_x1; x <= i_x2; x++ )
  68.         {
  69.             p_a[ x + i_pitch * i_y1 ] = 0xff;
  70.             p_a[ x + i_pitch * i_y2 ] = 0xff;
  71.         }
  72.     }
  73. }
  74. /*****************************************************************************
  75.  * Draws a triangle at the given position in the subpic.
  76.  * It may be filled (fill == STYLE_FILLED) or empty (fill == STYLE_EMPTY).
  77.  *****************************************************************************/
  78. static void DrawTriangle( subpicture_t *p_subpic, int i_x1, int i_y1,
  79.                           int i_x2, int i_y2, short fill )
  80. {
  81.     int x, y, i_mid, h;
  82.     uint8_t *p_a = p_subpic->p_region->picture.A_PIXELS;
  83.     int i_pitch = p_subpic->p_region->picture.Y_PITCH;
  84.     i_mid = i_y1 + ( ( i_y2 - i_y1 ) >> 1 );
  85.     if( i_x2 >= i_x1 )
  86.     {
  87.         if( fill == STYLE_FILLED )
  88.         {
  89.             for( y = i_y1; y <= i_mid; y++ )
  90.             {
  91.                 h = y - i_y1;
  92.                 for( x = i_x1; x <= i_x1 + h && x <= i_x2; x++ )
  93.                 {
  94.                     p_a[ x + i_pitch * y ] = 0xff;
  95.                     p_a[ x + i_pitch * ( i_y2 - h ) ] = 0xff;
  96.                 }
  97.             }
  98.         }
  99.         else
  100.         {
  101.             for( y = i_y1; y <= i_mid; y++ )
  102.             {
  103.                 h = y - i_y1;
  104.                 p_a[ i_x1 + i_pitch * y ] = 0xff;
  105.                 p_a[ i_x1 + h + i_pitch * y ] = 0xff;
  106.                 p_a[ i_x1 + i_pitch * ( i_y2 - h ) ] = 0xff;
  107.                 p_a[ i_x1 + h + i_pitch * ( i_y2 - h ) ] = 0xff;
  108.             }
  109.         }
  110.     }
  111.     else
  112.     {
  113.         if( fill == STYLE_FILLED )
  114.         {
  115.             for( y = i_y1; y <= i_mid; y++ )
  116.             {
  117.                 h = y - i_y1;
  118.                 for( x = i_x1; x >= i_x1 - h && x >= i_x2; x-- )
  119.                 {
  120.                     p_a[ x + i_pitch * y ] = 0xff;
  121.                     p_a[ x + i_pitch * ( i_y2 - h ) ] = 0xff;
  122.                 }
  123.             }
  124.         }
  125.         else
  126.         {
  127.             for( y = i_y1; y <= i_mid; y++ )
  128.             {
  129.                 h = y - i_y1;
  130.                 p_a[ i_x1 + i_pitch * y ] = 0xff;
  131.                 p_a[ i_x1 - h + i_pitch * y ] = 0xff;
  132.                 p_a[ i_x1 + i_pitch * ( i_y2 - h ) ] = 0xff;
  133.                 p_a[ i_x1 - h + i_pitch * ( i_y2 - h ) ] = 0xff;
  134.             }
  135.         }
  136.     }
  137. }
  138. /*****************************************************************************
  139.  * Create Picture: creates subpicture region and picture
  140.  *****************************************************************************/
  141. static void CreatePicture( spu_t *p_spu, subpicture_t *p_subpic,
  142.                            int i_x, int i_y, int i_width, int i_height )
  143. {
  144.     uint8_t *p_y, *p_u, *p_v, *p_a;
  145.     video_format_t fmt;
  146.     int i_pitch;
  147.     /* Create a new subpicture region */
  148.     memset( &fmt, 0, sizeof(video_format_t) );
  149.     fmt.i_chroma = VLC_FOURCC('Y','U','V','A');
  150.     fmt.i_aspect = VOUT_ASPECT_FACTOR;
  151.     fmt.i_width = fmt.i_visible_width = i_width;
  152.     fmt.i_height = fmt.i_visible_height = i_height;
  153.     fmt.i_x_offset = fmt.i_y_offset = 0;
  154.     p_subpic->p_region = p_subpic->pf_create_region( VLC_OBJECT(p_spu), &fmt );
  155.     if( !p_subpic->p_region )
  156.     {
  157.         msg_Err( p_spu, "cannot allocate SPU region" );
  158.         return;
  159.     }
  160.     p_subpic->p_region->i_x = i_x;
  161.     p_subpic->p_region->i_y = i_y;
  162.     p_y = p_subpic->p_region->picture.Y_PIXELS;
  163.     p_u = p_subpic->p_region->picture.U_PIXELS;
  164.     p_v = p_subpic->p_region->picture.V_PIXELS;
  165.     p_a = p_subpic->p_region->picture.A_PIXELS;
  166.     i_pitch = p_subpic->p_region->picture.Y_PITCH;
  167.     /* Initialize the region pixels (only the alpha will be changed later) */
  168.     memset( p_y, 0xff, i_pitch * p_subpic->p_region->fmt.i_height );
  169.     memset( p_u, 0x80, i_pitch * p_subpic->p_region->fmt.i_height );
  170.     memset( p_v, 0x80, i_pitch * p_subpic->p_region->fmt.i_height );
  171.     memset( p_a, 0x00, i_pitch * p_subpic->p_region->fmt.i_height );
  172. }
  173. /*****************************************************************************
  174.  * Creates and initializes an OSD widget.
  175.  *****************************************************************************/
  176. subpicture_t *vout_CreateWidget( spu_t *p_spu, int i_channel )
  177. {
  178.     subpicture_t *p_subpic;
  179.     mtime_t i_now = mdate();
  180.     /* Create and initialize a subpicture */
  181.     p_subpic = spu_CreateSubpicture( p_spu );
  182.     if( p_subpic == NULL ) return NULL;
  183.     p_subpic->i_channel = i_channel;
  184.     p_subpic->i_start = i_now;
  185.     p_subpic->i_stop = i_now + 1200000;
  186.     p_subpic->b_ephemer = VLC_TRUE;
  187.     p_subpic->b_fade = VLC_TRUE;
  188.     return p_subpic;
  189. }
  190. /*****************************************************************************
  191.  * Displays an OSD slider.
  192.  * Types are: OSD_HOR_SLIDER and OSD_VERT_SLIDER.
  193.  *****************************************************************************/
  194. void vout_OSDSlider( vlc_object_t *p_caller, int i_channel, int i_position,
  195.                      short i_type )
  196. {
  197.     vout_thread_t *p_vout = vlc_object_find( p_caller, VLC_OBJECT_VOUT,
  198.                                              FIND_ANYWHERE );
  199.     subpicture_t *p_subpic;
  200.     int i_x_margin, i_y_margin, i_x, i_y, i_width, i_height;
  201.     if( p_vout == NULL || !config_GetInt( p_caller, "osd" ) || i_position < 0 )
  202.     {
  203.         return;
  204.     }
  205.     p_subpic = vout_CreateWidget( p_vout->p_spu, i_channel );
  206.     if( p_subpic == NULL )
  207.     {
  208.         return;
  209.     }
  210.     i_y_margin = p_vout->render.i_height / 10;
  211.     i_x_margin = i_y_margin;
  212.     if( i_type == OSD_HOR_SLIDER )
  213.     {
  214.         i_width = p_vout->render.i_width - 2 * i_x_margin;
  215.         i_height = p_vout->render.i_height / 20;
  216.         i_x = i_x_margin;
  217.         i_y = p_vout->render.i_height - i_y_margin - i_height;
  218.     }
  219.     else
  220.     {
  221.         i_width = p_vout->render.i_width / 40;
  222.         i_height = p_vout->render.i_height - 2 * i_y_margin;
  223.         i_x = p_vout->render.i_width - i_x_margin - i_width;
  224.         i_y = i_y_margin;
  225.     }
  226.     /* Create subpicture region and picture */
  227.     CreatePicture( p_vout->p_spu, p_subpic, i_x, i_y, i_width, i_height );
  228.     if( i_type == OSD_HOR_SLIDER )
  229.     {
  230.         int i_x_pos = ( i_width - 2 ) * i_position / 100;
  231.         DrawRect( p_subpic, i_x_pos - 1, 2, i_x_pos + 1,
  232.                   i_height - 3, STYLE_FILLED );
  233.         DrawRect( p_subpic, 0, 0, i_width - 1, i_height - 1, STYLE_EMPTY );
  234.     }
  235.     else if( i_type == OSD_VERT_SLIDER )
  236.     {
  237.         int i_y_pos = i_height / 2;
  238.         DrawRect( p_subpic, 2, i_height - ( i_height - 2 ) * i_position / 100,
  239.                   i_width - 3, i_height - 3, STYLE_FILLED );
  240.         DrawRect( p_subpic, 1, i_y_pos, 1, i_y_pos, STYLE_FILLED );
  241.         DrawRect( p_subpic, i_width - 2, i_y_pos,
  242.                   i_width - 2, i_y_pos, STYLE_FILLED );
  243.         DrawRect( p_subpic, 0, 0, i_width - 1, i_height - 1, STYLE_EMPTY );
  244.     }
  245.     spu_DisplaySubpicture( p_vout->p_spu, p_subpic );
  246.     vlc_object_release( p_vout );
  247.     return;
  248. }
  249. /*****************************************************************************
  250.  * Displays an OSD icon.
  251.  * Types are: OSD_PLAY_ICON, OSD_PAUSE_ICON, OSD_SPEAKER_ICON, OSD_MUTE_ICON
  252.  *****************************************************************************/
  253. void vout_OSDIcon( vlc_object_t *p_caller, int i_channel, short i_type )
  254. {
  255.     vout_thread_t *p_vout = vlc_object_find( p_caller, VLC_OBJECT_VOUT,
  256.                                              FIND_ANYWHERE );
  257.     subpicture_t *p_subpic;
  258.     int i_x_margin, i_y_margin, i_x, i_y, i_width, i_height;
  259.     if( p_vout == NULL || !config_GetInt( p_caller, "osd" ) )
  260.     {
  261.         return;
  262.     }
  263.     p_subpic = vout_CreateWidget( p_vout->p_spu, i_channel );
  264.     if( p_subpic == NULL )
  265.     {
  266.         return;
  267.     }
  268.     i_y_margin = p_vout->render.i_height / 15;
  269.     i_x_margin = i_y_margin;
  270.     i_width = p_vout->render.i_width / 20;
  271.     i_height = i_width;
  272.     i_x = p_vout->render.i_width - i_x_margin - i_width;
  273.     i_y = i_y_margin;
  274.     /* Create subpicture region and picture */
  275.     CreatePicture( p_vout->p_spu, p_subpic, i_x, i_y, i_width, i_height );
  276.     if( i_type == OSD_PAUSE_ICON )
  277.     {
  278.         int i_bar_width = i_width / 3;
  279.         DrawRect( p_subpic, 0, 0, i_bar_width - 1, i_height -1, STYLE_FILLED );
  280.         DrawRect( p_subpic, i_width - i_bar_width, 0,
  281.                   i_width - 1, i_height - 1, STYLE_FILLED );
  282.     }
  283.     else if( i_type == OSD_PLAY_ICON )
  284.     {
  285.         int i_mid = i_height >> 1;
  286.         int i_delta = ( i_width - i_mid ) >> 1;
  287.         int i_y2 = ( ( i_height - 1 ) >> 1 ) * 2;
  288.         DrawTriangle( p_subpic, i_delta, 0, i_width - i_delta, i_y2,
  289.                       STYLE_FILLED );
  290.     }
  291.     else if( i_type == OSD_SPEAKER_ICON || i_type == OSD_MUTE_ICON )
  292.     {
  293.         int i_mid = i_height >> 1;
  294.         int i_delta = ( i_width - i_mid ) >> 1;
  295.         int i_y2 = ( ( i_height - 1 ) >> 1 ) * 2;
  296.         DrawRect( p_subpic, i_delta, i_mid / 2, i_width - i_delta,
  297.                   i_height - 1 - i_mid / 2, STYLE_FILLED );
  298.         DrawTriangle( p_subpic, i_width - i_delta, 0, i_delta, i_y2,
  299.                       STYLE_FILLED );
  300.         if( i_type == OSD_MUTE_ICON )
  301.         {
  302.             uint8_t *p_a = p_subpic->p_region->picture.A_PIXELS;
  303.             int i_pitch = p_subpic->p_region->picture.Y_PITCH;
  304.             int i;
  305.             for( i = 1; i < i_pitch; i++ )
  306.             {
  307.                 int k = i + ( i_height - i - 1 ) * i_pitch;
  308.                 p_a[ k ] = 0xff - p_a[ k ];
  309.             }
  310.         }
  311.     }
  312.     spu_DisplaySubpicture( p_vout->p_spu, p_subpic );
  313.     vlc_object_release( p_vout );
  314.     return;
  315. }