]> git.sesse.net Git - vlc/blob - modules/gui/qt4/menus.cpp
Rename directx-wallpaper to video-wallpaper
[vlc] / modules / gui / qt4 / menus.cpp
1 /*****************************************************************************
2  * menus.cpp : Qt menus
3  *****************************************************************************
4  * Copyright © 2006-2009 the VideoLAN team
5  * $Id$
6  *
7  * Authors: Clément Stenac <zorglub@videolan.org>
8  *          Jean-Baptiste Kempf <jb@videolan.org>
9  *          Jean-Philippe André <jpeg@videolan.org>
10  *
11  * This program is free software; you can redistribute it and/or modify
12  * it under the terms of the GNU General Public License as published by
13  * the Free Software Foundation; either version 2 of the License, or
14  * ( at your option ) any later version.
15  *
16  * This program is distributed in the hope that it will be useful,
17  * but WITHOUT ANY WARRANTY; without even the implied warranty of
18  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
19  * GNU General Public License for more details.
20  *
21  * You should have received a copy of the GNU General Public License
22  * along with this program; if not, write to the Free Software
23  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
24  *****************************************************************************/
25
26 /** \todo
27  * - Remove static currentGroup
28  */
29
30 #ifdef HAVE_CONFIG_H
31 # include "config.h"
32 #endif
33
34 #include <vlc_intf_strings.h>
35 #include <vlc_services_discovery.h>
36 #include <vlc_aout.h>
37 #include <vlc_vout.h>
38
39 #include "menus.hpp"
40
41 #include "main_interface.hpp"    /* View modifications */
42 #include "dialogs_provider.hpp"  /* Dialogs display */
43 #include "input_manager.hpp"     /* Input Management */
44 #include "recents.hpp"           /* Recent Items */
45 #include "actions_manager.hpp"
46
47 #include <QMenu>
48 #include <QMenuBar>
49 #include <QAction>
50 #include <QActionGroup>
51 #include <QSignalMapper>
52 #include <QSystemTrayIcon>
53 #include <QList>
54
55 /*
56   This file defines the main menus and the pop-up menu (right-click menu)
57   and the systray menu (in that order in the file)
58
59   There are 3 menus that have to be rebuilt everytime there are called:
60   Audio, Video, Navigation
61   3 functions are building those menus: AudioMenu, VideoMenu, NavigMenu
62   and 3 functions associated are collecting the objects :
63   InputAutoMenuBuilder, AudioAutoMenuBuilder, VideoAutoMenuBuilder.
64
65   A QSignalMapper decides when to rebuild those menus cf MenuFunc in the .hpp
66   Just before one of those menus are aboutToShow(), they are rebuild.
67   */
68
69 #define STATIC_ENTRY "__static__"
70 #define ENTRY_ALWAYS_ENABLED "__ignore__"
71
72 enum
73 {
74     ITEM_NORMAL,
75     ITEM_CHECK,
76     ITEM_RADIO
77 };
78
79 static QActionGroup *currentGroup;
80
81 QMenu *QVLCMenu::recentsMenu = NULL;
82
83 /****************************************************************************
84  * Menu code helpers:
85  ****************************************************************************
86  * Add static entries to DP in menus
87  ***************************************************************************/
88 void addDPStaticEntry( QMenu *menu,
89                        const QString& text,
90                        const char *icon,
91                        const char *member,
92                        const char *shortcut = NULL )
93 {
94     QAction *action = NULL;
95     if( !EMPTY_STR( icon ) )
96     {
97         if( !EMPTY_STR( shortcut ) )
98             action = menu->addAction( QIcon( icon ), text, THEDP,
99                                       member, qtr( shortcut ) );
100         else
101             action = menu->addAction( QIcon( icon ), text, THEDP, member );
102     }
103     else
104     {
105         if( !EMPTY_STR( shortcut ) )
106             action = menu->addAction( text, THEDP, member, qtr( shortcut ) );
107         else
108             action = menu->addAction( text, THEDP, member );
109     }
110     action->setData( STATIC_ENTRY );
111 }
112
113 /***
114  * Same for MIM
115  ***/
116 QAction* addMIMStaticEntry( intf_thread_t *p_intf,
117                             QMenu *menu,
118                             const QString& text,
119                             const char *icon,
120                             const char *member,
121                             bool bStatic = false )
122 {
123     QAction *action;
124     if( strlen( icon ) > 0 )
125     {
126         action = menu->addAction( text, THEMIM,  member );
127         action->setIcon( QIcon( icon ) );
128     }
129     else
130     {
131         action = menu->addAction( text, THEMIM, member );
132     }
133     action->setData( bStatic ? STATIC_ENTRY : ENTRY_ALWAYS_ENABLED );
134     return action;
135 }
136
137 /**
138  * @brief Enable all static entries, disable the others
139  * @param enable if false, disable all entries
140  */
141 void EnableStaticEntries( QMenu *menu, bool enable = true )
142 {
143     if( !menu ) return;
144
145     QList< QAction* > actions = menu->actions();
146     for( int i = 0; i < actions.size(); ++i )
147     {
148         actions[i]->setEnabled( actions[i]->data().toString()
149                                 == ENTRY_ALWAYS_ENABLED ||
150             /* Be careful here, because data("string").toBool is true */
151             ( enable && (actions[i]->data().toString() == STATIC_ENTRY ) ) );
152     }
153 }
154
155 /**
156  * \return Number of static entries
157  */
158 int DeleteNonStaticEntries( QMenu *menu )
159 {
160     if( !menu ) return VLC_EGENERIC;
161
162     int i_ret = 0;
163
164     QList< QAction* > actions = menu->actions();
165     for( int i = 0; i < actions.size(); ++i )
166     {
167         if( actions[i]->data().toString() != STATIC_ENTRY )
168             delete actions[i];
169         else
170             i_ret++;
171     }
172     return i_ret;
173 }
174
175 /**
176  * \return QAction associated to psz_var variable
177  **/
178 static QAction * FindActionWithVar( QMenu *menu, const char *psz_var )
179 {
180     QList< QAction* > actions = menu->actions();
181     for( int i = 0; i < actions.size(); ++i )
182     {
183         if( actions[i]->data().toString() == psz_var )
184             return actions[i];
185     }
186     return NULL;
187 }
188
189 /*****************************************************************************
190  * Definitions of variables for the dynamic menus
191  *****************************************************************************/
192 #define PUSH_VAR( var ) varnames.push_back( var ); \
193     objects.push_back( VLC_OBJECT(p_object) )
194
195 #define PUSH_INPUTVAR( var ) varnames.push_back( var ); \
196     objects.push_back( VLC_OBJECT(p_input) );
197
198 #define PUSH_SEPARATOR if( objects.size() != i_last_separator ) { \
199     objects.push_back( 0 ); varnames.push_back( "" ); \
200     i_last_separator = objects.size(); }
201
202 static int InputAutoMenuBuilder( input_thread_t *p_object,
203         vector<vlc_object_t *> &objects,
204         vector<const char *> &varnames )
205 {
206     PUSH_VAR( "bookmark" );
207     PUSH_VAR( "title" );
208     PUSH_VAR( "chapter" );
209     PUSH_VAR( "navigation" );
210     PUSH_VAR( "program" );
211     return VLC_SUCCESS;
212 }
213
214 static int VideoAutoMenuBuilder( vout_thread_t *p_object,
215         input_thread_t *p_input,
216         vector<vlc_object_t *> &objects,
217         vector<const char *> &varnames )
218 {
219     PUSH_INPUTVAR( "video-es" );
220     PUSH_INPUTVAR( "spu-es" );
221     PUSH_VAR( "fullscreen" );
222     PUSH_VAR( "video-on-top" );
223     PUSH_VAR( "video-wallpaper" );
224 #ifdef WIN32
225     PUSH_VAR( "direct3d-desktop" );
226 #endif
227     PUSH_VAR( "video-snapshot" );
228     PUSH_VAR( "zoom" );
229     PUSH_VAR( "autoscale" );
230     PUSH_VAR( "aspect-ratio" );
231     PUSH_VAR( "crop" );
232     PUSH_VAR( "deinterlace" );
233     PUSH_VAR( "deinterlace-mode" );
234     PUSH_VAR( "postprocess" );
235
236     return VLC_SUCCESS;
237 }
238
239 static int AudioAutoMenuBuilder( aout_instance_t *p_object,
240         input_thread_t *p_input,
241         vector<vlc_object_t *> &objects,
242         vector<const char *> &varnames )
243 {
244     PUSH_INPUTVAR( "audio-es" );
245     PUSH_VAR( "audio-channels" );
246     PUSH_VAR( "audio-device" );
247     PUSH_VAR( "visual" );
248     return VLC_SUCCESS;
249 }
250
251 /*****************************************************************************
252  * All normal menus
253  * Simple Code
254  *****************************************************************************/
255
256 #define BAR_ADD( func, title ) { \
257     QMenu *_menu = func; _menu->setTitle( title ); bar->addMenu( _menu ); }
258
259 #define BAR_DADD( func, title, id ) { \
260     QMenu *_menu = func; _menu->setTitle( title ); bar->addMenu( _menu ); \
261     MenuFunc *f = new MenuFunc( _menu, id ); \
262     CONNECT( _menu, aboutToShow(), THEDP->menusUpdateMapper, map() ); \
263     THEDP->menusUpdateMapper->setMapping( _menu, f ); }
264
265 #define ACT_ADD( _menu, val, title ) { \
266     QAction *_action = new QAction( title, _menu ); _action->setData( val ); \
267     _menu->addAction( _action ); }
268
269 #define ACT_ADDMENU( _menu, val, title ) { \
270     QAction *_action = new QAction( title, _menu ); _action->setData( val ); \
271     _action->setMenu( new QMenu( _menu ) ); _menu->addAction( _action ); }
272
273 #define ACT_ADDCHECK( _menu, val, title ) { \
274     QAction *_action = new QAction( title, _menu ); _action->setData( val ); \
275     _action->setCheckable( true ); _menu->addAction( _action ); }
276
277 /**
278  * Main Menu Bar Creation
279  **/
280 void QVLCMenu::createMenuBar( MainInterface *mi,
281                               intf_thread_t *p_intf )
282 {
283     /* QMainWindows->menuBar()
284        gives the QProcess::destroyed timeout issue on Cleanlooks style with
285        setDesktopAware set to false */
286     QMenuBar *bar = mi->menuBar();
287
288     BAR_ADD( FileMenu( p_intf, bar ), qtr( "&Media" ) );
289
290     /* Dynamic menus, rebuilt before being showed */
291     BAR_DADD( NavigMenu( p_intf, bar ), qtr( "P&layback" ), 3 );
292     BAR_DADD( AudioMenu( p_intf, bar ), qtr( "&Audio" ), 1 );
293     BAR_DADD( VideoMenu( p_intf, bar ), qtr( "&Video" ), 2 );
294
295     BAR_ADD( ToolsMenu( bar ), qtr( "&Tools" ) );
296     BAR_ADD( ViewMenu( p_intf, mi ), qtr( "V&iew" ) );
297     BAR_ADD( HelpMenu( bar ), qtr( "&Help" ) );
298 }
299 #undef BAR_ADD
300 #undef BAR_DADD
301
302 /**
303  * Media ( File ) Menu
304  * Opening, streaming and quit
305  **/
306 QMenu *QVLCMenu::FileMenu( intf_thread_t *p_intf, QWidget *parent )
307 {
308     QMenu *menu = new QMenu( parent );
309
310     addDPStaticEntry( menu, qtr( "&Open File..." ),
311         ":/type/file-asym", SLOT( simpleOpenDialog() ), "Ctrl+O" );
312     addDPStaticEntry( menu, qtr( "Advanced Open File..." ),
313         ":/type/file-asym", SLOT( openFileDialog() ), "Ctrl+Shift+O" );
314     addDPStaticEntry( menu, qtr( I_OPEN_FOLDER ),
315         ":/type/folder-grey", SLOT( PLOpenDir() ), "Ctrl+F" );
316     addDPStaticEntry( menu, qtr( "Open &Disc..." ),
317         ":/type/disc", SLOT( openDiscDialog() ), "Ctrl+D" );
318     addDPStaticEntry( menu, qtr( "Open &Network Stream..." ),
319         ":/type/network", SLOT( openNetDialog() ), "Ctrl+N" );
320     addDPStaticEntry( menu, qtr( "Open &Capture Device..." ),
321         ":/type/capture-card", SLOT( openCaptureDialog() ),
322         "Ctrl+C" );
323
324     menu->addSeparator();
325     addDPStaticEntry( menu, qtr( "Open &Location from clipboard" ),
326                       NULL, SLOT( openUrlDialog() ), "Ctrl+V" );
327
328     if( config_GetInt( p_intf, "qt-recentplay" ) )
329     {
330         recentsMenu = new QMenu( qtr( "&Recent Media" ), menu );
331         updateRecents( p_intf );
332         menu->addMenu( recentsMenu );
333     }
334     menu->addSeparator();
335
336     addDPStaticEntry( menu, qtr( I_PL_SAVE ), "", SLOT( saveAPlaylist() ),
337         "Ctrl+Y" );
338     menu->addSeparator();
339
340 #ifdef ENABLE_SOUT
341     addDPStaticEntry( menu, qtr( "Conve&rt / Save..." ), "",
342         SLOT( openAndTranscodingDialogs() ), "Ctrl+R" );
343     addDPStaticEntry( menu, qtr( "&Streaming..." ),
344         ":/menu/stream", SLOT( openAndStreamingDialogs() ),
345         "Ctrl+S" );
346     menu->addSeparator();
347 #endif
348
349     addDPStaticEntry( menu, qtr( "&Quit" ) ,
350         ":/menu/quit", SLOT( quit() ), "Ctrl+Q" );
351     return menu;
352 }
353
354 /**
355  * Tools, like Media Information, Preferences or Messages
356  **/
357 QMenu *QVLCMenu::ToolsMenu( QMenu *menu )
358 {
359     addDPStaticEntry( menu, qtr( "&Effects and Filters"), ":/menu/settings",
360             SLOT( extendedDialog() ), "Ctrl+E" );
361
362     addDPStaticEntry( menu, qtr( "&Track Synchronization"), ":/menu/settings",
363             SLOT( synchroDialog() ), "" );
364
365     addDPStaticEntry( menu, qtr( I_MENU_INFO ) , ":/menu/info",
366         SLOT( mediaInfoDialog() ), "Ctrl+I" );
367     addDPStaticEntry( menu, qtr( I_MENU_CODECINFO ) ,
368         ":/menu/info", SLOT( mediaCodecDialog() ), "Ctrl+J" );
369
370     addDPStaticEntry( menu, qtr( I_MENU_BOOKMARK ),"",
371                       SLOT( bookmarksDialog() ), "Ctrl+B" );
372 #ifdef ENABLE_VLM
373     addDPStaticEntry( menu, qtr( I_MENU_VLM ), "", SLOT( vlmDialog() ),
374         "Ctrl+W" );
375 #endif
376
377     addDPStaticEntry( menu, qtr( I_MENU_MSG ),
378         ":/menu/messages", SLOT( messagesDialog() ),
379         "Ctrl+M" );
380
381     addDPStaticEntry( menu, qtr( "Plu&gins and extensions" ),
382         "", SLOT( pluginDialog() ) );
383     menu->addSeparator();
384
385     addDPStaticEntry( menu, qtr( "&Preferences" ),
386         ":/menu/preferences", SLOT( prefsDialog() ), "Ctrl+P" );
387
388     return menu;
389 }
390
391 QMenu *QVLCMenu::ToolsMenu( QWidget *parent )
392 {
393     return ToolsMenu( new QMenu( parent ) );
394 }
395
396 /**
397  * View Menu
398  * Interface Modification
399  **/
400 QMenu *QVLCMenu::ViewMenu( intf_thread_t *p_intf,
401                             MainInterface *mi,
402                             bool with_intf )
403 {
404     QAction *action;
405
406     assert( mi );
407
408     QMenu *menu = new QMenu( qtr( "V&iew" ), mi );
409
410     menu->addAction( QIcon( ":/menu/playlist_menu" ),
411             qtr( "Play&list" ), mi,
412             SLOT( togglePlaylist() ), qtr( "Ctrl+L" ) );
413
414     /*menu->addSeparator();
415     menu->addAction( qtr( "Undock from Interface" ), mi,
416                      SLOT( undockPlaylist() ), qtr( "Ctrl+U" ) );*/
417
418     menu->addSeparator();
419
420     if( with_intf )
421     {
422         QMenu *intfmenu = InterfacesMenu( p_intf, menu );
423         MenuFunc *f = new MenuFunc( intfmenu, 4 );
424         CONNECT( intfmenu, aboutToShow(), THEDP->menusUpdateMapper, map() );
425         THEDP->menusUpdateMapper->setMapping( intfmenu, f );
426         menu->addSeparator();
427     }
428
429     /* Minimal View */
430     action = menu->addAction( qtr( "Mi&nimal View" ) );
431     action->setShortcut( qtr( "Ctrl+H" ) );
432     action->setCheckable( true );
433     action->setChecked( !with_intf &&
434             (mi->getControlsVisibilityStatus() & CONTROLS_HIDDEN ) );
435
436     CONNECT( action, triggered( bool ), mi, toggleMinimalView( bool ) );
437     CONNECT( mi, minimalViewToggled( bool ), action, setChecked( bool ) );
438
439     /* FullScreen View */
440     action = menu->addAction( qtr( "&Fullscreen Interface" ), mi,
441             SLOT( toggleFullScreen() ), QString( "F11" ) );
442     action->setCheckable( true );
443     action->setChecked( mi->isFullScreen() );
444     CONNECT( mi, fullscreenInterfaceToggled( bool ),
445              action, setChecked( bool ) );
446
447     /* Advanced Controls */
448     action = menu->addAction( qtr( "&Advanced Controls" ), mi,
449             SLOT( toggleAdvanced() ) );
450     action->setCheckable( true );
451     if( mi->getControlsVisibilityStatus() & CONTROLS_ADVANCED )
452         action->setChecked( true );
453
454     if( with_intf )
455     // I don't want to manage consistency between menus, so no popup-menu
456     {
457         action = menu->addAction( qtr( "Quit after Playback" ) );
458         action->setCheckable( true );
459         CONNECT( action, triggered( bool ), THEMIM, activatePlayQuit( bool ) );
460     }
461
462 #if 0 /* For Visualisations. Not yet working */
463     adv = menu->addAction( qtr( "Visualizations selector" ),
464             mi, SLOT( visual() ) );
465     adv->setCheckable( true );
466     if( visual_selector_enabled ) adv->setChecked( true );
467 #endif
468
469     menu->addSeparator();
470     addDPStaticEntry( menu, qtr( "Customi&ze Interface..." ),
471         ":/menu/preferences", SLOT( toolbarDialog() ) );
472     menu->addSeparator();
473
474     return menu;
475 }
476
477 /**
478  * Interface Sub-Menu, to list extras interface and skins
479  **/
480 QMenu *QVLCMenu::InterfacesMenu( intf_thread_t *p_intf, QMenu *current )
481 {
482     vector<vlc_object_t *> objects;
483     vector<const char *> varnames;
484     varnames.push_back( "intf-add" );
485     objects.push_back( VLC_OBJECT(p_intf) );
486
487     return Populate( p_intf, current, varnames, objects );
488 }
489
490 /**
491  * Main Audio Menu
492  **/
493 QMenu *QVLCMenu::AudioMenu( intf_thread_t *p_intf, QMenu * current )
494 {
495     vector<vlc_object_t *> objects;
496     vector<const char *> varnames;
497     aout_instance_t *p_aout;
498     input_thread_t *p_input;
499
500     if( current->isEmpty() )
501     {
502         ACT_ADDMENU( current, "audio-es", qtr( "Audio &Track" ) );
503         ACT_ADDMENU( current, "audio-channels", qtr( "Audio &Channels" ) );
504         ACT_ADDMENU( current, "audio-device", qtr( "Audio &Device" ) );
505         current->addSeparator();
506
507         ACT_ADDMENU( current, "visual", qtr( "&Visualizations" ) );
508         current->addSeparator();
509
510         QAction *action = current->addAction( qtr( "Increase Volume" ),
511                 ActionsManager::getInstance( p_intf ), SLOT( AudioUp() ) );
512         action->setData( STATIC_ENTRY );
513         action = current->addAction( qtr( "Decrease Volume" ),
514                 ActionsManager::getInstance( p_intf ), SLOT( AudioDown() ) );
515         action->setData( STATIC_ENTRY );
516         action = current->addAction( qtr( "Mute" ),
517                 ActionsManager::getInstance( p_intf ), SLOT( toggleMuteAudio() ) );
518         action->setData( STATIC_ENTRY );
519     }
520
521     p_input = THEMIM->getInput();
522     p_aout = THEMIM->getAout();
523     EnableStaticEntries( current, ( p_aout != NULL ) );
524     AudioAutoMenuBuilder( p_aout, p_input, objects, varnames );
525     if( p_aout )
526     {
527         vlc_object_release( p_aout );
528     }
529
530     return Populate( p_intf, current, varnames, objects );
531 }
532
533 QMenu *QVLCMenu::AudioMenu( intf_thread_t *p_intf, QWidget *parent )
534 {
535     return AudioMenu( p_intf, new QMenu( parent ) );
536 }
537
538 /**
539  * Main Video Menu
540  * Subtitles are part of Video.
541  **/
542 QMenu *QVLCMenu::VideoMenu( intf_thread_t *p_intf, QMenu *current )
543 {
544     vout_thread_t *p_vout;
545     input_thread_t *p_input;
546     vector<vlc_object_t *> objects;
547     vector<const char *> varnames;
548
549     if( current->isEmpty() )
550     {
551         ACT_ADDMENU( current, "video-es", qtr( "Video &Track" ) );
552
553         QAction *action;
554         QMenu *submenu = new QMenu( qtr( "&Subtitles Track" ), current );
555         action = current->addMenu( submenu );
556         action->setData( "spu-es" );
557         addDPStaticEntry( submenu, qtr( "Open File..." ), "",
558                           SLOT( loadSubtitlesFile() ) );
559         submenu->addSeparator();
560         current->addSeparator();
561
562         ACT_ADDCHECK( current, "fullscreen", qtr( "&Fullscreen" ) );
563         ACT_ADDCHECK( current, "video-on-top", qtr( "Always &On Top" ) );
564         ACT_ADDCHECK( current, "video-wallpaper", qtr( "DirectX Wallpaper" ) );
565 #ifdef WIN32
566         ACT_ADDCHECK( current, "direct3d-desktop", qtr( "Direct3D Desktop mode" ) );
567 #endif
568         ACT_ADD( current, "video-snapshot", qtr( "Sna&pshot" ) );
569
570         current->addSeparator();
571
572         ACT_ADDMENU( current, "zoom", qtr( "&Zoom" ) );
573         ACT_ADDCHECK( current, "autoscale", qtr( "Sca&le" ) );
574         ACT_ADDMENU( current, "aspect-ratio", qtr( "&Aspect Ratio" ) );
575         ACT_ADDMENU( current, "crop", qtr( "&Crop" ) );
576         ACT_ADDMENU( current, "deinterlace", qtr( "&Deinterlace" ) );
577         ACT_ADDMENU( current, "deinterlace-mode", qtr( "&Deinterlace mode" ) );
578         ACT_ADDMENU( current, "postprocess", qtr( "&Post processing" ) );
579     }
580
581     p_input = THEMIM->getInput();
582
583     p_vout = THEMIM->getVout();
584     VideoAutoMenuBuilder( p_vout, p_input, objects, varnames );
585
586     if( p_vout )
587         vlc_object_release( p_vout );
588
589     return Populate( p_intf, current, varnames, objects );
590 }
591
592 QMenu *QVLCMenu::VideoMenu( intf_thread_t *p_intf, QWidget *parent )
593 {
594     return VideoMenu( p_intf, new QMenu( parent ) );
595 }
596
597 /**
598  * Navigation Menu
599  * For DVD, MP4, MOV and other chapter based format
600  **/
601 QMenu *QVLCMenu::NavigMenu( intf_thread_t *p_intf, QMenu *menu )
602 {
603     QAction *action;
604
605     QMenu *submenu = new QMenu( qtr( "&Bookmarks" ), menu );
606     addDPStaticEntry( submenu, qtr( "Manage &bookmarks" ), "",
607                       SLOT( bookmarksDialog() ) );
608     submenu->addSeparator();
609     action = menu->addMenu( submenu );
610     action->setData( "bookmark" );
611
612     ACT_ADDMENU( menu, "title", qtr( "T&itle" ) );
613     ACT_ADDMENU( menu, "chapter", qtr( "&Chapter" ) );
614     ACT_ADDMENU( menu, "navigation", qtr( "&Navigation" ) );
615     ACT_ADDMENU( menu, "program", qtr( "&Program" ) );
616
617     menu->addSeparator();
618     PopupMenuPlaylistControlEntries( menu, p_intf );
619     PopupMenuControlEntries( menu, p_intf );
620
621     EnableStaticEntries( menu, ( THEMIM->getInput() != NULL ) );
622     return RebuildNavigMenu( p_intf, menu );
623 }
624
625 QMenu *QVLCMenu::RebuildNavigMenu( intf_thread_t *p_intf, QMenu *menu )
626 {
627     /* */
628     input_thread_t *p_object;
629     vector<vlc_object_t *> objects;
630     vector<const char *> varnames;
631
632     /* Get the input and hold it */
633     p_object = THEMIM->getInput();
634
635     InputAutoMenuBuilder( p_object, objects, varnames );
636
637     menu->addSeparator();
638
639     /* Title and so on */
640     PUSH_VAR( "prev-title" );
641     PUSH_VAR( "next-title" );
642     PUSH_VAR( "prev-chapter" );
643     PUSH_VAR( "next-chapter" );
644
645     EnableStaticEntries( menu, (p_object != NULL ) );
646     return Populate( p_intf, menu, varnames, objects );
647 }
648
649 QMenu *QVLCMenu::NavigMenu( intf_thread_t *p_intf, QWidget *parent )
650 {
651     return NavigMenu( p_intf, new QMenu( parent ) );
652 }
653
654 /**
655  * Service Discovery SubMenu
656  **/
657 QMenu *QVLCMenu::SDMenu( intf_thread_t *p_intf, QWidget *parent )
658 {
659     QMenu *menu = new QMenu( parent );
660     menu->setTitle( qtr( I_PL_SD ) );
661
662     char **ppsz_longnames;
663     char **ppsz_names = vlc_sd_GetNames( p_intf, &ppsz_longnames );
664     if( !ppsz_names )
665         return menu;
666
667     char **ppsz_name = ppsz_names, **ppsz_longname = ppsz_longnames;
668     for( ; *ppsz_name; ppsz_name++, ppsz_longname++ )
669     {
670         if( !strcmp( *ppsz_name, "podcast" ) )
671         {
672             QAction *b = new QAction( qtr( "Configure podcasts..." ), menu );
673             //b->setEnabled( a->isChecked() );
674             menu->addAction( b );
675             CONNECT( b, triggered(), THEDP, podcastConfigureDialog() );
676         }
677         free( *ppsz_name );
678         free( *ppsz_longname );
679     }
680     free( ppsz_names );
681     free( ppsz_longnames );
682     return menu;
683 }
684
685 /**
686  * Help/About Menu
687 **/
688 QMenu *QVLCMenu::HelpMenu( QWidget *parent )
689 {
690     QMenu *menu = new QMenu( parent );
691     addDPStaticEntry( menu, qtr( "&Help..." ) ,
692         ":/menu/help", SLOT( helpDialog() ), "F1" );
693 #ifdef UPDATE_CHECK
694     addDPStaticEntry( menu, qtr( "Check for &Updates..." ) , "",
695                       SLOT( updateDialog() ) );
696 #endif
697     menu->addSeparator();
698     addDPStaticEntry( menu, qtr( I_MENU_ABOUT ), ":/menu/info",
699             SLOT( aboutDialog() ), "Shift+F1" );
700     return menu;
701 }
702
703 /*****************************************************************************
704  * Popup menus - Right Click menus                                           *
705  *****************************************************************************/
706 #define POPUP_BOILERPLATE \
707     unsigned int i_last_separator = 0; \
708     vector<vlc_object_t *> objects; \
709     vector<const char *> varnames; \
710     input_thread_t *p_input = THEMIM->getInput();
711
712 #define CREATE_POPUP \
713     Populate( p_intf, menu, varnames, objects ); \
714     p_intf->p_sys->p_popup_menu = menu; \
715     menu->popup( QCursor::pos() ); \
716     p_intf->p_sys->p_popup_menu = NULL; \
717     i_last_separator = 0;
718
719 void QVLCMenu::PopupPlayEntries( QMenu *menu,
720                                         intf_thread_t *p_intf,
721                                         input_thread_t *p_input )
722 {
723     QAction *action;
724
725     /* Play or Pause action and icon */
726     if( !p_input || var_GetInteger( p_input, "state" ) != PLAYING_S )
727     {
728         action = menu->addAction( qtr( "Play" ),
729                 ActionsManager::getInstance( p_intf ), SLOT( play() ) );
730         action->setIcon( QIcon( ":/menu/play" ) );
731     }
732     else
733     {
734          addMIMStaticEntry( p_intf, menu, qtr( "Pause" ),
735                     ":/menu/pause", SLOT( togglePlayPause() ) );
736     }
737 }
738
739 void QVLCMenu::PopupMenuControlEntries( QMenu *menu, intf_thread_t *p_intf )
740 {
741     QAction *action;
742
743     /* Faster/Slower */
744     action = menu->addAction( qtr( "&Faster" ), THEMIM->getIM(),
745                               SLOT( faster() ) );
746     action->setIcon( QIcon( ":/toolbar/faster") );
747     action->setData( STATIC_ENTRY );
748
749     action = menu->addAction( qtr( "Faster (fine)" ), THEMIM->getIM(),
750                               SLOT( littlefaster() ) );
751     action->setData( STATIC_ENTRY );
752
753     action = menu->addAction( qtr( "N&ormal Speed" ), THEMIM->getIM(),
754                               SLOT( normalRate() ) );
755     action->setData( STATIC_ENTRY );
756
757     action = menu->addAction( qtr( "Slower (fine)" ), THEMIM->getIM(),
758                               SLOT( littleslower() ) );
759     action->setData( STATIC_ENTRY );
760
761     action = menu->addAction( qtr( "Slo&wer" ), THEMIM->getIM(),
762                               SLOT( slower() ) );
763     action->setIcon( QIcon( ":/toolbar/slower") );
764     action->setData( STATIC_ENTRY );
765
766     menu->addSeparator();
767
768     action = menu->addAction( qtr( "&Jump Forward" ), THEMIM->getIM(),
769              SLOT( jumpFwd() ) );
770     action->setIcon( QIcon( ":/toolbar/skip_fw") );
771     action->setData( STATIC_ENTRY );
772
773     action = menu->addAction( qtr( "Jump Bac&kward" ), THEMIM->getIM(),
774              SLOT( jumpBwd() ) );
775     action->setIcon( QIcon( ":/toolbar/skip_back") );
776     action->setData( STATIC_ENTRY );
777     addDPStaticEntry( menu, qtr( I_MENU_GOTOTIME ),"",
778                       SLOT( gotoTimeDialog() ), "Ctrl+T" );
779     menu->addSeparator();
780 }
781
782
783 void QVLCMenu::PopupMenuPlaylistControlEntries( QMenu *menu,
784                                                 intf_thread_t *p_intf )
785 {
786     bool bEnable = THEMIM->getInput() != NULL;
787     QAction *action =
788             addMIMStaticEntry( p_intf, menu, qtr( "&Stop" ), ":/menu/stop",
789                                SLOT( stop() ), true );
790     /* Disable Stop in the right-click popup menu */
791     if( !bEnable )
792         action->setEnabled( false );
793
794     /* Next / Previous */
795     addMIMStaticEntry( p_intf, menu, qtr( "Pre&vious" ),
796         ":/menu/previous", SLOT( prev() ) );
797     addMIMStaticEntry( p_intf, menu, qtr( "Ne&xt" ),
798         ":/menu/next", SLOT( next() ) );
799     menu->addSeparator();
800 }
801
802 void QVLCMenu::PopupMenuStaticEntries( QMenu *menu )
803 {
804     QMenu *openmenu = new QMenu( qtr( "Open Media" ), menu );
805     addDPStaticEntry( openmenu, qtr( "&Open File..." ),
806         ":/type/file-asym", SLOT( openFileDialog() ) );
807     addDPStaticEntry( openmenu, qtr( I_OPEN_FOLDER ),
808         ":/type/folder-grey", SLOT( PLOpenDir() ) );
809     addDPStaticEntry( openmenu, qtr( "Open &Disc..." ),
810         ":/type/disc", SLOT( openDiscDialog() ) );
811     addDPStaticEntry( openmenu, qtr( "Open &Network..." ),
812         ":/type/network", SLOT( openNetDialog() ) );
813     addDPStaticEntry( openmenu, qtr( "Open &Capture Device..." ),
814         ":/type/capture-card", SLOT( openCaptureDialog() ) );
815     menu->addMenu( openmenu );
816
817     menu->addSeparator();
818 #if 0
819     QMenu *helpmenu = HelpMenu( menu );
820     helpmenu->setTitle( qtr( "Help" ) );
821     menu->addMenu( helpmenu );
822 #endif
823
824     addDPStaticEntry( menu, qtr( "Quit" ), ":/menu/quit",
825                       SLOT( quit() ), "Ctrl+Q" );
826 }
827
828 /* Video Tracks and Subtitles tracks */
829 void QVLCMenu::VideoPopupMenu( intf_thread_t *p_intf )
830 {
831     POPUP_BOILERPLATE;
832     if( p_input )
833     {
834         vout_thread_t *p_vout = THEMIM->getVout();
835         if( p_vout )
836         {
837             VideoAutoMenuBuilder( p_vout, p_input, objects, varnames );
838             vlc_object_release( p_vout );
839         }
840     }
841     QMenu *menu = new QMenu();
842     CREATE_POPUP;
843 }
844
845 /* Audio Tracks */
846 void QVLCMenu::AudioPopupMenu( intf_thread_t *p_intf )
847 {
848     POPUP_BOILERPLATE;
849     if( p_input )
850     {
851         aout_instance_t *p_aout = THEMIM->getAout();
852         AudioAutoMenuBuilder( p_aout, p_input, objects, varnames );
853         if( p_aout )
854             vlc_object_release( p_aout );
855     }
856     QMenu *menu = new QMenu();
857     CREATE_POPUP;
858 }
859
860 /* Navigation stuff, and general menus ( open ), used only for skins */
861 void QVLCMenu::MiscPopupMenu( intf_thread_t *p_intf )
862 {
863     POPUP_BOILERPLATE;
864
865     if( p_input )
866     {
867         varnames.push_back( "audio-es" );
868         InputAutoMenuBuilder( p_input, objects, varnames );
869         PUSH_SEPARATOR;
870     }
871
872     QMenu *menu = new QMenu();
873     Populate( p_intf, menu, varnames, objects );
874
875     menu->addSeparator();
876     PopupPlayEntries( menu, p_intf, p_input );
877     PopupMenuPlaylistControlEntries( menu, p_intf);
878
879     menu->addSeparator();
880     PopupMenuControlEntries( menu, p_intf );
881
882     menu->addSeparator();
883     PopupMenuStaticEntries( menu );
884
885     p_intf->p_sys->p_popup_menu = menu;
886     menu->popup( QCursor::pos() );
887     p_intf->p_sys->p_popup_menu = NULL;
888 }
889
890 /* Main Menu that sticks everything together  */
891 void QVLCMenu::PopupMenu( intf_thread_t *p_intf, bool show )
892 {
893     /* Delete old popup if there is one */
894     delete p_intf->p_sys->p_popup_menu;
895
896     if( !show )
897     {
898         p_intf->p_sys->p_popup_menu = NULL;
899         return;
900     }
901
902     /* */
903     QMenu *menu = new QMenu();
904     QAction *action;
905     bool b_isFullscreen = false;
906     MainInterface *mi = p_intf->p_sys->p_mi;
907
908     POPUP_BOILERPLATE;
909
910     PopupPlayEntries( menu, p_intf, p_input );
911     PopupMenuPlaylistControlEntries( menu, p_intf );
912     menu->addSeparator();
913
914     if( p_input )
915     {
916         QMenu *submenu;
917         vout_thread_t *p_vout = THEMIM->getVout();
918
919         /* Add a fullscreen switch button, since it is the most used function */
920         if( p_vout )
921         {
922             vlc_value_t val; var_Get( p_vout, "fullscreen", &val );
923
924             b_isFullscreen = !( !val.b_bool );
925             if( b_isFullscreen )
926                 CreateAndConnect( menu, "fullscreen",
927                         qtr( "Leave Fullscreen" ),"" , ITEM_NORMAL,
928                         VLC_OBJECT(p_vout), val, VLC_VAR_BOOL, b_isFullscreen );
929             vlc_object_release( p_vout );
930
931             menu->addSeparator();
932         }
933
934         /* Input menu */
935         InputAutoMenuBuilder( p_input, objects, varnames );
936
937         /* Audio menu */
938         submenu = new QMenu( menu );
939         action = menu->addMenu( AudioMenu( p_intf, submenu ) );
940         action->setText( qtr( "&Audio" ) );
941         if( action->menu()->isEmpty() )
942             action->setEnabled( false );
943
944         /* Video menu */
945         submenu = new QMenu( menu );
946         action = menu->addMenu( VideoMenu( p_intf, submenu ) );
947         action->setText( qtr( "&Video" ) );
948         if( action->menu()->isEmpty() )
949             action->setEnabled( false );
950
951         /* Playback menu for chapters */
952         submenu = new QMenu( menu );
953         action = menu->addMenu( NavigMenu( p_intf, submenu ) );
954         action->setText( qtr( "&Playback" ) );
955         if( action->menu()->isEmpty() )
956             action->setEnabled( false );
957     }
958
959     menu->addSeparator();
960
961     /* Add some special entries for windowed mode: Interface Menu */
962     if( !b_isFullscreen )
963     {
964         QMenu *submenu = new QMenu( qtr( "Interface" ), menu );
965         /*QMenu *tools =*/ ToolsMenu( submenu );
966         submenu->addSeparator();
967
968         /* In skins interface, append some items */
969         if( !mi )
970         {
971             if( p_intf->p_sys->b_isDialogProvider )
972             {
973                 vlc_object_t* p_object = p_intf->p_parent;
974
975                 objects.clear(); varnames.clear();
976                 objects.push_back( p_object );
977                 varnames.push_back( "intf-skins" );
978                 Populate( p_intf, submenu, varnames, objects );
979
980                 objects.clear(); varnames.clear();
981                 objects.push_back( p_object );
982                 varnames.push_back( "intf-skins-interactive" );
983                 Populate( p_intf, submenu, varnames, objects );
984             }
985             else
986                 msg_Warn( p_intf, "could not find parent interface" );
987         }
988         else
989             menu->addMenu( ViewMenu( p_intf, mi, false ));
990
991         menu->addMenu( submenu );
992     }
993
994     /* Static entries for ending, like open */
995     PopupMenuStaticEntries( menu );
996
997     p_intf->p_sys->p_popup_menu = menu;
998     p_intf->p_sys->p_popup_menu->popup( QCursor::pos() );
999 }
1000
1001 #undef ACT_ADD
1002 #undef ACT_ADDMENU
1003 #undef ACT_ADDCHECK
1004
1005 #ifndef HAVE_MAEMO
1006 /************************************************************************
1007  * Systray Menu                                                         *
1008  ************************************************************************/
1009
1010 void QVLCMenu::updateSystrayMenu( MainInterface *mi,
1011                                   intf_thread_t *p_intf,
1012                                   bool b_force_visible )
1013 {
1014     POPUP_BOILERPLATE;
1015
1016     /* Get the systray menu and clean it */
1017     QMenu *sysMenu = mi->getSysTrayMenu();
1018     sysMenu->clear();
1019
1020     /* Hide / Show VLC and cone */
1021     if( mi->isVisible() || b_force_visible )
1022     {
1023         sysMenu->addAction( QIcon( ":/logo/vlc16.png" ),
1024                             qtr( "Hide VLC media player in taskbar" ), mi,
1025                             SLOT( toggleUpdateSystrayMenu() ) );
1026     }
1027     else
1028     {
1029         sysMenu->addAction( QIcon( ":/logo/vlc16.png" ),
1030                             qtr( "Show VLC media player" ), mi,
1031                             SLOT( toggleUpdateSystrayMenu() ) );
1032     }
1033
1034     sysMenu->addSeparator();
1035     PopupPlayEntries( sysMenu, p_intf, p_input );
1036     PopupMenuPlaylistControlEntries( sysMenu, p_intf);
1037     PopupMenuControlEntries( sysMenu, p_intf);
1038
1039     sysMenu->addSeparator();
1040     addDPStaticEntry( sysMenu, qtr( "&Open Media" ),
1041             ":/type/file-wide", SLOT( openFileDialog() ) );
1042     addDPStaticEntry( sysMenu, qtr( "&Quit" ) ,
1043             ":/menu/quit", SLOT( quit() ) );
1044
1045     /* Set the menu */
1046     mi->getSysTray()->setContextMenu( sysMenu );
1047 }
1048 #endif
1049
1050 #undef CREATE_POPUP
1051 #undef POPUP_BOILERPLATE
1052
1053 #undef PUSH_VAR
1054 #undef PUSH_SEPARATOR
1055
1056 /*************************************************************************
1057  * Builders for automenus
1058  *************************************************************************/
1059 QMenu * QVLCMenu::Populate( intf_thread_t *p_intf,
1060                             QMenu *current,
1061                             vector< const char *> & varnames,
1062                             vector<vlc_object_t *> & objects )
1063 {
1064     QMenu *menu = current;
1065     assert( menu );
1066
1067     currentGroup = NULL;
1068
1069     vlc_object_t *p_object;
1070
1071     for( int i = 0; i < ( int )objects.size() ; i++ )
1072     {
1073         if( !varnames[i] || !*varnames[i] )
1074         {
1075             menu->addSeparator();
1076             continue;
1077         }
1078         p_object = objects[i];
1079
1080         UpdateItem( p_intf, menu, varnames[i], p_object, true );
1081     }
1082     return menu;
1083 }
1084
1085 /*****************************************************************************
1086  * Private methods.
1087  *****************************************************************************/
1088
1089 static bool IsMenuEmpty( const char *psz_var,
1090                          vlc_object_t *p_object,
1091                          bool b_root = true )
1092 {
1093     vlc_value_t val, val_list;
1094     int i_type, i_result, i;
1095
1096     /* Check the type of the object variable */
1097     i_type = var_Type( p_object, psz_var );
1098
1099     /* Check if we want to display the variable */
1100     if( !( i_type & VLC_VAR_HASCHOICE ) ) return false;
1101
1102     var_Change( p_object, psz_var, VLC_VAR_CHOICESCOUNT, &val, NULL );
1103     if( val.i_int == 0 ) return true;
1104
1105     if( ( i_type & VLC_VAR_TYPE ) != VLC_VAR_VARIABLE )
1106     {
1107         if( val.i_int == 1 && b_root ) return true;
1108         else return false;
1109     }
1110
1111     /* Check children variables in case of VLC_VAR_VARIABLE */
1112     if( var_Change( p_object, psz_var, VLC_VAR_GETLIST, &val_list, NULL ) < 0 )
1113     {
1114         return true;
1115     }
1116
1117     for( i = 0, i_result = true; i < val_list.p_list->i_count; i++ )
1118     {
1119         if( !IsMenuEmpty( val_list.p_list->p_values[i].psz_string,
1120                     p_object, false ) )
1121         {
1122             i_result = false;
1123             break;
1124         }
1125     }
1126
1127     /* clean up everything */
1128     var_FreeList( &val_list, NULL );
1129
1130     return i_result;
1131 }
1132
1133 #define TEXT_OR_VAR qfu ( text.psz_string ? text.psz_string : psz_var )
1134
1135 void QVLCMenu::UpdateItem( intf_thread_t *p_intf, QMenu *menu,
1136         const char *psz_var, vlc_object_t *p_object, bool b_submenu )
1137 {
1138     vlc_value_t val, text;
1139     int i_type;
1140
1141     QAction *action = FindActionWithVar( menu, psz_var );
1142     if( action )
1143         DeleteNonStaticEntries( action->menu() );
1144
1145     if( !p_object )
1146     {
1147         if( action )
1148             action->setEnabled( false );
1149         return;
1150     }
1151
1152     /* Check the type of the object variable */
1153     /* This HACK is needed so we have a radio button for audio and video tracks
1154        instread of a checkbox */
1155     if( !strcmp( psz_var, "audio-es" )
1156      || !strcmp( psz_var, "video-es" ) )
1157         i_type = VLC_VAR_INTEGER | VLC_VAR_HASCHOICE;
1158     else
1159         i_type = var_Type( p_object, psz_var );
1160
1161     switch( i_type & VLC_VAR_TYPE )
1162     {
1163         case VLC_VAR_VOID:
1164         case VLC_VAR_BOOL:
1165         case VLC_VAR_VARIABLE:
1166         case VLC_VAR_STRING:
1167         case VLC_VAR_INTEGER:
1168         case VLC_VAR_FLOAT:
1169             break;
1170         default:
1171             /* Variable doesn't exist or isn't handled */
1172             if( action )
1173                 action->setEnabled( false );
1174             return;
1175     }
1176
1177     /* Make sure we want to display the variable */
1178     if( menu->isEmpty() && IsMenuEmpty( psz_var, p_object ) )
1179     {
1180         if( action )
1181             action->setEnabled( false );
1182         return;
1183     }
1184
1185     /* Get the descriptive name of the variable */
1186     int i_ret = var_Change( p_object, psz_var, VLC_VAR_GETTEXT, &text, NULL );
1187     if( i_ret != VLC_SUCCESS )
1188     {
1189         text.psz_string = NULL;
1190     }
1191
1192     if( !action )
1193     {
1194         action = new QAction( TEXT_OR_VAR, menu );
1195         menu->addAction( action );
1196         action->setData( psz_var );
1197     }
1198
1199     /* Some specific stuff */
1200     bool forceDisabled = false;
1201     if( !strcmp( psz_var, "spu-es" ) )
1202     {
1203         vout_thread_t *p_vout = THEMIM->getVout();
1204         forceDisabled = ( p_vout == NULL );
1205         if( p_vout )
1206             vlc_object_release( p_vout );
1207     }
1208
1209     if( i_type & VLC_VAR_HASCHOICE )
1210     {
1211         /* Append choices menu */
1212         if( b_submenu )
1213         {
1214             QMenu *submenu;
1215             submenu = action->menu();
1216             if( !submenu )
1217             {
1218                 submenu = new QMenu( menu );
1219                 action->setMenu( submenu );
1220             }
1221
1222             action->setEnabled(
1223                CreateChoicesMenu( submenu, psz_var, p_object, true ) == 0 );
1224             if( forceDisabled )
1225                 action->setEnabled( false );
1226         }
1227         else
1228         {
1229             action->setEnabled(
1230                 CreateChoicesMenu( menu, psz_var, p_object, true ) == 0 );
1231         }
1232         FREENULL( text.psz_string );
1233         return;
1234     }
1235
1236     switch( i_type & VLC_VAR_TYPE )
1237     {
1238         case VLC_VAR_VOID:
1239             var_Get( p_object, psz_var, &val );
1240             CreateAndConnect( menu, psz_var, TEXT_OR_VAR, "", ITEM_NORMAL,
1241                     p_object, val, i_type );
1242             break;
1243
1244         case VLC_VAR_BOOL:
1245             var_Get( p_object, psz_var, &val );
1246             val.b_bool = !val.b_bool;
1247             CreateAndConnect( menu, psz_var, TEXT_OR_VAR, "", ITEM_CHECK,
1248                     p_object, val, i_type, !val.b_bool );
1249             break;
1250     }
1251     FREENULL( text.psz_string );
1252 }
1253
1254 #undef TEXT_OR_VAR
1255
1256 /** HACK for the navigation submenu:
1257  * "title %2i" variables take the value 0 if not set
1258  */
1259 static bool CheckTitle( vlc_object_t *p_object, const char *psz_var )
1260 {
1261     int i_title = 0;
1262     if( sscanf( psz_var, "title %2i", &i_title ) <= 0 )
1263         return true;
1264
1265     int i_current_title = var_GetInteger( p_object, "title" );
1266     return ( i_title == i_current_title );
1267 }
1268
1269
1270 int QVLCMenu::CreateChoicesMenu( QMenu *submenu, const char *psz_var,
1271         vlc_object_t *p_object, bool b_root )
1272 {
1273     vlc_value_t val, val_list, text_list;
1274     int i_type, i;
1275
1276     /* Check the type of the object variable */
1277     i_type = var_Type( p_object, psz_var );
1278
1279     /* Make sure we want to display the variable */
1280     if( submenu->isEmpty() && IsMenuEmpty( psz_var, p_object, b_root ) )
1281         return VLC_EGENERIC;
1282
1283     switch( i_type & VLC_VAR_TYPE )
1284     {
1285         case VLC_VAR_VOID:
1286         case VLC_VAR_BOOL:
1287         case VLC_VAR_VARIABLE:
1288         case VLC_VAR_STRING:
1289         case VLC_VAR_INTEGER:
1290         case VLC_VAR_FLOAT:
1291             break;
1292         default:
1293             /* Variable doesn't exist or isn't handled */
1294             return VLC_EGENERIC;
1295     }
1296
1297     if( var_Change( p_object, psz_var, VLC_VAR_GETLIST,
1298                     &val_list, &text_list ) < 0 )
1299     {
1300         return VLC_EGENERIC;
1301     }
1302
1303 #define CURVAL val_list.p_list->p_values[i]
1304 #define CURTEXT text_list.p_list->p_values[i].psz_string
1305
1306     for( i = 0; i < val_list.p_list->i_count; i++ )
1307     {
1308         vlc_value_t another_val;
1309         QString menutext;
1310         QMenu *subsubmenu = new QMenu( submenu );
1311
1312         switch( i_type & VLC_VAR_TYPE )
1313         {
1314             case VLC_VAR_VARIABLE:
1315                 CreateChoicesMenu( subsubmenu, CURVAL.psz_string, p_object, false );
1316                 subsubmenu->setTitle( qfu( CURTEXT ? CURTEXT :CURVAL.psz_string ) );
1317                 submenu->addMenu( subsubmenu );
1318                 break;
1319
1320             case VLC_VAR_STRING:
1321                 var_Get( p_object, psz_var, &val );
1322                 another_val.psz_string = strdup( CURVAL.psz_string );
1323                 menutext = qfu( CURTEXT ? CURTEXT : another_val.psz_string );
1324                 CreateAndConnect( submenu, psz_var, menutext, "", ITEM_RADIO,
1325                         p_object, another_val, i_type,
1326                         val.psz_string && !strcmp( val.psz_string, CURVAL.psz_string ) );
1327
1328                 free( val.psz_string );
1329                 break;
1330
1331             case VLC_VAR_INTEGER:
1332                 var_Get( p_object, psz_var, &val );
1333                 if( CURTEXT ) menutext = qfu( CURTEXT );
1334                 else menutext.sprintf( "%d", CURVAL.i_int );
1335                 CreateAndConnect( submenu, psz_var, menutext, "", ITEM_RADIO,
1336                         p_object, CURVAL, i_type,
1337                         ( CURVAL.i_int == val.i_int )
1338                         && CheckTitle( p_object, psz_var ) );
1339                 break;
1340
1341             case VLC_VAR_FLOAT:
1342                 var_Get( p_object, psz_var, &val );
1343                 if( CURTEXT ) menutext = qfu( CURTEXT );
1344                 else menutext.sprintf( "%.2f", CURVAL.f_float );
1345                 CreateAndConnect( submenu, psz_var, menutext, "", ITEM_RADIO,
1346                         p_object, CURVAL, i_type,
1347                         CURVAL.f_float == val.f_float );
1348                 break;
1349
1350             default:
1351                 break;
1352         }
1353     }
1354     currentGroup = NULL;
1355
1356     /* clean up everything */
1357     var_FreeList( &val_list, &text_list );
1358
1359 #undef CURVAL
1360 #undef CURTEXT
1361     return submenu->isEmpty() ? VLC_EGENERIC : VLC_SUCCESS;
1362 }
1363
1364 void QVLCMenu::CreateAndConnect( QMenu *menu, const char *psz_var,
1365         const QString& text, const QString& help,
1366         int i_item_type, vlc_object_t *p_obj,
1367         vlc_value_t val, int i_val_type,
1368         bool checked )
1369 {
1370     QAction *action = FindActionWithVar( menu, psz_var );
1371
1372     bool b_new = false;
1373     if( !action )
1374     {
1375         action = new QAction( text, menu );
1376         menu->addAction( action );
1377         b_new = true;
1378     }
1379
1380     action->setToolTip( help );
1381     action->setEnabled( p_obj != NULL );
1382
1383     if( i_item_type == ITEM_CHECK )
1384     {
1385         action->setCheckable( true );
1386     }
1387     else if( i_item_type == ITEM_RADIO )
1388     {
1389         action->setCheckable( true );
1390         if( !currentGroup )
1391             currentGroup = new QActionGroup( menu );
1392         currentGroup->addAction( action );
1393     }
1394
1395     action->setChecked( checked );
1396
1397     MenuItemData *itemData = new MenuItemData( THEDP->menusMapper, p_obj, i_val_type,
1398             val, psz_var );
1399
1400     /* remove previous signal-slot connection(s) if any */
1401     action->disconnect( );
1402
1403     CONNECT( action, triggered(), THEDP->menusMapper, map() );
1404     THEDP->menusMapper->setMapping( action, itemData );
1405
1406     if( b_new )
1407         menu->addAction( action );
1408 }
1409
1410 void QVLCMenu::DoAction( QObject *data )
1411 {
1412     MenuItemData *itemData = qobject_cast<MenuItemData *>( data );
1413     vlc_object_t *p_object = itemData->p_obj;
1414     if( p_object == NULL ) return;
1415
1416     var_Set( p_object, itemData->psz_var, itemData->val );
1417 }
1418
1419 void QVLCMenu::updateRecents( intf_thread_t *p_intf )
1420 {
1421     if( recentsMenu )
1422     {
1423         QAction* action;
1424         RecentsMRL* rmrl = RecentsMRL::getInstance( p_intf );
1425         QStringList l = rmrl->recents();
1426
1427         recentsMenu->clear();
1428
1429         if( !l.size() )
1430         {
1431             action = recentsMenu->addAction( qtr(" - Empty - ") );
1432             action->setEnabled( false );
1433         }
1434         else
1435         {
1436             for( int i = 0; i < l.size(); ++i )
1437             {
1438                 action = recentsMenu->addAction(
1439                         QString( "&%1: " ).arg( i + 1 ) + l.at( i ),
1440                         rmrl->signalMapper, SLOT( map() ),
1441                         i <= 9 ? QString( "Ctrl+%1" ).arg( i + 1 ) : "" );
1442                 rmrl->signalMapper->setMapping( action, l.at( i ) );
1443             }
1444
1445             recentsMenu->addSeparator();
1446             recentsMenu->addAction( qtr("&Clear"), rmrl, SLOT( clear() ) );
1447         }
1448     }
1449 }