]> git.sesse.net Git - kdenlive/commitdiff
Cleanup data processing framework
authorJean-Baptiste Mardelle <jb@kdenlive.org>
Sun, 21 Oct 2012 09:48:18 +0000 (11:48 +0200)
committerJean-Baptiste Mardelle <jb@kdenlive.org>
Sun, 21 Oct 2012 09:48:18 +0000 (11:48 +0200)
20 files changed:
effects/automask.xml
effects/sox_gain.xml
src/CMakeLists.txt
src/clipproperties.cpp
src/clipproperties.h
src/customtrackview.cpp
src/customtrackview.h
src/effectstack/collapsibleeffect.cpp
src/effectstack/collapsibleeffect.h
src/effectstack/effectstackview2.cpp
src/effectstack/effectstackview2.h
src/effectstack/parametercontainer.cpp
src/effectstack/parametercontainer.h
src/mainwindow.cpp
src/mainwindow.h
src/projectlist.cpp
src/projectlist.h
src/projecttree/meltjob.cpp
src/projecttree/meltjob.h
src/widgets/scenecutdialog_ui.ui [new file with mode: 0644]

index d2bde1acc79f5efa3b24183a089d1b26d8b15357..1d5f0b2df71c25d522a7c571640cf88fb3c52d5f 100644 (file)
@@ -12,7 +12,7 @@
        <parameter type="bool" name="obscure" default="1">
                <name>Obscure</name>
        </parameter>
-       <parameter type="filterjob" filtertag="autotrack_rectangle:%geometry" filterparams="_serialize=1" consumer="null" consumerparams="all=1 terminate_on_pause=1" wantedproperties="motion_vector_list" finalfilter="autotrack_rectangle" extraparams="projecttreefilter">
+       <parameter type="filterjob" filtertag="autotrack_rectangle:%geometry" filterparams="_serialize=1" consumer="null" consumerparams="all=1 terminate_on_pause=1" finalfilter="autotrack_rectangle" extraparams="key:motion_vector_list projecttreefilter storedata">
                <name>Analyse</name>
        </parameter>
 </effect>
index d40552b5242a1a4c0a6ead625972c7e270088bcb..0a67a67403d81e7ef4fde3dfd5f375d258da3b90 100644 (file)
@@ -6,7 +6,7 @@
        <parameter type="double" name="gain" max="50" min="-50" default="5.00" decimals="2">
                <name>Gain</name>
        </parameter>
-       <parameter type="filterjob" filtertag="sox:analysis" filterparams="" consumer="null" consumerparams="video_off=1 all=1 terminate_on_pause=1" wantedproperties="gain" finalfilter="sox_gain">
+       <parameter type="filterjob" filtertag="sox:analysis" filterparams="" consumer="null" consumerparams="video_off=1 all=1 terminate_on_pause=1" extraparams="key:gain" finalfilter="sox_gain">
                <name>Normalize</name>
        </parameter>
 </effect>
index 597dfbb02a09efb692ef2b9d139a7fe3e808fa02..74c44596c678963f9c33951e8cd6a00ddc14cb11 100644 (file)
@@ -262,6 +262,7 @@ kde4_add_ui_files(kdenlive_UIS
   widgets/keywordval_ui.ui
   widgets/fontval_ui.ui
   widgets/cutjobdialog_ui.ui
+  widgets/scenecutdialog_ui.ui
 )
 
 if(OPENGL_FOUND)
index 7981f9537cb666953525e5fcc801e6b24da59875..1d5e39ee5dce7e62cce42b33e7d3b7fee1bf9933 100644 (file)
@@ -771,7 +771,9 @@ void ClipProperties::slotAddMarker()
     QPointer<MarkerDialog> d = new MarkerDialog(m_clip, marker,
                                           m_tc, i18n("Add Marker"), this);
     if (d->exec() == QDialog::Accepted) {
-        emit addMarker(m_clip->getId(), d->newMarker());
+       QList <CommentedTime> markers;
+       markers << d->newMarker();
+        emit addMarkers(m_clip->getId(), markers);
     }
     delete d;
 }
@@ -793,7 +795,9 @@ void ClipProperties::slotEditMarker()
     if (pos < 0 || pos > marks.count() - 1) return;
     MarkerDialog d(m_clip, marks.at(pos), m_tc, i18n("Edit Marker"), this);
     if (d.exec() == QDialog::Accepted) {
-        emit addMarker(m_clip->getId(), d.newMarker());
+       QList <CommentedTime> markers;
+       markers << d.newMarker();
+        emit addMarkers(m_clip->getId(), markers);
     }
 }
 
@@ -808,8 +812,7 @@ void ClipProperties::slotDeleteMarker()
            toDelete << marker;
        }
     }
-    for (int i = 0; i < toDelete.count(); i++)
-       emit addMarker(m_clip->getId(), toDelete.at(i));
+    emit addMarkers(m_clip->getId(), toDelete);
 }
 
 void ClipProperties::slotDeleteAnalysis()
index 97ac8dd6a9713834d246b6aea3482f36e6b2273d..ec0dee89ae091b2b8794af73bd731459d3a054c8 100644 (file)
@@ -93,7 +93,7 @@ private:
     QFrame* m_proxyContainer;
 
 signals:
-    void addMarker(const QString &, CommentedTime);
+    void addMarkers(const QString &, QList <CommentedTime>);
     void deleteProxy(const QString);
     void applyNewClipProperties(const QString, QMap <QString, QString> , QMap <QString, QString> , bool, bool);
     void saveMarkers(const QString &);
