]> git.sesse.net Git - kdenlive/blobdiff - src/mainwindow.cpp
Save vertical zoom in project:
[kdenlive] / src / mainwindow.cpp
index 52dc63f6b3e1aa23e62d71cecb99f586bd41a34a..ef285aa56a2e89a857a11d2ffc31cd033b5a3a22 100644 (file)
@@ -86,6 +86,7 @@
 #include <QTimer>
 #include <QAction>
 #include <QKeyEvent>
+#include <QInputDialog>
 
 #include <stdlib.h>
 
@@ -390,6 +391,22 @@ void MainWindow::queryQuit()
 //virtual
 bool MainWindow::queryClose()
 {
+    if (m_renderWidget) {
+        int waitingJobs = m_renderWidget->waitingJobsCount();
+        if (waitingJobs > 0) {
+            switch (KMessageBox::warningYesNoCancel(this, i18n("You have %1 rendering jobs waiting in the queue.\nWhat do you want to do with these jobs?", waitingJobs), QString(), KGuiItem(i18n("Start them now")), KGuiItem(i18n("Delete them")))) {
+            case KMessageBox::Yes :
+                // create script with waiting jobs and start it
+                if (m_renderWidget->startWaitingRenderJobs() == false) return false;
+                break;
+            case KMessageBox::No :
+                // Don't do anything, jobs will be deleted
+                break;
+            default:
+                return false;
+            }
+        }
+    }
     saveOptions();
     if (m_monitorManager) m_monitorManager->stopActiveMonitor();
     if (m_activeDocument && m_activeDocument->isModified()) {
@@ -941,11 +958,13 @@ void MainWindow::setupActions()
     connect(stickTransition, SIGNAL(triggered(bool)), this, SLOT(slotAutoTransition()));
 
     KAction* groupClip = new KAction(KIcon("object-group"), i18n("Group Clips"), this);
+    groupClip->setShortcut(Qt::CTRL + Qt::Key_G);
     collection->addAction("group_clip", groupClip);
     connect(groupClip, SIGNAL(triggered(bool)), this, SLOT(slotGroupClips()));
 
     KAction* ungroupClip = new KAction(KIcon("object-ungroup"), i18n("Ungroup Clips"), this);
     collection->addAction("ungroup_clip", ungroupClip);
+    ungroupClip->setShortcut(Qt::CTRL + Qt::SHIFT + Qt::Key_G);
     ungroupClip->setData("ungroup_clip");
     connect(ungroupClip, SIGNAL(triggered(bool)), this, SLOT(slotUnGroupClips()));
 
@@ -1499,7 +1518,7 @@ void MainWindow::parseProfiles(const QString &mltPath)
             KUrl mltPath = getUrl->selectedUrl();
             delete getUrl;
             if (mltPath.isEmpty()) ::exit(0);
-            KdenliveSettings::setMltpath(mltPath.path());
+            KdenliveSettings::setMltpath(mltPath.path(KUrl::AddTrailingSlash));
             QStringList profilesList = QDir(KdenliveSettings::mltpath()).entryList(profilesFilter, QDir::Files);
         }
     }
