renderwidget.cpp 120 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 21 22
#include "renderwidget.h"
#include "kdenlivesettings.h"
#include "ui_saveprofile_ui.h"
23
#include "timecode.h"
Vincent Pinon's avatar
Vincent Pinon committed
24
#include "dialogs/profilesdialog.h"
25
#include "utils/KoIconUtils.h"
26 27
#include "profiles/profilerepository.hpp"
#include "profiles/profilemodel.hpp"
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
28

29
#include "klocalizedstring.h"
30
#include <KMessageBox>
31
#include <KRun>
32
#include <KColorScheme>
33
#include <KNotification>
34 35
#include <KMimeTypeTrader>
#include <KIO/DesktopExecParser>
36
#include <knotifications_version.h>
37

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

Vincent Pinon's avatar
Vincent Pinon committed
54
#include <locale>
55 56 57
#ifdef Q_OS_MAC
#include <xlocale.h>
#endif
58

59
// Render profiles roles
60 61 62 63 64 65 66 67 68 69
enum {GroupRole = Qt::UserRole,
      ExtensionRole,
      StandardRole,
      RenderRole,
      ParamsRole,
      EditableRole,
      ExtraRole,
      BitratesRole,
      DefaultBitrateRole,
      AudioBitratesRole,
70
      DefaultAudioBitrateRole,
71
      SpeedsRole,
72
      ErrorRole
73
     };
74

75 76 77
// Render job roles
const int ParametersRole = Qt::UserRole + 1;
const int TimeRole = Qt::UserRole + 2;
78 79
const int ProgressRole = Qt::UserRole + 3;
const int ExtraInfoRole = Qt::UserRole + 5;
80

81 82
const int DirectRenderType = QTreeWidgetItem::Type;
const int ScriptRenderType = QTreeWidgetItem::UserType;
83

84
// Running job status
85 86 87 88 89 90 91 92
enum JOBSTATUS {
    WAITINGJOB = 0,
    STARTINGJOB,
    RUNNINGJOB,
    FINISHEDJOB,
    FAILEDJOB,
    ABORTEDJOB
};
93

94
#ifdef Q_OS_WIN
95 96 97 98 99 100 101 102 103
const QLatin1String ScriptFormat(".bat");
QString ScriptSetVar(const QString name, const QString value) { return QString("set ") + name + "=" + value; }
QString ScriptGetVar(const QString varName) { return QString('%') + varName + '%'; }
#else
const QLatin1String ScriptFormat(".sh");
QString ScriptSetVar(const QString name, const QString value) { return name + "=\"" + value + '\"'; }
QString ScriptGetVar(const QString varName) { return QString('$') + varName; }
#endif

104 105 106
static QStringList acodecsList;
static QStringList vcodecsList;
static QStringList supportedFormats;
107

108
RenderJobItem::RenderJobItem(QTreeWidget *parent, const QStringList &strings, int type)
Laurent Montel's avatar
Laurent Montel committed
109
    : QTreeWidgetItem(parent, strings, type),
110
      m_status(-1)
111
{
112
    setSizeHint(1, QSize(parent->columnWidth(1), parent->fontMetrics().height() * 3));
113 114 115 116 117
    setStatus(WAITINGJOB);
}

void RenderJobItem::setStatus(int status)
{
118
    if (m_status == status) {
Laurent Montel's avatar
Laurent Montel committed
119
        return;
120
    }
121 122
    m_status = status;
    switch (status) {
123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142
    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;
143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159
    }
}

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

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

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

161
RenderWidget::RenderWidget(const QString &projectfolder, bool enableProxy, const QString &profile, QWidget *parent) :
162 163 164 165
    QDialog(parent),
    m_projectFolder(projectfolder),
    m_profile(profile),
    m_blockProcessing(false)