index 6d6d548291807daf60ecf690ebcfffa59f5e8d7b..45834e5da57414971620ad65373f2949670a002d 100644 (file)
@@ -5269,15 +5269,31 @@ void CustomTrackView::slotAddClipExtraData(const QString &id, const QString &key
     if (!groupCommand) m_commandStack->push(command);
 }
 
-void CustomTrackView::slotAddClipMarker(const QString &id, CommentedTime newMarker, QUndoCommand *groupCommand)
-{
-    CommentedTime oldMarker = m_document->clipManager()->getClipById(id)->markerAt(newMarker.time());
-    if (oldMarker == CommentedTime()) {
-       oldMarker = newMarker;
-       oldMarker.setMarkerType(-1);
+void CustomTrackView::slotAddClipMarker(const QString &id, QList <CommentedTime> newMarkers, QUndoCommand *groupCommand)
+{
+    QUndoCommand *subCommand = NULL;
+    if (newMarkers.count() > 1 && groupCommand == NULL) {
+       subCommand = new QUndoCommand;
+       subCommand->setText("Add markers");
+    }
+    for (int i = 0; i < newMarkers.count(); i++) {
+       CommentedTime oldMarker = m_document->clipManager()->getClipById(id)->markerAt(newMarkers.at(i).time());
+       if (oldMarker == CommentedTime()) {
+           oldMarker = newMarkers.at(i);
+           oldMarker.setMarkerType(-1);
+       }
+       if (newMarkers.count() == 1 && !groupCommand) {
+           AddMarkerCommand *command = new AddMarkerCommand(this, oldMarker, newMarkers.at(i), id, groupCommand);
+           m_commandStack->push(command);
+       }
+       else if (groupCommand) {
+           (void) new AddMarkerCommand(this, oldMarker, newMarkers.at(i), id, groupCommand);
+       }
+       else {
+           (void) new AddMarkerCommand(this, oldMarker, newMarkers.at(i), id, subCommand);
+       }
     }
-    AddMarkerCommand *command = new AddMarkerCommand(this, oldMarker, newMarker, id, groupCommand);
-    if (!groupCommand) m_commandStack->push(command);
+    if (subCommand) m_commandStack->push(subCommand);
 }
 
 void CustomTrackView::slotDeleteClipMarker(const QString &comment, const QString &id, const GenTime &position)
@@ -5388,6 +5404,7 @@ void CustomTrackView::slotLoadClipMarkers(const QString &id)
     QUndoCommand *command = new QUndoCommand();
     command->setText("Load markers");
     QString markerText;
+    QList <CommentedTime> markersList;
     foreach(QString line, lines) {
        markerText.clear();
        values = line.split("\t", QString::SkipEmptyParts);
@@ -5418,13 +5435,14 @@ void CustomTrackView::slotLoadClipMarkers(const QString &id)
            // Marker found, add it
            //TODO: allow user to set a marker category
            CommentedTime marker1(GenTime(time1), markerText, category);
-           slotAddClipMarker(id, marker1, command);
+           markersList << marker1;
            if (time2 > 0 && time2 != time1) {
                CommentedTime marker2(GenTime(time2), markerText, category);
-               slotAddClipMarker(id, marker2, command);
+               markersList << marker2;
            }
        }
     }
+    if (!markersList.isEmpty()) slotAddClipMarker(id, markersList, command);
     if (command->childCount() > 0) m_commandStack->push(command);
     else delete command;
 }
@@ -7472,7 +7490,7 @@ void CustomTrackView::adjustEffects(ClipItem* item, ItemInfo oldInfo, QUndoComma
 }
 
 
