]> git.sesse.net Git - vlc/blob - modules/gui/wxwidgets/input_manager.cpp
Merge back branch 0.8.6-playlist-vlm to trunk.
[vlc] / modules / gui / wxwidgets / input_manager.cpp
1 /*****************************************************************************
2  * slider_manager.cpp : Manage an input slider
3  *****************************************************************************
4  * Copyright (C) 2000-2005 the VideoLAN team
5  * $Id$
6  *
7  * Authors: Gildas Bazin <gbazin@videolan.org>
8  *          ClĂ©ment Stenac <zorglub@videolan.org>
9  *
10  * This program is free software; you can redistribute it and/or modify
11  * it under the terms of the GNU General Public License as published by
12  * the Free Software Foundation; either version 2 of the License, or
13  * (at your option) any later version.
14  *
15  * This program is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18  * GNU General Public License for more details.
19  *
20  * You should have received a copy of the GNU General Public License
21  * along with this program; if not, write to the Free Software
22  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
23  *****************************************************************************/
24
25 #include "input_manager.hpp"
26 #include "interface.hpp"
27 #include "video.hpp"
28
29 #include <vlc_meta.h>
30
31 /* include the toolbar graphics */
32 #include "bitmaps/prev.xpm"
33 #include "bitmaps/next.xpm"
34 #include "bitmaps/playlist.xpm"
35
36 /* IDs for the controls */
37 enum
38 {
39     SliderScroll_Event = wxID_HIGHEST,
40
41     DiscMenu_Event,
42     DiscPrev_Event,
43     DiscNext_Event
44 };
45
46 BEGIN_EVENT_TABLE(InputManager, wxPanel)
47     /* Slider events */
48     EVT_COMMAND_SCROLL(SliderScroll_Event, InputManager::OnSliderUpdate)
49
50     /* Disc Buttons events */
51     EVT_BUTTON(DiscMenu_Event, InputManager::OnDiscMenu)
52     EVT_BUTTON(DiscPrev_Event, InputManager::OnDiscPrev)
53     EVT_BUTTON(DiscNext_Event, InputManager::OnDiscNext)
54
55 END_EVENT_TABLE()
56
57 #define STATUS_STOP 0
58 #define STATUS_PLAYING 1
59 #define STATUS_PAUSE 2
60
61 /*****************************************************************************
62  * Constructor.
63  *****************************************************************************/
64 InputManager::InputManager( intf_thread_t *_p_intf, Interface *_p_main_intf,
65                             wxWindow *p_parent )
66   : wxPanel( p_parent )
67 {
68     p_intf = _p_intf;
69     p_main_intf = _p_main_intf;
70     p_input = NULL;
71     i_old_playing_status = STATUS_STOP;
72     i_old_rate = INPUT_RATE_DEFAULT;
73     b_slider_free = VLC_TRUE;
74     i_input_hide_delay = 0;
75
76     /* Create slider */
77     slider = new wxSlider( this, SliderScroll_Event, 0, 0, SLIDER_MAX_POS );
78
79     /* Create disc buttons */
80     disc_frame = new wxPanel( this );
81
82     disc_menu_button = new wxBitmapButton( disc_frame, DiscMenu_Event,
83                                            wxBitmap( playlist_xpm ) );
84     disc_prev_button = new wxBitmapButton( disc_frame, DiscPrev_Event,
85                                            wxBitmap( prev_xpm ) );
86     disc_next_button = new wxBitmapButton( disc_frame, DiscNext_Event,
87                                            wxBitmap( next_xpm ) );
88
89     disc_sizer = new wxBoxSizer( wxHORIZONTAL );
90     disc_sizer->Add( disc_menu_button, 1, wxEXPAND | wxLEFT | wxRIGHT, 1 );
91     disc_sizer->Add( disc_prev_button, 1, wxEXPAND | wxLEFT | wxRIGHT, 1 );
92     disc_sizer->Add( disc_next_button, 1, wxEXPAND | wxLEFT | wxRIGHT, 1 );
93     disc_frame->SetSizer( disc_sizer );
94     disc_sizer->Layout();
95
96     /* Add everything to the panel */
97     sizer = new wxBoxSizer( wxHORIZONTAL );
98     SetSizer( sizer );
99     sizer->Add( slider, 1, wxEXPAND | wxALL, 5 );
100     sizer->Add( disc_frame, 0, wxALL, 2 );
101
102     /* Hide by default */
103     sizer->Hide( disc_frame );
104     sizer->Hide( slider );
105
106     sizer->Layout();
107     Fit();
108 }
109
110 InputManager::~InputManager()
111 {
112     vlc_mutex_lock( &p_intf->change_lock );
113     if( p_intf->p_sys->p_input ) vlc_object_release( p_intf->p_sys->p_input );
114     p_intf->p_sys->p_input = NULL;
115     vlc_mutex_unlock( &p_intf->change_lock );
116 }
117
118 /*****************************************************************************
119  * Public methods.
120  *****************************************************************************/
121 vlc_bool_t InputManager::IsPlaying()
122 {
123     return (p_input && !p_input->b_die);
124 }
125
126 /*****************************************************************************
127  * Private methods.
128  *****************************************************************************/
129 void InputManager::UpdateInput()
130 {
131     playlist_t *p_playlist =
132         (playlist_t *)vlc_object_find( p_intf, VLC_OBJECT_PLAYLIST,
133                                        FIND_ANYWHERE );
134     if( p_playlist != NULL )
135     {
136         LockPlaylist( p_intf->p_sys, p_playlist );
137         p_input = p_intf->p_sys->p_input = p_playlist->p_input;
138         if( p_intf->p_sys->p_input )
139              vlc_object_yield( p_intf->p_sys->p_input );
140         UnlockPlaylist( p_intf->p_sys, p_playlist );
141         vlc_object_release( p_playlist );
142     }
143 }
144
145 void InputManager::UpdateNowPlaying()
146 {
147     char *psz_now_playing = p_input->input.p_item->p_meta->psz_nowplaying ?
148                     strdup( p_input->input.p_item->p_meta->psz_nowplaying ):
149                     strdup( "" );
150     if( psz_now_playing && *psz_now_playing )
151     {
152         p_main_intf->statusbar->SetStatusText(
153                     wxString(wxU(psz_now_playing)) + wxT( " - " ) +
154                     wxU(p_input->input.p_item->psz_name), 2 );
155     }
156     else
157     {
158         p_main_intf->statusbar->SetStatusText(
159                    wxU(p_input->input.p_item->psz_name), 2 );
160     }
161     free( psz_now_playing );
162 }
163
164 void InputManager::UpdateButtons( vlc_bool_t b_play )
165 {
166     if( !b_play )
167     {
168         if( i_old_playing_status == STATUS_STOP ) return;
169
170         i_old_playing_status = STATUS_STOP;
171         p_main_intf->TogglePlayButton( PAUSE_S );
172         p_main_intf->statusbar->SetStatusText( wxT(""), 0 );
173         p_main_intf->statusbar->SetStatusText( wxT(""), 2 );
174
175 /* wxCocoa pretends to support this, but at least 2.6.x doesn't */
176 #ifndef __APPLE__
177 #ifdef wxHAS_TASK_BAR_ICON
178         if( p_main_intf->p_systray )
179         {
180             p_main_intf->p_systray->UpdateTooltip(
181                 wxString(wxT("VLC media player - ")) + wxU(_("Stopped")) );
182         }
183 #endif
184 #endif
185
186         return;
187     }
188
189     /* Manage Playing status */
190     vlc_value_t val;
191     var_Get( p_input, "state", &val );
192     val.i_int = val.i_int == PAUSE_S ? STATUS_PAUSE : STATUS_PLAYING;
193     if( i_old_playing_status != val.i_int )
194     {
195         i_old_playing_status = val.i_int;
196         p_main_intf->TogglePlayButton( val.i_int == STATUS_PAUSE ?
197                                        PAUSE_S : PLAYING_S );
198
199 /* wxCocoa pretends to support this, but at least 2.6.x doesn't */
200 #ifndef __APPLE__
201 #ifdef wxHAS_TASK_BAR_ICON
202         if( p_main_intf->p_systray )
203         {
204             p_main_intf->p_systray->UpdateTooltip(
205                 wxU(p_input->input.p_item->psz_name) + wxString(wxT(" - ")) +
206                 (val.i_int == PAUSE_S ? wxU(_("Paused")) : wxU(_("Playing"))));
207         }
208 #endif
209 #endif
210     }
211 }
212
213 void InputManager::UpdateDiscButtons()
214 {
215     vlc_value_t val;
216     var_Change( p_input, "title", VLC_VAR_CHOICESCOUNT, &val, NULL );
217     if( val.i_int > 0 && !disc_frame->IsShown() )
218     {
219         vlc_value_t val;
220
221         #define HELP_MENU N_("Menu")
222         #define HELP_PCH N_("Previous chapter")
223         #define HELP_NCH N_("Next chapter")
224         #define HELP_PTR N_("Previous track")
225         #define HELP_NTR N_("Next track")
226
227         var_Change( p_input, "chapter", VLC_VAR_CHOICESCOUNT, &val, NULL );
228
229         if( val.i_int > 0 )
230         {
231             disc_menu_button->Show();
232             disc_sizer->Show( disc_menu_button );
233             disc_sizer->Layout();
234             disc_sizer->Fit( disc_frame );
235             disc_menu_button->SetToolTip( wxU(_( HELP_MENU ) ) );
236             disc_prev_button->SetToolTip( wxU(_( HELP_PCH ) ) );
237             disc_next_button->SetToolTip( wxU(_( HELP_NCH ) ) );
238         }
239         else
240         {
241             disc_menu_button->Hide();
242             disc_sizer->Hide( disc_menu_button );
243             disc_prev_button->SetToolTip( wxU(_( HELP_PTR ) ) );
244             disc_next_button->SetToolTip( wxU(_( HELP_NTR ) ) );
245         }
246
247         ShowDiscFrame();
248     }
249     else if( val.i_int == 0 && disc_frame->IsShown() )
250     {
251         HideDiscFrame();
252     }
253 }
254
255 void InputManager::HideSlider()
256 {
257     ShowSlider( false );
258 }
259
260 void InputManager::HideDiscFrame()
261 {
262     ShowDiscFrame( false );
263 }
264
265 void InputManager::UpdateTime()
266 {
267     char psz_time[ MSTRTIME_MAX_SIZE ], psz_total[ MSTRTIME_MAX_SIZE ];
268     mtime_t i_seconds;
269
270     i_seconds = var_GetTime( p_intf->p_sys->p_input, "length" ) / 1000000;
271     secstotimestr( psz_total, i_seconds );
272
273     i_seconds = var_GetTime( p_intf->p_sys->p_input, "time" ) / 1000000;
274     secstotimestr( psz_time, i_seconds );
275
276     p_main_intf->statusbar->SetStatusText(
277         wxU(psz_time) + wxString(wxT(" / ")) +wxU(psz_total), 0 );
278 }
279
280 void InputManager::Update()
281 {
282     /* Update the input */
283     if( p_input == NULL )
284     {
285         UpdateInput();
286
287         if( p_input )
288         {
289             slider->SetValue( 0 );
290         }
291         else if( !i_input_hide_delay )
292         {
293             i_input_hide_delay = mdate() + 200000;
294         }
295         else if( i_input_hide_delay < mdate() )
296         {
297             if( disc_frame->IsShown() ) HideDiscFrame();
298             if( slider->IsShown() ) HideSlider();
299             i_input_hide_delay = 0;
300         }
301     }
302     else if( p_input->b_dead )
303     {
304         UpdateButtons( VLC_FALSE );
305         vlc_object_release( p_input );
306         p_input = NULL;
307     }
308     else
309     {
310         i_input_hide_delay = 0;
311     }
312
313     if( p_input && !p_input->b_die )
314     {
315         vlc_value_t pos, len;
316
317         UpdateTime();
318         UpdateButtons( VLC_TRUE );
319         UpdateNowPlaying();
320         UpdateDiscButtons();
321
322         /* Really manage the slider */
323         var_Get( p_input, "position", &pos );
324         var_Get( p_input, "length", &len );
325
326         if( pos.f_float > 0 && !slider->IsShown() ) ShowSlider();
327         else if(  pos.f_float <= 0 &&  slider->IsShown() ) HideSlider();
328
329         /* Update the slider if the user isn't dragging it. */
330         if( slider->IsShown() && b_slider_free )
331         {
332             i_slider_pos = (int)(SLIDER_MAX_POS * pos.f_float);
333             slider->SetValue( i_slider_pos );
334         }
335
336         /* Manage Speed status */
337         vlc_value_t val;
338         var_Get( p_input, "rate", &val );
339         if( i_old_rate != val.i_int )
340         {
341             p_main_intf->statusbar->SetStatusText(
342                 wxString::Format(wxT("x%.2f"),
343                 (float)INPUT_RATE_DEFAULT / val.i_int ), 1 );
344             i_old_rate = val.i_int;
345         }
346     }
347 }
348
349 /*****************************************************************************
350  * Event Handlers.
351  *****************************************************************************/
352 void InputManager::OnDiscMenu( wxCommandEvent& WXUNUSED(event) )
353 {
354     input_thread_t *p_input =
355         (input_thread_t *)vlc_object_find( p_intf, VLC_OBJECT_INPUT,
356                                            FIND_ANYWHERE );
357     if( p_input )
358     {
359         vlc_value_t val; val.i_int = 2;
360
361         var_Set( p_input, "title  0", val);
362         vlc_object_release( p_input );
363     }
364 }
365
366 void InputManager::OnDiscPrev( wxCommandEvent& WXUNUSED(event) )
367 {
368     input_thread_t *p_input =
369         (input_thread_t *)vlc_object_find( p_intf, VLC_OBJECT_INPUT,
370                                            FIND_ANYWHERE );
371     if( p_input )
372     {
373         int i_type = var_Type( p_input, "prev-chapter" );
374         vlc_value_t val; val.b_bool = VLC_TRUE;
375
376         var_Set( p_input, ( i_type & VLC_VAR_TYPE ) != 0 ?
377                  "prev-chapter" : "prev-title", val );
378
379         vlc_object_release( p_input );
380     }
381 }
382
383 void InputManager::OnDiscNext( wxCommandEvent& WXUNUSED(event) )
384 {
385     input_thread_t *p_input =
386         (input_thread_t *)vlc_object_find( p_intf, VLC_OBJECT_INPUT,
387                                            FIND_ANYWHERE );
388     if( p_input )
389     {
390         int i_type = var_Type( p_input, "next-chapter" );
391         vlc_value_t val; val.b_bool = VLC_TRUE;
392
393         var_Set( p_input, ( i_type & VLC_VAR_TYPE ) != 0 ?
394                  "next-chapter" : "next-title", val );
395
396         vlc_object_release( p_input );
397     }
398 }
399
400 void InputManager::OnSliderUpdate( wxScrollEvent& event )
401 {
402     vlc_mutex_lock( &p_intf->change_lock );
403
404 #ifdef WIN32
405     if( event.GetEventType() == wxEVT_SCROLL_THUMBRELEASE
406         || event.GetEventType() == wxEVT_SCROLL_ENDSCROLL )
407     {
408 #endif
409         if( i_slider_pos != event.GetPosition() && p_intf->p_sys->p_input )
410         {
411             vlc_value_t pos;
412             pos.f_float = (float)event.GetPosition() / (float)SLIDER_MAX_POS;
413             var_Set( p_intf->p_sys->p_input, "position", pos );
414         }
415
416 #ifdef WIN32
417         b_slider_free = VLC_TRUE;
418     }
419     else
420     {
421         b_slider_free = VLC_FALSE;
422         if( p_intf->p_sys->p_input ) UpdateTime();
423     }
424 #endif
425
426 #undef WIN32
427     vlc_mutex_unlock( &p_intf->change_lock );
428 }
429
430 void InputManager::ShowSlider( bool show )
431 {
432     if( !!show == !!slider->IsShown() ) return;
433
434     UpdateVideoWindow( p_intf, p_main_intf->video_window );
435
436     sizer->Show( slider, show );
437     sizer->Layout();
438
439     wxCommandEvent intf_event( wxEVT_INTF, 0 );
440     p_main_intf->AddPendingEvent( intf_event );
441 }
442
443 void InputManager::ShowDiscFrame( bool show )
444 {
445     if( !!show == !!disc_frame->IsShown() ) return;
446
447     UpdateVideoWindow( p_intf, p_main_intf->video_window );
448
449     sizer->Show( disc_frame, show );
450     sizer->Layout();
451
452     wxCommandEvent intf_event( wxEVT_INTF, 0 );
453     p_main_intf->AddPendingEvent( intf_event );
454 }