2816 lines
70 KiB
C++
2816 lines
70 KiB
C++
/*
|
|
* MainWin.cpp - Media Player for the Haiku Operating System
|
|
*
|
|
* Copyright (C) 2006 Marcus Overhagen <marcus@overhagen.de>
|
|
* Copyright (C) 2007-2010 Stephan Aßmus <superstippi@gmx.de> (GPL->MIT ok)
|
|
* Copyright (C) 2007-2009 Fredrik Modéen <[FirstName]@[LastName].se> (MIT ok)
|
|
*
|
|
* Released under the terms of the MIT license.
|
|
*/
|
|
|
|
|
|
#include "MainWin.h"
|
|
|
|
#include <math.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
#include <Alert.h>
|
|
#include <Application.h>
|
|
#include <Autolock.h>
|
|
#include <Catalog.h>
|
|
#include <Debug.h>
|
|
#include <Directory.h>
|
|
#include <Drivers.h>
|
|
#include <fs_attr.h>
|
|
#include <LayoutBuilder.h>
|
|
#include <Language.h>
|
|
#include <Locale.h>
|
|
#include <MediaRoster.h>
|
|
#include <Menu.h>
|
|
#include <MenuBar.h>
|
|
#include <MenuItem.h>
|
|
#include <MessageRunner.h>
|
|
#include <Messenger.h>
|
|
#include <PopUpMenu.h>
|
|
#include <PropertyInfo.h>
|
|
#include <RecentItems.h>
|
|
#include <Roster.h>
|
|
#include <Screen.h>
|
|
#include <String.h>
|
|
#include <TypeConstants.h>
|
|
#include <View.h>
|
|
|
|
#include "AudioProducer.h"
|
|
#include "ControllerObserver.h"
|
|
#include "DurationToString.h"
|
|
#include "FilePlaylistItem.h"
|
|
#include "MainApp.h"
|
|
#include "NetworkStreamWin.h"
|
|
#include "PeakView.h"
|
|
#include "PlaylistItem.h"
|
|
#include "PlaylistObserver.h"
|
|
#include "PlaylistWindow.h"
|
|
#include "Settings.h"
|
|
|
|
|
|
#undef B_TRANSLATION_CONTEXT
|
|
#define B_TRANSLATION_CONTEXT "MediaPlayer-Main"
|
|
#define MIN_WIDTH 250
|
|
|
|
|
|
int MainWin::sNoVideoWidth = MIN_WIDTH;
|
|
|
|
|
|
// XXX TODO: why is lround not defined?
|
|
#define lround(a) ((int)(0.99999 + (a)))
|
|
|
|
enum {
|
|
M_DUMMY = 0x100,
|
|
M_FILE_OPEN = 0x1000,
|
|
M_NETWORK_STREAM_OPEN,
|
|
M_EJECT_DEVICE,
|
|
M_FILE_INFO,
|
|
M_FILE_PLAYLIST,
|
|
M_FILE_CLOSE,
|
|
M_FILE_QUIT,
|
|
M_VIEW_SIZE,
|
|
M_TOGGLE_FULLSCREEN,
|
|
M_TOGGLE_ALWAYS_ON_TOP,
|
|
M_TOGGLE_NO_INTERFACE,
|
|
M_VOLUME_UP,
|
|
M_VOLUME_DOWN,
|
|
M_SKIP_NEXT,
|
|
M_SKIP_PREV,
|
|
M_WIND,
|
|
|
|
// The common display aspect ratios
|
|
M_ASPECT_SAME_AS_SOURCE,
|
|
M_ASPECT_NO_DISTORTION,
|
|
M_ASPECT_4_3,
|
|
M_ASPECT_16_9,
|
|
M_ASPECT_83_50,
|
|
M_ASPECT_7_4,
|
|
M_ASPECT_37_20,
|
|
M_ASPECT_47_20,
|
|
|
|
M_SELECT_AUDIO_TRACK = 0x00000800,
|
|
M_SELECT_AUDIO_TRACK_END = 0x00000fff,
|
|
M_SELECT_VIDEO_TRACK = 0x00010000,
|
|
M_SELECT_VIDEO_TRACK_END = 0x00010fff,
|
|
M_SELECT_SUB_TITLE_TRACK = 0x00020000,
|
|
M_SELECT_SUB_TITLE_TRACK_END = 0x00020fff,
|
|
|
|
M_SET_RATING,
|
|
|
|
M_SET_PLAYLIST_POSITION,
|
|
|
|
M_FILE_DELETE,
|
|
|
|
M_SLIDE_CONTROLS,
|
|
M_FINISH_SLIDING_CONTROLS
|
|
};
|
|
|
|
|
|
static property_info sPropertyInfo[] = {
|
|
{ "Next", { B_EXECUTE_PROPERTY },
|
|
{ B_DIRECT_SPECIFIER, 0 },
|
|
"Skip to the next track.", 0
|
|
},
|
|
{ "Prev", { B_EXECUTE_PROPERTY },
|
|
{ B_DIRECT_SPECIFIER, 0 },
|
|
"Skip to the previous track.", 0
|
|
},
|
|
{ "Play", { B_EXECUTE_PROPERTY },
|
|
{ B_DIRECT_SPECIFIER, 0 },
|
|
"Start playing.", 0
|
|
},
|
|
{ "Stop", { B_EXECUTE_PROPERTY },
|
|
{ B_DIRECT_SPECIFIER, 0 },
|
|
"Stop playing.", 0
|
|
},
|
|
{ "Pause", { B_EXECUTE_PROPERTY },
|
|
{ B_DIRECT_SPECIFIER, 0 },
|
|
"Pause playback.", 0
|
|
},
|
|
{ "TogglePlaying", { B_EXECUTE_PROPERTY },
|
|
{ B_DIRECT_SPECIFIER, 0 },
|
|
"Toggle pause/play.", 0
|
|
},
|
|
{ "Mute", { B_EXECUTE_PROPERTY },
|
|
{ B_DIRECT_SPECIFIER, 0 },
|
|
"Toggle mute.", 0
|
|
},
|
|
{ "Volume", { B_GET_PROPERTY, B_SET_PROPERTY, 0 },
|
|
{ B_DIRECT_SPECIFIER, 0 },
|
|
"Gets/sets the volume (0.0-2.0).", 0,
|
|
{ B_FLOAT_TYPE }
|
|
},
|
|
{ "URI", { B_GET_PROPERTY, 0 },
|
|
{ B_DIRECT_SPECIFIER, 0 },
|
|
"Gets the URI of the currently playing item.", 0,
|
|
{ B_STRING_TYPE }
|
|
},
|
|
{ "TrackNumber", { B_GET_PROPERTY, 0 },
|
|
{ B_DIRECT_SPECIFIER, 0 },
|
|
"Gets the number of the current track playing.", 0,
|
|
{ B_INT32_TYPE }
|
|
},
|
|
{ "ToggleFullscreen", { B_EXECUTE_PROPERTY },
|
|
{ B_DIRECT_SPECIFIER, 0 },
|
|
"Toggle fullscreen.", 0
|
|
},
|
|
{ "Duration", { B_GET_PROPERTY, 0 },
|
|
{ B_DIRECT_SPECIFIER, 0 },
|
|
"Gets the duration of the currently playing item "
|
|
"in microseconds.", 0,
|
|
{ B_INT64_TYPE }
|
|
},
|
|
{ "Position", { B_GET_PROPERTY, B_SET_PROPERTY, 0 },
|
|
{ B_DIRECT_SPECIFIER, 0 },
|
|
"Gets/sets the current playing position in microseconds.",
|
|
0, { B_INT64_TYPE }
|
|
},
|
|
{ "Seek", { B_SET_PROPERTY },
|
|
{ B_DIRECT_SPECIFIER, 0 },
|
|
"Seek by the specified amounts of microseconds.", 0,
|
|
{ B_INT64_TYPE }
|
|
},
|
|
{ "PlaylistTrackCount", { B_GET_PROPERTY, 0 },
|
|
{ B_DIRECT_SPECIFIER, 0 },
|
|
"Gets the number of tracks in Playlist.", 0,
|
|
{ B_INT16_TYPE }
|
|
},
|
|
{ "PlaylistTrackTitle", { B_GET_PROPERTY, 0 },
|
|
{ B_INDEX_SPECIFIER, 0 },
|
|
"Gets the title of the nth track in Playlist.", 0,
|
|
{ B_STRING_TYPE }
|
|
},
|
|
|
|
{ 0 }
|
|
};
|
|
|
|
|
|
static const char* kRatingAttrName = "Media:Rating";
|
|
|
|
static const char* kDisabledSeekMessage = B_TRANSLATE("Drop files to play");
|
|
|
|
static const char* kApplicationName = B_TRANSLATE_SYSTEM_NAME(NAME);
|
|
|
|
|
|
MainWin::MainWin(bool isFirstWindow, BMessage* message)
|
|
:
|
|
BWindow(BRect(100, 100, 400, 300), kApplicationName, B_TITLED_WINDOW,
|
|
B_ASYNCHRONOUS_CONTROLS),
|
|
fCreationTime(system_time()),
|
|
fInfoWin(NULL),
|
|
fPlaylistWindow(NULL),
|
|
fHasFile(false),
|
|
fHasVideo(false),
|
|
fHasAudio(false),
|
|
fPlaylist(new Playlist),
|
|
fPlaylistObserver(new PlaylistObserver(this)),
|
|
fController(new Controller),
|
|
fControllerObserver(new ControllerObserver(this,
|
|
OBSERVE_FILE_CHANGES | OBSERVE_TRACK_CHANGES
|
|
| OBSERVE_PLAYBACK_STATE_CHANGES | OBSERVE_POSITION_CHANGES
|
|
| OBSERVE_VOLUME_CHANGES)),
|
|
fIsFullscreen(false),
|
|
fAlwaysOnTop(false),
|
|
fNoInterface(false),
|
|
fShowsFullscreenControls(false),
|
|
fSourceWidth(-1),
|
|
fSourceHeight(-1),
|
|
fWidthAspect(0),
|
|
fHeightAspect(0),
|
|
fSavedFrame(),
|
|
fNoVideoFrame(),
|
|
|
|
fMouseDownTracking(false),
|
|
fLastMousePos(0, 0),
|
|
fLastMouseMovedTime(system_time()),
|
|
fMouseMoveDist(0),
|
|
|
|
fGlobalSettingsListener(this),
|
|
fInitialSeekPosition(0),
|
|
fAllowWinding(true)
|
|
{
|
|
// Handle window position and size depending on whether this is the
|
|
// first window or not. Use the window size from the window that was
|
|
// last resized by the user.
|
|
static int pos = 0;
|
|
MoveBy(pos * 25, pos * 25);
|
|
pos = (pos + 1) % 15;
|
|
|
|
BRect frame = Settings::Default()->AudioPlayerWindowFrame();
|
|
if (frame.IsValid()) {
|
|
if (isFirstWindow) {
|
|
if (message == NULL) {
|
|
MoveTo(frame.LeftTop());
|
|
ResizeTo(frame.Width(), frame.Height());
|
|
} else {
|
|
// Delay moving to the initial position, since we don't
|
|
// know if we will be playing audio at all.
|
|
message->AddRect("window frame", frame);
|
|
}
|
|
}
|
|
if (sNoVideoWidth == MIN_WIDTH)
|
|
sNoVideoWidth = frame.IntegerWidth();
|
|
} else if (sNoVideoWidth > MIN_WIDTH) {
|
|
ResizeTo(sNoVideoWidth, Bounds().Height());
|
|
}
|
|
fNoVideoWidth = sNoVideoWidth;
|
|
|
|
BRect rect = Bounds();
|
|
|
|
// background
|
|
fBackground = new BView(rect, "background", B_FOLLOW_ALL,
|
|
B_WILL_DRAW | B_FULL_UPDATE_ON_RESIZE);
|
|
fBackground->SetViewColor(0, 0, 0);
|
|
AddChild(fBackground);
|
|
|
|
// menu
|
|
fMenuBar = new BMenuBar(fBackground->Bounds(), "menu");
|
|
_CreateMenu();
|
|
fBackground->AddChild(fMenuBar);
|
|
fMenuBar->SetResizingMode(B_FOLLOW_NONE);
|
|
fMenuBar->ResizeToPreferred();
|
|
fMenuBarWidth = (int)fMenuBar->Frame().Width() + 1;
|
|
fMenuBarHeight = (int)fMenuBar->Frame().Height() + 1;
|
|
|
|
// video view
|
|
rect = BRect(0, fMenuBarHeight, fBackground->Bounds().right,
|
|
fMenuBarHeight + 10);
|
|
fVideoView = new VideoView(rect, "video display", B_FOLLOW_NONE);
|
|
fBackground->AddChild(fVideoView);
|
|
|
|
// controls
|
|
rect = BRect(0, fMenuBarHeight + 11, fBackground->Bounds().right,
|
|
fBackground->Bounds().bottom);
|
|
fControls = new ControllerView(rect, fController, fPlaylist);
|
|
fBackground->AddChild(fControls);
|
|
fControls->ResizeToPreferred();
|
|
fControlsHeight = (int)fControls->Frame().Height() + 1;
|
|
fControlsWidth = (int)fControls->Frame().Width() + 1;
|
|
fControls->SetResizingMode(B_FOLLOW_BOTTOM | B_FOLLOW_LEFT_RIGHT);
|
|
fControls->SetDisabledString(kDisabledSeekMessage);
|
|
|
|
fPlaylist->AddListener(fPlaylistObserver);
|
|
fController->SetVideoView(fVideoView);
|
|
fController->AddListener(fControllerObserver);
|
|
PeakView* peakView = fControls->GetPeakView();
|
|
peakView->SetPeakNotificationWhat(MSG_PEAK_NOTIFICATION);
|
|
fController->SetPeakListener(peakView);
|
|
|
|
_SetupWindow();
|
|
|
|
// setup the playlist window now, we need to have it
|
|
// running for the undo/redo playlist editing
|
|
fPlaylistWindow = new PlaylistWindow(BRect(150, 150, 500, 600), fPlaylist,
|
|
fController);
|
|
fPlaylistWindow->Hide();
|
|
fPlaylistWindow->Show();
|
|
// this makes sure the window thread is running without
|
|
// showing the window just yet
|
|
|
|
Settings::Default()->AddListener(&fGlobalSettingsListener);
|
|
_AdoptGlobalSettings();
|
|
|
|
AddShortcut('z', B_COMMAND_KEY, new BMessage(B_UNDO));
|
|
AddShortcut('y', B_COMMAND_KEY, new BMessage(B_UNDO));
|
|
AddShortcut('z', B_COMMAND_KEY | B_SHIFT_KEY, new BMessage(B_REDO));
|
|
AddShortcut('y', B_COMMAND_KEY | B_SHIFT_KEY, new BMessage(B_REDO));
|
|
|
|
Hide();
|
|
Show();
|
|
|
|
if (message != NULL)
|
|
PostMessage(message);
|
|
|
|
BMediaRoster* roster = BMediaRoster::Roster();
|
|
roster->StartWatching(BMessenger(this, this), B_MEDIA_SERVER_STARTED);
|
|
roster->StartWatching(BMessenger(this, this), B_MEDIA_SERVER_QUIT);
|
|
}
|
|
|
|
|
|
MainWin::~MainWin()
|
|
{
|
|
// printf("MainWin::~MainWin\n");
|
|
|
|
BMediaRoster* roster = BMediaRoster::CurrentRoster();
|
|
roster->StopWatching(BMessenger(this, this), B_MEDIA_SERVER_STARTED);
|
|
roster->StopWatching(BMessenger(this, this), B_MEDIA_SERVER_QUIT);
|
|
|
|
Settings::Default()->RemoveListener(&fGlobalSettingsListener);
|
|
fPlaylist->RemoveListener(fPlaylistObserver);
|
|
fController->Lock();
|
|
fController->RemoveListener(fControllerObserver);
|
|
fController->SetPeakListener(NULL);
|
|
fController->SetVideoTarget(NULL);
|
|
fController->Unlock();
|
|
|
|
// give the views a chance to detach from any notifiers
|
|
// before we delete them
|
|
fBackground->RemoveSelf();
|
|
delete fBackground;
|
|
|
|
if (fInfoWin && fInfoWin->Lock())
|
|
fInfoWin->Quit();
|
|
|
|
if (fPlaylistWindow && fPlaylistWindow->Lock())
|
|
fPlaylistWindow->Quit();
|
|
|
|
delete fPlaylist;
|
|
fPlaylist = NULL;
|
|
|
|
// quit the Controller looper thread
|
|
thread_id controllerThread = fController->Thread();
|
|
fController->PostMessage(B_QUIT_REQUESTED);
|
|
status_t exitValue;
|
|
wait_for_thread(controllerThread, &exitValue);
|
|
}
|
|
|
|
|
|
// #pragma mark -
|
|
|
|
|
|
void
|
|
MainWin::FrameResized(float newWidth, float newHeight)
|
|
{
|
|
if (newWidth != Bounds().Width() || newHeight != Bounds().Height()) {
|
|
debugger("size wrong\n");
|
|
}
|
|
|
|
bool noMenu = fNoInterface || fIsFullscreen;
|
|
bool noControls = fNoInterface || fIsFullscreen;
|
|
|
|
// printf("FrameResized enter: newWidth %.0f, newHeight %.0f\n",
|
|
// newWidth, newHeight);
|
|
|
|
if (!fHasVideo)
|
|
sNoVideoWidth = fNoVideoWidth = (int)newWidth;
|
|
|
|
int maxVideoWidth = int(newWidth) + 1;
|
|
int maxVideoHeight = int(newHeight) + 1
|
|
- (noMenu ? 0 : fMenuBarHeight)
|
|
- (noControls ? 0 : fControlsHeight);
|
|
|
|
ASSERT(maxVideoHeight >= 0);
|
|
|
|
int y = 0;
|
|
|
|
if (noMenu) {
|
|
if (!fMenuBar->IsHidden(fMenuBar))
|
|
fMenuBar->Hide();
|
|
} else {
|
|
fMenuBar->MoveTo(0, y);
|
|
fMenuBar->ResizeTo(newWidth, fMenuBarHeight - 1);
|
|
if (fMenuBar->IsHidden(fMenuBar))
|
|
fMenuBar->Show();
|
|
y += fMenuBarHeight;
|
|
}
|
|
|
|
if (maxVideoHeight == 0) {
|
|
if (!fVideoView->IsHidden(fVideoView))
|
|
fVideoView->Hide();
|
|
} else {
|
|
_ResizeVideoView(0, y, maxVideoWidth, maxVideoHeight);
|
|
if (fVideoView->IsHidden(fVideoView))
|
|
fVideoView->Show();
|
|
y += maxVideoHeight;
|
|
}
|
|
|
|
if (noControls) {
|
|
if (!fControls->IsHidden(fControls))
|
|
fControls->Hide();
|
|
} else {
|
|
fControls->MoveTo(0, y);
|
|
fControls->ResizeTo(newWidth, fControlsHeight - 1);
|
|
if (fControls->IsHidden(fControls))
|
|
fControls->Show();
|
|
// y += fControlsHeight;
|
|
}
|
|
|
|
// printf("FrameResized leave\n");
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::Zoom(BPoint /*position*/, float /*width*/, float /*height*/)
|
|
{
|
|
PostMessage(M_TOGGLE_FULLSCREEN);
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::DispatchMessage(BMessage* msg, BHandler* handler)
|
|
{
|
|
if ((msg->what == B_MOUSE_DOWN)
|
|
&& (handler == fBackground || handler == fVideoView
|
|
|| handler == fControls)) {
|
|
_MouseDown(msg, dynamic_cast<BView*>(handler));
|
|
}
|
|
|
|
if ((msg->what == B_MOUSE_MOVED)
|
|
&& (handler == fBackground || handler == fVideoView
|
|
|| handler == fControls)) {
|
|
_MouseMoved(msg, dynamic_cast<BView*>(handler));
|
|
}
|
|
|
|
if ((msg->what == B_MOUSE_UP)
|
|
&& (handler == fBackground || handler == fVideoView)) {
|
|
_MouseUp(msg);
|
|
}
|
|
|
|
if ((msg->what == B_KEY_DOWN)
|
|
&& (handler == fBackground || handler == fVideoView)) {
|
|
// special case for PrintScreen key
|
|
if (msg->FindInt32("key") == B_PRINT_KEY) {
|
|
fVideoView->OverlayScreenshotPrepare();
|
|
BWindow::DispatchMessage(msg, handler);
|
|
fVideoView->OverlayScreenshotCleanup();
|
|
return;
|
|
}
|
|
|
|
// every other key gets dispatched to our _KeyDown first
|
|
if (_KeyDown(msg)) {
|
|
// it got handled, don't pass it on
|
|
return;
|
|
}
|
|
}
|
|
|
|
BWindow::DispatchMessage(msg, handler);
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::MessageReceived(BMessage* msg)
|
|
{
|
|
// msg->PrintToStream();
|
|
switch (msg->what) {
|
|
case B_EXECUTE_PROPERTY:
|
|
case B_GET_PROPERTY:
|
|
case B_SET_PROPERTY:
|
|
{
|
|
BMessage reply(B_REPLY);
|
|
status_t result = B_BAD_SCRIPT_SYNTAX;
|
|
int32 index;
|
|
BMessage specifier;
|
|
int32 what;
|
|
const char* property;
|
|
|
|
if (msg->GetCurrentSpecifier(&index, &specifier, &what,
|
|
&property) != B_OK) {
|
|
return BWindow::MessageReceived(msg);
|
|
}
|
|
|
|
BPropertyInfo propertyInfo(sPropertyInfo);
|
|
switch (propertyInfo.FindMatch(msg, index, &specifier, what,
|
|
property)) {
|
|
case 0:
|
|
fControls->SkipForward();
|
|
result = B_OK;
|
|
break;
|
|
|
|
case 1:
|
|
fControls->SkipBackward();
|
|
result = B_OK;
|
|
break;
|
|
|
|
case 2:
|
|
fController->Play();
|
|
result = B_OK;
|
|
break;
|
|
|
|
case 3:
|
|
fController->Stop();
|
|
result = B_OK;
|
|
break;
|
|
|
|
case 4:
|
|
fController->Pause();
|
|
result = B_OK;
|
|
break;
|
|
|
|
case 5:
|
|
fController->TogglePlaying();
|
|
result = B_OK;
|
|
break;
|
|
|
|
case 6:
|
|
fController->ToggleMute();
|
|
result = B_OK;
|
|
break;
|
|
|
|
case 7:
|
|
{
|
|
if (msg->what == B_GET_PROPERTY) {
|
|
result = reply.AddFloat("result",
|
|
fController->Volume());
|
|
} else if (msg->what == B_SET_PROPERTY) {
|
|
float newVolume;
|
|
result = msg->FindFloat("data", &newVolume);
|
|
if (result == B_OK)
|
|
fController->SetVolume(newVolume);
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 8:
|
|
{
|
|
if (msg->what == B_GET_PROPERTY) {
|
|
BAutolock _(fPlaylist);
|
|
const PlaylistItem* item = fController->Item();
|
|
if (item == NULL) {
|
|
result = B_NO_INIT;
|
|
break;
|
|
}
|
|
|
|
result = reply.AddString("result", item->LocationURI());
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 9:
|
|
{
|
|
if (msg->what == B_GET_PROPERTY) {
|
|
BAutolock _(fPlaylist);
|
|
const PlaylistItem* item = fController->Item();
|
|
if (item == NULL) {
|
|
result = B_NO_INIT;
|
|
break;
|
|
}
|
|
|
|
result = reply.AddInt32("result", item->TrackNumber());
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 10:
|
|
PostMessage(M_TOGGLE_FULLSCREEN);
|
|
break;
|
|
|
|
case 11:
|
|
if (msg->what != B_GET_PROPERTY)
|
|
break;
|
|
|
|
result = reply.AddInt64("result",
|
|
fController->TimeDuration());
|
|
break;
|
|
|
|
case 12:
|
|
{
|
|
if (msg->what == B_GET_PROPERTY) {
|
|
result = reply.AddInt64("result",
|
|
fController->TimePosition());
|
|
} else if (msg->what == B_SET_PROPERTY) {
|
|
int64 newTime;
|
|
result = msg->FindInt64("data", &newTime);
|
|
if (result == B_OK)
|
|
fController->SetTimePosition(newTime);
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case 13:
|
|
{
|
|
if (msg->what != B_SET_PROPERTY)
|
|
break;
|
|
|
|
bigtime_t seekBy;
|
|
result = msg->FindInt64("data", &seekBy);
|
|
if (result != B_OK)
|
|
break;
|
|
|
|
_Wind(seekBy, 0);
|
|
break;
|
|
}
|
|
|
|
case 14:
|
|
result = reply.AddInt16("result", fPlaylist->CountItems());
|
|
break;
|
|
|
|
case 15:
|
|
{
|
|
int32 i = specifier.GetInt32("index", 0);
|
|
if (i >= fPlaylist->CountItems()) {
|
|
result = B_NO_INIT;
|
|
break;
|
|
}
|
|
|
|
BAutolock _(fPlaylist);
|
|
const PlaylistItem* item = fPlaylist->ItemAt(i);
|
|
result = item == NULL ? B_NO_INIT
|
|
: reply.AddString("result", item->Title());
|
|
break;
|
|
}
|
|
|
|
default:
|
|
return BWindow::MessageReceived(msg);
|
|
}
|
|
|
|
if (result != B_OK) {
|
|
reply.what = B_MESSAGE_NOT_UNDERSTOOD;
|
|
reply.AddString("message", strerror(result));
|
|
reply.AddInt32("error", result);
|
|
}
|
|
|
|
msg->SendReply(&reply);
|
|
break;
|
|
}
|
|
|
|
case B_REFS_RECEIVED:
|
|
case M_URL_RECEIVED:
|
|
_RefsReceived(msg);
|
|
break;
|
|
|
|
case B_SIMPLE_DATA:
|
|
if (msg->HasRef("refs"))
|
|
_RefsReceived(msg);
|
|
break;
|
|
case M_OPEN_PREVIOUS_PLAYLIST:
|
|
OpenPlaylist(msg);
|
|
break;
|
|
|
|
case B_UNDO:
|
|
case B_REDO:
|
|
fPlaylistWindow->PostMessage(msg);
|
|
break;
|
|
|
|
case B_MEDIA_SERVER_STARTED:
|
|
{
|
|
printf("TODO: implement B_MEDIA_SERVER_STARTED\n");
|
|
//
|
|
// BAutolock _(fPlaylist);
|
|
// BMessage fakePlaylistMessage(MSG_PLAYLIST_CURRENT_ITEM_CHANGED);
|
|
// fakePlaylistMessage.AddInt32("index",
|
|
// fPlaylist->CurrentItemIndex());
|
|
// PostMessage(&fakePlaylistMessage);
|
|
break;
|
|
}
|
|
|
|
case B_MEDIA_SERVER_QUIT:
|
|
printf("TODO: implement B_MEDIA_SERVER_QUIT\n");
|
|
// if (fController->Lock()) {
|
|
// fController->CleanupNodes();
|
|
// fController->Unlock();
|
|
// }
|
|
break;
|
|
|
|
// PlaylistObserver messages
|
|
case MSG_PLAYLIST_ITEM_ADDED:
|
|
{
|
|
PlaylistItem* item;
|
|
int32 index;
|
|
if (msg->FindPointer("item", (void**)&item) == B_OK
|
|
&& msg->FindInt32("index", &index) == B_OK) {
|
|
_AddPlaylistItem(item, index);
|
|
}
|
|
break;
|
|
}
|
|
case MSG_PLAYLIST_ITEM_REMOVED:
|
|
{
|
|
int32 index;
|
|
if (msg->FindInt32("index", &index) == B_OK)
|
|
_RemovePlaylistItem(index);
|
|
break;
|
|
}
|
|
case MSG_PLAYLIST_CURRENT_ITEM_CHANGED:
|
|
{
|
|
BAutolock _(fPlaylist);
|
|
|
|
int32 index;
|
|
// if false, the message was meant to only update the GUI
|
|
bool play;
|
|
if (msg->FindBool("play", &play) < B_OK || !play)
|
|
break;
|
|
if (msg->FindInt32("index", &index) < B_OK
|
|
|| index != fPlaylist->CurrentItemIndex())
|
|
break;
|
|
PlaylistItemRef item(fPlaylist->ItemAt(index));
|
|
if (item.IsSet()) {
|
|
printf("open playlist item: %s\n", item->Name().String());
|
|
OpenPlaylistItem(item);
|
|
_MarkPlaylistItem(index);
|
|
}
|
|
break;
|
|
}
|
|
case MSG_PLAYLIST_IMPORT_FAILED:
|
|
{
|
|
BAlert* alert = new BAlert(B_TRANSLATE("Nothing to Play"),
|
|
B_TRANSLATE("None of the files you wanted to play appear "
|
|
"to be media files."), B_TRANSLATE("OK"));
|
|
alert->SetFlags(alert->Flags() | B_CLOSE_ON_ESCAPE);
|
|
alert->Go();
|
|
fControls->SetDisabledString(kDisabledSeekMessage);
|
|
break;
|
|
}
|
|
|
|
// ControllerObserver messages
|
|
case MSG_CONTROLLER_FILE_FINISHED:
|
|
{
|
|
BAutolock _(fPlaylist);
|
|
|
|
//The file is finished. Open at start next time.
|
|
fController->SaveState(true);
|
|
|
|
bool hadNext = fPlaylist->SetCurrentItemIndex(
|
|
fPlaylist->CurrentItemIndex() + 1);
|
|
if (!hadNext) {
|
|
// Reached end of playlist
|
|
// Handle "quit when done" settings
|
|
if ((fHasVideo && fCloseWhenDonePlayingMovie)
|
|
|| (!fHasVideo && fCloseWhenDonePlayingSound))
|
|
PostMessage(B_QUIT_REQUESTED);
|
|
// Handle "loop by default" settings
|
|
if ((fHasVideo && fLoopMovies)
|
|
|| (!fHasVideo && fLoopSounds)) {
|
|
if (fPlaylist->CountItems() > 1)
|
|
fPlaylist->SetCurrentItemIndex(0);
|
|
else
|
|
fController->Play();
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case MSG_CONTROLLER_FILE_CHANGED:
|
|
{
|
|
status_t result = B_ERROR;
|
|
msg->FindInt32("result", &result);
|
|
PlaylistItemRef itemRef;
|
|
PlaylistItem* item;
|
|
if (msg->FindPointer("item", (void**)&item) == B_OK) {
|
|
itemRef.SetTo(item, true);
|
|
// The reference was passed along with the message.
|
|
} else {
|
|
BAutolock _(fPlaylist);
|
|
itemRef.SetTo(fPlaylist->ItemAt(
|
|
fPlaylist->CurrentItemIndex()));
|
|
}
|
|
_PlaylistItemOpened(itemRef, result);
|
|
break;
|
|
}
|
|
case MSG_CONTROLLER_VIDEO_TRACK_CHANGED:
|
|
{
|
|
int32 index;
|
|
if (msg->FindInt32("index", &index) == B_OK) {
|
|
int32 i = 0;
|
|
while (BMenuItem* item = fVideoTrackMenu->ItemAt(i)) {
|
|
item->SetMarked(i == index);
|
|
i++;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case MSG_CONTROLLER_AUDIO_TRACK_CHANGED:
|
|
{
|
|
int32 index;
|
|
if (msg->FindInt32("index", &index) == B_OK) {
|
|
int32 i = 0;
|
|
while (BMenuItem* item = fAudioTrackMenu->ItemAt(i)) {
|
|
item->SetMarked(i == index);
|
|
i++;
|
|
}
|
|
_UpdateAudioChannelCount(index);
|
|
}
|
|
break;
|
|
}
|
|
case MSG_CONTROLLER_SUB_TITLE_TRACK_CHANGED:
|
|
{
|
|
int32 index;
|
|
if (msg->FindInt32("index", &index) == B_OK) {
|
|
int32 i = 0;
|
|
while (BMenuItem* item = fSubTitleTrackMenu->ItemAt(i)) {
|
|
BMessage* message = item->Message();
|
|
if (message != NULL) {
|
|
item->SetMarked((int32)message->what
|
|
- M_SELECT_SUB_TITLE_TRACK == index);
|
|
}
|
|
i++;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case MSG_CONTROLLER_PLAYBACK_STATE_CHANGED:
|
|
{
|
|
uint32 state;
|
|
if (msg->FindInt32("state", (int32*)&state) == B_OK)
|
|
fControls->SetPlaybackState(state);
|
|
break;
|
|
}
|
|
case MSG_CONTROLLER_POSITION_CHANGED:
|
|
{
|
|
float position;
|
|
if (msg->FindFloat("position", &position) == B_OK) {
|
|
fControls->SetPosition(position, fController->TimePosition(),
|
|
fController->TimeDuration());
|
|
fAllowWinding = true;
|
|
}
|
|
break;
|
|
}
|
|
case MSG_CONTROLLER_SEEK_HANDLED:
|
|
break;
|
|
|
|
case MSG_CONTROLLER_VOLUME_CHANGED:
|
|
{
|
|
float volume;
|
|
if (msg->FindFloat("volume", &volume) == B_OK)
|
|
fControls->SetVolume(volume);
|
|
fController->SaveState();
|
|
break;
|
|
}
|
|
case MSG_CONTROLLER_MUTED_CHANGED:
|
|
{
|
|
bool muted;
|
|
if (msg->FindBool("muted", &muted) == B_OK)
|
|
fControls->SetMuted(muted);
|
|
break;
|
|
}
|
|
|
|
// menu item messages
|
|
case M_FILE_OPEN:
|
|
{
|
|
BMessenger target(this);
|
|
BMessage result(B_REFS_RECEIVED);
|
|
BMessage appMessage(M_SHOW_OPEN_PANEL);
|
|
appMessage.AddMessenger("target", target);
|
|
appMessage.AddMessage("message", &result);
|
|
appMessage.AddString("title", B_TRANSLATE("Open clips"));
|
|
appMessage.AddString("label", B_TRANSLATE("Open"));
|
|
be_app->PostMessage(&appMessage);
|
|
break;
|
|
}
|
|
|
|
case M_NETWORK_STREAM_OPEN:
|
|
{
|
|
BMessenger target(this);
|
|
NetworkStreamWin* win = new NetworkStreamWin(target);
|
|
win->Show();
|
|
break;
|
|
}
|
|
|
|
case M_EJECT_DEVICE:
|
|
Eject();
|
|
break;
|
|
|
|
case M_FILE_INFO:
|
|
ShowFileInfo();
|
|
break;
|
|
case M_FILE_PLAYLIST:
|
|
ShowPlaylistWindow();
|
|
break;
|
|
case M_FILE_CLOSE:
|
|
PostMessage(B_QUIT_REQUESTED);
|
|
break;
|
|
case M_FILE_QUIT:
|
|
be_app->PostMessage(B_QUIT_REQUESTED);
|
|
break;
|
|
|
|
case M_TOGGLE_FULLSCREEN:
|
|
_ToggleFullscreen();
|
|
break;
|
|
|
|
case M_TOGGLE_ALWAYS_ON_TOP:
|
|
_ToggleAlwaysOnTop();
|
|
break;
|
|
|
|
case M_TOGGLE_NO_INTERFACE:
|
|
_ToggleNoInterface();
|
|
break;
|
|
|
|
case M_VIEW_SIZE:
|
|
{
|
|
int32 size;
|
|
if (msg->FindInt32("size", &size) == B_OK) {
|
|
if (!fHasVideo)
|
|
break;
|
|
if (fIsFullscreen)
|
|
_ToggleFullscreen();
|
|
_ResizeWindow(size);
|
|
}
|
|
break;
|
|
}
|
|
|
|
/*
|
|
case B_ACQUIRE_OVERLAY_LOCK:
|
|
printf("B_ACQUIRE_OVERLAY_LOCK\n");
|
|
fVideoView->OverlayLockAcquire();
|
|
break;
|
|
|
|
case B_RELEASE_OVERLAY_LOCK:
|
|
printf("B_RELEASE_OVERLAY_LOCK\n");
|
|
fVideoView->OverlayLockRelease();
|
|
break;
|
|
*/
|
|
case B_MOUSE_WHEEL_CHANGED:
|
|
{
|
|
float dx = msg->FindFloat("be:wheel_delta_x");
|
|
float dy = msg->FindFloat("be:wheel_delta_y");
|
|
bool inv = modifiers() & B_COMMAND_KEY;
|
|
if (dx > 0.1)
|
|
PostMessage(inv ? M_VOLUME_DOWN : M_SKIP_PREV);
|
|
if (dx < -0.1)
|
|
PostMessage(inv ? M_VOLUME_UP : M_SKIP_NEXT);
|
|
if (dy > 0.1)
|
|
PostMessage(inv ? M_SKIP_PREV : M_VOLUME_DOWN);
|
|
if (dy < -0.1)
|
|
PostMessage(inv ? M_SKIP_NEXT : M_VOLUME_UP);
|
|
break;
|
|
}
|
|
|
|
case M_SKIP_NEXT:
|
|
fControls->SkipForward();
|
|
break;
|
|
|
|
case M_SKIP_PREV:
|
|
fControls->SkipBackward();
|
|
break;
|
|
|
|
case M_WIND:
|
|
{
|
|
bigtime_t howMuch;
|
|
int64 frames;
|
|
if (msg->FindInt64("how much", &howMuch) != B_OK
|
|
|| msg->FindInt64("frames", &frames) != B_OK) {
|
|
break;
|
|
}
|
|
|
|
_Wind(howMuch, frames);
|
|
break;
|
|
}
|
|
|
|
case M_VOLUME_UP:
|
|
fController->VolumeUp();
|
|
break;
|
|
|
|
case M_VOLUME_DOWN:
|
|
fController->VolumeDown();
|
|
break;
|
|
|
|
case M_ASPECT_SAME_AS_SOURCE:
|
|
if (fHasVideo) {
|
|
int width;
|
|
int height;
|
|
int widthAspect;
|
|
int heightAspect;
|
|
fController->GetSize(&width, &height,
|
|
&widthAspect, &heightAspect);
|
|
VideoFormatChange(width, height, widthAspect, heightAspect);
|
|
}
|
|
break;
|
|
|
|
case M_ASPECT_NO_DISTORTION:
|
|
if (fHasVideo) {
|
|
int width;
|
|
int height;
|
|
fController->GetSize(&width, &height);
|
|
VideoFormatChange(width, height, width, height);
|
|
}
|
|
break;
|
|
|
|
case M_ASPECT_4_3:
|
|
VideoAspectChange(4, 3);
|
|
break;
|
|
|
|
case M_ASPECT_16_9: // 1.77 : 1
|
|
VideoAspectChange(16, 9);
|
|
break;
|
|
|
|
case M_ASPECT_83_50: // 1.66 : 1
|
|
VideoAspectChange(83, 50);
|
|
break;
|
|
|
|
case M_ASPECT_7_4: // 1.75 : 1
|
|
VideoAspectChange(7, 4);
|
|
break;
|
|
|
|
case M_ASPECT_37_20: // 1.85 : 1
|
|
VideoAspectChange(37, 20);
|
|
break;
|
|
|
|
case M_ASPECT_47_20: // 2.35 : 1
|
|
VideoAspectChange(47, 20);
|
|
break;
|
|
|
|
case M_SET_PLAYLIST_POSITION:
|
|
{
|
|
BAutolock _(fPlaylist);
|
|
|
|
int32 index;
|
|
if (msg->FindInt32("index", &index) == B_OK)
|
|
fPlaylist->SetCurrentItemIndex(index);
|
|
break;
|
|
}
|
|
|
|
case MSG_OBJECT_CHANGED:
|
|
// received from fGlobalSettingsListener
|
|
// TODO: find out which object, if we ever watch more than
|
|
// the global settings instance...
|
|
_AdoptGlobalSettings();
|
|
break;
|
|
|
|
case M_SLIDE_CONTROLS:
|
|
{
|
|
float offset;
|
|
if (msg->FindFloat("offset", &offset) == B_OK) {
|
|
fControls->MoveBy(0, offset);
|
|
fVideoView->SetSubTitleMaxBottom(fControls->Frame().top - 1);
|
|
UpdateIfNeeded();
|
|
snooze(15000);
|
|
}
|
|
break;
|
|
}
|
|
case M_FINISH_SLIDING_CONTROLS:
|
|
{
|
|
float offset;
|
|
bool show;
|
|
if (msg->FindFloat("offset", &offset) == B_OK
|
|
&& msg->FindBool("show", &show) == B_OK) {
|
|
if (show) {
|
|
fControls->MoveTo(fControls->Frame().left, offset);
|
|
fVideoView->SetSubTitleMaxBottom(offset - 1);
|
|
} else {
|
|
fVideoView->SetSubTitleMaxBottom(
|
|
fVideoView->Bounds().bottom);
|
|
fControls->RemoveSelf();
|
|
fControls->MoveTo(fVideoView->Frame().left,
|
|
fVideoView->Frame().bottom + 1);
|
|
fBackground->AddChild(fControls);
|
|
fControls->SetSymbolScale(1.0f);
|
|
while (!fControls->IsHidden())
|
|
fControls->Hide();
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case M_HIDE_FULL_SCREEN_CONTROLS:
|
|
if (fIsFullscreen) {
|
|
BPoint videoViewWhere;
|
|
if (msg->FindPoint("where", &videoViewWhere) == B_OK) {
|
|
if (msg->FindBool("force")
|
|
|| !fControls->Frame().Contains(videoViewWhere)) {
|
|
_ShowFullscreenControls(false);
|
|
// hide the mouse cursor until the user moves it
|
|
be_app->ObscureCursor();
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
case M_SET_RATING:
|
|
{
|
|
int32 rating;
|
|
if (msg->FindInt32("rating", &rating) == B_OK)
|
|
_SetRating(rating);
|
|
break;
|
|
}
|
|
|
|
default:
|
|
if (msg->what >= M_SELECT_AUDIO_TRACK
|
|
&& msg->what <= M_SELECT_AUDIO_TRACK_END) {
|
|
fController->SelectAudioTrack(msg->what - M_SELECT_AUDIO_TRACK);
|
|
break;
|
|
}
|
|
if (msg->what >= M_SELECT_VIDEO_TRACK
|
|
&& msg->what <= M_SELECT_VIDEO_TRACK_END) {
|
|
fController->SelectVideoTrack(msg->what - M_SELECT_VIDEO_TRACK);
|
|
break;
|
|
}
|
|
if ((int32)msg->what >= M_SELECT_SUB_TITLE_TRACK - 1
|
|
&& msg->what <= M_SELECT_SUB_TITLE_TRACK_END) {
|
|
fController->SelectSubTitleTrack((int32)msg->what
|
|
- M_SELECT_SUB_TITLE_TRACK);
|
|
break;
|
|
}
|
|
// let BWindow handle the rest
|
|
BWindow::MessageReceived(msg);
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::WindowActivated(bool active)
|
|
{
|
|
fController->PlayerActivated(active);
|
|
}
|
|
|
|
|
|
bool
|
|
MainWin::QuitRequested()
|
|
{
|
|
fController->SaveState();
|
|
BMessage message(M_PLAYER_QUIT);
|
|
GetQuitMessage(&message);
|
|
be_app->PostMessage(&message);
|
|
return true;
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::MenusBeginning()
|
|
{
|
|
_SetupVideoAspectItems(fVideoAspectMenu);
|
|
}
|
|
|
|
|
|
// #pragma mark -
|
|
|
|
|
|
void
|
|
MainWin::OpenPlaylist(const BMessage* playlistArchive)
|
|
{
|
|
if (playlistArchive == NULL)
|
|
return;
|
|
|
|
BAutolock _(this);
|
|
BAutolock playlistLocker(fPlaylist);
|
|
|
|
if (fPlaylist->Unarchive(playlistArchive) != B_OK)
|
|
return;
|
|
|
|
int32 currentIndex;
|
|
if (playlistArchive->FindInt32("index", ¤tIndex) != B_OK)
|
|
currentIndex = 0;
|
|
fPlaylist->SetCurrentItemIndex(currentIndex);
|
|
|
|
playlistLocker.Unlock();
|
|
|
|
if (currentIndex != -1) {
|
|
// Restore the current play position only if we have something to play
|
|
playlistArchive->FindInt64("position", (int64*)&fInitialSeekPosition);
|
|
}
|
|
|
|
if (IsHidden())
|
|
Show();
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::OpenPlaylistItem(const PlaylistItemRef& item)
|
|
{
|
|
status_t ret = fController->SetToAsync(item);
|
|
if (ret != B_OK) {
|
|
fprintf(stderr, "MainWin::OpenPlaylistItem() - Failed to send message "
|
|
"to Controller.\n");
|
|
BString message = B_TRANSLATE("%app% encountered an internal error. "
|
|
"The file could not be opened.");
|
|
message.ReplaceFirst("%app%", kApplicationName);
|
|
BAlert* alert = new BAlert(kApplicationName, message.String(),
|
|
B_TRANSLATE("OK"), NULL, NULL, B_WIDTH_AS_USUAL, B_STOP_ALERT);
|
|
alert->SetFlags(alert->Flags() | B_CLOSE_ON_ESCAPE);
|
|
alert->Go();
|
|
_PlaylistItemOpened(item, ret);
|
|
} else {
|
|
BString string;
|
|
string.SetToFormat(B_TRANSLATE("Opening '%s'."), item->Name().String());
|
|
fControls->SetDisabledString(string.String());
|
|
}
|
|
}
|
|
|
|
|
|
static int
|
|
FindCdPlayerDevice(const char* directory)
|
|
{
|
|
BDirectory dir;
|
|
dir.SetTo(directory);
|
|
if (dir.InitCheck() != B_NO_ERROR)
|
|
return false;
|
|
dir.Rewind();
|
|
BEntry entry;
|
|
while (dir.GetNextEntry(&entry) >= 0) {
|
|
BPath path;
|
|
if (entry.GetPath(&path) != B_NO_ERROR)
|
|
continue;
|
|
const char* name = path.Path();
|
|
entry_ref e;
|
|
if (entry.GetRef(&e) != B_NO_ERROR)
|
|
continue;
|
|
if (entry.IsDirectory()) {
|
|
if (strcmp(e.name, "floppy") == 0)
|
|
continue; // ignore floppy
|
|
int deviceFD = FindCdPlayerDevice(name);
|
|
if (deviceFD >= 0)
|
|
return deviceFD;
|
|
} else {
|
|
if (strcmp(e.name, "raw") != 0)
|
|
continue;
|
|
int deviceFD = open(name, O_RDONLY);
|
|
if (deviceFD < 0)
|
|
continue;
|
|
device_geometry geometry;
|
|
if (ioctl(deviceFD, B_GET_GEOMETRY, &geometry, sizeof(geometry)) >= 0
|
|
&& geometry.device_type == B_CD)
|
|
return deviceFD;
|
|
close(deviceFD);
|
|
}
|
|
}
|
|
return B_ERROR;
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::Eject()
|
|
{
|
|
status_t mediaStatus = B_DEV_NO_MEDIA;
|
|
// find the cd player device
|
|
int cdPlayerFd = FindCdPlayerDevice("/dev/disk");
|
|
// get the status first
|
|
ioctl(cdPlayerFd, B_GET_MEDIA_STATUS, &mediaStatus, sizeof(mediaStatus));
|
|
// if door open, load the media, else eject the cd
|
|
status_t result = ioctl(cdPlayerFd,
|
|
mediaStatus == B_DEV_DOOR_OPEN ? B_LOAD_MEDIA : B_EJECT_DEVICE);
|
|
if (result != B_NO_ERROR)
|
|
printf("Error ejecting device");
|
|
close(cdPlayerFd);
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::ShowFileInfo()
|
|
{
|
|
if (!fInfoWin)
|
|
fInfoWin = new InfoWin(Frame().LeftTop(), fController);
|
|
|
|
if (fInfoWin->Lock()) {
|
|
if (fInfoWin->IsHidden())
|
|
fInfoWin->Show();
|
|
else
|
|
fInfoWin->Activate();
|
|
fInfoWin->Unlock();
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::ShowPlaylistWindow()
|
|
{
|
|
if (fPlaylistWindow->Lock()) {
|
|
// make sure the window shows on the same workspace as ourself
|
|
uint32 workspaces = Workspaces();
|
|
if (fPlaylistWindow->Workspaces() != workspaces)
|
|
fPlaylistWindow->SetWorkspaces(workspaces);
|
|
|
|
// show or activate
|
|
if (fPlaylistWindow->IsHidden())
|
|
fPlaylistWindow->Show();
|
|
else
|
|
fPlaylistWindow->Activate();
|
|
|
|
fPlaylistWindow->Unlock();
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::VideoAspectChange(int forcedWidth, int forcedHeight, float widthScale)
|
|
{
|
|
// Force specific source size and pixel width scale.
|
|
if (fHasVideo) {
|
|
int width;
|
|
int height;
|
|
fController->GetSize(&width, &height);
|
|
VideoFormatChange(forcedWidth, forcedHeight,
|
|
lround(width * widthScale), height);
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::VideoAspectChange(float widthScale)
|
|
{
|
|
// Called when video aspect ratio changes and the original
|
|
// width/height should be restored too, display aspect is not known,
|
|
// only pixel width scale.
|
|
if (fHasVideo) {
|
|
int width;
|
|
int height;
|
|
fController->GetSize(&width, &height);
|
|
VideoFormatChange(width, height, lround(width * widthScale), height);
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::VideoAspectChange(int widthAspect, int heightAspect)
|
|
{
|
|
// Called when video aspect ratio changes and the original
|
|
// width/height should be restored too.
|
|
if (fHasVideo) {
|
|
int width;
|
|
int height;
|
|
fController->GetSize(&width, &height);
|
|
VideoFormatChange(width, height, widthAspect, heightAspect);
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::VideoFormatChange(int width, int height, int widthAspect,
|
|
int heightAspect)
|
|
{
|
|
// Called when video format or aspect ratio changes.
|
|
|
|
printf("VideoFormatChange enter: width %d, height %d, "
|
|
"aspect ratio: %d:%d\n", width, height, widthAspect, heightAspect);
|
|
|
|
// remember current view scale
|
|
int percent = _CurrentVideoSizeInPercent();
|
|
|
|
fSourceWidth = width;
|
|
fSourceHeight = height;
|
|
fWidthAspect = widthAspect;
|
|
fHeightAspect = heightAspect;
|
|
|
|
if (percent == 100)
|
|
_ResizeWindow(100);
|
|
else
|
|
FrameResized(Bounds().Width(), Bounds().Height());
|
|
|
|
printf("VideoFormatChange leave\n");
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::GetQuitMessage(BMessage* message)
|
|
{
|
|
message->AddPointer("instance", this);
|
|
message->AddRect("window frame", Frame());
|
|
message->AddBool("audio only", !fHasVideo);
|
|
message->AddInt64("creation time", fCreationTime);
|
|
|
|
if (!fHasVideo && fHasAudio) {
|
|
// store playlist, current index and position if this is audio
|
|
BMessage playlistArchive;
|
|
|
|
BAutolock controllerLocker(fController);
|
|
playlistArchive.AddInt64("position", fController->TimePosition());
|
|
controllerLocker.Unlock();
|
|
|
|
if (!fPlaylist)
|
|
return;
|
|
|
|
BAutolock playlistLocker(fPlaylist);
|
|
if (fPlaylist->Archive(&playlistArchive) != B_OK
|
|
|| playlistArchive.AddInt32("index",
|
|
fPlaylist->CurrentItemIndex()) != B_OK
|
|
|| message->AddMessage("playlist", &playlistArchive) != B_OK) {
|
|
fprintf(stderr, "Failed to store current playlist.\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
BHandler*
|
|
MainWin::ResolveSpecifier(BMessage* message, int32 index, BMessage* specifier,
|
|
int32 what, const char* property)
|
|
{
|
|
BPropertyInfo propertyInfo(sPropertyInfo);
|
|
if (propertyInfo.FindMatch(message, index, specifier, what, property)
|
|
!= B_ERROR)
|
|
return this;
|
|
|
|
return BWindow::ResolveSpecifier(message, index, specifier, what, property);
|
|
}
|
|
|
|
|
|
status_t
|
|
MainWin::GetSupportedSuites(BMessage* data)
|
|
{
|
|
if (data == NULL)
|
|
return B_BAD_VALUE;
|
|
|
|
status_t status = data->AddString("suites", "suite/vnd.Haiku-MediaPlayer");
|
|
if (status != B_OK)
|
|
return status;
|
|
|
|
BPropertyInfo propertyInfo(sPropertyInfo);
|
|
status = data->AddFlat("messages", &propertyInfo);
|
|
if (status != B_OK)
|
|
return status;
|
|
|
|
return BWindow::GetSupportedSuites(data);
|
|
}
|
|
|
|
|
|
// #pragma mark -
|
|
|
|
|
|
void
|
|
MainWin::_RefsReceived(BMessage* message)
|
|
{
|
|
// the playlist is replaced by dropped files
|
|
// or the dropped files are appended to the end
|
|
// of the existing playlist if <shift> is pressed
|
|
bool append = false;
|
|
if (message->FindBool("append to playlist", &append) != B_OK)
|
|
append = modifiers() & B_SHIFT_KEY;
|
|
|
|
BAutolock _(fPlaylist);
|
|
int32 appendIndex = append ? APPEND_INDEX_APPEND_LAST
|
|
: APPEND_INDEX_REPLACE_PLAYLIST;
|
|
message->AddInt32("append_index", appendIndex);
|
|
|
|
// forward the message to the playlist window,
|
|
// so that undo/redo is used for modifying the playlist
|
|
fPlaylistWindow->PostMessage(message);
|
|
|
|
if (message->FindRect("window frame", &fNoVideoFrame) != B_OK)
|
|
fNoVideoFrame = BRect();
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_PlaylistItemOpened(const PlaylistItemRef& item, status_t result)
|
|
{
|
|
if (result != B_OK) {
|
|
BAutolock _(fPlaylist);
|
|
|
|
item->SetPlaybackFailed();
|
|
bool allItemsFailed = true;
|
|
int32 count = fPlaylist->CountItems();
|
|
for (int32 i = 0; i < count; i++) {
|
|
if (!fPlaylist->ItemAtFast(i)->PlaybackFailed()) {
|
|
allItemsFailed = false;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (allItemsFailed) {
|
|
// Display error if all files failed to play.
|
|
BString message(B_TRANSLATE(
|
|
"The file '%filename' could not be opened.\n\n"));;
|
|
message.ReplaceAll("%filename", item->Name());
|
|
|
|
if (result == B_MEDIA_NO_HANDLER) {
|
|
// give a more detailed message for the most likely of all
|
|
// errors
|
|
message << B_TRANSLATE(
|
|
"There is no decoder installed to handle the "
|
|
"file format, or the decoder has trouble with the "
|
|
"specific version of the format.");
|
|
} else {
|
|
message << B_TRANSLATE("Error: ") << strerror(result);
|
|
}
|
|
BAlert* alert = new BAlert("error", message.String(),
|
|
B_TRANSLATE("OK"), NULL, NULL, B_WIDTH_AS_USUAL, B_STOP_ALERT);
|
|
alert->SetFlags(alert->Flags() | B_CLOSE_ON_ESCAPE);
|
|
alert->Go();
|
|
fControls->SetDisabledString(kDisabledSeekMessage);
|
|
} else {
|
|
// Just go to the next file and don't bother user (yet)
|
|
fPlaylist->SetCurrentItemIndex(fPlaylist->CurrentItemIndex() + 1);
|
|
}
|
|
|
|
fHasFile = false;
|
|
fHasVideo = false;
|
|
fHasAudio = false;
|
|
SetTitle(kApplicationName);
|
|
} else {
|
|
fHasFile = true;
|
|
fHasVideo = fController->VideoTrackCount() != 0;
|
|
fHasAudio = fController->AudioTrackCount() != 0;
|
|
SetTitle(item->Name().String());
|
|
|
|
if (fInitialSeekPosition < 0) {
|
|
fInitialSeekPosition
|
|
= fController->TimeDuration() + fInitialSeekPosition;
|
|
}
|
|
fController->SetTimePosition(fInitialSeekPosition);
|
|
fInitialSeekPosition = 0;
|
|
|
|
if (fPlaylist->CountItems() == 1)
|
|
fController->RestoreState();
|
|
}
|
|
_SetupWindow();
|
|
|
|
if (result == B_OK)
|
|
_UpdatePlaylistItemFile();
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_SetupWindow()
|
|
{
|
|
// printf("MainWin::_SetupWindow\n");
|
|
// Populate the track menus
|
|
_SetupTrackMenus(fAudioTrackMenu, fVideoTrackMenu, fSubTitleTrackMenu);
|
|
_UpdateAudioChannelCount(fController->CurrentAudioTrack());
|
|
|
|
fVideoMenu->SetEnabled(fHasVideo);
|
|
fAudioMenu->SetEnabled(fHasAudio);
|
|
int previousSourceWidth = fSourceWidth;
|
|
int previousSourceHeight = fSourceHeight;
|
|
int previousWidthAspect = fWidthAspect;
|
|
int previousHeightAspect = fHeightAspect;
|
|
if (fHasVideo) {
|
|
fController->GetSize(&fSourceWidth, &fSourceHeight,
|
|
&fWidthAspect, &fHeightAspect);
|
|
} else {
|
|
fSourceWidth = 0;
|
|
fSourceHeight = 0;
|
|
fWidthAspect = 1;
|
|
fHeightAspect = 1;
|
|
}
|
|
_UpdateControlsEnabledStatus();
|
|
|
|
// Adopt the size and window layout if necessary
|
|
if (previousSourceWidth != fSourceWidth
|
|
|| previousSourceHeight != fSourceHeight
|
|
|| previousWidthAspect != fWidthAspect
|
|
|| previousHeightAspect != fHeightAspect) {
|
|
|
|
_SetWindowSizeLimits();
|
|
|
|
if (!fIsFullscreen) {
|
|
// Resize to 100% but stay on screen
|
|
_ResizeWindow(100, !fHasVideo, true);
|
|
} else {
|
|
// Make sure we relayout the video view when in full screen mode
|
|
FrameResized(Frame().Width(), Frame().Height());
|
|
}
|
|
}
|
|
|
|
_ShowIfNeeded();
|
|
|
|
fVideoView->MakeFocus();
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_CreateMenu()
|
|
{
|
|
fFileMenu = new BMenu(kApplicationName);
|
|
fPlaylistMenu = new BMenu(B_TRANSLATE("Playlist" B_UTF8_ELLIPSIS));
|
|
fAudioMenu = new BMenu(B_TRANSLATE("Audio"));
|
|
fVideoMenu = new BMenu(B_TRANSLATE("Video"));
|
|
fVideoAspectMenu = new BMenu(B_TRANSLATE("Aspect ratio"));
|
|
fAudioTrackMenu = new BMenu(B_TRANSLATE_CONTEXT("Track",
|
|
"Audio Track Menu"));
|
|
fVideoTrackMenu = new BMenu(B_TRANSLATE_CONTEXT("Track",
|
|
"Video Track Menu"));
|
|
fSubTitleTrackMenu = new BMenu(B_TRANSLATE("Subtitles"));
|
|
fAttributesMenu = new BMenu(B_TRANSLATE("Attributes"));
|
|
|
|
fMenuBar->AddItem(fFileMenu);
|
|
fMenuBar->AddItem(fAudioMenu);
|
|
fMenuBar->AddItem(fVideoMenu);
|
|
fMenuBar->AddItem(fAttributesMenu);
|
|
|
|
BMenuItem* item = new BMenuItem(B_TRANSLATE("New player" B_UTF8_ELLIPSIS),
|
|
new BMessage(M_NEW_PLAYER), 'N');
|
|
fFileMenu->AddItem(item);
|
|
item->SetTarget(be_app);
|
|
|
|
// Add recent files to "Open File" entry as sub-menu.
|
|
BRecentFilesList recentFiles(10, false, NULL, kAppSig);
|
|
item = new BMenuItem(recentFiles.NewFileListMenu(
|
|
B_TRANSLATE("Open file" B_UTF8_ELLIPSIS), NULL, NULL, this, 10, true,
|
|
NULL, kAppSig), new BMessage(M_FILE_OPEN));
|
|
item->SetShortcut('O', 0);
|
|
fFileMenu->AddItem(item);
|
|
|
|
item = new BMenuItem(B_TRANSLATE("Open network stream"),
|
|
new BMessage(M_NETWORK_STREAM_OPEN));
|
|
fFileMenu->AddItem(item);
|
|
|
|
item = new BMenuItem(B_TRANSLATE("Eject Device"),
|
|
new BMessage(M_EJECT_DEVICE));
|
|
fFileMenu->AddItem(item);
|
|
|
|
fFileMenu->AddSeparatorItem();
|
|
|
|
fFileMenu->AddItem(new BMenuItem(B_TRANSLATE("File info" B_UTF8_ELLIPSIS),
|
|
new BMessage(M_FILE_INFO), 'I'));
|
|
fFileMenu->AddItem(fPlaylistMenu);
|
|
fPlaylistMenu->Superitem()->SetShortcut('P', B_COMMAND_KEY);
|
|
fPlaylistMenu->Superitem()->SetMessage(new BMessage(M_FILE_PLAYLIST));
|
|
|
|
fFileMenu->AddSeparatorItem();
|
|
|
|
fNoInterfaceMenuItem = new BMenuItem(B_TRANSLATE("Hide interface"),
|
|
new BMessage(M_TOGGLE_NO_INTERFACE), 'H');
|
|
fFileMenu->AddItem(fNoInterfaceMenuItem);
|
|
fFileMenu->AddItem(new BMenuItem(B_TRANSLATE("Always on top"),
|
|
new BMessage(M_TOGGLE_ALWAYS_ON_TOP), 'A'));
|
|
|
|
item = new BMenuItem(B_TRANSLATE("Settings" B_UTF8_ELLIPSIS),
|
|
new BMessage(M_SETTINGS), ',');
|
|
fFileMenu->AddItem(item);
|
|
item->SetTarget(be_app);
|
|
|
|
fFileMenu->AddSeparatorItem();
|
|
|
|
fFileMenu->AddItem(new BMenuItem(B_TRANSLATE("Close"),
|
|
new BMessage(M_FILE_CLOSE), 'W'));
|
|
fFileMenu->AddItem(new BMenuItem(B_TRANSLATE("Quit"),
|
|
new BMessage(M_FILE_QUIT), 'Q'));
|
|
|
|
fPlaylistMenu->SetRadioMode(true);
|
|
|
|
fAudioMenu->AddItem(fAudioTrackMenu);
|
|
|
|
fVideoMenu->AddItem(fVideoTrackMenu);
|
|
fVideoMenu->AddItem(fSubTitleTrackMenu);
|
|
fVideoMenu->AddSeparatorItem();
|
|
BMessage* resizeMessage = new BMessage(M_VIEW_SIZE);
|
|
resizeMessage->AddInt32("size", 50);
|
|
fVideoMenu->AddItem(new BMenuItem(
|
|
B_TRANSLATE("50% scale"), resizeMessage, '0'));
|
|
|
|
resizeMessage = new BMessage(M_VIEW_SIZE);
|
|
resizeMessage->AddInt32("size", 100);
|
|
fVideoMenu->AddItem(new BMenuItem(
|
|
B_TRANSLATE("100% scale"), resizeMessage, '1'));
|
|
|
|
resizeMessage = new BMessage(M_VIEW_SIZE);
|
|
resizeMessage->AddInt32("size", 200);
|
|
fVideoMenu->AddItem(new BMenuItem(
|
|
B_TRANSLATE("200% scale"), resizeMessage, '2'));
|
|
|
|
resizeMessage = new BMessage(M_VIEW_SIZE);
|
|
resizeMessage->AddInt32("size", 300);
|
|
fVideoMenu->AddItem(new BMenuItem(
|
|
B_TRANSLATE("300% scale"), resizeMessage, '3'));
|
|
|
|
resizeMessage = new BMessage(M_VIEW_SIZE);
|
|
resizeMessage->AddInt32("size", 400);
|
|
fVideoMenu->AddItem(new BMenuItem(
|
|
B_TRANSLATE("400% scale"), resizeMessage, '4'));
|
|
|
|
fVideoMenu->AddSeparatorItem();
|
|
|
|
fVideoMenu->AddItem(new BMenuItem(B_TRANSLATE("Full screen"),
|
|
new BMessage(M_TOGGLE_FULLSCREEN), B_ENTER));
|
|
|
|
fVideoMenu->AddSeparatorItem();
|
|
|
|
_SetupVideoAspectItems(fVideoAspectMenu);
|
|
fVideoMenu->AddItem(fVideoAspectMenu);
|
|
|
|
fRatingMenu = new BMenu(B_TRANSLATE("Rating"));
|
|
fAttributesMenu->AddItem(fRatingMenu);
|
|
for (int32 i = 1; i <= 10; i++) {
|
|
char label[16];
|
|
snprintf(label, sizeof(label), "%" B_PRId32, i);
|
|
BMessage* setRatingMsg = new BMessage(M_SET_RATING);
|
|
setRatingMsg->AddInt32("rating", i);
|
|
fRatingMenu->AddItem(new BMenuItem(label, setRatingMsg));
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_SetupVideoAspectItems(BMenu* menu)
|
|
{
|
|
BMenuItem* item;
|
|
while ((item = menu->RemoveItem((int32)0)) != NULL)
|
|
delete item;
|
|
|
|
int width;
|
|
int height;
|
|
int widthAspect;
|
|
int heightAspect;
|
|
fController->GetSize(&width, &height, &widthAspect, &heightAspect);
|
|
// We don't care if there is a video track at all. In that
|
|
// case we should end up not marking any item.
|
|
|
|
// NOTE: The item marking may end up marking for example both
|
|
// "Stream Settings" and "16 : 9" if the stream settings happen to
|
|
// be "16 : 9".
|
|
|
|
menu->AddItem(item = new BMenuItem(B_TRANSLATE("Stream settings"),
|
|
new BMessage(M_ASPECT_SAME_AS_SOURCE), '1', B_SHIFT_KEY));
|
|
item->SetMarked(widthAspect == fWidthAspect
|
|
&& heightAspect == fHeightAspect);
|
|
|
|
menu->AddItem(item = new BMenuItem(B_TRANSLATE("No aspect correction"),
|
|
new BMessage(M_ASPECT_NO_DISTORTION), '0', B_SHIFT_KEY));
|
|
item->SetMarked(width == fWidthAspect && height == fHeightAspect);
|
|
|
|
menu->AddSeparatorItem();
|
|
|
|
menu->AddItem(item = new BMenuItem("4 : 3",
|
|
new BMessage(M_ASPECT_4_3), 2, B_SHIFT_KEY));
|
|
item->SetMarked(fWidthAspect == 4 && fHeightAspect == 3);
|
|
menu->AddItem(item = new BMenuItem("16 : 9",
|
|
new BMessage(M_ASPECT_16_9), 3, B_SHIFT_KEY));
|
|
item->SetMarked(fWidthAspect == 16 && fHeightAspect == 9);
|
|
|
|
menu->AddSeparatorItem();
|
|
|
|
menu->AddItem(item = new BMenuItem("1.66 : 1",
|
|
new BMessage(M_ASPECT_83_50)));
|
|
item->SetMarked(fWidthAspect == 83 && fHeightAspect == 50);
|
|
menu->AddItem(item = new BMenuItem("1.75 : 1",
|
|
new BMessage(M_ASPECT_7_4)));
|
|
item->SetMarked(fWidthAspect == 7 && fHeightAspect == 4);
|
|
menu->AddItem(item = new BMenuItem(B_TRANSLATE("1.85 : 1 (American)"),
|
|
new BMessage(M_ASPECT_37_20)));
|
|
item->SetMarked(fWidthAspect == 37 && fHeightAspect == 20);
|
|
menu->AddItem(item = new BMenuItem(B_TRANSLATE("2.35 : 1 (Cinemascope)"),
|
|
new BMessage(M_ASPECT_47_20)));
|
|
item->SetMarked(fWidthAspect == 47 && fHeightAspect == 20);
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_SetupTrackMenus(BMenu* audioTrackMenu, BMenu* videoTrackMenu,
|
|
BMenu* subTitleTrackMenu)
|
|
{
|
|
audioTrackMenu->RemoveItems(0, audioTrackMenu->CountItems(), true);
|
|
videoTrackMenu->RemoveItems(0, videoTrackMenu->CountItems(), true);
|
|
subTitleTrackMenu->RemoveItems(0, subTitleTrackMenu->CountItems(), true);
|
|
|
|
char s[100];
|
|
|
|
int count = fController->AudioTrackCount();
|
|
int current = fController->CurrentAudioTrack();
|
|
for (int i = 0; i < count; i++) {
|
|
BMessage metaData;
|
|
const char* languageString = NULL;
|
|
if (fController->GetAudioMetaData(i, &metaData) == B_OK)
|
|
metaData.FindString("language", &languageString);
|
|
if (languageString != NULL) {
|
|
BLanguage language(languageString);
|
|
BString languageName;
|
|
if (language.GetName(languageName) == B_OK)
|
|
languageString = languageName.String();
|
|
snprintf(s, sizeof(s), "%s", languageString);
|
|
} else
|
|
snprintf(s, sizeof(s), B_TRANSLATE("Track %d"), i + 1);
|
|
BMenuItem* item = new BMenuItem(s,
|
|
new BMessage(M_SELECT_AUDIO_TRACK + i));
|
|
item->SetMarked(i == current);
|
|
audioTrackMenu->AddItem(item);
|
|
}
|
|
if (count == 0) {
|
|
audioTrackMenu->AddItem(new BMenuItem(B_TRANSLATE_CONTEXT("none",
|
|
"Audio track menu"), new BMessage(M_DUMMY)));
|
|
audioTrackMenu->ItemAt(0)->SetMarked(true);
|
|
}
|
|
audioTrackMenu->SetEnabled(count > 1);
|
|
|
|
count = fController->VideoTrackCount();
|
|
current = fController->CurrentVideoTrack();
|
|
for (int i = 0; i < count; i++) {
|
|
snprintf(s, sizeof(s), B_TRANSLATE("Track %d"), i + 1);
|
|
BMenuItem* item = new BMenuItem(s,
|
|
new BMessage(M_SELECT_VIDEO_TRACK + i));
|
|
item->SetMarked(i == current);
|
|
videoTrackMenu->AddItem(item);
|
|
}
|
|
if (count == 0) {
|
|
videoTrackMenu->AddItem(new BMenuItem(B_TRANSLATE("none"),
|
|
new BMessage(M_DUMMY)));
|
|
videoTrackMenu->ItemAt(0)->SetMarked(true);
|
|
}
|
|
videoTrackMenu->SetEnabled(count > 1);
|
|
|
|
count = fController->SubTitleTrackCount();
|
|
if (count > 0) {
|
|
current = fController->CurrentSubTitleTrack();
|
|
BMenuItem* item = new BMenuItem(
|
|
B_TRANSLATE_CONTEXT("Off", "Subtitles menu"),
|
|
new BMessage(M_SELECT_SUB_TITLE_TRACK - 1));
|
|
subTitleTrackMenu->AddItem(item);
|
|
item->SetMarked(current == -1);
|
|
|
|
subTitleTrackMenu->AddSeparatorItem();
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
const char* name = fController->SubTitleTrackName(i);
|
|
if (name != NULL)
|
|
snprintf(s, sizeof(s), "%s", name);
|
|
else
|
|
snprintf(s, sizeof(s), B_TRANSLATE("Track %d"), i + 1);
|
|
item = new BMenuItem(s,
|
|
new BMessage(M_SELECT_SUB_TITLE_TRACK + i));
|
|
item->SetMarked(i == current);
|
|
subTitleTrackMenu->AddItem(item);
|
|
}
|
|
} else {
|
|
subTitleTrackMenu->AddItem(new BMenuItem(
|
|
B_TRANSLATE_CONTEXT("none", "Subtitles menu"),
|
|
new BMessage(M_DUMMY)));
|
|
subTitleTrackMenu->ItemAt(0)->SetMarked(true);
|
|
}
|
|
subTitleTrackMenu->SetEnabled(count > 0);
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_UpdateAudioChannelCount(int32 audioTrackIndex)
|
|
{
|
|
fControls->SetAudioChannelCount(fController->AudioTrackChannelCount());
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_GetMinimumWindowSize(int& width, int& height) const
|
|
{
|
|
width = MIN_WIDTH;
|
|
height = 0;
|
|
if (!fNoInterface) {
|
|
width = max_c(width, fMenuBarWidth);
|
|
width = max_c(width, fControlsWidth);
|
|
height = fMenuBarHeight + fControlsHeight;
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_GetUnscaledVideoSize(int& videoWidth, int& videoHeight) const
|
|
{
|
|
if (fWidthAspect != 0 && fHeightAspect != 0) {
|
|
videoWidth = fSourceHeight * fWidthAspect / fHeightAspect;
|
|
videoHeight = fSourceWidth * fHeightAspect / fWidthAspect;
|
|
// Use the scaling which produces an enlarged view.
|
|
if (videoWidth > fSourceWidth) {
|
|
// Enlarge width
|
|
videoHeight = fSourceHeight;
|
|
} else {
|
|
// Enlarge height
|
|
videoWidth = fSourceWidth;
|
|
}
|
|
} else {
|
|
videoWidth = fSourceWidth;
|
|
videoHeight = fSourceHeight;
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_SetWindowSizeLimits()
|
|
{
|
|
int minWidth;
|
|
int minHeight;
|
|
_GetMinimumWindowSize(minWidth, minHeight);
|
|
SetSizeLimits(minWidth - 1, 32000, minHeight - 1,
|
|
fHasVideo ? 32000 : minHeight - 1);
|
|
}
|
|
|
|
|
|
int
|
|
MainWin::_CurrentVideoSizeInPercent() const
|
|
{
|
|
if (!fHasVideo)
|
|
return 0;
|
|
|
|
int videoWidth;
|
|
int videoHeight;
|
|
_GetUnscaledVideoSize(videoWidth, videoHeight);
|
|
|
|
int viewWidth = fVideoView->Bounds().IntegerWidth() + 1;
|
|
int viewHeight = fVideoView->Bounds().IntegerHeight() + 1;
|
|
|
|
int widthPercent = viewWidth * 100 / videoWidth;
|
|
int heightPercent = viewHeight * 100 / videoHeight;
|
|
|
|
if (widthPercent > heightPercent)
|
|
return widthPercent;
|
|
return heightPercent;
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_ZoomVideoView(int percentDiff)
|
|
{
|
|
if (!fHasVideo)
|
|
return;
|
|
|
|
int percent = _CurrentVideoSizeInPercent();
|
|
int newSize = percent * (100 + percentDiff) / 100;
|
|
|
|
if (newSize < 25)
|
|
newSize = 25;
|
|
if (newSize > 400)
|
|
newSize = 400;
|
|
if (newSize != percent) {
|
|
BMessage message(M_VIEW_SIZE);
|
|
message.AddInt32("size", newSize);
|
|
PostMessage(&message);
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_ResizeWindow(int percent, bool useNoVideoWidth, bool stayOnScreen)
|
|
{
|
|
// Get required window size
|
|
int videoWidth;
|
|
int videoHeight;
|
|
_GetUnscaledVideoSize(videoWidth, videoHeight);
|
|
|
|
videoWidth = (videoWidth * percent) / 100;
|
|
videoHeight = (videoHeight * percent) / 100;
|
|
|
|
// Calculate and set the minimum window size
|
|
int width;
|
|
int height;
|
|
_GetMinimumWindowSize(width, height);
|
|
|
|
width = max_c(width, videoWidth) - 1;
|
|
if (useNoVideoWidth)
|
|
width = max_c(width, fNoVideoWidth);
|
|
height = height + videoHeight - 1;
|
|
|
|
if (stayOnScreen) {
|
|
BRect screenFrame(BScreen(this).Frame());
|
|
BRect frame(Frame());
|
|
BRect decoratorFrame(DecoratorFrame());
|
|
|
|
// Shrink the screen frame by the window border size
|
|
screenFrame.top += frame.top - decoratorFrame.top;
|
|
screenFrame.left += frame.left - decoratorFrame.left;
|
|
screenFrame.right += frame.right - decoratorFrame.right;
|
|
screenFrame.bottom += frame.bottom - decoratorFrame.bottom;
|
|
|
|
// Update frame to what the new size would be
|
|
frame.right = frame.left + width;
|
|
frame.bottom = frame.top + height;
|
|
|
|
if (!screenFrame.Contains(frame)) {
|
|
// Resize the window so it doesn't extend outside the current
|
|
// screen frame.
|
|
// We don't use BWindow::MoveOnScreen() in order to resize the
|
|
// window while keeping the same aspect ratio.
|
|
if (frame.Width() > screenFrame.Width()
|
|
|| frame.Height() > screenFrame.Height()) {
|
|
// too large
|
|
int widthDiff
|
|
= frame.IntegerWidth() - screenFrame.IntegerWidth();
|
|
int heightDiff
|
|
= frame.IntegerHeight() - screenFrame.IntegerHeight();
|
|
|
|
float shrinkScale;
|
|
if (widthDiff > heightDiff)
|
|
shrinkScale = (float)(width - widthDiff) / width;
|
|
else
|
|
shrinkScale = (float)(height - heightDiff) / height;
|
|
|
|
// Resize width/height and center window
|
|
width = lround(width * shrinkScale);
|
|
height = lround(height * shrinkScale);
|
|
MoveTo((screenFrame.left + screenFrame.right - width) / 2,
|
|
(screenFrame.top + screenFrame.bottom - height) / 2);
|
|
} else {
|
|
// just off-screen on one or more sides
|
|
int offsetX = 0;
|
|
int offsetY = 0;
|
|
if (frame.left < screenFrame.left)
|
|
offsetX = (int)(screenFrame.left - frame.left);
|
|
else if (frame.right > screenFrame.right)
|
|
offsetX = (int)(screenFrame.right - frame.right);
|
|
if (frame.top < screenFrame.top)
|
|
offsetY = (int)(screenFrame.top - frame.top);
|
|
else if (frame.bottom > screenFrame.bottom)
|
|
offsetY = (int)(screenFrame.bottom - frame.bottom);
|
|
MoveBy(offsetX, offsetY);
|
|
}
|
|
}
|
|
}
|
|
|
|
ResizeTo(width, height);
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_ResizeVideoView(int x, int y, int width, int height)
|
|
{
|
|
// Keep aspect ratio, place video view inside
|
|
// the background area (may create black bars).
|
|
int videoWidth;
|
|
int videoHeight;
|
|
_GetUnscaledVideoSize(videoWidth, videoHeight);
|
|
float scaledWidth = videoWidth;
|
|
float scaledHeight = videoHeight;
|
|
float factor = min_c(width / scaledWidth, height / scaledHeight);
|
|
int renderWidth = lround(scaledWidth * factor);
|
|
int renderHeight = lround(scaledHeight * factor);
|
|
if (renderWidth > width)
|
|
renderWidth = width;
|
|
if (renderHeight > height)
|
|
renderHeight = height;
|
|
|
|
int xOffset = (width - renderWidth) / 2;
|
|
int yOffset = (height - renderHeight) / 2;
|
|
|
|
fVideoView->MoveTo(x, y);
|
|
fVideoView->ResizeTo(width - 1, height - 1);
|
|
|
|
BRect videoFrame(xOffset, yOffset,
|
|
xOffset + renderWidth - 1, yOffset + renderHeight - 1);
|
|
|
|
fVideoView->SetVideoFrame(videoFrame);
|
|
fVideoView->SetSubTitleMaxBottom(height - 1);
|
|
}
|
|
|
|
|
|
// #pragma mark -
|
|
|
|
|
|
void
|
|
MainWin::_MouseDown(BMessage* msg, BView* originalHandler)
|
|
{
|
|
uint32 buttons = msg->FindInt32("buttons");
|
|
|
|
// On Zeta, only "screen_where" is reliable, "where" and "be:view_where"
|
|
// seem to be broken
|
|
BPoint screenWhere;
|
|
if (msg->FindPoint("screen_where", &screenWhere) != B_OK) {
|
|
// TODO: remove
|
|
// Workaround for BeOS R5, it has no "screen_where"
|
|
if (!originalHandler || msg->FindPoint("where", &screenWhere) < B_OK)
|
|
return;
|
|
originalHandler->ConvertToScreen(&screenWhere);
|
|
}
|
|
|
|
// double click handling
|
|
|
|
if (msg->FindInt32("clicks") % 2 == 0) {
|
|
BRect rect(screenWhere.x - 1, screenWhere.y - 1, screenWhere.x + 1,
|
|
screenWhere.y + 1);
|
|
if (rect.Contains(fMouseDownMousePos)) {
|
|
if (buttons == B_PRIMARY_MOUSE_BUTTON)
|
|
PostMessage(M_TOGGLE_FULLSCREEN);
|
|
else if (buttons == B_SECONDARY_MOUSE_BUTTON)
|
|
PostMessage(M_TOGGLE_NO_INTERFACE);
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
fMouseDownMousePos = screenWhere;
|
|
fMouseDownWindowPos = Frame().LeftTop();
|
|
|
|
if (buttons == B_PRIMARY_MOUSE_BUTTON && !fIsFullscreen) {
|
|
// start mouse tracking
|
|
fVideoView->SetMouseEventMask(B_POINTER_EVENTS | B_NO_POINTER_HISTORY
|
|
/* | B_LOCK_WINDOW_FOCUS */);
|
|
fMouseDownTracking = true;
|
|
}
|
|
|
|
// pop up a context menu if right mouse button is down
|
|
|
|
if ((buttons & B_SECONDARY_MOUSE_BUTTON) != 0)
|
|
_ShowContextMenu(screenWhere);
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_MouseMoved(BMessage* msg, BView* originalHandler)
|
|
{
|
|
// msg->PrintToStream();
|
|
|
|
BPoint mousePos;
|
|
uint32 buttons = msg->FindInt32("buttons");
|
|
// On Zeta, only "screen_where" is reliable, "where"
|
|
// and "be:view_where" seem to be broken
|
|
if (msg->FindPoint("screen_where", &mousePos) != B_OK) {
|
|
// TODO: remove
|
|
// Workaround for BeOS R5, it has no "screen_where"
|
|
if (!originalHandler || msg->FindPoint("where", &mousePos) < B_OK)
|
|
return;
|
|
originalHandler->ConvertToScreen(&mousePos);
|
|
}
|
|
|
|
if (buttons == B_PRIMARY_MOUSE_BUTTON && fMouseDownTracking
|
|
&& !fIsFullscreen) {
|
|
// printf("screen where: %.0f, %.0f => ", mousePos.x, mousePos.y);
|
|
float delta_x = mousePos.x - fMouseDownMousePos.x;
|
|
float delta_y = mousePos.y - fMouseDownMousePos.y;
|
|
float x = fMouseDownWindowPos.x + delta_x;
|
|
float y = fMouseDownWindowPos.y + delta_y;
|
|
// printf("move window to %.0f, %.0f\n", x, y);
|
|
MoveTo(x, y);
|
|
}
|
|
|
|
bigtime_t eventTime;
|
|
if (msg->FindInt64("when", &eventTime) != B_OK)
|
|
eventTime = system_time();
|
|
|
|
if (buttons == 0 && fIsFullscreen) {
|
|
BPoint moveDelta = mousePos - fLastMousePos;
|
|
float moveDeltaDist
|
|
= sqrtf(moveDelta.x * moveDelta.x + moveDelta.y * moveDelta.y);
|
|
if (eventTime - fLastMouseMovedTime < 200000)
|
|
fMouseMoveDist += moveDeltaDist;
|
|
else
|
|
fMouseMoveDist = moveDeltaDist;
|
|
if (fMouseMoveDist > 5)
|
|
_ShowFullscreenControls(true);
|
|
}
|
|
|
|
fLastMousePos = mousePos;
|
|
fLastMouseMovedTime =eventTime;
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_MouseUp(BMessage* msg)
|
|
{
|
|
fMouseDownTracking = false;
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_ShowContextMenu(const BPoint& screenPoint)
|
|
{
|
|
printf("Show context menu\n");
|
|
BPopUpMenu* menu = new BPopUpMenu("context menu", false, false);
|
|
BMenuItem* item;
|
|
menu->AddItem(item = new BMenuItem(B_TRANSLATE("Full screen"),
|
|
new BMessage(M_TOGGLE_FULLSCREEN), B_ENTER));
|
|
item->SetMarked(fIsFullscreen);
|
|
item->SetEnabled(fHasVideo);
|
|
|
|
menu->AddItem(item = new BMenuItem(B_TRANSLATE("Hide interface"),
|
|
new BMessage(M_TOGGLE_NO_INTERFACE), 'H'));
|
|
item->SetMarked(fNoInterface);
|
|
item->SetEnabled(fHasVideo && !fIsFullscreen);
|
|
|
|
menu->AddItem(item = new BMenuItem(B_TRANSLATE("Always on top"),
|
|
new BMessage(M_TOGGLE_ALWAYS_ON_TOP), 'A'));
|
|
item->SetMarked(fAlwaysOnTop);
|
|
item->SetEnabled(fHasVideo);
|
|
|
|
BMenu* aspectSubMenu = new BMenu(B_TRANSLATE("Aspect ratio"));
|
|
_SetupVideoAspectItems(aspectSubMenu);
|
|
aspectSubMenu->SetTargetForItems(this);
|
|
menu->AddItem(item = new BMenuItem(aspectSubMenu));
|
|
item->SetEnabled(fHasVideo);
|
|
|
|
menu->AddSeparatorItem();
|
|
|
|
// Add track selector menus
|
|
BMenu* audioTrackMenu = new BMenu(B_TRANSLATE("Audio track"));
|
|
BMenu* videoTrackMenu = new BMenu(B_TRANSLATE("Video track"));
|
|
BMenu* subTitleTrackMenu = new BMenu(B_TRANSLATE("Subtitles"));
|
|
_SetupTrackMenus(audioTrackMenu, videoTrackMenu, subTitleTrackMenu);
|
|
|
|
audioTrackMenu->SetTargetForItems(this);
|
|
videoTrackMenu->SetTargetForItems(this);
|
|
subTitleTrackMenu->SetTargetForItems(this);
|
|
|
|
menu->AddItem(item = new BMenuItem(audioTrackMenu));
|
|
item->SetEnabled(fHasAudio);
|
|
|
|
menu->AddItem(item = new BMenuItem(videoTrackMenu));
|
|
item->SetEnabled(fHasVideo);
|
|
|
|
menu->AddItem(item = new BMenuItem(subTitleTrackMenu));
|
|
item->SetEnabled(fHasVideo);
|
|
|
|
menu->AddSeparatorItem();
|
|
menu->AddItem(new BMenuItem(B_TRANSLATE("Quit"), new BMessage(M_FILE_QUIT), 'Q'));
|
|
|
|
menu->SetTargetForItems(this);
|
|
BRect rect(screenPoint.x - 5, screenPoint.y - 5, screenPoint.x + 5,
|
|
screenPoint.y + 5);
|
|
menu->Go(screenPoint, true, true, rect, true);
|
|
}
|
|
|
|
|
|
/*! Trap keys that are about to be send to background or renderer view.
|
|
Return true if it shouldn't be passed to the view.
|
|
*/
|
|
bool
|
|
MainWin::_KeyDown(BMessage* msg)
|
|
{
|
|
uint32 key = msg->FindInt32("key");
|
|
uint32 rawChar = msg->FindInt32("raw_char");
|
|
uint32 modifier = msg->FindInt32("modifiers");
|
|
|
|
// printf("key 0x%lx, rawChar 0x%lx, modifiers 0x%lx\n", key, rawChar,
|
|
// modifier);
|
|
|
|
// ignore the system modifier namespace
|
|
if ((modifier & (B_CONTROL_KEY | B_COMMAND_KEY))
|
|
== (B_CONTROL_KEY | B_COMMAND_KEY))
|
|
return false;
|
|
|
|
switch (rawChar) {
|
|
case B_SPACE:
|
|
fController->TogglePlaying();
|
|
return true;
|
|
|
|
case 'm':
|
|
fController->ToggleMute();
|
|
return true;
|
|
|
|
case B_ESCAPE:
|
|
if (!fIsFullscreen)
|
|
break;
|
|
|
|
PostMessage(M_TOGGLE_FULLSCREEN);
|
|
return true;
|
|
|
|
case B_ENTER: // Enter / Return
|
|
if ((modifier & B_COMMAND_KEY) != 0) {
|
|
PostMessage(M_TOGGLE_FULLSCREEN);
|
|
return true;
|
|
}
|
|
break;
|
|
|
|
case B_TAB:
|
|
case 'f':
|
|
if ((modifier & (B_COMMAND_KEY | B_CONTROL_KEY | B_OPTION_KEY
|
|
| B_MENU_KEY)) == 0) {
|
|
PostMessage(M_TOGGLE_FULLSCREEN);
|
|
return true;
|
|
}
|
|
break;
|
|
|
|
case B_UP_ARROW:
|
|
if ((modifier & B_COMMAND_KEY) != 0)
|
|
PostMessage(M_SKIP_NEXT);
|
|
else
|
|
PostMessage(M_VOLUME_UP);
|
|
return true;
|
|
|
|
case B_DOWN_ARROW:
|
|
if ((modifier & B_COMMAND_KEY) != 0)
|
|
PostMessage(M_SKIP_PREV);
|
|
else
|
|
PostMessage(M_VOLUME_DOWN);
|
|
return true;
|
|
|
|
case B_RIGHT_ARROW:
|
|
if ((modifier & B_COMMAND_KEY) != 0)
|
|
PostMessage(M_SKIP_NEXT);
|
|
else if (fAllowWinding) {
|
|
BMessage windMessage(M_WIND);
|
|
if ((modifier & B_SHIFT_KEY) != 0) {
|
|
windMessage.AddInt64("how much", 30000000LL);
|
|
windMessage.AddInt64("frames", 5);
|
|
} else {
|
|
windMessage.AddInt64("how much", 5000000LL);
|
|
windMessage.AddInt64("frames", 1);
|
|
}
|
|
PostMessage(&windMessage);
|
|
}
|
|
return true;
|
|
|
|
case B_LEFT_ARROW:
|
|
if ((modifier & B_COMMAND_KEY) != 0)
|
|
PostMessage(M_SKIP_PREV);
|
|
else if (fAllowWinding) {
|
|
BMessage windMessage(M_WIND);
|
|
if ((modifier & B_SHIFT_KEY) != 0) {
|
|
windMessage.AddInt64("how much", -30000000LL);
|
|
windMessage.AddInt64("frames", -5);
|
|
} else {
|
|
windMessage.AddInt64("how much", -5000000LL);
|
|
windMessage.AddInt64("frames", -1);
|
|
}
|
|
PostMessage(&windMessage);
|
|
}
|
|
return true;
|
|
|
|
case B_PAGE_UP:
|
|
PostMessage(M_SKIP_NEXT);
|
|
return true;
|
|
|
|
case B_PAGE_DOWN:
|
|
PostMessage(M_SKIP_PREV);
|
|
return true;
|
|
|
|
case '+':
|
|
if ((modifier & B_COMMAND_KEY) == 0) {
|
|
_ZoomVideoView(10);
|
|
return true;
|
|
}
|
|
break;
|
|
|
|
case '-':
|
|
if ((modifier & B_COMMAND_KEY) == 0) {
|
|
_ZoomVideoView(-10);
|
|
return true;
|
|
}
|
|
break;
|
|
|
|
case B_DELETE:
|
|
case 'd': // d for delete
|
|
case 't': // t for Trash
|
|
if ((modifiers() & B_COMMAND_KEY) != 0) {
|
|
BAutolock _(fPlaylist);
|
|
BMessage removeMessage(M_PLAYLIST_MOVE_TO_TRASH);
|
|
removeMessage.AddInt32("playlist index",
|
|
fPlaylist->CurrentItemIndex());
|
|
fPlaylistWindow->PostMessage(&removeMessage);
|
|
return true;
|
|
}
|
|
break;
|
|
}
|
|
|
|
switch (key) {
|
|
case 0x3a: // numeric keypad +
|
|
if ((modifier & B_COMMAND_KEY) == 0) {
|
|
_ZoomVideoView(10);
|
|
return true;
|
|
}
|
|
break;
|
|
|
|
case 0x25: // numeric keypad -
|
|
if ((modifier & B_COMMAND_KEY) == 0) {
|
|
_ZoomVideoView(-10);
|
|
return true;
|
|
}
|
|
break;
|
|
|
|
case 0x38: // numeric keypad up arrow
|
|
PostMessage(M_VOLUME_UP);
|
|
return true;
|
|
|
|
case 0x59: // numeric keypad down arrow
|
|
PostMessage(M_VOLUME_DOWN);
|
|
return true;
|
|
|
|
case 0x39: // numeric keypad page up
|
|
case 0x4a: // numeric keypad right arrow
|
|
PostMessage(M_SKIP_NEXT);
|
|
return true;
|
|
|
|
case 0x5a: // numeric keypad page down
|
|
case 0x48: // numeric keypad left arrow
|
|
PostMessage(M_SKIP_PREV);
|
|
return true;
|
|
|
|
// Playback controls along the bottom of the keyboard:
|
|
// Z X C (V) B for US International
|
|
case 0x4c:
|
|
PostMessage(M_SKIP_PREV);
|
|
return true;
|
|
case 0x4d:
|
|
fController->TogglePlaying();
|
|
return true;
|
|
case 0x4e:
|
|
fController->Pause();
|
|
return true;
|
|
case 0x4f:
|
|
fController->Stop();
|
|
return true;
|
|
case 0x50:
|
|
PostMessage(M_SKIP_NEXT);
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
// #pragma mark -
|
|
|
|
|
|
void
|
|
MainWin::_ToggleFullscreen()
|
|
{
|
|
printf("_ToggleFullscreen enter\n");
|
|
|
|
if (!fHasVideo) {
|
|
printf("_ToggleFullscreen - ignoring, as we don't have a video\n");
|
|
return;
|
|
}
|
|
|
|
fIsFullscreen = !fIsFullscreen;
|
|
|
|
if (fIsFullscreen) {
|
|
// switch to fullscreen
|
|
|
|
fSavedFrame = Frame();
|
|
printf("saving current frame: %d %d %d %d\n", int(fSavedFrame.left),
|
|
int(fSavedFrame.top), int(fSavedFrame.right),
|
|
int(fSavedFrame.bottom));
|
|
BScreen screen(this);
|
|
BRect rect(screen.Frame());
|
|
|
|
Hide();
|
|
MoveTo(rect.left, rect.top);
|
|
ResizeTo(rect.Width(), rect.Height());
|
|
Show();
|
|
|
|
} else {
|
|
// switch back from full screen mode
|
|
_ShowFullscreenControls(false, false);
|
|
|
|
Hide();
|
|
MoveTo(fSavedFrame.left, fSavedFrame.top);
|
|
ResizeTo(fSavedFrame.Width(), fSavedFrame.Height());
|
|
Show();
|
|
}
|
|
|
|
fVideoView->SetFullscreen(fIsFullscreen);
|
|
|
|
_MarkItem(fFileMenu, M_TOGGLE_FULLSCREEN, fIsFullscreen);
|
|
|
|
printf("_ToggleFullscreen leave\n");
|
|
}
|
|
|
|
void
|
|
MainWin::_ToggleAlwaysOnTop()
|
|
{
|
|
fAlwaysOnTop = !fAlwaysOnTop;
|
|
SetFeel(fAlwaysOnTop ? B_FLOATING_ALL_WINDOW_FEEL : B_NORMAL_WINDOW_FEEL);
|
|
|
|
_MarkItem(fFileMenu, M_TOGGLE_ALWAYS_ON_TOP, fAlwaysOnTop);
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_ToggleNoInterface()
|
|
{
|
|
printf("_ToggleNoInterface enter\n");
|
|
|
|
if (fIsFullscreen || !fHasVideo) {
|
|
// Fullscreen playback is always without interface and
|
|
// audio playback is always with interface. So we ignore these
|
|
// two states here.
|
|
printf("_ToggleNoControls leave, doing nothing, we are fullscreen\n");
|
|
return;
|
|
}
|
|
|
|
fNoInterface = !fNoInterface;
|
|
_SetWindowSizeLimits();
|
|
|
|
if (fNoInterface) {
|
|
MoveBy(0, fMenuBarHeight);
|
|
ResizeBy(0, -(fControlsHeight + fMenuBarHeight));
|
|
SetLook(B_BORDERED_WINDOW_LOOK);
|
|
} else {
|
|
MoveBy(0, -fMenuBarHeight);
|
|
ResizeBy(0, fControlsHeight + fMenuBarHeight);
|
|
SetLook(B_TITLED_WINDOW_LOOK);
|
|
}
|
|
|
|
_MarkItem(fFileMenu, M_TOGGLE_NO_INTERFACE, fNoInterface);
|
|
|
|
printf("_ToggleNoInterface leave\n");
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_ShowIfNeeded()
|
|
{
|
|
// Only proceed if the window is already running
|
|
if (find_thread(NULL) != Thread())
|
|
return;
|
|
|
|
if (!fHasVideo && fNoVideoFrame.IsValid()) {
|
|
MoveTo(fNoVideoFrame.LeftTop());
|
|
ResizeTo(fNoVideoFrame.Width(), fNoVideoFrame.Height());
|
|
MoveOnScreen(B_MOVE_IF_PARTIALLY_OFFSCREEN);
|
|
} else if (fHasVideo && IsHidden())
|
|
CenterOnScreen();
|
|
|
|
fNoVideoFrame = BRect();
|
|
|
|
if (IsHidden()) {
|
|
Show();
|
|
UpdateIfNeeded();
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_ShowFullscreenControls(bool show, bool animate)
|
|
{
|
|
if (fShowsFullscreenControls == show)
|
|
return;
|
|
|
|
fShowsFullscreenControls = show;
|
|
fVideoView->SetFullscreenControlsVisible(show);
|
|
|
|
if (show) {
|
|
fControls->RemoveSelf();
|
|
fControls->MoveTo(fVideoView->Bounds().left,
|
|
fVideoView->Bounds().bottom + 1);
|
|
fVideoView->AddChild(fControls);
|
|
if (fScaleFullscreenControls)
|
|
fControls->SetSymbolScale(1.5f);
|
|
|
|
while (fControls->IsHidden())
|
|
fControls->Show();
|
|
}
|
|
|
|
if (animate) {
|
|
// Slide the controls into view. We need to do this with
|
|
// messages, otherwise we block the video playback for the
|
|
// time of the animation.
|
|
const float kAnimationOffsets[] = { 0.05, 0.2, 0.5, 0.2, 0.05 };
|
|
const int32 steps = sizeof(kAnimationOffsets) / sizeof(float);
|
|
float height = fControls->Bounds().Height();
|
|
float moveDist = show ? -height : height;
|
|
float originalY = fControls->Frame().top;
|
|
for (int32 i = 0; i < steps; i++) {
|
|
BMessage message(M_SLIDE_CONTROLS);
|
|
message.AddFloat("offset",
|
|
floorf(moveDist * kAnimationOffsets[i]));
|
|
PostMessage(&message, this);
|
|
}
|
|
BMessage finalMessage(M_FINISH_SLIDING_CONTROLS);
|
|
finalMessage.AddFloat("offset", originalY + moveDist);
|
|
finalMessage.AddBool("show", show);
|
|
PostMessage(&finalMessage, this);
|
|
} else if (!show) {
|
|
fControls->RemoveSelf();
|
|
fControls->MoveTo(fVideoView->Frame().left,
|
|
fVideoView->Frame().bottom + 1);
|
|
fBackground->AddChild(fControls);
|
|
fControls->SetSymbolScale(1.0f);
|
|
|
|
while (!fControls->IsHidden())
|
|
fControls->Hide();
|
|
}
|
|
}
|
|
|
|
|
|
// #pragma mark -
|
|
|
|
|
|
void
|
|
MainWin::_Wind(bigtime_t howMuch, int64 frames)
|
|
{
|
|
if (!fAllowWinding || !fController->Lock())
|
|
return;
|
|
|
|
if (frames != 0 && fHasVideo && !fController->IsPlaying()) {
|
|
int64 newFrame = fController->CurrentFrame() + frames;
|
|
fController->SetFramePosition(newFrame);
|
|
} else {
|
|
bigtime_t seekTime = fController->TimePosition() + howMuch;
|
|
if (seekTime < 0) {
|
|
fInitialSeekPosition = seekTime;
|
|
PostMessage(M_SKIP_PREV);
|
|
} else if (seekTime > fController->TimeDuration()) {
|
|
fInitialSeekPosition = 0;
|
|
PostMessage(M_SKIP_NEXT);
|
|
} else
|
|
fController->SetTimePosition(seekTime);
|
|
}
|
|
|
|
fController->Unlock();
|
|
fAllowWinding = false;
|
|
}
|
|
|
|
|
|
// #pragma mark -
|
|
|
|
|
|
void
|
|
MainWin::_UpdatePlaylistItemFile()
|
|
{
|
|
BAutolock locker(fPlaylist);
|
|
const FilePlaylistItem* item
|
|
= dynamic_cast<const FilePlaylistItem*>(fController->Item());
|
|
if (item == NULL)
|
|
return;
|
|
|
|
if (!fHasVideo && !fHasAudio)
|
|
return;
|
|
|
|
BNode node(&item->Ref());
|
|
if (node.InitCheck())
|
|
return;
|
|
|
|
locker.Unlock();
|
|
|
|
// Set some standard attributes of the currently played file.
|
|
// This should only be a temporary solution.
|
|
|
|
// Write duration
|
|
const char* kDurationAttrName = "Media:Length";
|
|
attr_info info;
|
|
status_t status = node.GetAttrInfo(kDurationAttrName, &info);
|
|
if (status != B_OK || info.size == 0) {
|
|
bigtime_t duration = fController->TimeDuration();
|
|
// TODO: Tracker does not seem to care about endian for scalar types
|
|
node.WriteAttr(kDurationAttrName, B_INT64_TYPE, 0, &duration,
|
|
sizeof(int64));
|
|
}
|
|
|
|
// Write audio bitrate
|
|
if (fHasAudio) {
|
|
status = node.GetAttrInfo("Audio:Bitrate", &info);
|
|
if (status != B_OK || info.size == 0) {
|
|
media_format format;
|
|
if (fController->GetEncodedAudioFormat(&format) == B_OK
|
|
&& format.type == B_MEDIA_ENCODED_AUDIO) {
|
|
int32 bitrate = (int32)(format.u.encoded_audio.bit_rate
|
|
/ 1000);
|
|
char text[256];
|
|
snprintf(text, sizeof(text), "%" B_PRId32 " kbit", bitrate);
|
|
node.WriteAttr("Audio:Bitrate", B_STRING_TYPE, 0, text,
|
|
strlen(text) + 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Write video bitrate
|
|
if (fHasVideo) {
|
|
status = node.GetAttrInfo("Video:Bitrate", &info);
|
|
if (status != B_OK || info.size == 0) {
|
|
media_format format;
|
|
if (fController->GetEncodedVideoFormat(&format) == B_OK
|
|
&& format.type == B_MEDIA_ENCODED_VIDEO) {
|
|
int32 bitrate = (int32)(format.u.encoded_video.avg_bit_rate
|
|
/ 1000);
|
|
char text[256];
|
|
snprintf(text, sizeof(text), "%" B_PRId32 " kbit", bitrate);
|
|
node.WriteAttr("Video:Bitrate", B_STRING_TYPE, 0, text,
|
|
strlen(text) + 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
_UpdateAttributesMenu(node);
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_UpdateAttributesMenu(const BNode& node)
|
|
{
|
|
int32 rating = -1;
|
|
|
|
attr_info info;
|
|
status_t status = node.GetAttrInfo(kRatingAttrName, &info);
|
|
if (status == B_OK && info.type == B_INT32_TYPE) {
|
|
// Node has the Rating attribute.
|
|
node.ReadAttr(kRatingAttrName, B_INT32_TYPE, 0, &rating,
|
|
sizeof(rating));
|
|
}
|
|
|
|
for (int32 i = 0; BMenuItem* item = fRatingMenu->ItemAt(i); i++)
|
|
item->SetMarked(i + 1 == rating);
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_SetRating(int32 rating)
|
|
{
|
|
BAutolock locker(fPlaylist);
|
|
const FilePlaylistItem* item
|
|
= dynamic_cast<const FilePlaylistItem*>(fController->Item());
|
|
if (item == NULL)
|
|
return;
|
|
|
|
BNode node(&item->Ref());
|
|
if (node.InitCheck())
|
|
return;
|
|
|
|
locker.Unlock();
|
|
|
|
node.WriteAttr(kRatingAttrName, B_INT32_TYPE, 0, &rating, sizeof(rating));
|
|
|
|
// TODO: The whole mechnism should work like this:
|
|
// * There is already an attribute API for PlaylistItem, flesh it out!
|
|
// * FilePlaylistItem node-monitors it's file somehow.
|
|
// * FilePlaylistItem keeps attributes in sync and sends notications.
|
|
// * MainWin updates the menu according to FilePlaylistItem notifications.
|
|
// * PlaylistWin shows columns with attribute and other info.
|
|
// * PlaylistWin updates also upon FilePlaylistItem notifications.
|
|
// * This keeps attributes in sync when another app changes them.
|
|
|
|
_UpdateAttributesMenu(node);
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_UpdateControlsEnabledStatus()
|
|
{
|
|
uint32 enabledButtons = 0;
|
|
if (fHasVideo || fHasAudio) {
|
|
enabledButtons |= PLAYBACK_ENABLED | SEEK_ENABLED
|
|
| SEEK_BACK_ENABLED | SEEK_FORWARD_ENABLED;
|
|
}
|
|
if (fHasAudio)
|
|
enabledButtons |= VOLUME_ENABLED;
|
|
|
|
BAutolock _(fPlaylist);
|
|
bool canSkipPrevious, canSkipNext;
|
|
fPlaylist->GetSkipInfo(&canSkipPrevious, &canSkipNext);
|
|
if (canSkipPrevious)
|
|
enabledButtons |= SKIP_BACK_ENABLED;
|
|
if (canSkipNext)
|
|
enabledButtons |= SKIP_FORWARD_ENABLED;
|
|
|
|
fControls->SetEnabled(enabledButtons);
|
|
|
|
fNoInterfaceMenuItem->SetEnabled(fHasVideo);
|
|
fAttributesMenu->SetEnabled(fHasAudio || fHasVideo);
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_UpdatePlaylistMenu()
|
|
{
|
|
if (!fPlaylist->Lock())
|
|
return;
|
|
|
|
fPlaylistMenu->RemoveItems(0, fPlaylistMenu->CountItems(), true);
|
|
|
|
int32 count = fPlaylist->CountItems();
|
|
for (int32 i = 0; i < count; i++) {
|
|
PlaylistItem* item = fPlaylist->ItemAtFast(i);
|
|
_AddPlaylistItem(item, i);
|
|
}
|
|
fPlaylistMenu->SetTargetForItems(this);
|
|
|
|
_MarkPlaylistItem(fPlaylist->CurrentItemIndex());
|
|
|
|
fPlaylist->Unlock();
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_AddPlaylistItem(PlaylistItem* item, int32 index)
|
|
{
|
|
BMessage* message = new BMessage(M_SET_PLAYLIST_POSITION);
|
|
message->AddInt32("index", index);
|
|
BMenuItem* menuItem = new BMenuItem(item->Name().String(), message);
|
|
fPlaylistMenu->AddItem(menuItem, index);
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_RemovePlaylistItem(int32 index)
|
|
{
|
|
delete fPlaylistMenu->RemoveItem(index);
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_MarkPlaylistItem(int32 index)
|
|
{
|
|
if (BMenuItem* item = fPlaylistMenu->ItemAt(index)) {
|
|
item->SetMarked(true);
|
|
// ... and in case the menu is currently on screen:
|
|
if (fPlaylistMenu->LockLooper()) {
|
|
fPlaylistMenu->Invalidate();
|
|
fPlaylistMenu->UnlockLooper();
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_MarkItem(BMenu* menu, uint32 command, bool mark)
|
|
{
|
|
if (BMenuItem* item = menu->FindItem(command))
|
|
item->SetMarked(mark);
|
|
}
|
|
|
|
|
|
void
|
|
MainWin::_AdoptGlobalSettings()
|
|
{
|
|
mpSettings settings;
|
|
Settings::Default()->Get(settings);
|
|
|
|
fCloseWhenDonePlayingMovie = settings.closeWhenDonePlayingMovie;
|
|
fCloseWhenDonePlayingSound = settings.closeWhenDonePlayingSound;
|
|
fLoopMovies = settings.loopMovie;
|
|
fLoopSounds = settings.loopSound;
|
|
fScaleFullscreenControls = settings.scaleFullscreenControls;
|
|
}
|