-void CustomTrackView::slotGotFilterJobResults(const QString &/*id*/, int startPos, int track, const QString &filter, stringMap filterParams)
+void CustomTrackView::slotGotFilterJobResults(const QString &/*id*/, int startPos, int track, const QString &filter, stringMap filterParams, QStringList extra)
 {
     ClipItem *clip = getClipItemAt(GenTime(startPos, m_document->fps()), track);
     if (clip == NULL) {
index 6ba3bb2a919cffa4035e2622fdc5f2beef6b999a..0c950b54098811adf48efc3d4b9be0bc2ce0519f 100644 (file)
@@ -234,7 +234,7 @@ public slots:
      * @param id Id of the marker's clip
      * @param t Position of the marker
      * @param c Comment of the marker */
-    void slotAddClipMarker(const QString &id, CommentedTime newMarker, QUndoCommand *groupCommand = 0);
+    void slotAddClipMarker(const QString &id, QList <CommentedTime> newMarker, QUndoCommand *groupCommand = 0);
     void slotLoadClipMarkers(const QString &id);
     void slotSaveClipMarkers(const QString &id);
     bool addGuide(const GenTime &pos, const QString &comment);
@@ -494,7 +494,7 @@ private slots:
      *  @param resetThumbs Should we recreate the timeline thumbnails. */
     void slotRefreshThumbs(const QString &id, bool resetThumbs);
     /** @brief A Filter job producer results. */
-    void slotGotFilterJobResults(const QString &id, int startPos, int track, const QString &filter, stringMap filterParams);
+    void slotGotFilterJobResults(const QString &id, int startPos, int track, const QString &filter, stringMap filterParams, QStringList extra);
 
 
 signals:
index 22e2a02b85212c50d5e2c37bf2fd043215c6395f..5d953f19d1ad16cfc29a75a684653ee1cae07d81 100644 (file)
@@ -455,7 +455,7 @@ void CollapsibleEffect::setupWidget(ItemInfo info, EffectMetaInfo *metaInfo)
     }
     connect (m_paramWidget, SIGNAL(parameterChanged(const QDomElement, const QDomElement, int)), this, SIGNAL(parameterChanged(const QDomElement, const QDomElement, int)));
     
-    connect(m_paramWidget, SIGNAL(startFilterJob(QString,QString,QString,QString,QString,QString,QStringList)), this, SIGNAL(startFilterJob(QString,QString,QString,QString,QString,QString,QStringList)));
+    connect(m_paramWidget, SIGNAL(startFilterJob(QString,QString,QString,QString,QString,QStringList)), this, SIGNAL(startFilterJob(QString,QString,QString,QString,QString,QStringList)));
     
     connect (this, SIGNAL(syncEffectsPos(int)), m_paramWidget, SIGNAL(syncEffectsPos(int)));
     connect (m_paramWidget, SIGNAL(checkMonitorPosition(int)), this, SIGNAL(checkMonitorPosition(int)));
index 736cba36465fae775e324e03917abc8987a2d0c2..45398f272221d3e618bbab6d11cdd7e626cd21bb 100644 (file)
@@ -129,7 +129,7 @@ signals:
     void checkMonitorPosition(int);
     void seekTimeline(int);
     /** @brief Start an MLT filter job on this clip. */
-    void startFilterJob(QString filterName, QString filterParams, QString finalFilterName, QString consumer, QString consumerParams, QString properties, QStringList extraParams);
+    void startFilterJob(QString filterName, QString filterParams, QString finalFilterName, QString consumer, QString consumerParams, QStringList extraParams);
     /** @brief An effect was reset, trigger param reload. */
     void resetEffect(int ix);
     /** @brief Ask for creation of a group. */
index b82d96f7da0c3fee26709918e8bb960a9e643efd..f33ecc6c0099f07ad8df8b76dc7458c127b29454 100644 (file)
@@ -271,7 +271,7 @@ void EffectStackView2::connectEffect(CollapsibleEffect *currentEffect)
     // Check drag & drop
     currentEffect->installEventFilter( this );
     connect(currentEffect, SIGNAL(parameterChanged(const QDomElement, const QDomElement, int)), this , SLOT(slotUpdateEffectParams(const QDomElement, const QDomElement, int)));
-    connect(currentEffect, SIGNAL(startFilterJob(QString,QString,QString,QString,QString,QString,QStringList)), this , SLOT(slotStartFilterJob(QString,QString,QString,QString,QString,QString,QStringList)));
+    connect(currentEffect, SIGNAL(startFilterJob(QString,QString,QString,QString,QString,QStringList)), this , SLOT(slotStartFilterJob(QString,QString,QString,QString,QString,QStringList)));
     connect(currentEffect, SIGNAL(deleteEffect(const QDomElement)), this , SLOT(slotDeleteEffect(const QDomElement)));
     connect(currentEffect, SIGNAL(reloadEffects()), this , SIGNAL(reloadEffects()));
     connect(currentEffect, SIGNAL(resetEffect(int)), this , SLOT(slotResetEffect(int)));
@@ -643,10 +643,10 @@ void EffectStackView2::slotMoveEffectUp(QList <int> indexes, bool up)
     else emit changeEffectPosition(m_clipref, -1, indexes, endPos);
 }
 
-void EffectStackView2::slotStartFilterJob(const QString&filterName, const QString&filterParams, const QString&finalFilterName, const QString&consumer, const QString&consumerParams, const QString&properties, const QStringList &extraParams)
+void EffectStackView2::slotStartFilterJob(const QString&filterName, const QString&filterParams, const QString&finalFilterName, const QString&consumer, const QString&consumerParams, const QStringList &extraParams)
 {
     if (!m_clipref) return;
-    emit startFilterJob(m_clipref->info(), m_clipref->clipProducer(), filterName, filterParams, finalFilterName, consumer, consumerParams, properties, extraParams);
+    emit startFilterJob(m_clipref->info(), m_clipref->clipProducer(), filterName, filterParams, finalFilterName, consumer, consumerParams, extraParams);
 }
 
 void EffectStackView2::slotResetEffect(int ix)
index 419b07c6edc31b54754b063a8731df386a674a50..4ce9580f8a9137bba7e35f94f5f2b888df9595ad 100644 (file)
@@ -167,7 +167,7 @@ private slots:
     void slotSetCurrentEffect(int ix);
     
     /** @brief Triggers a filter job on this clip. */
-    void slotStartFilterJob(const QString&filterName, const QString&filterParams, const QString&finalFilterName, const QString&consumer, const QString&consumerParams, const QString&properties,const QStringList &extraParams);
+    void slotStartFilterJob(const QString&filterName, const QString&filterParams, const QString&finalFilterName, const QString&consumer, const QString&consumerParams, const QStringList &extraParams);
     
     /** @brief Reset an effect to its default values. */
     void slotResetEffect(int ix);
@@ -223,7 +223,7 @@ signals:
     void updateClipRegion(ClipItem*, int, QString);
     void displayMessage(const QString&, int);
     void showComments(bool show);
-    void startFilterJob(ItemInfo info, const QString &clipId, const QString &filterName, const QString &filterParams, const QString&finalFilterName, const QString &consumer, const QString &consumerParams, const QString &properties,const QStringList &extraParams);
+    void startFilterJob(ItemInfo info, const QString &clipId, const QString &filterName, const QString &filterParams, const QString&finalFilterName, const QString &consumer, const QString &consumerParams, const QStringList &extraParams);
     void addEffect(ClipItem*,QDomElement);
 };
 
index 8bd9f709c374dda5f5f6eff70d7bf98841dbb559..9aaf36c12d0c928eca41de8514de561a507b82f6 100644 (file)
@@ -816,10 +816,9 @@ void ParameterContainer::slotStartFilterJobAction()
                    kDebug()<<"// Setting geometry: "<<data<<", RES: "<<filtertag;
                }
            }
