]> git.sesse.net Git - vlc/blob - modules/gui/qt4/input_manager.cpp
Add extra debug to chase #934
[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 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 <assert.h>
25
26 #include "qt4.hpp"
27 #include "input_manager.hpp"
28 #include "dialogs_provider.hpp"
29
30 static int ChangeVideo( vlc_object_t *p_this, const char *var, vlc_value_t o,
31                         vlc_value_t n, void *param );
32 static int ChangeAudio( vlc_object_t *p_this, const char *var, vlc_value_t o,
33                         vlc_value_t n, void *param );
34
35 /**********************************************************************
36  * InputManager implementation
37  **********************************************************************/
38
39 InputManager::InputManager( QObject *parent, intf_thread_t *_p_intf) :
40                            QObject( parent ), p_intf( _p_intf )
41 {
42     i_old_playing_status = END_S;
43     p_input = NULL;
44     ON_TIMEOUT( update() );
45 }
46
47 InputManager::~InputManager()
48 {
49 }
50
51 void InputManager::setInput( input_thread_t *_p_input )
52 {
53     p_input = _p_input;
54     emit positionUpdated( 0.0,0,0 );
55     b_had_audio = b_had_video = b_has_audio = b_has_video = false;
56     if( p_input )
57     {
58         vlc_value_t val;
59         var_Change( p_input, "video-es", VLC_VAR_CHOICESCOUNT, &val, NULL );
60         b_has_video = val.i_int > 0;
61         var_Change( p_input, "audio-es", VLC_VAR_CHOICESCOUNT, &val, NULL );
62         b_has_audio = val.i_int > 0;
63         var_AddCallback( p_input, "audio-es", ChangeAudio, this );
64         var_AddCallback( p_input, "video-es", ChangeVideo, this );
65     }
66
67 }
68 void InputManager::delInput()
69 {
70     if( p_input )
71     {
72         var_DelCallback( p_input, "audio-es", ChangeAudio, this );
73         var_DelCallback( p_input, "video-es", ChangeVideo, this );
74     }
75 }
76
77 void InputManager::update()
78 {
79     /// \todo Emit the signals only if it changed
80     if( !p_input ) return;
81
82     if( p_input->b_dead || p_input->b_die )
83     {
84         emit positionUpdated( 0.0, 0, 0 );
85         msg_Dbg( p_intf, "*********** NAV 0");
86         emit navigationChanged( 0 );
87         emit statusChanged( 0 ); // 0 = STOPPED, 1 = PLAY, 2 = PAUSE
88     }
89
90     if( !b_had_audio && b_has_audio )
91         emit audioStarted();
92     if( !b_had_video && b_has_video )
93         emit videoStarted();
94
95     /* Update position */
96     mtime_t i_length, i_time;
97     float f_pos;
98     i_length = var_GetTime( p_input, "length" ) / 1000000;
99     i_time = var_GetTime( p_input, "time") / 1000000;
100     f_pos = var_GetFloat( p_input, "position" );
101     emit positionUpdated( f_pos, i_time, i_length );
102
103     /* Update disc status */
104     vlc_value_t val; val.i_int = 0;
105     var_Change( p_input, "title", VLC_VAR_CHOICESCOUNT, &val, NULL );
106     if( val.i_int > 0 )
107     {
108         val.i_int = 0;
109         var_Change( p_input, "chapter", VLC_VAR_CHOICESCOUNT, &val, NULL );
110         if( val.i_int > 0 )
111         {
112             msg_Dbg( p_intf, "******* CHAPTER");
113             emit navigationChanged( 1 ); // 1 = chapter, 2 = title, 0 = NO
114         }
115         else
116         {
117             msg_Dbg( p_intf, "******* TITLE");
118             emit navigationChanged( 2 );
119         }
120     }
121     else
122     {
123         msg_Dbg( p_intf, "******** NONE" );
124         emit navigationChanged( 0 );
125     }
126
127     /* Update text */
128     QString text;
129     if( input_GetItem(p_input)->p_meta &&
130         input_GetItem(p_input)->p_meta->psz_nowplaying &&
131         *input_GetItem(p_input)->p_meta->psz_nowplaying )
132     {
133         text.sprintf( "%s - %s",
134                   input_GetItem(p_input)->p_meta->psz_nowplaying,
135                   input_GetItem(p_input)->psz_name );
136     }
137     else
138     {
139         text.sprintf( "%s", input_GetItem(p_input)->psz_name );
140     }
141     emit nameChanged( text );
142
143     /* Update playing status */
144     var_Get( p_input, "state", &val );
145     val.i_int = val.i_int == PAUSE_S ? PAUSE_S : PLAYING_S;
146     if( i_old_playing_status != val.i_int )
147     {
148         i_old_playing_status = val.i_int;
149         emit statusChanged(  val.i_int == PAUSE_S ? PAUSE_S : PLAYING_S );
150     }
151 }
152
153 void InputManager::sliderUpdate( float new_pos )
154 {
155     if( hasInput() )
156         var_SetFloat( p_input, "position", new_pos );
157 }
158
159 void InputManager::togglePlayPause()
160 {
161     vlc_value_t state;
162     var_Get( p_input, "state", &state );
163     if( state.i_int != PAUSE_S )
164     {
165         /* A stream is being played, pause it */
166         state.i_int = PAUSE_S;
167     }
168     else
169     {
170         /* Stream is paused, resume it */
171         state.i_int = PLAYING_S;
172     }
173     var_Set( p_input, "state", state );
174     emit statusChanged( state.i_int );
175 }
176
177 void InputManager::sectionPrev()
178 {
179     if( hasInput() )
180     {
181         int i_type = var_Type( p_input, "prev-chapter" );
182         vlc_value_t val; val.b_bool = VLC_TRUE;
183         var_Set( p_input, (i_type & VLC_VAR_TYPE) != 0 ?
184                             "prev-chapter":"prev-title", val );
185     }
186 }
187
188 void InputManager::sectionNext()
189 {
190     if( hasInput() )
191     {
192         int i_type = var_Type( p_input, "prev-chapter" );
193         vlc_value_t val; val.b_bool = VLC_TRUE;
194         var_Set( p_input, (i_type & VLC_VAR_TYPE) != 0 ?
195                             "next-chapter":"next-title", val );
196     }
197 }
198
199 void InputManager::sectionMenu()
200 {
201     if( hasInput() )
202         var_SetInteger( p_input, "title 0", 2);
203 }
204
205 void InputManager::slower()
206 {
207     if( hasInput() )
208         var_SetVoid( p_input, "rate-slower" );
209 }
210
211 void InputManager::faster()
212 {
213     if( hasInput() )
214         var_SetVoid( p_input, "rate-faster" );
215 }
216
217 void InputManager::normalRate()
218 {
219     if( hasInput() )
220         var_SetInteger( p_input, "rate", INPUT_RATE_DEFAULT );
221 }
222
223 /**********************************************************************
224  * MainInputManager implementation. Wrap an input manager and
225  * take care of updating the main playlist input
226  **********************************************************************/
227 MainInputManager * MainInputManager::instance = NULL;
228
229 MainInputManager::MainInputManager( intf_thread_t *_p_intf ) : QObject(NULL),
230                                                 p_intf( _p_intf )
231 {
232     p_input = NULL;
233     im = new InputManager( this, p_intf );
234     ON_TIMEOUT( updateInput() );
235     /* Warn our embedded IM about input changes */
236     CONNECT( this, inputChanged( input_thread_t * ),
237              im,   setInput( input_thread_t * ) );
238 }
239
240 MainInputManager::~MainInputManager()
241 {
242     if( p_input ) vlc_object_release( p_input );
243 }
244
245 void MainInputManager::updateInput()
246 {
247     vlc_mutex_lock( &p_intf->change_lock );
248     if( p_input && p_input->b_dead )
249     {
250         vlc_object_release( p_input );
251         getIM()->delInput();
252         p_input = NULL;
253         emit inputChanged( NULL );
254     }
255
256     if( !p_input )
257     {
258         QPL_LOCK;
259         p_input = THEPL->p_input;
260         if( p_input )
261         {
262             vlc_object_yield( p_input );
263             emit inputChanged( p_input );
264         }
265         QPL_UNLOCK;
266     }
267     vlc_mutex_unlock( &p_intf->change_lock );
268 }
269
270 void MainInputManager::togglePlayPause()
271 {
272     if( p_input == NULL )
273     {
274         playlist_Play( THEPL );
275         return;
276     }
277     getIM()->togglePlayPause();
278 }
279
280
281 static int ChangeAudio( vlc_object_t *p_this, const char *var, vlc_value_t o,
282                         vlc_value_t n, void *param )
283 {
284     InputManager *im = (InputManager*)param;
285     im->b_has_audio = true;
286     return 0;
287 }
288
289 static int ChangeVideo( vlc_object_t *p_this, const char *var, vlc_value_t o,
290                         vlc_value_t n, void *param )
291 {
292     InputManager *im = (InputManager*)param;
293     im->b_has_video = true;
294     return 0;
295 }