renderwidget.cpp 124 KB
Newer Older
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
/***************************************************************************
 *   Copyright (C) 2008 by Jean-Baptiste Mardelle (jb@kdenlive.org)        *
 *                                                                         *
 *   This program is free software; you can redistribute it and/or modify  *
 *   it under the terms of the GNU General Public License as published by  *
 *   the Free Software Foundation; either version 2 of the License, or     *
 *   (at your option) any later version.                                   *
 *                                                                         *
 *   This program is distributed in the hope that it will be useful,       *
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of        *
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *
 *   GNU General Public License for more details.                          *
 *                                                                         *
 *   You should have received a copy of the GNU General Public License     *
 *   along with this program; if not, write to the                         *
 *   Free Software Foundation, Inc.,                                       *
 *   51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA          *
 ***************************************************************************/

20
#include "renderwidget.h"
Nicolas Carion's avatar
linting  
Nicolas Carion committed
21
#include "core.h"
Nicolas Carion's avatar
Nicolas Carion committed
22
#include "dialogs/profilesdialog.h"
23
#include "kdenlivesettings.h"
Nicolas Carion's avatar
Nicolas Carion committed
24 25
#include "profiles/profilemodel.hpp"
#include "profiles/profilerepository.hpp"
26
#include "timecode.h"
Nicolas Carion's avatar
Nicolas Carion committed
27
#include "ui_saveprofile_ui.h"
28
#include "utils/KoIconUtils.h"
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
29

30
#include "klocalizedstring.h"
31
#include <KColorScheme>
32
#include <KIO/DesktopExecParser>
Nicolas Carion's avatar
Nicolas Carion committed
33 34 35 36
#include <KMessageBox>
#include <KMimeTypeTrader>
#include <KNotification>
#include <KRun>
37
#include <kio_version.h>
Nicolas Carion's avatar
Nicolas Carion committed
38
#include <knotifications_version.h>
39

Laurent Montel's avatar
Laurent Montel committed
40
#include "kdenlive_debug.h"
Nicolas Carion's avatar
Nicolas Carion committed
41 42
#include <QDBusConnectionInterface>
#include <QDir>
43 44 45
#include <QDomDocument>
#include <QHeaderView>
#include <QInputDialog>
Nicolas Carion's avatar
Nicolas Carion committed
46 47
#include <QKeyEvent>
#include <QMimeDatabase>
48
#include <QProcess>
49
#include <QStandardPaths>
Nicolas Carion's avatar
Nicolas Carion committed
50 51 52
#include <QThread>
#include <QTimer>
#include <QTreeWidgetItem>
53
#include <QJsonObject>
54
#include <QJsonArray>
Nicolas Carion's avatar
Nicolas Carion committed
55 56
#include <qglobal.h>
#include <qstring.h>
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
57

58 59 60 61 62
#ifdef KF5_USE_PURPOSE
#include <Purpose/AlternativesModel>
#include <PurposeWidgets/Menu>
#endif

Vincent Pinon's avatar
Vincent Pinon committed
63
#include <locale>
64 65 66
#ifdef Q_OS_MAC
#include <xlocale.h>
#endif
67

68
// Render profiles roles
Nicolas Carion's avatar
Nicolas Carion committed
69 70 71 72 73 74 75 76 77 78 79 80 81 82 83
enum {
    GroupRole = Qt::UserRole,
    ExtensionRole,
    StandardRole,
    RenderRole,
    ParamsRole,
    EditableRole,
    ExtraRole,
    BitratesRole,
    DefaultBitrateRole,
    AudioBitratesRole,
    DefaultAudioBitrateRole,
    SpeedsRole,
    ErrorRole
};
84

85 86 87
// Render job roles
const int ParametersRole = Qt::UserRole + 1;
const int TimeRole = Qt::UserRole + 2;
88 89
const int ProgressRole = Qt::UserRole + 3;
const int ExtraInfoRole = Qt::UserRole + 5;
90

91 92
const int DirectRenderType = QTreeWidgetItem::Type;
const int ScriptRenderType = QTreeWidgetItem::UserType;
93

94
// Running job status
Nicolas Carion's avatar
Nicolas Carion committed
95
enum JOBSTATUS { WAITINGJOB = 0, STARTINGJOB, RUNNINGJOB, FINISHEDJOB, FAILEDJOB, ABORTEDJOB };
96

97
#ifdef Q_OS_WIN
98
const QLatin1String ScriptFormat(".bat");
Nicolas Carion's avatar
format  
Nicolas Carion committed
99 100 101 102 103 104 105 106
QString ScriptSetVar(const QString name, const QString value)
{
    return QString("set ") + name + "=" + value;
}
QString ScriptGetVar(const QString varName)
{
    return QString('%') + varName + '%';
}
107 108
#else
const QLatin1String ScriptFormat(".sh");
Nicolas Carion's avatar
linting  
Nicolas Carion committed
109
QString ScriptSetVar(const QString &name, const QString &value)
Nicolas Carion's avatar
format  
Nicolas Carion committed
110 111 112
{
    return name + "=\"" + value + '\"';
}
Nicolas Carion's avatar
linting  
Nicolas Carion committed
113
QString ScriptGetVar(const QString &varName)
Nicolas Carion's avatar
format  
Nicolas Carion committed
114 115 116
{
    return QString('$') + varName;
}
117 118
#endif

119 120 121
static QStringList acodecsList;
static QStringList vcodecsList;
static QStringList supportedFormats;
122

123 124 125
RenderJobItem::RenderJobItem(QTreeWidget *parent, const QStringList &strings, int type)
    : QTreeWidgetItem(parent, strings, type)
    , m_status(-1)
126
{
127
    setSizeHint(1, QSize(parent->columnWidth(1), parent->fontMetrics().height() * 3));
128 129 130 131 132
    setStatus(WAITINGJOB);
}