-           QStringList extra;
-           extra = pa.attribute("extraparams").split(' ', QString::SkipEmptyParts);
-            emit startFilterJob(filtertag, pa.attribute("filterparams"), pa.attribute("finalfilter"), pa.attribute("consumer"), pa.attribute("consumerparams"), pa.attribute("wantedproperties"), extra);
-            kDebug()<<" - - -PROPS:\n"<<"filtertag"<<"-"<< pa.attribute("filterparams")<<"-"<< pa.attribute("consumer")<<"-"<< pa.attribute("consumerparams")<<"-"<< pa.attribute("wantedproperties");
+           QStringList extra = pa.attribute("extraparams").split(' ', QString::SkipEmptyParts);
+            emit startFilterJob(filtertag, pa.attribute("filterparams"), pa.attribute("finalfilter"), pa.attribute("consumer"), pa.attribute("consumerparams"), extra);
+            kDebug()<<" - - -PROPS:\n"<<"filtertag"<<"-"<< pa.attribute("filterparams")<<"-"<< pa.attribute("consumer")<<"-"<< pa.attribute("consumerparams")<<"-"<< pa.attribute("extraparams");
             break;
         }
     }
index ba0c0f3811928cd505929727352670716e61e0fa..ae286af8ceac48eca096ac494b19cd4923b5e107 100644 (file)
@@ -102,7 +102,7 @@ signals:
     void seekTimeline(int);
     void showComments(bool);    
     /** @brief Start an MLT filter job on this clip. */
-    void startFilterJob(QString filterName, QString filterParams, QString finalFilterName, QString consumer, QString consumerParams, QString properties, QStringList extra);
+    void startFilterJob(QString filterName, QString filterParams, QString finalFilterName, QString consumer, QString consumerParams, QStringList extra);
     
 };
 
index 6945bf9c449e61751938b018cd0e9a068bd06033..8f2e08292f6d587fc62d1a4f529c38afd3a89c7b 100644 (file)
@@ -280,7 +280,7 @@ MainWindow::MainWindow(const QString &MltPath, const KUrl & Url, const QString &
     m_effectStack = new EffectStackView2(m_projectMonitor);
     m_effectStackDock->setWidget(m_effectStack);
     addDockWidget(Qt::TopDockWidgetArea, m_effectStackDock);
-    connect(m_effectStack, SIGNAL(startFilterJob(ItemInfo, const QString&,const QString&,const QString&,const QString&,const QString&,const QString&,const QString&,const QStringList&)), m_projectList, SLOT(slotStartFilterJob(ItemInfo, const QString&,const QString&,const QString&,const QString&,const QString&,const QString&,const QString&,const QStringList&)));
+    connect(m_effectStack, SIGNAL(startFilterJob(ItemInfo, const QString&,const QString&,const QString&,const QString&,const QString&,const QString&,const QStringList&)), m_projectList, SLOT(slotStartFilterJob(ItemInfo, const QString&,const QString&,const QString&,const QString&,const QString&,const QString&,const QStringList&)));
 
     m_transitionConfigDock = new QDockWidget(i18n("Transition"), this);
     m_transitionConfigDock->setObjectName("transition");
@@ -2523,7 +2523,7 @@ void MainWindow::connectDocument(TrackView *trackView, KdenliveDoc *doc)   //cha
             disconnect(m_activeTimeline->projectView(), SIGNAL(playMonitor()), m_projectMonitor, SLOT(slotPlay()));
             disconnect(m_activeTimeline->projectView(), SIGNAL(displayMessage(const QString&, MessageType)), m_messageLabel, SLOT(setMessage(const QString&, MessageType)));
             disconnect(m_activeTimeline->projectView(), SIGNAL(showClipFrame(DocClipBase *, QPoint, bool, const int)), m_clipMonitor, SLOT(slotSetClipProducer(DocClipBase *, QPoint, bool, const int)));
-            disconnect(m_projectList, SIGNAL(gotFilterJobResults(const QString &, int, int, const QString &, stringMap)), m_activeTimeline->projectView(), SLOT(slotGotFilterJobResults(const QString &, int, int, const QString &, stringMap)));
+            disconnect(m_projectList, SIGNAL(gotFilterJobResults(const QString &, int, int, const QString &, stringMap,QStringList)), m_activeTimeline->projectView(), SLOT(slotGotFilterJobResults(const QString &, int, int, const QString &, stringMap,QStringList)));
 
             disconnect(m_activeTimeline, SIGNAL(cursorMoved()), m_projectMonitor, SLOT(slotActivateMonitor()));
             disconnect(m_activeTimeline, SIGNAL(configTrack(int)), this, SLOT(slotConfigTrack(int)));
@@ -2542,6 +2542,7 @@ void MainWindow::connectDocument(TrackView *trackView, KdenliveDoc *doc)   //cha
             disconnect(m_activeTimeline, SIGNAL(zoneMoved(int, int)), this, SLOT(slotZoneMoved(int, int)));
             disconnect(m_projectList, SIGNAL(loadingIsOver()), m_activeTimeline->projectView(), SLOT(slotUpdateAllThumbs()));
             disconnect(m_projectList, SIGNAL(refreshClip(const QString &)), m_activeTimeline->projectView(), SLOT(slotRefreshThumbs(const QString &)));
+           disconnect(m_projectList, SIGNAL(addMarkers(const QString &, QList <CommentedTime>)), m_activeTimeline->projectView(), SLOT(slotAddClipMarker(const QString &, QList <CommentedTime>)));
             m_effectStack->clear();
         }
         //m_activeDocument->setRenderer(NULL);