166
{
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
167
    m_view.setupUi(this);
168 169
    int size = style()->pixelMetric(QStyle::PM_SmallIconSize);
    QSize iconSize(size, size);
170

171
    setWindowTitle(i18n("Rendering"));
172 173 174 175
    m_view.buttonDelete->setIconSize(iconSize);
    m_view.buttonEdit->setIconSize(iconSize);
    m_view.buttonSave->setIconSize(iconSize);
    m_view.buttonFavorite->setIconSize(iconSize);
176

177
    m_view.buttonDelete->setIcon(KoIconUtils::themedIcon(QStringLiteral("trash-empty")));
178 179 180
    m_view.buttonDelete->setToolTip(i18n("Delete profile"));
    m_view.buttonDelete->setEnabled(false);

181
    m_view.buttonEdit->setIcon(KoIconUtils::themedIcon(QStringLiteral("document-edit")));
182 183 184
    m_view.buttonEdit->setToolTip(i18n("Edit profile"));
    m_view.buttonEdit->setEnabled(false);

185
    m_view.buttonSave->setIcon(KoIconUtils::themedIcon(QStringLiteral("document-new")));
186 187
    m_view.buttonSave->setToolTip(i18n("Create new profile"));

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

190
    m_view.buttonFavorite->setIcon(KoIconUtils::themedIcon(QStringLiteral("favorite")));
191
    m_view.buttonFavorite->setToolTip(i18n("Copy profile to favorites"));
192

193
    m_view.out_file->button()->setToolTip(i18n("Select output destination"));
194
    m_view.advanced_params->setMaximumHeight(QFontMetrics(font()).lineSpacing() * 5);
195

196
    m_view.optionsGroup->setVisible(m_view.options->isChecked());
Laurent Montel's avatar
Laurent Montel committed
197
    connect(m_view.options, &QAbstractButton::toggled, m_view.optionsGroup, &QWidget::setVisible);
198
    m_view.videoLabel->setVisible(m_view.options->isChecked());
Laurent Montel's avatar
Laurent Montel committed
199
    connect(m_view.options, &QAbstractButton::toggled, m_view.videoLabel, &QWidget::setVisible);
200
    m_view.video->setVisible(m_view.options->isChecked());
Laurent Montel's avatar
Laurent Montel committed
201
    connect(m_view.options, &QAbstractButton::toggled, m_view.video, &QWidget::setVisible);
202
    m_view.audioLabel->setVisible(m_view.options->isChecked());
Laurent Montel's avatar
Laurent Montel committed
203
    connect(m_view.options, &QAbstractButton::toggled, m_view.audioLabel, &QWidget::setVisible);
204
    m_view.audio->setVisible(m_view.options->isChecked());
Laurent Montel's avatar
Laurent Montel committed
205 206
    connect(m_view.options, &QAbstractButton::toggled, m_view.audio, &QWidget::setVisible);
    connect(m_view.quality, &QAbstractSlider::valueChanged, this, &RenderWidget::adjustAVQualities);
207
    connect(m_view.video, SIGNAL(valueChanged(int)), this, SLOT(adjustQuality(int)));
Laurent Montel's avatar
Laurent Montel committed
208
    connect(m_view.speed, &QAbstractSlider::valueChanged, this, &RenderWidget::adjustSpeed);
209

210 211
    m_view.buttonRender->setEnabled(false);
    m_view.buttonGenerateScript->setEnabled(false);
212
    setRescaleEnabled(false);
213 214 215 216 217 218 219
    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);
220
    m_view.proxy_render->setHidden(!enableProxy);
221
    connect(m_view.proxy_render, &QCheckBox::toggled, this, &RenderWidget::slotProxyWarn);
222 223
    KColorScheme scheme(palette().currentColorGroup(), KColorScheme::Window, KSharedConfig::openConfig(KdenliveSettings::colortheme()));
    QColor bg = scheme.background(KColorScheme::NegativeBackground).color();
224
    m_view.errorBox->setStyleSheet(QStringLiteral("QGroupBox { background-color: rgb(%1, %2, %3); border-radius: 5px;}; ").arg(bg.red()).arg(bg.green()).arg(bg.blue()));
225
    int height = QFontInfo(font()).pixelSize();
226
    m_view.errorIcon->setPixmap(KoIconUtils::themedIcon(QStringLiteral("dialog-warning")).pixmap(height, height));
227 228 229
    m_view.errorBox->setHidden(true);

    m_infoMessage = new KMessageWidget;
230
    m_view.info->addWidget(m_infoMessage);
231 232 233
    m_infoMessage->setCloseButtonVisible(false);
    m_infoMessage->hide();

234 235 236 237
    m_view.encoder_threads->setMaximum(QThread::idealThreadCount());
    m_view.encoder_threads->setValue(KdenliveSettings::encodethreads());
    connect(m_view.encoder_threads, SIGNAL(valueChanged(int)), this, SLOT(slotUpdateEncodeThreads(int)));

238 239 240
    m_view.rescale_keep->setChecked(KdenliveSettings::rescalekeepratio());
    connect(m_view.rescale_width, SIGNAL(valueChanged(int)), this, SLOT(slotUpdateRescaleWidth(int)));
    connect(m_view.rescale_height, SIGNAL(valueChanged(int)), this, SLOT(slotUpdateRescaleHeight(int)));