void RenderJobItem::setStatus(int status)
{
133
    if (m_status == status) {
Laurent Montel's avatar
Laurent Montel committed
134
        return;
135
    }
136 137
    m_status = status;
    switch (status) {
138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157
    case WAITINGJOB:
        setIcon(0, KoIconUtils::themedIcon(QStringLiteral("media-playback-pause")));
        setData(1, Qt::UserRole, i18n("Waiting..."));
        break;
    case FINISHEDJOB:
        setData(1, Qt::UserRole, i18n("Rendering finished"));
        setIcon(0, KoIconUtils::themedIcon(QStringLiteral("dialog-ok")));
        setData(1, ProgressRole, 100);
        break;
    case FAILEDJOB:
        setData(1, Qt::UserRole, i18n("Rendering crashed"));
        setIcon(0, KoIconUtils::themedIcon(QStringLiteral("dialog-close")));
        setData(1, ProgressRole, 100);
        break;
    case ABORTEDJOB:
        setData(1, Qt::UserRole, i18n("Rendering aborted"));
        setIcon(0, KoIconUtils::themedIcon(QStringLiteral("dialog-cancel")));
        setData(1, ProgressRole, 100);
    default:
        break;
158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174
    }
}

int RenderJobItem::status() const
{
    return m_status;
}

void RenderJobItem::setMetadata(const QString &data)
{
    m_data = data;
}

const QString RenderJobItem::metadata() const
{
    return m_data;
}
175

176
RenderWidget::RenderWidget(const QString &projectfolder, bool enableProxy, QWidget *parent)
177 178 179
    : QDialog(parent)
    , m_projectFolder(projectfolder)
    , m_blockProcessing(false)
