]> git.sesse.net Git - vlc/blob - modules/gui/qt4/input_manager.cpp
f08fc7e992fc2e679c32f4961e021cbb072b678f
[vlc] / modules / gui / qt4 / input_manager.cpp
1 /*****************************************************************************
2  * input_manager.cpp : Manage an input and interact with its GUI elements
3  ****************************************************************************
4  * Copyright (C) 2006-2007 the VideoLAN team
5  * $Id$
6  *
7  * Authors: ClĂ©ment Stenac <zorglub@videolan.org>
8  *
9  * This program is free software; you can redistribute it and/or modify
10  * it under the terms of the GNU General Public License as published by
11  * the Free Software Foundation; either version 2 of the License, or
12  * (at your option) any later version.
13  *
14  * This program is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17  * GNU General Public License for more details.
18  *
19  * You should have received a copy of the GNU General Public License
20  * along with this program; if not, write to the Free Software
21  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22  *****************************************************************************/
23
24 #include "qt4.hpp"
25 #include "input_manager.hpp"
26 #include "dialogs_provider.hpp"
27
28 static int ChangeVideo( vlc_object_t *p_this, const char *var, vlc_value_t o,
29                         vlc_value_t n, void *param );
30 static int ChangeAudio( vlc_object_t *p_this, const char *var, vlc_value_t o,
31                         vlc_value_t n, void *param );
32
33 /**********************************************************************
34  * InputManager implementation
35  **********************************************************************/
36
37 InputManager::InputManager( QObject *parent, intf_thread_t *_p_intf) :
38                            QObject( parent ), p_intf( _p_intf )
39 {
40     i_old_playing_status = END_S;
41     old_name="";
42     p_input = NULL;
43     i_rate = 0;
44     ON_TIMEOUT( update() );
45 }
46
47 InputManager::~InputManager()
48 {
49     delInput();
50 }
51
52 void InputManager::setInput( input_thread_t *_p_input )
53 {
54     delInput();
55     p_input = _p_input;
56     emit positionUpdated( 0.0,0,0 );
57     b_had_audio = b_had_video = b_has_audio = b_has_video = false;
58     if( p_input )
59     {
60         vlc_object_yield( p_input );
61         vlc_value_t val;
62         var_Change( p_input, "video-es", VLC_VAR_CHOICESCOUNT, &val, NULL );
63         b_has_video = val.i_int > 0;
64         var_Change( p_input, "audio-es", VLC_VAR_CHOICESCOUNT, &val, NULL );
65         b_has_audio = val.i_int > 0;
66         var_AddCallback( p_input, "audio-es", ChangeAudio, this );
67         var_AddCallback( p_input, "video-es", ChangeVideo, this );
68     }
69 }
70
71 void InputManager::delInput()
72 {
73     if( p_input )
74     {
75         var_DelCallback( p_input, "audio-es", ChangeAudio, this );
76         var_DelCallback( p_input, "video-es", ChangeVideo, this );
77         vlc_object_release( p_input );
78         p_input = NULL;
79     }
80 }
81
82 void InputManager::update()
83 {
84     /// \todo Emit the signals only if it changed
85     if( !p_input ) return;
86
87     if( p_input->b_dead || p_input->b_die )
88     {
89         emit positionUpdated( 0.0, 0, 0 );
90         emit navigationChanged( 0 );
91         i_old_playing_status = 0;
92         emit statusChanged( END_S ); // see vlc_input.h, input_state_e enum
93         delInput();
94         return;
95     }
96
97     /* Update position */
98     mtime_t i_length, i_time;
99     float f_pos;
100     i_length = var_GetTime( p_input, "length" ) / 1000000;
101     i_time = var_GetTime( p_input, "time") / 1000000;
102     f_pos = var_GetFloat( p_input, "position" );
103     emit positionUpdated( f_pos, i_time, i_length );
104     
105     int i_new_rate = var_GetInteger( p_input, "rate");
106     if( i_new_rate != i_rate )
107     {
108         i_rate = i_new_rate;
109         /* Update rate */
110         emit rateChanged( i_rate );
111     }
112
113     /* Update navigation status */
114     vlc_value_t val; val.i_int = 0;
115     var_Change( p_input, "title", VLC_VAR_CHOICESCOUNT, &val, NULL );
116     if( val.i_int > 0 )
117     {
118         val.i_int = 0;
119         var_Change( p_input, "chapter", VLC_VAR_CHOICESCOUNT, &val, NULL );
120         if( val.i_int > 0 )
121         {
122             emit navigationChanged( 1 ); // 1 = chapter, 2 = title, 0 = NO
123         }
124         else
125         {
126             emit navigationChanged( 2 );
127         }
128     }
129     else
130     {
131         emit navigationChanged( 0 );
132     }
133
134     /* Update text */
135     QString text;
136     char *psz_name = input_item_GetTitle( input_GetItem( p_input ) );
137     char *psz_nowplaying =
138         input_item_GetNowPlaying( input_GetItem( p_input ) );
139     char *psz_artist = input_item_GetArtist( input_GetItem( p_input ) );
140     if( EMPTY_STR( psz_name ) )
141     {
142         free( psz_name );
143         psz_name = input_item_GetName( input_GetItem( p_input ) );
144     }
145     if( !EMPTY_STR( psz_nowplaying ) )
146     {
147         text.sprintf( "%s - %s", psz_nowplaying, psz_name );
148     }
149     else if( !EMPTY_STR( psz_artist ) )
150     {
151         text.sprintf( "%s - %s", psz_artist, psz_name );
152     }
153     else
154     {
155         text.sprintf( "%s", psz_name );
156     }
157     free( psz_name );
158     free( psz_nowplaying );
159     free( psz_artist );
160     if( old_name != text )
161     {
162         emit nameChanged( text );
163         old_name=text;
164     }
165     /* Update playing status */
166     var_Get( p_input, "state", &val );
167     val.i_int = val.i_int == PAUSE_S ? PAUSE_S : PLAYING_S;
168     if( i_old_playing_status != val.i_int )
169     {
170         i_old_playing_status = val.i_int;
171         emit statusChanged(  val.i_int == PAUSE_S ? PAUSE_S : PLAYING_S );
172     }
173 }
174
175 void InputManager::sliderUpdate( float new_pos )
176 {
177     if( hasInput() )
178         var_SetFloat( p_input, "position", new_pos );
179 }
180
181 void InputManager::togglePlayPause()
182 {
183     vlc_value_t state;
184     var_Get( p_input, "state", &state );
185     if( state.i_int != PAUSE_S )
186     {
187         /* A stream is being played, pause it */
188         state.i_int = PAUSE_S;
189     }
190     else
191     {
192         /* Stream is paused, resume it */
193         state.i_int = PLAYING_S;
194     }
195     var_Set( p_input, "state", state );
196     emit statusChanged( state.i_int );
197 }
198
199 void InputManager::sectionPrev()
200 {
201     if( hasInput() )
202     {
203         int i_type = var_Type( p_input, "next-chapter" );
204         vlc_value_t val; val.b_bool = VLC_TRUE;
205         var_Set( p_input, (i_type & VLC_VAR_TYPE) != 0 ?
206                             "prev-chapter":"prev-title", val );
207     }
208 }
209
210 void InputManager::sectionNext()
211 {
212     if( hasInput() )
213     {
214         int i_type = var_Type( p_input, "next-chapter" );
215         vlc_value_t val; val.b_bool = VLC_TRUE;
216         var_Set( p_input, (i_type & VLC_VAR_TYPE) != 0 ?
217                             "next-chapter":"next-title", val );
218     }
219 }
220
221 void InputManager::sectionMenu()
222 {
223     if( hasInput() )
224         var_SetInteger( p_input, "title 0", 2 );
225 }
226
227 void InputManager::slower()
228 {
229     if( hasInput() )
230         var_SetVoid( p_input, "rate-slower" );
231 }
232
233 void InputManager::faster()
234 {
235     if( hasInput() )
236         var_SetVoid( p_input, "rate-faster" );
237 }
238
239 void InputManager::normalRate()
240 {
241     if( hasInput() )
242         var_SetInteger( p_input, "rate", INPUT_RATE_DEFAULT );
243 }
244
245 void InputManager::setRate( int new_rate )
246 {
247     if( hasInput() )
248         var_SetInteger( p_input, "rate", new_rate );
249 }
250
251 /**********************************************************************
252  * MainInputManager implementation. Wrap an input manager and
253  * take care of updating the main playlist input
254  **********************************************************************/
255 MainInputManager * MainInputManager::instance = NULL;
256
257 MainInputManager::MainInputManager( intf_thread_t *_p_intf ) : QObject(NULL),
258                                                 p_intf( _p_intf )
259 {
260     p_input = NULL;
261     im = new InputManager( this, p_intf );
262     ON_TIMEOUT( updateInput() );
263     /* Warn our embedded IM about input changes */
264     CONNECT( this, inputChanged( input_thread_t * ),
265              im,   setInput( input_thread_t * ) );
266 }
267
268 MainInputManager::~MainInputManager()
269 {
270     if( p_input ) vlc_object_release( p_input );
271 }
272
273 void MainInputManager::updateInput()
274 {
275     if( VLC_OBJECT_INTF == p_intf->i_object_type )
276     {
277         vlc_mutex_lock( &p_intf->change_lock );
278         if( p_input && p_input->b_dead )
279         {
280             vlc_object_release( p_input );
281             getIM()->delInput();
282             p_input = NULL;
283             emit inputChanged( NULL );
284         }
285
286         if( !p_input )
287         {
288             QPL_LOCK;
289             p_input = THEPL->p_input;
290             if( p_input )
291             {
292                 vlc_object_yield( p_input );
293                 emit inputChanged( p_input );
294             }
295             QPL_UNLOCK;
296         }
297         vlc_mutex_unlock( &p_intf->change_lock );
298     }
299     else {
300         /* we are working as a dialogs provider */
301         playlist_t *p_playlist = (playlist_t *) vlc_object_find( p_intf,
302                                        VLC_OBJECT_PLAYLIST, FIND_ANYWHERE );
303         if( p_playlist )
304         {
305             p_input = p_playlist->p_input;
306             emit inputChanged( p_input );
307         }
308     }
309 }
310
311 void MainInputManager::stop()
312 {
313    playlist_Stop( THEPL );
314 }
315
316 void MainInputManager::next()
317 {
318    playlist_Next( THEPL );
319 }
320
321 void MainInputManager::prev()
322 {
323    playlist_Prev( THEPL );
324 }
325
326 void MainInputManager::togglePlayPause()
327 {
328     if( p_input == NULL )
329     {
330         playlist_Play( THEPL );
331         return;
332     }
333     getIM()->togglePlayPause();
334 }
335
336 static int ChangeAudio( vlc_object_t *p_this, const char *var, vlc_value_t o,
337                         vlc_value_t n, void *param )
338 {
339     InputManager *im = (InputManager*)param;
340     im->b_has_audio = true;
341     return VLC_SUCCESS;
342 }
343
344 static int ChangeVideo( vlc_object_t *p_this, const char *var, vlc_value_t o,
345                         vlc_value_t n, void *param )
346 {
347     InputManager *im = (InputManager*)param;
348     im->b_has_video = true;
349     return VLC_SUCCESS;
350 }