241
    m_view.rescale_keep->setIcon(KoIconUtils::themedIcon(QStringLiteral("edit-link")));
242
    m_view.rescale_keep->setToolTip(i18n("Preserve aspect ratio"));
Laurent Montel's avatar
Laurent Montel committed
243
    connect(m_view.rescale_keep, &QAbstractButton::clicked, this, &RenderWidget::slotSwitchAspectRatio);
244

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

248 249 250 251
    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
252
    connect(m_view.export_audio, &QCheckBox::stateChanged, this, &RenderWidget::slotUpdateAudioLabel);
253 254
    m_view.export_audio->setCheckState(Qt::PartiallyChecked);

255
    parseProfiles();
256
    parseScriptFiles();
257 258
    m_view.running_jobs->setUniformRowHeights(false);
    m_view.scripts_list->setUniformRowHeights(false);
Laurent Montel's avatar
Laurent Montel committed
259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278
    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
279
    connect(m_view.out_file, SIGNAL(textChanged(QString)), this, SLOT(slotUpdateButtons()));
280
    connect(m_view.out_file, SIGNAL(urlSelected(QUrl)), this, SLOT(slotUpdateButtons(QUrl)));
281

Laurent Montel's avatar
Laurent Montel committed
282 283
    connect(m_view.formats, &QTreeWidget::currentItemChanged, this, &RenderWidget::refreshParams);
    connect(m_view.formats, &QTreeWidget::itemDoubleClicked, this, &RenderWidget::slotEditItem);
284

Laurent Montel's avatar
Laurent Montel committed
285 286 287
    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);
288 289 290 291

    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
292
    connect(m_view.tc_overlay, &QAbstractButton::toggled, m_view.tc_type, &QWidget::setEnabled);
293

294 295
    //m_view.splitter->setStretchFactor(1, 5);
    //m_view.splitter->setStretchFactor(0, 2);
296

297
    m_view.out_file->setMode(KFile::File);
298 299 300 301 302

#if KXMLGUI_VERSION_MINOR > 32 || KXMLGUI_VERSION_MAJOR > 5
    m_view.out_file->setAcceptMode(QFileDialog::AcceptSave);
#endif

303
    m_view.out_file->setFocusPolicy(Qt::ClickFocus);
304

305
    m_jobsDelegate = new RenderViewDelegate(this);
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
306
    m_view.running_jobs->setHeaderLabels(QStringList() << QString() << i18n("File"));
307
    m_view.running_jobs->setItemDelegate(m_jobsDelegate);
308

309
    QHeaderView *header = m_view.running_jobs->header();
310
    header->setSectionResizeMode(0, QHeaderView::Fixed);
311
    header->resizeSection(0, size + 4);
312
    header->setSectionResizeMode(1, QHeaderView::Interactive);
313

314
    m_view.scripts_list->setHeaderLabels(QStringList() << QString() << i18n("Script Files"));
315 316
    m_scriptsDelegate = new RenderViewDelegate(this);
    m_view.scripts_list->setItemDelegate(m_scriptsDelegate);
317
    header = m_view.scripts_list->header();
318
    header->setSectionResizeMode(0, QHeaderView::Fixed);
319
    header->resizeSection(0, size + 4);
320

Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
321
    // Find path for Kdenlive renderer
322 323 324
#ifdef Q_OS_WIN
    m_renderer = QCoreApplication::applicationDirPath() + QStringLiteral("/kdenlive_render.exe");
#else
325
    m_renderer = QCoreApplication::applicationDirPath() + QStringLiteral("/kdenlive_render");
326
#endif
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
327
    if (!QFile::exists(m_renderer)) {
328
        m_renderer = QStandardPaths::findExecutable(QStringLiteral("kdenlive_render"));
329
        if (m_renderer.isEmpty()) {
330
            m_renderer = QStringLiteral("kdenlive_render");
331
        }
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
332 333
    }

334 335
    QDBusConnectionInterface *interface = QDBusConnection::sessionBus().interface();
    if (!interface || (!interface->isServiceRegistered(QStringLiteral("org.kde.ksmserver")) && !interface->isServiceRegistered(QStringLiteral("org.gnome.SessionManager")))) {
336
        m_view.shutdown->setEnabled(false);
337
    }
338
    checkCodecs();
339
    refreshView();
340
    focusFirstVisibleItem();
341
    adjustSize();
342 343 344 345 346 347
}

