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

midi

开发平台:

Unix_Linux

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