180
{
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
181
    m_view.setupUi(this);
182 183
    int size = style()->pixelMetric(QStyle::PM_SmallIconSize);
    QSize iconSize(size, size);
184

185
    setWindowTitle(i18n("Rendering"));
186 187 188 189
    m_view.buttonDelete->setIconSize(iconSize);
    m_view.buttonEdit->setIconSize(iconSize);
    m_view.buttonSave->setIconSize(iconSize);
    m_view.buttonFavorite->setIconSize(iconSize);
190

191
    m_view.buttonDelete->setIcon(KoIconUtils::themedIcon(QStringLiteral("trash-empty")));
192 193 194
    m_view.buttonDelete->setToolTip(i18n("Delete profile"));
    m_view.buttonDelete->setEnabled(false);

195
    m_view.buttonEdit->setIcon(KoIconUtils::themedIcon(QStringLiteral("document-edit")));
196 197 198
    m_view.buttonEdit->setToolTip(i18n("Edit profile"));
    m_view.buttonEdit->setEnabled(false);

199
    m_view.buttonSave->setIcon(KoIconUtils::themedIcon(QStringLiteral("document-new")));
200 201
    m_view.buttonSave->setToolTip(i18n("Create new profile"));

202
    m_view.hide_log->setIcon(KoIconUtils::themedIcon(QStringLiteral("go-down")));
203

204
    m_view.buttonFavorite->setIcon(KoIconUtils::themedIcon(QStringLiteral("favorite")));
205
    m_view.buttonFavorite->setToolTip(i18n("Copy profile to favorites"));
206

207
    m_view.out_file->button()->setToolTip(i18n("Select output destination"));
208
    m_view.advanced_params->setMaximumHeight(QFontMetrics(font()).lineSpacing() * 5);
209

210
    m_view.optionsGroup->setVisible(m_view.options->isChecked());
Laurent Montel's avatar
Laurent Montel committed
211
    connect(m_view.options, &QAbstractButton::toggled, m_view.optionsGroup, &QWidget::setVisible);
212
    m_view.videoLabel->setVisible(m_view.options->isChecked());
Laurent Montel's avatar
Laurent Montel committed
213
    connect(m_view.options, &QAbstractButton::toggled, m_view.videoLabel, &QWidget::setVisible);
214
    m_view.video->setVisible(m_view.options->isChecked());
Laurent Montel's avatar
Laurent Montel committed
215
    connect(m_view.options, &QAbstractButton::toggled, m_view.video, &QWidget::setVisible);
216
    m_view.audioLabel->setVisible(m_view.options->isChecked());
Laurent Montel's avatar
Laurent Montel committed
217
    connect(m_view.options, &QAbstractButton::toggled, m_view.audioLabel, &QWidget::setVisible);
218
    m_view.audio->setVisible(m_view.options->isChecked());
Laurent Montel's avatar
Laurent Montel committed
219 220
    connect(m_view.options, &QAbstractButton::toggled, m_view.audio, &QWidget::setVisible);
    connect(m_view.quality, &QAbstractSlider::valueChanged, this, &RenderWidget::adjustAVQualities);
221
    connect(m_view.video, static_cast<void (QSpinBox::*)(int)>(&QSpinBox::valueChanged), this, &RenderWidget::adjustQuality);
Laurent Montel's avatar
Laurent Montel committed
222
    connect(m_view.speed, &QAbstractSlider::valueChanged, this, &RenderWidget::adjustSpeed);
223

224 225
    m_view.buttonRender->setEnabled(false);
    m_view.buttonGenerateScript->setEnabled(false);
226
    setRescaleEnabled(false);
227 228 229 230 231 232 233
    m_view.guides_box->setVisible(false);
    m_view.open_dvd->setVisible(false);
    m_view.create_chapter->setVisible(false);
    m_view.open_browser->setVisible(false);
    m_view.error_box->setVisible(false);
    m_view.tc_type->setEnabled(false);
    m_view.checkTwoPass->setEnabled(false);
234
    m_view.proxy_render->setHidden(!enableProxy);
235
    connect(m_view.proxy_render, &QCheckBox::toggled, this, &RenderWidget::slotProxyWarn);
236 237
    KColorScheme scheme(palette().currentColorGroup(), KColorScheme::Window, KSharedConfig::openConfig(KdenliveSettings::colortheme()));
    QColor bg = scheme.background(KColorScheme::NegativeBackground).color();
Nicolas Carion's avatar
Nicolas Carion committed
238 239
    m_view.errorBox->setStyleSheet(
        QStringLiteral("QGroupBox { background-color: rgb(%1, %2, %3); border-radius: 5px;}; ").arg(bg.red()).arg(bg.green()).arg(bg.blue()));
240
    int height = QFontInfo(font()).pixelSize();
241
    m_view.errorIcon->setPixmap(KoIconUtils::themedIcon(QStringLiteral("dialog-warning")).pixmap(height, height));
242 243 244
    m_view.errorBox->setHidden(true);

    m_infoMessage = new KMessageWidget;
245
    m_view.info->addWidget(m_infoMessage);
246 247 248
    m_infoMessage->setCloseButtonVisible(false);
    m_infoMessage->hide();

249 250 251 252 253
    m_jobInfoMessage = new KMessageWidget;
    m_view.jobInfo->addWidget(m_jobInfoMessage);
    m_jobInfoMessage->setCloseButtonVisible(false);
    m_jobInfoMessage->hide();

254
    m_view.encoder_threads->setMinimum(0);
255
    m_view.encoder_threads->setMaximum(QThread::idealThreadCount());
256
    m_view.encoder_threads->setToolTip(i18n("Encoding threads (0 is automatic)"));
257
    m_view.encoder_threads->setValue(KdenliveSettings::encodethreads());
258
    connect(m_view.encoder_threads, static_cast<void (QSpinBox::*)(int)>(&QSpinBox::valueChanged), this, &RenderWidget::slotUpdateEncodeThreads);
259

260
    m_view.rescale_keep->setChecked(KdenliveSettings::rescalekeepratio());
261 262
    connect(m_view.rescale_width, static_cast<void (QSpinBox::*)(int)>(&QSpinBox::valueChanged), this, &RenderWidget::slotUpdateRescaleWidth);
    connect(m_view.rescale_height, static_cast<void (QSpinBox::*)(int)>(&QSpinBox::valueChanged), this, &RenderWidget::slotUpdateRescaleHeight);
263
    m_view.rescale_keep->setIcon(KoIconUtils::themedIcon(QStringLiteral("edit-link")));
264
    m_view.rescale_keep->setToolTip(i18n("Preserve aspect ratio"));
Laurent Montel's avatar
Laurent Montel committed
265
    connect(m_view.rescale_keep, &QAbstractButton::clicked, this, &RenderWidget::slotSwitchAspectRatio);
266

267
    connect(m_view.buttonRender, SIGNAL(clicked()), this, SLOT(slotPrepareExport()));
Laurent Montel's avatar
Laurent Montel committed
268
    connect(m_view.buttonGenerateScript, &QAbstractButton::clicked, this, &RenderWidget::slotGenerateScript);
269

270 271 272 273
    m_view.abort_job->setEnabled(false);
    m_view.start_script->setEnabled(false);
    m_view.delete_script->setEnabled(false);

Laurent Montel's avatar
Laurent Montel committed
274
    connect(m_view.export_audio, &QCheckBox::stateChanged, this, &RenderWidget::slotUpdateAudioLabel);
275 276
    m_view.export_audio->setCheckState(Qt::PartiallyChecked);

Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
277
    checkCodecs();
278
    parseProfiles();
279
    parseScriptFiles();
280 281
    m_view.running_jobs->setUniformRowHeights(false);
    m_view.scripts_list->setUniformRowHeights(false);
Laurent Montel's avatar
Laurent Montel committed
282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301
    connect(m_view.start_script, &QAbstractButton::clicked, this, &RenderWidget::slotStartScript);
    connect(m_view.delete_script, &QAbstractButton::clicked, this, &RenderWidget::slotDeleteScript);
    connect(m_view.scripts_list, &QTreeWidget::itemSelectionChanged, this, &RenderWidget::slotCheckScript);
    connect(m_view.running_jobs, &QTreeWidget::itemSelectionChanged, this, &RenderWidget::slotCheckJob);
    connect(m_view.running_jobs, &QTreeWidget::itemDoubleClicked, this, &RenderWidget::slotPlayRendering);

    connect(m_view.buttonSave, &QAbstractButton::clicked, this, &RenderWidget::slotSaveProfile);
    connect(m_view.buttonEdit, &QAbstractButton::clicked, this, &RenderWidget::slotEditProfile);
    connect(m_view.buttonDelete, &QAbstractButton::clicked, this, &RenderWidget::slotDeleteProfile);
    connect(m_view.buttonFavorite, &QAbstractButton::clicked, this, &RenderWidget::slotCopyToFavorites);

    connect(m_view.abort_job, &QAbstractButton::clicked, this, &RenderWidget::slotAbortCurrentJob);
    connect(m_view.start_job, &QAbstractButton::clicked, this, &RenderWidget::slotStartCurrentJob);
    connect(m_view.clean_up, &QAbstractButton::clicked, this, &RenderWidget::slotCLeanUpJobs);
    connect(m_view.hide_log, &QAbstractButton::clicked, this, &RenderWidget::slotHideLog);

    connect(m_view.buttonClose, &QAbstractButton::clicked, this, &QWidget::hide);
    connect(m_view.buttonClose2, &QAbstractButton::clicked, this, &QWidget::hide);
    connect(m_view.buttonClose3, &QAbstractButton::clicked, this, &QWidget::hide);
    connect(m_view.rescale, &QAbstractButton::toggled, this, &RenderWidget::setRescaleEnabled);
Laurent Montel's avatar
Laurent Montel committed
302
    connect(m_view.out_file, SIGNAL(textChanged(QString)), this, SLOT(slotUpdateButtons()));
303
    connect(m_view.out_file, SIGNAL(urlSelected(QUrl)), this, SLOT(slotUpdateButtons(QUrl)));
304

Laurent Montel's avatar
Laurent Montel committed
305 306
    connect(m_view.formats, &QTreeWidget::currentItemChanged, this, &RenderWidget::refreshParams);
    connect(m_view.formats, &QTreeWidget::itemDoubleClicked, this, &RenderWidget::slotEditItem);
307

Laurent Montel's avatar
Laurent Montel committed
308 309 310
    connect(m_view.render_guide, &QAbstractButton::clicked, this, &RenderWidget::slotUpdateGuideBox);
    connect(m_view.render_zone, &QAbstractButton::clicked, this, &RenderWidget::slotUpdateGuideBox);
    connect(m_view.render_full, &QAbstractButton::clicked, this, &RenderWidget::slotUpdateGuideBox);
311 312 313 314

    connect(m_view.guide_end, SIGNAL(activated(int)), this, SLOT(slotCheckStartGuidePosition()));
    connect(m_view.guide_start, SIGNAL(activated(int)), this, SLOT(slotCheckEndGuidePosition()));

Laurent Montel's avatar
Laurent Montel committed
315
    connect(m_view.tc_overlay, &QAbstractButton::toggled, m_view.tc_type, &QWidget::setEnabled);
316

Nicolas Carion's avatar
Nicolas Carion committed
317 318
    // m_view.splitter->setStretchFactor(1, 5);
    // m_view.splitter->setStretchFactor(0, 2);
319

320
    m_view.out_file->setMode(KFile::File);
Nicolas Carion's avatar
Nicolas Carion committed
321
#if KIO_VERSION >= QT_VERSION_CHECK(5, 33, 0)
322
    m_view.out_file->setAcceptMode(QFileDialog::AcceptSave);
323
#elif !defined(KIOWIDGETS_DEPRECATED)
324
    m_view.out_file->fileDialog()->setAcceptMode(QFileDialog::AcceptSave);
325 326
#endif

327
    m_view.out_file->setFocusPolicy(Qt::ClickFocus);
328

329
    m_jobsDelegate = new RenderViewDelegate(this);
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
330
    m_view.running_jobs->setHeaderLabels(QStringList() << QString() << i18n("File"));
331
    m_view.running_jobs->setItemDelegate(m_jobsDelegate);
332

333
    QHeaderView *header = m_view.running_jobs->header();
334
    header->setSectionResizeMode(0, QHeaderView::Fixed);
335
    header->resizeSection(0, size + 4);
336
    header->setSectionResizeMode(1, QHeaderView::Interactive);
337

338
    m_view.scripts_list->setHeaderLabels(QStringList() << QString() << i18n("Script Files"));
339 340
    m_scriptsDelegate = new RenderViewDelegate(this);
    m_view.scripts_list->setItemDelegate(m_scriptsDelegate);
341
    header = m_view.scripts_list->header();
342
    header->setSectionResizeMode(0, QHeaderView::Fixed);
343
    header->resizeSection(0, size + 4);
344

Nicolas Carion's avatar
Nicolas Carion committed
345
// Find path for Kdenlive renderer
346 347 348
#ifdef Q_OS_WIN
    m_renderer = QCoreApplication::applicationDirPath() + QStringLiteral("/kdenlive_render.exe");
#else
349
    m_renderer = QCoreApplication::applicationDirPath() + QStringLiteral("/kdenlive_render");
350
#endif
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
351
    if (!QFile::exists(m_renderer)) {
352
        m_renderer = QStandardPaths::findExecutable(QStringLiteral("kdenlive_render"));
353
        if (m_renderer.isEmpty()) {
354
            m_renderer = QStringLiteral("kdenlive_render");
355
        }
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
356 357
    }

358
    QDBusConnectionInterface *interface = QDBusConnection::sessionBus().interface();
Nicolas Carion's avatar
Nicolas Carion committed
359 360
    if ((interface == nullptr) ||
        (!interface->isServiceRegistered(QStringLiteral("org.kde.ksmserver")) && !interface->isServiceRegistered(QStringLiteral("org.gnome.SessionManager")))) {
361
        m_view.shutdown->setEnabled(false);
362
    }
363 364 365 366

#ifdef KF5_USE_PURPOSE
    m_shareMenu = new Purpose::Menu();
    m_view.shareButton->setMenu(m_shareMenu);
367
    m_view.shareButton->setIcon( KoIconUtils::themedIcon(QStringLiteral("document-share")));
368
    connect(m_shareMenu, &Purpose::Menu::finished, this, &RenderWidget::slotShareActionFinished);
369 370 371
#else
    m_view.shareButton->setEnabled(false);
#endif
372
    refreshView();
373
    focusFirstVisibleItem();
374
    adjustSize();
375 376
}