QSize RenderWidget::sizeHint() const
{
    // Make sure the widget has minimum size on opening
    return QSize(200, 200);
348 349
}

350 351 352 353 354 355 356 357
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;
358
    delete m_infoMessage;
359 360
}

361
void RenderWidget::slotEditItem(QTreeWidgetItem *item)
362
{
Laurent Montel's avatar
Laurent Montel committed
363
    if (item->parent() == nullptr) {
364 365 366
        // This is a top level item - group - don't edit
        return;
    }
367
    const QString edit = item->data(0, EditableRole).toString();
368
    if (edit.isEmpty() || !edit.endsWith(QLatin1String("customprofiles.xml"))) {
Laurent Montel's avatar
Laurent Montel committed
369
        slotSaveProfile();
370 371 372
    } else {
        slotEditProfile();
    }
373 374
}

375 376
void RenderWidget::showInfoPanel()
{
377 378 379 380 381 382 383
    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
384 385
}

386
void RenderWidget::setDocumentPath(const QString &path)
387
{
388
    if (m_view.out_file->url().adjusted(QUrl::RemoveFilename).toLocalFile() == QUrl::fromLocalFile(m_projectFolder).adjusted(QUrl::RemoveFilename).toLocalFile()) {
389
        const QString fileName = m_view.out_file->url().fileName();
390
        m_view.out_file->setUrl(QUrl::fromLocalFile(path + fileName));
391
    }
392
    m_projectFolder = path;
393
    parseScriptFiles();
394

395 396
}

397 398
void RenderWidget::slotUpdateGuideBox()
{
399 400 401
    m_view.guides_box->setVisible(m_view.render_guide->isChecked());
}

402 403
void RenderWidget::slotCheckStartGuidePosition()
{
404
    if (m_view.guide_start->currentIndex() > m_view.guide_end->currentIndex()) {
405
        m_view.guide_start->setCurrentIndex(m_view.guide_end->currentIndex());
406
    }
407 408
}

409 410
void RenderWidget::slotCheckEndGuidePosition()
{
411
    if (m_view.guide_end->currentIndex() < m_view.guide_start->currentIndex()) {
412
        m_view.guide_end->setCurrentIndex(m_view.guide_start->currentIndex());
413
    }
414 415
}

Laurent Montel's avatar
Laurent Montel committed
416
void RenderWidget::setGuides(const QMap<double, QString> &guidesData, double duration)
417
{
418 419
    m_view.guide_start->clear();
    m_view.guide_end->clear();
420
    if (!guidesData.isEmpty()) {
421
        m_view.guide_start->addItem(i18n("Beginning"), "0");
422
        m_view.render_guide->setEnabled(true);
423 424 425 426 427
        m_view.create_chapter->setEnabled(true);
    } else {
        m_view.render_guide->setEnabled(false);
        m_view.create_chapter->setEnabled(false);
    }
428
    double fps = ProfileRepository::get()->getProfile(m_profile)->fps();
429 430 431 432 433
    QMapIterator<double, QString> i(guidesData);
    while (i.hasNext()) {
        i.next();
        GenTime pos = GenTime(i.key());
        const QString guidePos = Timecode::getStringTimecode(pos.frames(fps), fps);
Laurent Montel's avatar
Laurent Montel committed
434 435
        m_view.guide_start->addItem(i.value() + QLatin1Char('/') + guidePos, i.key());
        m_view.guide_end->addItem(i.value() + QLatin1Char('/') + guidePos, i.key());
436
    }
437
    if (!guidesData.isEmpty()) {
438
        m_view.guide_end->addItem(i18n("End"), QString::number(duration));
439
    }
440 441
}

442
/**
Simon Eugster's avatar
Simon Eugster committed
443 444 445
 * Will be called when the user selects an output file via the file dialog.
 * File extension will be added automatically.
 */
446
void RenderWidget::slotUpdateButtons(const QUrl &url)
447
{
448 449 450 451
    if (m_view.out_file->url().isEmpty()) {
        m_view.buttonGenerateScript->setEnabled(false);
        m_view.buttonRender->setEnabled(false);
    } else {
452 453
        updateButtons(); // This also checks whether the selected format is available
    }
454
    if (url.isValid()) {
455 456
        QTreeWidgetItem *item = m_view.formats->currentItem();
        if (!item || !item->parent()) { // categories have no parent
457 458
            m_view.buttonRender->setEnabled(false);
            m_view.buttonGenerateScript->setEnabled(false);
459 460
            return;
        }
461
        const QString extension = item->data(0, ExtensionRole).toString();
Laurent Montel's avatar
Laurent Montel committed
462
        m_view.out_file->setUrl(filenameWithExtension(url, extension));
463 464 465
    }
}

