1 /*****************************************************************************
2 * actions_manager.cpp : Controller for the main interface
3 ****************************************************************************
4 * Copyright © 2009-2014 VideoLAN and VLC authors
7 * Authors: Jean-Baptiste Kempf <jb@videolan.org>
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * ( at your option ) any later version.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22 *****************************************************************************/
29 #include <vlc_actions.h>
30 #include <vlc_renderer_discovery.h>
32 #include "actions_manager.hpp"
34 #include "dialogs_provider.hpp" /* Opening Dialogs */
35 #include "input_manager.hpp" /* THEMIM */
36 #include "main_interface.hpp" /* Show playlist */
37 #include "components/controller.hpp" /* Toggle FSC controller width */
38 #include "components/extended_panels.hpp"
41 ActionsManager::ActionsManager( intf_thread_t
* _p_i
)
45 CONNECT( this, rendererItemAdded( vlc_renderer_item_t
* ),
46 this, onRendererItemAdded( vlc_renderer_item_t
* ) );
47 CONNECT( this, rendererItemRemoved( vlc_renderer_item_t
* ),
48 this, onRendererItemRemoved( vlc_renderer_item_t
* ) );
49 m_stop_scan_timer
.setSingleShot( true );
50 CONNECT( &m_stop_scan_timer
, timeout(), this, StopRendererScan() );
53 ActionsManager::~ActionsManager()
56 /* reset the list of renderers */
57 foreach (QAction
* action
, VLCMenuBar::rendererMenu
->actions())
59 QVariant data
= action
->data();
60 if (!data
.canConvert
<QVariantHash
>())
62 VLCMenuBar::rendererMenu
->removeAction(action
);
63 VLCMenuBar::rendererGroup
->removeAction(action
);
67 void ActionsManager::doAction( int id_action
)
74 THEMIM
->stop(); break;
76 THEDP
->openDialog(); break;
78 THEMIM
->prev(); break;
80 THEMIM
->next(); break;
82 THEMIM
->getIM()->slower(); break;
84 THEMIM
->getIM()->faster(); break;
85 case FULLSCREEN_ACTION
:
88 THEDP
->extendedDialog(); break;
98 THEMIM
->getIM()->setAtoB(); break;
100 THEMIM
->getIM()->reverse(); break;
101 case SKIP_BACK_ACTION
:
108 THEDP
->quit(); break;
110 THEMIM
->toggleRandom(); break;
112 THEDP
->mediaInfoDialog(); break;
113 case OPEN_SUB_ACTION
:
114 THEDP
->loadSubtitlesFile(); break;
115 case FULLWIDTH_ACTION
:
116 if( p_intf
->p_sys
->p_mi
)
117 p_intf
->p_sys
->p_mi
->getFullscreenControllerWidget()->toggleFullwidth();
120 msg_Warn( p_intf
, "Action not supported: %i", id_action
);
125 void ActionsManager::play()
127 if( THEPL
->current
.i_size
== 0 && THEPL
->items
.i_size
== 0 )
129 /* The playlist is empty, open a file requester */
130 THEDP
->openFileDialog();
133 THEMIM
->togglePlayPause();
138 * This functions toggle the fullscreen mode
139 * If there is no video, it should first activate Visualisations...
140 * This has also to be fixed in enableVideo()
142 void ActionsManager::fullscreen()
144 bool fs
= var_ToggleBool( THEPL
, "fullscreen" );
145 vout_thread_t
*p_vout
= THEMIM
->getVout();
148 var_SetBool( p_vout
, "fullscreen", fs
);
149 vlc_object_release( p_vout
);
153 void ActionsManager::snapshot()
155 vout_thread_t
*p_vout
= THEMIM
->getVout();
158 var_TriggerCallback( p_vout
, "video-snapshot" );
159 vlc_object_release( p_vout
);
163 void ActionsManager::playlist()
165 if( p_intf
->p_sys
->p_mi
)
166 p_intf
->p_sys
->p_mi
->togglePlaylist();
169 void ActionsManager::record()
171 input_thread_t
*p_input
= THEMIM
->getInput();
174 /* This method won't work fine if the stream can't be cut anywhere */
175 var_ToggleBool( p_input
, "record" );
179 /* 'record' access-filter is not loaded, we open Save dialog */
180 input_item_t
*p_item
= input_GetItem( p_input
);
184 char *psz
= input_item_GetURI( p_item
);
186 THEDP
->streamingDialog( NULL
, qfu(psz
), true );
192 void ActionsManager::frame()
194 input_thread_t
*p_input
= THEMIM
->getInput();
196 var_TriggerCallback( p_input
, "frame-next" );
199 void ActionsManager::toggleMuteAudio()
201 playlist_MuteToggle( THEPL
);
204 void ActionsManager::AudioUp()
206 playlist_VolumeUp( THEPL
, 1, NULL
);
209 void ActionsManager::AudioDown()
211 playlist_VolumeDown( THEPL
, 1, NULL
);
214 void ActionsManager::skipForward()
216 input_thread_t
*p_input
= THEMIM
->getInput();
218 THEMIM
->getIM()->jumpFwd();
221 void ActionsManager::skipBackward()
223 input_thread_t
*p_input
= THEMIM
->getInput();
225 THEMIM
->getIM()->jumpBwd();
228 vlc_renderer_item_t
* ActionsManager::getMatchingRenderer( const QVariant
&obj
, vlc_renderer_item_t
* p_item
)
230 if (!obj
.canConvert
<QVariantHash
>())
232 QVariantHash qvh
= obj
.value
<QVariantHash
>();
233 if (!qvh
.contains( "sout" ))
235 vlc_renderer_item_t
* p_existing
=
236 reinterpret_cast<vlc_renderer_item_t
*>( qvh
["sout"].value
<void*>() );
237 if ( !strcasecmp(vlc_renderer_item_sout( p_existing
),
238 vlc_renderer_item_sout( p_item
) ) )
243 void ActionsManager::onRendererItemAdded(vlc_renderer_item_t
* p_item
)
245 QAction
*firstSeparator
= NULL
;
247 foreach (QAction
* action
, VLCMenuBar::rendererMenu
->actions())
249 if (action
->isSeparator())
251 firstSeparator
= action
;
254 if (getMatchingRenderer( action
->data(), p_item
))
256 vlc_renderer_item_release( p_item
);
257 return; /* we already have this item */
261 QAction
*action
= new QAction( vlc_renderer_item_flags(p_item
) & VLC_RENDERER_CAN_VIDEO
? QIcon( ":/sidebar/movie.svg" ) : QIcon( ":/sidebar/music.svg" ),
262 vlc_renderer_item_name(p_item
), VLCMenuBar::rendererMenu
);
263 action
->setCheckable(true);
266 data
.insert( "sout", QVariant::fromValue( reinterpret_cast<void*>( p_item
) ) );
267 action
->setData( data
);
268 if (firstSeparator
!= NULL
)
270 VLCMenuBar::rendererMenu
->insertAction( firstSeparator
, action
);
271 VLCMenuBar::rendererGroup
->addAction(action
);
275 vlc_renderer_item_release( p_item
);
280 void ActionsManager::onRendererItemRemoved( vlc_renderer_item_t
* p_item
)
282 foreach (QAction
* action
, VLCMenuBar::rendererMenu
->actions())
284 if (action
->isSeparator())
286 vlc_renderer_item_t
*p_existing
= getMatchingRenderer( action
->data(), p_item
);
289 VLCMenuBar::rendererMenu
->removeAction( action
);
290 VLCMenuBar::rendererGroup
->removeAction( action
);
291 vlc_renderer_item_release( p_existing
);
295 // Always release the item as we acquired it before emiting the signal
296 vlc_renderer_item_release( p_item
);
299 void ActionsManager::renderer_event_item_added( vlc_renderer_discovery_t
* p_rd
,
300 vlc_renderer_item_t
*p_item
)
302 ActionsManager
*self
= reinterpret_cast<ActionsManager
*>( p_rd
->owner
.sys
);
303 vlc_renderer_item_hold( p_item
);
304 self
->emit
rendererItemAdded( p_item
);
307 void ActionsManager::renderer_event_item_removed( vlc_renderer_discovery_t
*p_rd
,
308 vlc_renderer_item_t
*p_item
)
310 ActionsManager
*self
= reinterpret_cast<ActionsManager
*>( p_rd
->owner
.sys
);
311 vlc_renderer_item_hold( p_item
);
312 self
->emit
rendererItemRemoved( p_item
);
315 void ActionsManager::StartRendererScan()
317 m_stop_scan_timer
.stop();
322 char **ppsz_longnames
;
324 if( vlc_rd_get_names( THEPL
, &ppsz_names
, &ppsz_longnames
) != VLC_SUCCESS
)
327 struct vlc_renderer_discovery_owner owner
=
330 renderer_event_item_added
,
331 renderer_event_item_removed
,
334 char **ppsz_name
= ppsz_names
, **ppsz_longname
= ppsz_longnames
;
335 for( ; *ppsz_name
; ppsz_name
++, ppsz_longname
++ )
337 msg_Dbg( p_intf
, "starting renderer discovery service %s", *ppsz_longname
);
338 vlc_renderer_discovery_t
* p_rd
= vlc_rd_new( VLC_OBJECT(p_intf
), *ppsz_name
, &owner
);
340 m_rds
.push_back( p_rd
);
342 free( *ppsz_longname
);
345 free( ppsz_longnames
);
349 void ActionsManager::RendererMenuCountdown()
351 m_stop_scan_timer
.start( 20000 );
354 void ActionsManager::StopRendererScan()
356 foreach ( vlc_renderer_discovery_t
* p_rd
, m_rds
)
357 vlc_rd_release( p_rd
);
362 void ActionsManager::RendererSelected( QAction
*selected
)
364 QVariant data
= selected
->data();
365 vlc_renderer_item_t
*p_item
= NULL
;
366 if (data
.canConvert
<QVariantHash
>())
368 QVariantHash hash
= data
.value
<QVariantHash
>();
369 if ( hash
.contains( "sout" ) )
370 p_item
= reinterpret_cast<vlc_renderer_item_t
*>(
371 hash
["sout"].value
<void*>() );
373 // If we failed to convert the action data to a vlc_renderer_item_t,
374 // assume the selected item was invalid, or most likely that "Local" was selected
375 playlist_SetRenderer( THEPL
, p_item
);