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

midi

开发平台:

Unix_Linux

  1. /*****************************************************************************
  2.  * osd.c - The OSD Menu core code.
  3.  *****************************************************************************
  4.  * Copyright (C) 2005-2008 M2X
  5.  * $Id: 9102c12999d7bac50bb66e1d865d44e0000f29af $
  6.  *
  7.  * Authors: Jean-Paul Saman <jpsaman #_at_# m2x dot nl>
  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_keys.h>
  31. #include <vlc_osd.h>
  32. #include <vlc_image.h>
  33. #include "libvlc.h"
  34. #undef OSD_MENU_DEBUG
  35. /*****************************************************************************
  36.  * Local prototypes
  37.  *****************************************************************************/
  38. static void osd_UpdateState( osd_menu_state_t *, int, int, int, int, picture_t * );
  39. static inline osd_state_t *osd_VolumeStateChange( osd_state_t *, int );
  40. static int osd_VolumeStep( vlc_object_t *, int, int );
  41. static bool osd_isVisible( osd_menu_t *p_osd );
  42. static bool osd_ParserLoad( osd_menu_t *, const char * );
  43. static void osd_ParserUnload( osd_menu_t * );
  44. static bool osd_isVisible( osd_menu_t *p_osd )
  45. {
  46.     vlc_value_t val;
  47.     var_Get( p_osd, "osd-menu-visible", &val );
  48.     return val.b_bool;
  49. }
  50. static vlc_mutex_t *osd_GetMutex( vlc_object_t *p_this )
  51. {
  52.     vlc_value_t lockval;
  53.     var_Create( p_this->p_libvlc, "osd_mutex", VLC_VAR_MUTEX );
  54.     var_Get( p_this->p_libvlc, "osd_mutex", &lockval );
  55.     return lockval.p_address;
  56. }
  57. /*****************************************************************************
  58.  * Wrappers for loading and unloading osd parser modules.
  59.  *****************************************************************************/
  60. static bool osd_ParserLoad( osd_menu_t *p_menu, const char *psz_file )
  61. {
  62.     /* Stuff needed for Parser */
  63.     p_menu->psz_file = strdup( psz_file );
  64.     p_menu->p_image = image_HandlerCreate( p_menu );
  65.     if( !p_menu->p_image || !p_menu->psz_file )
  66.     {
  67.         msg_Err( p_menu, "unable to load images, aborting .." );
  68.         return true;
  69.     }
  70.     else
  71.     {
  72.         char *psz_type;
  73.         char *psz_ext = strrchr( p_menu->psz_file, '.' );
  74.         if( psz_ext && !strcmp( psz_ext, ".cfg") )
  75.             psz_type = (char*)"import-osd";
  76.         else
  77.             psz_type = (char*)"import-osd-xml";
  78.         p_menu->p_parser = module_need( p_menu, "osd parser",
  79.                                         psz_type, true );
  80.         if( !p_menu->p_parser )
  81.         {
  82.             return false;
  83.         }
  84.     }
  85.     return true;
  86. }
  87. static void osd_ParserUnload( osd_menu_t *p_menu )
  88. {
  89.     if( p_menu->p_image )
  90.         image_HandlerDelete( p_menu->p_image );
  91.     if( p_menu->p_parser )
  92.         module_unneed( p_menu, p_menu->p_parser );
  93.     free( p_menu->psz_file );
  94. }
  95. /**
  96.  * Change state on an osd_button_t.
  97.  *
  98.  * This function selects the specified state and returns a pointer vlc_custom_createto it. The
  99.  * following states are currently supported:
  100.  * see OSD_BUTTON_UNSELECT
  101.  * see OSD_BUTTON_SELECT
  102.  * see OSD_BUTTON_PRESSED
  103.  */
  104. static osd_state_t *osd_StateChange( osd_button_t *p_button, const int i_state )
  105. {
  106.     osd_state_t *p_current = p_button->p_states;
  107.     osd_state_t *p_temp = NULL;
  108.     int i = 0;
  109.     for( i=0; p_current != NULL; i++ )
  110.     {
  111.         if( p_current->i_state == i_state )
  112.         {
  113.             p_button->i_x = p_current->i_x;
  114.             p_button->i_y = p_current->i_y;
  115.             p_button->i_width = p_current->i_width;
  116.             p_button->i_height = p_current->i_height;
  117.             return p_current;
  118.         }
  119.         p_temp = p_current->p_next;
  120.         p_current = p_temp;
  121.     }
  122.     return p_button->p_states;
  123. }
  124. /*****************************************************************************
  125.  * OSD menu Funtions
  126.  *****************************************************************************/
  127. osd_menu_t *__osd_MenuCreate( vlc_object_t *p_this, const char *psz_file )
  128. {
  129.     osd_menu_t  *p_osd = NULL;
  130.     vlc_value_t val;
  131.     vlc_mutex_t *p_lock;
  132.     int         i_volume = 0;
  133.     int         i_steps = 0;
  134.     /* to be sure to avoid multiple creation */
  135.     p_lock = osd_GetMutex( p_this );
  136.     vlc_mutex_lock( p_lock );
  137.     var_Create( p_this->p_libvlc, "osd", VLC_VAR_ADDRESS );
  138.     var_Get( p_this->p_libvlc, "osd", &val );
  139.     if( val.p_address == NULL )
  140.     {
  141.         static const char osdmenu_name[] = "osd menu";
  142.         p_osd = vlc_custom_create( p_this, sizeof( *p_osd ),
  143.                                    VLC_OBJECT_GENERIC, osdmenu_name );
  144.         if( !p_osd )
  145.             return NULL;
  146.         p_osd->p_parser = NULL;
  147.         vlc_object_attach( p_osd, p_this->p_libvlc );
  148.         /* Parse configuration file */
  149.         if ( !osd_ParserLoad( p_osd, psz_file ) )
  150.             goto error;
  151.         if( !p_osd->p_state )
  152.             goto error;
  153.         /* Setup default button (first button) */
  154.         p_osd->p_state->p_visible = p_osd->p_button;
  155.         p_osd->p_state->p_visible->p_current_state =
  156.             osd_StateChange( p_osd->p_state->p_visible, OSD_BUTTON_SELECT );
  157.         p_osd->i_width  = p_osd->p_state->p_visible->p_current_state->p_pic->p[Y_PLANE].i_visible_pitch;
  158.         p_osd->i_height = p_osd->p_state->p_visible->p_current_state->p_pic->p[Y_PLANE].i_visible_lines;
  159.         if( p_osd->p_state->p_volume )
  160.         {
  161.             /* Update the volume state images to match the current volume */
  162.             i_volume = config_GetInt( p_this, "volume" );
  163.             i_steps = osd_VolumeStep( p_this, i_volume, p_osd->p_state->p_volume->i_ranges );
  164.             p_osd->p_state->p_volume->p_current_state = osd_VolumeStateChange(
  165.                                     p_osd->p_state->p_volume->p_states, i_steps );
  166.         }
  167.         /* Initialize OSD state */
  168.         osd_UpdateState( p_osd->p_state, p_osd->i_x, p_osd->i_y,
  169.                          p_osd->i_width, p_osd->i_height, NULL );
  170.         /* Signal when an update of OSD menu is needed */
  171.         var_Create( p_osd, "osd-menu-update", VLC_VAR_BOOL );
  172.         var_Create( p_osd, "osd-menu-visible", VLC_VAR_BOOL );
  173.         var_SetBool( p_osd, "osd-menu-update", false );
  174.         var_SetBool( p_osd, "osd-menu-visible", false );
  175.         val.p_address = p_osd;
  176.         var_Set( p_this->p_libvlc, "osd", val );
  177.     }
  178.     else
  179.         p_osd = val.p_address;
  180.     vlc_object_hold( p_osd );
  181.     vlc_mutex_unlock( p_lock );
  182.     return p_osd;
  183. error:
  184.     vlc_mutex_unlock( p_lock );
  185.     __osd_MenuDelete( p_this, p_osd );
  186.     return NULL;
  187. }
  188. void __osd_MenuDelete( vlc_object_t *p_this, osd_menu_t *p_osd )
  189. {
  190.     vlc_mutex_t *p_lock;
  191.     if( !p_osd || !p_this ) return;
  192.     p_lock = osd_GetMutex( p_this );
  193.     vlc_mutex_lock( p_lock );
  194.     if( vlc_internals( VLC_OBJECT(p_osd) )->i_refcount == 1 )
  195.     {
  196.         vlc_value_t val;
  197.         var_Destroy( p_osd, "osd-menu-visible" );
  198.         var_Destroy( p_osd, "osd-menu-update" );
  199.         osd_ParserUnload( p_osd );
  200.         val.p_address = NULL;
  201.         var_Set( p_this->p_libvlc, "osd", val );
  202.     }
  203.     vlc_object_release( p_osd );
  204.     vlc_mutex_unlock( p_lock );
  205. }
  206. static osd_menu_t *osd_Find( vlc_object_t *p_this )
  207. {
  208.     vlc_value_t val;
  209.     if( var_Get( p_this->p_libvlc, "osd", &val ) )
  210.         return NULL;
  211.     return val.p_address;
  212. }
  213. /* The volume can be modified in another interface while the OSD Menu
  214.  * has not been instantiated yet. This routines updates the "volume OSD menu item"
  215.  * to reflect the current state of the GUI.
  216.  */
  217. static inline osd_state_t *osd_VolumeStateChange( osd_state_t *p_current, int i_steps )
  218. {
  219.     osd_state_t *p_temp = NULL;
  220.     int i;
  221.     if( i_steps < 0 ) i_steps = 0;
  222.     for( i=0; (i < i_steps) && (p_current != NULL); i++ )
  223.     {
  224.         p_temp = p_current->p_next;
  225.         if( !p_temp ) return p_current;
  226.         p_current = p_temp;
  227.     }
  228.     return (!p_temp) ? p_current : p_temp;
  229. }
  230. /* Update the state of the OSD Menu */
  231. static void osd_UpdateState( osd_menu_state_t *p_state, int i_x, int i_y,
  232.         int i_width, int i_height, picture_t *p_pic )
  233. {
  234.     p_state->i_x = i_x;
  235.     p_state->i_y = i_y;
  236.     p_state->i_width = i_width;
  237.     p_state->i_height = i_height;
  238.     p_state->p_pic = p_pic;
  239. }
  240. void __osd_MenuShow( vlc_object_t *p_this )
  241. {
  242.     osd_menu_t *p_osd;
  243.     osd_button_t *p_button = NULL;
  244.     vlc_mutex_t *p_lock = osd_GetMutex( p_this );
  245.     vlc_mutex_lock( p_lock );
  246.     p_osd = osd_Find( p_this );
  247.     if( p_osd == NULL )
  248.     {
  249.         vlc_mutex_unlock( p_lock );
  250.         msg_Err( p_this, "osd_MenuShow failed" );
  251.         return;
  252.     }
  253. #if defined(OSD_MENU_DEBUG)
  254.     msg_Dbg( p_osd, "menu on" );
  255. #endif
  256.     p_button = p_osd->p_state->p_visible;
  257.     if( p_button )
  258.     {
  259.         if( !p_button->b_range )
  260.             p_button->p_current_state = osd_StateChange( p_button, OSD_BUTTON_UNSELECT );
  261.         p_osd->p_state->p_visible = p_osd->p_button;
  262.         if( !p_osd->p_state->p_visible->b_range )
  263.             p_osd->p_state->p_visible->p_current_state =
  264.                 osd_StateChange( p_osd->p_state->p_visible, OSD_BUTTON_SELECT );
  265.         osd_UpdateState( p_osd->p_state,
  266.                 p_osd->p_state->p_visible->i_x, p_osd->p_state->p_visible->i_y,
  267.                 p_osd->p_state->p_visible->p_current_state->i_width,
  268.                 p_osd->p_state->p_visible->p_current_state->i_height,
  269.                 p_osd->p_state->p_visible->p_current_state->p_pic );
  270.         osd_SetMenuUpdate( p_osd, true );
  271.     }
  272.     osd_SetMenuVisible( p_osd, true );
  273.     vlc_mutex_unlock( p_lock );
  274. }
  275. void __osd_MenuHide( vlc_object_t *p_this )
  276. {
  277.     osd_menu_t *p_osd;
  278.     vlc_mutex_t *p_lock = osd_GetMutex( p_this );
  279.     vlc_mutex_lock( p_lock );
  280.     p_osd = osd_Find( p_this );
  281.     if( p_osd == NULL )
  282.     {
  283.         vlc_mutex_unlock( p_lock );
  284.         msg_Err( p_this, "osd_MenuHide failed" );
  285.         return;
  286.     }
  287. #if defined(OSD_MENU_DEBUG)
  288.     msg_Dbg( p_osd, "menu off" );
  289. #endif
  290.     osd_UpdateState( p_osd->p_state,
  291.                 p_osd->p_state->i_x, p_osd->p_state->i_y,
  292.                 0, 0, NULL );
  293.     osd_SetMenuUpdate( p_osd, true );
  294.     vlc_mutex_unlock( p_lock );
  295. }
  296. void __osd_MenuActivate( vlc_object_t *p_this )
  297. {
  298.     osd_menu_t *p_osd;
  299.     osd_button_t *p_button = NULL;
  300.     vlc_mutex_t *p_lock = osd_GetMutex( p_this );
  301.     vlc_mutex_lock( p_lock );
  302.     p_osd = osd_Find( p_this );
  303.     if( p_osd == NULL || !osd_isVisible( p_osd ) )
  304.     {
  305.         vlc_mutex_unlock( p_lock );
  306.         msg_Err( p_this, "osd_MenuActivate failed" );
  307.         return;
  308.     }
  309. #if defined(OSD_MENU_DEBUG)
  310.     msg_Dbg( p_osd, "select" );
  311. #endif
  312.     p_button = p_osd->p_state->p_visible;
  313.     /*
  314.      * Is there a menu item above or below? If so, then select it.
  315.      */
  316.     if( p_button && p_button->p_up )
  317.     {
  318.         vlc_mutex_unlock( p_lock );
  319.         __osd_MenuUp( p_this );   /* "menu select" means go to menu item above. */
  320.         return;
  321.     }
  322.     if( p_button && p_button->p_down )
  323.     {
  324.         vlc_mutex_unlock( p_lock );
  325.         __osd_MenuDown( p_this ); /* "menu select" means go to menu item below. */
  326.         return;
  327.     }
  328.     if( p_button && !p_button->b_range )
  329.     {
  330.         p_button->p_current_state = osd_StateChange( p_button, OSD_BUTTON_PRESSED );
  331.         osd_UpdateState( p_osd->p_state,
  332.                 p_button->i_x, p_button->i_y,
  333.                 p_osd->p_state->p_visible->p_current_state->i_width,
  334.                 p_osd->p_state->p_visible->p_current_state->i_height,
  335.                 p_button->p_current_state->p_pic );
  336.         osd_SetMenuUpdate( p_osd, true );
  337.         osd_SetMenuVisible( p_osd, true );
  338.         osd_SetKeyPressed( VLC_OBJECT(p_osd->p_libvlc), config_GetInt( p_osd, p_button->psz_action ) );
  339. #if defined(OSD_MENU_DEBUG)
  340.         msg_Dbg( p_osd, "select (%d, %s)", config_GetInt( p_osd, p_button->psz_action ), p_button->psz_action );
  341. #endif
  342.     }
  343.     vlc_mutex_unlock( p_lock );
  344. }
  345. void __osd_MenuNext( vlc_object_t *p_this )
  346. {
  347.     osd_menu_t *p_osd;
  348.     osd_button_t *p_button = NULL;
  349.     vlc_mutex_t *p_lock = osd_GetMutex( p_this );
  350.     vlc_mutex_lock( p_lock );
  351.     p_osd = osd_Find( p_this );
  352.     if( p_osd == NULL || !osd_isVisible( p_osd ) )
  353.     {
  354.         vlc_mutex_unlock( p_lock );
  355.         msg_Err( p_this, "osd_MenuNext failed" );
  356.         return;
  357.     }
  358.     p_button = p_osd->p_state->p_visible;
  359.     if( p_button )
  360.     {
  361.         if( !p_button->b_range )
  362.             p_button->p_current_state = osd_StateChange( p_button, OSD_BUTTON_UNSELECT );
  363.         if( p_button->p_next )
  364.             p_osd->p_state->p_visible = p_button->p_next;
  365.         else
  366.             p_osd->p_state->p_visible = p_osd->p_button;
  367.         if( !p_osd->p_state->p_visible->b_range )
  368.             p_osd->p_state->p_visible->p_current_state =
  369.                 osd_StateChange( p_osd->p_state->p_visible, OSD_BUTTON_SELECT );
  370.         osd_UpdateState( p_osd->p_state,
  371.                 p_osd->p_state->p_visible->i_x, p_osd->p_state->p_visible->i_y,
  372.                 p_osd->p_state->p_visible->p_current_state->i_width,
  373.                 p_osd->p_state->p_visible->p_current_state->i_height,
  374.                 p_osd->p_state->p_visible->p_current_state->p_pic );
  375.         osd_SetMenuUpdate( p_osd, true );
  376.     }
  377. #if defined(OSD_MENU_DEBUG)
  378.     msg_Dbg( p_osd, "direction right [button %s]", p_osd->p_state->p_visible->psz_action );
  379. #endif
  380.     vlc_mutex_unlock( p_lock );
  381. }
  382. void __osd_MenuPrev( vlc_object_t *p_this )
  383. {
  384.     osd_menu_t *p_osd;
  385.     osd_button_t *p_button = NULL;
  386.     vlc_mutex_t *p_lock = osd_GetMutex( p_this );
  387.     vlc_mutex_lock( p_lock );
  388.     p_osd = osd_Find( p_this );
  389.     if( p_osd == NULL || !osd_isVisible( p_osd ) )
  390.     {
  391.         vlc_mutex_unlock( p_lock );
  392.         msg_Err( p_this, "osd_MenuPrev failed" );
  393.         return;
  394.     }
  395.     p_button = p_osd->p_state->p_visible;
  396.     if( p_button )
  397.     {
  398.         if( !p_button->b_range )
  399.             p_button->p_current_state = osd_StateChange( p_button, OSD_BUTTON_UNSELECT );
  400.         if( p_button->p_prev )
  401.             p_osd->p_state->p_visible = p_button->p_prev;
  402.         else
  403.             p_osd->p_state->p_visible = p_osd->p_last_button;
  404.         if( !p_osd->p_state->p_visible->b_range )
  405.             p_osd->p_state->p_visible->p_current_state =
  406.                 osd_StateChange( p_osd->p_state->p_visible, OSD_BUTTON_SELECT );
  407.         osd_UpdateState( p_osd->p_state,
  408.                 p_osd->p_state->p_visible->i_x, p_osd->p_state->p_visible->i_y,
  409.                 p_osd->p_state->p_visible->p_current_state->i_width,
  410.                 p_osd->p_state->p_visible->p_current_state->i_height,
  411.                 p_osd->p_state->p_visible->p_current_state->p_pic );
  412.         osd_SetMenuUpdate( p_osd, true );
  413.     }
  414. #if defined(OSD_MENU_DEBUG)
  415.     msg_Dbg( p_osd, "direction left [button %s]", p_osd->p_state->p_visible->psz_action );
  416. #endif
  417.     vlc_mutex_unlock( p_lock );
  418. }
  419. void __osd_MenuUp( vlc_object_t *p_this )
  420. {
  421.     osd_menu_t *p_osd;
  422.     osd_button_t *p_button = NULL;
  423. #if defined(OSD_MENU_DEBUG)
  424.     vlc_value_t val;
  425. #endif
  426.     vlc_mutex_t *p_lock = osd_GetMutex( p_this );
  427.     vlc_mutex_lock( p_lock );
  428.     p_osd = osd_Find( p_this );
  429.     if( p_osd == NULL || !osd_isVisible( p_osd ) )
  430.     {
  431.         vlc_mutex_unlock( p_lock );
  432.         msg_Err( p_this, "osd_MenuActivate failed" );
  433.         return;
  434.     }
  435.     p_button = p_osd->p_state->p_visible;
  436.     if( p_button )
  437.     {
  438.         if( !p_button->b_range )
  439.         {
  440.             p_button->p_current_state = osd_StateChange( p_button, OSD_BUTTON_SELECT );
  441.             if( p_button->p_up )
  442.                 p_osd->p_state->p_visible = p_button->p_up;
  443.         }
  444.         if( p_button->b_range && p_osd->p_state->p_visible->b_range )
  445.         {
  446.             osd_state_t *p_temp = p_osd->p_state->p_visible->p_current_state;
  447.             if( p_temp && p_temp->p_next )
  448.                 p_osd->p_state->p_visible->p_current_state = p_temp->p_next;
  449.         }
  450.         else if( !p_osd->p_state->p_visible->b_range )
  451.         {
  452.             p_osd->p_state->p_visible->p_current_state =
  453.                 osd_StateChange( p_osd->p_state->p_visible, OSD_BUTTON_SELECT );
  454.         }
  455.         osd_UpdateState( p_osd->p_state,
  456.                 p_osd->p_state->p_visible->i_x, p_osd->p_state->p_visible->i_y,
  457.                 p_osd->p_state->p_visible->p_current_state->i_width,
  458.                 p_osd->p_state->p_visible->p_current_state->i_height,
  459.                 p_osd->p_state->p_visible->p_current_state->p_pic );
  460.         osd_SetMenuUpdate( p_osd, true );
  461.         /* If this is a range style action with associated images of only one state,
  462.             * then perform "menu select" on every menu navigation
  463.             */
  464.         if( p_button->b_range )
  465.         {
  466.             osd_SetKeyPressed( VLC_OBJECT(p_osd->p_libvlc), config_GetInt(p_osd, p_button->psz_action) );
  467. #if defined(OSD_MENU_DEBUG)
  468.             msg_Dbg( p_osd, "select (%d, %s)", val.i_int, p_button->psz_action );
  469. #endif
  470.         }
  471.     }
  472. #if defined(OSD_MENU_DEBUG)
  473.     msg_Dbg( p_osd, "direction up [button %s]", p_osd->p_state->p_visible->psz_action );
  474. #endif
  475.     vlc_mutex_unlock( p_lock );
  476. }
  477. void __osd_MenuDown( vlc_object_t *p_this )
  478. {
  479.     osd_menu_t *p_osd;
  480.     osd_button_t *p_button = NULL;
  481. #if defined(OSD_MENU_DEBUG)
  482.     vlc_value_t val;
  483. #endif
  484.     vlc_mutex_t *p_lock = osd_GetMutex( p_this );
  485.     vlc_mutex_lock( p_lock );
  486.     p_osd = osd_Find( p_this );
  487.     if( p_osd == NULL || !osd_isVisible( p_osd ) )
  488.     {
  489.         vlc_mutex_unlock( p_lock );
  490.         msg_Err( p_this, "osd_MenuActivate failed" );
  491.         return;
  492.     }
  493.     p_button = p_osd->p_state->p_visible;
  494.     if( p_button )
  495.     {
  496.         if( !p_button->b_range )
  497.         {
  498.             p_button->p_current_state = osd_StateChange( p_button, OSD_BUTTON_SELECT );
  499.             if( p_button->p_down )
  500.                 p_osd->p_state->p_visible = p_button->p_down;
  501.         }
  502.         if( p_button->b_range && p_osd->p_state->p_visible->b_range )
  503.         {
  504.             osd_state_t *p_temp = p_osd->p_state->p_visible->p_current_state;
  505.             if( p_temp && p_temp->p_prev )
  506.                 p_osd->p_state->p_visible->p_current_state = p_temp->p_prev;
  507.         }
  508.         else if( !p_osd->p_state->p_visible->b_range )
  509.         {
  510.             p_osd->p_state->p_visible->p_current_state =
  511.                 osd_StateChange( p_osd->p_state->p_visible, OSD_BUTTON_SELECT );
  512.         }
  513.         osd_UpdateState( p_osd->p_state,
  514.                 p_osd->p_state->p_visible->i_x, p_osd->p_state->p_visible->i_y,
  515.                 p_osd->p_state->p_visible->p_current_state->i_width,
  516.                 p_osd->p_state->p_visible->p_current_state->i_height,
  517.                 p_osd->p_state->p_visible->p_current_state->p_pic );
  518.         osd_SetMenuUpdate( p_osd, true );
  519.         /* If this is a range style action with associated images of only one state,
  520.          * then perform "menu select" on every menu navigation
  521.          */
  522.         if( p_button->b_range )
  523.         {
  524.             osd_SetKeyPressed( VLC_OBJECT(p_osd->p_libvlc), config_GetInt(p_osd, p_button->psz_action_down) );
  525. #if defined(OSD_MENU_DEBUG)
  526.             msg_Dbg( p_osd, "select (%d, %s)", val.i_int, p_button->psz_action_down );
  527. #endif
  528.         }
  529.     }
  530. #if defined(OSD_MENU_DEBUG)
  531.     msg_Dbg( p_osd, "direction down [button %s]", p_osd->p_state->p_visible->psz_action );
  532. #endif
  533.     vlc_mutex_unlock( p_lock );
  534. }
  535. static int osd_VolumeStep( vlc_object_t *p_this, int i_volume, int i_steps )
  536. {
  537.     int i_volume_step = 0;
  538.     (void)i_steps;
  539.     i_volume_step = config_GetInt( p_this->p_libvlc, "volume-step" );
  540.     return (i_volume/i_volume_step);
  541. }
  542. /**
  543.  * Display current audio volume bitmap
  544.  *
  545.  * The OSD Menu audio volume bar is updated to reflect the new audio volume. Call this function
  546.  * when the audio volume is updated outside the OSD menu command "menu up", "menu down" or "menu select".
  547.  */
  548. void __osd_Volume( vlc_object_t *p_this )
  549. {
  550.     osd_menu_t *p_osd;
  551.     osd_button_t *p_button = NULL;
  552.     vlc_mutex_t *p_lock = osd_GetMutex( p_this );
  553.     int i_volume = 0;
  554.     int i_steps = 0;
  555.     vlc_mutex_lock( p_lock );
  556.     p_osd = osd_Find( p_this );
  557.     if( p_osd == NULL )
  558.     {
  559.         vlc_mutex_unlock( p_lock );
  560.         msg_Err( p_this, "OSD menu volume update failed" );
  561.         return;
  562.     }
  563.     if( p_osd->p_state && p_osd->p_state->p_volume )
  564.     {
  565.         p_button = p_osd->p_state->p_volume;
  566.         if( p_osd->p_state->p_volume )
  567.             p_osd->p_state->p_visible = p_osd->p_state->p_volume;
  568.         if( p_button && p_button->b_range )
  569.         {
  570.             /* Update the volume state images to match the current volume */
  571.             i_volume = config_GetInt( p_this, "volume" );
  572.             i_steps = osd_VolumeStep( p_this, i_volume, p_button->i_ranges );
  573.             p_button->p_current_state = osd_VolumeStateChange( p_button->p_states, i_steps );
  574.             osd_UpdateState( p_osd->p_state,
  575.                     p_button->i_x, p_button->i_y,
  576.                     p_button->p_current_state->i_width,
  577.                     p_button->p_current_state->i_height,
  578.                     p_button->p_current_state->p_pic );
  579.             osd_SetMenuUpdate( p_osd, true );
  580.             osd_SetMenuVisible( p_osd, true );
  581.         }
  582.     }
  583.     vlc_mutex_unlock( p_lock );
  584. }
  585. osd_button_t *__osd_ButtonFind( vlc_object_t *p_this, int i_x, int i_y,
  586.     int i_window_height, int i_window_width,
  587.     int i_scale_width, int i_scale_height )
  588. {
  589.     osd_menu_t *p_osd;
  590.     osd_button_t *p_button;
  591.     vlc_mutex_t *p_lock = osd_GetMutex( p_this );
  592.     vlc_mutex_lock( p_lock );
  593.     p_osd = osd_Find( p_this );
  594.     if( p_osd == NULL || !osd_isVisible( p_osd ) )
  595.     {
  596.         vlc_mutex_unlock( p_lock );
  597.         msg_Err( p_this, "osd_ButtonFind failed" );
  598.         return NULL;
  599.     }
  600.     p_button = p_osd->p_button;
  601.     for( ; p_button != NULL; p_button = p_button->p_next )
  602.     {
  603.         int i_source_video_width  = ( i_window_width  * 1000 ) / i_scale_width;
  604.         int i_source_video_height = ( i_window_height * 1000 ) / i_scale_height;
  605.         int i_y_offset = p_button->i_y;
  606.         int i_x_offset = p_button->i_x;
  607.         int i_width = p_button->i_width;
  608.         int i_height = p_button->i_height;
  609.         if( p_osd->i_position > 0 )
  610.         {
  611.             int i_inv_scale_y = i_source_video_height;
  612.             int i_inv_scale_x = i_source_video_width;
  613.             int pi_x = 0;
  614.             if( p_osd->i_position & SUBPICTURE_ALIGN_BOTTOM )
  615.             {
  616.                 i_y_offset = i_window_height - p_button->i_height -
  617.                     (p_osd->i_y + p_button->i_y) * i_inv_scale_y / 1000;
  618.             }
  619.             else if ( !(p_osd->i_position & SUBPICTURE_ALIGN_TOP) )
  620.             {
  621.                 i_y_offset = i_window_height / 2 - p_button->i_height / 2;
  622.             }
  623.             if( p_osd->i_position & SUBPICTURE_ALIGN_RIGHT )
  624.             {
  625.                 i_x_offset = i_window_width - p_button->i_width -
  626.                     (pi_x + p_button->i_x)
  627.                     * i_inv_scale_x / 1000;
  628.             }
  629.             else if ( !(p_osd->i_position & SUBPICTURE_ALIGN_LEFT) )
  630.             {
  631.                 i_x_offset = i_window_width / 2 - p_button->i_width / 2;
  632.             }
  633.             i_width = i_window_width - p_button->i_width - i_inv_scale_x / 1000;
  634.             i_height = i_window_height - p_button->i_height - i_inv_scale_y / 1000;
  635.         }
  636.         // TODO: write for Up / Down case too.
  637.         // TODO: handle absolute positioning case
  638.         if( ( i_x >= i_x_offset ) && ( i_x <= i_x_offset + i_width ) &&
  639.             ( i_y >= i_y_offset ) && ( i_y <= i_y_offset + i_height ) )
  640.         {
  641.             vlc_mutex_unlock( p_lock );
  642.             return p_button;
  643.         }
  644.     }
  645.     vlc_mutex_unlock( p_lock );
  646.     return NULL;
  647. }
  648. /**
  649.  * Select the button provided as the new active button
  650.  */
  651. void __osd_ButtonSelect( vlc_object_t *p_this, osd_button_t *p_button )
  652. {
  653.     osd_menu_t *p_osd;
  654.     osd_button_t *p_old;
  655.     vlc_mutex_t *p_lock = osd_GetMutex( p_this );
  656.     vlc_mutex_lock( p_lock );
  657.     p_osd = osd_Find( p_this );
  658.     if( p_osd == NULL || !osd_isVisible( p_osd ) )
  659.     {
  660.         vlc_mutex_unlock( p_lock );
  661.         msg_Err( p_this, "osd_ButtonSelect failed" );
  662.         return;
  663.     }
  664.     p_old = p_osd->p_state->p_visible;
  665.     if( p_old )
  666.     {
  667.         if( !p_old->b_range )
  668.             p_old->p_current_state = osd_StateChange( p_old, OSD_BUTTON_UNSELECT );
  669.         p_osd->p_state->p_visible = p_button;
  670.         if( !p_osd->p_state->p_visible->b_range )
  671.             p_osd->p_state->p_visible->p_current_state =
  672.                 osd_StateChange( p_osd->p_state->p_visible, OSD_BUTTON_SELECT );
  673.         osd_UpdateState( p_osd->p_state,
  674.                 p_osd->p_state->p_visible->i_x, p_osd->p_state->p_visible->i_y,
  675.                 p_osd->p_state->p_visible->p_current_state->i_width,
  676.                 p_osd->p_state->p_visible->p_current_state->i_height,
  677.                 p_osd->p_state->p_visible->p_current_state->p_pic );
  678.         osd_SetMenuUpdate( p_osd, true );
  679.     }
  680. #if defined(OSD_MENU_DEBUG)
  681.     msg_Dbg( p_osd, "button selected is [button %s]", p_osd->p_state->p_visible->psz_action );
  682. #endif
  683.     vlc_mutex_unlock( p_lock );
  684. }