377 378
void RenderWidget::slotShareActionFinished(const QJsonObject &output, int error, const QString &message)
{
379
#ifdef KF5_USE_PURPOSE
380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396
    m_jobInfoMessage->hide();
    if (error) {
        KMessageBox::error(this, i18n("There was a problem sharing the document: %1", message),
                           i18n("Share"));
    } else {
        const QString url = output["url"].toString();
        if (url.isEmpty()) {
            m_jobInfoMessage->setMessageType(KMessageWidget::Positive);
            m_jobInfoMessage->setText(i18n("Document shared successfully"));
            m_jobInfoMessage->show();
        } else {
            KMessageBox::information(this, i18n("You can find the shared document at: <a href=\"%1\">%1</a>", url),
                                     i18n("Share"), QString(),
                                     KMessageBox::Notify | KMessageBox::AllowLink);
        }
    }
#endif
397
}
398

399 400 401 402
QSize RenderWidget::sizeHint() const
{
    // Make sure the widget has minimum size on opening
    return QSize(200, 200);
403 404
}

405 406 407 408 409 410 411 412
RenderWidget::~RenderWidget()
{
    m_view.running_jobs->blockSignals(true);
    m_view.scripts_list->blockSignals(true);
    m_view.running_jobs->clear();
    m_view.scripts_list->clear();
    delete m_jobsDelegate;
    delete m_scriptsDelegate;
413
    delete m_infoMessage;
414
    delete m_jobInfoMessage;
415 416
}