Simon Eugster's avatar
Simon Eugster committed
466 467 468 469
/**
 * 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!
 */
470 471
void RenderWidget::slotUpdateButtons()
{
472 473 474 475 476 477
    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
478 479
}

480 481
void RenderWidget::slotSaveProfile()
{
482
    Ui::SaveProfile_UI ui;
483
    QPointer<QDialog> d = new QDialog(this);
484
    ui.setupUi(d);
485

486
    QString customGroup;
487
    QStringList arguments = m_view.advanced_params->toPlainText().split(' ', QString::SkipEmptyParts);
488
    if (!arguments.isEmpty()) {
489
        ui.parameters->setText(arguments.join(QLatin1Char(' ')));
490
    }
491
    ui.profile_name->setFocus();
492 493
    QTreeWidgetItem *item = m_view.formats->currentItem();
    if (item && item->parent()) { //not a category
494
        // Duplicate current item settings
495 496
        customGroup = item->parent()->text(0);
        ui.extension->setText(item->data(0, ExtensionRole).toString());
497 498
        if (ui.parameters->toPlainText().contains(QStringLiteral("%bitrate")) || ui.parameters->toPlainText().contains(QStringLiteral("%quality"))) {
            if (ui.parameters->toPlainText().contains(QStringLiteral("%quality"))) {
499 500 501 502 503 504
                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"));
            }
505 506
            if (item->data(0, BitratesRole).canConvert(QVariant::StringList) && item->data(0, BitratesRole).toStringList().count()) {
                QStringList bitrates = item->data(0, BitratesRole).toStringList();
507
                ui.vbitrates_list->setText(bitrates.join(QLatin1Char(',')));
508
                if (item->data(0, DefaultBitrateRole).canConvert(QVariant::String)) {
509
                    ui.default_vbitrate->setValue(item->data(0, DefaultBitrateRole).toInt());
510
                }
511
            }
512 513
        } else {
            ui.vbitrates->setHidden(true);
514
        }
515 516
        if (ui.parameters->toPlainText().contains(QStringLiteral("%audiobitrate")) || ui.parameters->toPlainText().contains(QStringLiteral("%audioquality"))) {
            if (ui.parameters->toPlainText().contains(QStringLiteral("%audioquality"))) {
517 518 519 520 521 522
                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"));
            }
523
            if (item && item->data(0, AudioBitratesRole).canConvert(QVariant::StringList) && item->data(0, AudioBitratesRole).toStringList().count()) {
524
                QStringList bitrates = item->data(0, AudioBitratesRole).toStringList();
525
                ui.abitrates_list->setText(bitrates.join(QLatin1Char(',')));
526
                if (item->data(0, DefaultAudioBitrateRole).canConvert(QVariant::String)) {
527
                    ui.default_abitrate->setValue(item->data(0, DefaultAudioBitrateRole).toInt());
528
                }
529
            }
530 531
        } else {
            ui.abitrates->setHidden(true);
532
        }
533 534 535 536 537

        if (item->data(0, SpeedsRole).canConvert(QVariant::StringList) && item->data(0, SpeedsRole).toStringList().count()) {
            QStringList speeds = item->data(0, SpeedsRole).toStringList();
            ui.speeds_list->setText(speeds.join('\n'));
        }
538
    }
539

540 541 542
    if (customGroup.isEmpty()) {
        customGroup = i18nc("Group Name", "Custom");
    }
543
    ui.group_name->setText(customGroup);
544

545
    if (d->exec() == QDialog::Accepted && !ui.profile_name->text().simplified().isEmpty()) {
546 547
        QString newProfileName = ui.profile_name->text().simplified();
        QString newGroupName = ui.group_name->text().simplified();
548 549 550
        if (newGroupName.isEmpty()) {
            newGroupName = i18nc("Group Name", "Custom");
        }
551 552

        QDomDocument doc;
553 554 555 556
        QDomElement profileElement = doc.createElement(QStringLiteral("profile"));
        profileElement.setAttribute(QStringLiteral("name"), newProfileName);
        profileElement.setAttribute(QStringLiteral("category"), newGroupName);
        profileElement.setAttribute(QStringLiteral("extension"), ui.extension->text().simplified());
557
        QString args = ui.parameters->toPlainText().simplified();
558 559
        profileElement.setAttribute(QStringLiteral("args"), args);
        if (args.contains(QStringLiteral("%bitrate"))) {
560
            // profile has a variable bitrate
561 562 563 564 565
            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());
566
        }
567

568
        if (args.contains(QStringLiteral("%audiobitrate"))) {
569
            // profile has a variable bitrate
570 571 572
            profileElement.setAttribute(QStringLiteral("defaultaudiobitrate"), QString::number(ui.default_abitrate->value()));
            profileElement.setAttribute(QStringLiteral("audiobitrates"), ui.abitrates_list->text());
        } else if (args.contains(QStringLiteral("%audioquality"))) {
573
            // profile has a variable bitrate
574 575
            profileElement.setAttribute(QStringLiteral("defaultaudioquality"), QString::number(ui.default_abitrate->value()));
            profileElement.setAttribute(QStringLiteral("audioqualities"), ui.abitrates_list->text());
576
        }
577 578 579
        QString speeds_list_str = ui.speeds_list->toPlainText();
        if (!speeds_list_str.isEmpty()) {
            profileElement.setAttribute(QStringLiteral("speeds"), speeds_list_str.replace('\n', ';').simplified());
580
        }
581

582 583 584
        doc.appendChild(profileElement);
        saveProfile(doc.documentElement());

585
        parseProfiles();
586 587 588 589
    }
    delete d;
}

