]> git.sesse.net Git - kdenlive/blobdiff - src/cliptranscode.cpp
Integrate with the required MLT hooks for getting Movit to work.
[kdenlive] / src / cliptranscode.cpp
index e9ccc376c2211076579be27210edd84521ccf21b..4f6218c49747c765a8f2062d0d4fd61fc27e2983 100644 (file)
 
 
 #include "cliptranscode.h"
+#include "kdenlivesettings.h"
 
 #include <KDebug>
 #include <KGlobalSettings>
 #include <KMessageBox>
+#include <KFileDialog>
 
 
-ClipTranscode::ClipTranscode(const KUrl &src, const QString &params, QWidget * parent) :
-        QDialog(parent)
+ClipTranscode::ClipTranscode(const KUrl::List &urls, const QString &params, const QStringList &postParams, const QString &description, bool automaticMode, QWidget * parent) :
+    QDialog(parent), m_urls(urls), m_duration(0), m_automaticMode(automaticMode), m_postParams(postParams)
 {
     setFont(KGlobalSettings::toolBarFont());
-    m_view.setupUi(this);
+    setupUi(this);
     setAttribute(Qt::WA_DeleteOnClose);
+#if KDE_IS_VERSION(4,7,0)
+    m_infoMessage = new KMessageWidget;
+    QGridLayout *s =  static_cast <QGridLayout*> (layout());
+    s->addWidget(m_infoMessage, 10, 0, 1, -1);
+    m_infoMessage->setCloseButtonVisible(false);
+    m_infoMessage->hide();
+#endif
+    log_text->setHidden(true);
     setWindowTitle(i18n("Transcode Clip"));
+    if (m_automaticMode) {
+        auto_add->setHidden(true);
+    }
+    auto_add->setText(i18np("Add clip to project", "Add clips to project", m_urls.count()));
+    auto_add->setChecked(KdenliveSettings::add_new_clip());
 
-    QString fileName = src.path(); //.section('.', 0, -1);
-    QString newFile = params.section(' ', -1).replace("%1", fileName);
-    KUrl dest(newFile);
-    m_view.source_url->setUrl(src);
-    m_view.dest_url->setUrl(dest);
-    connect(m_view.source_url, SIGNAL(textChanged(const QString &)), this, SLOT(slotUpdateParams()));
+    if (m_urls.count() == 1) {
+        QString fileName = m_urls.at(0).path(); //.section('.', 0, -1);
+        QString newFile = params.section(' ', -1).replace("%1", fileName);
+        KUrl dest(newFile);
+        source_url->setUrl(m_urls.at(0));
+        dest_url->setMode(KFile::File);
+        dest_url->setUrl(dest);
+        dest_url->fileDialog()->setOperationMode(KFileDialog::Saving);
+        urls_list->setHidden(true);
+        connect(source_url, SIGNAL(textChanged(QString)), this, SLOT(slotUpdateParams()));
+    } else {
+        label_source->setHidden(true);
+        source_url->setHidden(true);
+        label_dest->setText(i18n("Destination folder"));
+        dest_url->setMode(KFile::Directory);
+        dest_url->setUrl(KUrl(m_urls.at(0).directory()));
+        dest_url->fileDialog()->setOperationMode(KFileDialog::Saving);
+        for (int i = 0; i < m_urls.count(); ++i)
+            urls_list->addItem(m_urls.at(i).path());
+    }
     if (!params.isEmpty()) {
-        m_view.label_profile->setHidden(true);
-        m_view.profile_list->setHidden(true);
-        m_view.params->setPlainText(params.simplified());
+        label_profile->setHidden(true);
+        profile_list->setHidden(true);
+        ffmpeg_params->setPlainText(params.simplified());
+        if (!description.isEmpty()) {
+            transcode_info->setText(description);
+        } else transcode_info->setHidden(true);
     } else {
         // load Profiles
-        KSharedConfigPtr config = KSharedConfig::openConfig("kdenlivetranscodingrc");
+        KSharedConfigPtr config = KSharedConfig::openConfig("kdenlivetranscodingrc", KConfig::CascadeConfig);
         KConfigGroup transConfig(config, "Transcoding");
         // read the entries
         QMap< QString, QString > profiles = transConfig.entryMap();
         QMapIterator<QString, QString> i(profiles);
-        connect(m_view.profile_list, SIGNAL(currentIndexChanged(int)), this, SLOT(slotUpdateParams(int)));
         while (i.hasNext()) {
             i.next();
-            m_view.profile_list->addItem(i.key(), i.value());
+            QStringList data = i.value().split(';');
+            profile_list->addItem(i.key(), data.at(0));
+            if (data.count() > 1) profile_list->setItemData(profile_list->count() - 1, data.at(1), Qt::UserRole + 1);
         }
+        connect(profile_list, SIGNAL(currentIndexChanged(int)), this, SLOT(slotUpdateParams(int)));
+        slotUpdateParams(0);
     }
 
-    kDebug() << "//PARAMS: " << params << "\n\nNAME: " << newFile;
-
-    connect(m_view.button_start, SIGNAL(clicked()), this, SLOT(slotStartTransCode()));
+    connect(button_start, SIGNAL(clicked()), this, SLOT(slotStartTransCode()));
 
     m_transcodeProcess.setProcessChannelMode(QProcess::MergedChannels);
     connect(&m_transcodeProcess, SIGNAL(readyReadStandardOutput()), this, SLOT(slotShowTranscodeInfo()));
-    connect(&m_transcodeProcess, SIGNAL(finished(int, QProcess::ExitStatus)), this, SLOT(slotTranscodeFinished(int, QProcess::ExitStatus)));
+    connect(&m_transcodeProcess, SIGNAL(finished(int,QProcess::ExitStatus)), this, SLOT(slotTranscodeFinished(int,QProcess::ExitStatus)));
+    
+    ffmpeg_params->setMaximumHeight(QFontMetrics(font()).lineSpacing() * 5);
 
-    //adjustSize();
+    adjustSize();
 }
 
 ClipTranscode::~ClipTranscode()
 {
+    KdenliveSettings::setAdd_new_clip(auto_add->isChecked());
     if (m_transcodeProcess.state() != QProcess::NotRunning) {
         m_transcodeProcess.close();
     }
+#if KDE_IS_VERSION(4,7,0)
+    delete m_infoMessage;
+#endif
 }
 
 void ClipTranscode::slotStartTransCode()