417
void RenderWidget::slotEditItem(QTreeWidgetItem *item)
418
{
Laurent Montel's avatar
Laurent Montel committed
419
    if (item->parent() == nullptr) {
420 421 422
        // This is a top level item - group - don't edit
        return;
    }
423
    const QString edit = item->data(0, EditableRole).toString();
424
    if (edit.isEmpty() || !edit.endsWith(QLatin1String("customprofiles.xml"))) {
Laurent Montel's avatar
Laurent Montel committed
425
        slotSaveProfile();
426 427 428
    } else {
        slotEditProfile();
    }
429 430
}

431 432
void RenderWidget::showInfoPanel()
{
433 434 435 436 437 438 439
    if (m_view.advanced_params->isVisible()) {
        m_view.advanced_params->setVisible(false);
        KdenliveSettings::setShowrenderparams(false);
    } else {
        m_view.advanced_params->setVisible(true);
        KdenliveSettings::setShowrenderparams(true);
    }
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
440 441
}

442
void RenderWidget::setDocumentPath(const QString &path)
443
{
Nicolas Carion's avatar
Nicolas Carion committed
444 445
    if (m_view.out_file->url().adjusted(QUrl::RemoveFilename).toLocalFile() ==
        QUrl::fromLocalFile(m_projectFolder).adjusted(QUrl::RemoveFilename).toLocalFile()) {
446
        const QString fileName = m_view.out_file->url().fileName();
447
        m_view.out_file->setUrl(QUrl::fromLocalFile(QDir(path).absoluteFilePath(fileName)));
448
    }
449
    m_projectFolder = QUrl::fromLocalFile(path).adjusted(QUrl::NormalizePathSegments | QUrl::StripTrailingSlash).toLocalFile() + QDir::separator();
450
    parseScriptFiles();
451 452
}

453 454
void RenderWidget::slotUpdateGuideBox()
{
455 456 457
    m_view.guides_box->setVisible(m_view.render_guide->isChecked());
}

458 459
void RenderWidget::slotCheckStartGuidePosition()
{
460
    if (m_view.guide_start->currentIndex() > m_view.guide_end->currentIndex()) {
461
        m_view.guide_start->setCurrentIndex(m_view.guide_end->currentIndex());
462
    }
463 464
}

465 466
void RenderWidget::slotCheckEndGuidePosition()
{
467
    if (m_view.guide_end->currentIndex() < m_view.guide_start->currentIndex()) {
468
        m_view.guide_end->setCurrentIndex(m_view.guide_start->currentIndex());
469
    }
470 471
}

472
void RenderWidget::setGuides(const QList<CommentedTime> &guidesList, double duration)
473
{
474 475
    m_view.guide_start->clear();
    m_view.guide_end->clear();
476
    if (!guidesList.isEmpty()) {
477
        m_view.guide_start->addItem(i18n("Beginning"), "0");
478
        m_view.render_guide->setEnabled(true);
479 480 481 482 483
        m_view.create_chapter->setEnabled(true);
    } else {
        m_view.render_guide->setEnabled(false);
        m_view.create_chapter->setEnabled(false);
    }
484
    double fps = pCore->getCurrentProfile()->fps();
485 486 487
    for (int i = 0; i < guidesList.count(); i++) {
        CommentedTime c = guidesList.at(i);
        GenTime pos = c.time();
488
        const QString guidePos = Timecode::getStringTimecode(pos.frames(fps), fps);
489 490
        m_view.guide_start->addItem(c.comment() + QLatin1Char('/') + guidePos, pos.seconds());
        m_view.guide_end->addItem(c.comment() + QLatin1Char('/') + guidePos, pos.seconds());
491
    }
492
    if (!guidesList.isEmpty()) {
493
        m_view.guide_end->addItem(i18n("End"), QString::number(duration));
494
    }
495 496
}

497
/**
Simon Eugster's avatar
Simon Eugster committed
498 499 500
 * Will be called when the user selects an output file via the file dialog.
 * File extension will be added automatically.
 */
501
void RenderWidget::slotUpdateButtons(const QUrl &url)
502
{
503 504 505 506
    if (m_view.out_file->url().isEmpty()) {
        m_view.buttonGenerateScript->setEnabled(false);
        m_view.buttonRender->setEnabled(false);
    } else {
507 508
        updateButtons(); // This also checks whether the selected format is available
    }
509
    if (url.isValid()) {
510
        QTreeWidgetItem *item = m_view.formats->currentItem();
Nicolas Carion's avatar
Nicolas Carion committed
511
        if ((item == nullptr) || (item->parent() == nullptr)) { // categories have no parent
512 513
            m_view.buttonRender->setEnabled(false);
            m_view.buttonGenerateScript->setEnabled(false);
514 515
            return;
        }
516
        const QString extension = item->data(0, ExtensionRole).toString();
Laurent Montel's avatar
Laurent Montel committed
517
        m_view.out_file->setUrl(filenameWithExtension(url, extension));
518 519 520
    }
}

Simon Eugster's avatar
Simon Eugster committed
521 522 523 524
/**
 * Will be called when the user changes the output file path in the text line.
 * File extension must NOT be added, would make editing impossible!
 */
525 526
void RenderWidget::slotUpdateButtons()
{
527 528 529 530 531 532
    if (m_view.out_file->url().isEmpty()) {
        m_view.buttonRender->setEnabled(false);
        m_view.buttonGenerateScript->setEnabled(false);
    } else {
        updateButtons(); // This also checks whether the selected format is available
    }
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
533 534
}

