* intf_beos.cpp: beos interface
*****************************************************************************
* Copyright (C) 1999, 2000, 2001 VideoLAN
- * $Id: intf_beos.cpp,v 1.13 2001/03/05 01:29:25 sam Exp $
+ * $Id: intf_beos.cpp,v 1.27 2001/05/01 04:18:17 sam Exp $
*
* Authors: Jean-Marc Dressler <polux@via.ecp.fr>
* Samuel Hocevar <sam@zoy.org>
+ * Tony Castley <tony@castley.net>
+ * Richard Shepherd <richard@rshepherd.demon.co.uk>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
#include <NodeInfo.h>
#include <Locker.h>
#include <DirectWindow.h>
-
+#include <Box.h>
+#include <Alert.h>
+#include <MenuBar.h>
+#include <MenuItem.h>
+#include <FilePanel.h>
+#include <Screen.h>
#include <malloc.h>
#include <string.h>
+#include <Directory.h>
+#include <Entry.h>
+#include <Path.h>
+#include <StorageDefs.h>
+#include <scsi.h>
+#include <scsiprobe_driver.h>
extern "C"
{
#include "input_ext-intf.h"
#include "interface.h"
-#include "intf_plst.h"
+#include "intf_playlist.h"
#include "intf_msg.h"
#include "audio_output.h"
#include "MsgVals.h"
-
#include "main.h"
}
: BWindow(frame, name, B_FLOATING_WINDOW_LOOK, B_NORMAL_WINDOW_FEEL,
B_NOT_RESIZABLE | B_NOT_ZOOMABLE | B_WILL_ACCEPT_FIRST_CLICK |B_ASYNCHRONOUS_CONTROLS)
{
+ file_panel = NULL;
p_intf = p_interface;
BRect ButtonRect;
- float xStart = 2.0;
- float yStart = 10.0;
+ float xStart = 5.0;
+ float yStart = 20.0;
SetName( "interface" );
SetTitle(VOUT_TITLE " (BeOS interface)");
+ BRect rect(0, 0, 0, 0);
- BView* p_view;
+ BMenuBar *menu_bar;
+ menu_bar = new BMenuBar(rect, "main menu");
+ AddChild( menu_bar );
+
+ BMenu *mFile;
+ BMenu *mAudio;
+ CDMenu *cd_menu;
+
+ BMenuItem *mItem;
+
+ menu_bar->AddItem( mFile = new BMenu("File") );
+ menu_bar->ResizeToPreferred();
+ mFile->AddItem(mItem = new BMenuItem("Open File" B_UTF8_ELLIPSIS, new BMessage(OPEN_FILE), 'O'));
+ cd_menu = new CDMenu("Open Disc");
+ mFile->AddItem(cd_menu);
+ mFile->AddSeparatorItem();
+ mFile->AddItem(mItem = new BMenuItem("About" B_UTF8_ELLIPSIS, new BMessage(B_ABOUT_REQUESTED), 'A'));
+ mItem->SetTarget( be_app );
+ mFile->AddItem(mItem = new BMenuItem("Quit", new BMessage(B_QUIT_REQUESTED), 'Q'));
+
+ menu_bar->AddItem ( mAudio = new BMenu("Audio") );
+ menu_bar->ResizeToPreferred();
+ mAudio->AddItem( new LanguageMenu("Language", AUDIO_ES, p_intf) );
+ mAudio->AddItem( new LanguageMenu("Subtitles", SPU_ES, p_intf) );
+
- /* Add the view */
- p_view = new BView( Bounds(), "", B_FOLLOW_ALL, B_WILL_DRAW );
- p_view->SetViewColor(216,216,216);
+ rect = Bounds();
+ rect.top += menu_bar->Bounds().IntegerHeight()+1;
+
+ BBox* p_view;
+ p_view = new BBox( rect, NULL, B_FOLLOW_ALL, B_WILL_DRAW, B_PLAIN_BORDER );
+ p_view->SetViewColor(ui_color(B_PANEL_BACKGROUND_COLOR));
/* Buttons */
/* Slow play */
kDisabledStopButtonBitmapBits,
new BMessage(STOP_PLAYBACK));
p_view->AddChild( p_stop );
-
+
+ ButtonRect.SetLeftTop(BPoint(xStart + 5, yStart + 6));
+ ButtonRect.SetRightBottom(ButtonRect.LeftTop() + kSpeakerButtonSize);
+ xStart += kSpeakerIconBitmapWidth;
+
+ TransportButton* p_mute = new TransportButton(ButtonRect, B_EMPTY_STRING,
+ kSpeakerIconBits,
+ kPressedSpeakerIconBits,
+ kSpeakerIconBits,
+ new BMessage(VOLUME_MUTE));
+
+ p_view->AddChild( p_mute );
+
/* Seek Status */
- p_seek = new SeekSlider(BRect(5,35,355,65), this, 0, 100,
+ rgb_color fill_color = {0,255,0};
+ p_seek = new SeekSlider(BRect(5,2,255,15), this, 0, 100,
B_TRIANGLE_THUMB);
p_seek->SetValue(0);
- p_seek->UseFillColor(TRUE);
+ p_seek->UseFillColor(true, &fill_color);
p_view->AddChild( p_seek );
/* Volume Slider */
- p_vol = new BSlider(BRect(xStart,2,300,20), "vol", "Volume",
- new BMessage(VOLUME_CHG), 0, VOLUME_MAX);
+ p_vol = new MediaSlider(BRect(xStart,20,255,30), new BMessage(VOLUME_CHG),
+ 0, VOLUME_MAX);
p_vol->SetValue(VOLUME_DEFAULT);
+ p_vol->UseFillColor(true, &fill_color);
p_view->AddChild( p_vol );
- /* Volume Mute */
- p_mute = new BCheckBox(BRect(300,10,355,25), "mute", "Mute",
- new BMessage(VOLUME_MUTE));
- p_view->AddChild( p_mute );
-
/* Set size and Show */
AddChild( p_view );
- ResizeTo(360,70);
+ ResizeTo(260,50 + menu_bar->Bounds().IntegerHeight()+1);
Show();
}
int vol_val = p_vol->Value(); // remember the current volume
static int playback_status; // remember playback state
+ BAlert *alert;
Activate();
switch( p_message->what )
{
+// case B_ABOUT_REQUESTED:
+// alert = new BAlert(VOUT_TITLE, "BeOS " VOUT_TITLE "\n\n<www.videolan.org>", "Ok");
+// alert->Go();
+// break;
+
+ case OPEN_FILE:
+ if(file_panel)
+ {
+ file_panel->Show();
+ break;
+ }
+ file_panel = new BFilePanel();
+ file_panel->SetTarget(this);
+ file_panel->Show();
+ break;
+
case OPEN_DVD:
+ const char **device;
+ char device_method_and_name[B_FILE_NAME_LENGTH + 4];
+ if(p_message->FindString("device", device) != B_ERROR)
+ {
+ sprintf(device_method_and_name, "dvd:%s", *device);
+ intf_PlaylistAdd( p_main->p_playlist, PLAYLIST_END, device_method_and_name );
+ }
break;
+
case STOP_PLAYBACK:
// this currently stops playback not nicely
if (p_intf->p_input != NULL )
// silence the sound, otherwise very horrible
if (p_main->p_aout != NULL)
{
- p_main->p_aout->vol = 0;
+ p_main->p_aout->i_vol = 0;
}
snooze(400000);
input_SetStatus(p_intf->p_input, INPUT_STATUS_END);
// {
// if (p_main->p_aout != NULL)
// {
-// p_main->p_aout->vol = vol_val;
+// p_main->p_aout->i_vol = vol_val;
// }
// snooze(400000);
// input_SetStatus(p_intf->p_input, INPUT_STATUS_PLAY);
{
if (p_main->p_aout != NULL)
{
- p_main->p_aout->vol = 0;
+ p_main->p_aout->i_vol = 0;
}
playback_status = PAUSED;
}
{
if (p_main->p_aout != NULL)
{
- p_main->p_aout->vol = vol_val;
+ p_main->p_aout->i_vol = vol_val;
}
playback_status = PLAYING;
}
{
if (p_main->p_aout != NULL)
{
- p_main->p_aout->vol = 0;
+ p_main->p_aout->i_vol = 0;
}
snooze(400000);
input_SetStatus(p_intf->p_input, INPUT_STATUS_FASTER);
{
if (p_main->p_aout != NULL)
{
- p_main->p_aout->vol = 0;
+ p_main->p_aout->i_vol = 0;
}
snooze(400000);
input_SetStatus(p_intf->p_input, INPUT_STATUS_SLOWER);
break;
case SEEK_PLAYBACK:
// handled by semaphores;
-/* if( p_intf->p_input != NULL )
- {
- float new_position;
- if (p_message->FindFloat("be:value", &new_position) == B_OK)
- {
- printf("%e\n", new_position);
- input_Seek( p_intf->p_input, new_position * 100 );
- }
- } */
break;
case VOLUME_CHG:
// adjust the volume
if (p_main->p_aout != NULL)
{
- p_main->p_aout->vol = vol_val;
+ p_main->p_aout->i_vol = vol_val;
}
break;
case VOLUME_MUTE:
// mute
if (p_main->p_aout != NULL)
{
- if (p_mute->Value() == B_CONTROL_OFF)
+ if (p_main->p_aout->i_vol == 0)
{
- p_main->p_aout->vol = vol_val;
+ p_vol->SetEnabled(true);
+ p_main->p_aout->i_vol = vol_val;
}
else
{
- p_main->p_aout->vol = 0;
+ p_vol->SetEnabled(false);
+ p_main->p_aout->i_vol = 0;
}
}
break;
case SELECT_CHANNEL:
+ {
+ int32 i = p_message->FindInt32("channel");
+ input_ChangeES(p_intf->p_input,
+ p_intf->p_input->stream.pp_es[i], 1);
+ }
break;
+ case SELECT_SUBTITLE:
+ {
+ int32 i = p_message->FindInt32("subtitle");
+ input_ChangeES(p_intf->p_input,
+ p_intf->p_input->stream.pp_es[i], 2);
+ }
+ break;
+ case B_REFS_RECEIVED:
case B_SIMPLE_DATA:
{
entry_ref ref;
if( p_message->FindRef( "refs", &ref ) == B_OK )
{
BPath path( &ref );
- char * psz_name = strdup(path.Path());
- intf_WarnMsg( 1, "intf: dropped text/uri-list data `%s'",
- psz_name );
- intf_PlstAdd( p_main->p_playlist, PLAYLIST_END, psz_name );
+ intf_PlaylistAdd( p_main->p_playlist,
+ PLAYLIST_END, path.Path() );
}
}
/*****************************************************************************
* InterfaceWindow::QuitRequested
*****************************************************************************/
-
bool InterfaceWindow::QuitRequested()
{
p_intf->b_die = 1;
return( false );
}
+/*****************************************************************************
+ * CDMenu::CDMenu
+ *****************************************************************************/
+CDMenu::CDMenu(const char *name)
+ : BMenu(name)
+{
+}
+
+/*****************************************************************************
+ * CDMenu::~CDMenu
+ *****************************************************************************/
+CDMenu::~CDMenu()
+{
+}
+
+/*****************************************************************************
+ * CDMenu::AttachedToWindow
+ *****************************************************************************/
+void CDMenu::AttachedToWindow(void)
+{
+ while (RemoveItem((long int)0) != NULL); // remove all items
+ GetCD("/dev/disk");
+ BMenu::AttachedToWindow();
+}
+
+/*****************************************************************************
+ * CDMenu::GetCD
+ *****************************************************************************/
+int CDMenu::GetCD(const char *directory)
+{
+ BDirectory dir;
+ dir.SetTo(directory);
+ if(dir.InitCheck() != B_NO_ERROR) {
+ return B_ERROR;
+ }
+ dir.Rewind();
+ BEntry entry;
+ while(dir.GetNextEntry(&entry) >= 0) {
+ BPath path;
+ const char *name;
+ entry_ref e;
+
+ if(entry.GetPath(&path) != B_NO_ERROR)
+ continue;
+ name = path.Path();
+
+
+ if(entry.GetRef(&e) != B_NO_ERROR)
+ continue;
+
+ if(entry.IsDirectory()) {
+ if(strcmp(e.name, "floppy") == 0)
+ continue; // ignore floppy (it is not silent)
+ int devfd = GetCD(name);
+ if(devfd >= 0)
+ {
+ return devfd;
+ }
+ }
+ else {
+ int devfd;
+ device_geometry g;
+ status_t m;
+
+ if(strcmp(e.name, "raw") != 0)
+ continue; // ignore partitions
+
+ devfd = open(name, O_RDONLY);
+ if(devfd < 0)
+ continue;
+
+ if(ioctl(devfd, B_GET_GEOMETRY, &g, sizeof(g)) >= 0) {
+ if(g.device_type == B_CD) //ensure the drive is a CD-ROM
+ {
+ if(ioctl(devfd, B_GET_MEDIA_STATUS, &m, sizeof(m)) >= 0 )
+ if(m == B_NO_ERROR) //ensure media is present
+ {
+ BMessage *msg;
+ msg = new BMessage(OPEN_DVD);
+ msg->AddString("device", name);
+ BMenuItem *menu_item;
+ menu_item = new BMenuItem(name, msg);
+ AddItem(menu_item);
+ continue;
+ }
+ }
+ }
+ close(devfd);
+ }
+ }
+ return B_ERROR;
+}
+
+/*****************************************************************************
+ * LanguageMenu::LanguageMenu
+ *****************************************************************************/
+LanguageMenu::LanguageMenu(const char *name, int menu_kind, intf_thread_t *p_interface)
+ :BMenu(name)
+{
+ kind = menu_kind;
+ p_intf = p_interface;
+}
+
+/*****************************************************************************
+ * LanguageMenu::~LanguageMenu
+ *****************************************************************************/
+LanguageMenu::~LanguageMenu()
+{
+}
+
+/*****************************************************************************
+ * LanguageMenu::AttachedToWindow
+ *****************************************************************************/
+void LanguageMenu::AttachedToWindow(void)
+{
+ while (RemoveItem((long int)0) != NULL); // remove all items
+ SetRadioMode(true);
+ GetChannels();
+ BMenu::AttachedToWindow();
+}
+
+/*****************************************************************************
+ * LanguageMenu::GetChannels
+ *****************************************************************************/
+int LanguageMenu::GetChannels()
+{
+ char* psz_name;
+ bool b_active;
+ bool b_found;
+ int32 i;
+ es_descriptor_t *p_es;
+
+ if (p_intf->p_input == NULL)
+ return 1;
+
+ for (i = 0; i < p_intf->p_input->stream.i_selected_es_number; i++)
+ {
+ if (kind ==
+ p_intf->p_input->stream.pp_selected_es[i]->i_cat)
+ {
+ p_es = p_intf->p_input->stream.pp_selected_es[i];
+ }
+ }
+ for (i = 0; i < p_intf->p_input->stream.i_es_number; i++)
+ {
+ if (kind == p_intf->p_input->stream.pp_es[i]->i_cat)
+ {
+ psz_name = p_intf->p_input->stream.pp_es[i]->psz_desc;
+ BMessage *msg;
+ if (kind == AUDIO_ES) //audio
+ {
+ msg = new BMessage(SELECT_CHANNEL);
+ msg->AddInt32("channel", i);
+ }
+ else
+ {
+ msg = new BMessage(SELECT_SUBTITLE);
+ msg->AddInt32("subtitle", i);
+ }
+ BMenuItem *menu_item;
+ menu_item = new BMenuItem(psz_name, msg);
+ AddItem(menu_item);
+ b_active = (p_es == p_intf->p_input->stream.pp_es[i]);
+ menu_item->SetMarked(b_active);
+ }
+ }
+}
+
+
+/*****************************************************************************
+ * MediaSlider
+ *****************************************************************************/
+MediaSlider::MediaSlider(BRect frame,
+ BMessage *message,
+ int32 minValue,
+ int32 maxValue)
+ :BSlider(frame, NULL, NULL, message, minValue, maxValue)
+{
+
+}
+
+MediaSlider::~MediaSlider()
+{
+
+}
+
+void MediaSlider::DrawThumb(void)
+{
+ BRect r;
+ BView *v;
+
+ rgb_color black = {0,0,0};
+ r = ThumbFrame();
+ v = OffscreenView();
+ if(IsEnabled())
+ v->SetHighColor(black);
+ else
+ v->SetHighColor(tint_color(black, B_LIGHTEN_2_TINT));
+ r.InsetBy(r.IntegerWidth()/4, r.IntegerHeight()/(4 * r.IntegerWidth() / r.IntegerHeight()));
+ v->StrokeEllipse(r);
+ if(IsEnabled())
+ v->SetHighColor(ui_color(B_PANEL_BACKGROUND_COLOR));
+ else
+ v->SetHighColor(tint_color(ui_color(B_PANEL_BACKGROUND_COLOR), B_LIGHTEN_2_TINT));
+ r.InsetBy(1,1);
+ v->FillEllipse(r);
+}
+
/*****************************************************************************
* SeekSlider
*****************************************************************************/
int32 minValue,
int32 maxValue,
thumb_style thumbType = B_TRIANGLE_THUMB)
- :BSlider(frame, B_EMPTY_STRING, B_EMPTY_STRING,
- NULL, minValue, maxValue, thumbType)
+ :MediaSlider(frame, NULL, minValue, maxValue)
{
fOwner = owner;
fMouseDown = false;
*****************************************************************************/
static int intf_Open( intf_thread_t *p_intf )
{
+ BScreen *screen;
+ screen = new BScreen();
+ BRect rect = screen->Frame();
+ rect.top = rect.bottom-100;
+ rect.bottom -= 50;
+ rect.left += 50;
+ rect.right = rect.left + 350;
+ delete screen;
+
/* Allocate instance and initialize some members */
p_intf->p_sys = (intf_sys_t*) malloc( sizeof( intf_sys_t ) );
if( p_intf->p_sys == NULL )
/* Create the interface window */
p_intf->p_sys->p_window =
- new InterfaceWindow( BRect( 50, 50, 400, 100 ),
+ new InterfaceWindow( rect,
VOUT_TITLE " (BeOS interface)", p_intf );
if( p_intf->p_sys->p_window == 0 )
{