590
bool RenderWidget::saveProfile(QDomElement newprofile)
591
{
Laurent Montel's avatar
Laurent Montel committed
592
    QDir dir(QStandardPaths::writableLocation(QStandardPaths::AppDataLocation) + QStringLiteral("/export/"));
593
    if (!dir.exists()) {
594
        dir.mkpath(QStringLiteral("."));
595
    }
596
    QDomDocument doc;
597
    QFile file(dir.absoluteFilePath(QStringLiteral("customprofiles.xml")));
598 599 600 601
    doc.setContent(&file, false);
    file.close();
    QDomElement documentElement;
    QDomElement profiles = doc.documentElement();
602
    if (profiles.isNull() || profiles.tagName() != QLatin1String("profiles")) {
603
        doc.clear();
604 605
        profiles = doc.createElement(QStringLiteral("profiles"));
        profiles.setAttribute(QStringLiteral("version"), 1);
606 607
        doc.appendChild(profiles);
    }
Laurent Montel's avatar
Laurent Montel committed
608
    int version = profiles.attribute(QStringLiteral("version"), nullptr).toInt();
609 610
    if (version < 1) {
        doc.clear();
611 612
        profiles = doc.createElement(QStringLiteral("profiles"));
        profiles.setAttribute(QStringLiteral("version"), 1);
613 614
        doc.appendChild(profiles);
    }
615

616
    QDomNodeList profilelist = doc.elementsByTagName(QStringLiteral("profile"));
617 618 619
    QString newProfileName = newprofile.attribute(QStringLiteral("name"));
    // Check existing profiles
    QStringList existingProfileNames;
620 621 622
    int i = 0;
    while (!profilelist.item(i).isNull()) {
        documentElement = profilelist.item(i).toElement();
623
        QString profileName = documentElement.attribute(QStringLiteral("name"));
624 625 626 627 628 629 630
        existingProfileNames << profileName;
        i++;
    }
    // Check if a profile with that same name already exists
    bool ok;
    while (existingProfileNames.contains(newProfileName)) {
        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);
631 632 633
        if (!ok) {
            return false;
        }
634 635 636 637 638 639 640
        if (updatedProfileName == newProfileName) {
            // remove previous profile
            profiles.removeChild(profilelist.item(existingProfileNames.indexOf(newProfileName)));
            break;
        } else {
            newProfileName = updatedProfileName;
            newprofile.setAttribute(QStringLiteral("name"), newProfileName);
641
        }
642
    }
643

644
    profiles.appendChild(newprofile);
645

646
    //QCString save = doc.toString().utf8();
647

648
    if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
649
        KMessageBox::sorry(this, i18n("Unable to write to file %1", dir.absoluteFilePath("customprofiles.xml")));
650
        return false;
651 652 653 654
    }
    QTextStream out(&file);
    out << doc.toString();
    if (file.error() != QFile::NoError) {
655
        KMessageBox::error(this, i18n("Cannot write to file %1", dir.absoluteFilePath("customprofiles.xml")));
656
        file.close();
657
        return false;
658 659
    }
    file.close();
660
    return true;
661 662 663 664
}

