mirror of
https://github.com/RetroShare/RetroShare.git
synced 2025-01-07 05:38:09 -05:00
87344de7d4
git-svn-id: http://svn.code.sf.net/p/retroshare/code/trunk@1850 b45a01b8-16f6-495d-af2f-9b41ad6348cc
3742 lines
90 KiB
C++
3742 lines
90 KiB
C++
/* smplayer, GUI front-end for mplayer.
|
|
Copyright (C) 2006-2008 Ricardo Villalba <rvm@escomposlinux.org>
|
|
|
|
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
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include "core.h"
|
|
#include <QDir>
|
|
#include <QFileInfo>
|
|
#include <QRegExp>
|
|
#include <QTextStream>
|
|
|
|
#include <cmath>
|
|
|
|
#include "mplayerwindow.h"
|
|
#include "desktopinfo.h"
|
|
#include "helper.h"
|
|
#include "paths.h"
|
|
#include "preferences.h"
|
|
#include "global.h"
|
|
#include "config.h"
|
|
#include "mplayerversion.h"
|
|
#include "constants.h"
|
|
#include "colorutils.h"
|
|
|
|
#ifdef Q_OS_WIN
|
|
#include <windows.h> // To change app priority
|
|
#include <QSysInfo> // To get Windows version
|
|
#include "screensaver.h"
|
|
#endif
|
|
|
|
#ifndef NO_USE_INI_FILES
|
|
#include "filesettings.h"
|
|
#include "filesettingshash.h"
|
|
#endif
|
|
|
|
using namespace Global;
|
|
|
|
Core::Core( MplayerWindow *mpw, QWidget* parent )
|
|
: QObject( parent )
|
|
{
|
|
qRegisterMetaType<Core::State>("Core::State");
|
|
|
|
mplayerwindow = mpw;
|
|
|
|
_state = Stopped;
|
|
|
|
we_are_restarting = false;
|
|
just_loaded_external_subs = false;
|
|
just_unloaded_external_subs = false;
|
|
change_volume_after_unpause = false;
|
|
|
|
#ifndef NO_USE_INI_FILES
|
|
// Create file_settings
|
|
#if NEW_SETTINGS_MANAGEMENT
|
|
file_settings = 0;
|
|
changeFileSettingsMethod(pref->file_settings_method);
|
|
#else
|
|
if (Paths::iniPath().isEmpty()) {
|
|
file_settings = new QSettings(QSettings::IniFormat, QSettings::UserScope,
|
|
QString(COMPANY), QString("smplayer_files") );
|
|
} else {
|
|
QString filename = Paths::iniPath() + "/smplayer_files.ini";
|
|
file_settings = new QSettings( filename, QSettings::IniFormat );
|
|
qDebug("Core::Core: file_settings: '%s'", filename.toUtf8().data());
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
proc = new MplayerProcess(this);
|
|
|
|
// Do this the first
|
|
connect( proc, SIGNAL(processExited()),
|
|
mplayerwindow->videoLayer(), SLOT(playingStopped()) );
|
|
|
|
connect( proc, SIGNAL(error(QProcess::ProcessError)),
|
|
mplayerwindow->videoLayer(), SLOT(playingStopped()) );
|
|
|
|
connect( proc, SIGNAL(receivedCurrentSec(double)),
|
|
this, SLOT(changeCurrentSec(double)) );
|
|
|
|
connect( proc, SIGNAL(receivedCurrentFrame(int)),
|
|
this, SIGNAL(showFrame(int)) );
|
|
|
|
connect( proc, SIGNAL(receivedPause()),
|
|
this, SLOT(changePause()) );
|
|
|
|
connect( proc, SIGNAL(processExited()),
|
|
this, SLOT(processFinished()), Qt::QueuedConnection );
|
|
|
|
connect( proc, SIGNAL(mplayerFullyLoaded()),
|
|
this, SLOT(finishRestart()), Qt::QueuedConnection );
|
|
|
|
connect( proc, SIGNAL(lineAvailable(QString)),
|
|
this, SIGNAL(logLineAvailable(QString)) );
|
|
|
|
connect( proc, SIGNAL(receivedCacheMessage(QString)),
|
|
this, SLOT(displayMessage(QString)) );
|
|
|
|
connect( proc, SIGNAL(receivedCreatingIndex(QString)),
|
|
this, SLOT(displayMessage(QString)) );
|
|
|
|
connect( proc, SIGNAL(receivedConnectingToMessage(QString)),
|
|
this, SLOT(displayMessage(QString)) );
|
|
|
|
connect( proc, SIGNAL(receivedResolvingMessage(QString)),
|
|
this, SLOT(displayMessage(QString)) );
|
|
|
|
connect( proc, SIGNAL(receivedScreenshot(QString)),
|
|
this, SLOT(displayScreenshotName(QString)) );
|
|
|
|
connect( proc, SIGNAL(receivedUpdatingFontCache()),
|
|
this, SLOT(displayUpdatingFontCache()) );
|
|
|
|
connect( proc, SIGNAL(receivedWindowResolution(int,int)),
|
|
this, SLOT(gotWindowResolution(int,int)) );
|
|
|
|
connect( proc, SIGNAL(receivedNoVideo()),
|
|
this, SLOT(gotNoVideo()) );
|
|
|
|
connect( proc, SIGNAL(receivedVO(QString)),
|
|
this, SLOT(gotVO(QString)) );
|
|
|
|
connect( proc, SIGNAL(receivedAO(QString)),
|
|
this, SLOT(gotAO(QString)) );
|
|
|
|
connect( proc, SIGNAL(receivedEndOfFile()),
|
|
this, SLOT(fileReachedEnd()), Qt::QueuedConnection );
|
|
|
|
connect( proc, SIGNAL(receivedStartingTime(double)),
|
|
this, SLOT(gotStartingTime(double)) );
|
|
|
|
connect( proc, SIGNAL(receivedStreamTitleAndUrl(QString,QString)),
|
|
this, SLOT(streamTitleAndUrlChanged(QString,QString)) );
|
|
|
|
connect( proc, SIGNAL(failedToParseMplayerVersion(QString)),
|
|
this, SIGNAL(failedToParseMplayerVersion(QString)) );
|
|
|
|
connect( this, SIGNAL(mediaLoaded()), this, SLOT(checkIfVideoIsHD()), Qt::QueuedConnection );
|
|
#if DELAYED_AUDIO_SETUP_ON_STARTUP
|
|
connect( this, SIGNAL(mediaLoaded()), this, SLOT(initAudioTrack()), Qt::QueuedConnection );
|
|
#endif
|
|
#if NOTIFY_SUB_CHANGES
|
|
connect( proc, SIGNAL(subtitleInfoChanged(const SubTracks &)),
|
|
this, SLOT(initSubtitleTrack(const SubTracks &)), Qt::QueuedConnection );
|
|
connect( proc, SIGNAL(subtitleInfoReceivedAgain(const SubTracks &)),
|
|
this, SLOT(setSubtitleTrackAgain(const SubTracks &)), Qt::QueuedConnection );
|
|
#endif
|
|
#if NOTIFY_AUDIO_CHANGES
|
|
connect( proc, SIGNAL(audioInfoChanged(const Tracks &)),
|
|
this, SLOT(initAudioTrack(const Tracks &)), Qt::QueuedConnection );
|
|
#endif
|
|
|
|
connect( this, SIGNAL(stateChanged(Core::State)),
|
|
this, SLOT(watchState(Core::State)) );
|
|
|
|
connect( proc, SIGNAL(error(QProcess::ProcessError)),
|
|
this, SIGNAL(mplayerFailed(QProcess::ProcessError)) );
|
|
|
|
//pref->load();
|
|
mset.reset();
|
|
|
|
// Mplayerwindow
|
|
connect( this, SIGNAL(aboutToStartPlaying()),
|
|
mplayerwindow->videoLayer(), SLOT(playingStarted()) );
|
|
|
|
#if REPAINT_BACKGROUND_OPTION
|
|
mplayerwindow->videoLayer()->setRepaintBackground(pref->repaint_video_background);
|
|
#endif
|
|
mplayerwindow->setMonitorAspect( pref->monitor_aspect_double() );
|
|
|
|
#ifdef Q_OS_WIN
|
|
// Windows screensaver
|
|
win_screensaver = new WinScreenSaver();
|
|
#endif
|
|
}
|
|
|
|
|
|
Core::~Core() {
|
|
#ifndef NO_USE_INI_FILES
|
|
saveMediaInfo();
|
|
#endif
|
|
|
|
if (proc->isRunning()) stopMplayer();
|
|
proc->terminate();
|
|
delete proc;
|
|
|
|
#ifndef NO_USE_INI_FILES
|
|
delete file_settings;
|
|
#endif
|
|
|
|
#ifdef Q_OS_WIN
|
|
delete win_screensaver;
|
|
#endif
|
|
}
|
|
|
|
#ifndef NO_USE_INI_FILES
|
|
void Core::changeFileSettingsMethod(QString method) {
|
|
#if NEW_SETTINGS_MANAGEMENT
|
|
qDebug("Core::changeFileSettingsMethod: %s", method.toUtf8().constData());
|
|
if (file_settings) delete file_settings;
|
|
|
|
if (method.toLower() == "hash")
|
|
file_settings = new FileSettingsHash(Paths::iniPath());
|
|
else
|
|
file_settings = new FileSettings(Paths::iniPath());
|
|
#endif
|
|
}
|
|
#endif
|
|
|
|
void Core::setState(State s) {
|
|
if (s != _state) {
|
|
_state = s;
|
|
emit stateChanged(_state);
|
|
}
|
|
}
|
|
|
|
QString Core::stateToString() {
|
|
if (state()==Playing) return "Playing";
|
|
else
|
|
if (state()==Stopped) return "Stopped";
|
|
else
|
|
if (state()==Paused) return "Paused";
|
|
else
|
|
return "Unknown";
|
|
}
|
|
|
|
// Public restart
|
|
void Core::restart() {
|
|
qDebug("Core::restart");
|
|
if (proc->isRunning()) {
|
|
restartPlay();
|
|
} else {
|
|
qDebug("Core::restart: mplayer is not running");
|
|
}
|
|
}
|
|
|
|
void Core::reload() {
|
|
qDebug("Core::reload");
|
|
|
|
stopMplayer();
|
|
we_are_restarting = false;
|
|
|
|
initPlaying();
|
|
}
|
|
|
|
#ifndef NO_USE_INI_FILES
|
|
#if !NEW_SETTINGS_MANAGEMENT
|
|
|
|
bool Core::checkHaveSettingsSaved(QString group_name) {
|
|
qDebug("Core::checkHaveSettingsSaved: group_name: '%s'", group_name.toUtf8().data());
|
|
|
|
file_settings->beginGroup( group_name );
|
|
bool saved = file_settings->value( "saved", false ).toBool();
|
|
file_settings->endGroup();
|
|
|
|
return saved;
|
|
}
|
|
|
|
void Core::loadMediaInfo(QString group_name) {
|
|
qDebug("Core::loadMediaInfo: '%s'", group_name.toUtf8().data() );
|
|
|
|
file_settings->beginGroup( group_name );
|
|
|
|
/*mdat.load(*settings);*/
|
|
mset.load(file_settings);
|
|
|
|
file_settings->endGroup();
|
|
}
|
|
|
|
#endif // NEW_SETTINGS_MANAGEMENT
|
|
|
|
void Core::saveMediaInfo() {
|
|
qDebug("Core::saveMediaInfo");
|
|
|
|
if (pref->dont_remember_media_settings) {
|
|
qDebug("Core::saveMediaInfo: not saving settings, disabled by user");
|
|
return;
|
|
}
|
|
|
|
#if NEW_SETTINGS_MANAGEMENT
|
|
if ( (mdat.type == TYPE_FILE) && (!mdat.filename.isEmpty()) ) {
|
|
file_settings->saveSettingsFor(mdat.filename, mset);
|
|
}
|
|
#else
|
|
QString group_name;
|
|
|
|
/*
|
|
if ( (mdat.type == TYPE_DVD) && (!mdat.dvd_id.isEmpty()) ) {
|
|
group_name = dvdForPref( mdat.dvd_id, mset.current_title_id );
|
|
}
|
|
else
|
|
*/
|
|
if ( (mdat.type == TYPE_FILE) && (!mdat.filename.isEmpty()) ) {
|
|
group_name = FileSettings::filenameToGroupname( mdat.filename );
|
|
}
|
|
|
|
if (!group_name.isEmpty()) {
|
|
file_settings->beginGroup( group_name );
|
|
file_settings->setValue( "saved", true);
|
|
|
|
/*mdat.save(*settings);*/
|
|
mset.save(file_settings);
|
|
|
|
file_settings->endGroup();
|
|
}
|
|
#endif // NEW_SETTINGS_MANAGEMENT
|
|
}
|
|
|
|
#endif // NO_USE_INI_FILES
|
|
|
|
void Core::initializeMenus() {
|
|
qDebug("Core::initializeMenus");
|
|
|
|
emit menusNeedInitialize();
|
|
}
|
|
|
|
|
|
void Core::updateWidgets() {
|
|
qDebug("Core::updateWidgets");
|
|
|
|
emit widgetsNeedUpdate();
|
|
}
|
|
|
|
|
|
void Core::tellmp(const QString & command) {
|
|
qDebug("Core::tellmp: '%s'", command.toUtf8().data());
|
|
|
|
//qDebug("Command: '%s'", command.toUtf8().data());
|
|
if (proc->isRunning()) {
|
|
proc->writeToStdin( command );
|
|
} else {
|
|
qWarning(" tellmp: no process running: %s", command.toUtf8().data());
|
|
}
|
|
}
|
|
|
|
// Generic open, autodetect type
|
|
void Core::open(QString file, int seek) {
|
|
qDebug("Core::open: '%s'", file.toUtf8().data());
|
|
|
|
QFileInfo fi(file);
|
|
|
|
if ( (fi.exists()) && (fi.suffix().toLower()=="iso") ) {
|
|
qDebug("Core::open: * identified as a dvd iso");
|
|
openDVD("dvd://1:" + file);
|
|
}
|
|
else
|
|
if ( (fi.exists()) && (!fi.isDir()) ) {
|
|
qDebug("Core::open: * identified as local file");
|
|
// Local file
|
|
file = QFileInfo(file).absoluteFilePath();
|
|
openFile(file, seek);
|
|
}
|
|
else
|
|
if ( (fi.exists()) && (fi.isDir()) ) {
|
|
// Directory
|
|
qDebug("Core::open: * identified as a directory");
|
|
qDebug("Core::open: checking if contains a dvd");
|
|
file = QFileInfo(file).absoluteFilePath();
|
|
if (Helper::directoryContainsDVD(file)) {
|
|
qDebug("Core::open: * directory contains a dvd");
|
|
openDVD("dvd://1:"+ file);
|
|
} else {
|
|
qDebug("Core::open: * directory doesn't contain a dvd");
|
|
qDebug("Core::open: opening nothing");
|
|
}
|
|
}
|
|
else
|
|
if (file.toLower().startsWith("dvd:")) {
|
|
qDebug("Core::open: * identified as dvd");
|
|
openDVD(file);
|
|
/*
|
|
QString f = file.lower();
|
|
QRegExp s("^dvd://(\\d+)");
|
|
if (s.indexIn(f) != -1) {
|
|
int title = s.cap(1).toInt();
|
|
openDVD(title);
|
|
} else {
|
|
qWarning("Core::open: couldn't parse dvd title, playing first one");
|
|
openDVD();
|
|
}
|
|
*/
|
|
}
|
|
else
|
|
if (file.toLower().startsWith("vcd:")) {
|
|
qDebug("Core::open: * identified as vcd");
|
|
|
|
QString f = file.toLower();
|
|
QRegExp s("^vcd://(\\d+)");
|
|
if (s.indexIn(f) != -1) {
|
|
int title = s.cap(1).toInt();
|
|
openVCD(title);
|
|
} else {
|
|
qWarning("Core::open: couldn't parse vcd title, playing first one");
|
|
openVCD();
|
|
}
|
|
}
|
|
else
|
|
if (file.toLower().startsWith("cdda:")) {
|
|
qDebug("Core::open: * identified as cdda");
|
|
|
|
QString f = file.toLower();
|
|
QRegExp s("^cdda://(\\d+)");
|
|
if (s.indexIn(f) != -1) {
|
|
int title = s.cap(1).toInt();
|
|
openAudioCD(title);
|
|
} else {
|
|
qWarning("Core::open: couldn't parse cdda title, playing first one");
|
|
openAudioCD();
|
|
}
|
|
}
|
|
else {
|
|
qDebug("Core::open: * not identified, playing as stream");
|
|
openStream(file);
|
|
}
|
|
}
|
|
|
|
void Core::openFile(QString filename, int seek) {
|
|
qDebug("Core::openFile: '%s'", filename.toUtf8().data());
|
|
|
|
QFileInfo fi(filename);
|
|
if (fi.exists()) {
|
|
playNewFile(fi.absoluteFilePath(), seek);
|
|
} else {
|
|
//File doesn't exists
|
|
//TODO: error message
|
|
}
|
|
}
|
|
|
|
|
|
void Core::loadSub(const QString & sub ) {
|
|
if ( !sub.isEmpty() ) {
|
|
//tellmp( "sub_load " + sub );
|
|
mset.external_subtitles = sub;
|
|
just_loaded_external_subs = true;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::unloadSub() {
|
|
if ( !mset.external_subtitles.isEmpty() ) {
|
|
mset.external_subtitles = "";
|
|
just_unloaded_external_subs = true;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::loadAudioFile(const QString & audiofile) {
|
|
if (!audiofile.isEmpty()) {
|
|
mset.external_audio = audiofile;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::unloadAudioFile() {
|
|
if (!mset.external_audio.isEmpty()) {
|
|
mset.external_audio = "";
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
/*
|
|
void Core::openDVD( bool from_folder, QString directory) {
|
|
qDebug("Core::openDVD");
|
|
|
|
if (from_folder) {
|
|
if (!directory.isEmpty()) {
|
|
QFileInfo fi(directory);
|
|
if ( (fi.exists()) && (fi.isDir()) ) {
|
|
pref->dvd_directory = directory;
|
|
pref->play_dvd_from_hd = TRUE;
|
|
openDVD();
|
|
} else {
|
|
qDebug("Core::openDVD: directory '%s' is not valid", directory.toUtf8().data());
|
|
}
|
|
} else {
|
|
qDebug("Core::openDVD: directory is empty");
|
|
}
|
|
} else {
|
|
pref->play_dvd_from_hd = FALSE;
|
|
openDVD();
|
|
}
|
|
}
|
|
|
|
void Core::openDVD() {
|
|
openDVD(1);
|
|
}
|
|
|
|
void Core::openDVD(int title) {
|
|
qDebug("Core::openDVD: %d", title);
|
|
|
|
if (proc->isRunning()) {
|
|
stopMplayer();
|
|
}
|
|
|
|
// Save data of previous file:
|
|
saveMediaInfo();
|
|
|
|
mdat.reset();
|
|
mdat.filename = "dvd://" + QString::number(title);
|
|
mdat.type = TYPE_DVD;
|
|
|
|
mset.reset();
|
|
|
|
mset.current_title_id = title;
|
|
mset.current_chapter_id = 1;
|
|
mset.current_angle_id = 1;
|
|
|
|
initializeMenus();
|
|
|
|
initPlaying();
|
|
}
|
|
*/
|
|
|
|
void Core::openVCD(int title) {
|
|
qDebug("Core::openVCD: %d", title);
|
|
|
|
if (title == -1) title = pref->vcd_initial_title;
|
|
|
|
if (proc->isRunning()) {
|
|
stopMplayer();
|
|
}
|
|
|
|
// Save data of previous file:
|
|
#ifndef NO_USE_INI_FILES
|
|
saveMediaInfo();
|
|
#endif
|
|
|
|
mdat.reset();
|
|
mdat.filename = "vcd://" + QString::number(title);
|
|
mdat.type = TYPE_VCD;
|
|
|
|
mset.reset();
|
|
|
|
mset.current_title_id = title;
|
|
mset.current_chapter_id = -1;
|
|
mset.current_angle_id = -1;
|
|
|
|
/* initializeMenus(); */
|
|
|
|
initPlaying();
|
|
}
|
|
|
|
void Core::openAudioCD(int title) {
|
|
qDebug("Core::openAudioCD: %d", title);
|
|
|
|
if (title == -1) title = 1;
|
|
|
|
if (proc->isRunning()) {
|
|
stopMplayer();
|
|
}
|
|
|
|
// Save data of previous file:
|
|
#ifndef NO_USE_INI_FILES
|
|
saveMediaInfo();
|
|
#endif
|
|
|
|
mdat.reset();
|
|
mdat.filename = "cdda://" + QString::number(title);
|
|
mdat.type = TYPE_AUDIO_CD;
|
|
|
|
mset.reset();
|
|
|
|
mset.current_title_id = title;
|
|
mset.current_chapter_id = -1;
|
|
mset.current_angle_id = -1;
|
|
|
|
/* initializeMenus(); */
|
|
|
|
initPlaying();
|
|
}
|
|
|
|
void Core::openDVD(QString dvd_url) {
|
|
qDebug("Core::openDVD: '%s'", dvd_url.toUtf8().data());
|
|
|
|
//Checks
|
|
QString folder = Helper::dvdSplitFolder(dvd_url);
|
|
int title = Helper::dvdSplitTitle(dvd_url);
|
|
|
|
if (title == -1) {
|
|
qWarning("Core::openDVD: title invalid, not playing dvd");
|
|
return;
|
|
}
|
|
|
|
if (folder.isEmpty()) {
|
|
qDebug("Core::openDVD: not folder");
|
|
} else {
|
|
QFileInfo fi(folder);
|
|
if ( (!fi.exists()) /*|| (!fi.isDir())*/ ) {
|
|
qWarning("Core::openDVD: folder invalid, not playing dvd");
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (proc->isRunning()) {
|
|
stopMplayer();
|
|
we_are_restarting = false;
|
|
}
|
|
|
|
// Save data of previous file:
|
|
#ifndef NO_USE_INI_FILES
|
|
saveMediaInfo();
|
|
#endif
|
|
|
|
mdat.reset();
|
|
mdat.filename = dvd_url;
|
|
mdat.type = TYPE_DVD;
|
|
|
|
mset.reset();
|
|
|
|
mset.current_title_id = title;
|
|
#if GENERIC_CHAPTER_SUPPORT
|
|
mset.current_chapter_id = firstChapter();
|
|
#else
|
|
mset.current_chapter_id = dvdFirstChapter();
|
|
#endif
|
|
mset.current_angle_id = 1;
|
|
|
|
/* initializeMenus(); */
|
|
|
|
initPlaying();
|
|
}
|
|
|
|
void Core::openStream(QString name) {
|
|
qDebug("Core::openStream: '%s'", name.toUtf8().data());
|
|
|
|
if (proc->isRunning()) {
|
|
stopMplayer();
|
|
we_are_restarting = false;
|
|
}
|
|
|
|
// Save data of previous file:
|
|
#ifndef NO_USE_INI_FILES
|
|
saveMediaInfo();
|
|
#endif
|
|
|
|
mdat.reset();
|
|
mdat.filename = name;
|
|
mdat.type = TYPE_STREAM;
|
|
|
|
mset.reset();
|
|
|
|
/* initializeMenus(); */
|
|
|
|
initPlaying();
|
|
}
|
|
|
|
|
|
void Core::playNewFile(QString file, int seek) {
|
|
qDebug("Core::playNewFile: '%s'", file.toUtf8().data());
|
|
|
|
if (proc->isRunning()) {
|
|
stopMplayer();
|
|
we_are_restarting = false;
|
|
}
|
|
|
|
// Save data of previous file:
|
|
#ifndef NO_USE_INI_FILES
|
|
saveMediaInfo();
|
|
#endif
|
|
|
|
mdat.reset();
|
|
mdat.filename = file;
|
|
mdat.type = TYPE_FILE;
|
|
|
|
int old_volume = mset.volume;
|
|
mset.reset();
|
|
|
|
#ifndef NO_USE_INI_FILES
|
|
// Check if we already have info about this file
|
|
#if NEW_SETTINGS_MANAGEMENT
|
|
if (file_settings->existSettingsFor(file)) {
|
|
#else
|
|
if (checkHaveSettingsSaved( FileSettings::filenameToGroupname(file) )) {
|
|
#endif
|
|
qDebug("Core::playNewFile: We have settings for this file!!!");
|
|
|
|
// In this case we read info from config
|
|
if (!pref->dont_remember_media_settings) {
|
|
#if NEW_SETTINGS_MANAGEMENT
|
|
file_settings->loadSettingsFor(file, mset);
|
|
#else
|
|
loadMediaInfo( FileSettings::filenameToGroupname(file) );
|
|
#endif
|
|
qDebug("Core::playNewFile: Media settings read");
|
|
|
|
// Resize the window and set the aspect as soon as possible
|
|
int saved_width = mset.win_width;
|
|
int saved_height = mset.win_height;
|
|
// 400x300 is the default size for win_width and win_height
|
|
// so we set them to 0 to avoid to resize the window on
|
|
// audio files
|
|
if ((saved_width == 400) && (saved_height == 300)) {
|
|
saved_width = 0;
|
|
saved_height = 0;
|
|
}
|
|
if ((saved_width > 0) && (saved_height > 0)) {
|
|
emit needResize(mset.win_width, mset.win_height);
|
|
changeAspectRatio(mset.aspect_ratio_id);
|
|
}
|
|
|
|
if (pref->dont_remember_time_pos) {
|
|
mset.current_sec = 0;
|
|
qDebug("Core::playNewFile: Time pos reset to 0");
|
|
}
|
|
} else {
|
|
qDebug("Core::playNewFile: Media settings have not read because of preferences setting");
|
|
}
|
|
} else {
|
|
// Recover volume
|
|
mset.volume = old_volume;
|
|
}
|
|
#else
|
|
// Recover volume
|
|
mset.volume = old_volume;
|
|
#endif // NO_USE_INI_FILES
|
|
|
|
/* initializeMenus(); */
|
|
|
|
qDebug("Core::playNewFile: volume: %d, old_volume: %d", mset.volume, old_volume);
|
|
initPlaying(seek);
|
|
}
|
|
|
|
|
|
void Core::restartPlay() {
|
|
we_are_restarting = true;
|
|
initPlaying();
|
|
}
|
|
|
|
void Core::initPlaying(int seek) {
|
|
qDebug("Core::initPlaying");
|
|
|
|
/*
|
|
mdat.list();
|
|
mset.list();
|
|
*/
|
|
|
|
/* updateWidgets(); */
|
|
|
|
mplayerwindow->showLogo(FALSE);
|
|
|
|
if (proc->isRunning()) {
|
|
stopMplayer();
|
|
}
|
|
|
|
int start_sec = (int) mset.current_sec;
|
|
if (seek > -1) start_sec = seek;
|
|
|
|
startMplayer( mdat.filename, start_sec );
|
|
}
|
|
|
|
// This is reached when a new video has just started playing
|
|
// and maybe we need to give some defaults
|
|
void Core::newMediaPlaying() {
|
|
qDebug("Core::newMediaPlaying: --- start ---");
|
|
|
|
QString file = mdat.filename;
|
|
int type = mdat.type;
|
|
mdat = proc->mediaData();
|
|
mdat.filename = file;
|
|
mdat.type = type;
|
|
|
|
initializeMenus(); // Old
|
|
|
|
// Video
|
|
if ( (mset.current_video_id == MediaSettings::NoneSelected) &&
|
|
(mdat.videos.numItems() > 0) )
|
|
{
|
|
changeVideo( mdat.videos.itemAt(0).ID(), false ); // Don't allow to restart
|
|
}
|
|
|
|
#if !DELAYED_AUDIO_SETUP_ON_STARTUP && !NOTIFY_AUDIO_CHANGES
|
|
// First audio if none selected
|
|
if ( (mset.current_audio_id == MediaSettings::NoneSelected) &&
|
|
(mdat.audios.numItems() > 0) )
|
|
{
|
|
// Don't set mset.current_audio_id here! changeAudio will do.
|
|
// Otherwise changeAudio will do nothing.
|
|
|
|
int audio = mdat.audios.itemAt(0).ID(); // First one
|
|
if (mdat.audios.existsItemAt(pref->initial_audio_track-1)) {
|
|
audio = mdat.audios.itemAt(pref->initial_audio_track-1).ID();
|
|
}
|
|
|
|
// Check if one of the audio tracks is the user preferred.
|
|
if (!pref->audio_lang.isEmpty()) {
|
|
int res = mdat.audios.findLang( pref->audio_lang );
|
|
if (res != -1) audio = res;
|
|
}
|
|
|
|
// Change the audio without restarting mplayer, it's not
|
|
// safe to do it here.
|
|
changeAudio( audio, false );
|
|
|
|
}
|
|
#endif
|
|
|
|
#if !NOTIFY_SUB_CHANGES
|
|
// Subtitles
|
|
if (mset.external_subtitles.isEmpty()) {
|
|
if (pref->autoload_sub) {
|
|
//Select first subtitle if none selected
|
|
if (mset.current_sub_id == MediaSettings::NoneSelected) {
|
|
int sub = mdat.subs.selectOne( pref->subtitle_lang, pref->initial_subtitle_track-1 );
|
|
changeSubtitle( sub );
|
|
}
|
|
} else {
|
|
changeSubtitle( MediaSettings::SubNone );
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#if GENERIC_CHAPTER_SUPPORT
|
|
if (mdat.chapters > 0) {
|
|
#else
|
|
// mkv chapters
|
|
if (mdat.mkv_chapters > 0) {
|
|
#endif
|
|
// Just to show the first chapter checked in the menu
|
|
mset.current_chapter_id = firstChapter();
|
|
}
|
|
|
|
mdat.initialized = TRUE;
|
|
|
|
// MPlayer doesn't display the length in ID_LENGTH for audio CDs...
|
|
if ((mdat.duration == 0) && (mdat.type == TYPE_AUDIO_CD)) {
|
|
/*
|
|
qDebug(" *** get duration here from title info *** ");
|
|
qDebug(" *** current title: %d", mset.current_title_id );
|
|
*/
|
|
if (mset.current_title_id > 0) {
|
|
mdat.duration = mdat.titles.item(mset.current_title_id).duration();
|
|
}
|
|
}
|
|
|
|
/* updateWidgets(); */
|
|
|
|
mdat.list();
|
|
mset.list();
|
|
|
|
qDebug("Core::newMediaPlaying: --- end ---");
|
|
}
|
|
|
|
void Core::finishRestart() {
|
|
qDebug("Core::finishRestart: --- start ---");
|
|
|
|
if (!we_are_restarting) {
|
|
newMediaPlaying();
|
|
//QTimer::singleShot(1000, this, SIGNAL(mediaStartPlay()));
|
|
emit mediaStartPlay();
|
|
}
|
|
|
|
if (we_are_restarting) {
|
|
// Update info about codecs and demuxer
|
|
mdat.video_codec = proc->mediaData().video_codec;
|
|
mdat.audio_codec = proc->mediaData().audio_codec;
|
|
mdat.demuxer = proc->mediaData().demuxer;
|
|
}
|
|
|
|
#if !NOTIFY_SUB_CHANGES
|
|
// Subtitles
|
|
//if (we_are_restarting) {
|
|
if ( (just_loaded_external_subs) || (just_unloaded_external_subs) ) {
|
|
qDebug("Core::finishRestart: processing new subtitles");
|
|
|
|
// Just to simplify things
|
|
if (mset.current_sub_id == MediaSettings::NoneSelected) {
|
|
mset.current_sub_id = MediaSettings::SubNone;
|
|
}
|
|
|
|
// Save current sub
|
|
SubData::Type type;
|
|
int ID;
|
|
int old_item = -1;
|
|
if ( mset.current_sub_id != MediaSettings::SubNone ) {
|
|
old_item = mset.current_sub_id;
|
|
type = mdat.subs.itemAt(old_item).type();
|
|
ID = mdat.subs.itemAt(old_item).ID();
|
|
}
|
|
|
|
// Use the subtitle info from mplayerprocess
|
|
qDebug( "Core::finishRestart: copying sub data from proc to mdat");
|
|
mdat.subs = proc->mediaData().subs;
|
|
initializeMenus();
|
|
int item = MediaSettings::SubNone;
|
|
|
|
// Try to recover old subtitle
|
|
if (just_unloaded_external_subs) {
|
|
if (old_item > -1) {
|
|
int new_item = mdat.subs.find(type, ID);
|
|
if (new_item > -1) item = new_item;
|
|
}
|
|
}
|
|
|
|
// If we've just loaded a subtitle file
|
|
// select one if the user wants to autoload
|
|
// one subtitle
|
|
if (just_loaded_external_subs) {
|
|
if ( (pref->autoload_sub) && (item == MediaSettings::SubNone) ) {
|
|
qDebug("Core::finishRestart: cannot find previous subtitle");
|
|
qDebug("Core::finishRestart: selecting a new one");
|
|
item = mdat.subs.selectOne( pref->subtitle_lang );
|
|
}
|
|
}
|
|
changeSubtitle( item );
|
|
just_loaded_external_subs = false;
|
|
just_unloaded_external_subs = false;
|
|
} else {
|
|
// Normal restart, subtitles haven't changed
|
|
// Recover current subtitle
|
|
changeSubtitle( mset.current_sub_id );
|
|
}
|
|
#endif
|
|
|
|
we_are_restarting = false;
|
|
|
|
#if NEW_ASPECT_CODE
|
|
changeAspectRatio(mset.aspect_ratio_id);
|
|
#else
|
|
if (mset.aspect_ratio_id < MediaSettings::Aspect43Letterbox) {
|
|
changeAspectRatio(mset.aspect_ratio_id);
|
|
}
|
|
#endif
|
|
|
|
bool isMuted = mset.mute;
|
|
if (!pref->dont_change_volume) {
|
|
setVolume( mset.volume, TRUE );
|
|
}
|
|
if (isMuted) mute(TRUE);
|
|
|
|
if (pref->change_video_equalizer_on_startup && (mset.gamma != 0)) {
|
|
int gamma = mset.gamma;
|
|
mset.gamma = -1000; // if mset.gamma == new value, mset.gamma is not changed!
|
|
setGamma( gamma );
|
|
}
|
|
// Hack to be sure that the equalizers are up to date
|
|
emit videoEqualizerNeedsUpdate();
|
|
emit audioEqualizerNeedsUpdate();
|
|
|
|
changePanscan(mset.panscan_factor);
|
|
|
|
emit mediaLoaded();
|
|
emit mediaInfoChanged();
|
|
|
|
updateWidgets(); // New
|
|
|
|
qDebug("Core::finishRestart: --- end ---");
|
|
}
|
|
|
|
|
|
void Core::stop()
|
|
{
|
|
qDebug("Core::stop");
|
|
qDebug("Core::stop: state: %s", stateToString().toUtf8().data());
|
|
|
|
if (state()==Stopped) {
|
|
// if pressed stop twice, reset video to the beginning
|
|
qDebug("Core::stop: mset.current_sec: %f", mset.current_sec);
|
|
mset.current_sec = 0;
|
|
emit showTime( mset.current_sec );
|
|
#ifdef SEEKBAR_RESOLUTION
|
|
emit positionChanged( 0 );
|
|
#else
|
|
emit posChanged( 0 );
|
|
#endif
|
|
//updateWidgets();
|
|
}
|
|
|
|
stopMplayer();
|
|
emit mediaStoppedByUser();
|
|
}
|
|
|
|
|
|
void Core::play()
|
|
{
|
|
qDebug("Core::play");
|
|
|
|
if ((proc->isRunning()) && (state()==Paused)) {
|
|
tellmp("pause"); // Unpauses
|
|
}
|
|
else
|
|
if ((proc->isRunning()) && (state()==Playing)) {
|
|
// nothing to do, continue playing
|
|
}
|
|
else {
|
|
// if we're stopped, play it again
|
|
if ( !mdat.filename.isEmpty() ) {
|
|
/*
|
|
qDebug( "current_sec: %f, duration: %f", mset.current_sec, mdat.duration);
|
|
if ( (floor(mset.current_sec)) >= (floor(mdat.duration)) ) {
|
|
mset.current_sec = 0;
|
|
}
|
|
*/
|
|
restartPlay();
|
|
}
|
|
}
|
|
}
|
|
|
|
void Core::pause_and_frame_step() {
|
|
qDebug("Core::pause_and_frame_step");
|
|
|
|
if (proc->isRunning()) {
|
|
if (state() == Paused) {
|
|
tellmp("frame_step");
|
|
}
|
|
else {
|
|
tellmp("pause");
|
|
}
|
|
}
|
|
}
|
|
|
|
void Core::pause() {
|
|
qDebug("Core::pause");
|
|
qDebug("Core::pause: current state: %s", stateToString().toUtf8().data());
|
|
|
|
if (proc->isRunning()) {
|
|
// Pauses and unpauses
|
|
tellmp("pause");
|
|
}
|
|
}
|
|
|
|
void Core::play_or_pause() {
|
|
if (proc->isRunning()) {
|
|
pause();
|
|
} else {
|
|
play();
|
|
}
|
|
}
|
|
|
|
void Core::frameStep() {
|
|
qDebug("Core::frameStep");
|
|
|
|
if (proc->isRunning()) {
|
|
tellmp("frame_step");
|
|
}
|
|
}
|
|
|
|
void Core::screenshot() {
|
|
qDebug("Core::screenshot");
|
|
|
|
if ( (!pref->screenshot_directory.isEmpty()) &&
|
|
(QFileInfo(pref->screenshot_directory).isDir()) )
|
|
{
|
|
tellmp( pausing_prefix() + " screenshot 0");
|
|
qDebug("Core::screenshot: taken screenshot");
|
|
} else {
|
|
qDebug("Core::screenshot: error: directory for screenshots not valid");
|
|
QString text = "Screenshot NOT taken, folder not configured";
|
|
tellmp("osd_show_text \"" + text + "\" 3000 1");
|
|
emit showMessage(text);
|
|
}
|
|
}
|
|
|
|
void Core::processFinished()
|
|
{
|
|
qDebug("Core::processFinished");
|
|
|
|
#ifdef Q_OS_WIN
|
|
// Restores the Windows screensaver
|
|
if (pref->disable_screensaver) {
|
|
win_screensaver->restore();
|
|
}
|
|
#endif
|
|
|
|
qDebug("Core::processFinished: we_are_restarting: %d", we_are_restarting);
|
|
|
|
//mset.current_sec = 0;
|
|
|
|
if (!we_are_restarting) {
|
|
qDebug("Core::processFinished: play has finished!");
|
|
setState(Stopped);
|
|
//emit stateChanged(state());
|
|
}
|
|
|
|
int exit_code = proc->exitCode();
|
|
qDebug("Core::processFinished: exit_code: %d", exit_code);
|
|
if (exit_code != 0) {
|
|
emit mplayerFinishedWithError(exit_code);
|
|
}
|
|
}
|
|
|
|
void Core::fileReachedEnd() {
|
|
/*
|
|
if (mdat.type == TYPE_VCD) {
|
|
// If the first vcd title has nothing, it doesn't start to play
|
|
// and menus are not initialized.
|
|
initializeMenus();
|
|
}
|
|
*/
|
|
|
|
// If we're at the end of the movie, reset to 0
|
|
mset.current_sec = 0;
|
|
updateWidgets();
|
|
|
|
emit mediaFinished();
|
|
}
|
|
|
|
#if SEEKBAR_RESOLUTION
|
|
void Core::goToPosition(int value) {
|
|
qDebug("Core::goToPosition: value: %d", value);
|
|
if (mdat.duration > 0) {
|
|
int jump_time = (int) mdat.duration * value / SEEKBAR_RESOLUTION;
|
|
goToSec(jump_time);
|
|
}
|
|
}
|
|
#else
|
|
void Core::goToPos(int perc) {
|
|
qDebug("Core::goToPos: per: %d", perc);
|
|
tellmp( "seek " + QString::number(perc) + " 1");
|
|
}
|
|
#endif
|
|
|
|
|
|
void Core::startMplayer( QString file, double seek ) {
|
|
qDebug("Core::startMplayer");
|
|
|
|
if (file.isEmpty()) {
|
|
qWarning("Core:startMplayer: file is empty!");
|
|
return;
|
|
}
|
|
|
|
if (proc->isRunning()) {
|
|
qWarning("Core::startMplayer: MPlayer still running!");
|
|
return;
|
|
}
|
|
|
|
#ifdef Q_OS_WIN
|
|
// Disable the Windows screensaver
|
|
if (pref->disable_screensaver) {
|
|
win_screensaver->disable();
|
|
}
|
|
#endif
|
|
|
|
bool is_mkv = (QFileInfo(file).suffix().toLower() == "mkv");
|
|
|
|
// DVD
|
|
QString dvd_folder;
|
|
int dvd_title = -1;
|
|
if (mdat.type==TYPE_DVD) {
|
|
dvd_folder = Helper::dvdSplitFolder(file);
|
|
if (dvd_folder.isEmpty()) dvd_folder = pref->dvd_device;
|
|
// Remove trailing "/"
|
|
if (dvd_folder.endsWith("/")) {
|
|
#ifdef Q_OS_WIN
|
|
QRegExp r("^[A-Z]:/$");
|
|
int pos = r.indexIn(dvd_folder);
|
|
qDebug("Core::startMplayer: drive check: '%s': regexp: %d", dvd_folder.toUtf8().data(), pos);
|
|
if (pos == -1)
|
|
#endif
|
|
dvd_folder = dvd_folder.remove( dvd_folder.length()-1, 1);
|
|
}
|
|
dvd_title = Helper::dvdSplitTitle(file);
|
|
file = "dvd://" + QString::number(dvd_title);
|
|
}
|
|
|
|
// URL
|
|
bool url_is_playlist = file.endsWith(IS_PLAYLIST_TAG);
|
|
if (url_is_playlist) file = file.remove( QRegExp(IS_PLAYLIST_TAG_RX) );
|
|
|
|
bool screenshot_enabled = ( (!pref->screenshot_directory.isEmpty()) &&
|
|
(QFileInfo(pref->screenshot_directory).isDir()) );
|
|
|
|
proc->clearArguments();
|
|
|
|
// Set working directory to screenshot directory
|
|
if (screenshot_enabled) {
|
|
qDebug("Core::startMplayer: setting working directory to '%s'", pref->screenshot_directory.toUtf8().data());
|
|
proc->setWorkingDirectory( pref->screenshot_directory );
|
|
}
|
|
|
|
// Use absolute path, otherwise after changing to the screenshot directory
|
|
// the mplayer path might not be found if it's a relative path
|
|
// (seems to be necessary only for linux)
|
|
QString mplayer_bin = pref->mplayer_bin;
|
|
QFileInfo fi(mplayer_bin);
|
|
if (fi.exists() && fi.isExecutable() && !fi.isDir()) {
|
|
mplayer_bin = fi.absoluteFilePath();
|
|
}
|
|
|
|
proc->addArgument( mplayer_bin );
|
|
|
|
proc->addArgument("-noquiet");
|
|
|
|
if (pref->fullscreen && pref->use_mplayer_window) {
|
|
proc->addArgument("-fs");
|
|
} else {
|
|
// No mplayer fullscreen mode
|
|
proc->addArgument("-nofs");
|
|
}
|
|
|
|
// Demuxer and audio and video codecs:
|
|
if (!mset.forced_demuxer.isEmpty()) {
|
|
proc->addArgument("-demuxer");
|
|
proc->addArgument(mset.forced_demuxer);
|
|
}
|
|
if (!mset.forced_audio_codec.isEmpty()) {
|
|
proc->addArgument("-ac");
|
|
proc->addArgument(mset.forced_audio_codec);
|
|
}
|
|
if (!mset.forced_video_codec.isEmpty()) {
|
|
proc->addArgument("-vc");
|
|
proc->addArgument(mset.forced_video_codec);
|
|
}
|
|
|
|
if (pref->use_hwac3) {
|
|
proc->addArgument("-afm");
|
|
proc->addArgument("hwac3");
|
|
}
|
|
|
|
|
|
QString lavdopts;
|
|
|
|
if ( (pref->h264_skip_loop_filter == Preferences::LoopDisabled) ||
|
|
((pref->h264_skip_loop_filter == Preferences::LoopDisabledOnHD) &&
|
|
(mset.is264andHD)) )
|
|
{
|
|
if (!lavdopts.isEmpty()) lavdopts += ":";
|
|
lavdopts += "skiploopfilter=all";
|
|
}
|
|
|
|
if (pref->show_motion_vectors) {
|
|
if (!lavdopts.isEmpty()) lavdopts += ":";
|
|
lavdopts += "vismv=7";
|
|
}
|
|
|
|
if (pref->threads > 1) {
|
|
if (!lavdopts.isEmpty()) lavdopts += ":";
|
|
lavdopts += "threads=" + QString::number(pref->threads);
|
|
}
|
|
|
|
if (!lavdopts.isEmpty()) {
|
|
proc->addArgument("-lavdopts");
|
|
proc->addArgument(lavdopts);
|
|
}
|
|
|
|
proc->addArgument("-sub-fuzziness");
|
|
proc->addArgument( QString::number(pref->subfuzziness) );
|
|
|
|
proc->addArgument("-identify");
|
|
|
|
#if GENERIC_CHAPTER_SUPPORT
|
|
if (MplayerVersion::isMplayerAtLeast(27667)) {
|
|
// From r27667 the number of chapters can be obtained from ID_CHAPTERS
|
|
mset.current_chapter_id = 0; // Reset chapters
|
|
} else {
|
|
#endif
|
|
// We need this to get info about mkv chapters
|
|
if (is_mkv) {
|
|
proc->addArgument("-msglevel");
|
|
proc->addArgument("demux=6");
|
|
|
|
// **** Reset chapter ***
|
|
// Select first chapter, otherwise we cannot
|
|
// resume playback at the same point
|
|
// (time would be relative to chapter)
|
|
mset.current_chapter_id = 0;
|
|
}
|
|
#if GENERIC_CHAPTER_SUPPORT
|
|
}
|
|
#endif
|
|
|
|
proc->addArgument("-slave");
|
|
|
|
if (!pref->vo.isEmpty()) {
|
|
proc->addArgument( "-vo");
|
|
proc->addArgument( pref->vo );
|
|
} else {
|
|
proc->addArgument("-vo");
|
|
#ifdef Q_OS_WIN
|
|
// On Windows Vista, the default vo is already set in preferences.cpp
|
|
proc->addArgument("directx,");
|
|
#else
|
|
proc->addArgument("xv,");
|
|
#endif
|
|
}
|
|
|
|
#if USE_ADAPTER
|
|
if (pref->adapter > -1) {
|
|
proc->addArgument("-adapter");
|
|
proc->addArgument(QString::number(pref->adapter));
|
|
}
|
|
#endif
|
|
|
|
if (!pref->ao.isEmpty()) {
|
|
proc->addArgument( "-ao");
|
|
proc->addArgument( pref->ao );
|
|
}
|
|
#ifndef Q_OS_WIN
|
|
else {
|
|
proc->addArgument( "-ao");
|
|
proc->addArgument( "alsa," );
|
|
}
|
|
#endif
|
|
|
|
proc->addArgument( "-zoom");
|
|
proc->addArgument("-nokeepaspect");
|
|
|
|
// Performance options
|
|
#ifdef Q_OS_WIN
|
|
QString p;
|
|
int app_p = NORMAL_PRIORITY_CLASS;
|
|
switch (pref->priority) {
|
|
case Preferences::Realtime: p = "realtime";
|
|
app_p = REALTIME_PRIORITY_CLASS;
|
|
break;
|
|
case Preferences::High: p = "high";
|
|
app_p = REALTIME_PRIORITY_CLASS;
|
|
break;
|
|
case Preferences::AboveNormal: p = "abovenormal";
|
|
app_p = HIGH_PRIORITY_CLASS;
|
|
break;
|
|
case Preferences::Normal: p = "normal";
|
|
app_p = ABOVE_NORMAL_PRIORITY_CLASS;
|
|
break;
|
|
case Preferences::BelowNormal: p = "belownormal"; break;
|
|
case Preferences::Idle: p = "idle"; break;
|
|
default: p = "normal";
|
|
}
|
|
proc->addArgument("-priority");
|
|
proc->addArgument( p );
|
|
SetPriorityClass(GetCurrentProcess(), app_p);
|
|
qDebug("Core::startMplayer: priority of smplayer process set to %d", app_p);
|
|
#endif
|
|
|
|
if (pref->frame_drop) {
|
|
proc->addArgument("-framedrop");
|
|
}
|
|
|
|
if (pref->hard_frame_drop) {
|
|
proc->addArgument("-hardframedrop");
|
|
}
|
|
|
|
if (pref->autosync) {
|
|
proc->addArgument("-autosync");
|
|
proc->addArgument( QString::number( pref->autosync_factor ) );
|
|
}
|
|
|
|
if (pref->use_direct_rendering) {
|
|
proc->addArgument("-dr");
|
|
} else {
|
|
proc->addArgument("-nodr");
|
|
}
|
|
|
|
if (pref->use_double_buffer) {
|
|
proc->addArgument("-double");
|
|
} else {
|
|
proc->addArgument("-nodouble");
|
|
}
|
|
|
|
#ifndef Q_OS_WIN
|
|
if (!pref->use_mplayer_window) {
|
|
proc->addArgument( "-input" );
|
|
proc->addArgument( "conf=" + Paths::dataPath() +"/input.conf" );
|
|
}
|
|
#endif
|
|
|
|
#ifdef Q_WS_X11
|
|
if (pref->disable_screensaver) {
|
|
proc->addArgument("-stop-xscreensaver");
|
|
} else {
|
|
proc->addArgument("-nostop-xscreensaver");
|
|
}
|
|
#endif
|
|
|
|
if (!pref->use_mplayer_window) {
|
|
proc->addArgument("-wid");
|
|
proc->addArgument( QString::number( (int) mplayerwindow->videoLayer()->winId() ) );
|
|
|
|
#if USE_COLORKEY
|
|
#ifdef Q_OS_WIN
|
|
if ((pref->vo.startsWith("directx")) || (pref->vo.isEmpty())) {
|
|
proc->addArgument("-colorkey");
|
|
//proc->addArgument( "0x"+QString::number(pref->color_key, 16) );
|
|
proc->addArgument( ColorUtils::colorToRGB(pref->color_key) );
|
|
} else {
|
|
#endif
|
|
qDebug("Core::startMplayer: * not using -colorkey for %s", pref->vo.toUtf8().data());
|
|
qDebug("Core::startMplayer: * report if you can't see the video");
|
|
#ifdef Q_OS_WIN
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
// Square pixels
|
|
proc->addArgument("-monitorpixelaspect");
|
|
proc->addArgument("1");
|
|
} else {
|
|
// no -wid
|
|
if (!pref->monitor_aspect.isEmpty()) {
|
|
proc->addArgument("-monitoraspect");
|
|
proc->addArgument( pref->monitor_aspect );
|
|
}
|
|
}
|
|
|
|
// Subtitles fonts
|
|
if ((pref->use_ass_subtitles) && (pref->freetype_support)) {
|
|
// ASS:
|
|
proc->addArgument("-ass");
|
|
proc->addArgument("-embeddedfonts");
|
|
|
|
proc->addArgument("-ass-line-spacing");
|
|
proc->addArgument(QString::number(pref->ass_line_spacing));
|
|
|
|
proc->addArgument( "-ass-font-scale");
|
|
proc->addArgument( QString::number(mset.sub_scale_ass) );
|
|
|
|
// Load the styles.ass file
|
|
if (!QFile::exists(Paths::subtitleStyleFile())) {
|
|
// If file doesn't exist, create it
|
|
pref->ass_styles.exportStyles(Paths::subtitleStyleFile());
|
|
}
|
|
if (QFile::exists(Paths::subtitleStyleFile())) {
|
|
proc->addArgument("-ass-styles");
|
|
proc->addArgument( Paths::subtitleStyleFile() );
|
|
} else {
|
|
qWarning("Core::startMplayer: '%s' doesn't exist", Paths::subtitleStyleFile().toUtf8().constData());
|
|
}
|
|
// Use the same font for OSD
|
|
if (!pref->ass_styles.fontname.isEmpty()) {
|
|
proc->addArgument("-fontconfig");
|
|
proc->addArgument("-font");
|
|
proc->addArgument( pref->ass_styles.fontname );
|
|
}
|
|
|
|
} else {
|
|
// NO ASS:
|
|
if (pref->freetype_support) proc->addArgument("-noass");
|
|
|
|
if ( (pref->use_fontconfig) && (!pref->font_name.isEmpty()) ) {
|
|
proc->addArgument("-fontconfig");
|
|
proc->addArgument("-font");
|
|
proc->addArgument( pref->font_name );
|
|
}
|
|
|
|
if ( (!pref->use_fontconfig) && (!pref->font_file.isEmpty()) ) {
|
|
proc->addArgument("-font");
|
|
proc->addArgument( pref->font_file );
|
|
}
|
|
|
|
if (pref->freetype_support) {
|
|
proc->addArgument( "-subfont-autoscale");
|
|
proc->addArgument( QString::number( pref->font_autoscale ) );
|
|
|
|
proc->addArgument( "-subfont-text-scale");
|
|
proc->addArgument( QString::number(mset.sub_scale) );
|
|
}
|
|
}
|
|
|
|
// Subtitle encoding
|
|
{
|
|
QString encoding;
|
|
if ( (pref->use_enca) && (!pref->enca_lang.isEmpty()) ) {
|
|
encoding = "enca:"+ pref->enca_lang;
|
|
if (!pref->subcp.isEmpty()) {
|
|
encoding += ":"+ pref->subcp;
|
|
}
|
|
}
|
|
else
|
|
if (!pref->subcp.isEmpty()) {
|
|
encoding = pref->subcp;
|
|
}
|
|
|
|
if (!encoding.isEmpty()) {
|
|
proc->addArgument("-subcp");
|
|
proc->addArgument( encoding );
|
|
}
|
|
}
|
|
|
|
if (pref->use_closed_caption_subs) {
|
|
proc->addArgument("-subcc");
|
|
}
|
|
|
|
if (pref->use_forced_subs_only) {
|
|
proc->addArgument("-forcedsubsonly");
|
|
}
|
|
|
|
if (mset.current_video_id != MediaSettings::NoneSelected) {
|
|
proc->addArgument("-vid");
|
|
proc->addArgument( QString::number( mset.current_video_id ) );
|
|
}
|
|
|
|
if (mset.current_audio_id != MediaSettings::NoneSelected) {
|
|
proc->addArgument("-aid");
|
|
proc->addArgument( QString::number( mset.current_audio_id ) );
|
|
}
|
|
|
|
if (!initial_subtitle.isEmpty()) {
|
|
mset.external_subtitles = initial_subtitle;
|
|
initial_subtitle = "";
|
|
just_loaded_external_subs = true; // Big ugly hack :(
|
|
}
|
|
if (!mset.external_subtitles.isEmpty()) {
|
|
if (QFileInfo(mset.external_subtitles).suffix().toLower()=="idx") {
|
|
// sub/idx subtitles
|
|
QFileInfo fi;
|
|
|
|
#ifdef Q_OS_WIN
|
|
if (pref->use_short_pathnames)
|
|
fi.setFile(Helper::shortPathName(mset.external_subtitles));
|
|
else
|
|
#endif
|
|
fi.setFile(mset.external_subtitles);
|
|
|
|
QString s = fi.path() +"/"+ fi.completeBaseName();
|
|
qDebug("Core::startMplayer: subtitle file without extension: '%s'", s.toUtf8().data());
|
|
proc->addArgument("-vobsub");
|
|
proc->addArgument( s );
|
|
} else {
|
|
proc->addArgument("-sub");
|
|
#ifdef Q_OS_WIN
|
|
if (pref->use_short_pathnames)
|
|
proc->addArgument(Helper::shortPathName(mset.external_subtitles));
|
|
else
|
|
#endif
|
|
proc->addArgument( mset.external_subtitles );
|
|
}
|
|
}
|
|
|
|
if (!mset.external_audio.isEmpty()) {
|
|
proc->addArgument("-audiofile");
|
|
#ifdef Q_OS_WIN
|
|
if (pref->use_short_pathnames)
|
|
proc->addArgument(Helper::shortPathName(mset.external_audio));
|
|
else
|
|
#endif
|
|
proc->addArgument( mset.external_audio );
|
|
}
|
|
|
|
proc->addArgument("-subpos");
|
|
proc->addArgument( QString::number(mset.sub_pos) );
|
|
|
|
if (mset.audio_delay!=0) {
|
|
proc->addArgument("-delay");
|
|
proc->addArgument( QString::number( (double) mset.audio_delay/1000 ) );
|
|
}
|
|
|
|
if (mset.sub_delay!=0) {
|
|
proc->addArgument("-subdelay");
|
|
proc->addArgument( QString::number( (double) mset.sub_delay/1000 ) );
|
|
}
|
|
|
|
// Contrast, brightness...
|
|
if (pref->change_video_equalizer_on_startup) {
|
|
if (mset.contrast != 0) {
|
|
proc->addArgument("-contrast");
|
|
proc->addArgument( QString::number( mset.contrast ) );
|
|
}
|
|
|
|
if (mset.brightness != 0) {
|
|
proc->addArgument("-brightness");
|
|
proc->addArgument( QString::number( mset.brightness ) );
|
|
}
|
|
|
|
if (mset.hue != 0) {
|
|
proc->addArgument("-hue");
|
|
proc->addArgument( QString::number( mset.hue ) );
|
|
}
|
|
|
|
if (mset.saturation != 0) {
|
|
proc->addArgument("-saturation");
|
|
proc->addArgument( QString::number( mset.saturation ) );
|
|
}
|
|
}
|
|
|
|
// Set volume, requires mplayer svn r27872
|
|
bool use_volume_option = (pref->use_volume_option == Preferences::Enabled);
|
|
if (pref->use_volume_option == Preferences::Detect) {
|
|
use_volume_option = (MplayerVersion::isMplayerAtLeast(27872));
|
|
}
|
|
if ((use_volume_option) && (!pref->dont_change_volume)) {
|
|
proc->addArgument("-volume");
|
|
// Note: mset.volume may not be right, it can be the volume of the previous video if
|
|
// playing a new one, but I think it's better to use anyway the current volume on
|
|
// startup than set it to 0 or something.
|
|
// The right volume will be set later, when the video starts to play.
|
|
proc->addArgument( QString::number( mset.volume ) );
|
|
}
|
|
|
|
|
|
if (mdat.type==TYPE_DVD) {
|
|
if (!dvd_folder.isEmpty()) {
|
|
proc->addArgument("-dvd-device");
|
|
proc->addArgument( dvd_folder );
|
|
} else {
|
|
qWarning("Core::startMplayer: dvd device is empty!");
|
|
}
|
|
}
|
|
|
|
if ((mdat.type==TYPE_VCD) || (mdat.type==TYPE_AUDIO_CD)) {
|
|
if (!pref->cdrom_device.isEmpty()) {
|
|
proc->addArgument("-cdrom-device");
|
|
proc->addArgument( pref->cdrom_device );
|
|
}
|
|
}
|
|
|
|
if (mset.current_chapter_id > 0) {
|
|
proc->addArgument("-chapter");
|
|
int chapter = mset.current_chapter_id;
|
|
// Fix for older versions of mplayer:
|
|
#if GENERIC_CHAPTER_SUPPORT
|
|
if ((mdat.type == TYPE_DVD) && (firstChapter() == 0)) chapter++;
|
|
#else
|
|
if ((mdat.type == TYPE_DVD) && (dvdFirstChapter() == 0)) chapter++;
|
|
#endif
|
|
proc->addArgument( QString::number( chapter ) );
|
|
}
|
|
|
|
if (mset.current_angle_id > 0) {
|
|
proc->addArgument("-dvdangle");
|
|
proc->addArgument( QString::number( mset.current_angle_id ) );
|
|
}
|
|
|
|
|
|
int cache = 0;
|
|
switch (mdat.type) {
|
|
case TYPE_FILE : cache = pref->cache_for_files; break;
|
|
case TYPE_DVD : cache = pref->cache_for_dvds; break;
|
|
case TYPE_STREAM : cache = pref->cache_for_streams; break;
|
|
case TYPE_VCD : cache = pref->cache_for_vcds; break;
|
|
case TYPE_AUDIO_CD : cache = pref->cache_for_audiocds; break;
|
|
default: cache = 0;
|
|
}
|
|
|
|
if (cache > 31) { // Minimum value for cache = 32
|
|
proc->addArgument("-cache");
|
|
proc->addArgument( QString::number( cache ) );
|
|
} else {
|
|
proc->addArgument("-nocache");
|
|
}
|
|
|
|
if (mset.speed != 1.0) {
|
|
proc->addArgument("-speed");
|
|
proc->addArgument( QString::number( mset.speed ) );
|
|
}
|
|
|
|
// If seek < 5 it's better to allow the video to start from the beginning
|
|
if ((seek >= 5) && (!pref->loop)) {
|
|
proc->addArgument("-ss");
|
|
proc->addArgument( QString::number( seek ) );
|
|
}
|
|
|
|
proc->addArgument("-osdlevel");
|
|
proc->addArgument( QString::number( pref->osd ) );
|
|
|
|
if (pref->use_idx) {
|
|
proc->addArgument("-idx");
|
|
}
|
|
|
|
if (mdat.type == TYPE_STREAM) {
|
|
if (pref->prefer_ipv4) {
|
|
proc->addArgument("-prefer-ipv4");
|
|
} else {
|
|
proc->addArgument("-prefer-ipv6");
|
|
}
|
|
}
|
|
|
|
if (pref->use_correct_pts) {
|
|
proc->addArgument("-correct-pts");
|
|
} else {
|
|
if (pref->mplayer_detected_version > 0) {
|
|
if (MplayerVersion::isMplayerAtLeast(26842)) {
|
|
proc->addArgument("-nocorrect-pts");
|
|
} else {
|
|
proc->addArgument("-no-correct-pts");
|
|
}
|
|
} else {
|
|
qDebug("Core::startMplayer: unknown version of mplayer, not passing -no(-)correct-pts");
|
|
}
|
|
}
|
|
|
|
// Video filters:
|
|
// Phase
|
|
if (mset.phase_filter) {
|
|
proc->addArgument("-vf-add");
|
|
proc->addArgument( "phase=A" );
|
|
}
|
|
|
|
// Deinterlace
|
|
if (mset.current_deinterlacer != MediaSettings::NoDeinterlace) {
|
|
proc->addArgument("-vf-add");
|
|
switch (mset.current_deinterlacer) {
|
|
case MediaSettings::L5: proc->addArgument("pp=l5"); break;
|
|
case MediaSettings::Yadif: proc->addArgument("yadif"); break;
|
|
case MediaSettings::LB: proc->addArgument("pp=lb"); break;
|
|
case MediaSettings::Yadif_1: proc->addArgument("yadif=1"); break;
|
|
case MediaSettings::Kerndeint: proc->addArgument("kerndeint=5"); break;
|
|
}
|
|
}
|
|
|
|
#if !NEW_ASPECT_CODE
|
|
// Panscan (crop)
|
|
if (!mset.panscan_filter.isEmpty()) {
|
|
proc->addArgument( "-vf-add" );
|
|
proc->addArgument( mset.panscan_filter );
|
|
}
|
|
|
|
// Crop 4:3 to 16:9
|
|
if (!mset.crop_43to169_filter.isEmpty()) {
|
|
proc->addArgument( "-vf-add" );
|
|
proc->addArgument( mset.crop_43to169_filter );
|
|
}
|
|
#endif
|
|
|
|
// Denoise
|
|
if (mset.current_denoiser != MediaSettings::NoDenoise) {
|
|
proc->addArgument("-vf-add");
|
|
if (mset.current_denoiser==MediaSettings::DenoiseSoft) {
|
|
proc->addArgument( "hqdn3d=2:1:2" );
|
|
} else {
|
|
proc->addArgument( "hqdn3d" );
|
|
}
|
|
}
|
|
|
|
// Deblock
|
|
if (mset.deblock_filter) {
|
|
proc->addArgument("-vf-add");
|
|
proc->addArgument( "pp=vb/hb" );
|
|
}
|
|
|
|
// Dering
|
|
if (mset.dering_filter) {
|
|
proc->addArgument("-vf-add");
|
|
proc->addArgument( "pp=dr" );
|
|
}
|
|
|
|
// Upscale
|
|
if (mset.upscaling_filter) {
|
|
int width = DesktopInfo::desktop_size(mplayerwindow).width();
|
|
proc->addArgument("-sws");
|
|
proc->addArgument("9");
|
|
proc->addArgument("-vf-add");
|
|
proc->addArgument("scale="+QString::number(width)+":-2");
|
|
}
|
|
|
|
// Addnoise
|
|
if (mset.noise_filter) {
|
|
proc->addArgument("-vf-add");
|
|
proc->addArgument( "noise=9ah:5ah" );
|
|
}
|
|
|
|
// Postprocessing
|
|
if (mset.postprocessing_filter) {
|
|
proc->addArgument("-vf-add");
|
|
proc->addArgument("pp");
|
|
proc->addArgument("-autoq");
|
|
proc->addArgument( QString::number(pref->autoq) );
|
|
}
|
|
|
|
|
|
// Letterbox (expand)
|
|
#if NEW_ASPECT_CODE
|
|
if ((mset.add_letterbox) || (pref->fullscreen && pref->add_blackborders_on_fullscreen)) {
|
|
proc->addArgument("-vf-add");
|
|
proc->addArgument( QString("expand=:::::%1,harddup").arg( DesktopInfo::desktop_aspectRatio(mplayerwindow)) );
|
|
// Note: on some videos (h264 for instance) the subtitles doesn't disappear,
|
|
// appearing the new ones on top of the old ones. It seems adding another
|
|
// filter after expand fixes the problem. I chose harddup 'cos I think
|
|
// it will be harmless in mplayer.
|
|
// Anyway, if you know a proper way to fix the problem, please tell me.
|
|
}
|
|
#else
|
|
if (mset.letterbox == MediaSettings::Letterbox_43) {
|
|
proc->addArgument("-vf-add");
|
|
proc->addArgument("expand=:::::4/3");
|
|
}
|
|
else
|
|
if (mset.letterbox == MediaSettings::Letterbox_169) {
|
|
proc->addArgument("-vf-add");
|
|
proc->addArgument("expand=:::::16/9");
|
|
}
|
|
#endif
|
|
|
|
// Software equalizer
|
|
if ( (pref->use_soft_video_eq) ) {
|
|
proc->addArgument("-vf-add");
|
|
QString eq_filter = "eq2,hue";
|
|
if ( (pref->vo == "gl") || (pref->vo == "gl2")
|
|
#ifdef Q_OS_WIN
|
|
|| (pref->vo == "directx:noaccel")
|
|
#endif
|
|
) eq_filter += ",scale";
|
|
proc->addArgument(eq_filter);
|
|
}
|
|
|
|
// Additional video filters, supplied by user
|
|
// File
|
|
if ( !mset.mplayer_additional_video_filters.isEmpty() ) {
|
|
proc->addArgument("-vf-add");
|
|
proc->addArgument( mset.mplayer_additional_video_filters );
|
|
}
|
|
// Global
|
|
if ( !pref->mplayer_additional_video_filters.isEmpty() ) {
|
|
proc->addArgument("-vf-add");
|
|
proc->addArgument( pref->mplayer_additional_video_filters );
|
|
}
|
|
|
|
bool force_noslices = false;
|
|
|
|
// Filters for subtitles on screenshots
|
|
if ((screenshot_enabled) && (pref->subtitles_on_screenshots))
|
|
{
|
|
if (pref->use_ass_subtitles) {
|
|
proc->addArgument("-vf-add");
|
|
proc->addArgument("ass");
|
|
} else {
|
|
proc->addArgument("-vf-add");
|
|
proc->addArgument("expand=osd=1");
|
|
//proc->addArgument("-noslices");
|
|
force_noslices = true;
|
|
}
|
|
}
|
|
|
|
// Rotate
|
|
if (mset.rotate != MediaSettings::NoRotate) {
|
|
proc->addArgument( "-vf-add" );
|
|
proc->addArgument( QString("rotate=%1").arg(mset.rotate) );
|
|
}
|
|
|
|
// Flip
|
|
if (mset.flip) {
|
|
proc->addArgument( "-vf-add" );
|
|
// expand + flip doesn't work well, a workaround is to add another
|
|
// filter between them, so that's why harddup is here
|
|
proc->addArgument("harddup,flip");
|
|
}
|
|
|
|
// Mirror
|
|
if (mset.mirror) {
|
|
proc->addArgument( "-vf-add" );
|
|
proc->addArgument("mirror");
|
|
}
|
|
|
|
// Screenshots
|
|
if (screenshot_enabled) {
|
|
proc->addArgument("-vf-add");
|
|
proc->addArgument("screenshot");
|
|
}
|
|
|
|
// slices
|
|
if ((pref->use_slices) && (!force_noslices)) {
|
|
proc->addArgument("-slices");
|
|
} else {
|
|
proc->addArgument("-noslices");
|
|
}
|
|
|
|
|
|
// Audio channels
|
|
if (mset.audio_use_channels != 0) {
|
|
proc->addArgument("-channels");
|
|
proc->addArgument( QString::number( mset.audio_use_channels ) );
|
|
}
|
|
|
|
// Stereo mode
|
|
if (mset.stereo_mode != 0) {
|
|
proc->addArgument("-stereo");
|
|
proc->addArgument( QString::number( mset.stereo_mode ) );
|
|
}
|
|
|
|
// Audio filters
|
|
QString af="";
|
|
if (mset.karaoke_filter) {
|
|
af="karaoke";
|
|
}
|
|
|
|
if (mset.extrastereo_filter) {
|
|
if (!af.isEmpty()) af += ",";
|
|
af += "extrastereo";
|
|
}
|
|
|
|
if (mset.volnorm_filter) {
|
|
if (!af.isEmpty()) af += ",";
|
|
af += "volnorm=2";
|
|
}
|
|
|
|
bool use_scaletempo = (pref->use_scaletempo == Preferences::Enabled);
|
|
if (pref->use_scaletempo == Preferences::Detect) {
|
|
use_scaletempo = (MplayerVersion::isMplayerAtLeast(24924));
|
|
}
|
|
if (use_scaletempo) {
|
|
if (!af.isEmpty()) af += ",";
|
|
af += "scaletempo";
|
|
}
|
|
|
|
// Audio equalizer
|
|
if (pref->use_audio_equalizer) {
|
|
if (!af.isEmpty()) af += ",";
|
|
af += "equalizer=" + Helper::equalizerListToString(mset.audio_equalizer);
|
|
}
|
|
|
|
|
|
// Additional audio filters, supplied by user
|
|
// File
|
|
if ( !pref->mplayer_additional_audio_filters.isEmpty() ) {
|
|
if (!af.isEmpty()) af += ",";
|
|
af += pref->mplayer_additional_audio_filters;
|
|
}
|
|
// Global
|
|
if ( !mset.mplayer_additional_audio_filters.isEmpty() ) {
|
|
if (!af.isEmpty()) af += ",";
|
|
af += mset.mplayer_additional_audio_filters;
|
|
}
|
|
|
|
if (!af.isEmpty()) {
|
|
proc->addArgument("-af");
|
|
proc->addArgument( af );
|
|
}
|
|
|
|
if (pref->use_soft_vol) {
|
|
proc->addArgument("-softvol");
|
|
proc->addArgument("-softvol-max");
|
|
proc->addArgument( QString::number(pref->softvol_max) );
|
|
}
|
|
|
|
// Load edl file
|
|
if (pref->use_edl_files) {
|
|
QString edl_f;
|
|
QFileInfo f(file);
|
|
QString basename = f.path() + "/" + f.completeBaseName();
|
|
|
|
qDebug("Core::startMplayer: file basename: '%s'", basename.toUtf8().data());
|
|
|
|
if (QFile::exists(basename+".edl"))
|
|
edl_f = basename+".edl";
|
|
else
|
|
if (QFile::exists(basename+".EDL"))
|
|
edl_f = basename+".EDL";
|
|
|
|
qDebug("Core::startMplayer: edl file: '%s'", edl_f.toUtf8().data());
|
|
if (!edl_f.isEmpty()) {
|
|
proc->addArgument("-edl");
|
|
proc->addArgument(edl_f);
|
|
}
|
|
}
|
|
|
|
// Additional options supplied by the user
|
|
// File
|
|
if (!mset.mplayer_additional_options.isEmpty()) {
|
|
QStringList args = mset.mplayer_additional_options.split(" ");
|
|
QStringList::Iterator it = args.begin();
|
|
while( it != args.end() ) {
|
|
proc->addArgument( (*it) );
|
|
++it;
|
|
}
|
|
}
|
|
// Global
|
|
if (!pref->mplayer_additional_options.isEmpty()) {
|
|
QStringList args = pref->mplayer_additional_options.split(" ");
|
|
QStringList::Iterator it = args.begin();
|
|
while( it != args.end() ) {
|
|
proc->addArgument( (*it) );
|
|
++it;
|
|
}
|
|
}
|
|
|
|
// File to play
|
|
if (url_is_playlist) {
|
|
proc->addArgument("-playlist");
|
|
}
|
|
|
|
#ifdef Q_OS_WIN
|
|
if (pref->use_short_pathnames)
|
|
proc->addArgument(Helper::shortPathName(file));
|
|
else
|
|
#endif
|
|
proc->addArgument( file );
|
|
|
|
// It seems the loop option must be after the filename
|
|
if (pref->loop) {
|
|
proc->addArgument("-loop");
|
|
proc->addArgument("0");
|
|
}
|
|
|
|
emit aboutToStartPlaying();
|
|
|
|
QString commandline = proc->arguments().join(" ");
|
|
qDebug("Core::startMplayer: command: '%s'", commandline.toUtf8().data());
|
|
|
|
//Log command
|
|
QString line_for_log = commandline + "\n";
|
|
emit logLineAvailable(line_for_log);
|
|
|
|
if ( !proc->start() ) {
|
|
// error handling
|
|
qWarning("Core::startMplayer: mplayer process didn't start");
|
|
}
|
|
|
|
}
|
|
|
|
void Core::stopMplayer() {
|
|
qDebug("Core::stopMplayer");
|
|
|
|
if (!proc->isRunning()) {
|
|
qWarning("Core::stopMplayer: mplayer in not running!");
|
|
return;
|
|
}
|
|
|
|
tellmp("quit");
|
|
|
|
qDebug("Core::stopMplayer: Waiting mplayer to finish...");
|
|
if (!proc->waitForFinished(5000)) {
|
|
qWarning("Core::stopMplayer: process didn't finish. Killing it...");
|
|
proc->kill();
|
|
}
|
|
|
|
qDebug("Core::stopMplayer: Finished. (I hope)");
|
|
}
|
|
|
|
|
|
void Core::goToSec( double sec ) {
|
|
qDebug("Core::goToSec: %f", sec);
|
|
|
|
if (sec < 0) sec = 0;
|
|
if (sec > mdat.duration ) sec = mdat.duration - 20;
|
|
tellmp("seek " + QString::number(sec) + " 2");
|
|
}
|
|
|
|
|
|
void Core::seek(int secs) {
|
|
qDebug("Core::seek: %d", secs);
|
|
if ( (proc->isRunning()) && (secs!=0) ) {
|
|
tellmp("seek " + QString::number(secs) + " 0");
|
|
}
|
|
}
|
|
|
|
void Core::sforward() {
|
|
qDebug("Core::sforward");
|
|
seek( pref->seeking1 ); // +10s
|
|
}
|
|
|
|
void Core::srewind() {
|
|
qDebug("Core::srewind");
|
|
seek( -pref->seeking1 ); // -10s
|
|
}
|
|
|
|
|
|
void Core::forward() {
|
|
qDebug("Core::forward");
|
|
seek( pref->seeking2 ); // +1m
|
|
}
|
|
|
|
|
|
void Core::rewind() {
|
|
qDebug("Core::rewind");
|
|
seek( -pref->seeking2 ); // -1m
|
|
}
|
|
|
|
|
|
void Core::fastforward() {
|
|
qDebug("Core::fastforward");
|
|
seek( pref->seeking3 ); // +10m
|
|
}
|
|
|
|
|
|
void Core::fastrewind() {
|
|
qDebug("Core::fastrewind");
|
|
seek( -pref->seeking3 ); // -10m
|
|
}
|
|
|
|
void Core::forward(int secs) {
|
|
qDebug("Core::forward: %d", secs);
|
|
seek(secs);
|
|
}
|
|
|
|
void Core::rewind(int secs) {
|
|
qDebug("Core::rewind: %d", secs);
|
|
seek(-secs);
|
|
}
|
|
|
|
void Core::wheelUp() {
|
|
qDebug("Core::wheelUp");
|
|
switch (pref->wheel_function) {
|
|
case Preferences::Volume : incVolume(); break;
|
|
case Preferences::Zoom : incPanscan(); break;
|
|
case Preferences::Seeking : forward( pref->seeking4 ); break;
|
|
case Preferences::ChangeSpeed : incSpeed10(); break;
|
|
default : {} // do nothing
|
|
}
|
|
}
|
|
|
|
void Core::wheelDown() {
|
|
qDebug("Core::wheelDown");
|
|
switch (pref->wheel_function) {
|
|
case Preferences::Volume : decVolume(); break;
|
|
case Preferences::Zoom : decPanscan(); break;
|
|
case Preferences::Seeking : rewind( pref->seeking4 ); break;
|
|
case Preferences::ChangeSpeed : decSpeed10(); break;
|
|
default : {} // do nothing
|
|
}
|
|
}
|
|
|
|
|
|
void Core::toggleRepeat() {
|
|
qDebug("Core::toggleRepeat");
|
|
toggleRepeat( !pref->loop );
|
|
}
|
|
|
|
void Core::toggleRepeat(bool b) {
|
|
qDebug("Core::toggleRepeat: %d", b);
|
|
if ( pref->loop != b ) {
|
|
pref->loop = b;
|
|
if (MplayerVersion::isMplayerAtLeast(23747)) {
|
|
// Use slave command
|
|
int v = -1; // no loop
|
|
if (pref->loop) v = 0; // infinite loop
|
|
tellmp( QString("loop %1 1").arg(v) );
|
|
} else {
|
|
// Restart mplayer
|
|
if (proc->isRunning()) restartPlay();
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void Core::toggleFlip() {
|
|
qDebug("Core::toggleFlip");
|
|
toggleFlip( !mset.flip );
|
|
}
|
|
|
|
void Core::toggleFlip(bool b) {
|
|
qDebug("Core::toggleFlip: %d", b);
|
|
|
|
if (mset.flip != b) {
|
|
mset.flip = b;
|
|
if (proc->isRunning()) restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::toggleMirror() {
|
|
qDebug("Core::toggleMirror");
|
|
toggleMirror( !mset.mirror );
|
|
}
|
|
|
|
void Core::toggleMirror(bool b) {
|
|
qDebug("Core::toggleMirror: %d", b);
|
|
|
|
if (mset.mirror != b) {
|
|
mset.mirror = b;
|
|
if (proc->isRunning()) restartPlay();
|
|
}
|
|
}
|
|
|
|
// Audio filters
|
|
void Core::toggleKaraoke() {
|
|
toggleKaraoke( !mset.karaoke_filter );
|
|
}
|
|
|
|
void Core::toggleKaraoke(bool b) {
|
|
qDebug("Core::toggleKaraoke: %d", b);
|
|
if (b != mset.karaoke_filter) {
|
|
mset.karaoke_filter = b;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::toggleExtrastereo() {
|
|
toggleExtrastereo( !mset.extrastereo_filter );
|
|
}
|
|
|
|
void Core::toggleExtrastereo(bool b) {
|
|
qDebug("Core::toggleExtrastereo: %d", b);
|
|
if (b != mset.extrastereo_filter) {
|
|
mset.extrastereo_filter = b;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::toggleVolnorm() {
|
|
toggleVolnorm( !mset.volnorm_filter );
|
|
}
|
|
|
|
void Core::toggleVolnorm(bool b) {
|
|
qDebug("Core::toggleVolnorm: %d", b);
|
|
if (b != mset.volnorm_filter) {
|
|
mset.volnorm_filter = b;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::setAudioChannels(int channels) {
|
|
qDebug("Core::setAudioChannels:%d", channels);
|
|
if (channels != mset.audio_use_channels ) {
|
|
mset.audio_use_channels = channels;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::setStereoMode(int mode) {
|
|
qDebug("Core::setStereoMode:%d", mode);
|
|
if (mode != mset.stereo_mode ) {
|
|
mset.stereo_mode = mode;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
|
|
// Video filters
|
|
void Core::toggleAutophase() {
|
|
toggleAutophase( !mset.phase_filter );
|
|
}
|
|
|
|
void Core::toggleAutophase( bool b ) {
|
|
qDebug("Core::toggleAutophase: %d", b);
|
|
if ( b != mset.phase_filter) {
|
|
mset.phase_filter = b;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::toggleDeblock() {
|
|
toggleDeblock( !mset.deblock_filter );
|
|
}
|
|
|
|
void Core::toggleDeblock(bool b) {
|
|
qDebug("Core::toggleDeblock: %d", b);
|
|
if ( b != mset.deblock_filter ) {
|
|
mset.deblock_filter = b;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::toggleDering() {
|
|
toggleDering( !mset.dering_filter );
|
|
}
|
|
|
|
void Core::toggleDering(bool b) {
|
|
qDebug("Core::toggleDering: %d", b);
|
|
if ( b != mset.dering_filter) {
|
|
mset.dering_filter = b;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::toggleNoise() {
|
|
toggleNoise( !mset.noise_filter );
|
|
}
|
|
|
|
void Core::toggleNoise(bool b) {
|
|
qDebug("Core::toggleNoise: %d", b);
|
|
if ( b!= mset.noise_filter ) {
|
|
mset.noise_filter = b;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::togglePostprocessing() {
|
|
togglePostprocessing( !mset.postprocessing_filter );
|
|
}
|
|
|
|
void Core::togglePostprocessing(bool b) {
|
|
qDebug("Core::togglePostprocessing: %d", b);
|
|
if ( b != mset.postprocessing_filter ) {
|
|
mset.postprocessing_filter = b;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::changeDenoise(int id) {
|
|
qDebug( "Core::changeDenoise: %d", id );
|
|
if (id != mset.current_denoiser) {
|
|
mset.current_denoiser = id;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::changeUpscale(bool b) {
|
|
qDebug( "Core::changeUpscale: %d", b );
|
|
if (mset.upscaling_filter != b) {
|
|
mset.upscaling_filter = b;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::setBrightness(int value) {
|
|
qDebug("Core::setBrightness: %d", value);
|
|
|
|
if (value > 100) value = 100;
|
|
if (value < -100) value = -100;
|
|
|
|
if (value != mset.brightness) {
|
|
tellmp(pausing_prefix() + " brightness " + QString::number(value) + " 1");
|
|
mset.brightness = value;
|
|
displayMessage( tr("Brightness: %1").arg(value) );
|
|
emit videoEqualizerNeedsUpdate();
|
|
}
|
|
}
|
|
|
|
|
|
void Core::setContrast(int value) {
|
|
qDebug("Core::setContrast: %d", value);
|
|
|
|
if (value > 100) value = 100;
|
|
if (value < -100) value = -100;
|
|
|
|
if (value != mset.contrast) {
|
|
tellmp(pausing_prefix() + " contrast " + QString::number(value) + " 1");
|
|
mset.contrast = value;
|
|
displayMessage( tr("Contrast: %1").arg(value) );
|
|
emit videoEqualizerNeedsUpdate();
|
|
}
|
|
}
|
|
|
|
void Core::setGamma(int value) {
|
|
qDebug("Core::setGamma: %d", value);
|
|
|
|
if (value > 100) value = 100;
|
|
if (value < -100) value = -100;
|
|
|
|
if (value != mset.gamma) {
|
|
tellmp(pausing_prefix() + " gamma " + QString::number(value) + " 1");
|
|
mset.gamma= value;
|
|
displayMessage( tr("Gamma: %1").arg(value) );
|
|
emit videoEqualizerNeedsUpdate();
|
|
}
|
|
}
|
|
|
|
void Core::setHue(int value) {
|
|
qDebug("Core::setHue: %d", value);
|
|
|
|
if (value > 100) value = 100;
|
|
if (value < -100) value = -100;
|
|
|
|
if (value != mset.hue) {
|
|
tellmp(pausing_prefix() + " hue " + QString::number(value) + " 1");
|
|
mset.hue = value;
|
|
displayMessage( tr("Hue: %1").arg(value) );
|
|
emit videoEqualizerNeedsUpdate();
|
|
}
|
|
}
|
|
|
|
void Core::setSaturation(int value) {
|
|
qDebug("Core::setSaturation: %d", value);
|
|
|
|
if (value > 100) value = 100;
|
|
if (value < -100) value = -100;
|
|
|
|
if (value != mset.saturation) {
|
|
tellmp(pausing_prefix() + " saturation " + QString::number(value) + " 1");
|
|
mset.saturation = value;
|
|
displayMessage( tr("Saturation: %1").arg(value) );
|
|
emit videoEqualizerNeedsUpdate();
|
|
}
|
|
}
|
|
|
|
void Core::incBrightness() {
|
|
setBrightness(mset.brightness + 4);
|
|
}
|
|
|
|
void Core::decBrightness() {
|
|
setBrightness(mset.brightness - 4);
|
|
}
|
|
|
|
void Core::incContrast() {
|
|
setContrast(mset.contrast + 4);
|
|
}
|
|
|
|
void Core::decContrast() {
|
|
setContrast(mset.contrast - 4);
|
|
}
|
|
|
|
void Core::incGamma() {
|
|
setGamma(mset.gamma + 4);
|
|
}
|
|
|
|
void Core::decGamma() {
|
|
setGamma(mset.gamma - 4);
|
|
}
|
|
|
|
void Core::incHue() {
|
|
setHue(mset.hue + 4);
|
|
}
|
|
|
|
void Core::decHue() {
|
|
setHue(mset.hue - 4);
|
|
}
|
|
|
|
void Core::incSaturation() {
|
|
setSaturation(mset.saturation + 4);
|
|
}
|
|
|
|
void Core::decSaturation() {
|
|
setSaturation(mset.saturation - 4);
|
|
}
|
|
|
|
void Core::setSpeed( double value ) {
|
|
qDebug("Core::setSpeed: %f", value);
|
|
|
|
if (value < 0.10) value = 0.10;
|
|
if (value > 100) value = 100;
|
|
|
|
mset.speed = value;
|
|
tellmp( "speed_set " + QString::number( value ) );
|
|
}
|
|
|
|
void Core::incSpeed10() {
|
|
qDebug("Core::incSpeed10");
|
|
setSpeed( (double) mset.speed + 0.1 );
|
|
}
|
|
|
|
void Core::decSpeed10() {
|
|
qDebug("Core::decSpeed10");
|
|
setSpeed( (double) mset.speed - 0.1 );
|
|
}
|
|
|
|
void Core::incSpeed4() {
|
|
qDebug("Core::incSpeed4");
|
|
setSpeed( (double) mset.speed + 0.04 );
|
|
}
|
|
|
|
void Core::decSpeed4() {
|
|
qDebug("Core::decSpeed4");
|
|
setSpeed( (double) mset.speed - 0.04 );
|
|
}
|
|
|
|
void Core::incSpeed1() {
|
|
qDebug("Core::incSpeed1");
|
|
setSpeed( (double) mset.speed + 0.01 );
|
|
}
|
|
|
|
void Core::decSpeed1() {
|
|
qDebug("Core::decSpeed1");
|
|
setSpeed( (double) mset.speed - 0.01 );
|
|
}
|
|
|
|
void Core::doubleSpeed() {
|
|
qDebug("Core::doubleSpeed");
|
|
setSpeed( (double) mset.speed * 2 );
|
|
}
|
|
|
|
void Core::halveSpeed() {
|
|
qDebug("Core::halveSpeed");
|
|
setSpeed( (double) mset.speed / 2 );
|
|
}
|
|
|
|
void Core::normalSpeed() {
|
|
setSpeed(1);
|
|
}
|
|
|
|
void Core::setVolume(int volume, bool force) {
|
|
qDebug("Core::setVolume: %d", volume);
|
|
|
|
if ((volume==mset.volume) && (!force)) return;
|
|
|
|
mset.volume = volume;
|
|
if (mset.volume > 100 ) mset.volume = 100;
|
|
if (mset.volume < 0 ) mset.volume = 0;
|
|
|
|
if (state() == Paused) {
|
|
// Change volume later, after quiting pause
|
|
change_volume_after_unpause = true;
|
|
} else {
|
|
tellmp("volume " + QString::number(volume) + " 1");
|
|
}
|
|
|
|
//if (mset.mute) mute(TRUE);
|
|
mset.mute=false;
|
|
|
|
updateWidgets();
|
|
|
|
displayMessage( tr("Volume: %1").arg(mset.volume) );
|
|
emit volumeChanged( mset.volume );
|
|
}
|
|
|
|
void Core::switchMute() {
|
|
qDebug("Core::switchMute");
|
|
|
|
mset.mute = !mset.mute;
|
|
mute(mset.mute);
|
|
}
|
|
|
|
void Core::mute(bool b) {
|
|
qDebug("Core::mute");
|
|
|
|
mset.mute = b;
|
|
|
|
int v = 0;
|
|
if (mset.mute) v = 1;
|
|
tellmp( pausing_prefix() + " mute " + QString::number(v) );
|
|
|
|
updateWidgets();
|
|
}
|
|
|
|
void Core::incVolume() {
|
|
qDebug("Core::incVolume");
|
|
setVolume(mset.volume + 4);
|
|
}
|
|
|
|
void Core::decVolume() {
|
|
qDebug("Core::incVolume");
|
|
setVolume(mset.volume-4);
|
|
}
|
|
|
|
void Core::incSubDelay() {
|
|
qDebug("Core::incSubDelay");
|
|
|
|
mset.sub_delay += 100;
|
|
tellmp("sub_delay " + QString::number( (double) mset.sub_delay/1000 ) +" 1");
|
|
}
|
|
|
|
void Core::decSubDelay() {
|
|
qDebug("Core::decSubDelay");
|
|
|
|
mset.sub_delay -= 100;
|
|
tellmp("sub_delay " + QString::number( (double) mset.sub_delay/1000 ) +" 1");
|
|
}
|
|
|
|
void Core::incAudioDelay() {
|
|
qDebug("Core::incAudioDelay");
|
|
|
|
mset.audio_delay += 100;
|
|
tellmp("audio_delay " + QString::number( (double) mset.audio_delay/1000 ) +" 1");
|
|
}
|
|
|
|
void Core::decAudioDelay() {
|
|
qDebug("Core::decAudioDelay");
|
|
|
|
mset.audio_delay -= 100;
|
|
tellmp("audio_delay " + QString::number( (double) mset.audio_delay/1000 ) +" 1");
|
|
}
|
|
|
|
void Core::incSubPos() {
|
|
qDebug("Core::incSubPos");
|
|
|
|
mset.sub_pos++;
|
|
if (mset.sub_pos > 100) mset.sub_pos = 100;
|
|
tellmp("sub_pos " + QString::number( mset.sub_pos ) + " 1");
|
|
}
|
|
|
|
void Core::decSubPos() {
|
|
qDebug("Core::decSubPos");
|
|
|
|
mset.sub_pos--;
|
|
if (mset.sub_pos < 0) mset.sub_pos = 0;
|
|
tellmp("sub_pos " + QString::number( mset.sub_pos ) + " 1");
|
|
}
|
|
|
|
bool Core::subscale_need_restart() {
|
|
bool need_restart = false;
|
|
|
|
need_restart = (pref->change_sub_scale_should_restart == Preferences::Enabled);
|
|
if (pref->change_sub_scale_should_restart == Preferences::Detect) {
|
|
if (pref->use_ass_subtitles)
|
|
need_restart = (!MplayerVersion::isMplayerAtLeast(25843));
|
|
else
|
|
need_restart = (!MplayerVersion::isMplayerAtLeast(23745));
|
|
}
|
|
return need_restart;
|
|
}
|
|
|
|
void Core::changeSubScale(double value) {
|
|
qDebug("Core::changeSubScale: %f", value);
|
|
|
|
bool need_restart = subscale_need_restart();
|
|
|
|
if (value < 0) value = 0;
|
|
|
|
if (pref->use_ass_subtitles) {
|
|
if (value != mset.sub_scale_ass) {
|
|
mset.sub_scale_ass = value;
|
|
if (need_restart) {
|
|
restartPlay();
|
|
} else {
|
|
tellmp("sub_scale " + QString::number( mset.sub_scale_ass ) + " 1");
|
|
}
|
|
displayMessage( tr("Font scale: %1").arg(mset.sub_scale_ass) );
|
|
}
|
|
} else {
|
|
// No ass
|
|
if (value != mset.sub_scale) {
|
|
mset.sub_scale = value;
|
|
if (need_restart) {
|
|
restartPlay();
|
|
} else {
|
|
tellmp("sub_scale " + QString::number( mset.sub_scale ) + " 1");
|
|
|
|
}
|
|
displayMessage( tr("Font scale: %1").arg(mset.sub_scale) );
|
|
}
|
|
}
|
|
}
|
|
|
|
void Core::incSubScale() {
|
|
double step = 0.20;
|
|
|
|
if (pref->use_ass_subtitles) {
|
|
changeSubScale( mset.sub_scale_ass + step );
|
|
} else {
|
|
if (subscale_need_restart()) step = 1;
|
|
changeSubScale( mset.sub_scale + step );
|
|
}
|
|
}
|
|
|
|
void Core::decSubScale() {
|
|
double step = 0.20;
|
|
|
|
if (pref->use_ass_subtitles) {
|
|
changeSubScale( mset.sub_scale_ass - step );
|
|
} else {
|
|
if (subscale_need_restart()) step = 1;
|
|
changeSubScale( mset.sub_scale - step );
|
|
}
|
|
}
|
|
|
|
void Core::incSubStep() {
|
|
qDebug("Core::incSubStep");
|
|
tellmp("sub_step +1");
|
|
}
|
|
|
|
void Core::decSubStep() {
|
|
qDebug("Core::decSubStep");
|
|
tellmp("sub_step -1");
|
|
}
|
|
|
|
// Audio equalizer functions
|
|
void Core::setAudioEqualizer(AudioEqualizerList values, bool restart) {
|
|
mset.audio_equalizer = values;
|
|
|
|
if (!restart) {
|
|
tellmp( "af_eq_set_bands " + Helper::equalizerListToString(values) );
|
|
} else {
|
|
restartPlay();
|
|
}
|
|
|
|
emit audioEqualizerNeedsUpdate();
|
|
}
|
|
|
|
void Core::updateAudioEqualizer() {
|
|
setAudioEqualizer(mset.audio_equalizer);
|
|
}
|
|
|
|
void Core::setAudioEq0(int value) {
|
|
mset.audio_equalizer[0] = value;
|
|
updateAudioEqualizer();
|
|
}
|
|
|
|
void Core::setAudioEq1(int value) {
|
|
mset.audio_equalizer[1] = value;
|
|
updateAudioEqualizer();
|
|
}
|
|
|
|
void Core::setAudioEq2(int value) {
|
|
mset.audio_equalizer[2] = value;
|
|
updateAudioEqualizer();
|
|
}
|
|
|
|
void Core::setAudioEq3(int value) {
|
|
mset.audio_equalizer[3] = value;
|
|
updateAudioEqualizer();
|
|
}
|
|
|
|
void Core::setAudioEq4(int value) {
|
|
mset.audio_equalizer[4] = value;
|
|
updateAudioEqualizer();
|
|
}
|
|
|
|
void Core::setAudioEq5(int value) {
|
|
mset.audio_equalizer[5] = value;
|
|
updateAudioEqualizer();
|
|
}
|
|
|
|
void Core::setAudioEq6(int value) {
|
|
mset.audio_equalizer[6] = value;
|
|
updateAudioEqualizer();
|
|
}
|
|
|
|
void Core::setAudioEq7(int value) {
|
|
mset.audio_equalizer[7] = value;
|
|
updateAudioEqualizer();
|
|
}
|
|
|
|
void Core::setAudioEq8(int value) {
|
|
mset.audio_equalizer[8] = value;
|
|
updateAudioEqualizer();
|
|
}
|
|
|
|
void Core::setAudioEq9(int value) {
|
|
mset.audio_equalizer[9] = value;
|
|
updateAudioEqualizer();
|
|
}
|
|
|
|
|
|
|
|
void Core::changeCurrentSec(double sec) {
|
|
mset.current_sec = sec;
|
|
|
|
if (mset.starting_time != -1) {
|
|
mset.current_sec -= mset.starting_time;
|
|
}
|
|
|
|
if (state() != Playing) {
|
|
setState(Playing);
|
|
qDebug("Core::changeCurrentSec: mplayer reports that now it's playing");
|
|
//emit mediaStartPlay();
|
|
//emit stateChanged(state());
|
|
}
|
|
|
|
emit showTime(mset.current_sec);
|
|
|
|
// Emit posChanged:
|
|
static int last_second = 0;
|
|
|
|
if (floor(sec)==last_second) return; // Update only once per second
|
|
last_second = (int) floor(sec);
|
|
|
|
#ifdef SEEKBAR_RESOLUTION
|
|
int value = 0;
|
|
if ( (mdat.duration > 1) && (mset.current_sec > 1) &&
|
|
(mdat.duration > mset.current_sec) )
|
|
{
|
|
value = ( (int) mset.current_sec * SEEKBAR_RESOLUTION) / (int) mdat.duration;
|
|
}
|
|
emit positionChanged(value);
|
|
#else
|
|
int perc = 0;
|
|
if ( (mdat.duration > 1) && (mset.current_sec > 1) &&
|
|
(mdat.duration > mset.current_sec) )
|
|
{
|
|
perc = ( (int) mset.current_sec * 100) / (int) mdat.duration;
|
|
}
|
|
emit posChanged( perc );
|
|
#endif
|
|
}
|
|
|
|
void Core::gotStartingTime(double time) {
|
|
qDebug("Core::gotStartingTime: %f", time);
|
|
qDebug("Core::gotStartingTime: current_sec: %f", mset.current_sec);
|
|
if ((mset.starting_time == -1.0) && (mset.current_sec == 0)) {
|
|
mset.starting_time = time;
|
|
qDebug("Core::gotStartingTime: starting time set to %f", time);
|
|
}
|
|
}
|
|
|
|
|
|
void Core::changePause() {
|
|
qDebug("Core::changePause");
|
|
qDebug("Core::changePause: mplayer reports that it's paused");
|
|
setState(Paused);
|
|
//emit stateChanged(state());
|
|
}
|
|
|
|
void Core::changeDeinterlace(int ID) {
|
|
qDebug("Core::changeDeinterlace: %d", ID);
|
|
|
|
if (ID!=mset.current_deinterlacer) {
|
|
mset.current_deinterlacer = ID;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void Core::changeSubtitle(int ID) {
|
|
qDebug("Core::changeSubtitle: %d", ID);
|
|
|
|
mset.current_sub_id = ID;
|
|
if (ID==MediaSettings::SubNone) {
|
|
ID=-1;
|
|
}
|
|
|
|
if (ID==MediaSettings::NoneSelected) {
|
|
ID=-1;
|
|
qDebug("Core::changeSubtitle: subtitle is NoneSelected, this shouldn't happen. ID set to -1.");
|
|
}
|
|
|
|
qDebug("Core::changeSubtitle: ID: %d", ID);
|
|
|
|
bool use_new_commands = (pref->use_new_sub_commands == Preferences::Enabled);
|
|
if (pref->use_new_sub_commands == Preferences::Detect) {
|
|
use_new_commands = (MplayerVersion::isMplayerAtLeast(25158));
|
|
}
|
|
|
|
if (!use_new_commands) {
|
|
// Old command sub_select
|
|
tellmp( "sub_select " + QString::number(ID) );
|
|
} else {
|
|
// New commands
|
|
int real_id = -1;
|
|
if (ID == -1) {
|
|
tellmp( "sub_source -1" );
|
|
} else {
|
|
bool valid_item = ( (ID >= 0) && (ID < mdat.subs.numItems()) );
|
|
if (!valid_item) qWarning("Core::changeSubtitle: ID: %d is not valid!", ID);
|
|
if ( (mdat.subs.numItems() > 0) && (valid_item) ) {
|
|
real_id = mdat.subs.itemAt(ID).ID();
|
|
switch (mdat.subs.itemAt(ID).type()) {
|
|
case SubData::Vob:
|
|
tellmp( "sub_vob " + QString::number(real_id) );
|
|
break;
|
|
case SubData::Sub:
|
|
tellmp( "sub_demux " + QString::number(real_id) );
|
|
break;
|
|
case SubData::File:
|
|
tellmp( "sub_file " + QString::number(real_id) );
|
|
break;
|
|
default: {
|
|
qWarning("Core::changeSubtitle: unknown type!");
|
|
}
|
|
}
|
|
} else {
|
|
qWarning("Core::changeSubtitle: subtitle list is empty!");
|
|
}
|
|
}
|
|
}
|
|
|
|
updateWidgets();
|
|
}
|
|
|
|
void Core::nextSubtitle() {
|
|
qDebug("Core::nextSubtitle");
|
|
|
|
if ( (mset.current_sub_id == MediaSettings::SubNone) &&
|
|
(mdat.subs.numItems() > 0) )
|
|
{
|
|
changeSubtitle(0);
|
|
}
|
|
else {
|
|
int item = mset.current_sub_id + 1;
|
|
if (item >= mdat.subs.numItems()) {
|
|
item = MediaSettings::SubNone;
|
|
}
|
|
changeSubtitle( item );
|
|
}
|
|
}
|
|
|
|
void Core::changeAudio(int ID, bool allow_restart) {
|
|
qDebug("Core::changeAudio: ID: %d, allow_restart: %d", ID, allow_restart);
|
|
|
|
if (ID!=mset.current_audio_id) {
|
|
mset.current_audio_id = ID;
|
|
qDebug("changeAudio: ID: %d", ID);
|
|
|
|
bool need_restart = false;
|
|
if (allow_restart) {
|
|
need_restart = (pref->fast_audio_change == Preferences::Disabled);
|
|
if (pref->fast_audio_change == Preferences::Detect) {
|
|
need_restart = (!MplayerVersion::isMplayerAtLeast(21441));
|
|
}
|
|
}
|
|
|
|
if (need_restart) {
|
|
restartPlay();
|
|
} else {
|
|
tellmp("switch_audio " + QString::number(ID) );
|
|
//#ifdef Q_OS_WIN
|
|
// Workaround for a mplayer problem in windows,
|
|
// volume is too loud after changing audio.
|
|
|
|
// Workaround too for a mplayer problem in linux,
|
|
// the volume is reduced if using -softvol-max.
|
|
if (!pref->dont_change_volume) {
|
|
setVolume( mset.volume, true );
|
|
}
|
|
//#endif
|
|
if (mset.mute) mute(true); // if muted, mute again
|
|
updateWidgets();
|
|
}
|
|
}
|
|
}
|
|
|
|
void Core::nextAudio() {
|
|
qDebug("Core::nextAudio");
|
|
|
|
int item = mdat.audios.find( mset.current_audio_id );
|
|
if (item == -1) {
|
|
qWarning("Core::nextAudio: audio ID %d not found!", mset.current_audio_id);
|
|
} else {
|
|
qDebug( "Core::nextAudio: numItems: %d, item: %d", mdat.audios.numItems(), item);
|
|
item++;
|
|
if (item >= mdat.audios.numItems()) item=0;
|
|
int ID = mdat.audios.itemAt(item).ID();
|
|
qDebug( "Core::nextAudio: item: %d, ID: %d", item, ID);
|
|
changeAudio( ID );
|
|
}
|
|
}
|
|
|
|
void Core::changeVideo(int ID, bool allow_restart) {
|
|
qDebug("Core::changeVideo: ID: %d, allow_restart: %d", ID, allow_restart);
|
|
|
|
if (ID != mset.current_video_id) {
|
|
mset.current_video_id = ID;
|
|
qDebug("Core::changeVideo: ID set to: %d", ID);
|
|
|
|
bool need_restart = false;
|
|
if (allow_restart) {
|
|
// afaik lavf doesn't require to restart, any other?
|
|
need_restart = (mdat.demuxer != "lavf");
|
|
}
|
|
|
|
if (need_restart) {
|
|
restartPlay();
|
|
} else {
|
|
tellmp("set_property switch_video " + QString::number(ID) );
|
|
}
|
|
}
|
|
}
|
|
|
|
void Core::nextVideo() {
|
|
qDebug("Core::nextVideo");
|
|
|
|
int item = mdat.videos.find( mset.current_video_id );
|
|
if (item == -1) {
|
|
qWarning("Core::nextVideo: video ID %d not found!", mset.current_video_id);
|
|
} else {
|
|
qDebug( "Core::nextVideo: numItems: %d, item: %d", mdat.videos.numItems(), item);
|
|
item++;
|
|
if (item >= mdat.videos.numItems()) item=0;
|
|
int ID = mdat.videos.itemAt(item).ID();
|
|
qDebug( "Core::nextVideo: item: %d, ID: %d", item, ID);
|
|
changeVideo( ID );
|
|
}
|
|
}
|
|
|
|
|
|
void Core::changeTitle(int ID) {
|
|
if (mdat.type == TYPE_VCD) {
|
|
// VCD
|
|
openVCD( ID );
|
|
}
|
|
else
|
|
if (mdat.type == TYPE_AUDIO_CD) {
|
|
// AUDIO CD
|
|
openAudioCD( ID );
|
|
}
|
|
else
|
|
if (mdat.type == TYPE_DVD) {
|
|
QString dvd_url = "dvd://" + QString::number(ID);
|
|
QString folder = Helper::dvdSplitFolder(mdat.filename);
|
|
if (!folder.isEmpty()) dvd_url += ":" + folder;
|
|
|
|
openDVD(dvd_url);
|
|
//openDVD( ID );
|
|
}
|
|
}
|
|
|
|
void Core::changeChapter(int ID) {
|
|
qDebug("Core::changeChapter: ID: %d", ID);
|
|
|
|
if (ID != mset.current_chapter_id) {
|
|
//if (QFileInfo(mdat.filename).extension().lower()=="mkv") {
|
|
#if GENERIC_CHAPTER_SUPPORT
|
|
if (mdat.type != TYPE_DVD) {
|
|
#else
|
|
if (mdat.mkv_chapters > 0) {
|
|
// mkv doesn't require to restart
|
|
#endif
|
|
tellmp("seek_chapter " + QString::number(ID) +" 1");
|
|
mset.current_chapter_id = ID;
|
|
updateWidgets();
|
|
} else {
|
|
#if SMART_DVD_CHAPTERS
|
|
if (pref->cache_for_dvds == 0) {
|
|
#else
|
|
if (pref->fast_chapter_change) {
|
|
#endif
|
|
tellmp("seek_chapter " + QString::number(ID) +" 1");
|
|
mset.current_chapter_id = ID;
|
|
updateWidgets();
|
|
} else {
|
|
stopMplayer();
|
|
mset.current_chapter_id = ID;
|
|
//goToPos(0);
|
|
mset.current_sec = 0;
|
|
restartPlay();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
int Core::firstChapter() {
|
|
if (MplayerVersion::isMplayerAtLeast(25391))
|
|
return 1;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
#if !GENERIC_CHAPTER_SUPPORT
|
|
int Core::dvdFirstChapter() {
|
|
// TODO: check if the change really happens in the same version as mkv
|
|
return firstChapter();
|
|
}
|
|
#endif
|
|
|
|
void Core::prevChapter() {
|
|
qDebug("Core::prevChapter");
|
|
|
|
#if GENERIC_CHAPTER_SUPPORT
|
|
int last_chapter = 0;
|
|
int first_chapter = firstChapter();
|
|
|
|
last_chapter = mdat.chapters + firstChapter() - 1;
|
|
|
|
int ID = mset.current_chapter_id - 1;
|
|
if (ID < first_chapter) {
|
|
ID = last_chapter;
|
|
}
|
|
changeChapter(ID);
|
|
#else
|
|
int last_chapter = 0;
|
|
bool matroshka = (mdat.mkv_chapters > 0);
|
|
|
|
int first_chapter = dvdFirstChapter();
|
|
if (matroshka) first_chapter = firstChapter();
|
|
|
|
// Matroshka chapters
|
|
if (matroshka) last_chapter = mdat.mkv_chapters + firstChapter() - 1;
|
|
else
|
|
// DVD chapters
|
|
if (mset.current_title_id > 0) {
|
|
last_chapter = mdat.titles.item(mset.current_title_id).chapters() + dvdFirstChapter() -1;
|
|
}
|
|
|
|
int ID = mset.current_chapter_id - 1;
|
|
if (ID < first_chapter) {
|
|
ID = last_chapter;
|
|
}
|
|
changeChapter(ID);
|
|
#endif
|
|
}
|
|
|
|
void Core::nextChapter() {
|
|
qDebug("Core::nextChapter");
|
|
|
|
#if GENERIC_CHAPTER_SUPPORT
|
|
int last_chapter = 0;
|
|
last_chapter = mdat.chapters + firstChapter() - 1;
|
|
|
|
int ID = mset.current_chapter_id + 1;
|
|
if (ID > last_chapter) {
|
|
ID = firstChapter();
|
|
}
|
|
changeChapter(ID);
|
|
#else
|
|
int last_chapter = 0;
|
|
bool matroshka = (mdat.mkv_chapters > 0);
|
|
|
|
// Matroshka chapters
|
|
if (matroshka) last_chapter = mdat.mkv_chapters + firstChapter() - 1;
|
|
else
|
|
// DVD chapters
|
|
if (mset.current_title_id > 0) {
|
|
last_chapter = mdat.titles.item(mset.current_title_id).chapters() + dvdFirstChapter() - 1;
|
|
}
|
|
|
|
int ID = mset.current_chapter_id + 1;
|
|
if (ID > last_chapter) {
|
|
if (matroshka) ID = firstChapter(); else ID = dvdFirstChapter();
|
|
}
|
|
changeChapter(ID);
|
|
#endif
|
|
}
|
|
|
|
void Core::changeAngle(int ID) {
|
|
qDebug("Core::changeAngle: ID: %d", ID);
|
|
|
|
if (ID != mset.current_angle_id) {
|
|
mset.current_angle_id = ID;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
#if NEW_ASPECT_CODE
|
|
void Core::changeAspectRatio( int ID ) {
|
|
qDebug("Core::changeAspectRatio: %d", ID);
|
|
|
|
mset.aspect_ratio_id = ID;
|
|
|
|
double asp = mset.aspectToNum( (MediaSettings::Aspect) ID);
|
|
|
|
if (!pref->use_mplayer_window) {
|
|
mplayerwindow->setAspect( asp );
|
|
} else {
|
|
// Using mplayer own window
|
|
if (!mdat.novideo) {
|
|
tellmp("switch_ratio " + QString::number(asp));
|
|
}
|
|
}
|
|
|
|
QString asp_name = MediaSettings::aspectToString( (MediaSettings::Aspect) mset.aspect_ratio_id);
|
|
displayMessage( tr("Aspect ratio: %1").arg(asp_name) );
|
|
}
|
|
|
|
void Core::nextAspectRatio() {
|
|
int ID = mset.aspect_ratio_id + 1;
|
|
if (ID > MediaSettings:: Aspect11) ID = MediaSettings::AspectNone;
|
|
changeAspectRatio(ID);
|
|
|
|
updateWidgets();
|
|
}
|
|
|
|
void Core::changeLetterbox(bool b) {
|
|
qDebug("Core::changeLetterbox: %d", b);
|
|
|
|
if (mset.add_letterbox != b) {
|
|
mset.add_letterbox = b;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
#else
|
|
void Core::changeAspectRatio( int ID ) {
|
|
qDebug("Core::changeAspectRatio: %d", ID);
|
|
|
|
int old_id = mset.aspect_ratio_id;
|
|
mset.aspect_ratio_id = ID;
|
|
bool need_restart = FALSE;
|
|
|
|
double asp = mdat.video_aspect; // Set a default
|
|
|
|
if (ID==MediaSettings::Aspect43Letterbox) {
|
|
need_restart = (old_id != MediaSettings::Aspect43Letterbox);
|
|
asp = (double) 4 / 3;
|
|
mset.letterbox = MediaSettings::Letterbox_43;
|
|
mset.panscan_filter = "";
|
|
mset.crop_43to169_filter = "";
|
|
}
|
|
else
|
|
if (ID==MediaSettings::Aspect169Letterbox) {
|
|
need_restart = (old_id != MediaSettings::Aspect169Letterbox);
|
|
asp = (double) 16 / 9;
|
|
mset.letterbox = MediaSettings::Letterbox_169;
|
|
mset.panscan_filter = "";
|
|
mset.crop_43to169_filter = "";
|
|
}
|
|
else
|
|
if (ID==MediaSettings::Aspect43Panscan) {
|
|
need_restart = (old_id != MediaSettings::Aspect43Panscan);
|
|
mset.crop_43to169_filter = "";
|
|
mset.letterbox = MediaSettings::NoLetterbox;
|
|
|
|
asp = (double) 4 / 3;
|
|
int real_width = (int) round(mdat.video_height * mdat.video_aspect);
|
|
mset.panscan_filter = QString("scale=%1:%2,").arg(real_width).arg(mdat.video_height);
|
|
mset.panscan_filter += QString("crop=%1:%2").arg(round(mdat.video_height * 4 /3)).arg(mdat.video_height);
|
|
//mset.crop = QSize( mdat.video_height * 4 /3, mdat.video_height );
|
|
qDebug("Core::changeAspectRatio: panscan_filter = '%s'", mset.panscan_filter.toUtf8().data() );
|
|
|
|
}
|
|
else
|
|
if (ID==MediaSettings::Aspect43To169) {
|
|
need_restart = (old_id != MediaSettings::Aspect43To169);
|
|
mset.panscan_filter = "";
|
|
mset.crop_43to169_filter = "";
|
|
mset.letterbox = MediaSettings::NoLetterbox;
|
|
|
|
int real_width = (int) round(mdat.video_height * mdat.video_aspect);
|
|
int height = (int) round(real_width * 9 / 16);
|
|
|
|
qDebug("Core::changeAspectRatio: video_width: %d, video_height: %d", real_width, mdat.video_height);
|
|
qDebug("Core::changeAspectRatio: crop: %d, %d", real_width, height );
|
|
|
|
if (height > mdat.video_height) {
|
|
// Invalid size, source video is not 4:3
|
|
need_restart = FALSE;
|
|
} else {
|
|
asp = (double) 16 / 9;
|
|
mset.crop_43to169_filter = QString("scale=%1:%2,").arg(real_width).arg(mdat.video_height);
|
|
mset.crop_43to169_filter += QString("crop=%1:%2").arg(real_width).arg(height);
|
|
qDebug("Core::changeAspectRatio: crop_43to169_filter = '%s'", mset.crop_43to169_filter.toUtf8().data() );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//need_restart = (mset.force_letterbox == TRUE);
|
|
need_restart = ( (old_id == MediaSettings::Aspect43Letterbox) ||
|
|
(old_id == MediaSettings::Aspect169Letterbox) ||
|
|
(old_id == MediaSettings::Aspect43Panscan) ||
|
|
(old_id == MediaSettings::Aspect43To169) );
|
|
mset.letterbox = MediaSettings::NoLetterbox;
|
|
mset.panscan_filter = "";
|
|
mset.crop_43to169_filter = "";
|
|
switch (ID) {
|
|
//case MediaSettings::AspectAuto: asp = mdat.video_aspect; break;
|
|
case MediaSettings::AspectAuto: {
|
|
qDebug("Core::changeAspectRatio: mset.win_width %d, mset.win_height: %d", mset.win_width, mset.win_height);
|
|
asp = mset.win_aspect(); break;
|
|
}
|
|
case MediaSettings::Aspect43: asp = (double) 4 / 3; break;
|
|
case MediaSettings::Aspect169: asp = (double) 16 / 9; break;
|
|
case MediaSettings::Aspect149: asp = (double) 14 / 9; break;
|
|
case MediaSettings::Aspect1610: asp = (double) 16 / 10; break;
|
|
case MediaSettings::Aspect54: asp = (double) 5 / 4; break;
|
|
case MediaSettings::Aspect235: asp = 2.35; break;
|
|
}
|
|
}
|
|
|
|
if (!pref->use_mplayer_window) {
|
|
mplayerwindow->setAspect( asp );
|
|
} else {
|
|
// Using mplayer own window
|
|
tellmp("switch_ratio " + QString::number(asp));
|
|
}
|
|
|
|
updateWidgets();
|
|
|
|
if (need_restart) {
|
|
/*mdat.calculateWinResolution(mset.force_letterbox);*/
|
|
restartPlay();
|
|
}
|
|
}
|
|
#endif
|
|
|
|
void Core::changeOSD(int v) {
|
|
qDebug("Core::changeOSD: %d", v);
|
|
|
|
pref->osd = v;
|
|
tellmp("osd " + QString::number( pref->osd ) );
|
|
updateWidgets();
|
|
}
|
|
|
|
void Core::nextOSD() {
|
|
int osd = pref->osd + 1;
|
|
if (osd > Preferences::SeekTimerTotal) {
|
|
osd = Preferences::None;
|
|
}
|
|
changeOSD( osd );
|
|
}
|
|
|
|
void Core::changeRotate(int r) {
|
|
qDebug("Core::changeRotate: %d", r);
|
|
|
|
if (mset.rotate != r) {
|
|
mset.rotate = r;
|
|
restartPlay();
|
|
}
|
|
}
|
|
|
|
#if USE_ADAPTER
|
|
void Core::changeAdapter(int n) {
|
|
qDebug("Core::changeScreen: %d", n);
|
|
|
|
if (pref->adapter != n) {
|
|
pref->adapter = n;
|
|
restartPlay();
|
|
}
|
|
}
|
|
#endif
|
|
|
|
void Core::changeSize(int n) {
|
|
if ( /*(n != pref->size_factor) &&*/ (!pref->use_mplayer_window) ) {
|
|
pref->size_factor = n;
|
|
|
|
emit needResize(mset.win_width, mset.win_height);
|
|
updateWidgets();
|
|
}
|
|
}
|
|
|
|
void Core::toggleDoubleSize() {
|
|
if (pref->size_factor != 100)
|
|
changeSize(100);
|
|
else
|
|
changeSize(200);
|
|
}
|
|
|
|
void Core::changePanscan(double p) {
|
|
qDebug("Core::changePanscan: %f", p);
|
|
if (p < ZOOM_MIN) p = ZOOM_MIN;
|
|
|
|
mset.panscan_factor = p;
|
|
mplayerwindow->setZoom(p);
|
|
displayMessage( tr("Zoom: %1").arg(mset.panscan_factor) );
|
|
}
|
|
|
|
void Core::resetPanscan() {
|
|
changePanscan(1.0);
|
|
}
|
|
|
|
void Core::autoPanscan() {
|
|
double video_aspect = mset.aspectToNum( (MediaSettings::Aspect) mset.aspect_ratio_id);
|
|
|
|
if (video_aspect <= 0) {
|
|
QSize w = mplayerwindow->videoLayer()->size();
|
|
video_aspect = (double) w.width() / w.height();
|
|
}
|
|
|
|
double screen_aspect = DesktopInfo::desktop_aspectRatio(mplayerwindow);
|
|
double zoom_factor;
|
|
|
|
if (video_aspect > screen_aspect)
|
|
zoom_factor = video_aspect / screen_aspect;
|
|
else
|
|
zoom_factor = screen_aspect / video_aspect;
|
|
|
|
qDebug("Core::autoPanscan: video_aspect: %f", video_aspect);
|
|
qDebug("Core::autoPanscan: screen_aspect: %f", screen_aspect);
|
|
qDebug("Core::autoPanscan: zoom_factor: %f", zoom_factor);
|
|
|
|
changePanscan(zoom_factor);
|
|
}
|
|
|
|
void Core::autoPanscanFromLetterbox(double aspect) {
|
|
qDebug("Core::autoPanscanFromLetterbox: %f", aspect);
|
|
|
|
// Probably there's a much easy way to do this, but I'm not good with maths...
|
|
|
|
QSize desktop = DesktopInfo::desktop_size(mplayerwindow);
|
|
|
|
double video_aspect = mset.aspectToNum( (MediaSettings::Aspect) mset.aspect_ratio_id);
|
|
|
|
if (video_aspect <= 0) {
|
|
QSize w = mplayerwindow->videoLayer()->size();
|
|
video_aspect = (double) w.width() / w.height();
|
|
}
|
|
|
|
// Calculate size of the video in fullscreen
|
|
QSize video;
|
|
video.setHeight( desktop.height() );;
|
|
video.setWidth( (int) (video.height() * video_aspect) );
|
|
if (video.width() > desktop.width()) {
|
|
video.setWidth( desktop.width() );;
|
|
video.setHeight( (int) (video.width() / video_aspect) );
|
|
}
|
|
|
|
qDebug("Core::autoPanscanFromLetterbox: max. size of video: %d %d", video.width(), video.height());
|
|
|
|
// Calculate the size of the actual video inside the letterbox
|
|
QSize actual_video;
|
|
actual_video.setWidth( video.width() );
|
|
actual_video.setHeight( (int) (actual_video.width() / aspect) );
|
|
|
|
qDebug("Core::autoPanscanFromLetterbox: calculated size of actual video for aspect %f: %d %d", aspect, actual_video.width(), actual_video.height());
|
|
|
|
double zoom_factor = (double) desktop.height() / actual_video.height();
|
|
|
|
qDebug("Core::autoPanscanFromLetterbox: calculated zoom factor: %f", zoom_factor);
|
|
changePanscan(zoom_factor);
|
|
}
|
|
|
|
void Core::autoPanscanFor169() {
|
|
autoPanscanFromLetterbox((double) 16 / 9);
|
|
}
|
|
|
|
void Core::autoPanscanFor235() {
|
|
autoPanscanFromLetterbox(2.35);
|
|
}
|
|
|
|
void Core::incPanscan() {
|
|
qDebug("Core::incPanscan");
|
|
changePanscan( mset.panscan_factor + ZOOM_STEP );
|
|
}
|
|
|
|
void Core::decPanscan() {
|
|
qDebug("Core::decPanscan");
|
|
changePanscan( mset.panscan_factor - ZOOM_STEP );
|
|
}
|
|
|
|
void Core::changeUseAss(bool b) {
|
|
qDebug("Core::changeUseAss: %d", b);
|
|
|
|
if (pref->use_ass_subtitles != b) {
|
|
pref->use_ass_subtitles = b;
|
|
if (proc->isRunning()) restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::toggleClosedCaption(bool b) {
|
|
qDebug("Core::toggleClosedCaption: %d", b);
|
|
|
|
if (pref->use_closed_caption_subs != b) {
|
|
pref->use_closed_caption_subs = b;
|
|
if (proc->isRunning()) restartPlay();
|
|
}
|
|
}
|
|
|
|
void Core::toggleForcedSubsOnly(bool b) {
|
|
qDebug("Core::toggleForcedSubsOnly: %d", b);
|
|
|
|
if (pref->use_forced_subs_only != b) {
|
|
pref->use_forced_subs_only = b;
|
|
//if (proc->isRunning()) restartPlay();
|
|
int v = 0;
|
|
if (b) v = 1;
|
|
tellmp( QString("forced_subs_only %1").arg(v) );
|
|
}
|
|
}
|
|
|
|
void Core::visualizeMotionVectors(bool b) {
|
|
qDebug("Core::visualizeMotionVectors: %d", b);
|
|
|
|
if (pref->show_motion_vectors != b) {
|
|
pref->show_motion_vectors = b;
|
|
if (proc->isRunning()) restartPlay();
|
|
}
|
|
}
|
|
|
|
#if DVDNAV_SUPPORT
|
|
// dvdnav buttons
|
|
void Core::dvdnavUp() {
|
|
qDebug("Core::dvdnavUp");
|
|
tellmp("dvdnav up");
|
|
}
|
|
|
|
void Core::dvdnavDown() {
|
|
qDebug("Core::dvdnavDown");
|
|
tellmp("dvdnav down");
|
|
}
|
|
|
|
void Core::dvdnavLeft() {
|
|
qDebug("Core::dvdnavLeft");
|
|
tellmp("dvdnav left");
|
|
}
|
|
|
|
void Core::dvdnavRight() {
|
|
qDebug("Core::dvdnavRight");
|
|
tellmp("dvdnav right");
|
|
}
|
|
|
|
void Core::dvdnavMenu() {
|
|
qDebug("Core::dvdnavMenu");
|
|
tellmp("dvdnav menu");
|
|
}
|
|
|
|
void Core::dvdnavSelect() {
|
|
qDebug("Core::dvdnavSelect");
|
|
tellmp("dvdnav select");
|
|
}
|
|
|
|
void Core::dvdnavPrev() {
|
|
qDebug("Core::dvdnavPrev");
|
|
tellmp("dvdnav prev");
|
|
}
|
|
|
|
void Core::dvdnavMouse() {
|
|
qDebug("Core::dvdnavMouse");
|
|
|
|
QPoint p = mplayerwindow->videoLayer()->mapFromGlobal(QCursor::pos());
|
|
|
|
tellmp(QString("set_mouse_pos %1 %2").arg(p.x()).arg(p.y()));
|
|
tellmp("dvdnav mouse");
|
|
}
|
|
#endif
|
|
|
|
void Core::displayMessage(QString text) {
|
|
qDebug("Core::displayMessage");
|
|
emit showMessage(text);
|
|
|
|
if ((pref->fullscreen) && (state() != Paused)) {
|
|
tellmp("osd_show_text \"" + text + "\" 3000 1");
|
|
}
|
|
}
|
|
|
|
void Core::displayScreenshotName(QString filename) {
|
|
qDebug("Core::displayScreenshotName");
|
|
//QString text = tr("Screenshot saved as %1").arg(filename);
|
|
QString text = QString("Screenshot saved as %1").arg(filename);
|
|
|
|
if (MplayerVersion::isMplayerAtLeast(27665)) {
|
|
tellmp( "pausing_keep_force osd_show_text \"" + text + "\" 3000 1");
|
|
}
|
|
else
|
|
if (state() != Paused) {
|
|
// Dont' show the message on OSD while in pause, otherwise
|
|
// the video goes forward a frame.
|
|
tellmp("pausing_keep osd_show_text \"" + text + "\" 3000 1");
|
|
}
|
|
|
|
emit showMessage(text);
|
|
}
|
|
|
|
void Core::displayUpdatingFontCache() {
|
|
qDebug("Core::displayUpdatingFontCache");
|
|
emit showMessage( tr("Updating the font cache. This may take some seconds...") );
|
|
}
|
|
|
|
void Core::gotWindowResolution(int w, int h) {
|
|
qDebug("Core::gotWindowResolution: %d, %d", w, h);
|
|
//double aspect = (double) w/h;
|
|
|
|
if (pref->use_mplayer_window) {
|
|
emit noVideo();
|
|
} else {
|
|
if ((pref->resize_method==Preferences::Afterload) && (we_are_restarting)) {
|
|
// Do nothing
|
|
} else {
|
|
emit needResize(w,h);
|
|
}
|
|
}
|
|
|
|
mset.win_width = w;
|
|
mset.win_height = h;
|
|
|
|
//Override aspect ratio, is this ok?
|
|
//mdat.video_aspect = mset.win_aspect();
|
|
|
|
mplayerwindow->setResolution( w, h );
|
|
mplayerwindow->setAspect( mset.win_aspect() );
|
|
}
|
|
|
|
void Core::gotNoVideo() {
|
|
// File has no video (a sound file)
|
|
|
|
// Reduce size of window
|
|
/*
|
|
mset.win_width = mplayerwindow->size().width();
|
|
mset.win_height = 0;
|
|
mplayerwindow->setResolution( mset.win_width, mset.win_height );
|
|
emit needResize( mset.win_width, mset.win_height );
|
|
*/
|
|
//mplayerwindow->showLogo(TRUE);
|
|
emit noVideo();
|
|
}
|
|
|
|
void Core::gotVO(QString vo) {
|
|
qDebug("Core::gotVO: '%s'", vo.toUtf8().data() );
|
|
|
|
if ( pref->vo.isEmpty()) {
|
|
qDebug("Core::gotVO: saving vo");
|
|
pref->vo = vo;
|
|
}
|
|
}
|
|
|
|
void Core::gotAO(QString ao) {
|
|
qDebug("Core::gotAO: '%s'", ao.toUtf8().data() );
|
|
|
|
if ( pref->ao.isEmpty()) {
|
|
qDebug("Core::gotAO: saving ao");
|
|
pref->ao = ao;
|
|
}
|
|
}
|
|
|
|
void Core::streamTitleAndUrlChanged(QString title, QString url) {
|
|
mdat.stream_title = title;
|
|
mdat.stream_url = url;
|
|
emit mediaInfoChanged();
|
|
}
|
|
|
|
//! Called when the state changes
|
|
void Core::watchState(Core::State state) {
|
|
if ((state == Playing) && (change_volume_after_unpause))
|
|
{
|
|
// Delayed volume change
|
|
qDebug("Core::watchState: delayed volume change");
|
|
tellmp("volume " + QString::number(mset.volume) + " 1");
|
|
change_volume_after_unpause = false;
|
|
}
|
|
}
|
|
|
|
void Core::checkIfVideoIsHD() {
|
|
qDebug("Core::checkIfVideoIsHD");
|
|
|
|
// Check if the video is in HD and uses ffh264 codec.
|
|
if ((mdat.video_codec=="ffh264") && (mset.win_height >= pref->HD_height)) {
|
|
qDebug("Core::checkIfVideoIsHD: video == ffh264 and height >= %d", pref->HD_height);
|
|
if (!mset.is264andHD) {
|
|
mset.is264andHD = true;
|
|
if (pref->h264_skip_loop_filter == Preferences::LoopDisabledOnHD) {
|
|
qDebug("Core::checkIfVideoIsHD: we're about to restart the video");
|
|
restartPlay();
|
|
}
|
|
}
|
|
} else {
|
|
mset.is264andHD = false;
|
|
// FIXME: if the video was previously marked as HD, and now it's not
|
|
// then the video should restart too.
|
|
}
|
|
}
|
|
|
|
#if DELAYED_AUDIO_SETUP_ON_STARTUP && NOTIFY_AUDIO_CHANGES
|
|
#error "DELAYED_AUDIO_SETUP_ON_STARTUP and NOTIFY_AUDIO_CHANGES can't be both defined"
|
|
#endif
|
|
|
|
#if DELAYED_AUDIO_SETUP_ON_STARTUP
|
|
void Core::initAudioTrack() {
|
|
qDebug("Core::initAudioTrack");
|
|
|
|
// First audio if none selected
|
|
if ( (mset.current_audio_id == MediaSettings::NoneSelected) &&
|
|
(mdat.audios.numItems() > 0) )
|
|
{
|
|
// Don't set mset.current_audio_id here! changeAudio will do.
|
|
// Otherwise changeAudio will do nothing.
|
|
|
|
int audio = mdat.audios.itemAt(0).ID(); // First one
|
|
if (mdat.audios.existsItemAt(pref->initial_audio_track-1)) {
|
|
audio = mdat.audios.itemAt(pref->initial_audio_track-1).ID();
|
|
}
|
|
|
|
// Check if one of the audio tracks is the user preferred.
|
|
if (!pref->audio_lang.isEmpty()) {
|
|
int res = mdat.audios.findLang( pref->audio_lang );
|
|
if (res != -1) audio = res;
|
|
}
|
|
|
|
changeAudio( audio );
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#if NOTIFY_AUDIO_CHANGES
|
|
void Core::initAudioTrack(const Tracks & audios) {
|
|
qDebug("Core::initAudioTrack");
|
|
|
|
qDebug("Core::initAudioTrack: num_items: %d", mdat.audios.numItems());
|
|
|
|
bool restore_audio = ((mdat.audios.numItems() > 0) ||
|
|
(mset.current_audio_id != MediaSettings::NoneSelected));
|
|
|
|
mdat.audios = audios;
|
|
|
|
qDebug("Core::initAudioTrack: list of audios:");
|
|
mdat.audios.list();
|
|
|
|
initializeMenus();
|
|
|
|
if (!restore_audio) {
|
|
// Select initial track
|
|
qDebug("Core::initAudioTrack: selecting initial track");
|
|
|
|
int audio = mdat.audios.itemAt(0).ID(); // First one
|
|
if (mdat.audios.existsItemAt(pref->initial_audio_track-1)) {
|
|
audio = mdat.audios.itemAt(pref->initial_audio_track-1).ID();
|
|
}
|
|
|
|
// Check if one of the audio tracks is the user preferred.
|
|
if (!pref->audio_lang.isEmpty()) {
|
|
int res = mdat.audios.findLang( pref->audio_lang );
|
|
if (res != -1) audio = res;
|
|
}
|
|
|
|
changeAudio( audio );
|
|
} else {
|
|
// Try to restore previous audio track
|
|
qDebug("Core::initAudioTrack: restoring audio");
|
|
// Nothing to do, the audio is already set with -aid
|
|
}
|
|
|
|
updateWidgets();
|
|
|
|
emit audioTracksChanged();
|
|
}
|
|
#endif
|
|
|
|
#if NOTIFY_SUB_CHANGES
|
|
void Core::initSubtitleTrack(const SubTracks & subs) {
|
|
qDebug("Core::initSubtitleTrack");
|
|
|
|
qDebug("Core::initSubtitleTrack: num_items: %d", mdat.subs.numItems());
|
|
|
|
bool restore_subs = ((mdat.subs.numItems() > 0) ||
|
|
(mset.current_sub_id != MediaSettings::NoneSelected));
|
|
|
|
// Save current sub
|
|
SubData::Type previous_sub_type = SubData::Sub;
|
|
int previous_sub_id = -1;
|
|
if (mdat.subs.numItems() > 0) {
|
|
if ((mset.current_sub_id != MediaSettings::SubNone) &&
|
|
(mset.current_sub_id != MediaSettings::NoneSelected))
|
|
{
|
|
previous_sub_type = mdat.subs.itemAt(mset.current_sub_id).type();
|
|
previous_sub_id = mdat.subs.itemAt(mset.current_sub_id).ID();
|
|
}
|
|
}
|
|
qDebug("Core::initSubtitleTrack: previous subtitle: type: %d id: %d", previous_sub_type, previous_sub_id);
|
|
|
|
mdat.subs = subs;
|
|
|
|
qDebug("Core::initSubtitleTrack: list of subtitles:");
|
|
mdat.subs.list();
|
|
|
|
initializeMenus();
|
|
|
|
if (just_unloaded_external_subs) {
|
|
qDebug("Core::initSubtitleTrack: just_unloaded_external_subs: true");
|
|
restore_subs = false;
|
|
just_unloaded_external_subs = false;
|
|
}
|
|
if (just_loaded_external_subs) {
|
|
qDebug("Core::initSubtitleTrack: just_loaded_external_subs: true");
|
|
restore_subs = false;
|
|
just_loaded_external_subs = false;
|
|
}
|
|
|
|
if (!restore_subs) {
|
|
// Select initial track
|
|
qDebug("Core::initSubtitleTrack: selecting initial track");
|
|
|
|
if (!pref->autoload_sub) {
|
|
changeSubtitle( MediaSettings::SubNone );
|
|
} else {
|
|
//Select first subtitle
|
|
int sub = mdat.subs.selectOne( pref->subtitle_lang, pref->initial_subtitle_track-1 );
|
|
changeSubtitle( sub );
|
|
}
|
|
} else {
|
|
// Try to restore previous subtitle track
|
|
qDebug("Core::initSubtitleTrack: restoring subtitle");
|
|
|
|
if (mset.current_sub_id == MediaSettings::SubNone) {
|
|
changeSubtitle( MediaSettings::SubNone );
|
|
}
|
|
else
|
|
if (mset.current_sub_id != MediaSettings::NoneSelected) {
|
|
// Try to find old subtitle
|
|
int item = mset.current_sub_id;
|
|
if (previous_sub_id != -1) {
|
|
int sub_item = mdat.subs.find(previous_sub_type, previous_sub_id);
|
|
if (sub_item > -1) {
|
|
item = sub_item;
|
|
qDebug("Core::initSubtitleTrack: previous subtitle found: %d", sub_item);
|
|
}
|
|
}
|
|
if (item > -1) {
|
|
changeSubtitle(item );
|
|
} else {
|
|
qDebug("Core::initSubtitleTrack: previous subtitle not found!");
|
|
}
|
|
}
|
|
}
|
|
|
|
updateWidgets();
|
|
}
|
|
|
|
void Core::setSubtitleTrackAgain(const SubTracks &) {
|
|
qDebug("Core::setSubtitleTrackAgain");
|
|
changeSubtitle( mset.current_sub_id );
|
|
}
|
|
#endif
|
|
|
|
QString Core::pausing_prefix() {
|
|
qDebug("Core::pausing_prefix");
|
|
|
|
if ( (pref->use_pausing_keep_force) &&
|
|
(MplayerVersion::isMplayerAtLeast(27665)) )
|
|
{
|
|
return "pausing_keep_force";
|
|
} else {
|
|
return "pausing_keep";
|
|
}
|
|
}
|
|
|
|
#include "moc_core.cpp"
|