1 /*****************************************************************************
2 * InterfaceWindow.cpp: beos interface
3 *****************************************************************************
4 * Copyright (C) 1999, 2000, 2001 VideoLAN
5 * $Id: InterfaceWindow.cpp,v 1.1 2001/06/02 09:42:26 tcastley Exp $
7 * Authors: Jean-Marc Dressler <polux@via.ecp.fr>
8 * Samuel Hocevar <sam@zoy.org>
9 * Tony Castley <tony@castley.net>
10 * Richard Shepherd <richard@rshepherd.demon.co.uk>
12 * This program is free software; you can redistribute it and/or modify
13 * it under the terms of the GNU General Public License as published by
14 * the Free Software Foundation; either version 2 of the License, or
15 * (at your option) any later version.
17 * This program is distributed in the hope that it will be useful,
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 * GNU General Public License for more details.
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, write to the Free Software
24 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111, USA.
25 *****************************************************************************/
30 #include <kernel/OS.h>
31 #include <InterfaceKit.h>
33 #include <StorageKit.h>
36 #include <scsiprobe_driver.h>
47 #include "stream_control.h"
48 #include "input_ext-intf.h"
49 #include "interface.h"
51 #include "intf_playlist.h"
52 #include "audio_output.h"
55 /* BeOS interface headers */
57 #include "InterfaceWindow.h"
59 #include "TransportButton.h"
61 /*****************************************************************************
63 *****************************************************************************/
65 InterfaceWindow::InterfaceWindow( BRect frame, const char *name,
66 intf_thread_t *p_interface )
67 : BWindow( frame, name, B_FLOATING_WINDOW_LOOK, B_NORMAL_WINDOW_FEEL,
68 B_NOT_RESIZABLE | B_NOT_ZOOMABLE | B_WILL_ACCEPT_FIRST_CLICK
69 | B_ASYNCHRONOUS_CONTROLS )
77 SetName( "interface" );
78 SetTitle(VOUT_TITLE " (BeOS interface)");
79 BRect rect(0, 0, 0, 0);
82 menu_bar = new BMenuBar(rect, "main menu");
91 menu_bar->AddItem( mFile = new BMenu( "File" ) );
92 menu_bar->ResizeToPreferred();
93 mFile->AddItem( mItem = new BMenuItem( "Open File" B_UTF8_ELLIPSIS,
94 new BMessage(OPEN_FILE), 'O') );
95 cd_menu = new CDMenu( "Open Disc" );
96 mFile->AddItem( cd_menu );
97 mFile->AddSeparatorItem();
98 mFile->AddItem( mItem = new BMenuItem( "About" B_UTF8_ELLIPSIS,
99 new BMessage(B_ABOUT_REQUESTED), 'A') );
100 mItem->SetTarget( be_app );
101 mFile->AddItem(mItem = new BMenuItem( "Quit",
102 new BMessage(B_QUIT_REQUESTED), 'Q') );
104 menu_bar->AddItem ( mAudio = new BMenu( "Audio" ) );
105 menu_bar->ResizeToPreferred();
106 mAudio->AddItem( new LanguageMenu( "Language", AUDIO_ES, p_intf ) );
107 mAudio->AddItem( new LanguageMenu( "Subtitles", SPU_ES, p_intf ) );
111 rect.top += menu_bar->Bounds().IntegerHeight() + 1;
114 p_view = new BBox( rect, NULL, B_FOLLOW_ALL, B_WILL_DRAW, B_PLAIN_BORDER );
115 p_view->SetViewColor( ui_color(B_PANEL_BACKGROUND_COLOR) );
119 ButtonRect.SetLeftTop(BPoint(xStart, yStart));
120 ButtonRect.SetRightBottom(ButtonRect.LeftTop() + kSkipButtonSize);
121 xStart += kRewindBitmapWidth;
122 TransportButton* p_slow = new TransportButton(ButtonRect, B_EMPTY_STRING,
124 kPressedSkipBackBitmapBits,
125 kDisabledSkipBackBitmapBits,
126 new BMessage(SLOWER_PLAY));
127 p_view->AddChild( p_slow );
130 ButtonRect.SetLeftTop(BPoint(xStart, yStart));
131 ButtonRect.SetRightBottom(ButtonRect.LeftTop() + kPlayButtonSize);
132 xStart += kPlayPauseBitmapWidth + 1.0;
133 PlayPauseButton* p_play = new PlayPauseButton(ButtonRect, B_EMPTY_STRING,
134 kPlayButtonBitmapBits,
135 kPressedPlayButtonBitmapBits,
136 kDisabledPlayButtonBitmapBits,
137 kPlayingPlayButtonBitmapBits,
138 kPressedPlayingPlayButtonBitmapBits,
139 kPausedPlayButtonBitmapBits,
140 kPressedPausedPlayButtonBitmapBits,
141 new BMessage(START_PLAYBACK));
143 p_view->AddChild( p_play );
144 /* p_play->SetPlaying(); */
147 ButtonRect.SetLeftTop(BPoint(xStart, yStart));
148 ButtonRect.SetRightBottom(ButtonRect.LeftTop() + kSkipButtonSize);
149 xStart += kRewindBitmapWidth;
150 TransportButton* p_fast = new TransportButton(ButtonRect, B_EMPTY_STRING,
151 kSkipForwardBitmapBits,
152 kPressedSkipForwardBitmapBits,
153 kDisabledSkipForwardBitmapBits,
154 new BMessage(FASTER_PLAY));
155 p_view->AddChild( p_fast );
158 ButtonRect.SetLeftTop(BPoint(xStart, yStart));
159 ButtonRect.SetRightBottom(ButtonRect.LeftTop() + kStopButtonSize);
160 xStart += kStopBitmapWidth;
161 TransportButton* p_stop = new TransportButton(ButtonRect, B_EMPTY_STRING,
162 kStopButtonBitmapBits,
163 kPressedStopButtonBitmapBits,
164 kDisabledStopButtonBitmapBits,
165 new BMessage(STOP_PLAYBACK));
166 p_view->AddChild( p_stop );
168 ButtonRect.SetLeftTop(BPoint(xStart + 5, yStart + 6));
169 ButtonRect.SetRightBottom(ButtonRect.LeftTop() + kSpeakerButtonSize);
170 xStart += kSpeakerIconBitmapWidth;
172 TransportButton* p_mute = new TransportButton(ButtonRect, B_EMPTY_STRING,
174 kPressedSpeakerIconBits,
176 new BMessage(VOLUME_MUTE));
178 p_view->AddChild( p_mute );
181 rgb_color fill_color = {0,255,0};
182 p_seek = new SeekSlider(BRect(5,2,255,15), this, 0, 100,
185 p_seek->UseFillColor(true, &fill_color);
186 p_view->AddChild( p_seek );
189 p_vol = new MediaSlider(BRect(xStart,20,255,30), new BMessage(VOLUME_CHG),
191 p_vol->SetValue(VOLUME_DEFAULT);
192 p_vol->UseFillColor(true, &fill_color);
193 p_view->AddChild( p_vol );
195 /* Set size and Show */
197 ResizeTo(260,50 + menu_bar->Bounds().IntegerHeight()+1);
201 InterfaceWindow::~InterfaceWindow()
205 /*****************************************************************************
206 * InterfaceWindow::MessageReceived
207 *****************************************************************************/
208 void InterfaceWindow::MessageReceived( BMessage * p_message )
210 int vol_val = p_vol->Value(); // remember the current volume
211 static int playback_status; // remember playback state
217 switch( p_message->what )
219 case B_ABOUT_REQUESTED:
220 alert = new BAlert(VOUT_TITLE, "BeOS " VOUT_TITLE "\n\n<www.videolan.org>", "Ok");
230 file_panel = new BFilePanel();
231 file_panel->SetTarget( this );
236 const char *psz_device;
237 char psz_source[ B_FILE_NAME_LENGTH + 4 ];
238 if( p_message->FindString("device", &psz_device) != B_ERROR )
240 snprintf( psz_source, B_FILE_NAME_LENGTH + 4,
241 "dvd:%s", psz_device );
242 psz_source[ strlen(psz_source) ] = '\0';
243 intf_PlaylistAdd( p_main->p_playlist, PLAYLIST_END, (char*)psz_source );
248 // this currently stops playback not nicely
249 if( p_intf->p_input != NULL )
251 // silence the sound, otherwise very horrible
252 vlc_mutex_lock( &p_aout_bank->lock );
253 for( i_index = 0 ; i_index < p_aout_bank->i_count ; i_index++ )
255 p_aout_bank->pp_aout[i_index]->i_savedvolume = p_aout_bank->pp_aout[i_index]->i_volume;
256 p_aout_bank->pp_aout[i_index]->i_volume = 0;
258 vlc_mutex_unlock( &p_aout_bank->lock );
261 /* end playing item */
262 p_intf->p_input->b_eof = 1;
264 /* update playlist */
265 vlc_mutex_lock( &p_main->p_playlist->change_lock );
266 p_main->p_playlist->i_index--;
267 p_main->p_playlist->b_stopped = 1;
268 vlc_mutex_unlock( &p_main->p_playlist->change_lock );
273 /* starts playing in normal mode */
274 /* if (p_intf->p_input != NULL )
276 if (p_main->p_aout != NULL)
278 p_main->p_aout->i_vol = vol_val;
281 input_SetStatus(p_intf->p_input, INPUT_STATUS_PLAY);
282 playback_status = PLAYING;
288 /* toggle between pause and play */
289 if( p_intf->p_input != NULL )
291 /* pause if currently playing */
292 if( playback_status == PLAYING )
295 vlc_mutex_lock( &p_aout_bank->lock );
296 for( i_index = 0 ; i_index < p_aout_bank->i_count ; i_index++ )
298 p_aout_bank->pp_aout[i_index]->i_savedvolume =
299 p_aout_bank->pp_aout[i_index]->i_volume;
300 p_aout_bank->pp_aout[i_index]->i_volume = 0;
302 vlc_mutex_unlock( &p_aout_bank->lock );
304 /* pause the movie */
305 input_SetStatus( p_intf->p_input, INPUT_STATUS_PAUSE );
306 vlc_mutex_lock( &p_main->p_playlist->change_lock );
307 p_main->p_playlist->b_stopped = 0;
308 vlc_mutex_unlock( &p_main->p_playlist->change_lock );
309 playback_status = PAUSED;
313 /* Play after pausing */
314 /* Restore the volume */
315 vlc_mutex_lock( &p_aout_bank->lock );
316 for( i_index = 0 ; i_index < p_aout_bank->i_count ; i_index++ )
318 p_aout_bank->pp_aout[i_index]->i_volume =
319 p_aout_bank->pp_aout[i_index]->i_savedvolume;
320 p_aout_bank->pp_aout[i_index]->i_savedvolume = 0;
322 vlc_mutex_unlock( &p_aout_bank->lock );
326 input_SetStatus( p_intf->p_input, INPUT_STATUS_PLAY );
327 p_main->p_playlist->b_stopped = 0;
328 playback_status = PLAYING;
333 /* Play a new file */
334 vlc_mutex_lock( &p_main->p_playlist->change_lock );
335 if( p_main->p_playlist->b_stopped )
337 if( p_main->p_playlist->i_size )
339 vlc_mutex_unlock( &p_main->p_playlist->change_lock );
340 intf_PlaylistJumpto( p_main->p_playlist,
341 p_main->p_playlist->i_index );
342 playback_status = PLAYING;
346 vlc_mutex_unlock( &p_main->p_playlist->change_lock );
353 /* cycle the fast playback modes */
354 if( p_intf->p_input != NULL )
357 vlc_mutex_lock( &p_aout_bank->lock );
358 for( i_index = 0 ; i_index < p_aout_bank->i_count ; i_index++ )
360 p_aout_bank->pp_aout[i_index]->i_savedvolume =
361 p_aout_bank->pp_aout[i_index]->i_volume;
362 p_aout_bank->pp_aout[i_index]->i_volume = 0;
364 vlc_mutex_unlock( &p_aout_bank->lock );
367 /* change the fast play mode */
368 input_SetStatus( p_intf->p_input, INPUT_STATUS_FASTER );
369 vlc_mutex_lock( &p_main->p_playlist->change_lock );
370 p_main->p_playlist->b_stopped = 0;
371 vlc_mutex_unlock( &p_main->p_playlist->change_lock );
376 /* cycle the slow playback modes */
377 if (p_intf->p_input != NULL )
380 vlc_mutex_lock( &p_aout_bank->lock );
381 for( i_index = 0 ; i_index < p_aout_bank->i_count ; i_index++ )
383 p_aout_bank->pp_aout[i_index]->i_savedvolume =
384 p_aout_bank->pp_aout[i_index]->i_volume;
385 p_aout_bank->pp_aout[i_index]->i_volume = 0;
387 vlc_mutex_unlock( &p_aout_bank->lock );
390 /* change the slower play */
391 input_SetStatus( p_intf->p_input, INPUT_STATUS_SLOWER );
392 vlc_mutex_lock( &p_main->p_playlist->change_lock );
393 p_main->p_playlist->b_stopped = 0;
394 vlc_mutex_unlock( &p_main->p_playlist->change_lock );
399 /* handled by semaphores */
403 /* adjust the volume */
404 vlc_mutex_lock( &p_aout_bank->lock );
405 for( i_index = 0 ; i_index < p_aout_bank->i_count ; i_index++ )
407 if( p_aout_bank->pp_aout[i_index]->i_savedvolume )
409 p_aout_bank->pp_aout[i_index]->i_savedvolume = vol_val;
413 p_aout_bank->pp_aout[i_index]->i_volume = vol_val;
416 vlc_mutex_unlock( &p_aout_bank->lock );
421 vlc_mutex_lock( &p_aout_bank->lock );
422 for( i_index = 0 ; i_index < p_aout_bank->i_count ; i_index++ )
424 if( p_aout_bank->pp_aout[i_index]->i_savedvolume )
426 p_aout_bank->pp_aout[i_index]->i_volume =
427 p_aout_bank->pp_aout[i_index]->i_savedvolume;
428 p_aout_bank->pp_aout[i_index]->i_savedvolume = 0;
432 p_aout_bank->pp_aout[i_index]->i_savedvolume =
433 p_aout_bank->pp_aout[i_index]->i_volume;
434 p_aout_bank->pp_aout[i_index]->i_volume = 0;
437 vlc_mutex_unlock( &p_aout_bank->lock );
442 int32 i = p_message->FindInt32( "channel" );
445 input_ChangeES( p_intf->p_input, NULL, AUDIO_ES );
449 input_ChangeES( p_intf->p_input,
450 p_intf->p_input->stream.pp_es[i], AUDIO_ES );
455 case SELECT_SUBTITLE:
457 int32 i = p_message->FindInt32( "subtitle" );
460 input_ChangeES( p_intf->p_input, NULL, SPU_ES);
464 input_ChangeES( p_intf->p_input,
465 p_intf->p_input->stream.pp_es[i], SPU_ES );
470 case B_REFS_RECEIVED:
474 if( p_message->FindRef( "refs", &ref ) == B_OK )
477 intf_PlaylistAdd( p_main->p_playlist,
478 PLAYLIST_END, (char*)path.Path() );
484 BWindow::MessageReceived( p_message );
489 /*****************************************************************************
490 * InterfaceWindow::updateInterface
491 *****************************************************************************/
492 void InterfaceWindow::updateInterface()
495 bool seekNeeded = false;
497 if( acquire_sem(fScrubSem) == B_OK )
503 uint32 seekTo = (p_seek->Value() *
504 p_intf->p_input->stream.p_selected_area->i_size) / 100;
505 input_Seek( p_intf->p_input, seekTo );
510 progress = (100. * p_intf->p_input->stream.p_selected_area->i_tell) /
511 p_intf->p_input->stream.p_selected_area->i_size;
512 p_seek->SetValue(progress);
517 /*****************************************************************************
518 * InterfaceWindow::QuitRequested
519 *****************************************************************************/
520 bool InterfaceWindow::QuitRequested()
527 /*****************************************************************************
529 *****************************************************************************/
530 CDMenu::CDMenu(const char *name)
535 /*****************************************************************************
537 *****************************************************************************/
542 /*****************************************************************************
543 * CDMenu::AttachedToWindow
544 *****************************************************************************/
545 void CDMenu::AttachedToWindow(void)
547 while (RemoveItem((long int)0) != NULL); // remove all items
549 BMenu::AttachedToWindow();
552 /*****************************************************************************
554 *****************************************************************************/
555 int CDMenu::GetCD( const char *directory )
559 dir.SetTo( directory );
561 if( dir.InitCheck() != B_NO_ERROR )
569 while( dir.GetNextEntry(&entry) >= 0 )
575 if( entry.GetPath(&path) != B_NO_ERROR )
582 if( entry.GetRef(&e) != B_NO_ERROR )
587 if( entry.IsDirectory() )
589 if( strcmp(e.name, "floppy") == 0 )
591 continue; // ignore floppy (it is not silent)
594 i_dev = GetCD( name );
606 if( strcmp(e.name, "raw") != 0 )
608 continue; // ignore partitions
611 i_dev = open( name, O_RDONLY );
618 if( ioctl(i_dev, B_GET_GEOMETRY, &g, sizeof(g)) >= 0 )
620 if( g.device_type == B_CD ) //ensure the drive is a CD-ROM
622 if( ioctl(i_dev, B_GET_MEDIA_STATUS, &m, sizeof(m)) >= 0 )
624 if( m == B_NO_ERROR ) //ensure media is present
627 msg = new BMessage( OPEN_DVD );
628 msg->AddString( "device", name );
629 BMenuItem *menu_item;
630 menu_item = new BMenuItem( name, msg );
631 AddItem( menu_item );
643 /*****************************************************************************
644 * LanguageMenu::LanguageMenu
645 *****************************************************************************/
646 LanguageMenu::LanguageMenu(const char *name, int menu_kind, intf_thread_t *p_interface)
650 p_intf = p_interface;
653 /*****************************************************************************
654 * LanguageMenu::~LanguageMenu
655 *****************************************************************************/
656 LanguageMenu::~LanguageMenu()
660 /*****************************************************************************
661 * LanguageMenu::AttachedToWindow
662 *****************************************************************************/
663 void LanguageMenu::AttachedToWindow(void)
665 while( RemoveItem((long int)0) != NULL )
667 ; // remove all items
672 BMenu::AttachedToWindow();
675 /*****************************************************************************
676 * LanguageMenu::GetChannels
677 *****************************************************************************/
678 int LanguageMenu::GetChannels()
684 es_descriptor_t *p_es = NULL;
686 /* Insert the null */
687 if( kind == AUDIO_ES ) //audio
689 msg = new BMessage(SELECT_CHANNEL);
690 msg->AddInt32("channel", -1);
694 msg = new BMessage(SELECT_SUBTITLE);
695 msg->AddInt32("subtitle", -1);
697 BMenuItem *menu_item;
698 menu_item = new BMenuItem("None", msg);
700 menu_item->SetMarked(TRUE);
702 if( p_intf->p_input == NULL )
708 vlc_mutex_lock( &p_intf->p_input->stream.stream_lock );
709 for( i = 0; i < p_intf->p_input->stream.i_selected_es_number; i++ )
711 if( kind == p_intf->p_input->stream.pp_selected_es[i]->i_cat )
713 p_es = p_intf->p_input->stream.pp_selected_es[i];
717 for( i = 0; i < p_intf->p_input->stream.i_es_number; i++ )
719 if( kind == p_intf->p_input->stream.pp_es[i]->i_cat )
721 psz_name = p_intf->p_input->stream.pp_es[i]->psz_desc;
722 if( kind == AUDIO_ES ) //audio
724 msg = new BMessage(SELECT_CHANNEL);
725 msg->AddInt32("channel", i);
729 msg = new BMessage(SELECT_SUBTITLE);
730 msg->AddInt32("subtitle", i);
732 BMenuItem *menu_item;
733 menu_item = new BMenuItem(psz_name, msg);
735 b_active = (p_es == p_intf->p_input->stream.pp_es[i]);
736 menu_item->SetMarked(b_active);
739 vlc_mutex_unlock( &p_intf->p_input->stream.stream_lock );
743 /*****************************************************************************
745 *****************************************************************************/
746 MediaSlider::MediaSlider( BRect frame, BMessage *p_message,
747 int32 i_min, int32 i_max )
748 :BSlider(frame, NULL, NULL, p_message, i_min, i_max )
753 MediaSlider::~MediaSlider()
758 void MediaSlider::DrawThumb(void)
763 rgb_color black = {0,0,0};
769 v->SetHighColor(black);
773 v->SetHighColor(tint_color(black, B_LIGHTEN_2_TINT));
776 r.InsetBy(r.IntegerWidth()/4, r.IntegerHeight()/(4 * r.IntegerWidth() / r.IntegerHeight()));
781 v->SetHighColor(ui_color(B_PANEL_BACKGROUND_COLOR));
785 v->SetHighColor(tint_color(ui_color(B_PANEL_BACKGROUND_COLOR), B_LIGHTEN_2_TINT));
792 /*****************************************************************************
794 *****************************************************************************/
795 SeekSlider::SeekSlider( BRect frame, InterfaceWindow *p_owner, int32 i_min,
796 int32 i_max, thumb_style thumbType = B_TRIANGLE_THUMB )
797 :MediaSlider( frame, NULL, i_min, i_max )
803 SeekSlider::~SeekSlider()
807 /*****************************************************************************
808 * SeekSlider::MouseDown
809 *****************************************************************************/
810 void SeekSlider::MouseDown(BPoint where)
812 BSlider::MouseDown(where);
813 fOwner->fScrubSem = create_sem(1, "Vlc::fScrubSem");
817 /*****************************************************************************
818 * SeekSlider::MouseUp
819 *****************************************************************************/
820 void SeekSlider::MouseMoved(BPoint where, uint32 code, const BMessage *message)
822 BSlider::MouseMoved(where, code, message);
825 release_sem(fOwner->fScrubSem);
828 /*****************************************************************************
829 * SeekSlider::MouseUp
830 *****************************************************************************/
831 void SeekSlider::MouseUp(BPoint where)
833 BSlider::MouseUp(where);
834 delete_sem(fOwner->fScrubSem);
835 fOwner->fScrubSem = B_ERROR;