@@ -2599,7 +2600,9 @@ void MainWindow::connectDocument(TrackView *trackView, KdenliveDoc *doc)   //cha
 
     connect(trackView->projectView(), SIGNAL(transitionItemSelected(Transition*, int, QPoint, bool)), m_projectMonitor, SLOT(slotSetSelectedClip(Transition*)));
 
-    connect(m_projectList, SIGNAL(gotFilterJobResults(const QString &, int, int, const QString &, stringMap)), trackView->projectView(), SLOT(slotGotFilterJobResults(const QString &, int, int, const QString &, stringMap)));
+    connect(m_projectList, SIGNAL(gotFilterJobResults(const QString &, int, int, const QString &, stringMap,QStringList)), trackView->projectView(), SLOT(slotGotFilterJobResults(const QString &, int, int, const QString &, stringMap,QStringList)));
+
+    connect(m_projectList, SIGNAL(addMarkers(const QString &, QList <CommentedTime>)), trackView->projectView(), SLOT(slotAddClipMarker(const QString &, QList <CommentedTime>)));
 
     // Effect stack signals
     connect(m_effectStack, SIGNAL(updateEffect(ClipItem*, int, QDomElement, QDomElement, int,bool)), trackView->projectView(), SLOT(slotUpdateClipEffect(ClipItem*, int, QDomElement, QDomElement, int,bool)));
@@ -2839,7 +2842,7 @@ void MainWindow::slotAddClipMarker()
     QPointer<MarkerDialog> d = new MarkerDialog(clip, marker,
                        m_activeDocument->timecode(), i18n("Add Marker"), this);
     if (d->exec() == QDialog::Accepted)
-        m_activeTimeline->projectView()->slotAddClipMarker(id, d->newMarker());
+        m_activeTimeline->projectView()->slotAddClipMarker(id, QList <CommentedTime>() << d->newMarker());
     delete d;
 }
 
@@ -2924,11 +2927,11 @@ void MainWindow::slotEditClipMarker()
     QPointer<MarkerDialog> d = new MarkerDialog(clip, oldMarker,
                       m_activeDocument->timecode(), i18n("Edit Marker"), this);
     if (d->exec() == QDialog::Accepted) {
-        m_activeTimeline->projectView()->slotAddClipMarker(id, d->newMarker());
+        m_activeTimeline->projectView()->slotAddClipMarker(id, QList <CommentedTime>() <<d->newMarker());
         if (d->newMarker().time() != pos) {
             // remove old marker
             oldMarker.setMarkerType(-1);
-            m_activeTimeline->projectView()->slotAddClipMarker(id, oldMarker);
+            m_activeTimeline->projectView()->slotAddClipMarker(id, QList <CommentedTime>() <<oldMarker);
         }
     }
     delete d;
@@ -2949,7 +2952,7 @@ void MainWindow::slotAddMarkerGuideQuickly()
         }
         //TODO: allow user to set default marker category
        CommentedTime marker(pos, m_activeDocument->timecode().getDisplayTimecode(pos, false), KdenliveSettings::default_marker_type());
-        m_activeTimeline->projectView()->slotAddClipMarker(clip->getId(), marker);
+        m_activeTimeline->projectView()->slotAddClipMarker(clip->getId(), QList <CommentedTime>() <<marker);
     } else {
         m_activeTimeline->projectView()->slotAddGuide(false);
     }
@@ -3312,7 +3315,7 @@ void MainWindow::slotShowClipProperties(DocClipBase *clip)
 
     // any type of clip but a title
     ClipProperties *dia = new ClipProperties(clip, m_activeDocument->timecode(), m_activeDocument->fps(), this);
-    connect(dia, SIGNAL(addMarker(const QString &, CommentedTime)), m_activeTimeline->projectView(), SLOT(slotAddClipMarker(const QString &, CommentedTime)));
+    connect(dia, SIGNAL(addMarkers(const QString &, QList <CommentedTime>)), m_activeTimeline->projectView(), SLOT(slotAddClipMarker(const QString &, QList <CommentedTime>)));
     connect(dia, SIGNAL(deleteAnalysis(QString,QString)), m_activeTimeline->projectView(), SLOT(slotAddClipExtraData(QString,QString)));
     connect(m_activeTimeline->projectView(), SIGNAL(updateClipMarkers(DocClipBase *)), dia, SLOT(slotFillMarkersList(DocClipBase *)));
     connect(m_activeTimeline->projectView(), SIGNAL(updateClipExtraData(DocClipBase *)), dia, SLOT(slotUpdateAnalysisData(DocClipBase *)));
@@ -3902,21 +3905,21 @@ void MainWindow::loadClipActions()
                        delete filter;
                        QAction *action=actionMenu->addAction(i18n("Stabilize (vstab)"));
                        action->setData("videostab");
-                       connect(action,SIGNAL(triggered()), this, SLOT(slotStabilize()));
+                       connect(action,SIGNAL(triggered()), this, SLOT(slotStartClipAction()));
                }
                filter = Mlt::Factory::filter(profile,(char*)"videostab2");
                if (filter) {
                        delete filter;
                        QAction *action=actionMenu->addAction(i18n("Stabilize (transcode)"));
                        action->setData("videostab2");
-                       connect(action,SIGNAL(triggered()), this, SLOT(slotStabilize()));
+                       connect(action,SIGNAL(triggered()), this, SLOT(slotStartClipAction()));
                }
                filter = Mlt::Factory::filter(profile,(char*)"motion_est");
                if (filter) {
                        delete filter;
                        QAction *action=actionMenu->addAction(i18n("Automatic scene split"));
                        action->setData("motion_est");
-                       connect(action,SIGNAL(triggered()), this, SLOT(slotStabilize()));
+                       connect(action,SIGNAL(triggered()), this, SLOT(slotStartClipAction()));
                }
        }
 