@@ -80,57 +119,170 @@ void ClipTranscode::slotStartTransCode()
     if (m_transcodeProcess.state() != QProcess::NotRunning) {
         return;
     }
+    m_duration = 0;
+    m_destination.clear();
+#if KDE_IS_VERSION(4,7,0)
+    m_infoMessage->animatedHide();
+#endif
     QStringList parameters;
-    parameters << "-i" << m_view.source_url->url().path();
-
-    if (QFile::exists(m_view.dest_url->url().path())) {
-        if (KMessageBox::questionYesNo(this, i18n("File %1 already exists.\nDo you want to overwrite it?", m_view.dest_url->url().path())) == KMessageBox::No) return;
+    QString destination;
+    QString params = ffmpeg_params->toPlainText().simplified();
+    if (!m_urls.isEmpty() && urls_list->count() > 0) {
+        // We are processing multiple clips
+        source_url->setUrl(m_urls.takeFirst());
+        destination = dest_url->url().path(KUrl::AddTrailingSlash) + source_url->url().fileName();
+        QList<QListWidgetItem *> matching = urls_list->findItems(source_url->url().path(), Qt::MatchExactly);
+        if (matching.count() > 0) {
+            matching.at(0)->setFlags(Qt::ItemIsSelectable);
+            urls_list->setCurrentItem(matching.at(0));
+        }
+    } else {
+        destination = dest_url->url().path().section('.', 0, -2);
+    }
+    QString extension = params.section("%1", 1, 1).section(' ', 0, 0);
+    QString s_url = source_url->url().path();
+    parameters << "-i" << s_url;
+    if (QFile::exists(destination + extension)) {
+        if (KMessageBox::questionYesNo(this, i18n("File %1 already exists.\nDo you want to overwrite it?", destination + extension)) == KMessageBox::No) {
+            // Abort operation
+            if (m_automaticMode) {
+                // inform caller that we aborted
+                emit transcodedClip(source_url->url(), KUrl());
+                close();
+            }
+            return;
+        }
         parameters << "-y";
     }
-    m_view.buttonBox->button(QDialogButtonBox::Abort)->setText(i18n("Abort"));
-    QString params = m_view.params->toPlainText().simplified();
-    params = params.section(' ', 0, -2);
-    parameters << params.split(' ') << m_view.dest_url->url().path();
 
-    kDebug() << "/// FFMPEG ARGS: " << parameters;
+    bool replaceVfParams = false;
+    foreach(QString s, params.split(' ')) {
+        if (replaceVfParams) {
+            s= m_postParams.at(1);
+            replaceVfParams = false;
+        }
+        parameters << s.replace("%1", destination);
+        if (s == "-vf") {
+            replaceVfParams = true;
+        }
+    }
+    
+    buttonBox->button(QDialogButtonBox::Abort)->setText(i18n("Abort"));
 
-    m_transcodeProcess.start("ffmpeg", parameters);
-    m_view.button_start->setEnabled(false);
+    m_destination = destination + extension;
+    m_transcodeProcess.start(KdenliveSettings::ffmpegpath(), parameters);
+    source_url->setEnabled(false);
+    dest_url->setEnabled(false);
+    button_start->setEnabled(false);
 
 }
 
 void ClipTranscode::slotShowTranscodeInfo()
 {
     QString log = QString(m_transcodeProcess.readAll());
-    //kDebug() << "//LOG: " << log;
-    m_view.log->setPlainText(log);
+    if (m_duration == 0) {
+        if (log.contains("Duration:")) {
+            QString data = log.section("Duration:", 1, 1).section(',', 0, 0).simplified();
+            QStringList numbers = data.split(':');
+            if (numbers.size() < 3) return;
+            m_duration = numbers.at(0).toInt() * 3600 + numbers.at(1).toInt() * 60 + numbers.at(2).toDouble();
+            log_text->setHidden(true);
+            job_progress->setHidden(false);
+        }
+        else {
+            log_text->setHidden(false);
+            job_progress->setHidden(true);
+        }
+    }
+    else if (log.contains("time=")) {
+        int progress;
+        QString time = log.section("time=", 1, 1).simplified().section(' ', 0, 0);
+        if (time.contains(':')) {
+            QStringList numbers = time.split(':');
+            if (numbers.size() < 3) return;
+            progress = numbers.at(0).toInt() * 3600 + numbers.at(1).toInt() * 60 + numbers.at(2).toDouble();
+        }
+        else progress = (int) time.toDouble();
+        job_progress->setValue((int) (100.0 * progress / m_duration));
+    }
+    log_text->setPlainText(log);
 }
 
 void ClipTranscode::slotTranscodeFinished(int exitCode, QProcess::ExitStatus exitStatus)
 {
-    m_view.buttonBox->button(QDialogButtonBox::Abort)->setText(i18n("Close"));
-    m_view.button_start->setEnabled(true);
+    buttonBox->button(QDialogButtonBox::Abort)->setText(i18n("Close"));
+    button_start->setEnabled(true);
+    source_url->setEnabled(true);
+    dest_url->setEnabled(true);
+    m_duration = 0;
 
+    if (QFileInfo(m_destination).size() <= 0) {
+        // Destination file does not exist, transcoding failed
+        exitCode = 1;
+    }
     if (exitCode == 0 && exitStatus == QProcess::NormalExit) {
-        m_view.log->setHtml(m_view.log->toPlainText() + "<br><b>" + i18n("Transcoding finished."));
-        if (m_view.auto_add->isChecked()) emit addClip(m_view.dest_url->url());
-        if (m_view.auto_close->isChecked()) accept();
+        log_text->setHtml(log_text->toPlainText() + "<br /><b>" + i18n("Transcoding finished."));
+        if (auto_add->isChecked() || m_automaticMode) {
+            KUrl url;
+            if (urls_list->count() > 0) {
+                QString params = ffmpeg_params->toPlainText().simplified();
+                QString extension = params.section("%1", 1, 1).section(' ', 0, 0);
+                url = KUrl(dest_url->url().path(KUrl::AddTrailingSlash) + source_url->url().fileName() + extension);
+            } else url = dest_url->url();
+            if (m_automaticMode) emit transcodedClip(source_url->url(), url);
+            else emit addClip(url);
+        }
+        if (urls_list->count() > 0 && m_urls.count() > 0) {
+            m_transcodeProcess.close();
+            slotStartTransCode();
+            return;
+        } else if (auto_close->isChecked()) accept();
+        else {
+#if KDE_IS_VERSION(4,7,0)
+            m_infoMessage->setMessageType(KMessageWidget::Positive);
+            m_infoMessage->setText(i18n("Transcoding finished."));
+            m_infoMessage->animatedShow();
+#else
+            log_text->setVisible(true);
+#endif
+        }
     } else {
-        m_view.log->setHtml(m_view.log->toPlainText() + "<br><b>" + i18n("Transcoding FAILED!"));
+#if KDE_IS_VERSION(4,7,0)
+        m_infoMessage->setMessageType(KMessageWidget::Warning);
+        m_infoMessage->setText(i18n("Transcoding failed!"));
+        m_infoMessage->animatedShow();
+#else
+        log_text->setHtml(log_text->toPlainText() + "<br /><b>" + i18n("Transcoding failed!"));
+#endif
+        log_text->setVisible(true);
     }
-
     m_transcodeProcess.close();
+    
+    //Refill url list in case user wants to transcode to another format
+    if (urls_list->count() > 0) {
+        m_urls.clear();
+        for (int i = 0; i < urls_list->count(); ++i)
+            m_urls << urls_list->item(i)->text();
+    }
 }
 
 void ClipTranscode::slotUpdateParams(int ix)
 {
-    if (ix == -1) ix = m_view.profile_list->currentIndex();
-    QString fileName = m_view.source_url->url().path();
-    QString params = m_view.profile_list->itemData(ix).toString();
-    QString newFile = params.section(' ', -1).replace("%1", fileName);
-    KUrl dest(newFile);
-    m_view.dest_url->setUrl(dest);
-    m_view.params->setPlainText(params.simplified());
+    QString fileName = source_url->url().path();
+    if (ix != -1) {
+        QString params = profile_list->itemData(ix).toString();
+        ffmpeg_params->setPlainText(params.simplified());
+        QString desc = profile_list->itemData(ix, Qt::UserRole + 1).toString();
+        if (!desc.isEmpty()) {
+            transcode_info->setText(desc);
+            transcode_info->setHidden(false);
+        } else transcode_info->setHidden(true);
+    }
+    if (urls_list->count() == 0) {
+        QString newFile = ffmpeg_params->toPlainText().simplified().section(' ', -1).replace("%1", fileName);
+        dest_url->setUrl(KUrl(newFile));
+    }
+
 }
 
 #include "cliptranscode.moc"