1 ///////////////////////////////////////////////////////////////////////////////
2 // Simple x264 Launcher
3 // Copyright (C) 2004-2014 LoRd_MuldeR <MuldeR2@GMX.de>
5 // This program is free software; you can redistribute it and/or modify
6 // it under the terms of the GNU General Public License as published by
7 // the Free Software Foundation; either version 2 of the License, or
8 // (at your option) any later version.
10 // This program is distributed in the hope that it will be useful,
11 // but WITHOUT ANY WARRANTY; without even the implied warranty of
12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 // GNU General Public License for more details.
15 // You should have received a copy of the GNU General Public License along
16 // with this program; if not, write to the Free Software Foundation, Inc.,
17 // 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
19 // http://www.gnu.org/licenses/gpl-2.0.txt
20 ///////////////////////////////////////////////////////////////////////////////
22 #include "win_addJob.h"
23 #include "uic_win_addJob.h"
26 #include "model_options.h"
27 #include "model_preferences.h"
28 #include "model_sysinfo.h"
29 #include "model_recently.h"
30 #include "encoder_x264.h"
31 #include "encoder_x265.h"
33 #include "win_editor.h"
37 #include <QCloseEvent>
38 #include <QMessageBox>
39 #include <QFileDialog>
40 #include <QDesktopServices>
43 #include <QInputDialog>
50 #define ARRAY_SIZE(ARRAY) (sizeof((ARRAY))/sizeof((ARRAY[0])))
51 #define VALID_DIR(PATH) ((!(PATH).isEmpty()) && QFileInfo(PATH).exists() && QFileInfo(PATH).isDir())
53 #define REMOVE_USAFED_ITEM \
55 for(int i = 0; i < ui->cbxTemplate->count(); i++) \
57 const OptionsModel* temp = reinterpret_cast<const OptionsModel*>(ui->cbxTemplate->itemData(i).value<const void*>()); \
60 ui->cbxTemplate->blockSignals(true); \
61 ui->cbxTemplate->removeItem(i); \
62 ui->cbxTemplate->blockSignals(false); \
68 #define ADD_CONTEXTMENU_ACTION(WIDGET, ICON, TEXT, SLOTNAME) \
70 QAction *_action = new QAction((ICON), (TEXT), this); \
71 _action->setData(QVariant::fromValue<void*>(WIDGET)); \
72 WIDGET->addAction(_action); \
73 connect(_action, SIGNAL(triggered(bool)), this, SLOT(SLOTNAME())); \
76 #define ADD_CONTEXTMENU_SEPARATOR(WIDGET) \
78 QAction *_action = new QAction(this); \
79 _action->setSeparator(true); \
80 WIDGET->addAction(_action); \
83 #define BLOCK_SIGNALS(FLAG) do \
85 ui->cbxEncoderType->blockSignals(FLAG); \
86 ui->cbxEncoderArch->blockSignals(FLAG); \
87 ui->cbxEncoderVariant->blockSignals(FLAG); \
88 ui->cbxRateControlMode->blockSignals(FLAG); \
89 ui->spinQuantizer->blockSignals(FLAG); \
90 ui->spinBitrate->blockSignals(FLAG); \
91 ui->cbxPreset->blockSignals(FLAG); \
92 ui->cbxTuning->blockSignals(FLAG); \
93 ui->cbxProfile->blockSignals(FLAG); \
94 ui->editCustomX264Params->blockSignals(FLAG); \
95 ui->editCustomAvs2YUVParams->blockSignals(FLAG); \
99 Q_DECLARE_METATYPE(const void*)
101 ///////////////////////////////////////////////////////////////////////////////
103 ///////////////////////////////////////////////////////////////////////////////
105 class StringValidator : public QValidator
108 StringValidator(QLabel *notifier, QLabel *icon)
110 m_notifier(notifier), m_icon(icon)
116 virtual State validate(QString &input, int &pos) const = 0;
118 virtual void fixup(QString &input) const
120 input = input.simplified();
124 QLabel *const m_notifier, *const m_icon;
126 bool checkParam(const QString &input, const QString ¶m, const bool doubleMinus) const
128 static const char c[20] = {' ', '*', '?', '<', '>', '/', '\\', '"', '\'', '!', '+', '#', '&', '%', '=', ',', ';', '.', 'ยด', '`'};
129 const QString prefix = doubleMinus ? QLatin1String("--") : QLatin1String("-");
132 if(param.length() > 1)
134 flag = flag || input.endsWith(QString("%1%2").arg(prefix, param), Qt::CaseInsensitive);
135 for(size_t i = 0; i < sizeof(c); i++)
137 flag = flag || input.contains(QString("%1%2%3").arg(prefix, param, QChar::fromLatin1(c[i])), Qt::CaseInsensitive);
142 flag = flag || input.startsWith(QString("-%1").arg(param));
143 for(size_t i = 0; i < sizeof(c); i++)
145 flag = flag || input.contains(QString("%1-%2").arg(QChar::fromLatin1(c[i]), param), Qt::CaseSensitive);
148 if((flag) && (m_notifier))
150 m_notifier->setText(tr("Invalid parameter: %1").arg((param.length() > 1) ? QString("%1%2").arg(prefix, param) : QString("-%1").arg(param)));
155 const bool &setStatus(const bool &flag, const QString &toolName) const
161 if(m_notifier->isHidden()) m_notifier->show();
162 if(m_icon) { if(m_icon->isHidden()) m_icon->show(); }
163 if(QWidget *w = m_notifier->topLevelWidget()->focusWidget())
165 QToolTip::showText(static_cast<QWidget*>(w->parent())->mapToGlobal(w->pos()), QString("<nobr>%1</nobr>").arg(tr("<b>Warning:</b> You entered a parameter that is incomaptible with using %1 from a GUI.<br>Please note that the GUI will automatically set <i>this</i> parameter for you (if required).").arg(toolName)), m_notifier, QRect());
173 if(m_notifier->isVisible()) m_notifier->hide();
174 if(m_icon) { if(m_icon->isVisible()) m_icon->hide(); }
175 QToolTip::hideText();
182 class StringValidatorX264 : public StringValidator
185 StringValidatorX264(QLabel *notifier, QLabel *icon) : StringValidator(notifier, icon) {}
187 virtual State validate(QString &input, int &pos) const
189 static const char* p[] = {"B", "o", "h", "p", "q", /*"fps", "frames",*/ "preset", "tune", "profile",
190 "stdin", "crf", "bitrate", "qp", "pass", "stats", "output", "help","quiet", NULL};
192 bool invalid = false;
194 for(size_t i = 0; p[i] && (!invalid); i++)
196 invalid = invalid || checkParam(input, QString::fromLatin1(p[i]), true);
199 return setStatus(invalid, "x264") ? QValidator::Intermediate : QValidator::Acceptable;
203 class StringValidatorAvs2YUV : public StringValidator
206 StringValidatorAvs2YUV(QLabel *notifier, QLabel *icon) : StringValidator(notifier, icon) {}
208 virtual State validate(QString &input, int &pos) const
210 static const char* p[] = {"o", "frames", "seek", "raw", "hfyu", "slave", NULL};
212 bool invalid = false;
214 for(size_t i = 0; p[i] && (!invalid); i++)
216 invalid = invalid || checkParam(input, QString::fromLatin1(p[i]), false);
219 return setStatus(invalid, "Avs2YUV") ? QValidator::Intermediate : QValidator::Acceptable;
223 ///////////////////////////////////////////////////////////////////////////////
224 // Constructor & Destructor
225 ///////////////////////////////////////////////////////////////////////////////
227 AddJobDialog::AddJobDialog(QWidget *parent, OptionsModel *const options, RecentlyUsed *const recentlyUsed, const SysinfoModel *const sysinfo, const PreferencesModel *const preferences)
231 m_recentlyUsed(recentlyUsed),
233 m_preferences(preferences),
234 m_defaults(new OptionsModel(sysinfo)),
235 ui(new Ui::AddJobDialog())
237 //Init the dialog, from the .ui file
239 setWindowFlags(windowFlags() & (~Qt::WindowContextHelpButtonHint));
242 ui->buttonSaveTemplate->setMaximumHeight(20);
243 ui->buttonDeleteTemplate->setMaximumHeight(20);
244 resize(width(), minimumHeight());
245 setMinimumSize(size());
246 setMaximumHeight(height());
248 //Hide optional controls
249 ui->checkBoxApplyToAll->setVisible(false);
251 //Monitor combobox changes
252 connect(ui->cbxEncoderType, SIGNAL(currentIndexChanged(int)), this, SLOT(encoderIndexChanged(int)));
253 connect(ui->cbxEncoderVariant, SIGNAL(currentIndexChanged(int)), this, SLOT(variantIndexChanged(int)));
254 connect(ui->cbxRateControlMode, SIGNAL(currentIndexChanged(int)), this, SLOT(modeIndexChanged(int)));
257 connect(ui->buttonBrowseSource, SIGNAL(clicked()), this, SLOT(browseButtonClicked()));
258 connect(ui->buttonBrowseOutput, SIGNAL(clicked()), this, SLOT(browseButtonClicked()));
259 connect(ui->buttonSaveTemplate, SIGNAL(clicked()), this, SLOT(saveTemplateButtonClicked()));
260 connect(ui->buttonDeleteTemplate, SIGNAL(clicked()), this, SLOT(deleteTemplateButtonClicked()));
263 ui->editCustomX264Params->installEventFilter(this);
264 ui->editCustomX264Params->setValidator(new StringValidatorX264(ui->labelNotificationX264, ui->iconNotificationX264));
265 ui->editCustomX264Params->clear();
266 ui->editCustomAvs2YUVParams->installEventFilter(this);
267 ui->editCustomAvs2YUVParams->setValidator(new StringValidatorAvs2YUV(ui->labelNotificationAvs2YUV, ui->iconNotificationAvs2YUV));
268 ui->editCustomAvs2YUVParams->clear();
270 //Install event filter
271 ui->labelHelpScreenX264->installEventFilter(this);
272 ui->labelHelpScreenAvs2YUV->installEventFilter(this);
274 //Monitor for options changes
275 connect(ui->cbxEncoderType, SIGNAL(currentIndexChanged(int)), this, SLOT(configurationChanged()));
276 connect(ui->cbxEncoderArch, SIGNAL(currentIndexChanged(int)), this, SLOT(configurationChanged()));
277 connect(ui->cbxEncoderVariant, SIGNAL(currentIndexChanged(int)), this, SLOT(configurationChanged()));
278 connect(ui->cbxRateControlMode, SIGNAL(currentIndexChanged(int)), this, SLOT(configurationChanged()));
279 connect(ui->spinQuantizer, SIGNAL(valueChanged(double)), this, SLOT(configurationChanged()));
280 connect(ui->spinBitrate, SIGNAL(valueChanged(int)), this, SLOT(configurationChanged()));
281 connect(ui->cbxPreset, SIGNAL(currentIndexChanged(int)), this, SLOT(configurationChanged()));
282 connect(ui->cbxTuning, SIGNAL(currentIndexChanged(int)), this, SLOT(configurationChanged()));
283 connect(ui->cbxProfile, SIGNAL(currentIndexChanged(int)), this, SLOT(configurationChanged()));
284 connect(ui->editCustomX264Params, SIGNAL(textChanged(QString)), this, SLOT(configurationChanged()));
285 connect(ui->editCustomAvs2YUVParams, SIGNAL(textChanged(QString)), this, SLOT(configurationChanged()));
287 //Create context menus
288 ADD_CONTEXTMENU_ACTION(ui->editCustomX264Params, QIcon(":/buttons/page_edit.png"), tr("Open the Text-Editor"), editorActionTriggered);
289 ADD_CONTEXTMENU_ACTION(ui->editCustomAvs2YUVParams, QIcon(":/buttons/page_edit.png"), tr("Open the Text-Editor"), editorActionTriggered);
290 ADD_CONTEXTMENU_SEPARATOR(ui->editCustomX264Params);
291 ADD_CONTEXTMENU_SEPARATOR(ui->editCustomAvs2YUVParams);
292 ADD_CONTEXTMENU_ACTION(ui->editCustomX264Params, QIcon(":/buttons/page_copy.png"), tr("Copy to Clipboard"), copyActionTriggered);
293 ADD_CONTEXTMENU_ACTION(ui->editCustomAvs2YUVParams, QIcon(":/buttons/page_copy.png"), tr("Copy to Clipboard"), copyActionTriggered);
294 ADD_CONTEXTMENU_ACTION(ui->editCustomX264Params, QIcon(":/buttons/page_paste.png"), tr("Paste from Clipboard"), pasteActionTriggered);
295 ADD_CONTEXTMENU_ACTION(ui->editCustomAvs2YUVParams, QIcon(":/buttons/page_paste.png"), tr("Paste from Clipboard"), pasteActionTriggered);
297 //Setup template selector
299 connect(ui->cbxTemplate, SIGNAL(currentIndexChanged(int)), this, SLOT(templateSelected()));
302 AddJobDialog::~AddJobDialog(void)
305 for(int i = 0; i < ui->cbxTemplate->model()->rowCount(); i++)
307 if(ui->cbxTemplate->itemText(i).startsWith("<") || ui->cbxTemplate->itemText(i).endsWith(">"))
311 const OptionsModel *item = reinterpret_cast<const OptionsModel*>(ui->cbxTemplate->itemData(i).value<const void*>());
312 ui->cbxTemplate->setItemData(i, QVariant::fromValue<const void*>(NULL));
317 if(const QValidator *tmp = ui->editCustomX264Params->validator())
319 ui->editCustomX264Params->setValidator(NULL);
322 if(const QValidator *tmp = ui->editCustomAvs2YUVParams->validator())
324 ui->editCustomAvs2YUVParams->setValidator(NULL);
328 X264_DELETE(m_defaults);
332 ///////////////////////////////////////////////////////////////////////////////
334 ///////////////////////////////////////////////////////////////////////////////
336 void AddJobDialog::showEvent(QShowEvent *event)
338 QDialog::showEvent(event);
341 if((!ui->editSource->text().isEmpty()) && ui->editOutput->text().isEmpty())
343 QString outPath = generateOutputFileName(QDir::fromNativeSeparators(ui->editSource->text()), m_recentlyUsed->outputDirectory(), m_recentlyUsed->filterIndex(), m_preferences->getSaveToSourcePath());
344 ui->editOutput->setText(QDir::toNativeSeparators(outPath));
345 ui->buttonAccept->setFocus();
348 ui->labelNotificationX264->hide();
349 ui->iconNotificationX264->hide();
350 ui->labelNotificationAvs2YUV->hide();
351 ui->iconNotificationAvs2YUV->hide();
353 //Enable drag&drop support for this window, required for Qt v4.8.4+
354 setAcceptDrops(true);
357 bool AddJobDialog::eventFilter(QObject *o, QEvent *e)
359 if((o == ui->labelHelpScreenX264) && (e->type() == QEvent::MouseButtonPress))
361 OptionsModel options(m_sysinfo); saveOptions(&options);
362 HelpDialog *helpScreen = new HelpDialog(this, false, m_sysinfo, &options, m_preferences);
364 X264_DELETE(helpScreen);
366 else if((o == ui->labelHelpScreenAvs2YUV) && (e->type() == QEvent::MouseButtonPress))
368 HelpDialog *helpScreen = new HelpDialog(this, false, m_sysinfo, m_defaults, m_preferences);
370 X264_DELETE(helpScreen);
372 else if((o == ui->editCustomX264Params) && (e->type() == QEvent::FocusOut))
374 ui->editCustomX264Params->setText(ui->editCustomX264Params->text().simplified());
376 else if((o == ui->editCustomAvs2YUVParams) && (e->type() == QEvent::FocusOut))
378 ui->editCustomAvs2YUVParams->setText(ui->editCustomAvs2YUVParams->text().simplified());
383 void AddJobDialog::dragEnterEvent(QDragEnterEvent *event)
385 bool accept[2] = {false, false};
387 foreach(const QString &fmt, event->mimeData()->formats())
389 accept[0] = accept[0] || fmt.contains("text/uri-list", Qt::CaseInsensitive);
390 accept[1] = accept[1] || fmt.contains("FileNameW", Qt::CaseInsensitive);
393 if(accept[0] && accept[1])
395 event->acceptProposedAction();
399 void AddJobDialog::dropEvent(QDropEvent *event)
402 QList<QUrl> urls = event->mimeData()->urls();
406 QDragEnterEvent dragEvent(event->pos(), event->proposedAction(), event->mimeData(), Qt::NoButton, Qt::NoModifier);
407 if(qApp->notify(parent(), &dragEvent))
409 qApp->notify(parent(), event);
414 while((!urls.isEmpty()) && droppedFile.isEmpty())
416 QUrl currentUrl = urls.takeFirst();
417 QFileInfo file(currentUrl.toLocalFile());
418 if(file.exists() && file.isFile())
420 qDebug("AddJobDialog::dropEvent: %s", file.canonicalFilePath().toUtf8().constData());
421 droppedFile = file.canonicalFilePath();
425 if(!droppedFile.isEmpty())
427 const QString outFileName = generateOutputFileName(droppedFile, currentOutputPath(), currentOutputIndx(), m_preferences->getSaveToSourcePath());
428 ui->editSource->setText(QDir::toNativeSeparators(droppedFile));
429 ui->editOutput->setText(QDir::toNativeSeparators(outFileName));
433 ///////////////////////////////////////////////////////////////////////////////
435 ///////////////////////////////////////////////////////////////////////////////
437 void AddJobDialog::encoderIndexChanged(int index)
439 const bool isX265 = (index > 0);
440 const bool noProf = isX265 || (ui->cbxEncoderVariant->currentIndex() > 0);
442 ui->cbxEncoderVariant->setItemText(1, isX265 ? tr("16-Bit") : tr("10-Bit"));
443 ui->labelProfile->setEnabled(!noProf);
444 ui->cbxProfile->setEnabled(!noProf);
445 if(noProf) ui->cbxProfile->setCurrentIndex(0);
448 void AddJobDialog::variantIndexChanged(int index)
450 const bool noProf = (index > 0) || (ui->cbxEncoderType->currentIndex() > 0);
452 ui->labelProfile->setEnabled(!noProf);
453 ui->cbxProfile->setEnabled(!noProf);
454 if(noProf) ui->cbxProfile->setCurrentIndex(0);
457 void AddJobDialog::modeIndexChanged(int index)
459 ui->spinQuantizer->setEnabled(index == 0 || index == 1);
460 ui->spinBitrate->setEnabled(index == 2 || index == 3);
463 void AddJobDialog::accept(void)
466 if((ui->cbxEncoderType->currentIndex() == OptionsModel::EncType_X265) && (!m_sysinfo->has256Support()))
468 QMessageBox::warning(this, tr("x265 unsupported"), tr("<nobr>Sorry, the x265 encoder is <b>not</b> currently available on this computer!<br>Please see the Readme file on how to obtain and install x265...</nobr>"));
469 ui->cbxEncoderType->setCurrentIndex(OptionsModel::EncType_X264);
473 //Check 64-Bit support
474 if((ui->cbxEncoderArch->currentIndex() == OptionsModel::EncArch_x64) && (!m_sysinfo->hasX64Support()))
476 QMessageBox::warning(this, tr("64-Bit unsupported!"), tr("<nobr>Sorry, this computer does <b>not</b> support 64-Bit encoders!</nobr>"));
477 ui->cbxEncoderArch->setCurrentIndex(OptionsModel::EncArch_x32);
481 //Selection complete?
482 if(ui->editSource->text().trimmed().isEmpty())
484 QMessageBox::warning(this, tr("Not Found!"), tr("<nobr>Please select a valid source file first!<(nobr>"));
487 if(ui->editOutput->text().trimmed().isEmpty())
489 QMessageBox::warning(this, tr("Not Selected!"), tr("<nobr>Please select a valid output file first!</nobr>"));
494 QFileInfo sourceFile = QFileInfo(this->sourceFile());
495 if(!(sourceFile.exists() && sourceFile.isFile()))
497 QMessageBox::warning(this, tr("Not Found!"), tr("<nobr>The selected source file could not be found!</nobr>"));
502 const AbstractEncoderInfo &encoderInfo = getEncoderInfo(ui->cbxEncoderType->currentIndex());
504 //Is the type of the source file supported? (as far as we can tell)
505 if(sourceFile.suffix().compare("AVS", Qt::CaseInsensitive) == 0)
507 if(!m_sysinfo->hasAVSSupport())
509 if(QMessageBox::warning(this, tr("Avisynth unsupported!"), tr("<nobr>An Avisynth script was selected as input, although Avisynth is <b>not</b> available!</nobr>"), tr("Abort"), tr("Ingnore (at your own risk!)")) != 1)
515 else if((sourceFile.suffix().compare("VPY", Qt::CaseInsensitive) == 0) || (sourceFile.suffix().compare("PY", Qt::CaseInsensitive) == 0))
517 if(!m_sysinfo->hasVPSSupport())
519 if(QMessageBox::warning(this, tr("VapurSynth unsupported!"), tr("<nobr>A VapourSynth script was selected as input, although VapourSynth is <b>not/<b> available!</nobr>"), tr("Abort"), tr("Ingnore (at your own risk!)")) != 1)
527 const QStringList inputFormats = encoderInfo.supportedInputFormats();
528 if(!inputFormats.contains(sourceFile.suffix(), Qt::CaseInsensitive))
530 if(QMessageBox::warning(this, tr("Unsupported input format"), tr("<nobr>The selected encoder does <b>not</b> support the selected input format!</nobr>"), tr("Abort"), tr("Ingnore (at your own risk!)")) != 1)
538 //Is output file extension supported by encoder?
539 const QStringList outputFormats = encoderInfo.supportedOutputFormats();
540 QFileInfo outputFile = QFileInfo(this->outputFile());
541 if(!outputFormats.contains(outputFile.suffix(), Qt::CaseInsensitive))
543 QMessageBox::warning(this, tr("Unsupported output format"), tr("<nobr>Sorry, the selected encoder does not support the selected output format!</nobr>"));
544 ui->editOutput->setText(QDir::toNativeSeparators(QString("%1/%2.%3").arg(outputFile.absolutePath(), outputFile.completeBaseName(), outputFormats.first())));
548 //Does output file already exist?
549 if(outputFile.exists() && outputFile.isFile())
551 int ret = QMessageBox::question(this, tr("Already Exists!"), tr("<nobr>Output file already exists! Overwrite?</nobr>"), QMessageBox::Yes | QMessageBox::No, QMessageBox::No);
552 if(ret != QMessageBox::Yes) return;
554 if(outputFile.exists() && (!outputFile.isFile()))
556 QMessageBox::warning(this, tr("Not a File!"), tr("<nobr>Selected output file does not appear to be a valid file!</nobr>"));
560 //Is destination dir writable?
561 QFileInfo outputDir = QFileInfo(outputFile.absolutePath());
562 if(!(outputDir.exists() && outputDir.isDir() && outputDir.isWritable()))
564 QMessageBox::warning(this, tr("Not Writable!"), tr("<nobr>Output directory does not exist or is not writable!</nobr>"));
568 //Custom parameters okay?
569 if(!ui->editCustomX264Params->hasAcceptableInput())
571 int ret = QMessageBox::warning(this, tr("Invalid Params"), tr("<nobr>Your custom parameters are invalid and will be discarded!</nobr>"), QMessageBox::Ignore | QMessageBox::Cancel, QMessageBox::Cancel);
572 if(ret != QMessageBox::Ignore) return;
575 //Update recently used
576 m_recentlyUsed->setFilterIndex(currentOutputIndx());
577 m_recentlyUsed->setSourceDirectory(currentSourcePath());
578 m_recentlyUsed->setOutputDirectory(currentOutputPath());
579 RecentlyUsed::saveRecentlyUsed(m_recentlyUsed);
582 saveOptions(m_options);
586 void AddJobDialog::browseButtonClicked(void)
588 if(QObject::sender() == ui->buttonBrowseSource)
590 QString filePath = QFileDialog::getOpenFileName(this, tr("Open Source File"), currentSourcePath(true), getInputFilterLst(), NULL, QFileDialog::DontUseNativeDialog);
591 if(!(filePath.isNull() || filePath.isEmpty()))
593 QString destFile = generateOutputFileName(filePath, currentOutputPath(), currentOutputIndx(), m_preferences->getSaveToSourcePath());
594 ui->editSource->setText(QDir::toNativeSeparators(filePath));
595 ui->editOutput->setText(QDir::toNativeSeparators(destFile));
598 else if(QObject::sender() == ui->buttonBrowseOutput)
600 QString selectedType = getFilterStr(currentOutputIndx());
601 QString filePath = QFileDialog::getSaveFileName(this, tr("Choose Output File"), currentOutputPath(true), getFilterLst(), &selectedType, QFileDialog::DontUseNativeDialog | QFileDialog::DontConfirmOverwrite);
603 if(!(filePath.isNull() || filePath.isEmpty()))
605 if(getFilterIdx(QFileInfo(filePath).suffix()) < 0)
608 QRegExp regExp("\\(\\*\\.(\\w+)\\)");
609 if(regExp.lastIndexIn(selectedType) >= 0)
611 tempIndex = getFilterIdx(regExp.cap(1));
615 tempIndex = m_recentlyUsed->filterIndex();
617 filePath = QString("%1.%2").arg(filePath, getFilterExt(tempIndex));
619 ui->editOutput->setText(QDir::toNativeSeparators(filePath));
624 void AddJobDialog::configurationChanged(void)
626 const OptionsModel* options = reinterpret_cast<const OptionsModel*>(ui->cbxTemplate->itemData(ui->cbxTemplate->currentIndex()).value<const void*>());
629 ui->cbxTemplate->blockSignals(true);
630 ui->cbxTemplate->insertItem(0, tr("<Unsaved Configuration>"), QVariant::fromValue<const void*>(NULL));
631 ui->cbxTemplate->setCurrentIndex(0);
632 ui->cbxTemplate->blockSignals(false);
636 void AddJobDialog::templateSelected(void)
638 const OptionsModel* options = reinterpret_cast<const OptionsModel*>(ui->cbxTemplate->itemData(ui->cbxTemplate->currentIndex()).value<const void*>());
641 qDebug("Loading options!");
643 restoreOptions(options);
647 encoderIndexChanged(ui->cbxEncoderType->currentIndex());
648 variantIndexChanged(ui->cbxEncoderVariant->currentIndex());
649 modeIndexChanged(ui->cbxRateControlMode->currentIndex());
652 void AddJobDialog::saveTemplateButtonClicked(void)
654 qDebug("Saving template");
655 QString name = tr("New Template");
658 while(OptionsModel::templateExists(name))
660 name = tr("New Template (%1)").arg(QString::number(n++));
663 OptionsModel *options = new OptionsModel(m_sysinfo);
664 saveOptions(options);
666 if(options->equals(m_defaults))
668 QMessageBox::warning (this, tr("Oups"), tr("<nobr>It makes no sense to save the default settings!</nobr>"));
669 ui->cbxTemplate->blockSignals(true);
670 ui->cbxTemplate->setCurrentIndex(0);
671 ui->cbxTemplate->blockSignals(false);
673 X264_DELETE(options);
677 for(int i = 0; i < ui->cbxTemplate->count(); i++)
679 const QString tempName = ui->cbxTemplate->itemText(i);
680 if(tempName.contains('<') || tempName.contains('>'))
684 const OptionsModel* test = reinterpret_cast<const OptionsModel*>(ui->cbxTemplate->itemData(i).value<const void*>());
687 if(options->equals(test))
689 QMessageBox::warning (this, tr("Oups"), tr("<nobr>There already is a template for the current settings!</nobr>"));
690 ui->cbxTemplate->blockSignals(true);
691 ui->cbxTemplate->setCurrentIndex(i);
692 ui->cbxTemplate->blockSignals(false);
694 X264_DELETE(options);
703 name = QInputDialog::getText(this, tr("Save Template"), tr("Please enter the name of the template:").leftJustified(144, ' '), QLineEdit::Normal, name, &ok).simplified();
706 X264_DELETE(options);
709 if(name.contains('<') || name.contains('>') || name.contains('\\') || name.contains('/') || name.contains('"'))
711 QMessageBox::warning (this, tr("Invalid Name"), tr("<nobr>Sorry, the name you have entered is invalid!</nobr>"));
712 while(name.contains('<')) name.remove('<');
713 while(name.contains('>')) name.remove('>');
714 while(name.contains('\\')) name.remove('\\');
715 while(name.contains('/')) name.remove('/');
716 while(name.contains('"')) name.remove('"');
717 name = name.simplified();
720 if(OptionsModel::templateExists(name))
722 int ret = QMessageBox::warning (this, tr("Already Exists"), tr("<nobr>A template of that name already exists! Overwrite?</nobr>"), QMessageBox::Yes | QMessageBox::No, QMessageBox::No);
723 if(ret != QMessageBox::Yes)
731 if(!OptionsModel::saveTemplate(options, name))
733 QMessageBox::critical(this, tr("Save Failed"), tr("Sorry, the template could not be saved!"));
734 X264_DELETE(options);
738 int index = ui->cbxTemplate->model()->rowCount();
739 ui->cbxTemplate->blockSignals(true);
740 for(int i = 0; i < ui->cbxTemplate->count(); i++)
742 if(ui->cbxTemplate->itemText(i).compare(name, Qt::CaseInsensitive) == 0)
744 index = -1; //Do not append new template
745 const OptionsModel *oldItem = reinterpret_cast<const OptionsModel*>(ui->cbxTemplate->itemData(i).value<const void*>());
746 ui->cbxTemplate->setItemData(i, QVariant::fromValue<const void*>(options));
747 ui->cbxTemplate->setCurrentIndex(i);
748 X264_DELETE(oldItem);
753 ui->cbxTemplate->insertItem(index, name, QVariant::fromValue<const void*>(options));
754 ui->cbxTemplate->setCurrentIndex(index);
756 ui->cbxTemplate->blockSignals(false);
761 void AddJobDialog::deleteTemplateButtonClicked(void)
763 const int index = ui->cbxTemplate->currentIndex();
764 QString name = ui->cbxTemplate->itemText(index);
766 if(name.contains('<') || name.contains('>') || name.contains('\\') || name.contains('/'))
768 QMessageBox::warning (this, tr("Invalid Item"), tr("Sorry, the selected item cannot be deleted!"));
772 int ret = QMessageBox::question (this, tr("Delete Template"), tr("<nobr>Do you really want to delete the selected template?</nobr>"), QMessageBox::Yes | QMessageBox::No, QMessageBox::No);
773 if(ret != QMessageBox::Yes)
779 OptionsModel::deleteTemplate(name);
780 const OptionsModel *item = reinterpret_cast<const OptionsModel*>(ui->cbxTemplate->itemData(index).value<const void*>());
781 ui->cbxTemplate->removeItem(index);
785 void AddJobDialog::editorActionTriggered(void)
788 if(QAction *action = dynamic_cast<QAction*>(QObject::sender()))
790 QLineEdit *lineEdit = reinterpret_cast<QLineEdit*>(action->data().value<void*>());
792 EditorDialog *editor = new EditorDialog(this);
793 editor->setEditText(lineEdit->text());
795 if(editor->exec() == QDialog::Accepted)
797 lineEdit->setText(editor->getEditText());
804 void AddJobDialog::copyActionTriggered(void)
806 if(QAction *action = dynamic_cast<QAction*>(QObject::sender()))
808 QClipboard *clipboard = QApplication::clipboard();
809 QLineEdit *lineEdit = reinterpret_cast<QLineEdit*>(action->data().value<void*>());
810 QString text = lineEdit->hasSelectedText() ? lineEdit->selectedText() : lineEdit->text();
811 clipboard->setText(text);
815 void AddJobDialog::pasteActionTriggered(void)
817 if(QAction *action = dynamic_cast<QAction*>(QObject::sender()))
819 QClipboard *clipboard = QApplication::clipboard();
820 QLineEdit *lineEdit = reinterpret_cast<QLineEdit*>(action->data().value<void*>());
821 QString text = clipboard->text();
822 if(!text.isEmpty()) lineEdit->setText(text);
826 ///////////////////////////////////////////////////////////////////////////////
828 ///////////////////////////////////////////////////////////////////////////////
830 QString AddJobDialog::sourceFile(void)
832 return QDir::fromNativeSeparators(ui->editSource->text());
835 QString AddJobDialog::outputFile(void)
837 return QDir::fromNativeSeparators(ui->editOutput->text());
840 bool AddJobDialog::runImmediately(void)
842 return ui->checkBoxRun->isChecked();
845 bool AddJobDialog::applyToAll(void)
847 return ui->checkBoxApplyToAll->isChecked();
850 void AddJobDialog::setRunImmediately(bool run)
852 ui->checkBoxRun->setChecked(run);
855 void AddJobDialog::setSourceFile(const QString &path)
857 ui->editSource->setText(QDir::toNativeSeparators(path));
860 void AddJobDialog::setOutputFile(const QString &path)
862 ui->editOutput->setText(QDir::toNativeSeparators(path));}
864 void AddJobDialog::setSourceEditable(const bool editable)
866 ui->buttonBrowseSource->setEnabled(editable);
869 void AddJobDialog::setApplyToAllVisible(const bool visible)
871 ui->checkBoxApplyToAll->setVisible(visible);
874 ///////////////////////////////////////////////////////////////////////////////
876 ///////////////////////////////////////////////////////////////////////////////
878 void AddJobDialog::loadTemplateList(void)
880 ui->cbxTemplate->addItem(tr("<Default>"), QVariant::fromValue<const void*>(m_defaults));
881 ui->cbxTemplate->setCurrentIndex(0);
883 QMap<QString, OptionsModel*> templates = OptionsModel::loadAllTemplates(m_sysinfo);
884 QStringList templateNames = templates.keys();
885 templateNames.sort();
887 for(QStringList::ConstIterator current = templateNames.constBegin(); current != templateNames.constEnd(); current++)
889 OptionsModel *currentTemplate = templates.take(*current);
890 ui->cbxTemplate->addItem(*current, QVariant::fromValue<const void*>(currentTemplate));
891 if(currentTemplate->equals(m_options))
893 ui->cbxTemplate->setCurrentIndex(ui->cbxTemplate->count() - 1);
897 if((ui->cbxTemplate->currentIndex() == 0) && (!m_options->equals(m_defaults)))
899 qWarning("Not the default -> recently used!");
900 ui->cbxTemplate->insertItem(1, tr("<Recently Used>"), QVariant::fromValue<const void*>(m_options));
901 ui->cbxTemplate->setCurrentIndex(1);
905 void AddJobDialog::updateComboBox(QComboBox *cbox, const QString &text)
908 if(QAbstractItemModel *model = cbox->model())
910 for(int i = 0; i < cbox->model()->rowCount(); i++)
912 if(model->data(model->index(i, 0, QModelIndex())).toString().compare(text, Qt::CaseInsensitive) == 0)
919 cbox->setCurrentIndex(index);
922 void AddJobDialog::restoreOptions(const OptionsModel *options)
926 ui->cbxEncoderType->setCurrentIndex(options->encType());
927 ui->cbxEncoderArch->setCurrentIndex(options->encArch());
928 ui->cbxEncoderVariant->setCurrentIndex(options->encVariant());
929 ui->cbxRateControlMode->setCurrentIndex(options->rcMode());
930 ui->spinQuantizer->setValue(options->quantizer());
931 ui->spinBitrate->setValue(options->bitrate());
932 updateComboBox(ui->cbxPreset, options->preset());
933 updateComboBox(ui->cbxTuning, options->tune());
934 updateComboBox(ui->cbxProfile, options->profile());
935 ui->editCustomX264Params->setText(options->customEncParams());
936 ui->editCustomAvs2YUVParams->setText(options->customAvs2YUV());
938 BLOCK_SIGNALS(false);
941 void AddJobDialog::saveOptions(OptionsModel *options)
943 options->setEncType(static_cast<OptionsModel::EncType>(ui->cbxEncoderType->currentIndex()));
944 options->setEncArch(static_cast<OptionsModel::EncArch>(ui->cbxEncoderArch->currentIndex()));
945 options->setEncVariant(static_cast<OptionsModel::EncVariant>(ui->cbxEncoderVariant->currentIndex()));
946 options->setRCMode(static_cast<OptionsModel::RCMode>(ui->cbxRateControlMode->currentIndex()));
947 options->setQuantizer(ui->spinQuantizer->value());
948 options->setBitrate(ui->spinBitrate->value());
949 options->setPreset(ui->cbxPreset->model()->data(ui->cbxPreset->model()->index(ui->cbxPreset->currentIndex(), 0)).toString());
950 options->setTune(ui->cbxTuning->model()->data(ui->cbxTuning->model()->index(ui->cbxTuning->currentIndex(), 0)).toString());
951 options->setProfile(ui->cbxProfile->model()->data(ui->cbxProfile->model()->index(ui->cbxProfile->currentIndex(), 0)).toString());
952 options->setCustomEncParams(ui->editCustomX264Params->hasAcceptableInput() ? ui->editCustomX264Params->text().simplified() : QString());
953 options->setCustomAvs2YUV(ui->editCustomAvs2YUVParams->hasAcceptableInput() ? ui->editCustomAvs2YUVParams->text().simplified() : QString());
956 QString AddJobDialog::currentSourcePath(const bool bWithName)
958 QString path = m_recentlyUsed->sourceDirectory();
959 QString currentSourceFile = this->sourceFile();
961 if(!currentSourceFile.isEmpty())
963 QString currentSourceDir = QFileInfo(currentSourceFile).absolutePath();
964 if(VALID_DIR(currentSourceDir))
966 path = currentSourceDir;
970 path.append("/").append(QFileInfo(currentSourceFile).fileName());
977 QString AddJobDialog::currentOutputPath(const bool bWithName)
979 QString path = m_recentlyUsed->outputDirectory();
980 QString currentOutputFile = this->outputFile();
982 if(!currentOutputFile.isEmpty())
984 QString currentOutputDir = QFileInfo(currentOutputFile).absolutePath();
985 if(VALID_DIR(currentOutputDir))
987 path = currentOutputDir;
991 path.append("/").append(QFileInfo(currentOutputFile).fileName());
998 int AddJobDialog::currentOutputIndx(void)
1000 if(ui->cbxEncoderType->currentIndex() == OptionsModel::EncType_X265)
1002 return ARRAY_SIZE(X264_FILE_TYPE_FILTERS) - 1;
1005 int index = m_recentlyUsed->filterIndex();
1006 const QString currentOutputFile = this->outputFile();
1008 if(!currentOutputFile.isEmpty())
1010 const QString currentOutputExtn = QFileInfo(currentOutputFile).suffix();
1011 const int tempIndex = getFilterIdx(currentOutputExtn);
1021 ///////////////////////////////////////////////////////////////////////////////
1023 ///////////////////////////////////////////////////////////////////////////////
1025 QString AddJobDialog::generateOutputFileName(const QString &sourceFilePath, const QString &destinationDirectory, const int filterIndex, const bool saveToSourceDir)
1027 QString name = QFileInfo(sourceFilePath).completeBaseName();
1028 QString path = saveToSourceDir ? QFileInfo(sourceFilePath).canonicalPath() : destinationDirectory;
1029 QString fext = getFilterExt(filterIndex);
1031 if(!VALID_DIR(path))
1033 RecentlyUsed defaults;
1034 path = defaults.outputDirectory();
1037 QString outPath = QString("%1/%2.%3").arg(path, name, fext);
1040 while(QFileInfo(outPath).exists())
1042 outPath = QString("%1/%2 (%3).%4").arg(path, name, QString::number(n++), fext);
1048 /* ------------------------------------------------------------------------- */
1050 QString AddJobDialog::getFilterExt(const int filterIndex)
1052 const int count = ARRAY_SIZE(X264_FILE_TYPE_FILTERS);
1054 if((filterIndex >= 0) && (filterIndex < count))
1056 return QString::fromLatin1(X264_FILE_TYPE_FILTERS[filterIndex].pcExt);
1059 return QString::fromLatin1(X264_FILE_TYPE_FILTERS[0].pcExt);
1062 int AddJobDialog::getFilterIdx(const QString &fileExt)
1064 const int count = ARRAY_SIZE(X264_FILE_TYPE_FILTERS);
1066 for(int i = 0; i < count; i++)
1068 if(fileExt.compare(QString::fromLatin1(X264_FILE_TYPE_FILTERS[i].pcExt), Qt::CaseInsensitive) == 0)
1077 QString AddJobDialog::getFilterStr(const int filterIndex)
1079 const int count = ARRAY_SIZE(X264_FILE_TYPE_FILTERS);
1081 if((filterIndex >= 0) && (filterIndex < count))
1083 return QString("%1 (*.%2)").arg(QString::fromLatin1(X264_FILE_TYPE_FILTERS[filterIndex].pcStr), QString::fromLatin1(X264_FILE_TYPE_FILTERS[filterIndex].pcExt));
1086 return QString("%1 (*.%2)").arg(QString::fromLatin1(X264_FILE_TYPE_FILTERS[0].pcStr), QString::fromLatin1(X264_FILE_TYPE_FILTERS[0].pcExt));
1089 QString AddJobDialog::getFilterLst(void)
1091 QStringList filters;
1092 const int count = ARRAY_SIZE(X264_FILE_TYPE_FILTERS);
1094 for(int i = 0; i < count; i++)
1096 filters << QString("%1 (*.%2)").arg(QString::fromLatin1(X264_FILE_TYPE_FILTERS[i].pcStr), QString::fromLatin1(X264_FILE_TYPE_FILTERS[i].pcExt));
1099 return filters.join(";;");
1102 QString AddJobDialog::getInputFilterLst(void)
1111 {"Avisynth Scripts", "avs"},
1112 {"VapourSynth Scripts", "vpy"},
1113 {"Matroska Files", "mkv"},
1114 {"MPEG-4 Part 14 Container", "mp4"},
1115 {"Audio Video Interleaved", "avi"},
1116 {"Flash Video", "flv"},
1117 {"YUV4MPEG2 Stream", "y4m"},
1118 {"Uncompresses YUV Data", "yuv"},
1121 const int count = ARRAY_SIZE(s_filters);
1124 for(size_t index = 0; index < count; index++)
1127 allTypes += QString((index > 0) ? " *.%1" : "*.%1").arg(QString::fromLatin1(s_filters[index].fext));
1130 QStringList filters;
1131 filters << QString("All supported files (%1)").arg(allTypes);
1133 for(size_t index = 0; index < count; index++)
1135 filters << QString("%1 (*.%2)").arg(QString::fromLatin1(s_filters[index].name), QString::fromLatin1(s_filters[index].fext));
1138 filters << QString("All files (*.*)");
1139 return filters.join(";;");
1142 const AbstractEncoderInfo& AddJobDialog::getEncoderInfo(const int &encoder)
1146 case OptionsModel::EncType_X264:
1147 return X264Encoder::getEncoderInfo();
1148 case OptionsModel::EncType_X265:
1149 return X265Encoder::getEncoderInfo();
1152 THROW("Unsupported encoder type!");