@@ -3952,7 +3955,7 @@ void MainWindow::loadTranscoders()
     }
 }
 
-void MainWindow::slotStabilize()
+void MainWindow::slotStartClipAction()
 {
     QString condition,filtername;
     QStringList ids;
index dff7177a77a4b7159de30a56387e54b6022d6278..c7c6c83b29bcfd5f46ff44a8081618228b971e90 100644 (file)
@@ -499,7 +499,7 @@ private slots:
     void slotShowTimeline(bool show);
     void slotMaximizeCurrent(bool show);
     void slotTranscode(KUrl::List urls = KUrl::List());
-    void slotStabilize();
+    void slotStartClipAction();
     void slotTranscodeClip();
     /** @brief Archive project: creates a copy of the project file with all clips in a new folder. */
     void slotArchiveProject();
index a702d4422dcbe1ea372363eb3aca6039882e1b2f..319a15243e418b792aa46b009d5490f4fa20e382 100644 (file)
@@ -44,6 +44,7 @@
 
 #include "ui_templateclip_ui.h"
 #include "ui_cutjobdialog_ui.h"
+#include "ui_scenecutdialog_ui.h"
 
 #include <KDebug>
 #include <KAction>
@@ -586,7 +587,7 @@ void ProjectList::editClipSelection(QList<QTreeWidgetItem *> list)
         kDebug() << "Result: " << p.key() << " = " << p.value();
     }*/
     if (clipList.isEmpty()) {
-        emit displayMessage(i18n("No available clip selected"), -2);        
+        emit displayMessage(i18n("No available clip selected"), -2);
     }
     else emit showClipProperties(clipList, commonproperties);
 }
@@ -2994,9 +2995,9 @@ void ProjectList::slotProcessJobs()
             MeltJob *jb = static_cast<MeltJob *> (job);
             jb->setProducer(currentClip->getProducer(), currentClip->fileURL());
            if (jb->isProjectFilter())
-             connect(job, SIGNAL(gotFilterJobResults(QString,int, int, QString,stringMap)), this, SLOT(slotGotFilterJobResults(QString,int, int, QString,stringMap)));
+             connect(job, SIGNAL(gotFilterJobResults(QString,int, int, QString,stringMap,QStringList)), this, SLOT(slotGotFilterJobResults(QString,int, int, QString,stringMap,QStringList)));
            else
-               connect(job, SIGNAL(gotFilterJobResults(QString,int, int, QString,stringMap)), this, SIGNAL(gotFilterJobResults(QString,int, int, QString,stringMap)));
+               connect(job, SIGNAL(gotFilterJobResults(QString,int, int, QString,stringMap,QStringList)), this, SIGNAL(gotFilterJobResults(QString,int, int, QString,stringMap,QStringList)));
         }
         job->startJob();
         if (job->jobStatus == JOBDONE) {
@@ -3421,13 +3422,13 @@ void ProjectList::discardJobs(const QString &id, JOBTYPE type) {
     }
 }
 