535 536
void RenderWidget::slotSaveProfile()
{
537
    Ui::SaveProfile_UI ui;
538
    QPointer<QDialog> d = new QDialog(this);
539
    ui.setupUi(d);
540

541
    QString customGroup;
542
    QStringList arguments = m_view.advanced_params->toPlainText().split(' ', QString::SkipEmptyParts);
543
    if (!arguments.isEmpty()) {
544
        ui.parameters->setText(arguments.join(QLatin1Char(' ')));
545
    }
546
    ui.profile_name->setFocus();
547
    QTreeWidgetItem *item = m_view.formats->currentItem();
Nicolas Carion's avatar
Nicolas Carion committed
548
    if ((item != nullptr) && (item->parent() != nullptr)) { // not a category
549
        // Duplicate current item settings
550 551
        customGroup = item->parent()->text(0);
        ui.extension->setText(item->data(0, ExtensionRole).toString());
552 553
        if (ui.parameters->toPlainText().contains(QStringLiteral("%bitrate")) || ui.parameters->toPlainText().contains(QStringLiteral("%quality"))) {
            if (ui.parameters->toPlainText().contains(QStringLiteral("%quality"))) {
554 555 556 557 558 559
                ui.vbitrates_label->setText(i18n("Qualities"));
                ui.default_vbitrate_label->setText(i18n("Default quality"));
            } else {
                ui.vbitrates_label->setText(i18n("Bitrates"));
                ui.default_vbitrate_label->setText(i18n("Default bitrate"));
            }
Nicolas Carion's avatar
Nicolas Carion committed
560
            if (item->data(0, BitratesRole).canConvert(QVariant::StringList) && (item->data(0, BitratesRole).toStringList().count() != 0)) {
561
                QStringList bitrates = item->data(0, BitratesRole).toStringList();
562
                ui.vbitrates_list->setText(bitrates.join(QLatin1Char(',')));
563
                if (item->data(0, DefaultBitrateRole).canConvert(QVariant::String)) {
564
                    ui.default_vbitrate->setValue(item->data(0, DefaultBitrateRole).toInt());
565
                }
566
            }
567 568
        } else {
            ui.vbitrates->setHidden(true);
569
        }
570 571
        if (ui.parameters->toPlainText().contains(QStringLiteral("%audiobitrate")) || ui.parameters->toPlainText().contains(QStringLiteral("%audioquality"))) {
            if (ui.parameters->toPlainText().contains(QStringLiteral("%audioquality"))) {
572 573 574 575 576 577
                ui.abitrates_label->setText(i18n("Qualities"));
                ui.default_abitrate_label->setText(i18n("Default quality"));
            } else {
                ui.abitrates_label->setText(i18n("Bitrates"));
                ui.default_abitrate_label->setText(i18n("Default bitrate"));
            }
Nicolas Carion's avatar
Nicolas Carion committed
578 579
            if ((item != nullptr) && item->data(0, AudioBitratesRole).canConvert(QVariant::StringList) &&
                (item->data(0, AudioBitratesRole).toStringList().count() != 0)) {
580
                QStringList bitrates = item->data(0, AudioBitratesRole).toStringList();
581
                ui.abitrates_list->setText(bitrates.join(QLatin1Char(',')));
582
                if (item->data(0, DefaultAudioBitrateRole).canConvert(QVariant::String)) {
583
                    ui.default_abitrate->setValue(item->data(0, DefaultAudioBitrateRole).toInt());
584
                }
585
            }
586 587
        } else {
            ui.abitrates->setHidden(true);
588
        }
589

Nicolas Carion's avatar
Nicolas Carion committed
590
        if (item->data(0, SpeedsRole).canConvert(QVariant::StringList) && (item->data(0, SpeedsRole).toStringList().count() != 0)) {
591 592 593
            QStringList speeds = item->data(0, SpeedsRole).toStringList();
            ui.speeds_list->setText(speeds.join('\n'));
        }
594
    }
595

596 597 598
    if (customGroup.isEmpty()) {
        customGroup = i18nc("Group Name", "Custom");
    }
599
    ui.group_name->setText(customGroup);
600

601
    if (d->exec() == QDialog::Accepted && !ui.profile_name->text().simplified().isEmpty()) {
602 603
        QString newProfileName = ui.profile_name->text().simplified();
        QString newGroupName = ui.group_name->text().simplified();
604 605 606
        if (newGroupName.isEmpty()) {
            newGroupName = i18nc("Group Name", "Custom");
        }
607 608

        QDomDocument doc;
609 610 611 612
        QDomElement profileElement = doc.createElement(QStringLiteral("profile"));
        profileElement.setAttribute(QStringLiteral("name"), newProfileName);
        profileElement.setAttribute(QStringLiteral("category"), newGroupName);
        profileElement.setAttribute(QStringLiteral("extension"), ui.extension->text().simplified());
613
        QString args = ui.parameters->toPlainText().simplified();
614 615
        profileElement.setAttribute(QStringLiteral("args"), args);
        if (args.contains(QStringLiteral("%bitrate"))) {
616
            // profile has a variable bitrate
617 618 619 620 621
            profileElement.setAttribute(QStringLiteral("defaultbitrate"), QString::number(ui.default_vbitrate->value()));
            profileElement.setAttribute(QStringLiteral("bitrates"), ui.vbitrates_list->text());
        } else if (args.contains(QStringLiteral("%quality"))) {
            profileElement.setAttribute(QStringLiteral("defaultquality"), QString::number(ui.default_vbitrate->value()));
            profileElement.setAttribute(QStringLiteral("qualities"), ui.vbitrates_list->text());
622
        }
623

624
        if (args.contains(QStringLiteral("%audiobitrate"))) {
625
            // profile has a variable bitrate
626 627 628
            profileElement.setAttribute(QStringLiteral("defaultaudiobitrate"), QString::number(ui.default_abitrate->value()));
            profileElement.setAttribute(QStringLiteral("audiobitrates"), ui.abitrates_list->text());
        } else if (args.contains(QStringLiteral("%audioquality"))) {
629
            // profile has a variable bitrate
630 631
            profileElement.setAttribute(QStringLiteral("defaultaudioquality"), QString::number(ui.default_abitrate->value()));
            profileElement.setAttribute(QStringLiteral("audioqualities"), ui.abitrates_list->text());
632
        }
633 634 635
        QString speeds_list_str = ui.speeds_list->toPlainText();
        if (!speeds_list_str.isEmpty()) {
            profileElement.setAttribute(QStringLiteral("speeds"), speeds_list_str.replace('\n', ';').simplified());
636
        }
637

638 639 640
        doc.appendChild(profileElement);
        saveProfile(doc.documentElement());

641
        parseProfiles();
642 643 644 645
    }
    delete d;
}