@@ -1566,7 +1585,7 @@ void MainWindow::slotEditProjectSettings()
     if (w->exec() == QDialog::Accepted) {
         QString profile = w->selectedProfile();
         m_activeDocument->setProjectFolder(w->selectedFolder());
-        if (m_renderWidget) m_renderWidget->setDocumentPath(w->selectedFolder().path());
+        if (m_renderWidget) m_renderWidget->setDocumentPath(w->selectedFolder().path(KUrl::AddTrailingSlash));
         if (m_activeDocument->profilePath() != profile) {
             // Profile was changed
             double dar = m_activeDocument->dar();
@@ -1589,16 +1608,17 @@ void MainWindow::slotEditProjectSettings()
 void MainWindow::slotRenderProject()
 {
     if (!m_renderWidget) {
-        QString projectfolder = m_activeDocument ? m_activeDocument->projectFolder().path() : KdenliveSettings::defaultprojectfolder();
+        QString projectfolder = m_activeDocument ? m_activeDocument->projectFolder().path(KUrl::AddTrailingSlash) : KdenliveSettings::defaultprojectfolder();
         m_renderWidget = new RenderWidget(projectfolder, this);
-        connect(m_renderWidget, SIGNAL(doRender(const QStringList&, const QStringList&)), this, SLOT(slotDoRender(const QStringList&, const QStringList&)));
         connect(m_renderWidget, SIGNAL(selectedRenderProfile(const QString &, const QString &)), this, SLOT(slotSetDocumentRenderProfile(const QString &, const QString &)));
+        connect(m_renderWidget, SIGNAL(prepareRenderingData(bool, bool, const QString&)), this, SLOT(slotPrepareRendering(bool, bool, const QString&)));
         connect(m_renderWidget, SIGNAL(abortProcess(const QString &)), this, SIGNAL(abortRenderJob(const QString &)));
         connect(m_renderWidget, SIGNAL(openDvdWizard(const QString &, const QString &)), this, SLOT(slotDvdWizard(const QString &, const QString &)));
         if (m_activeDocument) {
             m_renderWidget->setProfile(m_activeDocument->mltProfile());
             m_renderWidget->setGuides(m_activeDocument->guidesXml(), m_activeDocument->projectDuration());
             m_renderWidget->setRenderProfile(m_activeDocument->getDocumentProperty("renderdestination"), m_activeDocument->getDocumentProperty("renderprofile"));
+            m_renderWidget->setDocumentPath(m_activeDocument->projectFolder().path(KUrl::AddTrailingSlash));
         }
     }
     /*TrackView *currentTab = (TrackView *) m_timelineArea->currentWidget();
@@ -1608,156 +1628,6 @@ void MainWindow::slotRenderProject()
     m_renderWidget->showNormal();
 }
 
-void MainWindow::slotDoRender(const QStringList args, const QStringList overlay_args)
-{
-    QString dest = args.at(0);
-    QString render = args.at(1);
-    QStringList avformat_args = args.at(2).split(' ');
-    bool zoneOnly = args.at(3).toInt();
-    bool playAfter = args.at(4).toInt();
-    double guideStart = args.at(5).toDouble();
-    double guideEnd = args.at(6).toDouble();
-    bool resizeProfile = args.at(7).toInt();
-    QString scriptExport = args.at(8);
-    bool createChapterFile = args.at(9).toInt();
-
-    if (dest.isEmpty()) return;
-    int in = 0;
-    int out = 0;
-
-    if (m_activeTimeline && zoneOnly) {
-        in = m_activeTimeline->inPoint();
-        out = m_activeTimeline->outPoint();
-    }
-
-    KTemporaryFile temp;
-    temp.setAutoRemove(false);
-    temp.setSuffix(".mlt");
-    if (!scriptExport.isEmpty() || temp.open()) {
-        if (KdenliveSettings::dropbframes()) {
-            KdenliveSettings::setDropbframes(false);
-            m_activeDocument->clipManager()->updatePreviewSettings();
-            if (!scriptExport.isEmpty()) m_projectMonitor->saveSceneList(scriptExport + ".mlt");
-            else m_projectMonitor->saveSceneList(temp.fileName());
-            KdenliveSettings::setDropbframes(true);
-            m_activeDocument->clipManager()->updatePreviewSettings();
-        } else {
-            if (!scriptExport.isEmpty()) m_projectMonitor->saveSceneList(scriptExport + ".mlt");
-            else m_projectMonitor->saveSceneList(temp.fileName());
-        }
-
-        QStringList args;
-        if (scriptExport.isEmpty()) args << "-erase";
-        if (KdenliveSettings::usekuiserver()) args << "-kuiserver";
-        if (zoneOnly) args << "in=" + QString::number(in) << "out=" + QString::number(out);
-        else if (guideStart != -1) {
-            args << "in=" + QString::number(GenTime(guideStart).frames(m_activeDocument->fps())) << "out=" + QString::number(GenTime(guideEnd).frames(m_activeDocument->fps()));
-        }
-        if (!overlay_args.isEmpty()) args << "preargs=" + overlay_args.join(" ");
-        QString videoPlayer = "-";
-        if (playAfter) {
-            videoPlayer = KdenliveSettings::defaultplayerapp();
-            if (videoPlayer.isEmpty()) KMessageBox::sorry(this, i18n("Cannot play video after rendering because the default video player application is not set.\nPlease define it in Kdenlive settings dialog."));
-        }
-        if (!QFile::exists(KdenliveSettings::rendererpath())) {
-            KMessageBox::sorry(this, i18n("Cannot find the melt program required for rendering (part of Mlt)"));
-            setRenderingProgress(dest, -3);
-            return;
-        }
-        args << KdenliveSettings::rendererpath() << m_activeDocument->profilePath() << render << videoPlayer;
-
-        for (int i = 0; i < avformat_args.count(); i++) {
-            if (avformat_args.at(i).startsWith("profile=")) {
-                if (avformat_args.at(i).section('=', 1) != m_activeDocument->profilePath()) resizeProfile = true;
-                break;
-            }
-        }
-
-        if (resizeProfile) {
-            // The rendering profile is different from project profile, so use MLT's special producer_consumer
-            if (scriptExport.isEmpty()) args << "consumer:" + temp.fileName();
-            else args << "consumer:$SOURCE";
-        } else {
-            if (scriptExport.isEmpty()) args << temp.fileName();
-            else args << "$SOURCE";
-        }
-        if (scriptExport.isEmpty()) args << dest;
-        else args << "$TARGET";
-        args << avformat_args;
-        QString renderer = QCoreApplication::applicationDirPath() + QString("/kdenlive_render");
-        if (!QFile::exists(renderer)) renderer = "kdenlive_render";
-        if (scriptExport.isEmpty()) {
-            QProcess::startDetached(renderer, args);
-            KNotification::event("RenderStarted", i18n("Rendering <i>%1</i> started", dest), QPixmap(), this);
-        } else {
-            // Generate script file
-            QFile file(scriptExport);
-            if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
-                KMessageBox::error(this, i18n("Cannot write to file %1", scriptExport));
-                return;
-            }
-
-            QTextStream outStream(&file);
-            outStream << "#! /bin/sh" << "\n" << "\n";
-            outStream << "SOURCE=" << "\"" + scriptExport + ".mlt\"" << "\n";
-            outStream << "TARGET=" << "\"" + dest + "\"" << "\n";
-            outStream << renderer << " " << args.join(" ") << "\n" << "\n";
-            if (file.error() != QFile::NoError) {
-                KMessageBox::error(this, i18n("Cannot write to file %1", scriptExport));
-                file.close();
-                return;
-            }
-            file.close();
-            QFile::setPermissions(scriptExport, file.permissions() | QFile::ExeUser);
-        }
-
-        if (createChapterFile) {
-            QDomDocument doc;
-            QDomElement chapters = doc.createElement("chapters");
-            chapters.setAttribute("fps", m_activeDocument->fps());
-            doc.appendChild(chapters);
-
-            QDomElement guidesxml = m_activeDocument->guidesXml();
-            if (!zoneOnly) out = (int) GenTime(m_activeDocument->projectDuration()).frames(m_activeDocument->fps());
-
-            QDomNodeList nodes = guidesxml.elementsByTagName("guide");
-            for (int i = 0; i < nodes.count(); i++) {
-                QDomElement e = nodes.item(i).toElement();
-                if (!e.isNull()) {
-                    QString comment = e.attribute("comment");
-                    int time = (int) GenTime(e.attribute("time").toDouble()).frames(m_activeDocument->fps());
-                    if (time >= in && time < out) {
-                        if (zoneOnly) time = time - in;
-                        QDomElement chapter = doc.createElement("chapter");
-                        chapters.appendChild(chapter);
-                        chapter.setAttribute("title", comment);
-                        chapter.setAttribute("time", time);
-                    }
-                }
-            }
-            if (chapters.childNodes().count() > 0) {
-                if (m_activeTimeline->projectView()->hasGuide(out, 0) == -1) {
-                    // Always insert a guide in pos 0
-                    QDomElement chapter = doc.createElement("chapter");
-                    chapters.insertBefore(chapter, QDomNode());
-                    chapter.setAttribute("title", i18n("Start"));
-                    chapter.setAttribute("time", "0");
-                }
-                // save chapters file
-                QFile file(dest + ".dvdchapter");
-                if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
-                    kWarning() << "//////  ERROR writing DVD CHAPTER file: " << dest + ".dvdchapter";
-                } else {
-                    file.write(doc.toString().toUtf8());
-                    if (file.error() != QFile::NoError)
-                        kWarning() << "//////  ERROR writing DVD CHAPTER file: " << dest + ".dvdchapter";
-                    file.close();
-                }
-            }
-        }
-    }
-}
-
 void MainWindow::setRenderingProgress(const QString &url, int progress)
 {
     if (m_renderWidget) m_renderWidget->setRenderJob(url, progress);
@@ -1891,7 +1761,7 @@ void MainWindow::connectDocument(TrackView *trackView, KdenliveDoc *doc)   //cha
     connect(trackView->projectView(), SIGNAL(clipItemSelected(ClipItem*, int)), this, SLOT(slotActivateEffectStackView()));
     connect(trackView->projectView(), SIGNAL(transitionItemSelected(Transition*, int, QPoint, bool)), m_transitionConfig, SLOT(slotTransitionItemSelected(Transition*, int, QPoint, bool)));
     connect(trackView->projectView(), SIGNAL(transitionItemSelected(Transition*, int, QPoint, bool)), this, SLOT(slotActivateTransitionView(Transition *)));
-    m_zoomSlider->setValue(doc->zoom());
+    m_zoomSlider->setValue(doc->zoom().x());
     connect(m_zoomSlider, SIGNAL(valueChanged(int)), trackView, SLOT(slotChangeZoom(int)));
     connect(trackView->projectView(), SIGNAL(zoomIn()), this, SLOT(slotZoomIn()));
     connect(trackView->projectView(), SIGNAL(zoomOut()), this, SLOT(slotZoomOut()));
@@ -1917,7 +1787,8 @@ void MainWindow::connectDocument(TrackView *trackView, KdenliveDoc *doc)   //cha
     m_activeTimeline = trackView;
     if (m_renderWidget) {
         m_renderWidget->setProfile(doc->mltProfile());
-        m_renderWidget->setDocumentPath(doc->projectFolder().path());
+        m_renderWidget->setGuides(doc->guidesXml(), doc->projectDuration());
+        m_renderWidget->setDocumentPath(doc->projectFolder().path(KUrl::AddTrailingSlash));
         m_renderWidget->setRenderProfile(doc->getDocumentProperty("renderdestination"), doc->getDocumentProperty("renderprofile"));
     }
     //doc->setRenderer(m_projectMonitor->render);
@@ -2330,7 +2201,7 @@ void MainWindow::slotGotProgressInfo(const QString &message, int progress)
 void MainWindow::slotShowClipProperties(DocClipBase *clip)
 {
     if (clip->clipType() == TEXT) {
-        QString titlepath = m_activeDocument->projectFolder().path() + "/titles/";
+        QString titlepath = m_activeDocument->projectFolder().path(KUrl::AddTrailingSlash) + "titles/";
         if (!clip->getProperty("xmltemplate").isEmpty()) {
             // template text clip
 
@@ -2812,4 +2683,86 @@ void MainWindow::slotSetDocumentRenderProfile(const QString &dest, const QString
     m_activeDocument->setModified(true);
 }
 
+
+void MainWindow::slotPrepareRendering(bool scriptExport, bool zoneOnly, const QString &chapterFile)
+{
+    if (m_activeDocument == NULL || m_renderWidget == NULL) return;
+    QString scriptPath;
+    QString playlistPath;
+    if (scriptExport) {
+        bool ok;
+        QString scriptsFolder = m_activeDocument->projectFolder().path() + "/scripts/";
+        QString path = m_renderWidget->getFreeScriptName();
+        scriptPath = QInputDialog::getText(this, i18n("Create Render Script"), i18n("Script name (will be saved in: %1)", scriptsFolder), QLineEdit::Normal, KUrl(path).fileName(), &ok);
+        if (!ok || scriptPath.isEmpty()) return;
+        scriptPath.prepend(scriptsFolder);
+        QFile f(scriptPath);
+        if (f.exists()) {
+            if (KMessageBox::warningYesNo(this, i18n("Script file already exists. Do you want to overwrite it?")) != KMessageBox::Yes)
+                return;
+        }
+        playlistPath = scriptPath + ".mlt";
+        m_projectMonitor->saveSceneList(playlistPath);
+    } else {
+        KTemporaryFile temp;
+        temp.setAutoRemove(false);
+        temp.setSuffix(".mlt");
+        temp.open();
+        playlistPath = temp.fileName();
+        m_projectMonitor->saveSceneList(playlistPath);
+    }
+
+    if (!chapterFile.isEmpty()) {
+        int in = 0;
+        int out;
+        if (!zoneOnly) out = (int) GenTime(m_activeDocument->projectDuration()).frames(m_activeDocument->fps());
+        else {
+            in = m_activeTimeline->inPoint();
+            out = m_activeTimeline->outPoint();
+        }
+        QDomDocument doc;
+        QDomElement chapters = doc.createElement("chapters");
+        chapters.setAttribute("fps", m_activeDocument->fps());
+        doc.appendChild(chapters);
+
+        QDomElement guidesxml = m_activeDocument->guidesXml();
+        QDomNodeList nodes = guidesxml.elementsByTagName("guide");
+        for (int i = 0; i < nodes.count(); i++) {
+            QDomElement e = nodes.item(i).toElement();
+            if (!e.isNull()) {
+                QString comment = e.attribute("comment");
+                int time = (int) GenTime(e.attribute("time").toDouble()).frames(m_activeDocument->fps());
+                if (time >= in && time < out) {
+                    if (zoneOnly) time = time - in;
+                    QDomElement chapter = doc.createElement("chapter");
+                    chapters.appendChild(chapter);
+                    chapter.setAttribute("title", comment);
+                    chapter.setAttribute("time", time);
+                }
+            }
+        }
+        if (chapters.childNodes().count() > 0) {
+            if (m_activeTimeline->projectView()->hasGuide(out, 0) == -1) {
+                // Always insert a guide in pos 0
+                QDomElement chapter = doc.createElement("chapter");
+                chapters.insertBefore(chapter, QDomNode());
+                chapter.setAttribute("title", i18n("Start"));
+                chapter.setAttribute("time", "0");
+            }
+            // save chapters file
+            QFile file(chapterFile);
+            if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
+                kWarning() << "//////  ERROR writing DVD CHAPTER file: " << chapterFile;
+            } else {
+                file.write(doc.toString().toUtf8());
+                if (file.error() != QFile::NoError)
+                    kWarning() << "//////  ERROR writing DVD CHAPTER file: " << chapterFile;
+                file.close();
+            }
+        }
+    }
+
+    m_renderWidget->slotExport(scriptExport, m_activeTimeline->inPoint(), m_activeTimeline->outPoint(), playlistPath, scriptPath);
+}
+
 #include "mainwindow.moc"