-void ProjectList::slotStartFilterJob(ItemInfo info, const QString&id, const QString&filterName, const QString&filterParams, const QString&finalFilterName, const QString&consumer, const QString&consumerParams, const QString&properties, const QStringList &extraParams)
+void ProjectList::slotStartFilterJob(ItemInfo info, const QString&id, const QString&filterName, const QString&filterParams, const QString&finalFilterName, const QString&consumer, const QString&consumerParams, const QStringList &extraParams)
 {
     ProjectItem *item = getItemById(id);
     if (!item) return;
     QStringList jobParams;
     jobParams << QString::number(info.cropStart.frames(m_fps)) << QString::number((info.cropStart + info.cropDuration).frames(m_fps));
-    jobParams << QString() << filterName << filterParams << consumer << consumerParams << properties << QString::number(info.startPos.frames(m_fps)) << QString::number(info.track) << finalFilterName;
+    jobParams << QString() << filterName << filterParams << consumer << consumerParams << QString::number(info.startPos.frames(m_fps)) << QString::number(info.track) << finalFilterName;
     MeltJob *job = new MeltJob(item->clipType(), id, jobParams, extraParams);
     if (job->isExclusive() && hasPendingJob(item, job->jobType)) {
         delete job;
@@ -3455,6 +3456,20 @@ void ProjectList::startClipFilterJob(const QString &filterName, const QString &c
         destination = item->clipUrl().directory();
     }
     if (filterName == "motion_est") {
+       // Show config dialog
+       QPointer<QDialog> d = new QDialog(this);
+       Ui::SceneCutDialog_UI ui;
+       ui.setupUi(d);
+       // Set  up categories
+       for (int i = 0; i < 5; ++i) {
+           ui.marker_type->insertItem(i, i18n("Category %1", i));
+           ui.marker_type->setItemData(i, CommentedTime::markerColor(i), Qt::DecorationRole);
+       }
+       ui.marker_type->setCurrentIndex(KdenliveSettings::default_marker_type());
+       if (d->exec() != QDialog::Accepted) {
+           delete d;
+           return;
+       }
        // Autosplit filter
        QStringList jobParams;
        // Producer params
@@ -3463,10 +3478,18 @@ void ProjectList::startClipFilterJob(const QString &filterName, const QString &c
        jobParams << filterName << "bounding=\"25%x25%:25%x25\" shot_change_list=0";
        // Consumer
        jobParams << "null" << "all=1 terminate_on_pause=1 real_time=-1";
-       // Keys
-       jobParams << "shot_change_list";
        QStringList extraParams;
+       extraParams << "key:shot_change_list";
        extraParams << "projecttreefilter";
+       if (ui.add_markers->isChecked()) {
+           // We want to create markers
+           extraParams << QString("addmarkers:%1").arg(ui.marker_type->currentIndex());
+       }
+       if (ui.cut_scenes->isChecked()) {
+           // We want to cut scenes
+           extraParams << "cutscenes";
+       }
+       
        processClipJob(ids, QString(), false, jobParams, i18n("Auto split"), extraParams);
     }
     else {
@@ -3574,15 +3597,37 @@ void ProjectList::slotClosePopup()
     m_errorLog.clear();
 }
 
-void ProjectList::slotGotFilterJobResults(QString id, int , int , QString filter, stringMap results)
+void ProjectList::slotGotFilterJobResults(QString id, int , int , QString filter, stringMap results, QStringList extra)
 {
-    if (filter == "motion_est") {
-       // Autosplit filter, add sub zones
-       QStringList cuts = results.value("shot_change_list").split(';', QString::SkipEmptyParts);
+    ProjectItem *clip = getItemById(id);
+    if (!clip) return;
+    // Check for return value
+    QString returnKey;
+    int markersType = -1;
+    for (int i = 0; i < extra.count(); i++) {
+       if (extra.at(i).startsWith("key:"))
+           returnKey = extra.at(i).section(':', 1);
+       if (extra.at(i).startsWith("addmarkers:")) {
+           markersType = extra.at(i).section(':', 1).toInt();
+       }
+    }
+    if (returnKey.isEmpty()) {
+       emit displayMessage(i18n("No data returned from clip analysis"), 2);
+       return;
+    }
+    QStringList returnData = results.value(returnKey).split(';', QString::SkipEmptyParts);
+    if (returnData.isEmpty()) {
+       emit displayMessage(i18n("No data returned from clip analysis"), 2);
+       return;
+    }
+    bool dataProcessed = false;
+    if (extra.contains("cutscenes")) {
+       // Check if we want to cut scenes from returned data
+       dataProcessed = true;
        int cutPos = 0;
        QUndoCommand *command = new QUndoCommand();
        command->setText(i18n("Auto Split Clip"));
-       foreach (QString pos, cuts) {
+       foreach (QString pos, returnData) {
            if (!pos.contains("=")) continue;
            int newPos = pos.section("=", 0, 0).toInt();
            // Don't use scenes shorter than 1 second
@@ -3594,15 +3639,32 @@ void ProjectList::slotGotFilterJobResults(QString id, int , int , QString filter
            delete command;
        else m_commandStack->push(command);
     }
-    else if (filter.startsWith("autotrack_rectangle")) {
-       QString cuts = results.value("motion_vector_list");
-       ProjectItem *clip = getItemById(id);
-       if (clip) {
-           clip->referencedClip()->setAnalysisData(i18n("Motion vectors"), cuts);
-           emit updateAnalysisData(clip->referencedClip());
+    if (markersType >= 0) {
+       // Add markers from returned data
+       dataProcessed = true;
+       int cutPos = 0;
+       QUndoCommand *command = new QUndoCommand();
+       command->setText(i18n("Add Markers"));
+       QList <CommentedTime> markersList;
+       int index = 1;
+       foreach (QString pos, returnData) {
+           if (!pos.contains("=")) continue;
+           int newPos = pos.section("=", 0, 0).toInt();
+           // Don't use scenes shorter than 1 second
+           if (newPos - cutPos < 24) continue;
+           CommentedTime m(GenTime(newPos, m_fps), QString::number(index), markersType);
+           markersList << m;
+           index++;
+           cutPos = newPos;
        }
+       emit addMarkers(id, markersList);
+    }
+    if (!dataProcessed || extra.contains("storedata")) {
+       // Store returned data as clip extra data
+       
+       clip->referencedClip()->setAnalysisData(i18n("Motion vectors"), results.value(returnKey));
+       emit updateAnalysisData(clip->referencedClip());
     }
-    
 }
 
 #include "projectlist.moc"
index 9f3d91e8f1b31a20471f44422440f25d365d3862..7019595c7fec367d52a47f15a9e08c46781ca689 100644 (file)
@@ -330,7 +330,7 @@ public slots:
     /** @brief Start transcoding selected clips. */
     void slotTranscodeClipJob(const QString &condition, QString params, QString desc);
     /** @brief Start an MLT process job. */
-    void slotStartFilterJob(ItemInfo, const QString&,const QString&,const QString&,const QString&,const QString&,const QString&,const QString&,const QStringList&);
+    void slotStartFilterJob(ItemInfo, const QString&,const QString&,const QString&,const QString&,const QString&,const QString&,const QStringList&);
     
 
 private:
@@ -491,7 +491,7 @@ private slots:
     /** @brief close warning info passive popup. */
     void slotClosePopup();
     /** @brief process clip job result. */
-    void slotGotFilterJobResults(QString ,int , int, QString, stringMap);
+    void slotGotFilterJobResults(QString ,int , int, QString, stringMap, QStringList);
 
 signals:
     void clipSelected(DocClipBase *, QPoint zone = QPoint(), bool forceUpdate = false);
@@ -524,9 +524,10 @@ signals:
     void gotProxy(const QString);
     void checkJobProcess();
     /** @brief A Filter Job produced results, send them back to the clip. */
-    void gotFilterJobResults(const QString &id, int startPos, int track, const QString &filterName, stringMap params);
+    void gotFilterJobResults(const QString &id, int startPos, int track, const QString &filterName, stringMap params, QStringList extre);
     void pauseMonitor();
     void updateAnalysisData(DocClipBase *);
+    void addMarkers(const QString &, QList <CommentedTime>);
 };
 
 #endif
index 1e24caffc1582b4647c7640b61bf890a7f82895f..ed170fdbe132fe3a4d71a41f0d9e7d946a1cbd39 100644 (file)
@@ -76,8 +76,6 @@ void MeltJob::startJob()
     QString consumerParams = m_params.takeFirst();
     
     // optional params
-    QString properties;
-    if (!m_params.isEmpty()) properties = m_params.takeFirst();
     int startPos = -1;
     if (!m_params.isEmpty()) startPos = m_params.takeFirst().toInt();
     int track = -1;
@@ -86,6 +84,15 @@ void MeltJob::startJob()
     if (!m_params.isEmpty()) finalFilter = m_params.takeFirst();
     else finalFilter = filter;
 
+    // Check if we want to return analysis data
+    QString properties;
+    for (int i = 0; i < m_extra.count(); i++) {
+       if (m_extra.at(i).startsWith("key:")) {
+           properties = m_extra.at(i).section(':', 1);
+           break;
+       }
+    }
+
 
     if (out != -1 && out <= in) {
         m_errorMessage.append(i18n("Clip zone undefined (%1 - %2).", in, out));
@@ -173,7 +180,7 @@ void MeltJob::startJob()
         QString value = mltFilter.get(key.toUtf8().constData());
         jobResults.insert(key, value);
     }
-    if (!jobResults.isEmpty()) emit gotFilterJobResults(m_clipId, startPos, track, finalFilter, jobResults);
+    if (!jobResults.isEmpty() && jobStatus != JOBABORTED) emit gotFilterJobResults(m_clipId, startPos, track, finalFilter, jobResults, m_extra);
     setStatus(JOBDONE);
     delete m_consumer;
     delete prod;
index 63a34d24b4e5c21b693d037d58c2d0b5f81c7803..d37365038635d7d475d8d9cd3f5c3db023cce7d7 100644 (file)
@@ -65,7 +65,7 @@ private:
     QStringList m_extra;
 
 signals:
-    void gotFilterJobResults(const QString &id, int startPos, int track, const QString &filterName, stringMap params);
+    void gotFilterJobResults(const QString &id, int startPos, int track, const QString &filterName, stringMap params, QStringList extraParam);
 };
 
 #endif
diff --git a/src/widgets/scenecutdialog_ui.ui b/src/widgets/scenecutdialog_ui.ui
new file mode 100644 (file)
index 0000000..462ff59
--- /dev/null
@@ -0,0 +1,97 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<ui version="4.0">
+ <class>SceneCutDialog_UI</class>
+ <widget class="QDialog" name="SceneCutDialog_UI">
+  <property name="geometry">
+   <rect>
+    <x>0</x>
+    <y>0</y>
+    <width>352</width>
+    <height>90</height>
+   </rect>
+  </property>
+  <property name="windowTitle">
+   <string>Scene Cut</string>
+  </property>
+  <layout class="QGridLayout" name="gridLayout">
+   <item row="3" column="0" colspan="3">
+    <widget class="QDialogButtonBox" name="buttonBox">
+     <property name="orientation">
+      <enum>Qt::Horizontal</enum>
+     </property>
+     <property name="standardButtons">
+      <set>QDialogButtonBox::Cancel|QDialogButtonBox::Ok</set>
+     </property>
+    </widget>
+   </item>
+   <item row="2" column="1">
+    <spacer name="verticalSpacer">
+     <property name="orientation">
+      <enum>Qt::Vertical</enum>
+     </property>
+     <property name="sizeHint" stdset="0">
+      <size>
+       <width>218</width>
+       <height>2</height>
+      </size>
+     </property>
+    </spacer>
+   </item>
+   <item row="0" column="0">
+    <widget class="QCheckBox" name="add_markers">
+     <property name="text">
+      <string>Add clip markers</string>
+     </property>
+     <property name="checked">
+      <bool>true</bool>
+     </property>
+    </widget>
+   </item>
+   <item row="0" column="1" colspan="2">
+    <widget class="QComboBox" name="marker_type"/>
+   </item>
+   <item row="1" column="0" colspan="3">
+    <widget class="QCheckBox" name="cut_scenes">
+     <property name="text">
+      <string>Cut scenes</string>
+     </property>
+    </widget>
+   </item>
+  </layout>
+ </widget>
+ <resources/>
+ <connections>
+  <connection>
+   <sender>buttonBox</sender>
+   <signal>accepted()</signal>
+   <receiver>SceneCutDialog_UI</receiver>
+   <slot>accept()</slot>
+   <hints>
+    <hint type="sourcelabel">
+     <x>248</x>
+     <y>254</y>
+    </hint>
+    <hint type="destinationlabel">
+     <x>157</x>
+     <y>274</y>
+    </hint>
+   </hints>
+  </connection>
+  <connection>
+   <sender>buttonBox</sender>
+   <signal>rejected()</signal>
+   <receiver>SceneCutDialog_UI</receiver>
+   <slot>reject()</slot>
+   <hints>
+    <hint type="sourcelabel">
+     <x>316</x>
+     <y>260</y>
+    </hint>
+    <hint type="destinationlabel">
+     <x>286</x>
+     <y>274</y>
+    </hint>
+   </hints>
+  </connection>
+ </connections>
+</ui>