void RenderWidget::slotCopyToFavorites()
{
665
    QTreeWidgetItem *item = m_view.formats->currentItem();
666
    if (!item || !item->parent()) {
Laurent Montel's avatar
Laurent Montel committed
667
        return;
668
    }
669

670 671 672
    QString params = item->data(0, ParamsRole).toString();
    QString extension = item->data(0, ExtensionRole).toString();
    QString currentProfile = item->text(0);
673
    QDomDocument doc;
674 675 676 677 678 679 680
    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"))) {
681
        // profile has a variable bitrate
682
        profileElement.setAttribute(QStringLiteral("defaultbitrate"), item->data(0, DefaultBitrateRole).toString());
683
        profileElement.setAttribute(QStringLiteral("bitrates"), item->data(0, BitratesRole).toStringList().join(QLatin1Char(',')));
684
    } else if (params.contains(QStringLiteral("%quality"))) {
685
        profileElement.setAttribute(QStringLiteral("defaultquality"), item->data(0, DefaultBitrateRole).toString());
686
        profileElement.setAttribute(QStringLiteral("qualities"), item->data(0, BitratesRole).toStringList().join(QLatin1Char(',')));
687
    }
688
    if (params.contains(QStringLiteral("%audiobitrate"))) {
689
        // profile has a variable bitrate
690
        profileElement.setAttribute(QStringLiteral("defaultaudiobitrate"), item->data(0, DefaultAudioBitrateRole).toString());
691
        profileElement.setAttribute(QStringLiteral("audiobitrates"), item->data(0, AudioBitratesRole).toStringList().join(QLatin1Char(',')));
692
    } else if (params.contains(QStringLiteral("%audioquality"))) {
693
        // profile has a variable bitrate
694
        profileElement.setAttribute(QStringLiteral("defaultaudioquality"), item->data(0, DefaultAudioBitrateRole).toString());
695
        profileElement.setAttribute(QStringLiteral("audioqualities"), item->data(0, AudioBitratesRole).toStringList().join(QLatin1Char(',')));
696
    }
697
    if (item->data(0, SpeedsRole).canConvert(QVariant::StringList) && item->data(0, SpeedsRole).toStringList().count()) {
698
        // profile has a variable speed
699
        profileElement.setAttribute(QStringLiteral("speeds"), item->data(0, SpeedsRole).toStringList().join(QLatin1Char(';')));
700
    }
701
    doc.appendChild(profileElement);
702
    if (saveProfile(doc.documentElement())) {
703
        parseProfiles(profileElement.attribute(QStringLiteral("name")));
704
    }
705 706
}