646
bool RenderWidget::saveProfile(QDomElement newprofile)
647
{
Laurent Montel's avatar
Laurent Montel committed
648
    QDir dir(QStandardPaths::writableLocation(QStandardPaths::AppDataLocation) + QStringLiteral("/export/"));
649
    if (!dir.exists()) {
650
        dir.mkpath(QStringLiteral("."));
651
    }
652
    QDomDocument doc;
653
    QFile file(dir.absoluteFilePath(QStringLiteral("customprofiles.xml")));
654 655 656 657
    doc.setContent(&file, false);
    file.close();
    QDomElement documentElement;
    QDomElement profiles = doc.documentElement();
658
    if (profiles.isNull() || profiles.tagName() != QLatin1String("profiles")) {
659
        doc.clear();
660 661
        profiles = doc.createElement(QStringLiteral("profiles"));
        profiles.setAttribute(QStringLiteral("version"), 1);
662 663
        doc.appendChild(profiles);
    }
Laurent Montel's avatar
Laurent Montel committed
664
    int version = profiles.attribute(QStringLiteral("version"), nullptr).toInt();
665 666
    if (version < 1) {
        doc.clear();
667 668
        profiles = doc.createElement(QStringLiteral("profiles"));
        profiles.setAttribute(QStringLiteral("version"), 1);
669 670
        doc.appendChild(profiles);
    }
671

672
    QDomNodeList profilelist = doc.elementsByTagName(QStringLiteral("profile"));
673 674 675
    QString newProfileName = newprofile.attribute(QStringLiteral("name"));
    // Check existing profiles
    QStringList existingProfileNames;
676 677 678
    int i = 0;
    while (!profilelist.item(i).isNull()) {
        documentElement = profilelist.item(i).toElement();
679
        QString profileName = documentElement.attribute(QStringLiteral("name"));
680 681 682 683 684 685
        existingProfileNames << profileName;
        i++;
    }
    // Check if a profile with that same name already exists
    bool ok;
    while (existingProfileNames.contains(newProfileName)) {
Nicolas Carion's avatar
Nicolas Carion committed
686 687 688
        QString updatedProfileName = QInputDialog::getText(this, i18n("Profile already exists"),
                                                           i18n("This profile name already exists. Change the name if you don't want to overwrite it."),
                                                           QLineEdit::Normal, newProfileName, &ok);
689 690 691
        if (!ok) {
            return false;
        }
692 693 694 695 696 697 698
        if (updatedProfileName == newProfileName) {
            // remove previous profile
            profiles.removeChild(profilelist.item(existingProfileNames.indexOf(newProfileName)));
            break;
        } else {
            newProfileName = updatedProfileName;
            newprofile.setAttribute(QStringLiteral("name"), newProfileName);
699
        }
700
    }
701

702
    profiles.appendChild(newprofile);
703

Nicolas Carion's avatar
Nicolas Carion committed
704
    // QCString save = doc.toString().utf8();
705

706
    if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
707
        KMessageBox::sorry(this, i18n("Unable to write to file %1", dir.absoluteFilePath("customprofiles.xml")));
708
        return false;
709 710 711 712
    }
    QTextStream out(&file);
    out << doc.toString();
    if (file.error() != QFile::NoError) {
713
        KMessageBox::error(this, i18n("Cannot write to file %1", dir.absoluteFilePath("customprofiles.xml")));
714
        file.close();
715
        return false;
716 717
    }
    file.close();
718
    return true;
719 720 721 722
}

