3f677ef439
All credit goes to FRtranslator.
2865 lines
102 KiB
C++
2865 lines
102 KiB
C++
#include "tsmuxerwindow.h"
|
|
|
|
#include <QColorDialog>
|
|
#include <QDesktopServices>
|
|
#include <QDir>
|
|
#include <QDropEvent>
|
|
#include <QFileDialog>
|
|
#include <QFontDialog>
|
|
#include <QLibraryInfo>
|
|
#include <QMessageBox>
|
|
#include <QMimeData>
|
|
#include <QSettings>
|
|
#include <QSound>
|
|
#include <QTemporaryFile>
|
|
#include <QTime>
|
|
|
|
#include "checkboxedheaderview.h"
|
|
#include "codecinfo.h"
|
|
#include "lang_codes.h"
|
|
#include "muxForm.h"
|
|
#include "ui_tsmuxerwindow.h"
|
|
|
|
namespace
|
|
{
|
|
QString fileDialogFilter()
|
|
{
|
|
return TsMuxerWindow::tr(
|
|
"All supported media files (*.aac *.mpv *.mpa *.avc *.mvc *.264 *.h264 *.ac3 *.dts *.dtshd *.ts *.m2ts *.mts *.ssif *.mpg *.mpeg *.vob *.evo *.mkv *.mka *.mks *.mp4 *.m4a *.m4v *.mov *.sup *.wav *.w64 *.pcm *.m1v *.m2v *.vc1 *.hevc *.hvc *.265 *.h265 *.mpls *.mpl *.srt);;\
|
|
AC3/E-AC3 (*.ac3 *.ddp);;\
|
|
AAC (advanced audio coding) (*.aac);;\
|
|
AVC/MVC/H.264 elementary stream (*.avc *.mvc *.264 *.h264);;\
|
|
HEVC (High Efficiency Video Codec) (*.hevc *.hvc *.265 *.h265);;\
|
|
Digital Theater System (*.dts);;\
|
|
DTS-HD Master Audio (*.dtshd);;\
|
|
Mpeg video elementary stream (*.mpv *.m1v *.m2v);;\
|
|
Mpeg audio elementary stream (*.mpa);;\
|
|
Transport Stream (*.ts);;\
|
|
BDAV Transport Stream (*.m2ts *.mts *.ssif);;\
|
|
Program Stream (*.mpg *.mpeg *.vob *.evo);;\
|
|
Matroska audio/video files (*.mkv *.mka *.mks);;\
|
|
MP4 audio/video files (*.mp4 *.m4a *.m4v);;\
|
|
Quick time audio/video files (*.mov);;\
|
|
Blu-ray play list (*.mpls *.mpl);;\
|
|
Blu-ray PGS subtitles (*.sup);;\
|
|
Text subtitles (*.srt);;\
|
|
WAVE - Uncompressed PCM audio (*.wav *.w64);;\
|
|
RAW LPCM Stream (*.pcm);;\
|
|
All files (*.*)");
|
|
}
|
|
|
|
QString TI_DEFAULT_TAB_NAME() { return TsMuxerWindow::tr("General track options"); }
|
|
|
|
QString TI_DEMUX_TAB_NAME() { return TsMuxerWindow::tr("Demux options"); }
|
|
|
|
QString TS_SAVE_DIALOG_FILTER() { return TsMuxerWindow::tr("Transport stream (*.ts);;all files (*.*)"); }
|
|
|
|
QString M2TS_SAVE_DIALOG_FILTER() { return TsMuxerWindow::tr("BDAV Transport Stream (*.m2ts);;all files (*.*)"); }
|
|
|
|
QString ISO_SAVE_DIALOG_FILTER() { return TsMuxerWindow::tr("Disk image (*.iso);;all files (*.*)"); }
|
|
|
|
QSettings *settings = nullptr;
|
|
|
|
enum FileCustomData
|
|
{
|
|
MplsItemRole = Qt::UserRole,
|
|
FileNameRole,
|
|
ChaptersRole,
|
|
FileDurationRole
|
|
};
|
|
|
|
static const QString FILE_JOIN_PREFIX(" ++ ");
|
|
|
|
bool doubleCompare(double a, double b) { return qAbs(a - b) < 1e-6; }
|
|
|
|
QString closeDirPath(const QString &src)
|
|
{
|
|
if (src.isEmpty())
|
|
return src;
|
|
if (src[src.length() - 1] == '/' || src[src.length() - 1] == '\\')
|
|
return src;
|
|
return src + QDir::separator();
|
|
}
|
|
|
|
QString unquoteStr(QString val)
|
|
{
|
|
val = val.trimmed();
|
|
if (val.isEmpty())
|
|
return val;
|
|
if (val.at(0) == '\"')
|
|
{
|
|
if (val.right(1) == "\"")
|
|
return val.mid(1, val.length() - 2);
|
|
else
|
|
return val.mid(1, val.length() - 1);
|
|
}
|
|
else
|
|
{
|
|
if (val.right(1) == "\"")
|
|
return val.mid(0, val.length() - 1);
|
|
else
|
|
return val;
|
|
}
|
|
}
|
|
|
|
bool isVideoCodec(const QString &displayName)
|
|
{
|
|
return displayName == "H.264" || displayName == "MVC" || displayName == "VC-1" || displayName == "MPEG-2" ||
|
|
displayName == "HEVC";
|
|
}
|
|
|
|
QString floatToTime(double time, char msSeparator = '.')
|
|
{
|
|
int iTime = (int)time;
|
|
int hour = iTime / 3600;
|
|
iTime -= hour * 3600;
|
|
int min = iTime / 60;
|
|
iTime -= min * 60;
|
|
int sec = iTime;
|
|
int msec = (int)((time - (int)time) * 1000.0);
|
|
QString str;
|
|
str += QString::number(hour).rightJustified(2, '0');
|
|
str += ':';
|
|
str += QString::number(min).rightJustified(2, '0');
|
|
str += ':';
|
|
str += QString::number(sec).rightJustified(2, '0');
|
|
str += msSeparator;
|
|
str += QString::number(msec).rightJustified(3, '0');
|
|
return str;
|
|
}
|
|
|
|
QTime qTimeFromFloat(double secondsF)
|
|
{
|
|
int seconds = (int)secondsF;
|
|
int ms = (secondsF - seconds) * 1000.0;
|
|
int hours = seconds / 3600;
|
|
seconds -= hours * 3600;
|
|
int minute = seconds / 60;
|
|
seconds -= minute * 60;
|
|
return QTime(hours, minute, seconds, ms);
|
|
}
|
|
|
|
int qTimeToMsec(const QTime &time)
|
|
{
|
|
return (time.hour() * 3600 + time.minute() * 60 + time.second()) * 1000 + time.msec();
|
|
}
|
|
|
|
double timeToFloat(const QString &chapterStr)
|
|
{
|
|
if (chapterStr.size() == 0)
|
|
return 0;
|
|
QStringList timeParts = chapterStr.split(':');
|
|
double sec = 0;
|
|
if (timeParts.size() > 0)
|
|
sec = timeParts[timeParts.size() - 1].toDouble();
|
|
int min = 0;
|
|
if (timeParts.size() > 1)
|
|
min = timeParts[timeParts.size() - 2].toInt();
|
|
int hour = 0;
|
|
if (timeParts.size() > 2)
|
|
hour = timeParts[timeParts.size() - 3].toInt();
|
|
return hour * 3600 + min * 60 + sec;
|
|
}
|
|
|
|
QString changeFileExt(const QString &value, const QString &newExt)
|
|
{
|
|
QFileInfo fi(unquoteStr(value));
|
|
if (fi.suffix().length() > 0 || (!value.isEmpty() && value.right(1) == "."))
|
|
return unquoteStr(value.left(value.length() - fi.suffix().length()) + newExt);
|
|
else
|
|
return unquoteStr(value) + "." + newExt;
|
|
}
|
|
|
|
QString fpsTextToFpsStr(const QString &fpsText)
|
|
{
|
|
int p = fpsText.indexOf('/');
|
|
if (p >= 0)
|
|
{
|
|
auto left = fpsText.mid(0, p).toFloat();
|
|
auto right = fpsText.mid(p + 1).toFloat();
|
|
return QString::number(left / right, 'f', 3);
|
|
}
|
|
else
|
|
return fpsText;
|
|
}
|
|
|
|
float extractFloatFromDescr(const QString &str, const QString &pattern)
|
|
{
|
|
try
|
|
{
|
|
int p = 0;
|
|
if (!pattern.isEmpty())
|
|
p = str.indexOf(pattern);
|
|
if (p >= 0)
|
|
{
|
|
p += pattern.length();
|
|
int p2 = p;
|
|
while (p2 < str.length() &&
|
|
((str.at(p2) >= '0' && str.at(p2) <= '9') || str.at(p2) == '.' || str.at(p2) == '-'))
|
|
p2++;
|
|
return str.mid(p, p2 - p).toFloat();
|
|
}
|
|
}
|
|
catch (...)
|
|
{
|
|
return 0;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
QString quoteStr(const QString &val)
|
|
{
|
|
QString rez;
|
|
if (val.isEmpty())
|
|
return "";
|
|
if (val.at(0) != '\"')
|
|
rez = '\"' + val;
|
|
else
|
|
rez = val;
|
|
if (val.at(val.length() - 1) != '\"')
|
|
rez += '\"';
|
|
return rez;
|
|
}
|
|
|
|
QString myUnquoteStr(const QString &val) { return unquoteStr(val); }
|
|
|
|
QString getComboBoxTrackText(int idx, const QtvCodecInfo &codecInfo)
|
|
{
|
|
auto text = QString("[%1] %2").arg(idx + 1).arg(codecInfo.displayName);
|
|
if (!codecInfo.lang.isEmpty())
|
|
{
|
|
text.append(", lang : ");
|
|
text.append(codecInfo.lang);
|
|
}
|
|
text.append(", ");
|
|
text.append(codecInfo.descr);
|
|
return text;
|
|
}
|
|
|
|
void initLanguageComboBox(QComboBox *comboBox)
|
|
{
|
|
comboBox->addItem("English", "en"); // 0th index is also used as default if the language isn't set in the settings.
|
|
comboBox->addItem(QString::fromUtf8("Русский"), "ru");
|
|
comboBox->addItem(QString::fromUtf8("Français"), "fr");
|
|
comboBox->setCurrentIndex(-1); // makes sure currentIndexChanged() is emitted when reading settings.
|
|
}
|
|
|
|
} // namespace
|
|
|
|
// ----------------------- TsMuxerWindow -------------------------------------
|
|
|
|
QString TsMuxerWindow::getOutputDir() const
|
|
{
|
|
QString result = ui->radioButtonOutoutInInput->isChecked() ? lastSourceDir : lastOutputDir;
|
|
if (!result.isEmpty())
|
|
result = QDir::toNativeSeparators(closeDirPath(result));
|
|
return result;
|
|
}
|
|
|
|
QString TsMuxerWindow::getDefaultOutputFileName() const
|
|
{
|
|
QString prefix = getOutputDir();
|
|
|
|
if (ui->radioButtonTS->isChecked())
|
|
return prefix + QString("default.ts");
|
|
else if (ui->radioButtonM2TS->isChecked())
|
|
return prefix + QString("default.m2ts");
|
|
else if (ui->radioButtonBluRayISO->isChecked())
|
|
return prefix + QString("default.iso");
|
|
else
|
|
return prefix;
|
|
}
|
|
|
|
TsMuxerWindow::TsMuxerWindow()
|
|
: ui(new Ui::TsMuxerWindow),
|
|
disableUpdatesCnt(0),
|
|
outFileNameModified(false),
|
|
outFileNameDisableChange(false),
|
|
muxForm(new MuxForm(this)),
|
|
m_updateMeta(true),
|
|
m_3dMode(false)
|
|
{
|
|
ui->setupUi(this);
|
|
setUiMetaItemsData();
|
|
qApp->installTranslator(&qtCoreTranslator);
|
|
qApp->installTranslator(&tsMuxerTranslator);
|
|
initLanguageComboBox(ui->languageSelectComboBox);
|
|
setWindowTitle("tsMuxeR GUI " TSMUXER_VERSION);
|
|
lastInputDir = QDir::homePath();
|
|
lastOutputDir = QDir::homePath();
|
|
|
|
void (QComboBox::*comboBoxIndexChanged)(int) = &QComboBox::currentIndexChanged;
|
|
connect(ui->languageSelectComboBox, comboBoxIndexChanged, this, &TsMuxerWindow::onLanguageComboBoxIndexChanged);
|
|
|
|
QString path = QFileInfo(QApplication::arguments()[0]).absolutePath();
|
|
QString iniName = QDir::toNativeSeparators(path) + QDir::separator() + QString("tsMuxerGUI.ini");
|
|
|
|
settings = new QSettings(QSettings::UserScope, "Network Optix", "tsMuxeR");
|
|
readSettings();
|
|
|
|
if (QFile::exists(iniName))
|
|
{
|
|
delete settings;
|
|
settings = new QSettings(iniName, QSettings::IniFormat);
|
|
settings->setIniCodec("UTF-8");
|
|
if (!readSettings())
|
|
writeSettings(); // copy current registry settings to the ini file
|
|
}
|
|
|
|
ui->outFileName->setText(getDefaultOutputFileName());
|
|
|
|
m_header = new QnCheckBoxedHeaderView(this);
|
|
ui->trackLV->setHorizontalHeader(m_header);
|
|
ui->trackLV->horizontalHeader()->setStretchLastSection(true);
|
|
// ui->trackLV->model()->setHeaderData(0, Qt::Vertical, "#");
|
|
m_header->setVisible(true);
|
|
m_header->setSectionsClickable(true);
|
|
|
|
connect(m_header, &QnCheckBoxedHeaderView::checkStateChanged, this, &TsMuxerWindow::at_sectionCheckstateChanged);
|
|
|
|
/////////////////////////////////////////////////////////////
|
|
for (int i = 0; i <= 3600; i += 5 * 60) ui->memoChapters->insertPlainText(floatToTime(i, '.') + '\n');
|
|
|
|
mSaveDialogFilter = TS_SAVE_DIALOG_FILTER();
|
|
const static int colWidths[] = {28, 200, 62, 38, 10};
|
|
for (unsigned i = 0u; i < sizeof(colWidths) / sizeof(int); ++i)
|
|
ui->trackLV->horizontalHeader()->resizeSection(i, colWidths[i]);
|
|
ui->trackLV->setWordWrap(false);
|
|
|
|
ui->listViewFont->horizontalHeader()->resizeSection(0, 65);
|
|
ui->listViewFont->horizontalHeader()->resizeSection(1, 185);
|
|
for (int i = 0; i < ui->listViewFont->rowCount(); ++i)
|
|
{
|
|
ui->listViewFont->setRowHeight(i, 16);
|
|
ui->listViewFont->item(i, 0)->setFlags(ui->listViewFont->item(i, 0)->flags() & (~Qt::ItemIsEditable));
|
|
ui->listViewFont->item(i, 1)->setFlags(ui->listViewFont->item(i, 0)->flags() & (~Qt::ItemIsEditable));
|
|
}
|
|
void (QSpinBox::*spinBoxValueChanged)(int) = &QSpinBox::valueChanged;
|
|
void (QDoubleSpinBox::*doubleSpinBoxValueChanged)(double) = &QDoubleSpinBox::valueChanged;
|
|
connect(&opacityTimer, &QTimer::timeout, this, &TsMuxerWindow::onOpacityTimer);
|
|
connect(ui->trackLV, &QTableWidget::itemSelectionChanged, this, &TsMuxerWindow::trackLVItemSelectionChanged);
|
|
connect(ui->trackLV, &QTableWidget::itemChanged, this, &TsMuxerWindow::trackLVItemChanged);
|
|
connect(ui->inputFilesLV, &QListWidget::currentRowChanged, this, &TsMuxerWindow::inputFilesLVChanged);
|
|
connect(ui->addBtn, &QPushButton::clicked, this, &TsMuxerWindow::onAddBtnClick);
|
|
connect(ui->btnAppend, &QPushButton::clicked, this, &TsMuxerWindow::onAppendButtonClick);
|
|
connect(ui->removeFile, &QPushButton::clicked, this, &TsMuxerWindow::onRemoveBtnClick);
|
|
connect(ui->removeTrackBtn, &QPushButton::clicked, this, &TsMuxerWindow::onRemoveTrackButtonClick);
|
|
connect(ui->moveupBtn, &QPushButton::clicked, this, &TsMuxerWindow::onMoveUpButtonCLick);
|
|
connect(ui->movedownBtn, &QPushButton::clicked, this, &TsMuxerWindow::onMoveDownButtonCLick);
|
|
connect(ui->checkFPS, &QCheckBox::stateChanged, this, &TsMuxerWindow::onVideoCheckBoxChanged);
|
|
connect(ui->checkBoxLevel, &QCheckBox::stateChanged, this, &TsMuxerWindow::onVideoCheckBoxChanged);
|
|
connect(ui->comboBoxSEI, comboBoxIndexChanged, this, &TsMuxerWindow::onVideoCheckBoxChanged);
|
|
connect(ui->checkBoxSecondaryVideo, &QCheckBox::stateChanged, this, &TsMuxerWindow::onVideoCheckBoxChanged);
|
|
connect(ui->checkBoxSPS, &QCheckBox::stateChanged, this, &TsMuxerWindow::onVideoCheckBoxChanged);
|
|
connect(ui->checkBoxRemovePulldown, &QCheckBox::stateChanged, this, &TsMuxerWindow::onPulldownCheckBoxChanged);
|
|
connect(ui->comboBoxFPS, comboBoxIndexChanged, this, &TsMuxerWindow::onVideoComboBoxChanged);
|
|
connect(ui->comboBoxLevel, comboBoxIndexChanged, this, &TsMuxerWindow::onVideoComboBoxChanged);
|
|
connect(ui->comboBoxAR, comboBoxIndexChanged, this, &TsMuxerWindow::onVideoComboBoxChanged);
|
|
connect(ui->checkBoxKeepFps, &QCheckBox::stateChanged, this, &TsMuxerWindow::onAudioSubtitlesParamsChanged);
|
|
connect(ui->dtsDwnConvert, &QCheckBox::stateChanged, this, &TsMuxerWindow::onAudioSubtitlesParamsChanged);
|
|
connect(ui->secondaryCheckBox, &QCheckBox::stateChanged, this, &TsMuxerWindow::onAudioSubtitlesParamsChanged);
|
|
connect(ui->langComboBox, comboBoxIndexChanged, this, &TsMuxerWindow::onAudioSubtitlesParamsChanged);
|
|
connect(ui->offsetsComboBox, comboBoxIndexChanged, this, &TsMuxerWindow::onAudioSubtitlesParamsChanged);
|
|
connect(ui->comboBoxPipCorner, comboBoxIndexChanged, this, &TsMuxerWindow::onSavedParamChanged);
|
|
connect(ui->comboBoxPipSize, comboBoxIndexChanged, this, &TsMuxerWindow::onSavedParamChanged);
|
|
connect(ui->spinBoxPipOffsetH, spinBoxValueChanged, this, &TsMuxerWindow::onSavedParamChanged);
|
|
connect(ui->spinBoxPipOffsetV, spinBoxValueChanged, this, &TsMuxerWindow::onSavedParamChanged);
|
|
connect(ui->checkBoxSound, &QCheckBox::stateChanged, this, &TsMuxerWindow::onSavedParamChanged);
|
|
connect(ui->editDelay, spinBoxValueChanged, this, &TsMuxerWindow::onEditDelayChanged);
|
|
connect(ui->muxTimeEdit, &QTimeEdit::timeChanged, this, &TsMuxerWindow::updateMuxTime1);
|
|
connect(ui->muxTimeClock, spinBoxValueChanged, this, &TsMuxerWindow::updateMuxTime2);
|
|
connect(ui->fontButton, &QPushButton::clicked, this, &TsMuxerWindow::onFontBtnClicked);
|
|
connect(ui->colorButton, &QPushButton::clicked, this, &TsMuxerWindow::onColorBtnClicked);
|
|
connect(ui->checkBoxVBR, &QPushButton::clicked, this, &TsMuxerWindow::onGeneralCheckboxClicked);
|
|
connect(ui->spinBoxMplsNum, spinBoxValueChanged, this, &TsMuxerWindow::onGeneralCheckboxClicked);
|
|
connect(ui->spinBoxM2tsNum, spinBoxValueChanged, this, &TsMuxerWindow::onGeneralCheckboxClicked);
|
|
connect(ui->checkBoxBlankPL, &QPushButton::clicked, this, &TsMuxerWindow::onSavedParamChanged);
|
|
connect(ui->checkBoxV3, &QCheckBox::stateChanged, this, &TsMuxerWindow::updateMetaLines);
|
|
connect(ui->BlackplaylistCombo, spinBoxValueChanged, this, &TsMuxerWindow::onSavedParamChanged);
|
|
connect(ui->checkBoxNewAudioPes, &QAbstractButton::clicked, this, &TsMuxerWindow::onSavedParamChanged);
|
|
connect(ui->checkBoxCrop, &QCheckBox::stateChanged, this, &TsMuxerWindow::onSavedParamChanged);
|
|
connect(ui->checkBoxRVBR, &QAbstractButton::clicked, this, &TsMuxerWindow::onGeneralCheckboxClicked);
|
|
connect(ui->checkBoxCBR, &QAbstractButton::clicked, this, &TsMuxerWindow::onGeneralCheckboxClicked);
|
|
connect(ui->radioButtonStoreOutput, &QAbstractButton::clicked, this, &TsMuxerWindow::onSavedParamChanged);
|
|
connect(ui->radioButtonOutoutInInput, &QAbstractButton::clicked, this, &TsMuxerWindow::onSavedParamChanged);
|
|
connect(ui->editVBVLen, spinBoxValueChanged, this, &TsMuxerWindow::onGeneralSpinboxValueChanged);
|
|
connect(ui->editMaxBitrate, doubleSpinBoxValueChanged, this, &TsMuxerWindow::onGeneralSpinboxValueChanged);
|
|
connect(ui->editMinBitrate, doubleSpinBoxValueChanged, this, &TsMuxerWindow::onGeneralSpinboxValueChanged);
|
|
connect(ui->editCBRBitrate, doubleSpinBoxValueChanged, this, &TsMuxerWindow::onGeneralSpinboxValueChanged);
|
|
connect(ui->rightEyeCheckBox, &QCheckBox::stateChanged, this, &TsMuxerWindow::updateMetaLines);
|
|
connect(ui->radioButtonAutoChapter, &QAbstractButton::clicked, this, &TsMuxerWindow::onChapterParamsChanged);
|
|
connect(ui->radioButtonNoChapters, &QAbstractButton::clicked, this, &TsMuxerWindow::onChapterParamsChanged);
|
|
connect(ui->radioButtonCustomChapters, &QAbstractButton::clicked, this, &TsMuxerWindow::onChapterParamsChanged);
|
|
connect(ui->spinEditChapterLen, spinBoxValueChanged, this, &TsMuxerWindow::onChapterParamsChanged);
|
|
connect(ui->memoChapters, &QPlainTextEdit::textChanged, this, &TsMuxerWindow::onChapterParamsChanged);
|
|
connect(ui->noSplit, &QAbstractButton::clicked, this, &TsMuxerWindow::onSplitCutParamsChanged);
|
|
connect(ui->splitByDuration, &QAbstractButton::clicked, this, &TsMuxerWindow::onSplitCutParamsChanged);
|
|
connect(ui->splitBySize, &QAbstractButton::clicked, this, &TsMuxerWindow::onSplitCutParamsChanged);
|
|
connect(ui->spinEditSplitDuration, spinBoxValueChanged, this, &TsMuxerWindow::onSplitCutParamsChanged);
|
|
connect(ui->editSplitSize, doubleSpinBoxValueChanged, this, &TsMuxerWindow::onSplitCutParamsChanged);
|
|
connect(ui->comboBoxMeasure, comboBoxIndexChanged, this, &TsMuxerWindow::onSplitCutParamsChanged);
|
|
connect(ui->checkBoxCut, &QCheckBox::stateChanged, this, &TsMuxerWindow::onSplitCutParamsChanged);
|
|
connect(ui->cutStartTimeEdit, &QTimeEdit::timeChanged, this, &TsMuxerWindow::onSplitCutParamsChanged);
|
|
connect(ui->cutEndTimeEdit, &QTimeEdit::timeChanged, this, &TsMuxerWindow::onSplitCutParamsChanged);
|
|
connect(ui->spinEditBorder, spinBoxValueChanged, this, &TsMuxerWindow::onSavedParamChanged);
|
|
connect(ui->comboBoxAnimation, comboBoxIndexChanged, this, &TsMuxerWindow::onSavedParamChanged);
|
|
connect(ui->lineSpacing, doubleSpinBoxValueChanged, this, &TsMuxerWindow::onSavedParamChanged);
|
|
connect(ui->spinEditOffset, spinBoxValueChanged, this, &TsMuxerWindow::onSavedParamChanged);
|
|
connect(ui->radioButtonTS, &QAbstractButton::clicked, this, &TsMuxerWindow::RadioButtonMuxClick);
|
|
connect(ui->radioButtonM2TS, &QAbstractButton::clicked, this, &TsMuxerWindow::RadioButtonMuxClick);
|
|
connect(ui->radioButtonBluRay, &QAbstractButton::clicked, this, &TsMuxerWindow::RadioButtonMuxClick);
|
|
connect(ui->radioButtonBluRayISO, &QAbstractButton::clicked, this, &TsMuxerWindow::RadioButtonMuxClick);
|
|
connect(ui->radioButtonAVCHD, &QAbstractButton::clicked, this, &TsMuxerWindow::RadioButtonMuxClick);
|
|
connect(ui->radioButtonDemux, &QAbstractButton::clicked, this, &TsMuxerWindow::RadioButtonMuxClick);
|
|
connect(ui->outFileName, &QLineEdit::textChanged, this, &TsMuxerWindow::outFileNameChanged);
|
|
connect(ui->DiskLabelEdit, &QLineEdit::textChanged, this, &TsMuxerWindow::onGeneralCheckboxClicked);
|
|
connect(ui->btnBrowse, &QAbstractButton::clicked, this, &TsMuxerWindow::saveFileDialog);
|
|
connect(ui->buttonMux, &QAbstractButton::clicked, this, &TsMuxerWindow::startMuxing);
|
|
connect(ui->buttonSaveMeta, &QAbstractButton::clicked, this, &TsMuxerWindow::saveMetaFileBtnClick);
|
|
connect(ui->radioButtonOutoutInInput, &QAbstractButton::clicked, this, &TsMuxerWindow::onSavedParamChanged);
|
|
connect(ui->defaultAudioTrackComboBox, comboBoxIndexChanged, this, &TsMuxerWindow::updateMetaLines);
|
|
connect(ui->defaultSubTrackComboBox, comboBoxIndexChanged, this, &TsMuxerWindow::updateMetaLines);
|
|
connect(ui->defaultAudioTrackCheckBox, &QCheckBox::stateChanged, this, &TsMuxerWindow::updateMetaLines);
|
|
connect(ui->defaultSubTrackCheckBox, &QCheckBox::stateChanged, this, &TsMuxerWindow::updateMetaLines);
|
|
connect(ui->defaultSubTrackForcedOnlyCheckBox, &QCheckBox::stateChanged, this, &TsMuxerWindow::updateMetaLines);
|
|
connect(ui->pipTransparencySpinBox, spinBoxValueChanged, this, &TsMuxerWindow::updateMetaLines);
|
|
|
|
connect(&proc, &QProcess::readyReadStandardOutput, this, &TsMuxerWindow::readFromStdout);
|
|
connect(&proc, &QProcess::readyReadStandardError, this, &TsMuxerWindow::readFromStderr);
|
|
void (QProcess::*processFinished)(int, QProcess::ExitStatus) = &QProcess::finished;
|
|
connect(&proc, processFinished, this, &TsMuxerWindow::onProcessFinished);
|
|
void (QProcess::*processError)(QProcess::ProcessError) = &QProcess::error;
|
|
connect(&proc, processError, this, &TsMuxerWindow::onProcessError);
|
|
|
|
ui->DiskLabel->setVisible(false);
|
|
ui->DiskLabelEdit->setVisible(false);
|
|
|
|
ui->label_Donate->installEventFilter(this);
|
|
|
|
ui->langComboBox->addItem("und (Undetermined)");
|
|
ui->langComboBox->addItem("--------- common ---------");
|
|
for (auto &&lang : shortLangList)
|
|
{
|
|
ui->langComboBox->addItem(QString("%1 (%2)").arg(lang.code).arg(lang.lang), QString::fromUtf8(lang.code));
|
|
}
|
|
ui->langComboBox->addItem("---------- all ----------");
|
|
for (auto &&lang : fullLangList)
|
|
{
|
|
ui->langComboBox->addItem(QString("%1 (%2)").arg(lang.code).arg(lang.lang), QString::fromUtf8(lang.code));
|
|
}
|
|
trackLVItemSelectionChanged();
|
|
|
|
ui->trackSplitter->setStretchFactor(0, 10);
|
|
ui->trackSplitter->setStretchFactor(1, 100);
|
|
|
|
writeSettings();
|
|
}
|
|
|
|
TsMuxerWindow::~TsMuxerWindow() { writeSettings(); }
|
|
|
|
void TsMuxerWindow::onTsMuxerCodecInfoReceived()
|
|
{
|
|
m_updateMeta = false;
|
|
codecList.clear();
|
|
int p;
|
|
QtvCodecInfo *codecInfo = 0;
|
|
int lastTrackID = 0;
|
|
QString tmpStr;
|
|
bool firstMark = true;
|
|
codecList.clear();
|
|
mplsFileList.clear();
|
|
chapters.clear();
|
|
fileDuration = 0;
|
|
for (int i = 0; i < procStdOutput.size(); ++i)
|
|
{
|
|
p = procStdOutput[i].indexOf("Track ID: ");
|
|
if (p >= 0)
|
|
{
|
|
lastTrackID = procStdOutput[i].mid(QString("Track ID: ").length()).toInt();
|
|
codecList << QtvCodecInfo();
|
|
codecInfo = &(codecList.back());
|
|
codecInfo->trackID = lastTrackID;
|
|
}
|
|
|
|
p = procStdOutput[i].indexOf("Stream type: ");
|
|
if (p >= 0)
|
|
{
|
|
if (lastTrackID == 0)
|
|
{
|
|
codecList << QtvCodecInfo();
|
|
codecInfo = &(codecList.back());
|
|
}
|
|
codecInfo->descr = "Can't detect codec";
|
|
codecInfo->displayName = procStdOutput[i].mid(QString("Stream type: ").length());
|
|
if (codecInfo->displayName != "H.264" && codecInfo->displayName != "MVC")
|
|
{
|
|
codecInfo->addSEIMethod = 0;
|
|
codecInfo->addSPS = false;
|
|
}
|
|
if (codecInfo->displayName == "HEVC" && !ui->checkBoxV3->isChecked())
|
|
ui->checkBoxV3->setChecked(true);
|
|
lastTrackID = 0;
|
|
}
|
|
p = procStdOutput[i].indexOf("Stream ID: ");
|
|
if (p >= 0)
|
|
codecInfo->programName = procStdOutput[i].mid(QString("Stream ID: ").length());
|
|
p = procStdOutput[i].indexOf("Stream info: ");
|
|
if (p >= 0)
|
|
codecInfo->descr = procStdOutput[i].mid(QString("Stream info: ").length());
|
|
p = procStdOutput[i].indexOf("Stream lang: ");
|
|
if (p >= 0)
|
|
codecInfo->lang = procStdOutput[i].mid(QString("Stream lang: ").length());
|
|
|
|
p = procStdOutput[i].indexOf("Stream delay: ");
|
|
if (p >= 0)
|
|
{
|
|
tmpStr = procStdOutput[i].mid(QString("Stream delay: ").length());
|
|
codecInfo->delay = tmpStr.toInt();
|
|
}
|
|
|
|
p = procStdOutput[i].indexOf("subTrack: ");
|
|
if (p >= 0)
|
|
{
|
|
tmpStr = procStdOutput[i].mid(QString("subTrack: ").length());
|
|
codecInfo->subTrack = tmpStr.toInt();
|
|
}
|
|
|
|
p = procStdOutput[i].indexOf("Secondary: 1");
|
|
if (p == 0)
|
|
{
|
|
codecInfo->isSecondary = true;
|
|
}
|
|
p = procStdOutput[i].indexOf("Unselected: 1");
|
|
if (p == 0)
|
|
{
|
|
codecInfo->enabledByDefault = false;
|
|
}
|
|
|
|
if (procStdOutput[i].startsWith("Error: "))
|
|
{
|
|
tmpStr = procStdOutput[i].mid(QString("Error: ").length());
|
|
QMessageBox msgBox(this);
|
|
msgBox.setWindowTitle(tr("Not supported"));
|
|
msgBox.setText(tmpStr);
|
|
msgBox.setIcon(QMessageBox::Warning);
|
|
msgBox.setStandardButtons(QMessageBox::Ok);
|
|
msgBox.exec();
|
|
}
|
|
else if (procStdOutput[i].startsWith("File #"))
|
|
{
|
|
tmpStr = procStdOutput[i].mid(17); // todo: check here
|
|
mplsFileList << MPLSFileInfo(tmpStr, 0.0);
|
|
}
|
|
else if (procStdOutput[i].startsWith("Duration:"))
|
|
{
|
|
tmpStr = procStdOutput[i].mid(10); // todo: check here
|
|
if (!mplsFileList.empty())
|
|
{
|
|
mplsFileList.last().duration = timeToFloat(tmpStr);
|
|
}
|
|
else
|
|
{
|
|
fileDuration = timeToFloat(tmpStr);
|
|
}
|
|
}
|
|
else if (procStdOutput[i].startsWith("Base view: "))
|
|
{
|
|
tmpStr = procStdOutput[i].mid(11); // todo: check here
|
|
ui->rightEyeCheckBox->setChecked(tmpStr.trimmed() == "right-eye");
|
|
}
|
|
else if (procStdOutput[i].startsWith("start-time: "))
|
|
{
|
|
tmpStr = procStdOutput[i].mid(12);
|
|
if (tmpStr.contains(':'))
|
|
{
|
|
double secondsF = timeToFloat(tmpStr);
|
|
ui->muxTimeEdit->setTime(qTimeFromFloat(secondsF));
|
|
}
|
|
else
|
|
{
|
|
ui->muxTimeClock->setValue(tmpStr.toInt());
|
|
}
|
|
}
|
|
|
|
p = procStdOutput[i].indexOf("Marks: ");
|
|
if (p == 0)
|
|
{
|
|
if (firstMark)
|
|
{
|
|
firstMark = false;
|
|
ui->radioButtonCustomChapters->setChecked(true);
|
|
}
|
|
QStringList stringList = procStdOutput[i].mid(7).split(' ');
|
|
for (int k = 0; k < stringList.size(); ++k)
|
|
if (!stringList[k].isEmpty())
|
|
chapters << timeToFloat(stringList[k]);
|
|
}
|
|
}
|
|
if (fileDuration == 0 && !mplsFileList.isEmpty())
|
|
{
|
|
foreach (const MPLSFileInfo &mplsFile, mplsFileList)
|
|
fileDuration += mplsFile.duration;
|
|
}
|
|
|
|
m_updateMeta = true;
|
|
if (codecList.isEmpty())
|
|
{
|
|
QMessageBox msgBox(this);
|
|
msgBox.setWindowTitle(tr("Unsupported format"));
|
|
msgBox.setText(tr("Can't detect stream type. File name: \"%1\"").arg(newFileName));
|
|
msgBox.setIcon(QMessageBox::Warning);
|
|
msgBox.setStandardButtons(QMessageBox::Ok);
|
|
msgBox.exec();
|
|
return;
|
|
}
|
|
updateMetaLines();
|
|
emit codecListReady();
|
|
}
|
|
|
|
int TsMuxerWindow::findLangByCode(const QString &code)
|
|
{
|
|
QString addr;
|
|
for (int i = 0; i < ui->langComboBox->count(); ++i)
|
|
{
|
|
addr = ui->langComboBox->itemData(i).toString();
|
|
if (!addr.isEmpty() && code == addr)
|
|
{
|
|
return i;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
QtvCodecInfo *TsMuxerWindow::getCodecInfo(int idx)
|
|
{
|
|
auto iCodec = ui->trackLV->item(idx, 0)->data(Qt::UserRole).toLongLong();
|
|
return reinterpret_cast<QtvCodecInfo *>(iCodec);
|
|
}
|
|
|
|
QtvCodecInfo *TsMuxerWindow::getCurrentCodec()
|
|
{
|
|
auto row = ui->trackLV->currentRow();
|
|
if (row == -1)
|
|
return nullptr;
|
|
return getCodecInfo(row);
|
|
}
|
|
|
|
void TsMuxerWindow::onVideoComboBoxChanged(int)
|
|
{
|
|
if (disableUpdatesCnt)
|
|
return;
|
|
QtvCodecInfo *codecInfo = getCurrentCodec();
|
|
if (!codecInfo)
|
|
return;
|
|
codecInfo->fpsText = ui->comboBoxFPS->itemText(ui->comboBoxFPS->currentIndex());
|
|
codecInfo->levelText = ui->comboBoxLevel->itemText(ui->comboBoxLevel->currentIndex());
|
|
codecInfo->arText = ui->comboBoxAR->currentData().toString();
|
|
updateMetaLines();
|
|
}
|
|
|
|
void TsMuxerWindow::onVideoCheckBoxChanged(int state)
|
|
{
|
|
Q_UNUSED(state);
|
|
if (disableUpdatesCnt)
|
|
return;
|
|
QtvCodecInfo *codecInfo = getCurrentCodec();
|
|
if (codecInfo == nullptr)
|
|
return;
|
|
ui->comboBoxFPS->setEnabled(ui->checkFPS->isChecked());
|
|
codecInfo->checkFPS = ui->checkFPS->isChecked();
|
|
ui->comboBoxLevel->setEnabled(ui->checkBoxLevel->isChecked());
|
|
codecInfo->checkLevel = ui->checkBoxLevel->isChecked();
|
|
codecInfo->addSEIMethod = ui->comboBoxSEI->currentIndex();
|
|
codecInfo->addSPS = ui->checkBoxSPS->isChecked();
|
|
codecInfo->isSecondary = ui->checkBoxSecondaryVideo->isChecked();
|
|
colorizeCurrentRow(codecInfo);
|
|
updateMetaLines();
|
|
}
|
|
|
|
void TsMuxerWindow::updateCurrentColor(int dr, int dg, int db, int row)
|
|
{
|
|
if (row == -1)
|
|
row = ui->trackLV->currentRow();
|
|
if (row == -1)
|
|
return;
|
|
QColor color = ui->trackLV->palette().color(QPalette::Base);
|
|
color.setRed(qBound(0, color.red() + dr, 255));
|
|
color.setGreen(qBound(0, color.green() + dg, 255));
|
|
color.setBlue(qBound(0, color.blue() + db, 255));
|
|
for (int i = 0; i < 5; ++i)
|
|
{
|
|
QModelIndex index = ui->trackLV->model()->index(row, i);
|
|
ui->trackLV->model()->setData(index, QBrush(color), Qt::BackgroundColorRole);
|
|
}
|
|
}
|
|
|
|
void TsMuxerWindow::colorizeCurrentRow(const QtvCodecInfo *codecInfo, int rowIndex)
|
|
{
|
|
if (codecInfo->isSecondary)
|
|
updateCurrentColor(-40, 0, 0, rowIndex);
|
|
else
|
|
updateCurrentColor(0, 0, 0, rowIndex);
|
|
}
|
|
|
|
void TsMuxerWindow::addTrackToDefaultComboBox(int trackRowIdx)
|
|
{
|
|
auto codecInfo = getCodecInfo(trackRowIdx);
|
|
auto text = getComboBoxTrackText(trackRowIdx, *codecInfo);
|
|
if (codecInfo->programName.startsWith('A'))
|
|
{
|
|
ui->defaultAudioTrackComboBox->addItem(text, trackRowIdx);
|
|
}
|
|
else if (codecInfo->programName.startsWith('S'))
|
|
{
|
|
ui->defaultSubTrackComboBox->addItem(text, trackRowIdx);
|
|
}
|
|
}
|
|
|
|
void TsMuxerWindow::removeTrackFromDefaultComboBox(QComboBox *targetComboBox, QCheckBox *targetCheckBox,
|
|
int comboBoxIdx, int trackRowIdx)
|
|
{
|
|
if (targetComboBox->currentData().toInt() == trackRowIdx)
|
|
{
|
|
targetCheckBox->setChecked(false);
|
|
}
|
|
for (int i = comboBoxIdx + 1; i < targetComboBox->count(); ++i)
|
|
{
|
|
auto curTrackIdx = targetComboBox->itemData(i).toInt();
|
|
targetComboBox->setItemData(i, curTrackIdx - 1);
|
|
}
|
|
targetComboBox->removeItem(comboBoxIdx);
|
|
updateTracksComboBox(targetComboBox);
|
|
}
|
|
|
|
void TsMuxerWindow::removeTrackFromDefaultComboBox(int trackRowIdx)
|
|
{
|
|
auto comboBoxIdx = ui->defaultAudioTrackComboBox->findData(trackRowIdx);
|
|
if (comboBoxIdx != -1)
|
|
{
|
|
removeTrackFromDefaultComboBox(ui->defaultAudioTrackComboBox, ui->defaultAudioTrackCheckBox, comboBoxIdx,
|
|
trackRowIdx);
|
|
}
|
|
comboBoxIdx = ui->defaultSubTrackComboBox->findData(trackRowIdx);
|
|
if (comboBoxIdx != -1)
|
|
{
|
|
removeTrackFromDefaultComboBox(ui->defaultSubTrackComboBox, ui->defaultSubTrackCheckBox, comboBoxIdx,
|
|
trackRowIdx);
|
|
}
|
|
}
|
|
|
|
void TsMuxerWindow::updateTracksComboBox(QComboBox *comboBox)
|
|
{
|
|
for (int i = 0; i < comboBox->count(); ++i)
|
|
{
|
|
auto trackRowIdx = comboBox->itemData(i).toInt();
|
|
auto codecInfo = getCodecInfo(trackRowIdx);
|
|
comboBox->setItemText(i, getComboBoxTrackText(trackRowIdx, *codecInfo));
|
|
}
|
|
}
|
|
|
|
#include <QDebug>
|
|
|
|
void TsMuxerWindow::moveTrackInDefaultComboBox(int oldTrackRowIdx, int newTrackRowIdx)
|
|
{
|
|
qDebug() << oldTrackRowIdx << newTrackRowIdx;
|
|
auto currentSubTrack = ui->defaultSubTrackComboBox->currentData();
|
|
auto currentAudioTrack = ui->defaultAudioTrackComboBox->currentData();
|
|
ui->defaultSubTrackComboBox->clear();
|
|
ui->defaultAudioTrackComboBox->clear();
|
|
for (int i = 0; i < ui->trackLV->rowCount(); ++i)
|
|
{
|
|
addTrackToDefaultComboBox(i);
|
|
}
|
|
postMoveComboBoxUpdate(ui->defaultAudioTrackComboBox, currentAudioTrack, oldTrackRowIdx, newTrackRowIdx);
|
|
postMoveComboBoxUpdate(ui->defaultSubTrackComboBox, currentSubTrack, oldTrackRowIdx, newTrackRowIdx);
|
|
}
|
|
|
|
void TsMuxerWindow::postMoveComboBoxUpdate(QComboBox *comboBox, const QVariant &preMoveIndex, int oldIndex,
|
|
int newIndex)
|
|
{
|
|
if (!preMoveIndex.isValid())
|
|
{
|
|
return;
|
|
}
|
|
auto curTrackIdx = preMoveIndex.toInt();
|
|
if (curTrackIdx == oldIndex)
|
|
{
|
|
curTrackIdx = newIndex;
|
|
}
|
|
else if (curTrackIdx == newIndex)
|
|
{
|
|
curTrackIdx = oldIndex;
|
|
}
|
|
auto idx = comboBox->findData(curTrackIdx);
|
|
Q_ASSERT(idx != -1);
|
|
comboBox->setCurrentIndex(idx);
|
|
}
|
|
|
|
void TsMuxerWindow::setUiMetaItemsData()
|
|
{
|
|
// unfortunately, the .ui files don't allow the user to specify "item data" for combo boxes, which is the most
|
|
// convenient way to associate some extra data that's not displayed in the UI in a combo box item without having
|
|
// to resort to some kind of external containers which need to be kept synchronised.
|
|
// as some of the combo boxes are taken as input for the meta file, it would end up having translated strings in it
|
|
// if a non-English translation is active, and thus being invalid. item data for these UI items should always
|
|
// contain the valid metafile tokens, as they are the actual things incorporated into the metafile content.
|
|
ui->comboBoxAR->setItemData(0, QString());
|
|
ui->comboBoxAR->setItemData(1, "1:1");
|
|
ui->comboBoxAR->setItemData(2, "4:3");
|
|
ui->comboBoxAR->setItemData(3, "16:9");
|
|
ui->comboBoxAR->setItemData(4, "2.21:1");
|
|
ui->comboBoxMeasure->setItemData(0, "KB");
|
|
ui->comboBoxMeasure->setItemData(1, "KiB");
|
|
ui->comboBoxMeasure->setItemData(2, "MB");
|
|
ui->comboBoxMeasure->setItemData(3, "MiB");
|
|
ui->comboBoxMeasure->setItemData(4, "GB");
|
|
ui->comboBoxMeasure->setItemData(5, "GiB");
|
|
}
|
|
|
|
void TsMuxerWindow::onAudioSubtitlesParamsChanged()
|
|
{
|
|
if (disableUpdatesCnt)
|
|
return;
|
|
QtvCodecInfo *codecInfo = getCurrentCodec();
|
|
if (!codecInfo)
|
|
return;
|
|
codecInfo->bindFps = ui->checkBoxKeepFps->isChecked();
|
|
codecInfo->dtsDownconvert = ui->dtsDwnConvert->isChecked();
|
|
codecInfo->isSecondary = ui->secondaryCheckBox->isChecked();
|
|
codecInfo->offsetId = ui->offsetsComboBox->currentIndex() - 1;
|
|
QString addr = ui->langComboBox->itemData(ui->langComboBox->currentIndex()).toString();
|
|
if (!addr.isEmpty())
|
|
{
|
|
codecInfo->lang = addr;
|
|
}
|
|
else
|
|
{
|
|
codecInfo->lang.clear();
|
|
}
|
|
ui->trackLV->item(ui->trackLV->currentRow(), 3)->setText(codecInfo->lang);
|
|
colorizeCurrentRow(codecInfo);
|
|
|
|
updateMetaLines();
|
|
}
|
|
|
|
void TsMuxerWindow::onEditDelayChanged(int i)
|
|
{
|
|
Q_UNUSED(i);
|
|
if (disableUpdatesCnt)
|
|
return;
|
|
QtvCodecInfo *codecInfo = getCurrentCodec();
|
|
if (!codecInfo)
|
|
return;
|
|
codecInfo->delay = ui->editDelay->value();
|
|
updateMetaLines();
|
|
}
|
|
|
|
void TsMuxerWindow::onPulldownCheckBoxChanged(int state)
|
|
{
|
|
Q_UNUSED(state);
|
|
if (disableUpdatesCnt)
|
|
return;
|
|
QtvCodecInfo *codecInfo = getCurrentCodec();
|
|
if (!codecInfo)
|
|
return;
|
|
if (ui->checkBoxRemovePulldown->isEnabled())
|
|
{
|
|
if (ui->checkBoxRemovePulldown->isChecked())
|
|
{
|
|
codecInfo->delPulldown = 1;
|
|
if (codecInfo->fpsTextOrig == "29.97")
|
|
{
|
|
ui->checkFPS->setChecked(true);
|
|
ui->checkFPS->setEnabled(true);
|
|
codecInfo->checkFPS = true;
|
|
ui->comboBoxFPS->setEnabled(true);
|
|
ui->comboBoxFPS->setCurrentIndex(3);
|
|
codecInfo->fpsText = "24000/1001";
|
|
setComboBoxText(ui->comboBoxFPS, "24000/1001");
|
|
}
|
|
}
|
|
else
|
|
codecInfo->delPulldown = 0;
|
|
}
|
|
else
|
|
codecInfo->delPulldown = -1;
|
|
updateMetaLines();
|
|
}
|
|
|
|
void TsMuxerWindow::addFiles(const QList<QUrl> &files)
|
|
{
|
|
addFileList.clear();
|
|
addFileList = files;
|
|
addFile();
|
|
}
|
|
|
|
void TsMuxerWindow::onAddBtnClick()
|
|
{
|
|
QString fileName = QDir::toNativeSeparators(
|
|
QFileDialog::getOpenFileName(this, tr("Add media file"), lastInputDir, fileDialogFilter()));
|
|
if (fileName.isEmpty())
|
|
return;
|
|
lastInputDir = fileName;
|
|
// disconnect();
|
|
addFileList.clear();
|
|
addFileList << QUrl::fromLocalFile(fileName);
|
|
addFile();
|
|
}
|
|
|
|
void TsMuxerWindow::addFile()
|
|
{
|
|
if (addFileList.isEmpty())
|
|
return;
|
|
newFileName = QDir::toNativeSeparators(addFileList[0].toLocalFile());
|
|
if (lastSourceDir.isEmpty())
|
|
lastSourceDir = QFileInfo(newFileName).absolutePath();
|
|
addFileList.removeAt(0);
|
|
if (!checkFileDuplicate(newFileName))
|
|
return;
|
|
// disconnect(this, SIGNAL(tsMuxerSuccessFinished()));
|
|
// disconnect(this, SIGNAL(codecListReady()));
|
|
disconnect();
|
|
connect(this, &TsMuxerWindow::tsMuxerSuccessFinished, this, &TsMuxerWindow::onTsMuxerCodecInfoReceived);
|
|
connect(this, &TsMuxerWindow::codecListReady, this, &TsMuxerWindow::continueAddFile);
|
|
connect(this, &TsMuxerWindow::fileAdded, this, &TsMuxerWindow::addFile);
|
|
runInMuxMode = false;
|
|
tsMuxerExecute(QStringList() << newFileName);
|
|
}
|
|
|
|
bool TsMuxerWindow::checkFileDuplicate(const QString &fileName)
|
|
{
|
|
QString t = myUnquoteStr(fileName).trimmed();
|
|
for (int i = 0; i < ui->inputFilesLV->count(); ++i)
|
|
if (myUnquoteStr(ui->inputFilesLV->item(i)->data(FileNameRole).toString()) == t)
|
|
{
|
|
QMessageBox msgBox(this);
|
|
msgBox.setWindowTitle(tr("File already exists"));
|
|
msgBox.setText(tr("File \"%1\" already exists").arg(fileName));
|
|
msgBox.setIcon(QMessageBox::Warning);
|
|
msgBox.setStandardButtons(QMessageBox::Ok);
|
|
msgBox.exec();
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void TsMuxerWindow::setComboBoxText(QComboBox *comboBox, const QString &text)
|
|
{
|
|
for (int k = 0; k < comboBox->count(); ++k)
|
|
{
|
|
if (comboBox->itemText(k) == text)
|
|
{
|
|
comboBox->setCurrentIndex(k);
|
|
return;
|
|
}
|
|
}
|
|
comboBox->addItem(text);
|
|
comboBox->setCurrentIndex(comboBox->count() - 1);
|
|
}
|
|
|
|
void TsMuxerWindow::trackLVItemSelectionChanged()
|
|
{
|
|
if (disableUpdatesCnt)
|
|
return;
|
|
while (ui->tabWidgetTracks->count()) ui->tabWidgetTracks->removeTab(0);
|
|
if (ui->trackLV->currentRow() == -1)
|
|
{
|
|
ui->tabWidgetTracks->addTab(ui->tabSheetFake, TI_DEFAULT_TAB_NAME());
|
|
return;
|
|
}
|
|
QtvCodecInfo *codecInfo = getCurrentCodec();
|
|
if (!codecInfo)
|
|
return;
|
|
disableUpdatesCnt++;
|
|
if (ui->trackLV->rowCount() >= 1)
|
|
{
|
|
if (isVideoCodec(codecInfo->displayName))
|
|
{
|
|
ui->tabWidgetTracks->addTab(ui->tabSheetVideo, TI_DEFAULT_TAB_NAME());
|
|
|
|
ui->checkFPS->setChecked(codecInfo->checkFPS);
|
|
ui->checkBoxLevel->setChecked(codecInfo->checkLevel);
|
|
ui->comboBoxFPS->setEnabled(ui->checkFPS->isChecked());
|
|
ui->comboBoxLevel->setEnabled(ui->checkBoxLevel->isChecked());
|
|
ui->comboBoxSEI->setCurrentIndex(codecInfo->addSEIMethod);
|
|
ui->checkBoxSecondaryVideo->setChecked(codecInfo->isSecondary);
|
|
ui->checkBoxSPS->setChecked(codecInfo->addSPS);
|
|
ui->checkBoxRemovePulldown->setChecked(codecInfo->delPulldown == 1);
|
|
ui->checkBoxRemovePulldown->setEnabled(codecInfo->delPulldown >= 0);
|
|
|
|
setComboBoxText(ui->comboBoxFPS, codecInfo->fpsText);
|
|
if (!codecInfo->arText.isEmpty())
|
|
setComboBoxText(ui->comboBoxAR, codecInfo->arText);
|
|
else
|
|
ui->comboBoxAR->setCurrentIndex(0);
|
|
ui->checkBoxLevel->setEnabled(codecInfo->displayName == "H.264" || codecInfo->displayName == "MVC");
|
|
if (ui->checkBoxLevel->isEnabled())
|
|
setComboBoxText(ui->comboBoxLevel, codecInfo->levelText);
|
|
else
|
|
ui->comboBoxLevel->setCurrentIndex(0);
|
|
ui->comboBoxSEI->setEnabled(ui->checkBoxLevel->isEnabled());
|
|
ui->checkBoxSPS->setEnabled(ui->checkBoxLevel->isEnabled());
|
|
ui->comboBoxAR->setEnabled(codecInfo->displayName == "MPEG-2");
|
|
ui->comboBoxAR->setEnabled(true);
|
|
ui->labelAR->setEnabled(ui->comboBoxAR->isEnabled());
|
|
ui->checkBoxSecondaryVideo->setEnabled(codecInfo->displayName != "MVC");
|
|
}
|
|
else
|
|
{
|
|
ui->tabWidgetTracks->addTab(ui->tabSheetAudio, TI_DEFAULT_TAB_NAME());
|
|
if (codecInfo->displayName == "LPCM")
|
|
ui->tabWidgetTracks->addTab(ui->demuxLpcmOptions, TI_DEMUX_TAB_NAME());
|
|
|
|
if (codecInfo->displayName == "DTS-HD")
|
|
ui->dtsDwnConvert->setText(tr("Downconvert DTS-HD to DTS"));
|
|
else if (codecInfo->displayName == "TRUE-HD")
|
|
ui->dtsDwnConvert->setText(tr("Downconvert TRUE-HD to AC3"));
|
|
else if (codecInfo->displayName == "E-AC3 (DD+)")
|
|
ui->dtsDwnConvert->setText(tr("Downconvert E-AC3 to AC3"));
|
|
else
|
|
ui->dtsDwnConvert->setText(tr("Downconvert HD audio"));
|
|
ui->dtsDwnConvert->setEnabled(codecInfo->displayName == "DTS-HD" || codecInfo->displayName == "TRUE-HD" ||
|
|
codecInfo->displayName == "E-AC3 (DD+)");
|
|
ui->secondaryCheckBox->setEnabled(codecInfo->descr.contains("(DTS Express)") ||
|
|
codecInfo->descr.contains("(DTS Express 24bit)") ||
|
|
codecInfo->displayName == "E-AC3 (DD+)");
|
|
|
|
if (!ui->secondaryCheckBox->isEnabled())
|
|
ui->secondaryCheckBox->setChecked(false);
|
|
|
|
ui->langComboBox->setCurrentIndex(findLangByCode(codecInfo->lang));
|
|
ui->offsetsComboBox->setCurrentIndex(codecInfo->offsetId + 1);
|
|
ui->dtsDwnConvert->setVisible(codecInfo->displayName != "PGS" && codecInfo->displayName != "SRT");
|
|
ui->secondaryCheckBox->setVisible(ui->dtsDwnConvert->isVisible());
|
|
|
|
bool isPGS = codecInfo->displayName == "PGS";
|
|
ui->checkBoxKeepFps->setVisible(isPGS);
|
|
ui->offsetsLabel->setVisible(isPGS);
|
|
ui->offsetsComboBox->setVisible(isPGS);
|
|
|
|
ui->imageSubtitles->setVisible(!ui->dtsDwnConvert->isVisible());
|
|
ui->imageAudio->setVisible(ui->dtsDwnConvert->isVisible());
|
|
|
|
ui->editDelay->setValue(codecInfo->delay);
|
|
ui->dtsDwnConvert->setChecked(codecInfo->dtsDownconvert);
|
|
ui->secondaryCheckBox->setChecked(codecInfo->isSecondary);
|
|
ui->checkBoxKeepFps->setChecked(codecInfo->bindFps);
|
|
ui->editDelay->setEnabled(!ui->radioButtonDemux->isChecked());
|
|
}
|
|
}
|
|
|
|
disableUpdatesCnt--;
|
|
trackLVItemChanged(0);
|
|
}
|
|
|
|
void TsMuxerWindow::trackLVItemChanged(QTableWidgetItem *item)
|
|
{
|
|
if (disableUpdatesCnt > 0)
|
|
return;
|
|
|
|
Q_UNUSED(item);
|
|
updateMetaLines();
|
|
ui->moveupBtn->setEnabled(ui->trackLV->currentItem() != 0);
|
|
ui->movedownBtn->setEnabled(ui->trackLV->currentItem() != 0);
|
|
ui->removeTrackBtn->setEnabled(ui->trackLV->currentItem() != 0);
|
|
if (ui->trackLV->rowCount() == 0)
|
|
oldFileName.clear();
|
|
|
|
disableUpdatesCnt++;
|
|
bool checkedExist = false;
|
|
bool uncheckedExist = false;
|
|
for (int i = 0; i < ui->trackLV->rowCount(); ++i)
|
|
{
|
|
if (ui->trackLV->item(i, 0)->checkState() == Qt::Checked)
|
|
checkedExist = true;
|
|
else
|
|
uncheckedExist = true;
|
|
}
|
|
if (checkedExist && uncheckedExist)
|
|
m_header->setCheckState(Qt::PartiallyChecked);
|
|
else if (checkedExist)
|
|
m_header->setCheckState(Qt::Checked);
|
|
else
|
|
m_header->setCheckState(Qt::Unchecked);
|
|
update();
|
|
disableUpdatesCnt--;
|
|
}
|
|
|
|
void TsMuxerWindow::inputFilesLVChanged()
|
|
{
|
|
QListWidgetItem *itm = ui->inputFilesLV->currentItem();
|
|
if (!itm)
|
|
{
|
|
ui->btnAppend->setEnabled(false);
|
|
ui->removeFile->setEnabled(false);
|
|
return;
|
|
}
|
|
ui->btnAppend->setEnabled(itm->data(MplsItemRole).toInt() != MPLS_M2TS && ui->buttonMux->isEnabled());
|
|
ui->removeFile->setEnabled(itm->data(MplsItemRole).toInt() != MPLS_M2TS);
|
|
}
|
|
|
|
void TsMuxerWindow::modifyOutFileName(const QString fileName)
|
|
{
|
|
QFileInfo fi(unquoteStr(fileName));
|
|
QString name = fi.completeBaseName();
|
|
|
|
QString existingName = QDir::toNativeSeparators(ui->outFileName->text());
|
|
QFileInfo fiDst(existingName);
|
|
if (fiDst.completeBaseName() == "default")
|
|
{
|
|
QString dstPath;
|
|
if (existingName.contains(QDir::separator()))
|
|
dstPath = QDir::toNativeSeparators(fiDst.absolutePath());
|
|
else
|
|
dstPath = getOutputDir();
|
|
|
|
if (ui->radioButtonTS->isChecked())
|
|
ui->outFileName->setText(dstPath + QDir::separator() + name + ".ts");
|
|
else if (ui->radioButtonM2TS->isChecked())
|
|
ui->outFileName->setText(dstPath + QDir::separator() + name + ".m2ts");
|
|
else if (ui->radioButtonBluRayISO->isChecked())
|
|
ui->outFileName->setText(dstPath + QDir::separator() + name + ".iso");
|
|
else
|
|
ui->outFileName->setText(dstPath);
|
|
if (ui->outFileName->text() == fileName)
|
|
ui->outFileName->setText(dstPath + QDir::separator() + name + "_new." + fi.suffix());
|
|
}
|
|
}
|
|
|
|
void TsMuxerWindow::continueAddFile()
|
|
{
|
|
double fps;
|
|
double level;
|
|
disableUpdatesCnt++;
|
|
bool firstWarn = true;
|
|
int firstAddedIndex = -1;
|
|
for (int i = 0; i < codecList.size(); ++i)
|
|
{
|
|
if (codecList[i].displayName == "PGS (depended view)")
|
|
continue;
|
|
|
|
QtvCodecInfo info = codecList[i];
|
|
if (info.displayName.isEmpty())
|
|
{
|
|
QMessageBox msgBox(this);
|
|
msgBox.setWindowTitle(tr("Unsupported format"));
|
|
msgBox.setIcon(QMessageBox::Warning);
|
|
msgBox.setStandardButtons(QMessageBox::Ok);
|
|
if (codecList.size() == 0)
|
|
{
|
|
msgBox.setText(
|
|
tr("Unsupported format or all tracks are not recognized. File name: \"%1\"").arg(newFileName));
|
|
msgBox.exec();
|
|
disableUpdatesCnt--;
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
if (firstWarn)
|
|
{
|
|
msgBox.setText(
|
|
tr("Track %1 was not recognized and ignored. File name: \"%2\"").arg(i).arg(newFileName));
|
|
msgBox.exec();
|
|
firstWarn = false;
|
|
}
|
|
continue;
|
|
}
|
|
}
|
|
if (mplsFileList.isEmpty())
|
|
info.fileList << newFileName;
|
|
else
|
|
{
|
|
info.fileList << mplsFileList[0].name;
|
|
QFileInfo fileInfo(unquoteStr(newFileName));
|
|
// info.mplsFile = fileInfo.baseName();
|
|
info.mplsFiles << unquoteStr(newFileName);
|
|
}
|
|
if (info.descr.indexOf("not found") >= 0)
|
|
{
|
|
fps = 23.976;
|
|
info.checkFPS = true;
|
|
}
|
|
else
|
|
fps = extractFloatFromDescr(info.descr, "Frame rate: ");
|
|
info.width = extractFloatFromDescr(info.descr, "Resolution: ") + 0.5;
|
|
info.height = extractFloatFromDescr(info.descr, QString::number(info.width) + ":") + 0.5;
|
|
if (doubleCompare(fps, 23.976))
|
|
info.fpsText = "24000/1001";
|
|
else if (doubleCompare(fps, 29.97))
|
|
info.fpsText = "30000/1001";
|
|
else if (doubleCompare(fps, 59.94))
|
|
info.fpsText = "60000/1001";
|
|
else
|
|
info.fpsText = QString::number(fps);
|
|
info.fpsTextOrig = QString::number(fps);
|
|
level = extractFloatFromDescr(info.descr, "@");
|
|
info.levelText = QString::number(level);
|
|
if (info.descr.indexOf("pulldown") >= 0)
|
|
info.delPulldown = 0;
|
|
|
|
info.maxPgOffsets = extractFloatFromDescr(info.descr, "3d-pg-planes: ");
|
|
if (info.maxPgOffsets > 0)
|
|
m_3dMode = true;
|
|
|
|
if (info.descr.contains("3d-plane: zero"))
|
|
info.offsetId = -1;
|
|
else if (info.descr.contains("3d-plane: "))
|
|
info.offsetId = extractFloatFromDescr(info.descr, "3d-plane: ");
|
|
else if (m_3dMode)
|
|
info.offsetId = 0;
|
|
|
|
auto newTrackRowIdx = ui->trackLV->rowCount();
|
|
ui->trackLV->setRowCount(newTrackRowIdx + 1);
|
|
ui->trackLV->setRowHeight(newTrackRowIdx, 18);
|
|
QTableWidgetItem *item = new QTableWidgetItem("");
|
|
item->setCheckState(info.enabledByDefault ? Qt::Checked : Qt::Unchecked);
|
|
item->setData(Qt::UserRole, reinterpret_cast<qlonglong>(new QtvCodecInfo(info)));
|
|
ui->trackLV->setCurrentItem(item);
|
|
|
|
ui->trackLV->setItem(newTrackRowIdx, 0, item);
|
|
item = new QTableWidgetItem(newFileName);
|
|
item->setFlags(item->flags() & (~Qt::ItemIsEditable));
|
|
ui->trackLV->setItem(newTrackRowIdx, 1, item);
|
|
item = new QTableWidgetItem(info.displayName);
|
|
item->setFlags(item->flags() & (~Qt::ItemIsEditable));
|
|
ui->trackLV->setItem(newTrackRowIdx, 2, item);
|
|
item = new QTableWidgetItem(info.lang);
|
|
item->setFlags(item->flags() & (~Qt::ItemIsEditable));
|
|
ui->trackLV->setItem(newTrackRowIdx, 3, item);
|
|
item = new QTableWidgetItem(info.descr);
|
|
item->setFlags(item->flags() & (~Qt::ItemIsEditable));
|
|
ui->trackLV->setItem(newTrackRowIdx, 4, item);
|
|
if (firstAddedIndex == -1)
|
|
firstAddedIndex = newTrackRowIdx;
|
|
colorizeCurrentRow(&info, newTrackRowIdx);
|
|
addTrackToDefaultComboBox(newTrackRowIdx);
|
|
}
|
|
if (firstAddedIndex >= 0)
|
|
{
|
|
ui->trackLV->setRangeSelected(QTableWidgetSelectionRange(firstAddedIndex, 0, ui->trackLV->rowCount() - 1, 4),
|
|
true);
|
|
ui->trackLV->setCurrentCell(firstAddedIndex, 0);
|
|
}
|
|
|
|
QString displayName = newFileName;
|
|
if (fileDuration > 0)
|
|
displayName += QString(" (%1)").arg(floatToTime(fileDuration));
|
|
ui->inputFilesLV->addItem(displayName);
|
|
|
|
int index = ui->inputFilesLV->count() - 1;
|
|
QListWidgetItem *fileItem = ui->inputFilesLV->item(index);
|
|
if (!mplsFileList.empty())
|
|
fileItem->setData(MplsItemRole, MPLS_PRIMARY);
|
|
fileItem->setData(FileNameRole, newFileName);
|
|
QVariant v;
|
|
v.setValue<ChapterList>(chapters);
|
|
fileItem->setData(ChaptersRole, v);
|
|
fileItem->setData(FileDurationRole, fileDuration);
|
|
chapters.clear();
|
|
fileDuration = 0.0;
|
|
|
|
ui->inputFilesLV->setCurrentItem(fileItem);
|
|
if (mplsFileList.size() > 0)
|
|
doAppendInt(mplsFileList[0].name, newFileName, mplsFileList[0].duration, false, MPLS_M2TS);
|
|
for (int mplsCnt = 1; mplsCnt < mplsFileList.size(); ++mplsCnt)
|
|
doAppendInt(mplsFileList[mplsCnt].name, mplsFileList[mplsCnt - 1].name, mplsFileList[mplsCnt].duration, false,
|
|
MPLS_M2TS);
|
|
ui->inputFilesLV->setCurrentItem(ui->inputFilesLV->item(index));
|
|
updateMetaLines();
|
|
ui->moveupBtn->setEnabled(ui->trackLV->currentRow() >= 0);
|
|
ui->movedownBtn->setEnabled(ui->trackLV->currentRow() >= 0);
|
|
ui->removeTrackBtn->setEnabled(ui->trackLV->currentRow() >= 0);
|
|
if (!outFileNameModified)
|
|
{
|
|
modifyOutFileName(newFileName);
|
|
outFileNameModified = true;
|
|
}
|
|
updateMaxOffsets();
|
|
updateCustomChapters();
|
|
disableUpdatesCnt--;
|
|
trackLVItemSelectionChanged();
|
|
emit fileAdded();
|
|
}
|
|
|
|
void TsMuxerWindow::updateCustomChapters()
|
|
{
|
|
QSet<qint64> chaptersSet;
|
|
double prevDuration = 0.0;
|
|
double offset = 0.0;
|
|
for (int i = 0; i < ui->inputFilesLV->count(); ++i)
|
|
{
|
|
QListWidgetItem *item = ui->inputFilesLV->item(i);
|
|
if (item->data(MplsItemRole).toInt() == MPLS_M2TS)
|
|
continue;
|
|
if (item->text().left(4) == FILE_JOIN_PREFIX)
|
|
offset += prevDuration;
|
|
else
|
|
offset = 0;
|
|
|
|
ChapterList chapters = item->data(ChaptersRole).value<ChapterList>();
|
|
foreach (double chapter, chapters)
|
|
chaptersSet << qint64((chapter + offset) * 1000000);
|
|
prevDuration = item->data(FileDurationRole).toDouble();
|
|
}
|
|
|
|
ui->memoChapters->clear();
|
|
QList<qint64> mergedChapterList = chaptersSet.toList();
|
|
std::sort(std::begin(mergedChapterList), std::end(mergedChapterList));
|
|
for (auto chapter : mergedChapterList)
|
|
ui->memoChapters->insertPlainText(floatToTime(chapter / 1000000.0) + QString('\n'));
|
|
}
|
|
|
|
void splitLines(const QString &str, QList<QString> &strList)
|
|
{
|
|
strList = str.split('\n');
|
|
for (int i = 0; i < strList.size(); ++i)
|
|
{
|
|
if (!strList[i].isEmpty() && strList[i].at(0) == '\r')
|
|
strList[i] = strList[i].mid(1);
|
|
else if (!strList[i].isEmpty() && strList[i].at(strList[i].length() - 1) == '\r')
|
|
strList[i] = strList[i].mid(0, strList[i].length() - 1);
|
|
}
|
|
}
|
|
|
|
void TsMuxerWindow::addLines(const QByteArray &arr, QList<QString> &outList, bool isError)
|
|
{
|
|
QString str = QString::fromUtf8(arr);
|
|
QList<QString> strList;
|
|
splitLines(str, strList);
|
|
QString text;
|
|
for (int i = 0; i < strList.size(); ++i)
|
|
{
|
|
if (strList[i].trimmed().isEmpty())
|
|
continue;
|
|
int p = strList[i].indexOf("% complete");
|
|
if (p >= 0)
|
|
{
|
|
int numStartPos = 0;
|
|
for (int j = 0; j < strList[i].length(); ++j)
|
|
{
|
|
if (strList[i].at(j) >= '0' && strList[i].at(j) <= '9')
|
|
{
|
|
numStartPos = j;
|
|
break;
|
|
}
|
|
}
|
|
QString progress = strList[i].mid(numStartPos, p - numStartPos);
|
|
float tmpVal = progress.toFloat();
|
|
if (qAbs(tmpVal) > 0 && runInMuxMode)
|
|
muxForm->setProgress(int(double(tmpVal * 10) + 0.5)); // todo: uncomment here
|
|
}
|
|
else
|
|
{
|
|
if (runInMuxMode)
|
|
{
|
|
if (!text.isEmpty())
|
|
text += '\n';
|
|
text += strList[i];
|
|
}
|
|
else
|
|
outList << strList[i];
|
|
}
|
|
}
|
|
if (runInMuxMode && !text.isEmpty())
|
|
{
|
|
if (isError)
|
|
muxForm->addStdErrLine(text);
|
|
else
|
|
muxForm->addStdOutLine(text);
|
|
}
|
|
}
|
|
|
|
void TsMuxerWindow::readFromStdout() { addLines(proc.readAllStandardOutput(), procStdOutput, false); }
|
|
|
|
void TsMuxerWindow::readFromStderr()
|
|
{
|
|
QByteArray data(proc.readAllStandardError());
|
|
addLines(data, procErrOutput, true);
|
|
}
|
|
|
|
void TsMuxerWindow::myPlaySound(const QString &fileName)
|
|
{
|
|
sound.setSource(QUrl(QString("qrc%1").arg(fileName)));
|
|
sound.play();
|
|
}
|
|
|
|
void TsMuxerWindow::onProcessFinished(int exitCode, QProcess::ExitStatus exitStatus)
|
|
{
|
|
processFinished = true;
|
|
if (!metaName.isEmpty())
|
|
{
|
|
QFile::remove(metaName);
|
|
metaName.clear();
|
|
if (ui->checkBoxSound->isChecked())
|
|
{
|
|
if (exitCode == 0)
|
|
myPlaySound(":/sounds/success.wav");
|
|
else
|
|
myPlaySound(":/sounds/fail.wav");
|
|
}
|
|
}
|
|
processExitCode = exitCode;
|
|
processExitStatus = exitStatus;
|
|
muxForm->muxFinished(processExitCode, ui->radioButtonDemux->isChecked() ? tr("Demux") : tr("Mux"));
|
|
ui->buttonMux->setEnabled(true);
|
|
ui->addBtn->setEnabled(true);
|
|
inputFilesLVChanged();
|
|
if (processExitCode == 0)
|
|
emit tsMuxerSuccessFinished();
|
|
}
|
|
|
|
void TsMuxerWindow::onProcessError(QProcess::ProcessError error)
|
|
{
|
|
processExitCode = -1;
|
|
if (!metaName.isEmpty())
|
|
{
|
|
QFile::remove(metaName);
|
|
metaName.clear();
|
|
}
|
|
processFinished = true;
|
|
QMessageBox msgBox(this);
|
|
QString text;
|
|
msgBox.setWindowTitle(tr("tsMuxeR error"));
|
|
switch (error)
|
|
{
|
|
case QProcess::FailedToStart:
|
|
msgBox.setText(tr("tsMuxeR not found!"));
|
|
ui->buttonMux->setEnabled(true);
|
|
ui->addBtn->setEnabled(true);
|
|
inputFilesLVChanged();
|
|
break;
|
|
case QProcess::Crashed:
|
|
// process killed
|
|
if (runInMuxMode)
|
|
return;
|
|
for (int i = 0; i < procErrOutput.size(); ++i)
|
|
{
|
|
if (i > 0)
|
|
text += '\n';
|
|
text += procErrOutput[i];
|
|
}
|
|
msgBox.setText(text);
|
|
break;
|
|
default:
|
|
msgBox.setText(tr("Can't execute tsMuxeR!"));
|
|
break;
|
|
}
|
|
msgBox.setIcon(QMessageBox::Critical);
|
|
msgBox.setStandardButtons(QMessageBox::Ok);
|
|
msgBox.exec();
|
|
}
|
|
|
|
static QString getTsMuxerBinaryPath()
|
|
{
|
|
const auto applicationDirPath =
|
|
QDir::toNativeSeparators(QCoreApplication::applicationDirPath()) + QDir::separator();
|
|
for (auto binaryName : {"tsmuxer", "tsMuxeR"})
|
|
{
|
|
auto binaryPath = QStandardPaths::findExecutable(binaryName, {applicationDirPath});
|
|
if (!binaryPath.isEmpty())
|
|
{
|
|
return binaryPath;
|
|
}
|
|
}
|
|
return QString();
|
|
}
|
|
|
|
void TsMuxerWindow::tsMuxerExecute(const QStringList &args)
|
|
{
|
|
const auto exePath = getTsMuxerBinaryPath();
|
|
ui->buttonMux->setEnabled(false);
|
|
procStdOutput.clear();
|
|
procErrOutput.clear();
|
|
processFinished = false;
|
|
processExitCode = -1;
|
|
proc.start(exePath, args);
|
|
if (muxForm->isVisible())
|
|
muxForm->setProcess(&proc);
|
|
}
|
|
|
|
void TsMuxerWindow::doAppendInt(const QString &fileName, const QString &parentFileName, double duration,
|
|
bool doublePrefix, MplsType mplsRole)
|
|
{
|
|
QString itemName = FILE_JOIN_PREFIX;
|
|
if (doublePrefix)
|
|
itemName += FILE_JOIN_PREFIX;
|
|
itemName += fileName;
|
|
if (duration > 0)
|
|
itemName += QString(" ( %1)").arg(floatToTime(duration));
|
|
QListWidgetItem *item = new QListWidgetItem(itemName);
|
|
ui->inputFilesLV->insertItem(ui->inputFilesLV->currentRow() + 1, item);
|
|
item->setData(MplsItemRole, mplsRole);
|
|
item->setData(FileNameRole, fileName);
|
|
if (duration > 0)
|
|
item->setData(FileDurationRole, duration);
|
|
QVariant v;
|
|
v.setValue<ChapterList>(chapters);
|
|
item->setData(ChaptersRole, v);
|
|
|
|
ui->inputFilesLV->setCurrentItem(item);
|
|
|
|
for (int i = 0; i < ui->trackLV->rowCount(); ++i)
|
|
{
|
|
auto info = getCodecInfo(i);
|
|
if (!info)
|
|
continue;
|
|
if (mplsRole == MPLS_PRIMARY)
|
|
{
|
|
for (int j = 0; j < info->mplsFiles.size(); ++j)
|
|
{
|
|
if (info->mplsFiles[j] == parentFileName)
|
|
{
|
|
info->mplsFiles.insert(j + 1, fileName);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (int j = 0; j < info->fileList.size(); ++j)
|
|
{
|
|
if (info->fileList[j] == parentFileName)
|
|
{
|
|
info->fileList.insert(j + 1, fileName);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
bool TsMuxerWindow::isVideoCropped()
|
|
{
|
|
for (int i = 0; i < ui->trackLV->rowCount(); ++i)
|
|
{
|
|
auto codecInfo = getCodecInfo(i);
|
|
if (!codecInfo)
|
|
continue;
|
|
|
|
if (isVideoCodec(codecInfo->displayName))
|
|
{
|
|
if (codecInfo->height < 1080 && codecInfo->height != 720 && codecInfo->height != 576 &&
|
|
codecInfo->height != 480)
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool TsMuxerWindow::isDiskOutput() const
|
|
{
|
|
return ui->radioButtonAVCHD->isChecked() || ui->radioButtonBluRay->isChecked() ||
|
|
ui->radioButtonBluRayISO->isChecked();
|
|
}
|
|
|
|
QString TsMuxerWindow::getMuxOpts()
|
|
{
|
|
QString rez = "MUXOPT --no-pcr-on-video-pid ";
|
|
if (ui->checkBoxNewAudioPes->isChecked())
|
|
rez += "--new-audio-pes --hdmv-descriptors ";
|
|
if (ui->radioButtonBluRay->isChecked())
|
|
rez += (ui->checkBoxV3->isChecked() ? "--blu-ray-v3 " : "--blu-ray ");
|
|
else if (ui->radioButtonBluRayISO->isChecked())
|
|
{
|
|
rez += (ui->checkBoxV3->isChecked() ? "--blu-ray-v3 " : "--blu-ray ");
|
|
if (!ui->DiskLabelEdit->text().isEmpty())
|
|
rez += QString("--label=\"%1\" ").arg(ui->DiskLabelEdit->text());
|
|
}
|
|
else if (ui->radioButtonAVCHD->isChecked())
|
|
rez += "--avchd ";
|
|
else if (ui->radioButtonDemux->isChecked())
|
|
rez += "--demux ";
|
|
if (ui->checkBoxCBR->isChecked())
|
|
rez += "--cbr --bitrate=" + QString::number(ui->editCBRBitrate->value(), 'f', 3);
|
|
else
|
|
{
|
|
rez += "--vbr ";
|
|
if (ui->checkBoxRVBR->isChecked())
|
|
{
|
|
rez += QString("--minbitrate=") + QString::number(ui->editMinBitrate->value(), 'f', 3);
|
|
rez += QString(" --maxbitrate=") + QString::number(ui->editMaxBitrate->value(), 'f', 3);
|
|
}
|
|
}
|
|
// if (!ui->checkBoxuseAsynIO->isChecked())
|
|
// rez += " --no-asyncio";
|
|
if (isDiskOutput())
|
|
{
|
|
if (ui->checkBoxBlankPL->isChecked() && isVideoCropped())
|
|
{
|
|
rez += " --insertBlankPL";
|
|
if (ui->BlackplaylistCombo->value() != 1900)
|
|
rez += QString(" --blankOffset=") + QString::number(ui->BlackplaylistCombo->value());
|
|
}
|
|
if (ui->spinBoxMplsNum->value() > 0)
|
|
rez += " --mplsOffset=" + QString::number(ui->spinBoxMplsNum->value());
|
|
if (ui->spinBoxM2tsNum->value() > 0)
|
|
rez += " --m2tsOffset=" + QString::number(ui->spinBoxM2tsNum->value());
|
|
}
|
|
if (isDiskOutput())
|
|
{
|
|
if (ui->radioButtonAutoChapter->isChecked())
|
|
rez += " --auto-chapters=" + QString::number(ui->spinEditChapterLen->value());
|
|
if (ui->radioButtonCustomChapters->isChecked())
|
|
{
|
|
QString custChapStr;
|
|
QList<QString> lines;
|
|
splitLines(ui->memoChapters->toPlainText(), lines);
|
|
for (int i = 0; i < lines.size(); ++i)
|
|
{
|
|
QString tmpStr = lines[i].trimmed();
|
|
if (!tmpStr.isEmpty())
|
|
{
|
|
if (!custChapStr.isEmpty())
|
|
custChapStr += ';';
|
|
custChapStr += tmpStr;
|
|
}
|
|
}
|
|
rez += QString(" --custom-chapters=") + custChapStr;
|
|
}
|
|
}
|
|
if (ui->splitByDuration->isChecked())
|
|
rez += QString(" --split-duration=") + ui->spinEditSplitDuration->text();
|
|
if (ui->splitBySize->isChecked())
|
|
rez += QString(" --split-size=") + QString::number(ui->editSplitSize->value(), 'f', 3) +
|
|
ui->comboBoxMeasure->currentData().toString();
|
|
|
|
int startCut = qTimeToMsec(ui->cutStartTimeEdit->time());
|
|
int endCut = qTimeToMsec(ui->cutEndTimeEdit->time());
|
|
if (startCut > 0 && ui->checkBoxCut->isChecked())
|
|
rez += QString(" --cut-start=%1ms").arg(startCut);
|
|
if (endCut > 0 && ui->checkBoxCut->isChecked())
|
|
rez += QString(" --cut-end=%1ms").arg(endCut);
|
|
|
|
int vbvLen = ui->editVBVLen->value();
|
|
if (vbvLen > 0)
|
|
rez += " --vbv-len=" + QString::number(vbvLen);
|
|
|
|
if (isDiskOutput() && ui->rightEyeCheckBox->isChecked())
|
|
rez += " --right-eye";
|
|
/*
|
|
QString muxTimeStr = ui->muxTimeEdit->time().toString("hh:mm:ss.zzz");
|
|
if (muxTimeStr != "00:00:00.000")
|
|
rez += " --start-time=" + muxTimeStr;
|
|
*/
|
|
if (ui->muxTimeClock->value())
|
|
rez += QString(" --start-time=%1").arg(ui->muxTimeClock->value());
|
|
return rez;
|
|
}
|
|
|
|
int getCharsetCode(const QString &name)
|
|
{
|
|
Q_UNUSED(name);
|
|
return 0; // todo: refactor this function
|
|
}
|
|
|
|
double TsMuxerWindow::getRendererAnimationTime() const
|
|
{
|
|
switch (ui->comboBoxAnimation->currentIndex())
|
|
{
|
|
case 1:
|
|
return 0.1; // fast
|
|
break;
|
|
case 2:
|
|
return 0.25; // medium
|
|
break;
|
|
case 3:
|
|
return 0.5; // slow
|
|
break;
|
|
case 4:
|
|
return 1.0; // very slow
|
|
}
|
|
return 0.0;
|
|
}
|
|
|
|
void TsMuxerWindow::setRendererAnimationTime(double value)
|
|
{
|
|
int index = 0;
|
|
if (doubleCompare(value, 0.1))
|
|
index = 1;
|
|
else if (doubleCompare(value, 0.25))
|
|
index = 2;
|
|
else if (doubleCompare(value, 0.5))
|
|
index = 3;
|
|
else if (doubleCompare(value, 1.0))
|
|
index = 4;
|
|
|
|
ui->comboBoxAnimation->setCurrentIndex(index);
|
|
}
|
|
|
|
QString TsMuxerWindow::getSrtParams()
|
|
{
|
|
QString rez;
|
|
if (ui->listViewFont->rowCount() < 5)
|
|
return rez;
|
|
rez = QString(",font-name=\"") + ui->listViewFont->item(0, 1)->text();
|
|
rez += QString("\",font-size=") + ui->listViewFont->item(1, 1)->text();
|
|
rez += QString(",font-color=") + ui->listViewFont->item(2, 1)->text();
|
|
int charsetCode = getCharsetCode(ui->listViewFont->item(3, 1)->text());
|
|
if (charsetCode)
|
|
rez += QString(",font-charset=") + QString::number(charsetCode);
|
|
if (ui->lineSpacing->value() != 1.0)
|
|
rez += ",line-spacing=" + QString::number(ui->lineSpacing->value());
|
|
|
|
if (ui->listViewFont->item(4, 1)->text().indexOf("Italic") >= 0)
|
|
rez += ",font-italic";
|
|
if (ui->listViewFont->item(4, 1)->text().indexOf("Bold") >= 0)
|
|
rez += ",font-bold";
|
|
if (ui->listViewFont->item(4, 1)->text().indexOf("Underline") >= 0)
|
|
rez += ",font-underline";
|
|
if (ui->listViewFont->item(4, 1)->text().indexOf("Strikeout") >= 0)
|
|
rez += ",font-strikeout";
|
|
rez += QString(",bottom-offset=") + QString::number(ui->spinEditOffset->value()) +
|
|
",font-border=" + QString::number(ui->spinEditBorder->value());
|
|
if (ui->rbhLeft->isChecked())
|
|
rez += ",text-align=left";
|
|
else if (ui->rbhRight->isChecked())
|
|
rez += ",text-align=right";
|
|
else
|
|
rez += ",text-align=center";
|
|
|
|
double animationTime = getRendererAnimationTime();
|
|
if (animationTime > 0.0)
|
|
rez += QString(",fadein-time=%1,fadeout-time=%1").arg(animationTime);
|
|
|
|
for (int i = 0; i < ui->trackLV->rowCount(); ++i)
|
|
{
|
|
if (ui->trackLV->item(i, 0)->checkState() == Qt::Checked)
|
|
{
|
|
auto codecInfo = getCodecInfo(i);
|
|
if (!codecInfo)
|
|
continue;
|
|
|
|
if (isVideoCodec(codecInfo->displayName))
|
|
{
|
|
rez += QString(",video-width=") + QString::number(codecInfo->width);
|
|
rez += QString(",video-height=") + QString::number(codecInfo->height);
|
|
rez += QString(",fps=") + fpsTextToFpsStr(codecInfo->fpsText);
|
|
return rez;
|
|
}
|
|
}
|
|
}
|
|
rez += ",video-width=1920,video-height=1080,fps=23.976";
|
|
return rez;
|
|
}
|
|
|
|
QString TsMuxerWindow::getFileList(QtvCodecInfo *codecInfo)
|
|
{
|
|
QString rezStr;
|
|
|
|
if (codecInfo->mplsFiles.isEmpty())
|
|
{
|
|
for (int i = 0; i < codecInfo->fileList.size(); ++i)
|
|
{
|
|
if (i > 0)
|
|
rezStr += '+';
|
|
rezStr += QString("\"") + codecInfo->fileList[i] + "\"";
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (int i = 0; i < codecInfo->mplsFiles.size(); ++i)
|
|
{
|
|
if (i > 0)
|
|
rezStr += '+';
|
|
rezStr += QString("\"") + codecInfo->mplsFiles[i] + "\"";
|
|
}
|
|
}
|
|
|
|
return rezStr;
|
|
}
|
|
|
|
QString cornerToStr(int corner)
|
|
{
|
|
if (corner == 0)
|
|
return "topLeft";
|
|
else if (corner == 1)
|
|
return "topRight";
|
|
else if (corner == 2)
|
|
return "bottomRight";
|
|
else
|
|
return "bottomLeft";
|
|
}
|
|
|
|
QString toPipScaleStr(int scaleIdx)
|
|
{
|
|
if (scaleIdx == 0)
|
|
return "1";
|
|
if (scaleIdx == 1)
|
|
return "1/2";
|
|
else if (scaleIdx == 2)
|
|
return "1/4";
|
|
else if (scaleIdx == 3)
|
|
return "1.5";
|
|
else
|
|
return "fullScreen";
|
|
}
|
|
|
|
QString TsMuxerWindow::getVideoMetaInfo(QtvCodecInfo *codecInfo)
|
|
{
|
|
QString fpsStr;
|
|
QString levelStr;
|
|
QString rezStr = codecInfo->programName + ", ";
|
|
|
|
rezStr += getFileList(codecInfo);
|
|
|
|
if (codecInfo->checkFPS)
|
|
fpsStr = fpsTextToFpsStr(codecInfo->fpsText);
|
|
if (codecInfo->checkLevel)
|
|
levelStr = QString::number(codecInfo->levelText.toDouble(), 'f', 1);
|
|
|
|
if (ui->checkBoxCrop->isChecked() && ui->checkBoxCrop->isEnabled())
|
|
rezStr += QString(", ") + "restoreCrop";
|
|
if (!fpsStr.isEmpty())
|
|
rezStr += QString(", ") + "fps=" + fpsStr;
|
|
if (!levelStr.isEmpty())
|
|
rezStr += QString(", ") + "level=" + levelStr;
|
|
if (codecInfo->addSEIMethod == 1)
|
|
rezStr += QString(", ") + "insertSEI";
|
|
else if (codecInfo->addSEIMethod == 2)
|
|
rezStr += QString(", ") + "forceSEI";
|
|
if (codecInfo->addSPS)
|
|
rezStr += QString(", ") + "contSPS";
|
|
if (codecInfo->delPulldown == 1)
|
|
rezStr += QString(", ") + "delPulldown";
|
|
if (!codecInfo->arText.isEmpty())
|
|
rezStr += QString(", ") + "ar=" + codecInfo->arText;
|
|
|
|
if (codecInfo->isSecondary)
|
|
{
|
|
rezStr += QString(", secondary");
|
|
rezStr += QString(", pipCorner=%1").arg(cornerToStr(ui->comboBoxPipCorner->currentIndex()));
|
|
rezStr += QString(" ,pipHOffset=%1").arg(ui->spinBoxPipOffsetH->value());
|
|
rezStr += QString(" ,pipVOffset=%1").arg(ui->spinBoxPipOffsetV->value());
|
|
rezStr += QString(", pipScale=%1").arg(toPipScaleStr(ui->comboBoxPipSize->currentIndex()));
|
|
rezStr += QString(", pipLumma=%1").arg(ui->pipTransparencySpinBox->value());
|
|
}
|
|
|
|
return rezStr;
|
|
}
|
|
|
|
QString TsMuxerWindow::getAudioMetaInfo(QtvCodecInfo *codecInfo)
|
|
{
|
|
QString rezStr = codecInfo->programName + ", ";
|
|
rezStr += getFileList(codecInfo);
|
|
if (codecInfo->delay != 0)
|
|
rezStr += QString(", timeshift=") + QString::number(codecInfo->delay) + "ms";
|
|
if (codecInfo->dtsDownconvert && codecInfo->programName == "A_DTS")
|
|
rezStr += ", down-to-dts";
|
|
else if (codecInfo->dtsDownconvert && codecInfo->programName == "A_AC3")
|
|
rezStr += ", down-to-ac3";
|
|
if (codecInfo->isSecondary)
|
|
rezStr += ", secondary";
|
|
return rezStr;
|
|
}
|
|
|
|
void TsMuxerWindow::updateMuxTime1()
|
|
{
|
|
QTime t = ui->muxTimeEdit->time();
|
|
int clock = (t.hour() * 3600 + t.minute() * 60 + t.second()) * 45000ll + t.msec() * 45ll;
|
|
ui->muxTimeClock->blockSignals(true);
|
|
ui->muxTimeClock->setValue(clock);
|
|
ui->muxTimeClock->blockSignals(false);
|
|
updateMetaLines();
|
|
}
|
|
|
|
void TsMuxerWindow::updateMuxTime2()
|
|
{
|
|
double timeF = ui->muxTimeClock->value() / 45000.0;
|
|
ui->muxTimeEdit->blockSignals(true);
|
|
ui->muxTimeEdit->setTime(qTimeFromFloat(timeF));
|
|
ui->muxTimeEdit->blockSignals(false);
|
|
updateMetaLines();
|
|
}
|
|
|
|
void TsMuxerWindow::onLanguageComboBoxIndexChanged(int idx)
|
|
{
|
|
auto lang = ui->languageSelectComboBox->itemData(idx).toString();
|
|
qtCoreTranslator.load(QString("qtbase_%1").arg(lang), QLibraryInfo::location(QLibraryInfo::TranslationsPath));
|
|
tsMuxerTranslator.load(QString("tsmuxergui_%1").arg(lang), ":/i18n");
|
|
QFile aboutContent(QString(":/about_%1.html").arg(lang));
|
|
if (aboutContent.open(QIODevice::ReadOnly))
|
|
{
|
|
ui->textEdit->setHtml(QString::fromUtf8(aboutContent.readAll()));
|
|
}
|
|
else
|
|
{
|
|
qWarning() << "Failed to open about.html for language" << lang << aboutContent.errorString();
|
|
ui->textEdit->clear();
|
|
}
|
|
writeSettings();
|
|
}
|
|
|
|
void TsMuxerWindow::updateMetaLines()
|
|
{
|
|
if (!m_updateMeta || disableUpdatesCnt > 0)
|
|
return;
|
|
|
|
ui->memoMeta->clear();
|
|
QString metaContent;
|
|
metaContent.append(getMuxOpts() + '\n');
|
|
QString tmpFps;
|
|
for (int i = 0; i < ui->trackLV->rowCount(); ++i)
|
|
{
|
|
auto codecInfo = getCodecInfo(i);
|
|
if (!codecInfo)
|
|
continue;
|
|
|
|
if (isVideoCodec(codecInfo->displayName))
|
|
{
|
|
if (codecInfo->checkFPS)
|
|
tmpFps = fpsTextToFpsStr(codecInfo->fpsText);
|
|
else
|
|
tmpFps = codecInfo->fpsTextOrig;
|
|
break;
|
|
}
|
|
}
|
|
|
|
QString prefix;
|
|
QString postfix;
|
|
|
|
bool bluray3D = isDiskOutput() && m_3dMode;
|
|
|
|
for (int i = 0; i < ui->trackLV->rowCount(); ++i)
|
|
{
|
|
if (ui->trackLV->item(i, 0)->checkState() == Qt::Checked)
|
|
prefix = "";
|
|
else
|
|
prefix = "#";
|
|
auto codecInfo = getCodecInfo(i);
|
|
if (!codecInfo)
|
|
continue;
|
|
|
|
postfix.clear();
|
|
if (codecInfo->programName.startsWith('S'))
|
|
{
|
|
if (isDiskOutput() && ui->defaultSubTrackCheckBox->isChecked() &&
|
|
ui->defaultSubTrackComboBox->currentData().toInt() == i)
|
|
{
|
|
postfix +=
|
|
QString(", default=") + (ui->defaultSubTrackForcedOnlyCheckBox->isChecked() ? "forced" : "all");
|
|
}
|
|
}
|
|
if (codecInfo->displayName == "PGS")
|
|
{
|
|
if (codecInfo->bindFps && !tmpFps.isEmpty())
|
|
postfix += QString(", fps=") + tmpFps;
|
|
if (bluray3D && codecInfo->offsetId >= 0)
|
|
postfix += QString(", 3d-plane=%1").arg(codecInfo->offsetId);
|
|
}
|
|
if (codecInfo->displayName == "SRT")
|
|
{
|
|
postfix += getSrtParams();
|
|
if (bluray3D && codecInfo->offsetId >= 0)
|
|
postfix += QString(", offset=%1").arg(codecInfo->offsetId);
|
|
}
|
|
if (codecInfo->trackID != 0)
|
|
postfix += QString(", track=") + QString::number(codecInfo->trackID);
|
|
if (!codecInfo->lang.isEmpty())
|
|
postfix += QString(", lang=") + codecInfo->lang;
|
|
// if (!codecInfo->mplsFile.isEmpty())
|
|
// postfix += QString(", mplsFile=") + codecInfo->mplsFile;
|
|
if (codecInfo->subTrack != 0)
|
|
postfix += QString(", subTrack=") + QString::number(codecInfo->subTrack);
|
|
if (isVideoCodec(codecInfo->displayName))
|
|
metaContent.append(prefix + getVideoMetaInfo(codecInfo) + postfix + '\n');
|
|
else
|
|
{
|
|
if (isDiskOutput() && ui->defaultAudioTrackCheckBox->isChecked() &&
|
|
ui->defaultAudioTrackComboBox->currentData().toInt() == i && codecInfo->programName.startsWith('A'))
|
|
{
|
|
postfix += QString(", default");
|
|
}
|
|
metaContent.append(prefix + getAudioMetaInfo(codecInfo) + postfix + '\n');
|
|
}
|
|
}
|
|
ui->memoMeta->setPlainText(metaContent);
|
|
}
|
|
|
|
void TsMuxerWindow::onFontBtnClicked()
|
|
{
|
|
bool ok;
|
|
QFont font;
|
|
font.setFamily(ui->listViewFont->item(0, 1)->text());
|
|
font.setPointSize((ui->listViewFont->item(1, 1)->text()).toInt());
|
|
font.setItalic(ui->listViewFont->item(4, 1)->text().indexOf("Italic") >= 0);
|
|
font.setBold(ui->listViewFont->item(4, 1)->text().indexOf("Bold") >= 0);
|
|
font.setUnderline(ui->listViewFont->item(4, 1)->text().indexOf("Underline") >= 0);
|
|
font.setStrikeOut(ui->listViewFont->item(4, 1)->text().indexOf("Strikeout") >= 0);
|
|
font = QFontDialog::getFont(&ok, font, this);
|
|
if (ok)
|
|
{
|
|
// FT_Face fontFace = fontfreetypeFace();
|
|
ui->listViewFont->item(0, 1)->setText(font.family());
|
|
ui->listViewFont->item(1, 1)->setText(QString::number(font.pointSize()));
|
|
QString optStr;
|
|
if (font.italic())
|
|
optStr += "Italic";
|
|
if (font.bold())
|
|
{
|
|
if (!optStr.isEmpty())
|
|
optStr += ',';
|
|
optStr += "Bold";
|
|
}
|
|
if (font.underline())
|
|
{
|
|
if (!optStr.isEmpty())
|
|
optStr += ',';
|
|
optStr += "Underline";
|
|
}
|
|
if (font.strikeOut())
|
|
{
|
|
if (!optStr.isEmpty())
|
|
optStr += ',';
|
|
optStr += "Strikeout";
|
|
}
|
|
ui->listViewFont->item(4, 1)->setText(optStr);
|
|
writeSettings();
|
|
updateMetaLines();
|
|
}
|
|
}
|
|
|
|
quint32 bswap(quint32 val)
|
|
{
|
|
return val; // ntohl(val);
|
|
}
|
|
|
|
int colorLight(QColor color) { return (0.257 * color.red()) + (0.504 * color.green()) + (0.098 * color.blue()) + 16; }
|
|
|
|
void TsMuxerWindow::setTextItemColor(QString str)
|
|
{
|
|
while (str.length() < 8) str = '0' + str;
|
|
QColor color = bswap(str.toLongLong(0, 16));
|
|
QTableWidgetItem *item = ui->listViewFont->item(2, 1);
|
|
item->setBackground(QBrush(color));
|
|
if (colorLight(color) < 128)
|
|
item->setForeground(QBrush(QColor(255, 255, 255, 255)));
|
|
else
|
|
item->setForeground(QBrush(QColor(0, 0, 0, 255)));
|
|
item->setText(QString("0x") + str);
|
|
}
|
|
|
|
void TsMuxerWindow::onColorBtnClicked()
|
|
{
|
|
QColor color = bswap(ui->listViewFont->item(2, 1)->text().toLongLong(0, 16));
|
|
color = QColorDialog::getColor(color, this);
|
|
QString str = QString::number(bswap(color.rgba()), 16);
|
|
setTextItemColor(str);
|
|
|
|
writeSettings();
|
|
updateMetaLines();
|
|
}
|
|
|
|
void TsMuxerWindow::onGeneralCheckboxClicked()
|
|
{
|
|
ui->editMaxBitrate->setEnabled(ui->checkBoxRVBR->isChecked());
|
|
ui->editMinBitrate->setEnabled(ui->checkBoxRVBR->isChecked());
|
|
ui->editCBRBitrate->setEnabled(ui->checkBoxCBR->isChecked());
|
|
ui->BlackplaylistCombo->setEnabled(ui->checkBoxBlankPL->isChecked());
|
|
ui->BlackplaylistLabel->setEnabled(ui->checkBoxBlankPL->isChecked());
|
|
updateMetaLines();
|
|
}
|
|
|
|
void TsMuxerWindow::onGeneralSpinboxValueChanged() { updateMetaLines(); }
|
|
|
|
void TsMuxerWindow::onChapterParamsChanged()
|
|
{
|
|
ui->memoChapters->setEnabled(ui->radioButtonCustomChapters->isChecked());
|
|
ui->spinEditChapterLen->setEnabled(ui->radioButtonAutoChapter->isChecked());
|
|
updateMetaLines();
|
|
}
|
|
|
|
void TsMuxerWindow::onSplitCutParamsChanged()
|
|
{
|
|
ui->spinEditSplitDuration->setEnabled(ui->splitByDuration->isChecked());
|
|
ui->labelSplitByDur->setEnabled(ui->splitByDuration->isChecked());
|
|
ui->editSplitSize->setEnabled(ui->splitBySize->isChecked());
|
|
ui->comboBoxMeasure->setEnabled(ui->splitBySize->isChecked());
|
|
|
|
ui->cutStartTimeEdit->setEnabled(ui->checkBoxCut->isChecked());
|
|
ui->cutEndTimeEdit->setEnabled(ui->checkBoxCut->isChecked());
|
|
updateMetaLines();
|
|
}
|
|
|
|
void TsMuxerWindow::onSavedParamChanged()
|
|
{
|
|
writeSettings();
|
|
updateMetaLines();
|
|
}
|
|
|
|
void TsMuxerWindow::onFontParamsChanged() { updateMetaLines(); }
|
|
|
|
void TsMuxerWindow::onRemoveBtnClick()
|
|
{
|
|
if (!ui->inputFilesLV->currentItem())
|
|
return;
|
|
int idx = ui->inputFilesLV->currentRow();
|
|
bool delMplsM2ts = false;
|
|
|
|
if (idx < ui->inputFilesLV->count() - 1)
|
|
{
|
|
if (ui->inputFilesLV->currentItem()->data(MplsItemRole).toInt() == MPLS_PRIMARY)
|
|
delMplsM2ts = true;
|
|
else if (ui->inputFilesLV->currentItem()->text().left(4) != FILE_JOIN_PREFIX)
|
|
{
|
|
QString text = ui->inputFilesLV->item(idx + 1)->text();
|
|
if (text.length() >= 4 && text.left(4) == FILE_JOIN_PREFIX)
|
|
ui->inputFilesLV->item(idx + 1)->setText(text.mid(4));
|
|
}
|
|
}
|
|
|
|
delTracksByFileName(myUnquoteStr(ui->inputFilesLV->currentItem()->data(FileNameRole).toString()));
|
|
ui->inputFilesLV->takeItem(idx);
|
|
if (idx >= ui->inputFilesLV->count())
|
|
idx--;
|
|
if (delMplsM2ts)
|
|
{
|
|
while (idx < ui->inputFilesLV->count())
|
|
{
|
|
QString text = ui->inputFilesLV->item(idx)->text();
|
|
if (text.length() >= 4 && text.left(4) == FILE_JOIN_PREFIX)
|
|
{
|
|
delTracksByFileName(myUnquoteStr(ui->inputFilesLV->item(idx)->data(FileNameRole).toString()));
|
|
ui->inputFilesLV->takeItem(idx);
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
}
|
|
if (ui->inputFilesLV->count() > 0)
|
|
ui->inputFilesLV->setCurrentRow(idx);
|
|
updateCustomChapters();
|
|
}
|
|
|
|
void TsMuxerWindow::delTracksByFileName(const QString &fileName)
|
|
{
|
|
for (int i = ui->trackLV->rowCount() - 1; i >= 0; --i)
|
|
{
|
|
if (auto info = getCodecInfo(i))
|
|
{
|
|
for (int j = info->fileList.count() - 1; j >= 0; --j)
|
|
{
|
|
if (info->fileList[j] == fileName)
|
|
{
|
|
info->fileList.removeAt(j);
|
|
break;
|
|
}
|
|
}
|
|
for (int j = info->mplsFiles.count() - 1; j >= 0; --j)
|
|
{
|
|
if (info->mplsFiles[j] == fileName)
|
|
{
|
|
info->mplsFiles.removeAt(j);
|
|
break;
|
|
}
|
|
}
|
|
if (info->fileList.count() == 0)
|
|
deleteTrack(i);
|
|
}
|
|
}
|
|
updateMaxOffsets();
|
|
updateMetaLines();
|
|
}
|
|
|
|
void TsMuxerWindow::deleteTrack(int idx)
|
|
{
|
|
disableUpdatesCnt++;
|
|
delete getCodecInfo(idx);
|
|
int lastItemIndex = idx; // trackLV.items[idx].index;
|
|
ui->trackLV->removeRow(idx);
|
|
if (ui->trackLV->rowCount() == 0)
|
|
{
|
|
lastSourceDir.clear();
|
|
while (ui->tabWidgetTracks->count()) ui->tabWidgetTracks->removeTab(0);
|
|
ui->tabWidgetTracks->addTab(ui->tabSheetFake, TI_DEFAULT_TAB_NAME());
|
|
ui->outFileName->setText(getDefaultOutputFileName());
|
|
outFileNameModified = false;
|
|
}
|
|
else
|
|
{
|
|
if (lastItemIndex > ui->trackLV->rowCount() - 1)
|
|
--lastItemIndex;
|
|
if (lastItemIndex >= 0)
|
|
{
|
|
ui->trackLV->setCurrentCell(lastItemIndex, 0);
|
|
ui->trackLV->setRangeSelected(QTableWidgetSelectionRange(lastItemIndex, 0, lastItemIndex, 4), true);
|
|
}
|
|
updateNum();
|
|
}
|
|
updateMaxOffsets();
|
|
updateMetaLines();
|
|
ui->moveupBtn->setEnabled(ui->trackLV->currentItem() != 0);
|
|
ui->movedownBtn->setEnabled(ui->trackLV->currentItem() != 0);
|
|
ui->removeTrackBtn->setEnabled(ui->trackLV->currentItem() != 0);
|
|
disableUpdatesCnt--;
|
|
trackLVItemSelectionChanged();
|
|
removeTrackFromDefaultComboBox(idx);
|
|
}
|
|
|
|
void TsMuxerWindow::updateNum()
|
|
{
|
|
// for (int i = 0; i < ui->trackLV->rowCount(); ++i)
|
|
// ui->trackLV->item(i,0)->setText(QString::number(i+1));
|
|
}
|
|
|
|
void TsMuxerWindow::onAppendButtonClick()
|
|
{
|
|
QList<QtvCodecInfo> codecList;
|
|
if (ui->inputFilesLV->currentItem() == 0)
|
|
{
|
|
QMessageBox msgBox(this);
|
|
msgBox.setWindowTitle(tr("No track selected"));
|
|
msgBox.setText(tr("No track selected"));
|
|
msgBox.setIcon(QMessageBox::Information);
|
|
msgBox.setStandardButtons(QMessageBox::Ok);
|
|
msgBox.exec();
|
|
return;
|
|
}
|
|
QString fileName = QDir::toNativeSeparators(
|
|
QFileDialog::getOpenFileName(this, tr("Append media file"), lastInputDir, fileDialogFilter()));
|
|
if (fileName.isEmpty())
|
|
return;
|
|
lastInputDir = fileName;
|
|
addFileList.clear();
|
|
addFileList << QUrl::fromLocalFile(fileName);
|
|
appendFile();
|
|
}
|
|
|
|
void TsMuxerWindow::appendFile()
|
|
{
|
|
if (addFileList.isEmpty())
|
|
return;
|
|
newFileName = QDir::toNativeSeparators(addFileList[0].toLocalFile());
|
|
if (lastSourceDir.isEmpty())
|
|
lastSourceDir = QFileInfo(newFileName).absolutePath();
|
|
addFileList.removeAt(0);
|
|
if (!checkFileDuplicate(newFileName))
|
|
return;
|
|
// disconnect(this, SIGNAL(tsMuxerSuccessFinished()));
|
|
// disconnect(this, SIGNAL(codecListReady()));
|
|
disconnect();
|
|
connect(this, &TsMuxerWindow::tsMuxerSuccessFinished, this, &TsMuxerWindow::onTsMuxerCodecInfoReceived);
|
|
connect(this, &TsMuxerWindow::codecListReady, this, &TsMuxerWindow::continueAppendFile);
|
|
connect(this, &TsMuxerWindow::fileAppended, this, &TsMuxerWindow::appendFile);
|
|
runInMuxMode = false;
|
|
tsMuxerExecute(QStringList() << newFileName);
|
|
}
|
|
|
|
void TsMuxerWindow::continueAppendFile()
|
|
{
|
|
QString parentFileName = myUnquoteStr((ui->inputFilesLV->currentItem()->data(FileNameRole).toString()));
|
|
QFileInfo newFi(unquoteStr(newFileName));
|
|
QFileInfo oldFi(unquoteStr(parentFileName));
|
|
if (newFi.suffix().toUpper() != oldFi.suffix().toUpper())
|
|
{
|
|
QMessageBox msgBox(this);
|
|
msgBox.setWindowTitle(tr("Invalid file extension"));
|
|
msgBox.setText(tr("Appended file must have same file extension."));
|
|
msgBox.setIcon(QMessageBox::Information);
|
|
msgBox.setStandardButtons(QMessageBox::Ok);
|
|
msgBox.exec();
|
|
return;
|
|
}
|
|
|
|
disableUpdatesCnt++;
|
|
|
|
int idx = ui->inputFilesLV->currentRow();
|
|
bool firstStep = true;
|
|
bool doublePrefix = false;
|
|
for (; idx < ui->inputFilesLV->count(); ++idx)
|
|
{
|
|
int mplsRole = ui->inputFilesLV->item(idx)->data(MplsItemRole).toInt();
|
|
if ((mplsRole == MPLS_PRIMARY && firstStep) || mplsRole == MPLS_M2TS)
|
|
{
|
|
ui->inputFilesLV->setCurrentRow(idx);
|
|
doublePrefix = true;
|
|
}
|
|
else
|
|
break;
|
|
firstStep = false;
|
|
}
|
|
|
|
doAppendInt(newFileName, parentFileName, fileDuration, false, doublePrefix ? MPLS_PRIMARY : MPLS_NONE);
|
|
|
|
if (mplsFileList.size() > 0)
|
|
doAppendInt(mplsFileList[0].name, newFileName, mplsFileList[0].duration, doublePrefix, MPLS_M2TS);
|
|
for (int mplsCnt = 1; mplsCnt < mplsFileList.size(); ++mplsCnt)
|
|
doAppendInt(mplsFileList[mplsCnt].name, mplsFileList[mplsCnt - 1].name, mplsFileList[mplsCnt].duration,
|
|
doublePrefix, MPLS_M2TS);
|
|
|
|
updateMaxOffsets();
|
|
if (!outFileNameModified)
|
|
{
|
|
modifyOutFileName(newFileName);
|
|
outFileNameModified = true;
|
|
}
|
|
disableUpdatesCnt--;
|
|
updateMetaLines();
|
|
updateCustomChapters();
|
|
emit fileAppended();
|
|
}
|
|
|
|
void TsMuxerWindow::onRemoveTrackButtonClick()
|
|
{
|
|
if (ui->trackLV->currentItem())
|
|
deleteTrack(ui->trackLV->currentRow());
|
|
}
|
|
|
|
void TsMuxerWindow::onMoveUpButtonCLick()
|
|
{
|
|
if (ui->trackLV->currentItem() == 0 || ui->trackLV->currentRow() < 1)
|
|
return;
|
|
disableUpdatesCnt++;
|
|
auto preMoveRow = ui->trackLV->currentRow();
|
|
moveRow(preMoveRow, preMoveRow - 1);
|
|
moveTrackInDefaultComboBox(preMoveRow, preMoveRow - 1);
|
|
updateMetaLines();
|
|
updateNum();
|
|
disableUpdatesCnt--;
|
|
}
|
|
|
|
void TsMuxerWindow::onMoveDownButtonCLick()
|
|
{
|
|
if (ui->trackLV->currentItem() == 0 || ui->trackLV->rowCount() == 0 ||
|
|
ui->trackLV->currentRow() == ui->trackLV->rowCount() - 1)
|
|
return;
|
|
disableUpdatesCnt++;
|
|
auto preMoveRow = ui->trackLV->currentRow();
|
|
moveRow(preMoveRow, preMoveRow + 2);
|
|
moveTrackInDefaultComboBox(preMoveRow, preMoveRow + 1);
|
|
updateMetaLines();
|
|
updateNum();
|
|
disableUpdatesCnt--;
|
|
}
|
|
|
|
void TsMuxerWindow::moveRow(int index, int index2)
|
|
{
|
|
ui->trackLV->insertRow(index2);
|
|
ui->trackLV->setRowHeight(index2, 18);
|
|
if (index2 < index)
|
|
index++;
|
|
for (int i = 0; i < ui->trackLV->columnCount(); ++i)
|
|
ui->trackLV->setItem(index2, i, ui->trackLV->item(index, i)->clone());
|
|
ui->trackLV->removeRow(index);
|
|
if (index2 > index)
|
|
index2--;
|
|
ui->trackLV->setRangeSelected(QTableWidgetSelectionRange(index2, 0, index2, 4), true);
|
|
ui->trackLV->setCurrentCell(index2, 0);
|
|
}
|
|
|
|
void TsMuxerWindow::RadioButtonMuxClick()
|
|
{
|
|
if (outFileNameDisableChange)
|
|
return;
|
|
if (ui->radioButtonDemux->isChecked())
|
|
ui->buttonMux->setText(tr("Sta&rt demuxing"));
|
|
else
|
|
ui->buttonMux->setText(tr("Sta&rt muxing"));
|
|
ui->checkBoxNewAudioPes->setChecked(!ui->radioButtonTS->isChecked());
|
|
ui->checkBoxNewAudioPes->setEnabled(ui->radioButtonTS->isChecked() || ui->radioButtonM2TS->isChecked());
|
|
outFileNameDisableChange = true;
|
|
if (ui->radioButtonBluRay->isChecked() || ui->radioButtonDemux->isChecked() || ui->radioButtonAVCHD->isChecked())
|
|
{
|
|
QFileInfo fi(unquoteStr(ui->outFileName->text()));
|
|
if (!fi.suffix().isEmpty())
|
|
{
|
|
oldFileName = fi.fileName();
|
|
ui->outFileName->setText(QDir::toNativeSeparators(fi.absolutePath()) + QDir::separator());
|
|
}
|
|
ui->FilenameLabel->setText(tr("Folder"));
|
|
}
|
|
else
|
|
{
|
|
ui->FilenameLabel->setText(tr("File name"));
|
|
if (!oldFileName.isEmpty())
|
|
{
|
|
ui->outFileName->setText(QDir::toNativeSeparators(ui->outFileName->text()));
|
|
if (!ui->outFileName->text().isEmpty() && ui->outFileName->text().right(1) != QDir::separator())
|
|
ui->outFileName->setText(ui->outFileName->text() + QDir::separator());
|
|
ui->outFileName->setText(ui->outFileName->text() + oldFileName);
|
|
oldFileName.clear();
|
|
}
|
|
if (ui->radioButtonTS->isChecked())
|
|
{
|
|
ui->outFileName->setText(changeFileExt(ui->outFileName->text(), "ts"));
|
|
mSaveDialogFilter = TS_SAVE_DIALOG_FILTER();
|
|
}
|
|
else if (ui->radioButtonBluRayISO->isChecked())
|
|
{
|
|
ui->outFileName->setText(changeFileExt(ui->outFileName->text(), "iso"));
|
|
mSaveDialogFilter = ISO_SAVE_DIALOG_FILTER();
|
|
}
|
|
else
|
|
{
|
|
ui->outFileName->setText(changeFileExt(ui->outFileName->text(), "m2ts"));
|
|
mSaveDialogFilter = M2TS_SAVE_DIALOG_FILTER();
|
|
}
|
|
}
|
|
ui->DiskLabel->setVisible(ui->radioButtonBluRayISO->isChecked());
|
|
ui->DiskLabelEdit->setVisible(ui->radioButtonBluRayISO->isChecked());
|
|
ui->editDelay->setEnabled(!ui->radioButtonDemux->isChecked());
|
|
updateMetaLines();
|
|
outFileNameDisableChange = false;
|
|
}
|
|
|
|
void TsMuxerWindow::outFileNameChanged()
|
|
{
|
|
outFileNameModified = true;
|
|
if (outFileNameDisableChange)
|
|
return;
|
|
if (ui->radioButtonDemux->isChecked() || ui->radioButtonBluRay->isChecked() || ui->radioButtonAVCHD->isChecked())
|
|
return;
|
|
|
|
outFileNameDisableChange = true;
|
|
QFileInfo fi(unquoteStr(ui->outFileName->text().trimmed()));
|
|
QString ext = fi.suffix().toUpper();
|
|
|
|
bool isISOMode = ui->radioButtonBluRayISO->isChecked();
|
|
|
|
if (ext == "M2TS" || ext == "M2TS\"")
|
|
ui->radioButtonM2TS->setChecked(true);
|
|
else if (ext == "ISO" || ext == "ISO\"")
|
|
{
|
|
ui->radioButtonBluRayISO->setChecked(true);
|
|
}
|
|
else
|
|
ui->radioButtonTS->setChecked(true);
|
|
|
|
bool isISOModeNew = ui->radioButtonBluRayISO->isChecked();
|
|
|
|
ui->DiskLabel->setVisible(ui->radioButtonBluRayISO->isChecked());
|
|
ui->DiskLabelEdit->setVisible(ui->radioButtonBluRayISO->isChecked());
|
|
if (isISOMode != isISOModeNew)
|
|
updateMetaLines();
|
|
outFileNameDisableChange = false;
|
|
}
|
|
|
|
void TsMuxerWindow::saveFileDialog()
|
|
{
|
|
if (ui->radioButtonDemux->isChecked() || ui->radioButtonBluRay->isChecked() || ui->radioButtonAVCHD->isChecked())
|
|
{
|
|
QString folder = QDir::toNativeSeparators(QFileDialog::getExistingDirectory(this, getOutputDir()));
|
|
if (!folder.isEmpty())
|
|
{
|
|
ui->outFileName->setText(folder + QDir::separator());
|
|
outFileNameModified = true;
|
|
lastOutputDir = folder;
|
|
writeSettings();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
QString path = getOutputDir();
|
|
if (path.isEmpty())
|
|
{
|
|
QString fileName = unquoteStr(ui->outFileName->text());
|
|
path = QFileInfo(fileName).absolutePath();
|
|
}
|
|
QString fileName = QDir::toNativeSeparators(
|
|
QFileDialog::getSaveFileName(this, tr("Select file for muxing"), path, mSaveDialogFilter));
|
|
if (!fileName.isEmpty())
|
|
{
|
|
ui->outFileName->setText(fileName);
|
|
lastOutputDir = QFileInfo(fileName).absolutePath();
|
|
writeSettings();
|
|
}
|
|
}
|
|
}
|
|
|
|
void TsMuxerWindow::startMuxing()
|
|
{
|
|
QString outputName = unquoteStr(ui->outFileName->text().trimmed());
|
|
ui->outFileName->setText(outputName);
|
|
lastOutputDir = QFileInfo(outputName).absolutePath();
|
|
writeSettings();
|
|
if (ui->radioButtonM2TS->isChecked())
|
|
{
|
|
QFileInfo fi(ui->outFileName->text());
|
|
if (fi.suffix().toUpper() != "M2TS")
|
|
{
|
|
QMessageBox msgBox(this);
|
|
msgBox.setWindowTitle(tr("Invalid file name"));
|
|
msgBox.setText(tr("The output file \"%1\" has invalid extension. Please, change file extension "
|
|
"to \".m2ts\"")
|
|
.arg(ui->outFileName->text()));
|
|
msgBox.setIcon(QMessageBox::Warning);
|
|
msgBox.setStandardButtons(QMessageBox::Ok);
|
|
msgBox.exec();
|
|
return;
|
|
}
|
|
}
|
|
else if (ui->radioButtonBluRayISO->isChecked())
|
|
{
|
|
QFileInfo fi(ui->outFileName->text());
|
|
if (fi.suffix().toUpper() != "ISO")
|
|
{
|
|
QMessageBox msgBox(this);
|
|
msgBox.setWindowTitle(tr("Invalid file name"));
|
|
msgBox.setText(tr("The output file \"%1\" has invalid extension. Please, change file extension to \".iso\"")
|
|
.arg(ui->outFileName->text()));
|
|
msgBox.setIcon(QMessageBox::Warning);
|
|
msgBox.setStandardButtons(QMessageBox::Ok);
|
|
msgBox.exec();
|
|
return;
|
|
}
|
|
}
|
|
|
|
bool isFile =
|
|
ui->radioButtonM2TS->isChecked() || ui->radioButtonTS->isChecked() || ui->radioButtonBluRayISO->isChecked();
|
|
if (isFile && QFile::exists(ui->outFileName->text()))
|
|
{
|
|
//: Used in expressions "Overwrite existing %1" and "The output %1 already exists".
|
|
auto fileOrDir = isFile ? tr("file") : tr("directory");
|
|
QMessageBox msgBox(this);
|
|
msgBox.setWindowTitle(tr("Overwrite existing %1?").arg(fileOrDir));
|
|
msgBox.setText(tr("The output %1 \"%2\" already exists. Do you want to overwrite it?")
|
|
.arg(fileOrDir)
|
|
.arg(ui->outFileName->text()));
|
|
msgBox.setIcon(QMessageBox::Question);
|
|
msgBox.setStandardButtons(QMessageBox::Yes | QMessageBox::No);
|
|
if (msgBox.exec() != QMessageBox::Yes)
|
|
return;
|
|
}
|
|
|
|
QFileInfo fi(ui->outFileName->text());
|
|
metaName =
|
|
QDir::toNativeSeparators(QDir::tempPath()) + QDir::separator() + "tsMuxeR_" + fi.completeBaseName() + ".meta";
|
|
if (!saveMetaFile(metaName))
|
|
{
|
|
metaName.clear();
|
|
return;
|
|
}
|
|
muxForm->prepare(!ui->radioButtonDemux->isChecked() ? tr("Muxing in progress") : tr("Demuxing in progress"));
|
|
ui->buttonMux->setEnabled(false);
|
|
ui->addBtn->setEnabled(false);
|
|
ui->btnAppend->setEnabled(false);
|
|
muxForm->show();
|
|
disconnect();
|
|
// QCoreApplication::dir
|
|
runInMuxMode = true;
|
|
tsMuxerExecute(QStringList() << metaName << quoteStr(ui->outFileName->text()));
|
|
}
|
|
|
|
void TsMuxerWindow::saveMetaFileBtnClick()
|
|
{
|
|
QString metaName = QFileDialog::getSaveFileName(this, "", changeFileExt(ui->outFileName->text(), "meta"),
|
|
tr("tsMuxeR project file (*.meta);;All files (*.*)"));
|
|
if (metaName.isEmpty())
|
|
return;
|
|
QFileInfo fi(metaName);
|
|
QDir dir;
|
|
dir.mkpath(fi.absolutePath());
|
|
saveMetaFile(metaName);
|
|
}
|
|
|
|
bool TsMuxerWindow::saveMetaFile(const QString &metaName)
|
|
{
|
|
QFile file(metaName);
|
|
if (!file.open(QIODevice::WriteOnly))
|
|
{
|
|
QMessageBox msgBox(this);
|
|
msgBox.setWindowTitle(tr("Can't create temporary meta file"));
|
|
msgBox.setText(tr("Can't create temporary meta file \"%1\"").arg(metaName));
|
|
msgBox.setIcon(QMessageBox::Warning);
|
|
msgBox.setStandardButtons(QMessageBox::Ok);
|
|
msgBox.exec();
|
|
return false;
|
|
}
|
|
QByteArray metaText = ui->memoMeta->toPlainText().toUtf8();
|
|
file.write(metaText);
|
|
file.close();
|
|
return true;
|
|
}
|
|
|
|
void TsMuxerWindow::closeEvent(QCloseEvent *event)
|
|
{
|
|
if (!metaName.isEmpty())
|
|
{
|
|
QFile::remove(metaName);
|
|
metaName.clear();
|
|
}
|
|
muxForm->close();
|
|
event->accept();
|
|
}
|
|
|
|
void TsMuxerWindow::changeEvent(QEvent *event)
|
|
{
|
|
if (event->type() == QEvent::LanguageChange)
|
|
{
|
|
ui->retranslateUi(this);
|
|
}
|
|
QWidget::changeEvent(event);
|
|
}
|
|
|
|
void TsMuxerWindow::dragEnterEvent(QDragEnterEvent *event)
|
|
{
|
|
if (event->mimeData()->hasFormat("text/plain") || event->mimeData()->hasFormat("text/uri-list"))
|
|
{
|
|
if (ui->addBtn->isEnabled())
|
|
{
|
|
opacityTimer.stop();
|
|
setWindowOpacity(0.9);
|
|
event->acceptProposedAction();
|
|
QWidget *w = childAt(event->pos());
|
|
updateBtns(w);
|
|
}
|
|
}
|
|
}
|
|
|
|
void TsMuxerWindow::dropEvent(QDropEvent *event)
|
|
{
|
|
setWindowOpacity(1.0);
|
|
updateBtns(0);
|
|
QWidget *w = childAt(event->pos());
|
|
QString wName;
|
|
if (w)
|
|
wName = w->objectName();
|
|
if (event->mimeData()->hasFormat("text/uri-list"))
|
|
{
|
|
addFileList = event->mimeData()->urls();
|
|
event->acceptProposedAction();
|
|
}
|
|
else if (event->mimeData()->hasFormat("text/plain"))
|
|
{
|
|
QList<QString> strList;
|
|
addFileList.clear();
|
|
splitLines(event->mimeData()->text(), strList);
|
|
QList<QUrl> urls = event->mimeData()->urls();
|
|
for (int i = 0; i < strList.size(); ++i) addFileList << QUrl::fromLocalFile(strList[i]);
|
|
event->acceptProposedAction();
|
|
}
|
|
if (addFileList.isEmpty())
|
|
return;
|
|
if (wName == "btnAppend" && ui->btnAppend->isEnabled())
|
|
appendFile();
|
|
else if (ui->addBtn->isEnabled())
|
|
addFile();
|
|
}
|
|
|
|
void TsMuxerWindow::dragMoveEvent(QDragMoveEvent *event)
|
|
{
|
|
event->acceptProposedAction();
|
|
QWidget *w = childAt(event->pos());
|
|
updateBtns(w);
|
|
}
|
|
|
|
void TsMuxerWindow::updateBtns(QWidget *w)
|
|
{
|
|
if (w)
|
|
{
|
|
QString wName = w->objectName();
|
|
ui->btnAppend->setDefault(wName == "btnAppend" && ui->btnAppend->isEnabled());
|
|
ui->addBtn->setDefault(wName == "addBtn" && ui->addBtn->isEnabled());
|
|
}
|
|
else
|
|
{
|
|
ui->btnAppend->setDefault(false);
|
|
ui->addBtn->setDefault(false);
|
|
}
|
|
QFont font = ui->removeFile->font();
|
|
QFont bFont(font);
|
|
bFont.setBold(true);
|
|
if (ui->btnAppend->isDefault())
|
|
ui->btnAppend->setFont(bFont);
|
|
else
|
|
ui->btnAppend->setFont(font);
|
|
if (ui->addBtn->isDefault())
|
|
ui->addBtn->setFont(bFont);
|
|
else
|
|
ui->addBtn->setFont(font);
|
|
}
|
|
|
|
void TsMuxerWindow::dragLeaveEvent(QDragLeaveEvent *) { opacityTimer.start(100); }
|
|
|
|
void TsMuxerWindow::onOpacityTimer()
|
|
{
|
|
opacityTimer.stop();
|
|
setWindowOpacity(1.0);
|
|
updateBtns(0);
|
|
}
|
|
|
|
void TsMuxerWindow::updateMaxOffsets()
|
|
{
|
|
int maxPGOffsets = 0;
|
|
m_3dMode = false;
|
|
|
|
for (int i = 0; i < ui->trackLV->rowCount(); ++i)
|
|
{
|
|
auto codecInfo = getCodecInfo(i);
|
|
if (!codecInfo)
|
|
continue;
|
|
|
|
if (codecInfo->displayName == "MVC")
|
|
{
|
|
m_3dMode = true;
|
|
maxPGOffsets = qMax(maxPGOffsets, codecInfo->maxPgOffsets);
|
|
}
|
|
}
|
|
|
|
disableUpdatesCnt++;
|
|
|
|
int oldIndex = ui->offsetsComboBox->currentIndex();
|
|
ui->offsetsComboBox->clear();
|
|
ui->offsetsComboBox->addItem(QString("zero"));
|
|
for (int i = 0; i < maxPGOffsets; ++i) ui->offsetsComboBox->addItem(QString("plane #%1").arg(i));
|
|
if (oldIndex >= 0 && oldIndex < ui->offsetsComboBox->count())
|
|
ui->offsetsComboBox->setCurrentIndex(oldIndex);
|
|
|
|
disableUpdatesCnt--;
|
|
}
|
|
|
|
bool TsMuxerWindow::eventFilter(QObject *obj, QEvent *event)
|
|
{
|
|
if (obj == ui->label_Donate && event->type() == QEvent::MouseButtonPress)
|
|
{
|
|
QDesktopServices::openUrl(QUrl("https://github.com/justdan96/tsMuxer"));
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return QWidget::eventFilter(obj, event);
|
|
}
|
|
}
|
|
|
|
void TsMuxerWindow::at_sectionCheckstateChanged(Qt::CheckState state)
|
|
{
|
|
if (disableUpdatesCnt > 0)
|
|
return;
|
|
|
|
disableUpdatesCnt++;
|
|
for (int i = 0; i < ui->trackLV->rowCount(); ++i) ui->trackLV->item(i, 0)->setCheckState(state);
|
|
disableUpdatesCnt--;
|
|
trackLVItemSelectionChanged();
|
|
}
|
|
|
|
void TsMuxerWindow::writeSettings()
|
|
{
|
|
if (disableUpdatesCnt > 0)
|
|
return;
|
|
|
|
disableUpdatesCnt++;
|
|
|
|
settings->beginGroup("main");
|
|
// settings->setValue("asyncIO", ui->checkBoxuseAsynIO->isChecked());
|
|
settings->setValue("soundEnabled", ui->checkBoxSound->isChecked());
|
|
settings->setValue("hdmvPES", ui->checkBoxNewAudioPes->isChecked());
|
|
if (ui->checkBoxCrop->isEnabled())
|
|
settings->setValue("restoreCropEnabled", ui->checkBoxCrop->isChecked());
|
|
settings->setValue("inputDir", lastInputDir);
|
|
settings->setValue("outputDir", lastOutputDir);
|
|
settings->setValue("useBlankPL", ui->checkBoxBlankPL->isChecked());
|
|
settings->setValue("blankPLNum", ui->BlackplaylistCombo->value());
|
|
|
|
settings->setValue("outputToInputFolder", ui->radioButtonOutoutInInput->isChecked());
|
|
settings->setValue("language", ui->languageSelectComboBox->currentText());
|
|
|
|
settings->endGroup();
|
|
|
|
settings->beginGroup("subtitles");
|
|
settings->setValue("fontBorder", ui->spinEditBorder->value());
|
|
settings->setValue("fontLineSpacing", ui->lineSpacing->value());
|
|
settings->setValue("offset", ui->spinEditOffset->value());
|
|
settings->setValue("fadeTime", getRendererAnimationTime());
|
|
settings->setValue("family", ui->listViewFont->item(0, 1)->text());
|
|
settings->setValue("size", ui->listViewFont->item(1, 1)->text().toUInt());
|
|
settings->setValue("color", ui->listViewFont->item(2, 1)->text().mid(2).toUInt(0, 16));
|
|
settings->setValue("options", ui->listViewFont->item(4, 1)->text());
|
|
settings->endGroup();
|
|
|
|
settings->beginGroup("pip");
|
|
settings->setValue("corner", ui->comboBoxPipCorner->currentIndex());
|
|
settings->setValue("h_offset", ui->spinBoxPipOffsetH->value());
|
|
settings->setValue("v_offset", ui->spinBoxPipOffsetV->value());
|
|
settings->setValue("size", ui->comboBoxPipSize->currentIndex());
|
|
settings->endGroup();
|
|
|
|
disableUpdatesCnt--;
|
|
}
|
|
|
|
bool TsMuxerWindow::readSettings()
|
|
{
|
|
// due to QTBUG-28893, the settings saved under "general" are not accessible
|
|
// when using .ini files for storage - those are used on Linux by default.
|
|
// newer GUI versions will save the settings under the "main" group to avoid
|
|
// that. the "general" group is still read in order to import the old settings
|
|
// on non-Linux systems where the bug doesn't occur.
|
|
if (!readGeneralSettings("main") && !readGeneralSettings("general"))
|
|
{
|
|
return false; // no settings still written
|
|
}
|
|
// checkBoxVBR checkBoxRVBR editMaxBitrate editMinBitrate checkBoxCBR
|
|
// editCBRBitrate editVBVLen
|
|
|
|
settings->beginGroup("subtitles");
|
|
ui->spinEditBorder->setValue(settings->value("fontBorder").toInt());
|
|
ui->lineSpacing->setValue(settings->value("fontLineSpacing").toDouble());
|
|
setRendererAnimationTime(settings->value("fadeTime").toDouble());
|
|
ui->spinEditOffset->setValue(settings->value("offset").toInt());
|
|
// keep backward compatibility with versions < 2.6.15 which contain "famaly" key
|
|
if (settings->contains("famaly"))
|
|
{
|
|
settings->setValue("family", settings->value("famaly"));
|
|
settings->remove("famaly");
|
|
}
|
|
QString fontName = settings->value("family").toString();
|
|
if (!fontName.isEmpty())
|
|
ui->listViewFont->item(0, 1)->setText(fontName);
|
|
int fontSize = settings->value("size").toInt();
|
|
if (fontSize > 0)
|
|
ui->listViewFont->item(1, 1)->setText(QString::number(fontSize));
|
|
if (!settings->value("color").isNull())
|
|
{
|
|
quint32 color = settings->value("color").toUInt();
|
|
setTextItemColor(QString::number(color, 16));
|
|
}
|
|
ui->listViewFont->item(4, 1)->setText(settings->value("options").toString());
|
|
settings->endGroup();
|
|
|
|
settings->beginGroup("pip");
|
|
ui->comboBoxPipCorner->setCurrentIndex(settings->value("corner").toInt());
|
|
ui->spinBoxPipOffsetH->setValue(settings->value("h_offset").toInt());
|
|
ui->spinBoxPipOffsetV->setValue(settings->value("v_offset").toInt());
|
|
ui->comboBoxPipSize->setCurrentIndex(settings->value("size").toInt());
|
|
settings->endGroup();
|
|
|
|
return true;
|
|
}
|
|
|
|
bool TsMuxerWindow::readGeneralSettings(const QString &prefix)
|
|
{
|
|
settings->beginGroup(prefix);
|
|
|
|
auto lang = settings->value("language");
|
|
if (lang.isValid())
|
|
{
|
|
ui->languageSelectComboBox->setCurrentText(lang.toString());
|
|
}
|
|
else
|
|
{
|
|
ui->languageSelectComboBox->setCurrentIndex(0);
|
|
}
|
|
|
|
if (!settings->contains("outputDir"))
|
|
{
|
|
settings->endGroup();
|
|
return false;
|
|
}
|
|
|
|
lastInputDir = settings->value("inputDir").toString();
|
|
lastOutputDir = settings->value("outputDir").toString();
|
|
|
|
// ui->checkBoxuseAsynIO->setChecked(settings->value("asyncIO").toBool());
|
|
ui->checkBoxSound->setChecked(settings->value("soundEnabled").toBool());
|
|
ui->checkBoxNewAudioPes->setChecked(settings->value("hdmvPES").toBool());
|
|
ui->checkBoxCrop->setChecked(settings->value("restoreCropEnabled").toBool());
|
|
ui->checkBoxBlankPL->setChecked(settings->value("useBlankPL").toBool());
|
|
int plNum = settings->value("blankPLNum").toInt();
|
|
if (plNum)
|
|
ui->BlackplaylistCombo->setValue(plNum);
|
|
|
|
ui->radioButtonOutoutInInput->setChecked(settings->value("outputToInputFolder").toBool());
|
|
ui->radioButtonStoreOutput->setChecked(!ui->radioButtonOutoutInInput->isChecked());
|
|
|
|
settings->endGroup();
|
|
return true;
|
|
}
|