707 708
void RenderWidget::slotEditProfile()
{
709
    QTreeWidgetItem *item = m_view.formats->currentItem();
710 711 712
    if (!item || !item->parent()) {
        return;
    }
713
    QString params = item->data(0, ParamsRole).toString();
714 715

    Ui::SaveProfile_UI ui;
716
    QPointer<QDialog> d = new QDialog(this);
717
    ui.setupUi(d);
718

719
    QString customGroup = item->parent()->text(0);
720 721 722
    if (customGroup.isEmpty()) {
        customGroup = i18nc("Group Name", "Custom");
    }
723 724
    ui.group_name->setText(customGroup);

725 726
    ui.profile_name->setText(item->text(0));
    ui.extension->setText(item->data(0, ExtensionRole).toString());
727 728
    ui.parameters->setText(params);
    ui.profile_name->setFocus();
729 730
    if (params.contains(QStringLiteral("%bitrate")) || ui.parameters->toPlainText().contains(QStringLiteral("%quality"))) {
        if (params.contains(QStringLiteral("%quality"))) {
731 732 733 734 735 736
            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"));
        }
737
        if (item->data(0, BitratesRole).canConvert(QVariant::StringList) && item->data(0, BitratesRole).toStringList().count()) {
738
            QStringList bitrates = item->data(0, BitratesRole).toStringList();
739
            ui.vbitrates_list->setText(bitrates.join(QLatin1Char(',')));
740
            if (item->data(0, DefaultBitrateRole).canConvert(QVariant::String)) {
741
                ui.default_vbitrate->setValue(item->data(0, DefaultBitrateRole).toInt());
742
            }
743
        }
Laurent Montel's avatar
Laurent Montel committed
744 745
    } else {
        ui.vbitrates->setHidden(true);
746
    }
Laurent Montel's avatar
Laurent Montel committed
747

748 749
    if (params.contains(QStringLiteral("%audiobitrate")) || ui.parameters->toPlainText().contains(QStringLiteral("%audioquality"))) {
        if (params.contains(QStringLiteral("%audioquality"))) {
750 751 752 753 754 755
            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"));
        }
756
        if (item->data(0, AudioBitratesRole).canConvert(QVariant::StringList) && item->data(0, AudioBitratesRole).toStringList().count()) {
757
            QStringList bitrates = item->data(0, AudioBitratesRole).toStringList();
758
            ui.abitrates_list->setText(bitrates.join(QLatin1Char(',')));
759
            if (item->data(0, DefaultAudioBitrateRole).canConvert(QVariant::String)) {
760
                ui.default_abitrate->setValue(item->data(0, DefaultAudioBitrateRole).toInt());
761
            }
762
        }
763 764
    } else {
        ui.abitrates->setHidden(true);
765
    }
766

767
    if (item->data(0, SpeedsRole).canConvert(QVariant::StringList) && item->data(0, SpeedsRole).toStringList().count()) {
768 769 770 771
        QStringList speeds = item->data(0, SpeedsRole).toStringList();
        ui.speeds_list->setText(speeds.join('\n'));
    }

772
    d->setWindowTitle(i18n("Edit Profile"));
773

774
    if (d->exec() == QDialog::Accepted) {
775
        slotDeleteProfile(false);
776
        QString exportFile = QStandardPaths::writableLocation(QStandardPaths::AppDataLocation) + QStringLiteral("/export/customprofiles.xml");
777 778 779 780 781
        QDomDocument doc;
        QFile file(exportFile);
        doc.setContent(&file, false);
        file.close();
        QDomElement documentElement;
782 783
        QDomElement profiles = doc.documentElement();

784
        if (profiles.isNull() || profiles.tagName() != QLatin1String("profiles")) {
785
            doc.clear();
786 787
            profiles = doc.createElement(QStringLiteral("profiles"));
            profiles.setAttribute(QStringLiteral("version"), 1);
788 789 790
            doc.appendChild(profiles);
        }

Laurent Montel's avatar
Laurent Montel committed
791
        int version = profiles.attribute(QStringLiteral("version"), nullptr).toInt();
792 793
        if (version < 1) {
            doc.clear();
794 795
            profiles = doc.createElement(QStringLiteral("profiles"));
            profiles.setAttribute(QStringLiteral("version"), 1);
796
            doc.appendChild(profiles);
797 798 799 800
        }

        QString newProfileName = ui.profile_name->text().simplified();
        QString newGroupName = ui.group_name->text().simplified();
801 802 803
        if (newGroupName.isEmpty()) {
            newGroupName = i18nc("Group Name", "Custom");
        }
804
        QDomNodeList profilelist = doc.elementsByTagName(QStringLiteral("profile"));
805 806 807 808
        int i = 0;
        while (!profilelist.item(i).isNull()) {
            // make sure a profile with same name doesn't exist
            documentElement = profilelist.item(i).toElement();
809
            QString profileName = documentElement.attribute(QStringLiteral("name"));
810 811 812
            if (profileName == newProfileName) {
                // a profile with that same name already exists
                bool ok;
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
813
                newProfileName = 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);
814 815 816
                if (!ok) {
                    return;
                }
817 818
                if (profileName == newProfileName) {
                    profiles.removeChild(profilelist.item(i));
819 820 821
                    break;
                }
            }
822
            ++i;
823
        }
824

825 826 827 828
        QDomElement profileElement = doc.createElement(QStringLiteral("profile"));
        profileElement.setAttribute(QStringLiteral("name"), newProfileName);
        profileElement.setAttribute(QStringLiteral("category"), newGroupName);
        profileElement.setAttribute(QStringLiteral("extension"), ui.extension->text().simplified());
829
        QString args = ui.parameters->toPlainText().simplified();
830 831
        profileElement.setAttribute(QStringLiteral("args"), args);
        if (args.contains(QStringLiteral("%bitrate"))) {
832
            // profile has a variable bitrate
833 834
            profileElement.setAttribute(QStringLiteral("defaultbitrate"), QString::number(ui.default_vbitrate->value()));
            profileElement.setAttribute(QStringLiteral("bitrates"), ui.vbitrates_list->text());
835
        } else if (args.contains(QStringLiteral("%quality"))) {
836 837
            profileElement.setAttribute(QStringLiteral("defaultquality"), QString::number(ui.default_vbitrate->value()));
            profileElement.setAttribute(QStringLiteral("qualities"), ui.vbitrates_list->text());
Jean-Baptiste Mardelle's avatar