1 /*****************************************************************************
2 * menu.c : functions to handle menu items.
3 *****************************************************************************
4 * Copyright (C) 2000, 2001 VideoLAN
5 * $Id: menu.c,v 1.5 2003/01/22 00:32:32 fenrir Exp $
7 * Authors: Samuel Hocevar <sam@zoy.org>
8 * Stéphane Borel <stef@via.ecp.fr>
9 * Johan Bilien <jobi@via.ecp.fr>
10 * Laurent Aimar <fenrir@via.ecp.fr>
12 * This program is free software; you can redistribute it and/or modify
13 * it under the terms of the GNU General Public License as published by
14 * the Free Software Foundation; either version 2 of the License, or
15 * (at your option) any later version.
17 * This program is distributed in the hope that it will be useful,
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 * GNU General Public License for more details.
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, write to the Free Software
24 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111, USA.
25 *****************************************************************************/
27 /*****************************************************************************
29 *****************************************************************************/
30 #include <sys/types.h> /* off_t */
38 #ifdef MODULE_NAME_IS_gnome
46 #include "gtk_callbacks.h"
47 #include "gtk_interface.h"
48 #include "gtk_support.h"
56 static gint GtkLanguageMenus( gpointer , GtkWidget *, es_descriptor_t *, gint,
57 void(*pf_toggle )( GtkCheckMenuItem *, gpointer ) );
59 void GtkMenubarAudioToggle ( GtkCheckMenuItem *, gpointer );
60 void GtkPopupAudioToggle ( GtkCheckMenuItem *, gpointer );
61 void GtkMenubarSubtitleToggle( GtkCheckMenuItem *, gpointer );
62 void GtkPopupSubtitleToggle ( GtkCheckMenuItem *, gpointer );
63 static gint GtkTitleMenu( gpointer, GtkWidget *,
64 void(*pf_toggle )( GtkCheckMenuItem *, gpointer ) );
65 static gint GtkRadioMenu( intf_thread_t *, GtkWidget *, GSList *,
66 char *, int, int, int,
67 void( *pf_toggle )( GtkCheckMenuItem *, gpointer ) );
69 static void GtkMenubarDeinterlaceToggle( GtkCheckMenuItem * menuitem, gpointer user_data );
70 static void GtkPopupDeinterlaceToggle( GtkCheckMenuItem * menuitem, gpointer user_data );
71 static gint GtkDeinterlaceMenus( gpointer p_data,
73 void(*pf_toggle )( GtkCheckMenuItem *, gpointer ) );
75 gint GtkSetupMenus( intf_thread_t * p_intf );
77 /****************************************************************************
78 * Gtk*Toggle: callbacks to toggle the value of a checkmenuitem
79 ****************************************************************************
80 * We need separate functions for menubar and popup here since we can't use
81 * user_data to transmit intf_* and we need to refresh the other menu.
82 ****************************************************************************/
84 #define GTKLANGTOGGLE( window, menu, type, callback, b_update ) \
85 intf_thread_t * p_intf; \
87 es_descriptor_t * p_es; \
89 p_intf = GtkGetIntf( menuitem ); \
91 if( !p_intf->p_sys->b_update ) \
93 p_menu = GTK_WIDGET( gtk_object_get_data( \
94 GTK_OBJECT( p_intf->p_sys->window ), (menu) ) ); \
95 p_es = (es_descriptor_t*)user_data; \
97 input_ToggleES( p_intf->p_sys->p_input, \
98 p_es, menuitem->active ); \
100 p_intf->p_sys->b_update = menuitem->active; \
102 if( p_intf->p_sys->b_update ) \
104 GtkLanguageMenus( p_intf, p_menu, p_es, type, callback ); \
107 p_intf->p_sys->b_update = VLC_FALSE; \
114 void GtkMenubarAudioToggle( GtkCheckMenuItem * menuitem, gpointer user_data )
116 GTKLANGTOGGLE( p_popup, "popup_audio", AUDIO_ES,
117 GtkPopupAudioToggle, b_audio_update );
120 void GtkPopupAudioToggle( GtkCheckMenuItem * menuitem, gpointer user_data )
122 GTKLANGTOGGLE( p_window, "menubar_audio", AUDIO_ES,
123 GtkMenubarAudioToggle, b_audio_update );
130 void GtkMenubarSubtitleToggle( GtkCheckMenuItem * menuitem, gpointer user_data )
132 GTKLANGTOGGLE( p_popup, "popup_subpictures", SPU_ES,
133 GtkPopupSubtitleToggle, b_spu_update );
136 void GtkPopupSubtitleToggle( GtkCheckMenuItem * menuitem, gpointer user_data )
138 GTKLANGTOGGLE( p_window, "menubar_subpictures", SPU_ES,
139 GtkMenubarSubtitleToggle, b_spu_update );
148 void GtkPopupNavigationToggle( GtkCheckMenuItem * menuitem,
151 intf_thread_t * p_intf = GtkGetIntf( menuitem );
153 if( menuitem->active &&
154 !p_intf->p_sys->b_title_update &&
155 !p_intf->p_sys->b_chapter_update )
157 input_area_t *p_area;
159 guint i_title = DATA2TITLE( user_data );
160 guint i_chapter = DATA2CHAPTER( user_data );
162 vlc_mutex_lock( &p_intf->p_sys->p_input->stream.stream_lock );
163 p_area = p_intf->p_sys->p_input->stream.p_selected_area;
165 i_title = __MIN( i_title,
166 p_intf->p_sys->p_input->stream.i_area_nb - 1 );
167 i_title = __MAX( i_title, 1 );
169 if( p_area != p_intf->p_sys->p_input->stream.pp_areas[i_title] )
171 p_area = p_intf->p_sys->p_input->stream.pp_areas[i_title];
172 p_intf->p_sys->b_title_update = VLC_TRUE;
175 i_chapter = __MIN( i_chapter, p_area->i_part_nb - 1 );
176 i_chapter = __MAX( i_chapter, 1 );
177 p_area->i_part = i_chapter;
179 vlc_mutex_unlock( &p_intf->p_sys->p_input->stream.stream_lock );
181 input_ChangeArea( p_intf->p_sys->p_input, (input_area_t*)p_area );
183 p_intf->p_sys->b_chapter_update = VLC_TRUE;
184 vlc_mutex_lock( &p_intf->p_sys->p_input->stream.stream_lock );
185 GtkSetupMenus( p_intf );
186 vlc_mutex_unlock( &p_intf->p_sys->p_input->stream.stream_lock );
188 input_SetStatus( p_intf->p_sys->p_input, INPUT_STATUS_PLAY );
195 #define GTKPROGRAMTOGGLE( ) \
196 intf_thread_t * p_intf = GtkGetIntf( menuitem ); \
198 if( menuitem->active && !p_intf->p_sys->b_program_update ) \
200 int i_program_id = (ptrdiff_t)user_data; \
202 input_ChangeProgram( p_intf->p_sys->p_input, i_program_id ); \
204 p_intf->p_sys->b_program_update = VLC_TRUE; \
206 vlc_mutex_lock( &p_intf->p_sys->p_input->stream.stream_lock ); \
207 GtkSetupMenus( p_intf ); \
208 vlc_mutex_unlock( &p_intf->p_sys->p_input->stream.stream_lock ); \
210 p_intf->p_sys->b_program_update = VLC_FALSE; \
212 input_SetStatus( p_intf->p_sys->p_input, INPUT_STATUS_PLAY ); \
215 void GtkMenubarProgramToggle( GtkCheckMenuItem * menuitem, gpointer user_data )
220 void GtkPopupProgramToggle( GtkCheckMenuItem * menuitem, gpointer user_data )
229 void GtkMenubarTitleToggle( GtkCheckMenuItem * menuitem, gpointer user_data )
231 intf_thread_t * p_intf = GtkGetIntf( menuitem );
233 if( menuitem->active && !p_intf->p_sys->b_title_update )
235 guint i_title = (ptrdiff_t)user_data;
237 vlc_mutex_lock( &p_intf->p_sys->p_input->stream.stream_lock );
238 i_title = __MIN( i_title,
239 p_intf->p_sys->p_input->stream.i_area_nb - 1 );
240 i_title = __MAX( i_title, 1 );
241 vlc_mutex_unlock( &p_intf->p_sys->p_input->stream.stream_lock );
243 input_ChangeArea( p_intf->p_sys->p_input,
244 p_intf->p_sys->p_input->stream.pp_areas[i_title] );
246 p_intf->p_sys->b_title_update = VLC_TRUE;
247 vlc_mutex_lock( &p_intf->p_sys->p_input->stream.stream_lock );
248 GtkSetupMenus( p_intf );
249 vlc_mutex_unlock( &p_intf->p_sys->p_input->stream.stream_lock );
250 p_intf->p_sys->b_title_update = VLC_FALSE;
252 input_SetStatus( p_intf->p_sys->p_input, INPUT_STATUS_PLAY );
260 void GtkMenubarChapterToggle( GtkCheckMenuItem * menuitem, gpointer user_data )
262 intf_thread_t * p_intf;
263 input_area_t * p_area;
265 GtkWidget * p_popup_menu;
267 p_intf = GtkGetIntf( menuitem );
268 p_area = p_intf->p_sys->p_input->stream.p_selected_area;
269 i_chapter = (ptrdiff_t)user_data;
271 if( menuitem->active && !p_intf->p_sys->b_chapter_update )
273 vlc_mutex_lock( &p_intf->p_sys->p_input->stream.stream_lock );
274 i_chapter = __MIN( i_chapter, p_area->i_part_nb - 1 );
275 i_chapter = __MAX( i_chapter, 1 );
276 p_area->i_part = i_chapter;
277 vlc_mutex_unlock( &p_intf->p_sys->p_input->stream.stream_lock );
279 input_ChangeArea( p_intf->p_sys->p_input, (input_area_t*)p_area );
281 p_intf->p_sys->b_chapter_update = VLC_TRUE;
282 p_popup_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
283 p_intf->p_sys->p_popup ), "popup_navigation" ) );
285 vlc_mutex_lock( &p_intf->p_sys->p_input->stream.stream_lock );
286 GtkTitleMenu( p_intf, p_popup_menu, GtkPopupNavigationToggle );
287 vlc_mutex_unlock( &p_intf->p_sys->p_input->stream.stream_lock );
289 p_intf->p_sys->b_chapter_update = VLC_FALSE;
291 input_SetStatus( p_intf->p_sys->p_input, INPUT_STATUS_PLAY );
296 static void GtkPopupObjectToggle( GtkCheckMenuItem * menuitem, gpointer user_data,
297 int i_object_type, char *psz_variable )
299 intf_thread_t *p_intf = GtkGetIntf( menuitem );
302 p_label = GTK_LABEL( ( GTK_BIN( menuitem )->child ) );
304 if( menuitem->active && !p_intf->p_sys->b_aout_update && !p_intf->p_sys->b_vout_update )
306 vlc_object_t * p_obj;
308 p_obj = (vlc_object_t *)vlc_object_find( p_intf, i_object_type,
314 gtk_label_get( p_label, &val.psz_string );
316 if( var_Set( p_obj, psz_variable, val ) < 0 )
318 msg_Warn( p_obj, "cannot set variable (%s)", val.psz_string );
320 vlc_object_release( p_obj );
324 static void GtkPopupAoutChannelsToggle( GtkCheckMenuItem * menuitem, gpointer user_data )
326 GtkPopupObjectToggle( menuitem, user_data, VLC_OBJECT_AOUT, "audio-channels" );
329 static void GtkPopupAoutDeviceToggle( GtkCheckMenuItem * menuitem, gpointer user_data )
331 GtkPopupObjectToggle( menuitem, user_data, VLC_OBJECT_AOUT, "audio-device" );
335 static void GtkPopupVoutDeviceToggle( GtkCheckMenuItem * menuitem, gpointer user_data )
337 GtkPopupObjectToggle( menuitem, user_data, VLC_OBJECT_VOUT, "video-device" );
341 static void GtkDeinterlaceUpdate( intf_thread_t *p_intf, char *psz_mode )
346 psz_filter = config_GetPsz( p_intf, "filter" );
348 if( !strcmp( psz_mode, "None" ) )
350 config_PutPsz( p_intf, "filter", "" );
354 if( !psz_filter || !*psz_filter )
356 config_PutPsz( p_intf, "filter", "deinterlace" );
360 if( strstr( psz_filter, "deinterlace" ) == NULL )
362 psz_filter = realloc( psz_filter, strlen( psz_filter ) + 20 );
363 strcat( psz_filter, ",deinterlace" );
365 config_PutPsz( p_intf, "filter", psz_filter );
368 config_PutPsz( p_intf, "deinterlace-mode", psz_mode );
374 /* now restart all video stream */
375 if( p_intf->p_sys->p_input )
377 vlc_mutex_lock( &p_intf->p_sys->p_input->stream.stream_lock );
378 #define ES p_intf->p_sys->p_input->stream.pp_es[i]
379 /* create a set of language buttons and append them to the container */
380 for( i = 0 ; i < p_intf->p_sys->p_input->stream.i_es_number ; i++ )
382 if( ( ES->i_cat == VIDEO_ES ) &&
383 ES->p_decoder_fifo != NULL )
385 input_UnselectES( p_intf->p_sys->p_input, ES );
386 input_SelectES( p_intf->p_sys->p_input, ES );
390 vlc_mutex_unlock( &p_intf->p_sys->p_input->stream.stream_lock );
394 static void GtkMenubarDeinterlaceToggle( GtkCheckMenuItem * menuitem, gpointer user_data )
396 intf_thread_t *p_intf = GtkGetIntf( menuitem );
399 GtkWidget *p_popup_menu;
401 p_label = GTK_LABEL( ( GTK_BIN( menuitem )->child ) );
403 if( !p_intf->p_sys->b_deinterlace_update && menuitem->active )
405 gtk_label_get( p_label, &psz_mode );
406 GtkDeinterlaceUpdate( p_intf, psz_mode );
408 p_intf->p_sys->b_deinterlace_update = VLC_TRUE;
410 p_popup_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
411 p_intf->p_sys->p_popup ), "popup_deinterlace" ) );
413 GtkDeinterlaceMenus( p_intf, p_popup_menu, GtkPopupDeinterlaceToggle );
415 p_intf->p_sys->b_deinterlace_update = VLC_FALSE;
420 static void GtkPopupDeinterlaceToggle( GtkCheckMenuItem * menuitem, gpointer user_data )
422 intf_thread_t *p_intf = GtkGetIntf( menuitem );
425 GtkWidget *p_menubar_menu;
427 p_label = GTK_LABEL( ( GTK_BIN( menuitem )->child ) );
429 if( !p_intf->p_sys->b_deinterlace_update && menuitem->active )
431 gtk_label_get( p_label, &psz_mode );
432 GtkDeinterlaceUpdate( p_intf, psz_mode );
434 p_intf->p_sys->b_deinterlace_update = VLC_TRUE;
436 p_menubar_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
437 p_intf->p_sys->p_window ), "menubar_deinterlace" ) );
439 GtkDeinterlaceMenus( p_intf, p_menubar_menu, GtkMenubarDeinterlaceToggle );
441 p_intf->p_sys->b_deinterlace_update = VLC_FALSE;
445 /****************************************************************************
446 * Functions to generate menus
447 ****************************************************************************/
449 /*****************************************************************************
450 * GtkRadioMenu: update interactive menus of the interface
451 *****************************************************************************
452 * Sets up menus with information from input
453 * Warning: since this function is designed to be called by management
454 * function, the interface lock has to be taken
455 *****************************************************************************/
456 static gint GtkRadioMenu( intf_thread_t * p_intf,
457 GtkWidget * p_root, GSList * p_menu_group,
458 char * psz_item_name,
459 int i_start, int i_end, int i_selected,
460 void( *pf_toggle )( GtkCheckMenuItem *, gpointer ) )
462 char psz_name[ GTK_MENU_LABEL_SIZE ];
464 GtkWidget * p_submenu;
465 GtkWidget * p_item_group;
467 GtkWidget * p_item_selected;
471 /* temporary hack to avoid blank menu when an open menu is removed */
472 if( GTK_MENU_ITEM(p_root)->submenu != NULL )
474 gtk_menu_popdown( GTK_MENU( GTK_MENU_ITEM(p_root)->submenu ) );
476 /* removes previous menu */
477 gtk_menu_item_remove_submenu( GTK_MENU_ITEM( p_root ) );
478 gtk_widget_set_sensitive( p_root, FALSE );
482 p_item_selected = NULL;
483 p_group = p_menu_group;
485 p_menu = gtk_menu_new();
486 gtk_object_set_data( GTK_OBJECT( p_menu ), "p_intf", p_intf );
488 for( i_item = i_start ; i_item <= i_end ; i_item++ )
490 /* we group chapters in packets of ten for small screens */
491 if( ( i_item % 10 == i_start ) && ( i_end > i_start + 20 ) )
493 if( i_item != i_start )
495 gtk_menu_item_set_submenu( GTK_MENU_ITEM( p_item_group ),
497 gtk_menu_append( GTK_MENU( p_menu ), p_item_group );
500 snprintf( psz_name, GTK_MENU_LABEL_SIZE,
501 "%ss %d to %d", psz_item_name, i_item, i_item + 9 );
502 psz_name[ GTK_MENU_LABEL_SIZE - 1 ] = '\0';
503 p_item_group = gtk_menu_item_new_with_label( psz_name );
504 gtk_widget_show( p_item_group );
505 p_submenu = gtk_menu_new();
506 gtk_object_set_data( GTK_OBJECT( p_submenu ), "p_intf", p_intf );
509 snprintf( psz_name, GTK_MENU_LABEL_SIZE, "%s %d",
510 psz_item_name, i_item );
511 psz_name[ GTK_MENU_LABEL_SIZE - 1 ] = '\0';
513 p_item = gtk_radio_menu_item_new_with_label( p_group, psz_name );
514 p_group = gtk_radio_menu_item_group( GTK_RADIO_MENU_ITEM( p_item ) );
516 if( i_selected == i_item )
518 p_item_selected = p_item;
521 gtk_widget_show( p_item );
523 /* setup signal hanling */
524 gtk_signal_connect( GTK_OBJECT( p_item ),
526 GTK_SIGNAL_FUNC( pf_toggle ),
527 (gpointer)((long)(i_item)) );
529 if( i_end > i_start + 20 )
531 gtk_menu_append( GTK_MENU( p_submenu ), p_item );
535 gtk_menu_append( GTK_MENU( p_menu ), p_item );
539 if( i_end > i_start + 20 )
541 gtk_menu_item_set_submenu( GTK_MENU_ITEM( p_item_group ), p_submenu );
542 gtk_menu_append( GTK_MENU( p_menu ), p_item_group );
545 /* link the new menu to the title menu item */
546 gtk_menu_item_set_submenu( GTK_MENU_ITEM( p_root ), p_menu );
548 /* toggle currently selected chapter
549 * We have to release the lock since input_ToggleES needs it */
550 if( p_item_selected != NULL )
552 gtk_check_menu_item_set_active( GTK_CHECK_MENU_ITEM( p_item_selected ),
556 /* be sure that menu is sensitive, if there are several items */
557 if( i_end > i_start )
559 gtk_widget_set_sensitive( p_root, TRUE );
565 /*****************************************************************************
566 * GtkProgramMenu: update the programs menu of the interface
567 *****************************************************************************
568 * Builds the program menu according to what have been found in the PAT
569 * by the input. Usefull for multi-programs streams such as DVB ones.
570 *****************************************************************************/
571 static gint GtkProgramMenu( gpointer p_data,
573 pgrm_descriptor_t * p_pgrm,
574 void(*pf_toggle )( GtkCheckMenuItem *, gpointer ) )
576 intf_thread_t * p_intf;
579 GtkWidget * p_item_active;
581 char psz_name[ GTK_MENU_LABEL_SIZE ];
585 p_intf = (intf_thread_t *)p_data;
587 /* temporary hack to avoid blank menu when an open menu is removed */
588 if( GTK_MENU_ITEM(p_root)->submenu != NULL )
590 gtk_menu_popdown( GTK_MENU( GTK_MENU_ITEM(p_root)->submenu ) );
592 /* removes previous menu */
593 gtk_menu_item_remove_submenu( GTK_MENU_ITEM( p_root ) );
594 gtk_widget_set_sensitive( p_root, FALSE );
599 p_menu = gtk_menu_new();
600 gtk_object_set_data( GTK_OBJECT( p_menu ), "p_intf", p_intf );
602 p_item_active = NULL;
604 /* create a set of program buttons and append them to the container */
605 for( i = 0 ; i < p_intf->p_sys->p_input->stream.i_pgrm_number ; i++ )
607 snprintf( psz_name, GTK_MENU_LABEL_SIZE, "id %d",
608 p_intf->p_sys->p_input->stream.pp_programs[i]->i_number );
609 psz_name[GTK_MENU_LABEL_SIZE-1] = '\0';
611 p_item = gtk_radio_menu_item_new_with_label( p_group, psz_name );
613 gtk_radio_menu_item_group( GTK_RADIO_MENU_ITEM( p_item ) );
615 if( p_pgrm == p_intf->p_sys->p_input->stream.pp_programs[i] )
617 /* don't lose p_item when we append into menu */
618 p_item_active = p_item;
621 gtk_widget_show( p_item );
623 /* setup signal hanling */
624 gtk_signal_connect( GTK_OBJECT( p_item ), "toggled",
625 GTK_SIGNAL_FUNC( pf_toggle ),
626 (gpointer)(ptrdiff_t)( p_intf->p_sys->p_input->
627 stream.pp_programs[i]->i_number ) );
629 gtk_menu_append( GTK_MENU( p_menu ), p_item );
632 /* link the new menu to the menubar item */
633 gtk_menu_item_set_submenu( GTK_MENU_ITEM( p_root ), p_menu );
635 /* activation will call signals so we can only do it
636 * when submenu is attached to menu - to get intf_window
637 * We have to release the lock since input_ToggleES needs it */
638 if( p_item_active != NULL )
640 gtk_check_menu_item_set_active( GTK_CHECK_MENU_ITEM( p_item_active ),
644 /* be sure that menu is sensitive if more than 1 program */
645 if( p_intf->p_sys->p_input->stream.i_pgrm_number > 1 )
647 gtk_widget_set_sensitive( p_root, TRUE );
653 /*****************************************************************************
654 * GtkLanguageMenus: update interactive menus of the interface
655 *****************************************************************************
656 * Sets up menus with information from input:
659 * Warning: since this function is designed to be called by management
660 * function, the interface lock has to be taken
661 *****************************************************************************/
662 static gint GtkLanguageMenus( gpointer p_data,
664 es_descriptor_t * p_es,
666 void(*pf_toggle )( GtkCheckMenuItem *, gpointer ) )
668 intf_thread_t * p_intf;
670 GtkWidget * p_separator;
672 GtkWidget * p_item_active;
674 char psz_name[ GTK_MENU_LABEL_SIZE ];
678 p_intf = (intf_thread_t *)p_data;
680 /* temporary hack to avoid blank menu when an open menu is removed */
681 if( GTK_MENU_ITEM(p_root)->submenu != NULL )
683 gtk_menu_popdown( GTK_MENU( GTK_MENU_ITEM(p_root)->submenu ) );
685 /* removes previous menu */
686 gtk_menu_item_remove_submenu( GTK_MENU_ITEM( p_root ) );
687 gtk_widget_set_sensitive( p_root, FALSE );
692 p_menu = gtk_menu_new();
693 gtk_object_set_data( GTK_OBJECT( p_menu ), "p_intf", p_intf );
695 /* special case for "off" item */
696 snprintf( psz_name, GTK_MENU_LABEL_SIZE, _("None") );
697 psz_name[ GTK_MENU_LABEL_SIZE - 1 ] = '\0';
699 p_item = gtk_radio_menu_item_new_with_label( p_group, psz_name );
700 p_group = gtk_radio_menu_item_group( GTK_RADIO_MENU_ITEM( p_item ) );
702 gtk_widget_show( p_item );
704 /* signal hanling for off */
705 gtk_signal_connect( GTK_OBJECT( p_item ), "toggled",
706 GTK_SIGNAL_FUNC ( pf_toggle ), NULL );
708 gtk_menu_append( GTK_MENU( p_menu ), p_item );
710 p_separator = gtk_menu_item_new();
711 gtk_widget_set_sensitive( p_separator, FALSE );
712 gtk_widget_show( p_separator );
713 gtk_menu_append( GTK_MENU( p_menu ), p_separator );
715 p_item_active = NULL;
718 vlc_mutex_lock( &p_intf->p_sys->p_input->stream.stream_lock );
720 #define ES p_intf->p_sys->p_input->stream.pp_es[i]
721 /* create a set of language buttons and append them to the container */
722 for( i = 0 ; i < p_intf->p_sys->p_input->stream.i_es_number ; i++ )
724 if( ( ES->i_cat == i_cat ) &&
727 p_intf->p_sys->p_input->stream.p_selected_program ) )
731 p_intf->p_sys->p_input->stream.pp_es[i]->psz_desc );
732 if( psz_name[0] == '\0' )
734 snprintf( psz_name, GTK_MENU_LABEL_SIZE,
735 "Language %d", i_item );
736 psz_name[ GTK_MENU_LABEL_SIZE - 1 ] = '\0';
739 p_item = gtk_radio_menu_item_new_with_label( p_group, psz_name );
741 gtk_radio_menu_item_group( GTK_RADIO_MENU_ITEM( p_item ) );
743 if( p_es == p_intf->p_sys->p_input->stream.pp_es[i] )
745 /* don't lose p_item when we append into menu */
746 p_item_active = p_item;
749 gtk_widget_show( p_item );
751 /* setup signal hanling */
752 gtk_signal_connect( GTK_OBJECT( p_item ), "toggled",
753 GTK_SIGNAL_FUNC( pf_toggle ),
754 (gpointer)( p_intf->p_sys->p_input->stream.pp_es[i] ) );
756 gtk_menu_append( GTK_MENU( p_menu ), p_item );
760 vlc_mutex_unlock( &p_intf->p_sys->p_input->stream.stream_lock );
762 /* link the new menu to the menubar item */
763 gtk_menu_item_set_submenu( GTK_MENU_ITEM( p_root ), p_menu );
765 /* acitvation will call signals so we can only do it
766 * when submenu is attached to menu - to get intf_window
767 * We have to release the lock since input_ToggleES needs it */
768 if( p_item_active != NULL )
770 gtk_check_menu_item_set_active( GTK_CHECK_MENU_ITEM( p_item_active ),
774 /* be sure that menu is sensitive if non empty */
777 gtk_widget_set_sensitive( p_root, TRUE );
783 /*****************************************************************************
784 * GtkTitleMenu: sets menus for titles and chapters selection
785 *****************************************************************************
786 * Generates two types of menus:
787 * -simple list of titles
788 * -cascaded lists of chapters for each title
789 *****************************************************************************/
790 static gint GtkTitleMenu( gpointer p_data,
791 GtkWidget * p_navigation,
792 void(*pf_toggle )( GtkCheckMenuItem *, gpointer ) )
794 intf_thread_t * p_intf;
795 char psz_name[ GTK_MENU_LABEL_SIZE ];
796 GtkWidget * p_title_menu;
797 GtkWidget * p_title_submenu;
798 GtkWidget * p_title_item;
799 GtkWidget * p_item_active;
800 GtkWidget * p_chapter_menu;
801 GtkWidget * p_chapter_submenu;
802 GtkWidget * p_title_menu_item;
803 GtkWidget * p_chapter_menu_item;
805 GSList * p_title_group;
806 GSList * p_chapter_group;
813 p_intf = (intf_thread_t*)p_data;
815 /* temporary hack to avoid blank menu when an open menu is removed */
816 if( GTK_MENU_ITEM(p_navigation)->submenu != NULL )
818 gtk_menu_popdown( GTK_MENU( GTK_MENU_ITEM(p_navigation)->submenu ) );
820 /* removes previous menu */
821 gtk_menu_item_remove_submenu( GTK_MENU_ITEM( p_navigation ) );
822 gtk_widget_set_sensitive( p_navigation, FALSE );
824 p_title_menu = gtk_menu_new();
825 p_title_group = NULL;
826 p_title_submenu = NULL;
827 p_title_menu_item = NULL;
828 p_chapter_group = NULL;
829 p_chapter_submenu = NULL;
830 p_chapter_menu_item = NULL;
831 p_item_active = NULL;
832 i_title_nb = p_intf->p_sys->p_input->stream.i_area_nb - 1;
834 gtk_object_set_data( GTK_OBJECT( p_title_menu ), "p_intf", p_intf );
837 for( i_title = 1 ; i_title <= i_title_nb ; i_title++ )
839 /* we group titles in packets of ten for small screens */
840 if( ( i_title % 10 == 1 ) && ( i_title_nb > 20 ) )
844 gtk_menu_item_set_submenu( GTK_MENU_ITEM( p_title_menu_item ),
846 gtk_menu_append( GTK_MENU( p_title_menu ), p_title_menu_item );
849 snprintf( psz_name, GTK_MENU_LABEL_SIZE,
850 "%d - %d", i_title, i_title + 9 );
851 psz_name[ GTK_MENU_LABEL_SIZE - 1 ] = '\0';
852 p_title_menu_item = gtk_menu_item_new_with_label( psz_name );
853 gtk_widget_show( p_title_menu_item );
854 p_title_submenu = gtk_menu_new();
855 gtk_object_set_data( GTK_OBJECT( p_title_submenu ),
859 snprintf( psz_name, GTK_MENU_LABEL_SIZE, _("Title %d (%d)"), i_title,
860 p_intf->p_sys->p_input->stream.pp_areas[i_title]->i_part_nb - 1);
861 psz_name[ GTK_MENU_LABEL_SIZE - 1 ] = '\0';
863 if( pf_toggle == on_menubar_title_toggle )
865 p_title_item = gtk_radio_menu_item_new_with_label( p_title_group,
868 gtk_radio_menu_item_group( GTK_RADIO_MENU_ITEM( p_title_item ) );
870 if( p_intf->p_sys->p_input->stream.pp_areas[i_title] ==
871 p_intf->p_sys->p_input->stream.p_selected_area )
873 p_item_active = p_title_item;
876 /* setup signal hanling */
877 gtk_signal_connect( GTK_OBJECT( p_title_item ),
879 GTK_SIGNAL_FUNC( pf_toggle ),
880 (gpointer)(p_intf->p_sys->p_input->stream.pp_areas[i_title]) );
882 if( p_intf->p_sys->p_input->stream.i_area_nb > 1 )
884 /* be sure that menu is sensitive */
885 gtk_widget_set_sensitive( p_navigation, TRUE );
891 p_title_item = gtk_menu_item_new_with_label( psz_name );
894 p_chapter_menu = gtk_menu_new();
895 gtk_object_set_data( GTK_OBJECT( p_chapter_menu ),
898 p_intf->p_sys->p_input->stream.pp_areas[i_title]->i_part_nb - 1;
900 for( i_chapter = 1 ; i_chapter <= i_chapter_nb ; i_chapter++ )
902 /* we group chapters in packets of ten for small screens */
903 if( ( i_chapter % 10 == 1 ) && ( i_chapter_nb > 20 ) )
907 gtk_menu_item_set_submenu(
908 GTK_MENU_ITEM( p_chapter_menu_item ),
910 gtk_menu_append( GTK_MENU( p_chapter_menu ),
911 p_chapter_menu_item );
914 snprintf( psz_name, GTK_MENU_LABEL_SIZE,
915 "%d - %d", i_chapter, i_chapter + 9 );
916 psz_name[ GTK_MENU_LABEL_SIZE - 1 ] = '\0';
917 p_chapter_menu_item =
918 gtk_menu_item_new_with_label( psz_name );
919 gtk_widget_show( p_chapter_menu_item );
920 p_chapter_submenu = gtk_menu_new();
921 gtk_object_set_data( GTK_OBJECT( p_chapter_submenu ),
925 snprintf( psz_name, GTK_MENU_LABEL_SIZE,
926 _("Chapter %d"), i_chapter );
927 psz_name[ GTK_MENU_LABEL_SIZE - 1 ] = '\0';
929 p_item = gtk_radio_menu_item_new_with_label(
930 p_chapter_group, psz_name );
931 p_chapter_group = gtk_radio_menu_item_group(
932 GTK_RADIO_MENU_ITEM( p_item ) );
933 gtk_widget_show( p_item );
935 #define p_area p_intf->p_sys->p_input->stream.pp_areas[i_title]
937 p_intf->p_sys->p_input->stream.p_selected_area ) &&
938 ( p_area->i_part == i_chapter ) )
940 p_item_active = p_item;
944 /* setup signal hanling */
945 gtk_signal_connect( GTK_OBJECT( p_item ),
947 GTK_SIGNAL_FUNC( pf_toggle ),
948 (gpointer)POS2DATA( i_title, i_chapter ) );
950 if( i_chapter_nb > 20 )
952 gtk_menu_append( GTK_MENU( p_chapter_submenu ), p_item );
956 gtk_menu_append( GTK_MENU( p_chapter_menu ), p_item );
960 if( i_chapter_nb > 20 )
962 gtk_menu_item_set_submenu( GTK_MENU_ITEM( p_chapter_menu_item ),
964 gtk_menu_append( GTK_MENU( p_chapter_menu ),
965 p_chapter_menu_item );
968 /* link the new menu to the title menu item */
969 gtk_menu_item_set_submenu( GTK_MENU_ITEM( p_title_item ),
972 if( p_intf->p_sys->p_input->stream.pp_areas[i_title]->i_part_nb > 1 )
974 /* be sure that menu is sensitive */
975 gtk_widget_set_sensitive( p_navigation, TRUE );
978 GtkRadioMenu( p_intf, p_title_item, p_chapter_group, _("Chapter"),
979 p_intf->p_sys->p_input->stream.pp_areas[i_title]->i_part_nb - 1,
981 p_intf->p_sys->p_input->stream.p_selected_area->i_part +
982 p_intf->p_sys->p_input->stream.p_selected_area->i_id *100,
987 gtk_widget_show( p_title_item );
989 if( i_title_nb > 20 )
991 gtk_menu_append( GTK_MENU( p_title_submenu ), p_title_item );
995 gtk_menu_append( GTK_MENU( p_title_menu ), p_title_item );
999 if( i_title_nb > 20 )
1001 gtk_menu_item_set_submenu( GTK_MENU_ITEM( p_title_menu_item ),
1003 gtk_menu_append( GTK_MENU( p_title_menu ), p_title_menu_item );
1006 /* be sure that menu is sensitive */
1007 gtk_widget_set_sensitive( p_title_menu, TRUE );
1009 /* link the new menu to the menubar item */
1010 gtk_menu_item_set_submenu( GTK_MENU_ITEM( p_navigation ), p_title_menu );
1012 /* Default selected chapter
1013 * We have to release the lock since input_ToggleES needs it */
1014 if( p_item_active != NULL )
1016 gtk_check_menu_item_set_active( GTK_CHECK_MENU_ITEM( p_item_active ),
1020 if( p_intf->p_sys->p_input->stream.i_area_nb > 1 )
1022 /* be sure that menu is sensitive */
1023 gtk_widget_set_sensitive( p_navigation, TRUE );
1030 /*****************************************************************************
1032 *****************************************************************************
1034 *****************************************************************************/
1035 static gint GtkSetupVarMenu( intf_thread_t * p_intf,
1036 vlc_object_t * p_object,
1038 char * psz_variable,
1039 void(*pf_toggle )( GtkCheckMenuItem *, gpointer ) )
1044 GSList * p_group = NULL;
1046 GtkWidget * p_item_active = NULL;
1050 /* temporary hack to avoid blank menu when an open menu is removed */
1051 if( GTK_MENU_ITEM(p_root)->submenu != NULL )
1053 gtk_menu_popdown( GTK_MENU( GTK_MENU_ITEM(p_root)->submenu ) );
1055 /* removes previous menu */
1056 gtk_menu_item_remove_submenu( GTK_MENU_ITEM( p_root ) );
1057 gtk_widget_set_sensitive( p_root, FALSE );
1059 /* get the current value */
1060 if( var_Get( p_object, psz_variable, &val ) < 0 )
1064 psz_value = val.psz_string;
1066 if( var_Change( p_object, psz_variable, VLC_VAR_GETLIST, &val ) < 0 )
1072 /* menu container */
1073 p_menu = gtk_menu_new();
1074 gtk_object_set_data( GTK_OBJECT( p_menu ), "p_intf", p_intf );
1076 for( i_item = 0; i_item < val.p_list->i_count; i_item++ )
1078 p_item = gtk_radio_menu_item_new_with_label( p_group,
1079 val.p_list->p_values[i_item].psz_string );
1080 p_group = gtk_radio_menu_item_group( GTK_RADIO_MENU_ITEM( p_item ) );
1082 if( !strcmp( psz_value, val.p_list->p_values[i_item].psz_string ) )
1084 p_item_active = p_item;
1087 gtk_widget_show( p_item );
1089 /* signal hanling for off */
1090 gtk_signal_connect( GTK_OBJECT( p_item ), "toggled",
1091 GTK_SIGNAL_FUNC ( pf_toggle ), NULL );
1093 gtk_menu_append( GTK_MENU( p_menu ), p_item );
1096 /* link the new menu to the menubar item */
1097 gtk_menu_item_set_submenu( GTK_MENU_ITEM( p_root ), p_menu );
1101 gtk_check_menu_item_set_active (GTK_CHECK_MENU_ITEM (p_item_active), TRUE);
1104 if( val.p_list->i_count > 0 )
1106 gtk_widget_set_sensitive( p_root, TRUE );
1109 /* clean up everything */
1110 var_Change( p_object, psz_variable, VLC_VAR_FREELIST, &val );
1115 /*****************************************************************************
1116 * GtkDeinterlaceMenus: update interactive menus of the interface
1117 *****************************************************************************
1118 *****************************************************************************/
1119 static gint GtkDeinterlaceMenus( gpointer p_data,
1121 void(*pf_toggle )( GtkCheckMenuItem *, gpointer ) )
1123 intf_thread_t * p_intf;
1125 GtkWidget * p_separator;
1127 GtkWidget * p_item_active;
1131 char *ppsz_deinterlace_mode[] = { "discard", "blend", "mean", "bob", "linear", NULL };
1132 char *psz_deinterlace_option;
1135 p_intf = (intf_thread_t *)p_data;
1137 /* temporary hack to avoid blank menu when an open menu is removed */
1138 if( GTK_MENU_ITEM(p_root)->submenu != NULL )
1140 gtk_menu_popdown( GTK_MENU( GTK_MENU_ITEM(p_root)->submenu ) );
1142 /* removes previous menu */
1143 gtk_menu_item_remove_submenu( GTK_MENU_ITEM( p_root ) );
1144 gtk_widget_set_sensitive( p_root, FALSE );
1148 /* menu container */
1149 p_menu = gtk_menu_new();
1150 gtk_object_set_data( GTK_OBJECT( p_menu ), "p_intf", p_intf );
1152 /* special case for "off" item */
1153 p_item = gtk_radio_menu_item_new_with_label( p_group, "None" );
1154 p_group = gtk_radio_menu_item_group( GTK_RADIO_MENU_ITEM( p_item ) );
1156 gtk_widget_show( p_item );
1158 /* signal hanling for off */
1159 gtk_signal_connect( GTK_OBJECT( p_item ), "toggled",
1160 GTK_SIGNAL_FUNC ( pf_toggle ), NULL );
1162 gtk_menu_append( GTK_MENU( p_menu ), p_item );
1164 p_separator = gtk_menu_item_new();
1165 gtk_widget_set_sensitive( p_separator, FALSE );
1166 gtk_widget_show( p_separator );
1167 gtk_menu_append( GTK_MENU( p_menu ), p_separator );
1170 /* search actual deinterlace mode */
1171 psz_filter = config_GetPsz( p_intf, "filter" );
1172 psz_deinterlace_option = strdup( "None" );
1174 if( psz_filter && *psz_filter )
1176 if( strstr ( psz_filter, "deinterlace" ) )
1178 free( psz_deinterlace_option );
1179 psz_deinterlace_option = config_GetPsz( p_intf, "deinterlace-mode" );
1180 if( !psz_deinterlace_option )
1181 psz_deinterlace_option = strdup( "None" );
1187 p_item_active = NULL;
1190 /* create a set of deinteralce buttons and append them to the container */
1191 for( i = 0; ppsz_deinterlace_mode[i] != NULL; i++ )
1195 p_item = gtk_radio_menu_item_new_with_label( p_group, ppsz_deinterlace_mode[i] );
1196 p_group = gtk_radio_menu_item_group( GTK_RADIO_MENU_ITEM( p_item ) );
1197 gtk_widget_show( p_item );
1199 if( !strcmp( ppsz_deinterlace_mode[i], psz_deinterlace_option ) )
1201 p_item_active = p_item;
1203 /* setup signal hanling */
1204 gtk_signal_connect( GTK_OBJECT( p_item ), "toggled",
1205 GTK_SIGNAL_FUNC( pf_toggle ),
1208 gtk_menu_append( GTK_MENU( p_menu ), p_item );
1212 /* link the new menu to the menubar item */
1213 gtk_menu_item_set_submenu( GTK_MENU_ITEM( p_root ), p_menu );
1215 /* acitvation will call signals so we can only do it
1216 * when submenu is attached to menu - to get intf_window
1217 * We have to release the lock since input_ToggleES needs it */
1218 if( p_item_active != NULL )
1220 gtk_check_menu_item_set_active( GTK_CHECK_MENU_ITEM( p_item_active ),
1224 /* be sure that menu is sensitive if non empty */
1227 gtk_widget_set_sensitive( p_root, TRUE );
1235 /*****************************************************************************
1236 * GtkSetupMenus: function that generates title/chapter/audio/subpic
1237 * menus with help from preceding functions
1238 *****************************************************************************
1239 * Function called with the lock on stream
1240 *****************************************************************************/
1241 gint GtkSetupMenus( intf_thread_t * p_intf )
1243 es_descriptor_t * p_audio_es;
1244 es_descriptor_t * p_spu_es;
1245 GtkWidget * p_menubar_menu;
1246 GtkWidget * p_popup_menu;
1249 p_intf->p_sys->b_chapter_update |= p_intf->p_sys->b_title_update;
1250 p_intf->p_sys->b_audio_update |= p_intf->p_sys->b_title_update |
1251 p_intf->p_sys->b_program_update;
1252 p_intf->p_sys->b_spu_update |= p_intf->p_sys->b_title_update |
1253 p_intf->p_sys->b_program_update;
1257 p_menubar_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1258 p_intf->p_sys->p_window ), "menubar_deinterlace" ) );
1259 p_popup_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1260 p_intf->p_sys->p_popup ), "popup_deinterlace" ) );
1262 p_intf->p_sys->b_deinterlace_update = VLC_TRUE;
1263 GtkDeinterlaceMenus( p_intf, p_menubar_menu, GtkMenubarDeinterlaceToggle );
1264 p_intf->p_sys->b_deinterlace_update = VLC_TRUE;
1265 GtkDeinterlaceMenus( p_intf, p_popup_menu, GtkPopupDeinterlaceToggle );
1267 p_intf->p_sys->b_deinterlace_update = VLC_FALSE;
1270 if( p_intf->p_sys->b_program_update )
1272 pgrm_descriptor_t * p_pgrm;
1274 if( p_intf->p_sys->p_input->stream.p_new_program )
1276 p_pgrm = p_intf->p_sys->p_input->stream.p_new_program;
1280 p_pgrm = p_intf->p_sys->p_input->stream.p_selected_program;
1283 p_menubar_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1284 p_intf->p_sys->p_window ), "menubar_program" ) );
1285 GtkProgramMenu( p_intf, p_menubar_menu, p_pgrm,
1286 GtkMenubarProgramToggle );
1288 p_intf->p_sys->b_program_update = VLC_TRUE;
1289 p_popup_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1290 p_intf->p_sys->p_popup ), "popup_program" ) );
1291 GtkProgramMenu( p_intf, p_popup_menu, p_pgrm,
1292 GtkPopupProgramToggle );
1294 p_intf->p_sys->b_program_update = VLC_FALSE;
1297 if( p_intf->p_sys->b_title_update )
1301 p_menubar_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1302 p_intf->p_sys->p_window ), "menubar_title" ) );
1303 GtkRadioMenu( p_intf, p_menubar_menu, NULL, _("Title"), 1,
1304 p_intf->p_sys->p_input->stream.i_area_nb - 1,
1305 p_intf->p_sys->p_input->stream.p_selected_area->i_id,
1306 GtkMenubarTitleToggle );
1308 snprintf( psz_title, 4, "%d",
1309 p_intf->p_sys->p_input->stream.p_selected_area->i_id );
1310 psz_title[ 4 ] = '\0';
1311 gtk_label_set_text( p_intf->p_sys->p_label_title, psz_title );
1313 p_intf->p_sys->b_title_update = VLC_FALSE;
1316 if( p_intf->p_sys->b_chapter_update )
1318 char psz_chapter[5];
1320 p_popup_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1321 p_intf->p_sys->p_popup ), "popup_navigation" ) );
1322 GtkTitleMenu( p_intf, p_popup_menu, GtkPopupNavigationToggle );
1324 GtkRadioMenu( p_intf, p_menubar_menu, NULL, _("Title"), 1,
1325 p_intf->p_sys->p_input->stream.i_area_nb - 1,
1326 p_intf->p_sys->p_input->stream.p_selected_area->i_id,
1327 on_menubar_chapter_toggle );
1330 p_menubar_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1331 p_intf->p_sys->p_window ), "menubar_chapter" ) );
1333 GtkRadioMenu( p_intf, p_menubar_menu, NULL, _("Chapter"), 1,
1334 p_intf->p_sys->p_input->stream.p_selected_area->i_part_nb - 1,
1335 p_intf->p_sys->p_input->stream.p_selected_area->i_part,
1336 GtkMenubarChapterToggle );
1339 snprintf( psz_chapter, 4, "%d",
1340 p_intf->p_sys->p_input->stream.p_selected_area->i_part );
1341 psz_chapter[ 4 ] = '\0';
1342 gtk_label_set_text( p_intf->p_sys->p_label_chapter, psz_chapter );
1344 p_intf->p_sys->i_part =
1345 p_intf->p_sys->p_input->stream.p_selected_area->i_part;
1347 p_intf->p_sys->b_chapter_update = VLC_FALSE;
1350 /* look for selected ES */
1354 for( i = 0 ; i < p_intf->p_sys->p_input->stream.i_selected_es_number ; i++ )
1356 if( p_intf->p_sys->p_input->stream.pp_selected_es[i]->i_cat == AUDIO_ES )
1358 p_audio_es = p_intf->p_sys->p_input->stream.pp_selected_es[i];
1361 if( p_intf->p_sys->p_input->stream.pp_selected_es[i]->i_cat == SPU_ES )
1363 p_spu_es = p_intf->p_sys->p_input->stream.pp_selected_es[i];
1367 vlc_mutex_unlock( &p_intf->p_sys->p_input->stream.stream_lock );
1370 if( p_intf->p_sys->b_audio_update )
1372 /* find audio root menu */
1373 p_menubar_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1374 p_intf->p_sys->p_window ), "menubar_audio" ) );
1376 p_popup_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1377 p_intf->p_sys->p_popup ), "popup_audio" ) );
1379 p_intf->p_sys->b_audio_update = VLC_TRUE;
1380 GtkLanguageMenus( p_intf, p_menubar_menu, p_audio_es, AUDIO_ES,
1381 GtkMenubarAudioToggle );
1382 p_intf->p_sys->b_audio_update = VLC_TRUE;
1383 GtkLanguageMenus( p_intf, p_popup_menu, p_audio_es, AUDIO_ES,
1384 GtkPopupAudioToggle );
1386 p_intf->p_sys->b_audio_update = VLC_FALSE;
1389 /* sub picture menus */
1390 if( p_intf->p_sys->b_spu_update )
1392 /* find spu root menu */
1393 p_menubar_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1394 p_intf->p_sys->p_window ), "menubar_subpictures" ) );
1396 p_popup_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1397 p_intf->p_sys->p_popup ), "popup_subpictures" ) );
1399 p_intf->p_sys->b_spu_update = VLC_TRUE;
1400 GtkLanguageMenus( p_intf, p_menubar_menu, p_spu_es, SPU_ES,
1401 GtkMenubarSubtitleToggle );
1402 p_intf->p_sys->b_spu_update = VLC_TRUE;
1403 GtkLanguageMenus( p_intf, p_popup_menu, p_spu_es, SPU_ES,
1404 GtkPopupSubtitleToggle );
1406 p_intf->p_sys->b_spu_update = VLC_FALSE;
1408 /* create audio channels and device menu (in menubar _and_ popup */
1409 if( p_intf->p_sys->b_aout_update )
1411 aout_instance_t *p_aout;
1413 p_aout = (aout_instance_t*)vlc_object_find( p_intf, VLC_OBJECT_AOUT, FIND_ANYWHERE );
1415 if( p_aout != NULL )
1418 val.b_bool = VLC_FALSE;
1420 var_Set( (vlc_object_t *)p_aout, "intf-change", val );
1422 /* audio-channels */
1423 p_menubar_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1424 p_intf->p_sys->p_window ), "menubar_audio_channels" ) );
1425 p_popup_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1426 p_intf->p_sys->p_popup ), "popup_audio_channels" ) );
1427 GtkSetupVarMenu( p_intf, (vlc_object_t *)p_aout, p_popup_menu,
1428 "audio-channels", GtkPopupAoutChannelsToggle );
1429 GtkSetupVarMenu( p_intf, (vlc_object_t *)p_aout, p_menubar_menu,
1430 "audio-channels", GtkPopupAoutChannelsToggle );
1433 p_menubar_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1434 p_intf->p_sys->p_window ), "menubar_audio_device" ) );
1435 p_popup_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1436 p_intf->p_sys->p_popup ), "popup_audio_device" ) );
1437 GtkSetupVarMenu( p_intf, (vlc_object_t *)p_aout, p_popup_menu,
1438 "audio-device", GtkPopupAoutDeviceToggle );
1439 GtkSetupVarMenu( p_intf, (vlc_object_t *)p_aout, p_menubar_menu,
1440 "audio-device", GtkPopupAoutDeviceToggle );
1442 vlc_object_release( (vlc_object_t *)p_aout );
1444 p_intf->p_sys->b_aout_update = VLC_FALSE;
1447 if( p_intf->p_sys->b_vout_update )
1449 vout_thread_t *p_vout;
1451 p_vout = (vout_thread_t*)vlc_object_find( p_intf, VLC_OBJECT_VOUT, FIND_ANYWHERE );
1453 if( p_vout != NULL )
1456 val.b_bool = VLC_FALSE;
1458 var_Set( (vlc_object_t *)p_vout, "intf-change", val );
1461 p_menubar_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1462 p_intf->p_sys->p_window ), "menubar_video_device" ) );
1463 p_popup_menu = GTK_WIDGET( gtk_object_get_data( GTK_OBJECT(
1464 p_intf->p_sys->p_popup ), "popup_video_device" ) );
1465 GtkSetupVarMenu( p_intf, (vlc_object_t *)p_vout, p_popup_menu,
1466 "video-device", GtkPopupVoutDeviceToggle );
1467 GtkSetupVarMenu( p_intf, (vlc_object_t *)p_vout, p_menubar_menu,
1468 "video-device", GtkPopupVoutDeviceToggle );
1471 vlc_object_release( (vlc_object_t *)p_vout );
1473 p_intf->p_sys->b_vout_update = VLC_FALSE;
1475 vlc_mutex_lock( &p_intf->p_sys->p_input->stream.stream_lock );