void RenderWidget::slotCopyToFavorites()
{
723
    QTreeWidgetItem *item = m_view.formats->currentItem();
Nicolas Carion's avatar
Nicolas Carion committed
724
    if ((item == nullptr) || (item->parent() == nullptr)) {
Laurent Montel's avatar
Laurent Montel committed
725
        return;
726
    }
727

728 729 730
    QString params = item->data(0, ParamsRole).toString();
    QString extension = item->data(0, ExtensionRole).toString();
    QString currentProfile = item->text(0);
731
    QDomDocument doc;
732 733 734 735 736 737 738
    QDomElement profileElement = doc.createElement(QStringLiteral("profile"));
    profileElement.setAttribute(QStringLiteral("name"), currentProfile);
    profileElement.setAttribute(QStringLiteral("category"), i18nc("Category Name", "Custom"));
    profileElement.setAttribute(QStringLiteral("destinationid"), QStringLiteral("favorites"));
    profileElement.setAttribute(QStringLiteral("extension"), extension);
    profileElement.setAttribute(QStringLiteral("args"), params);
    if (params.contains(QStringLiteral("%bitrate"))) {
739
        // profile has a variable bitrate
740
        profileElement.setAttribute(QStringLiteral("defaultbitrate"), item->data(0, DefaultBitrateRole).toString());
741
        profileElement.setAttribute(QStringLiteral("bitrates"), item->data(0, BitratesRole).toStringList().join(QLatin1Char(',')));
742
    } else if (params.contains(QStringLiteral("%quality"))) {
743
        profileElement.setAttribute(QStringLiteral("defaultquality"), item->data(0, DefaultBitrateRole).toString());
744
        profileElement.setAttribute(QStringLiteral("qualities"), item->data(0, BitratesRole).toStringList().join(QLatin1Char(',')));
745
    }
746
    if (params.contains(QStringLiteral("%audiobitrate"))) {
747
        // profile has a variable bitrate
748
        profileElement.setAttribute(QStringLiteral("defaultaudiobitrate"), item->data(0, DefaultAudioBitrateRole).toString());
749
        profileElement.setAttribute(QStringLiteral("audiobitrates"), item->data(0, AudioBitratesRole).toStringList().join(QLatin1Char(',')));
750
    } else if (params.contains(QStringLiteral("%audioquality"))) {
751
        // profile has a variable bitrate
752
        profileElement.setAttribute(QStringLiteral("defaultaudioquality"), item->data(0, DefaultAudioBitrateRole).toString());
753
        profileElement.setAttribute(QStringLiteral("audioqualities"), item->data(0, AudioBitratesRole).toStringList().join(QLatin1Char(',')));
754
    }
Nicolas Carion's avatar
Nicolas Carion committed
755
    if (item->data(0, SpeedsRole).canConvert(QVariant::StringList) && (item->data(0, SpeedsRole).toStringList().count() != 0)) {
756
        // profile has a variable speed
757
        profileElement.setAttribute(QStringLiteral("speeds"), item->data(0, SpeedsRole).toStringList().join(QLatin1Char(';')));
758
    }
759
    doc.appendChild(profileElement);
760
    if (saveProfile(doc.documentElement())) {
761
        parseProfiles(profileElement.attribute(QStringLiteral("name")));
762
    }
763 764
}

765 766
void RenderWidget::slotEditProfile()
{
767
    QTreeWidgetItem *item = m_view.formats->currentItem();
Nicolas Carion's avatar
Nicolas Carion committed
768
    if ((item == nullptr) || (item->parent() == nullptr)) {
769 770
        return;
    }
771
    QString params = item->data(0, ParamsRole).toString();
772 773

    Ui::SaveProfile_UI ui;
774
    QPointer<QDialog> d = new QDialog(this);
775
    ui.setupUi(d);
776

777
    QString customGroup = item->parent()->text(0);
778 779 780
    if (customGroup.isEmpty()) {
        customGroup = i18nc("Group Name", "Custom");
    }
781 782
    ui.group_name->setText(customGroup);

783 784
    ui.profile_name->setText(item->text(0));
    ui.extension->setText(item->data(0, ExtensionRole).toString());
785 786
    ui.parameters->setText(params);
    ui.profile_name->setFocus();
787 788
    if (params.contains(QStringLiteral("%bitrate")) || ui.parameters->toPlainText().contains(QStringLiteral("%quality"))) {
        if (params.contains(QStringLiteral("%quality"))) {
789 790 791 792 793 794
            ui.vbitrates_label->setText(i18n("Qualities"));
            ui.default_vbitrate_label->setText(i18n("Default quality"));
        } else {
            ui.vbitrates_label->setText(i18n("Bitrates"));
            ui.default_vbitrate_label->setText(i18n("Default bitrate"));
        }
Nicolas Carion's avatar
Nicolas Carion committed
795
        if (item->data(0, BitratesRole).canConvert(QVariant::StringList) && (item->data(0, BitratesRole).toStringList().count() != 0)) {
796
            QStringList bitrates = item->data(0, BitratesRole).toStringList();
797
            ui.vbitrates_list->setText(bitrates.join(QLatin1Char(',')));
798
            if (item->data(0, DefaultBitrateRole).canConvert(QVariant::String)) {
799
                ui.default_vbitrate->setValue(item->data(0, DefaultBitrateRole).toInt());
800
            }
801
        }
Laurent Montel's avatar
Laurent Montel committed
802 803
    } else {
        ui.vbitrates->setHidden(true);
804
    }
Laurent Montel's avatar
Laurent Montel committed
805

806 807
    if (params.contains(QStringLiteral("%audiobitrate")) || ui.parameters->toPlainText().contains(QStringLiteral("%audioquality"))) {
        if (params.contains(QStringLiteral("%audioquality"))) {
808 809 810 811 812 813
            ui.abitrates_label->setText(i18n("Qualities"));
            ui.default_abitrate_label->setText(i18n("Default quality"));
        } else {
            ui.abitrates_label->setText(i18n("Bitrates"));
            ui.default_abitrate_label->setText(i18n("Default bitrate"));
        }
Nicolas Carion's avatar
Nicolas Carion committed
814
        if (item->data(0, AudioBitratesRole).canConvert(QVariant::StringList) && (item->data(0, AudioBitratesRole).toStringList().count() != 0)) {
815
            QStringList bitrates = item->data(0, AudioBitratesRole).toStringList();
816
            ui.abitrates_list->setText(bitrates.join(QLatin1Char(',')));
817
            if (item->data(0, DefaultAudioBitrateRole).canConvert(QVariant::String)) {
818
                ui.default_abitrate->setValue(item->data(0, DefaultAudioBitrateRole).toInt());
819
            }
820
        }
821 822
    } else {
        ui.abitrates->setHidden(true);
823
    }
824

Nicolas Carion's avatar
Nicolas Carion committed
825
    if (item->data(0, SpeedsRole).canConvert(QVariant::StringList) && (item->data(0, SpeedsRole).toStringList().count() != 0)) {
826 827 828 829
        QStringList speeds = item->data(0, SpeedsRole).toStringList();
        ui.speeds_list->setText(speeds.join('\n'));
    }

830
    d->setWindowTitle(i18n("Edit Profile"));
831

832
    if (d->exec() == QDialog::Accepted) {
833
        slotDeleteProfile(true);
834
        QString exportFile = QStandardPaths::writableLocation(QStandardPaths::AppDataLocation) + QStringLiteral("/export/customprofiles.xml");