]> git.sesse.net Git - kdenlive/blob - src/mainwindow.cpp
ugly hack to read all parameters to set.
[kdenlive] / src / mainwindow.cpp
1 /***************************************************************************
2  *   Copyright (C) 2007 by Jean-Baptiste Mardelle (jb@kdenlive.org)        *
3  *                                                                         *
4  *   This program is free software; you can redistribute it and/or modify  *
5  *   it under the terms of the GNU General Public License as published by  *
6  *   the Free Software Foundation; either version 2 of the License, or     *
7  *   (at your option) any later version.                                   *
8  *                                                                         *
9  *   This program is distributed in the hope that it will be useful,       *
10  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        *
11  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *
12  *   GNU General Public License for more details.                          *
13  *                                                                         *
14  *   You should have received a copy of the GNU General Public License     *
15  *   along with this program; if not, write to the                         *
16  *   Free Software Foundation, Inc.,                                       *
17  *   51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA          *
18  ***************************************************************************/
19
20
21 #include "mainwindow.h"
22 #include "mainwindowadaptor.h"
23 #include "kdenlivesettings.h"
24 #include "kdenlivesettingsdialog.h"
25 #include "initeffects.h"
26 #include "profilesdialog.h"
27 #include "projectsettings.h"
28 #include "events.h"
29 #include "clipmanager.h"
30 #include "projectlist.h"
31 #include "monitor.h"
32 #include "recmonitor.h"
33 #include "monitormanager.h"
34 #include "kdenlivedoc.h"
35 #include "trackview.h"
36 #include "customtrackview.h"
37 #include "effectslistview.h"
38 #include "effectstackview.h"
39 #include "transitionsettings.h"
40 #include "renderwidget.h"
41 #include "renderer.h"
42 #include "audiosignal.h"
43 #ifdef USE_JOGSHUTTLE
44 #include "jogshuttle.h"
45 #include "jogaction.h"
46 #include "jogshuttleconfig.h"
47 #endif
48 #include "clipproperties.h"
49 #include "wizard.h"
50 #include "commands/editclipcommand.h"
51 #include "titlewidget.h"
52 #include "markerdialog.h"
53 #include "clipitem.h"
54 #include "interfaces.h"
55 #include "config-kdenlive.h"
56 #include "cliptranscode.h"
57 #include "clipstabilize.h"
58 #include "ui_templateclip_ui.h"
59 #include "colorscopes/vectorscope.h"
60 #include "colorscopes/waveform.h"
61 #include "colorscopes/rgbparade.h"
62 #include "colorscopes/histogram.h"
63 #include "audioscopes/audiospectrum.h"
64 #include "audioscopes/spectrogram.h"
65 #include "archivewidget.h"
66 #include "databackup/backupwidget.h"
67
68
69 #include <KApplication>
70 #include <KAction>
71 #include <KLocale>
72 #include <KGlobal>
73 #include <KActionCollection>
74 #include <KActionCategory>
75 #include <KStandardAction>
76 #include <KShortcutsDialog>
77 #include <KFileDialog>
78 #include <KMessageBox>
79 #include <KDebug>
80 #include <KIO/NetAccess>
81 #include <KSaveFile>
82 #include <KRuler>
83 #include <KConfigDialog>
84 #include <KXMLGUIFactory>
85 #include <KStatusBar>
86 #include <kstandarddirs.h>
87 #include <KUrlRequesterDialog>
88 #include <KTemporaryFile>
89 #include <KProcess>
90 #include <KActionMenu>
91 #include <KMenu>
92 #include <ktogglefullscreenaction.h>
93 #include <KFileItem>
94 #include <KNotification>
95 #include <KNotifyConfigWidget>
96 #if KDE_IS_VERSION(4,3,80)
97 #include <knewstuff3/downloaddialog.h>
98 #include <knewstuff3/knewstuffaction.h>
99 #else
100 #include <knewstuff2/engine.h>
101 #include <knewstuff2/ui/knewstuffaction.h>
102 #define KNS3 KNS
103 #endif
104 #include <KToolBar>
105 #include <KColorScheme>
106 #include <KProgressDialog>
107
108 #include <QTextStream>
109 #include <QTimer>
110 #include <QAction>
111 #include <QKeyEvent>
112 #include <QInputDialog>
113 #include <QDesktopWidget>
114 #include <QBitmap>
115
116 #include <stdlib.h>
117
118 // Uncomment for deeper debugging
119 //#define DEBUG_MAINW
120
121 #ifdef DEBUG_MAINW
122 #include <QDebug>
123 #endif
124
125 static const char version[] = VERSION;
126
127 static const int ID_TIMELINE_POS = 0;
128 namespace Mlt
129 {
130 class Producer;
131 };
132
133 Q_DECLARE_METATYPE(QVector<int16_t>)
134
135
136 EffectsList MainWindow::videoEffects;
137 EffectsList MainWindow::audioEffects;
138 EffectsList MainWindow::customEffects;
139 EffectsList MainWindow::transitions;
140
141 MainWindow::MainWindow(const QString &MltPath, const KUrl & Url, const QString & clipsToLoad, QWidget *parent) :
142     KXmlGuiWindow(parent),
143     m_activeDocument(NULL),
144     m_activeTimeline(NULL),
145     m_recMonitor(NULL),
146     m_renderWidget(NULL),
147 #ifdef USE_JOGSHUTTLE
148     m_jogProcess(NULL),
149     m_jogShuttle(NULL),
150 #endif
151     m_findActivated(false),
152     m_stopmotion(NULL)
153 {   
154     qRegisterMetaType<QVector<int16_t> > ();
155     qRegisterMetaType<stringMap> ("stringMap");
156     qRegisterMetaType<audioByteArray> ("audioByteArray");
157
158     // Init locale
159     QLocale systemLocale = QLocale();
160     systemLocale.setNumberOptions(QLocale::OmitGroupSeparator);
161     QLocale::setDefault(systemLocale);
162
163     // Create DBus interface
164     new MainWindowAdaptor(this);
165     QDBusConnection dbus = QDBusConnection::sessionBus();
166     dbus.registerObject("/MainWindow", this);
167
168     if (!KdenliveSettings::colortheme().isEmpty()) slotChangePalette(NULL, KdenliveSettings::colortheme());
169     setFont(KGlobalSettings::toolBarFont());
170     parseProfiles(MltPath);
171     KdenliveSettings::setCurrent_profile(KdenliveSettings::default_profile());
172     m_commandStack = new QUndoGroup;
173     setDockNestingEnabled(true);
174     m_timelineArea = new KTabWidget(this);
175     m_timelineArea->setTabReorderingEnabled(true);
176     m_timelineArea->setTabBarHidden(true);
177     m_timelineArea->setSizePolicy(QSizePolicy::Maximum, QSizePolicy::Preferred);
178     m_timelineArea->setMinimumHeight(200);
179
180     QToolButton *closeTabButton = new QToolButton;
181     connect(closeTabButton, SIGNAL(clicked()), this, SLOT(closeCurrentDocument()));
182     closeTabButton->setIcon(KIcon("tab-close"));
183     closeTabButton->adjustSize();
184     closeTabButton->setToolTip(i18n("Close the current tab"));
185     m_timelineArea->setCornerWidget(closeTabButton);
186     //connect(m_timelineArea, SIGNAL(currentChanged(int)), this, SLOT(activateDocument()));
187
188     connect(&m_findTimer, SIGNAL(timeout()), this, SLOT(findTimeout()));
189     m_findTimer.setSingleShot(true);
190
191     // FIXME: the next call returns a newly allocated object, which leaks
192     initEffects::parseEffectFiles();
193     //initEffects::parseCustomEffectsFile();
194
195     m_monitorManager = new MonitorManager();
196
197     m_shortcutRemoveFocus = new QShortcut(QKeySequence("Esc"), this);
198     connect(m_shortcutRemoveFocus, SIGNAL(activated()), this, SLOT(slotRemoveFocus()));
199
200
201     /// Add Widgets ///
202
203     m_projectListDock = new QDockWidget(i18n("Project Tree"), this);
204     m_projectListDock->setObjectName("project_tree");
205     m_projectList = new ProjectList();
206     m_projectListDock->setWidget(m_projectList);
207     addDockWidget(Qt::TopDockWidgetArea, m_projectListDock);
208
209     m_clipMonitorDock = new QDockWidget(i18n("Clip Monitor"), this);
210     m_clipMonitorDock->setObjectName("clip_monitor");
211     m_clipMonitor = new Monitor("clip", m_monitorManager, QString(), m_timelineArea);
212     m_clipMonitorDock->setWidget(m_clipMonitor);
213     
214     // Connect the project list
215     connect(m_projectList, SIGNAL(clipSelected(DocClipBase *, QPoint)), m_clipMonitor, SLOT(slotSetClipProducer(DocClipBase *, QPoint)));
216     connect(m_projectList, SIGNAL(raiseClipMonitor()), m_clipMonitor, SLOT(activateMonitor()));
217     connect(m_projectList, SIGNAL(loadingIsOver()), this, SLOT(slotElapsedTime()));
218     connect(m_projectList, SIGNAL(displayMessage(const QString&, int)), this, SLOT(slotGotProgressInfo(const QString&, int)));
219     connect(m_projectList, SIGNAL(updateRenderStatus()), this, SLOT(slotCheckRenderStatus()));
220     connect(m_projectList, SIGNAL(clipNeedsReload(const QString&)),this, SLOT(slotUpdateClip(const QString &)));
221     connect(m_projectList, SIGNAL(updateProfile(const QString &)), this, SLOT(slotUpdateProjectProfile(const QString &)));
222     connect(m_projectList, SIGNAL(refreshClip(const QString &, bool)), m_monitorManager, SLOT(slotRefreshCurrentMonitor()));
223     connect(m_projectList, SIGNAL(findInTimeline(const QString&)), this, SLOT(slotClipInTimeline(const QString&)));
224     connect(m_clipMonitor, SIGNAL(zoneUpdated(QPoint)), m_projectList, SLOT(slotUpdateClipCut(QPoint)));
225
226     m_projectMonitorDock = new QDockWidget(i18n("Project Monitor"), this);
227     m_projectMonitorDock->setObjectName("project_monitor");
228     m_projectMonitor = new Monitor("project", m_monitorManager, QString());
229     m_projectMonitorDock->setWidget(m_projectMonitor);
230
231 #ifndef Q_WS_MAC
232     m_recMonitorDock = new QDockWidget(i18n("Record Monitor"), this);
233     m_recMonitorDock->setObjectName("record_monitor");
234     m_recMonitor = new RecMonitor("record", m_monitorManager);
235     m_recMonitorDock->setWidget(m_recMonitor);
236     connect(m_recMonitor, SIGNAL(addProjectClip(KUrl)), this, SLOT(slotAddProjectClip(KUrl)));
237     connect(m_recMonitor, SIGNAL(addProjectClipList(KUrl::List)), this, SLOT(slotAddProjectClipList(KUrl::List)));
238     connect(m_recMonitor, SIGNAL(showConfigDialog(int, int)), this, SLOT(slotPreferences(int, int)));
239 #endif /* ! Q_WS_MAC */
240     m_monitorManager->initMonitors(m_clipMonitor, m_projectMonitor, m_recMonitor);
241
242     m_notesDock = new QDockWidget(i18n("Project Notes"), this);
243     m_notesDock->setObjectName("notes_widget");
244     m_notesWidget = new NotesWidget();
245     connect(m_notesWidget, SIGNAL(insertNotesTimecode()), this, SLOT(slotInsertNotesTimecode()));
246     connect(m_notesWidget, SIGNAL(seekProject(int)), m_projectMonitor->render, SLOT(seekToFrame(int)));
247     
248     m_notesWidget->setTabChangesFocus(true);
249 #if KDE_IS_VERSION(4,4,0)
250     m_notesWidget->setClickMessage(i18n("Enter your project notes here ..."));
251 #endif
252     m_notesDock->setWidget(m_notesWidget);
253     addDockWidget(Qt::TopDockWidgetArea, m_notesDock);
254
255     m_effectStackDock = new QDockWidget(i18n("Effect Stack"), this);
256     m_effectStackDock->setObjectName("effect_stack");
257     m_effectStack = new EffectStackView(m_projectMonitor);
258     m_effectStackDock->setWidget(m_effectStack);
259     addDockWidget(Qt::TopDockWidgetArea, m_effectStackDock);
260
261     m_transitionConfigDock = new QDockWidget(i18n("Transition"), this);
262     m_transitionConfigDock->setObjectName("transition");
263     m_transitionConfig = new TransitionSettings(m_projectMonitor);
264     m_transitionConfigDock->setWidget(m_transitionConfig);
265     addDockWidget(Qt::TopDockWidgetArea, m_transitionConfigDock);
266
267     m_effectListDock = new QDockWidget(i18n("Effect List"), this);
268     m_effectListDock->setObjectName("effect_list");
269     m_effectList = new EffectsListView();
270     m_effectListDock->setWidget(m_effectList);
271     addDockWidget(Qt::TopDockWidgetArea, m_effectListDock);
272
273     m_vectorscope = new Vectorscope(m_monitorManager);
274     m_vectorscopeDock = new QDockWidget(i18n("Vectorscope"), this);
275     m_vectorscopeDock->setObjectName(m_vectorscope->widgetName());
276     m_vectorscopeDock->setWidget(m_vectorscope);
277     addDockWidget(Qt::TopDockWidgetArea, m_vectorscopeDock);
278     connect(m_vectorscopeDock, SIGNAL(visibilityChanged(bool)), m_vectorscope, SLOT(forceUpdate(bool)));
279     connect(m_vectorscopeDock, SIGNAL(visibilityChanged(bool)), this, SLOT(slotUpdateGfxScopeFrameRequest()));
280     connect(m_vectorscope, SIGNAL(requestAutoRefresh(bool)), this, SLOT(slotUpdateGfxScopeFrameRequest()));
281     m_gfxScopesList.append(m_vectorscopeDock);
282
283     m_waveform = new Waveform(m_monitorManager);
284     m_waveformDock = new QDockWidget(i18n("Waveform"), this);
285     m_waveformDock->setObjectName(m_waveform->widgetName());
286     m_waveformDock->setWidget(m_waveform);
287     addDockWidget(Qt::TopDockWidgetArea, m_waveformDock);
288     connect(m_waveformDock, SIGNAL(visibilityChanged(bool)), m_waveform, SLOT(forceUpdate(bool)));
289     connect(m_waveformDock, SIGNAL(visibilityChanged(bool)), this, SLOT(slotUpdateGfxScopeFrameRequest()));
290     connect(m_waveform, SIGNAL(requestAutoRefresh(bool)), this, SLOT(slotUpdateGfxScopeFrameRequest()));
291     m_gfxScopesList.append(m_waveformDock);
292
293     m_RGBParade = new RGBParade(m_monitorManager);
294     m_RGBParadeDock = new QDockWidget(i18n("RGB Parade"), this);
295     m_RGBParadeDock->setObjectName(m_RGBParade->widgetName());
296     m_RGBParadeDock->setWidget(m_RGBParade);
297     addDockWidget(Qt::TopDockWidgetArea, m_RGBParadeDock);
298     connect(m_RGBParadeDock, SIGNAL(visibilityChanged(bool)), m_RGBParade, SLOT(forceUpdate(bool)));
299     connect(m_RGBParadeDock, SIGNAL(visibilityChanged(bool)), this, SLOT(slotUpdateGfxScopeFrameRequest()));
300     connect(m_RGBParade, SIGNAL(requestAutoRefresh(bool)), this, SLOT(slotUpdateGfxScopeFrameRequest()));
301     m_gfxScopesList.append(m_RGBParadeDock);
302
303     m_histogram = new Histogram(m_monitorManager);
304     m_histogramDock = new QDockWidget(i18n("Histogram"), this);
305     m_histogramDock->setObjectName(m_histogram->widgetName());
306     m_histogramDock->setWidget(m_histogram);
307     addDockWidget(Qt::TopDockWidgetArea, m_histogramDock);
308     connect(m_histogramDock, SIGNAL(visibilityChanged(bool)), m_histogram, SLOT(forceUpdate(bool)));
309     connect(m_histogramDock, SIGNAL(visibilityChanged(bool)), this, SLOT(slotUpdateGfxScopeFrameRequest()));
310     connect(m_histogram, SIGNAL(requestAutoRefresh(bool)), this, SLOT(slotUpdateGfxScopeFrameRequest()));
311     m_gfxScopesList.append(m_histogramDock);
312
313
314     m_audiosignal = new AudioSignal;
315     m_audiosignalDock = new QDockWidget(i18n("Audio Signal"), this);
316     m_audiosignalDock->setObjectName("audiosignal");
317     m_audiosignalDock->setWidget(m_audiosignal);
318     addDockWidget(Qt::TopDockWidgetArea, m_audiosignalDock);
319     connect(m_audiosignalDock, SIGNAL(visibilityChanged(bool)), this, SLOT(slotUpdateAudioScopeFrameRequest()));
320     connect(m_audiosignal, SIGNAL(updateAudioMonitoring()), this, SLOT(slotUpdateAudioScopeFrameRequest()));
321
322     m_audioSpectrum = new AudioSpectrum();
323     m_audioSpectrumDock = new QDockWidget(i18n("AudioSpectrum"), this);
324     m_audioSpectrumDock->setObjectName(m_audioSpectrum->widgetName());
325     m_audioSpectrumDock->setWidget(m_audioSpectrum);
326     addDockWidget(Qt::TopDockWidgetArea, m_audioSpectrumDock);
327     m_audioScopesList.append(m_audioSpectrum);
328     connect(m_audioSpectrumDock, SIGNAL(visibilityChanged(bool)), this, SLOT(slotUpdateAudioScopeFrameRequest()));
329     connect(m_audioSpectrum, SIGNAL(requestAutoRefresh(bool)), this, SLOT(slotUpdateAudioScopeFrameRequest()));
330
331     m_spectrogram = new Spectrogram();
332     m_spectrogramDock = new QDockWidget(i18n("Spectrogram"), this);
333     m_spectrogramDock->setObjectName(m_spectrogram->widgetName());
334     m_spectrogramDock->setWidget(m_spectrogram);
335     addDockWidget(Qt::TopDockWidgetArea, m_spectrogramDock);
336     m_audioScopesList.append(m_spectrogram);
337     connect(m_audioSpectrumDock, SIGNAL(visibilityChanged(bool)), this, SLOT(slotUpdateAudioScopeFrameRequest()));
338     connect(m_audioSpectrum, SIGNAL(requestAutoRefresh(bool)), this, SLOT(slotUpdateAudioScopeFrameRequest()));
339
340     // Connect the audio signal to the audio scope slots
341     bool b = true;
342     if (m_projectMonitor) {
343         qDebug() << "project monitor connected";
344         b &= connect(m_projectMonitor->render, SIGNAL(audioSamplesSignal(QVector<int16_t>, int, int, int)),
345                      m_audioSpectrum, SLOT(slotReceiveAudio(QVector<int16_t>, int, int, int)));
346         b &= connect(m_projectMonitor->render, SIGNAL(audioSamplesSignal(const QVector<int16_t>&, const int&, const int&, const int&)),
347                      m_audiosignal, SLOT(slotReceiveAudio(const QVector<int16_t>&, const int&, const int&, const int&)));
348         b &= connect(m_projectMonitor->render, SIGNAL(audioSamplesSignal(QVector<int16_t>,int,int,int)),
349                      m_spectrogram, SLOT(slotReceiveAudio(QVector<int16_t>,int,int,int)));
350     }
351     if (m_clipMonitor) {
352         qDebug() << "clip monitor connected";
353         b &= connect(m_clipMonitor->render, SIGNAL(audioSamplesSignal(QVector<int16_t>, int, int, int)),
354                      m_audioSpectrum, SLOT(slotReceiveAudio(QVector<int16_t>, int, int, int)));
355         b &= connect(m_clipMonitor->render, SIGNAL(audioSamplesSignal(const QVector<int16_t>&, int, int, int)),
356                      m_audiosignal, SLOT(slotReceiveAudio(const QVector<int16_t>&, int, int, int)));
357         b &= connect(m_clipMonitor->render, SIGNAL(audioSamplesSignal(QVector<int16_t>,int,int,int)),
358                      m_spectrogram, SLOT(slotReceiveAudio(QVector<int16_t>,int,int,int)));
359     }
360     // Ensure connections were set up correctly
361     Q_ASSERT(b);
362
363
364
365     // Add monitors here to keep them at the right of the window
366     addDockWidget(Qt::TopDockWidgetArea, m_clipMonitorDock);
367     addDockWidget(Qt::TopDockWidgetArea, m_projectMonitorDock);
368 #ifndef Q_WS_MAC
369     addDockWidget(Qt::TopDockWidgetArea, m_recMonitorDock);
370 #endif
371
372     m_undoViewDock = new QDockWidget(i18n("Undo History"), this);
373     m_undoViewDock->setObjectName("undo_history");
374     m_undoView = new QUndoView();
375     m_undoView->setCleanIcon(KIcon("edit-clear"));
376     m_undoView->setEmptyLabel(i18n("Clean"));
377     m_undoViewDock->setWidget(m_undoView);
378     m_undoView->setGroup(m_commandStack);
379     addDockWidget(Qt::TopDockWidgetArea, m_undoViewDock);
380
381
382     setupActions();
383     connect(m_commandStack, SIGNAL(cleanChanged(bool)), m_saveAction, SLOT(setDisabled(bool)));
384
385
386     // Close non-general docks for the initial layout
387     // only show important ones
388     m_histogramDock->close();
389     m_RGBParadeDock->close();
390     m_waveformDock->close();
391     m_vectorscopeDock->close();
392
393     m_audioSpectrumDock->close();
394     m_spectrogramDock->close();
395     m_audiosignalDock->close();
396
397     m_undoViewDock->close();
398
399
400
401     /// Tabify Widgets ///
402     tabifyDockWidget(m_effectListDock, m_effectStackDock);
403     tabifyDockWidget(m_effectListDock, m_transitionConfigDock);
404     tabifyDockWidget(m_projectListDock, m_notesDock);
405
406     tabifyDockWidget(m_clipMonitorDock, m_projectMonitorDock);
407 #ifndef Q_WS_MAC
408     tabifyDockWidget(m_clipMonitorDock, m_recMonitorDock);
409 #endif
410
411
412
413
414     setCentralWidget(m_timelineArea);
415
416
417     m_fileOpenRecent = KStandardAction::openRecent(this, SLOT(openFile(const KUrl &)), actionCollection());
418     readOptions();
419     m_fileRevert = KStandardAction::revert(this, SLOT(slotRevert()), actionCollection());
420     m_fileRevert->setEnabled(false);
421
422     // Prepare layout actions
423     KActionCategory *layoutActions = new KActionCategory(i18n("Layouts"), actionCollection());
424     m_loadLayout = new KSelectAction(i18n("Load Layout"), actionCollection());
425     for (int i = 1; i < 5; i++) {
426         KAction *load = new KAction(KIcon(), i18n("Layout %1").arg(i), this);
427         load->setData("_" + QString::number(i));
428         layoutActions->addAction("load_layout" + QString::number(i), load);
429         m_loadLayout->addAction(load);
430         KAction *save = new KAction(KIcon(), i18n("Save As Layout %1").arg(i), this);
431         save->setData("_" + QString::number(i));
432         layoutActions->addAction("save_layout" + QString::number(i), save);
433     }
434     layoutActions->addAction("load_layouts", m_loadLayout);
435     connect(m_loadLayout, SIGNAL(triggered(QAction*)), this, SLOT(slotLoadLayout(QAction*)));
436
437     KAction *action;
438     // Stop motion actions. Beware of the order, we MUST use the same order in stopmotion/stopmotion.cpp
439     m_stopmotion_actions = new KActionCategory(i18n("Stop Motion"), actionCollection());
440     action = new KAction(KIcon("media-record"), i18n("Capture frame"), this);
441     //action->setShortcutContext(Qt::WidgetWithChildrenShortcut);
442     m_stopmotion_actions->addAction("stopmotion_capture", action);
443     action = new KAction(i18n("Switch live / captured frame"), this);
444     //action->setShortcutContext(Qt::WidgetWithChildrenShortcut);
445     m_stopmotion_actions->addAction("stopmotion_switch", action);
446     action = new KAction(KIcon("edit-paste"), i18n("Show last frame over video"), this);
447     action->setCheckable(true);
448     action->setChecked(false);
449     m_stopmotion_actions->addAction("stopmotion_overlay", action);
450
451     // Build effects menu
452     m_effectsMenu = new QMenu(i18n("Add Effect"));
453     m_effectActions = new KActionCategory(i18n("Effects"), actionCollection());
454     m_effectList->reloadEffectList(m_effectsMenu, m_effectActions);
455     m_effectsActionCollection->readSettings();
456     
457     setupGUI();
458
459     // Find QDockWidget tab bars and show / hide widget title bars on right click
460     QList <QTabBar *> tabs = findChildren<QTabBar *>();
461     for (int i = 0; i < tabs.count(); i++) {
462         tabs.at(i)->setContextMenuPolicy(Qt::CustomContextMenu);
463         connect(tabs.at(i), SIGNAL(customContextMenuRequested(const QPoint &)), this, SLOT(slotSwitchTitles()));
464     }
465
466     /*ScriptingPart* sp = new ScriptingPart(this, QStringList());
467     guiFactory()->addClient(sp);*/
468
469     QMenu *saveLayout = (QMenu*)(factory()->container("layout_save_as", this));
470     if (saveLayout)
471         connect(saveLayout, SIGNAL(triggered(QAction*)), this, SLOT(slotSaveLayout(QAction*)));
472
473
474     // Load layout names from config file
475     loadLayouts();
476
477     loadPlugins();
478     loadTranscoders();
479     loadStabilize();
480
481     m_projectMonitor->setupMenu(static_cast<QMenu*>(factory()->container("monitor_go", this)), m_playZone, m_loopZone, NULL, m_loopClip);
482     m_clipMonitor->setupMenu(static_cast<QMenu*>(factory()->container("monitor_go", this)), m_playZone, m_loopZone, static_cast<QMenu*>(factory()->container("marker_menu", this)));
483
484     QMenu *clipInTimeline = static_cast<QMenu*>(factory()->container("clip_in_timeline", this));
485     clipInTimeline->setIcon(KIcon("go-jump"));
486         QHash<QString,QMenu*> menus;
487         menus.insert("addMenu",static_cast<QMenu*>(factory()->container("generators", this)));
488         menus.insert("transcodeMenu",static_cast<QMenu*>(factory()->container("transcoders", this)));
489         menus.insert("stabilizeMenu",static_cast<QMenu*>(factory()->container("stabilize", this)));
490         menus.insert("inTimelineMenu",clipInTimeline);
491     m_projectList->setupGeneratorMenu(menus);
492
493     // build themes menus
494     QMenu *themesMenu = static_cast<QMenu*>(factory()->container("themes_menu", this));
495     QActionGroup *themegroup = new QActionGroup(this);
496     themegroup->setExclusive(true);
497     action = new KAction(i18n("Default"), this);
498     action->setCheckable(true);
499     themegroup->addAction(action);
500     if (KdenliveSettings::colortheme().isEmpty()) action->setChecked(true);
501
502     const QStringList schemeFiles = KGlobal::dirs()->findAllResources("data", "color-schemes/*.colors", KStandardDirs::NoDuplicates);
503
504     for (int i = 0; i < schemeFiles.size(); ++i) {
505         // get the file name
506         const QString filename = schemeFiles.at(i);
507         const QFileInfo info(filename);
508
509         // add the entry
510         KSharedConfigPtr config = KSharedConfig::openConfig(filename);
511         QIcon icon = createSchemePreviewIcon(config);
512         KConfigGroup group(config, "General");
513         const QString name = group.readEntry("Name", info.baseName());
514         action = new KAction(name, this);
515         action->setData(filename);
516         action->setIcon(icon);
517         action->setCheckable(true);
518         themegroup->addAction(action);
519         if (KdenliveSettings::colortheme() == filename) action->setChecked(true);
520     }
521
522     /*KGlobal::dirs()->addResourceDir("themes", KStandardDirs::installPath("data") + QString("kdenlive/themes"));
523     QStringList themes = KGlobal::dirs()->findAllResources("themes", QString(), KStandardDirs::Recursive | KStandardDirs::NoDuplicates);
524     for (QStringList::const_iterator it = themes.constBegin(); it != themes.constEnd(); ++it)
525     {
526     QFileInfo fi(*it);
527         action = new QAction(fi.fileName(), this);
528         action->setData(*it);
529     action->setCheckable(true);
530     themegroup->addAction(action);
531     if (KdenliveSettings::colortheme() == *it) action->setChecked(true);
532     }*/
533     themesMenu->addActions(themegroup->actions());
534     connect(themesMenu, SIGNAL(triggered(QAction *)), this, SLOT(slotChangePalette(QAction*)));
535
536     // Setup and fill effects and transitions menus.
537
538
539     QMenu *m = static_cast<QMenu*>(factory()->container("video_effects_menu", this));
540     m->addActions(m_effectsMenu->actions());
541
542
543     m_transitionsMenu = new QMenu(i18n("Add Transition"), this);
544     for (int i = 0; i < transitions.count(); ++i)
545         m_transitionsMenu->addAction(m_transitions[i]);
546
547     connect(m, SIGNAL(triggered(QAction *)), this, SLOT(slotAddVideoEffect(QAction *)));
548     connect(m_effectsMenu, SIGNAL(triggered(QAction *)), this, SLOT(slotAddVideoEffect(QAction *)));
549     connect(m_transitionsMenu, SIGNAL(triggered(QAction *)), this, SLOT(slotAddTransition(QAction *)));
550
551     m_effectStack->setMenu(m_effectsMenu);
552
553     QMenu *viewMenu = static_cast<QMenu*>(factory()->container("dockwindows", this));
554     const QList<QAction *> viewActions = createPopupMenu()->actions();
555     viewMenu->insertActions(NULL, viewActions);
556
557     m_timelineContextMenu = new QMenu(this);
558     m_timelineContextClipMenu = new QMenu(this);
559     m_timelineContextTransitionMenu = new QMenu(this);
560
561     m_timelineContextMenu->addAction(actionCollection()->action("insert_space"));
562     m_timelineContextMenu->addAction(actionCollection()->action("delete_space"));
563     m_timelineContextMenu->addAction(actionCollection()->action(KStandardAction::name(KStandardAction::Paste)));
564
565     m_timelineContextClipMenu->addAction(actionCollection()->action("clip_in_project_tree"));
566     //m_timelineContextClipMenu->addAction(actionCollection()->action("clip_to_project_tree"));
567     m_timelineContextClipMenu->addAction(actionCollection()->action("edit_item_duration"));
568     m_timelineContextClipMenu->addAction(actionCollection()->action("delete_item"));
569     m_timelineContextClipMenu->addSeparator();
570     m_timelineContextClipMenu->addAction(actionCollection()->action("group_clip"));
571     m_timelineContextClipMenu->addAction(actionCollection()->action("ungroup_clip"));
572     m_timelineContextClipMenu->addAction(actionCollection()->action("split_audio"));
573     m_timelineContextClipMenu->addSeparator();
574     m_timelineContextClipMenu->addAction(actionCollection()->action("cut_timeline_clip"));
575     m_timelineContextClipMenu->addAction(actionCollection()->action(KStandardAction::name(KStandardAction::Copy)));
576     m_timelineContextClipMenu->addAction(actionCollection()->action("paste_effects"));
577     m_timelineContextClipMenu->addSeparator();
578
579     QMenu *markersMenu = (QMenu*)(factory()->container("marker_menu", this));
580     m_timelineContextClipMenu->addMenu(markersMenu);
581     m_timelineContextClipMenu->addSeparator();
582     m_timelineContextClipMenu->addMenu(m_transitionsMenu);
583     m_timelineContextClipMenu->addMenu(m_effectsMenu);
584
585     m_timelineContextTransitionMenu->addAction(actionCollection()->action("edit_item_duration"));
586     m_timelineContextTransitionMenu->addAction(actionCollection()->action("delete_item"));
587     m_timelineContextTransitionMenu->addAction(actionCollection()->action(KStandardAction::name(KStandardAction::Copy)));
588
589     m_timelineContextTransitionMenu->addAction(actionCollection()->action("auto_transition"));
590
591     connect(m_projectMonitorDock, SIGNAL(visibilityChanged(bool)), m_projectMonitor, SLOT(refreshMonitor(bool)));
592     connect(m_clipMonitorDock, SIGNAL(visibilityChanged(bool)), m_clipMonitor, SLOT(refreshMonitor(bool)));
593     connect(m_monitorManager, SIGNAL(checkColorScopes()), this, SLOT(slotUpdateColorScopes()));
594     connect(m_monitorManager, SIGNAL(clearScopes()), this, SLOT(slotClearColorScopes()));
595     connect(m_effectList, SIGNAL(addEffect(const QDomElement)), this, SLOT(slotAddEffect(const QDomElement)));
596     connect(m_effectList, SIGNAL(reloadEffects()), this, SLOT(slotReloadEffects()));
597
598     slotConnectMonitors();
599
600     // Open or create a file.  Command line argument passed in Url has
601     // precedence, then "openlastproject", then just a plain empty file.
602     // If opening Url fails, openlastproject will _not_ be used.
603     if (!Url.isEmpty()) {
604         // delay loading so that the window shows up
605         m_startUrl = Url;
606         QTimer::singleShot(500, this, SLOT(openFile()));
607     } else if (KdenliveSettings::openlastproject()) {
608         QTimer::singleShot(500, this, SLOT(openLastFile()));
609     } else { //if (m_timelineArea->count() == 0) {
610         newFile(false);
611     }
612
613     if (!clipsToLoad.isEmpty() && m_activeDocument) {
614         QStringList list = clipsToLoad.split(',');
615         QList <QUrl> urls;
616         foreach(QString path, list) {
617             kDebug() << QDir::current().absoluteFilePath(path);
618             urls << QUrl::fromLocalFile(QDir::current().absoluteFilePath(path));
619         }
620         m_projectList->slotAddClip(urls);
621     }
622
623 #ifdef USE_JOGSHUTTLE
624     activateShuttleDevice();
625 #endif
626     m_projectListDock->raise();
627
628     actionCollection()->addAssociatedWidget(m_clipMonitor->container());
629     actionCollection()->addAssociatedWidget(m_projectMonitor->container());
630     
631     // Populate encoding profiles
632     KConfig conf("encodingprofiles.rc", KConfig::FullConfig, "appdata");
633     if (KdenliveSettings::proxyparams().isEmpty() || KdenliveSettings::proxyextension().isEmpty()) {
634         KConfigGroup group(&conf, "proxy");
635         QMap< QString, QString > values = group.entryMap();
636         QMapIterator<QString, QString> i(values);
637         if (i.hasNext()) {
638             i.next();
639             QString data = i.value();
640             KdenliveSettings::setProxyparams(data.section(';', 0, 0));
641             KdenliveSettings::setProxyextension(data.section(';', 1, 1));
642         }
643     }
644     if (KdenliveSettings::v4l_parameters().isEmpty() || KdenliveSettings::v4l_extension().isEmpty()) {
645         KConfigGroup group(&conf, "video4linux");
646         QMap< QString, QString > values = group.entryMap();
647         QMapIterator<QString, QString> i(values);
648         if (i.hasNext()) {
649             i.next();
650             QString data = i.value();
651             KdenliveSettings::setV4l_parameters(data.section(';', 0, 0));
652             KdenliveSettings::setV4l_extension(data.section(';', 1, 1));
653         }
654     }
655     if (KdenliveSettings::decklink_parameters().isEmpty() || KdenliveSettings::decklink_extension().isEmpty()) {
656         KConfigGroup group(&conf, "decklink");
657         QMap< QString, QString > values = group.entryMap();
658         QMapIterator<QString, QString> i(values);
659         if (i.hasNext()) {
660             i.next();
661             QString data = i.value();
662             KdenliveSettings::setDecklink_parameters(data.section(';', 0, 0));
663             KdenliveSettings::setDecklink_extension(data.section(';', 1, 1));
664         }
665     }
666 }
667
668 MainWindow::~MainWindow()
669 {
670     if (m_stopmotion) {
671         delete m_stopmotion;
672     }
673     m_effectStack->slotClipItemSelected(NULL, 0);
674     m_transitionConfig->slotTransitionItemSelected(NULL, 0, QPoint(), false);
675
676     if (m_projectMonitor) m_projectMonitor->stop();
677     if (m_clipMonitor) m_clipMonitor->stop();
678
679     delete m_activeTimeline;
680     delete m_effectStack;
681     delete m_transitionConfig;
682     delete m_activeDocument;
683     delete m_projectMonitor;
684     delete m_clipMonitor;
685     delete m_shortcutRemoveFocus;
686     delete[] m_transitions;
687     Mlt::Factory::close();
688 }
689
690 //virtual
691 bool MainWindow::queryClose()
692 {
693     if (m_renderWidget) {
694         int waitingJobs = m_renderWidget->waitingJobsCount();
695         if (waitingJobs > 0) {
696             switch (KMessageBox::warningYesNoCancel(this, i18np("You have 1 rendering job waiting in the queue.\nWhat do you want to do with this job?", "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")))) {
697             case KMessageBox::Yes :
698                 // create script with waiting jobs and start it
699                 if (m_renderWidget->startWaitingRenderJobs() == false) return false;
700                 break;
701             case KMessageBox::No :
702                 // Don't do anything, jobs will be deleted
703                 break;
704             default:
705                 return false;
706             }
707         }
708     }
709     saveOptions();
710     if (m_monitorManager) m_monitorManager->stopActiveMonitor();
711     // warn the user to save if document is modified and we have clips in our project list
712     if (m_activeDocument && m_activeDocument->isModified() &&
713             ((m_projectList->documentClipList().isEmpty() && !m_activeDocument->url().isEmpty()) ||
714              !m_projectList->documentClipList().isEmpty())) {
715         raise();
716         activateWindow();
717         QString message;
718         if (m_activeDocument->url().fileName().isEmpty())
719             message = i18n("Save changes to document?");
720         else
721             message = i18n("The project <b>\"%1\"</b> has been changed.\nDo you want to save your changes?").arg(m_activeDocument->url().fileName());
722         switch (KMessageBox::warningYesNoCancel(this, message)) {
723         case KMessageBox::Yes :
724             // save document here. If saving fails, return false;
725             return saveFile();
726         case KMessageBox::No :
727             // User does not want to save the changes, clear recovery files
728             m_activeDocument->m_autosave->resize(0);
729             return true;
730         default: // cancel
731             return false;
732         }
733     }
734     return true;
735 }
736
737 void MainWindow::loadPlugins()
738 {
739     foreach(QObject * plugin, QPluginLoader::staticInstances()) {
740         populateMenus(plugin);
741     }
742
743     QStringList directories = KGlobal::dirs()->findDirs("module", QString());
744     QStringList filters;
745     filters << "libkdenlive*";
746     foreach(const QString & folder, directories) {
747         kDebug() << "Parsing plugin folder: " << folder;
748         QDir pluginsDir(folder);
749         foreach(const QString & fileName,
750                 pluginsDir.entryList(filters, QDir::Files)) {
751             /*
752              * Avoid loading the same plugin twice when there is more than one
753              * installation.
754              */
755             if (!m_pluginFileNames.contains(fileName)) {
756                 kDebug() << "Found plugin: " << fileName;
757                 QPluginLoader loader(pluginsDir.absoluteFilePath(fileName));
758                 QObject *plugin = loader.instance();
759                 if (plugin) {
760                     populateMenus(plugin);
761                     m_pluginFileNames += fileName;
762                 } else
763                     kDebug() << "Error loading plugin: " << fileName << ", " << loader.errorString();
764             }
765         }
766     }
767 }
768
769 void MainWindow::populateMenus(QObject *plugin)
770 {
771     QMenu *addMenu = static_cast<QMenu*>(factory()->container("generators", this));
772     ClipGenerator *iGenerator = qobject_cast<ClipGenerator *>(plugin);
773     if (iGenerator)
774         addToMenu(plugin, iGenerator->generators(KdenliveSettings::producerslist()), addMenu, SLOT(generateClip()),
775                   NULL);
776 }
777
778 void MainWindow::addToMenu(QObject *plugin, const QStringList &texts,
779                            QMenu *menu, const char *member,
780                            QActionGroup *actionGroup)
781 {
782     kDebug() << "// ADD to MENU" << texts;
783     foreach(const QString & text, texts) {
784         QAction *action = new QAction(text, plugin);
785         action->setData(text);
786         connect(action, SIGNAL(triggered()), this, member);
787         menu->addAction(action);
788
789         if (actionGroup) {
790             action->setCheckable(true);
791             actionGroup->addAction(action);
792         }
793     }
794 }
795
796 void MainWindow::aboutPlugins()
797 {
798     //PluginDialog dialog(pluginsDir.path(), m_pluginFileNames, this);
799     //dialog.exec();
800 }
801
802
803 void MainWindow::generateClip()
804 {
805     QAction *action = qobject_cast<QAction *>(sender());
806     ClipGenerator *iGenerator = qobject_cast<ClipGenerator *>(action->parent());
807
808     KUrl clipUrl = iGenerator->generatedClip(action->data().toString(), m_activeDocument->projectFolder(),
809                    QStringList(), QStringList(), m_activeDocument->fps(), m_activeDocument->width(), m_activeDocument->height());
810     if (!clipUrl.isEmpty()) {
811         m_projectList->slotAddClip(QList <QUrl> () << clipUrl);
812     }
813 }
814
815 void MainWindow::saveProperties(KConfigGroup &config)
816 {
817     // save properties here,used by session management
818     saveFile();
819     KMainWindow::saveProperties(config);
820 }
821
822
823 void MainWindow::readProperties(const KConfigGroup &config)
824 {
825     // read properties here,used by session management
826     KMainWindow::readProperties(config);
827     QString Lastproject = config.group("Recent Files").readPathEntry("File1", QString());
828     openFile(KUrl(Lastproject));
829 }
830
831 void MainWindow::slotReloadEffects()
832 {
833     initEffects::parseCustomEffectsFile();
834     m_effectList->reloadEffectList(m_effectsMenu, m_effectActions);
835 }
836
837 #ifdef USE_JOGSHUTTLE
838 void MainWindow::activateShuttleDevice()
839 {
840     delete m_jogShuttle;
841     m_jogShuttle = NULL;
842     delete m_jogProcess;
843     m_jogProcess = NULL;
844     if (KdenliveSettings::enableshuttle() == false) return;
845     
846     m_jogProcess = new JogShuttle(KdenliveSettings::shuttledevice());
847     m_jogShuttle = new JogShuttleAction(m_jogProcess, JogShuttleConfig::actionMap(KdenliveSettings::shuttlebuttons()));
848     
849     connect(m_jogShuttle, SIGNAL(rewindOneFrame()), m_monitorManager, SLOT(slotRewindOneFrame()));
850     connect(m_jogShuttle, SIGNAL(forwardOneFrame()), m_monitorManager, SLOT(slotForwardOneFrame()));
851     connect(m_jogShuttle, SIGNAL(rewind(double)), m_monitorManager, SLOT(slotRewind(double)));
852     connect(m_jogShuttle, SIGNAL(forward(double)), m_monitorManager, SLOT(slotForward(double)));
853     connect(m_jogShuttle, SIGNAL(action(const QString&)), this, SLOT(slotDoAction(const QString&)));
854 }
855 #endif /* USE_JOGSHUTTLE */
856
857 void MainWindow::slotDoAction(const QString& action_name)
858 {
859     QAction* action = actionCollection()->action(action_name);
860     if (!action) {
861         fprintf(stderr, "%s", QString("shuttle action '%1' unknown\n").arg(action_name).toAscii().constData());
862         return;
863     }
864     action->trigger();
865 }
866
867 void MainWindow::configureNotifications()
868 {
869     KNotifyConfigWidget::configure(this);
870 }
871
872 void MainWindow::slotFullScreen()
873 {
874     KToggleFullScreenAction::setFullScreen(this, actionCollection()->action("fullscreen")->isChecked());
875 }
876
877 void MainWindow::slotAddEffect(const QDomElement effect)
878 {
879     if (!m_activeDocument) return;
880     if (effect.isNull()) {
881         kDebug() << "--- ERROR, TRYING TO APPEND NULL EFFECT";
882         return;
883     }
884     QDomElement effectToAdd = effect.cloneNode().toElement();
885     bool ok;
886     int ix = m_effectStack->isTrackMode(&ok);
887     if (ok) m_activeTimeline->projectView()->slotAddTrackEffect(effectToAdd, m_activeDocument->tracksCount() - ix);
888     else m_activeTimeline->projectView()->slotAddEffect(effectToAdd, GenTime(), -1);
889 }
890
891 void MainWindow::slotUpdateClip(const QString &id)
892 {
893     if (!m_activeDocument) return;
894     DocClipBase *clip = m_activeDocument->clipManager()->getClipById(id);
895     if (!clip) return;
896     if (clip->numReferences() > 0) m_activeTimeline->projectView()->slotUpdateClip(id);
897     if (m_clipMonitor->activeClip() && m_clipMonitor->activeClip()->getId() == id) {
898         Mlt::Producer *monitorProducer = clip->getCloneProducer();
899         m_clipMonitor->updateClipProducer(monitorProducer);
900     }
901     clip->cleanupProducers();
902 }
903
904 void MainWindow::slotConnectMonitors()
905 {
906     m_projectList->setRenderer(m_projectMonitor->render);
907     connect(m_projectList, SIGNAL(deleteProjectClips(QStringList, QMap<QString, QString>)), this, SLOT(slotDeleteProjectClips(QStringList, QMap<QString, QString>)));
908     connect(m_projectList, SIGNAL(showClipProperties(DocClipBase *)), this, SLOT(slotShowClipProperties(DocClipBase *)));
909     connect(m_projectList, SIGNAL(showClipProperties(QList <DocClipBase *>, QMap<QString, QString>)), this, SLOT(slotShowClipProperties(QList <DocClipBase *>, QMap<QString, QString>)));
910     connect(m_projectMonitor->render, SIGNAL(replyGetImage(const QString &, const QString &, int, int)), m_projectList, SLOT(slotReplyGetImage(const QString &, const QString &, int, int)));
911     connect(m_projectMonitor->render, SIGNAL(replyGetImage(const QString &, const QImage &)), m_projectList, SLOT(slotReplyGetImage(const QString &, const QImage &)));
912     connect(m_projectMonitor->render, SIGNAL(replyGetFileProperties(const QString &, Mlt::Producer*, const stringMap &, const stringMap &, bool)), m_projectList, SLOT(slotReplyGetFileProperties(const QString &, Mlt::Producer*, const stringMap &, const stringMap &, bool)));
913
914     connect(m_projectMonitor->render, SIGNAL(removeInvalidClip(const QString &, bool)), m_projectList, SLOT(slotRemoveInvalidClip(const QString &, bool)));
915     
916     connect(m_projectMonitor->render, SIGNAL(removeInvalidProxy(const QString &, bool)), m_projectList, SLOT(slotRemoveInvalidProxy(const QString &, bool)));
917
918     connect(m_clipMonitor, SIGNAL(refreshClipThumbnail(const QString &, bool)), m_projectList, SLOT(slotRefreshClipThumbnail(const QString &, bool)));
919
920     connect(m_clipMonitor, SIGNAL(adjustMonitorSize()), this, SLOT(slotAdjustClipMonitor()));
921     connect(m_projectMonitor, SIGNAL(adjustMonitorSize()), this, SLOT(slotAdjustProjectMonitor()));
922
923     connect(m_projectMonitor, SIGNAL(requestFrameForAnalysis(bool)), this, SLOT(slotMonitorRequestRenderFrame(bool)));
924
925     connect(m_clipMonitor, SIGNAL(saveZone(Render *, QPoint, DocClipBase *)), this, SLOT(slotSaveZone(Render *, QPoint, DocClipBase *)));
926     connect(m_projectMonitor, SIGNAL(saveZone(Render *, QPoint, DocClipBase *)), this, SLOT(slotSaveZone(Render *, QPoint, DocClipBase *)));
927 }
928
929 void MainWindow::slotAdjustClipMonitor()
930 {
931     m_clipMonitorDock->updateGeometry();
932     m_clipMonitorDock->adjustSize();
933     m_clipMonitor->resetSize();
934 }
935
936 void MainWindow::slotAdjustProjectMonitor()
937 {
938     m_projectMonitorDock->updateGeometry();
939     m_projectMonitorDock->adjustSize();
940     m_projectMonitor->resetSize();
941 }
942
943
944 class NameGrabbingKActionCollection {
945 public:
946   NameGrabbingKActionCollection(KActionCollection* collection, QStringList& action_names)
947     : m_collection(collection), m_action_names(action_names) {
948       m_action_names.clear();
949     }
950   KAction* addAction(const QString& action_name) {
951     m_action_names << action_name;
952     return m_collection->addAction(action_name);
953   }
954   void addAction(const QString& action_name, QAction* action) {
955     m_action_names << action_name;
956     m_collection->addAction(action_name, action);
957   }
958   operator KActionCollection*() { return m_collection; }
959   const QStringList& actionNames() const { return m_action_names; }
960 private:
961   KActionCollection* m_collection;
962   QStringList& m_action_names;
963 };
964
965 void MainWindow::setupActions()
966 {
967
968     NameGrabbingKActionCollection collection(actionCollection(), m_action_names);
969     m_timecodeFormat = new KComboBox(this);
970     m_timecodeFormat->addItem(i18n("hh:mm:ss:ff"));
971     m_timecodeFormat->addItem(i18n("Frames"));
972     if (KdenliveSettings::frametimecode()) m_timecodeFormat->setCurrentIndex(1);
973     connect(m_timecodeFormat, SIGNAL(activated(int)), this, SLOT(slotUpdateTimecodeFormat(int)));
974
975     m_statusProgressBar = new QProgressBar(this);
976     m_statusProgressBar->setMinimum(0);
977     m_statusProgressBar->setMaximum(100);
978     m_statusProgressBar->setMaximumWidth(150);
979     m_statusProgressBar->setVisible(false);
980
981     KToolBar *toolbar = new KToolBar("statusToolBar", this, Qt::BottomToolBarArea);
982     toolbar->setMovable(false);
983     KColorScheme scheme(palette().currentColorGroup(), KColorScheme::Window, KSharedConfig::openConfig(KdenliveSettings::colortheme()));
984     QColor buttonBg = scheme.background(KColorScheme::LinkBackground).color();
985     QColor buttonBord = scheme.foreground(KColorScheme::LinkText).color();
986     QColor buttonBord2 = scheme.shade(KColorScheme::LightShade);
987     statusBar()->setStyleSheet(QString("QStatusBar QLabel {font-size:%1pt;} QStatusBar::item { border: 0px; font-size:%1pt;padding:0px; }").arg(statusBar()->font().pointSize()));
988     QString style1 = QString("QToolBar { border: 0px } QToolButton { border-style: inset; border:1px solid transparent;border-radius: 3px;margin: 0px 3px;padding: 0px;} QToolButton:hover { background: rgb(%7, %8, %9);border-style: inset; border:1px solid rgb(%7, %8, %9);border-radius: 3px;} QToolButton:checked { background-color: rgb(%1, %2, %3); border-style: inset; border:1px solid rgb(%4, %5, %6);border-radius: 3px;}").arg(buttonBg.red()).arg(buttonBg.green()).arg(buttonBg.blue()).arg(buttonBord.red()).arg(buttonBord.green()).arg(buttonBord.blue()).arg(buttonBord2.red()).arg(buttonBord2.green()).arg(buttonBord2.blue());
989     QString styleBorderless = "QToolButton { border-width: 0px;margin: 1px 3px 0px;padding: 0px;}";
990
991     //create edit mode buttons
992     m_normalEditTool = new KAction(KIcon("kdenlive-normal-edit"), i18n("Normal mode"), this);
993     m_normalEditTool->setShortcut(i18nc("Normal editing", "n"));
994     toolbar->addAction(m_normalEditTool);
995     m_normalEditTool->setCheckable(true);
996     m_normalEditTool->setChecked(true);
997
998     m_overwriteEditTool = new KAction(KIcon("kdenlive-overwrite-edit"), i18n("Overwrite mode"), this);
999     //m_overwriteEditTool->setShortcut(i18nc("Overwrite mode shortcut", "o"));
1000     toolbar->addAction(m_overwriteEditTool);
1001     m_overwriteEditTool->setCheckable(true);
1002     m_overwriteEditTool->setChecked(false);
1003
1004     m_insertEditTool = new KAction(KIcon("kdenlive-insert-edit"), i18n("Insert mode"), this);
1005     //m_insertEditTool->setShortcut(i18nc("Insert mode shortcut", "i"));
1006     toolbar->addAction(m_insertEditTool);
1007     m_insertEditTool->setCheckable(true);
1008     m_insertEditTool->setChecked(false);
1009     // not implemented yet
1010     m_insertEditTool->setEnabled(false);
1011
1012     QActionGroup *editGroup = new QActionGroup(this);
1013     editGroup->addAction(m_normalEditTool);
1014     editGroup->addAction(m_overwriteEditTool);
1015     editGroup->addAction(m_insertEditTool);
1016     editGroup->setExclusive(true);
1017     connect(editGroup, SIGNAL(triggered(QAction *)), this, SLOT(slotChangeEdit(QAction *)));
1018     //connect(m_overwriteEditTool, SIGNAL(toggled(bool)), this, SLOT(slotSetOverwriteMode(bool)));
1019
1020     toolbar->addSeparator();
1021
1022     // create tools buttons
1023     m_buttonSelectTool = new KAction(KIcon("kdenlive-select-tool"), i18n("Selection tool"), this);
1024     m_buttonSelectTool->setShortcut(i18nc("Selection tool shortcut", "s"));
1025     toolbar->addAction(m_buttonSelectTool);
1026     m_buttonSelectTool->setCheckable(true);
1027     m_buttonSelectTool->setChecked(true);
1028
1029     m_buttonRazorTool = new KAction(KIcon("edit-cut"), i18n("Razor tool"), this);
1030     m_buttonRazorTool->setShortcut(i18nc("Razor tool shortcut", "x"));
1031     toolbar->addAction(m_buttonRazorTool);
1032     m_buttonRazorTool->setCheckable(true);
1033     m_buttonRazorTool->setChecked(false);
1034
1035     m_buttonSpacerTool = new KAction(KIcon("kdenlive-spacer-tool"), i18n("Spacer tool"), this);
1036     m_buttonSpacerTool->setShortcut(i18nc("Spacer tool shortcut", "m"));
1037     toolbar->addAction(m_buttonSpacerTool);
1038     m_buttonSpacerTool->setCheckable(true);
1039     m_buttonSpacerTool->setChecked(false);
1040
1041     QActionGroup *toolGroup = new QActionGroup(this);
1042     toolGroup->addAction(m_buttonSelectTool);
1043     toolGroup->addAction(m_buttonRazorTool);
1044     toolGroup->addAction(m_buttonSpacerTool);
1045     toolGroup->setExclusive(true);
1046     toolbar->setToolButtonStyle(Qt::ToolButtonIconOnly);
1047
1048     QWidget * actionWidget;
1049     int max = toolbar->iconSizeDefault() + 2;
1050     actionWidget = toolbar->widgetForAction(m_normalEditTool);
1051     actionWidget->setMaximumWidth(max);
1052     actionWidget->setMaximumHeight(max - 4);
1053
1054     actionWidget = toolbar->widgetForAction(m_insertEditTool);
1055     actionWidget->setMaximumWidth(max);
1056     actionWidget->setMaximumHeight(max - 4);
1057
1058     actionWidget = toolbar->widgetForAction(m_overwriteEditTool);
1059     actionWidget->setMaximumWidth(max);
1060     actionWidget->setMaximumHeight(max - 4);
1061
1062     actionWidget = toolbar->widgetForAction(m_buttonSelectTool);
1063     actionWidget->setMaximumWidth(max);
1064     actionWidget->setMaximumHeight(max - 4);
1065
1066     actionWidget = toolbar->widgetForAction(m_buttonRazorTool);
1067     actionWidget->setMaximumWidth(max);
1068     actionWidget->setMaximumHeight(max - 4);
1069
1070     actionWidget = toolbar->widgetForAction(m_buttonSpacerTool);
1071     actionWidget->setMaximumWidth(max);
1072     actionWidget->setMaximumHeight(max - 4);
1073
1074     toolbar->setStyleSheet(style1);
1075     connect(toolGroup, SIGNAL(triggered(QAction *)), this, SLOT(slotChangeTool(QAction *)));
1076
1077     toolbar->addSeparator();
1078     m_buttonFitZoom = new KAction(KIcon("zoom-fit-best"), i18n("Fit zoom to project"), this);
1079     toolbar->addAction(m_buttonFitZoom);
1080     m_buttonFitZoom->setCheckable(false);
1081
1082     m_zoomOut = new KAction(KIcon("zoom-out"), i18n("Zoom Out"), this);
1083     toolbar->addAction(m_zoomOut);
1084     m_zoomOut->setShortcut(Qt::CTRL + Qt::Key_Minus);
1085
1086     m_zoomSlider = new QSlider(Qt::Horizontal, this);
1087     m_zoomSlider->setMaximum(13);
1088     m_zoomSlider->setPageStep(1);
1089     m_zoomSlider->setInvertedAppearance(true);
1090
1091     m_zoomSlider->setMaximumWidth(150);
1092     m_zoomSlider->setMinimumWidth(100);
1093     toolbar->addWidget(m_zoomSlider);
1094
1095     m_zoomIn = new KAction(KIcon("zoom-in"), i18n("Zoom In"), this);
1096     toolbar->addAction(m_zoomIn);
1097     m_zoomIn->setShortcut(Qt::CTRL + Qt::Key_Plus);
1098
1099     actionWidget = toolbar->widgetForAction(m_buttonFitZoom);
1100     actionWidget->setMaximumWidth(max);
1101     actionWidget->setMaximumHeight(max - 4);
1102     actionWidget->setStyleSheet(styleBorderless);
1103
1104     actionWidget = toolbar->widgetForAction(m_zoomIn);
1105     actionWidget->setMaximumWidth(max);
1106     actionWidget->setMaximumHeight(max - 4);
1107     actionWidget->setStyleSheet(styleBorderless);
1108
1109     actionWidget = toolbar->widgetForAction(m_zoomOut);
1110     actionWidget->setMaximumWidth(max);
1111     actionWidget->setMaximumHeight(max - 4);
1112     actionWidget->setStyleSheet(styleBorderless);
1113
1114     connect(m_zoomSlider, SIGNAL(valueChanged(int)), this, SLOT(slotSetZoom(int)));
1115     connect(m_zoomSlider, SIGNAL(sliderMoved(int)), this, SLOT(slotShowZoomSliderToolTip(int)));
1116     connect(m_buttonFitZoom, SIGNAL(triggered()), this, SLOT(slotFitZoom()));
1117     connect(m_zoomIn, SIGNAL(triggered(bool)), this, SLOT(slotZoomIn()));
1118     connect(m_zoomOut, SIGNAL(triggered(bool)), this, SLOT(slotZoomOut()));
1119
1120     toolbar->addSeparator();
1121
1122     //create automatic audio split button
1123     m_buttonAutomaticSplitAudio = new KAction(KIcon("kdenlive-split-audio"), i18n("Split audio and video automatically"), this);
1124     toolbar->addAction(m_buttonAutomaticSplitAudio);
1125     m_buttonAutomaticSplitAudio->setCheckable(true);
1126     m_buttonAutomaticSplitAudio->setChecked(KdenliveSettings::splitaudio());
1127     connect(m_buttonAutomaticSplitAudio, SIGNAL(triggered()), this, SLOT(slotSwitchSplitAudio()));
1128
1129     m_buttonVideoThumbs = new KAction(KIcon("kdenlive-show-videothumb"), i18n("Show video thumbnails"), this);
1130     toolbar->addAction(m_buttonVideoThumbs);
1131     m_buttonVideoThumbs->setCheckable(true);
1132     m_buttonVideoThumbs->setChecked(KdenliveSettings::videothumbnails());
1133     connect(m_buttonVideoThumbs, SIGNAL(triggered()), this, SLOT(slotSwitchVideoThumbs()));
1134
1135     m_buttonAudioThumbs = new KAction(KIcon("kdenlive-show-audiothumb"), i18n("Show audio thumbnails"), this);
1136     toolbar->addAction(m_buttonAudioThumbs);
1137     m_buttonAudioThumbs->setCheckable(true);
1138     m_buttonAudioThumbs->setChecked(KdenliveSettings::audiothumbnails());
1139     connect(m_buttonAudioThumbs, SIGNAL(triggered()), this, SLOT(slotSwitchAudioThumbs()));
1140
1141     m_buttonShowMarkers = new KAction(KIcon("kdenlive-show-markers"), i18n("Show markers comments"), this);
1142     toolbar->addAction(m_buttonShowMarkers);
1143     m_buttonShowMarkers->setCheckable(true);
1144     m_buttonShowMarkers->setChecked(KdenliveSettings::showmarkers());
1145     connect(m_buttonShowMarkers, SIGNAL(triggered()), this, SLOT(slotSwitchMarkersComments()));
1146
1147     m_buttonSnap = new KAction(KIcon("kdenlive-snap"), i18n("Snap"), this);
1148     toolbar->addAction(m_buttonSnap);
1149     m_buttonSnap->setCheckable(true);
1150     m_buttonSnap->setChecked(KdenliveSettings::snaptopoints());
1151     connect(m_buttonSnap, SIGNAL(triggered()), this, SLOT(slotSwitchSnap()));
1152
1153     actionWidget = toolbar->widgetForAction(m_buttonAutomaticSplitAudio);
1154     actionWidget->setMaximumWidth(max);
1155     actionWidget->setMaximumHeight(max - 4);
1156
1157     actionWidget = toolbar->widgetForAction(m_buttonVideoThumbs);
1158     actionWidget->setMaximumWidth(max);
1159     actionWidget->setMaximumHeight(max - 4);
1160
1161     actionWidget = toolbar->widgetForAction(m_buttonAudioThumbs);
1162     actionWidget->setMaximumWidth(max);
1163     actionWidget->setMaximumHeight(max - 4);
1164
1165     actionWidget = toolbar->widgetForAction(m_buttonShowMarkers);
1166     actionWidget->setMaximumWidth(max);
1167     actionWidget->setMaximumHeight(max - 4);
1168
1169     actionWidget = toolbar->widgetForAction(m_buttonSnap);
1170     actionWidget->setMaximumWidth(max);
1171     actionWidget->setMaximumHeight(max - 4);
1172
1173     m_messageLabel = new StatusBarMessageLabel(this);
1174     m_messageLabel->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::MinimumExpanding);
1175
1176     statusBar()->addWidget(m_messageLabel, 10);
1177     statusBar()->addWidget(m_statusProgressBar, 0);
1178     statusBar()->addPermanentWidget(toolbar);
1179     statusBar()->insertPermanentFixedItem("00:00:00:00", ID_TIMELINE_POS);
1180     statusBar()->addPermanentWidget(m_timecodeFormat);
1181     //statusBar()->setMaximumHeight(statusBar()->font().pointSize() * 3);
1182
1183     collection.addAction("normal_mode", m_normalEditTool);
1184     collection.addAction("overwrite_mode", m_overwriteEditTool);
1185     collection.addAction("insert_mode", m_insertEditTool);
1186     collection.addAction("select_tool", m_buttonSelectTool);
1187     collection.addAction("razor_tool", m_buttonRazorTool);
1188     collection.addAction("spacer_tool", m_buttonSpacerTool);
1189
1190     collection.addAction("automatic_split_audio", m_buttonAutomaticSplitAudio);
1191     collection.addAction("show_video_thumbs", m_buttonVideoThumbs);
1192     collection.addAction("show_audio_thumbs", m_buttonAudioThumbs);
1193     collection.addAction("show_markers", m_buttonShowMarkers);
1194     collection.addAction("snap", m_buttonSnap);
1195     collection.addAction("zoom_fit", m_buttonFitZoom);
1196     collection.addAction("zoom_in", m_zoomIn);
1197     collection.addAction("zoom_out", m_zoomOut);
1198
1199     m_projectSearch = new KAction(KIcon("edit-find"), i18n("Find"), this);
1200     collection.addAction("project_find", m_projectSearch);
1201     connect(m_projectSearch, SIGNAL(triggered(bool)), this, SLOT(slotFind()));
1202     m_projectSearch->setShortcut(Qt::Key_Slash);
1203
1204     m_projectSearchNext = new KAction(KIcon("go-down-search"), i18n("Find Next"), this);
1205     collection.addAction("project_find_next", m_projectSearchNext);
1206     connect(m_projectSearchNext, SIGNAL(triggered(bool)), this, SLOT(slotFindNext()));
1207     m_projectSearchNext->setShortcut(Qt::Key_F3);
1208     m_projectSearchNext->setEnabled(false);
1209
1210     KAction* profilesAction = new KAction(KIcon("document-new"), i18n("Manage Project Profiles"), this);
1211     collection.addAction("manage_profiles", profilesAction);
1212     connect(profilesAction, SIGNAL(triggered(bool)), this, SLOT(slotEditProfiles()));
1213
1214     KNS3::standardAction(i18n("Download New Wipes..."),            this, SLOT(slotGetNewLumaStuff()),       actionCollection(), "get_new_lumas");
1215     KNS3::standardAction(i18n("Download New Render Profiles..."),  this, SLOT(slotGetNewRenderStuff()),     actionCollection(), "get_new_profiles");
1216     KNS3::standardAction(i18n("Download New Project Profiles..."), this, SLOT(slotGetNewMltProfileStuff()), actionCollection(), "get_new_mlt_profiles");
1217     KNS3::standardAction(i18n("Download New Title Templates..."),  this, SLOT(slotGetNewTitleStuff()),      actionCollection(), "get_new_titles");
1218
1219     KAction* wizAction = new KAction(KIcon("configure"), i18n("Run Config Wizard"), this);
1220     collection.addAction("run_wizard", wizAction);
1221     connect(wizAction, SIGNAL(triggered(bool)), this, SLOT(slotRunWizard()));
1222
1223     KAction* projectAction = new KAction(KIcon("configure"), i18n("Project Settings"), this);
1224     collection.addAction("project_settings", projectAction);
1225     connect(projectAction, SIGNAL(triggered(bool)), this, SLOT(slotEditProjectSettings()));
1226
1227     KAction* backupAction = new KAction(KIcon("edit-undo"), i18n("Open Backup File"), this);
1228     collection.addAction("open_backup", backupAction);
1229     connect(backupAction, SIGNAL(triggered(bool)), this, SLOT(slotOpenBackupDialog()));
1230
1231     KAction* projectRender = new KAction(KIcon("media-record"), i18n("Render"), this);
1232     collection.addAction("project_render", projectRender);
1233     projectRender->setShortcut(Qt::CTRL + Qt::Key_Return);
1234     connect(projectRender, SIGNAL(triggered(bool)), this, SLOT(slotRenderProject()));
1235
1236     KAction* projectClean = new KAction(KIcon("edit-clear"), i18n("Clean Project"), this);
1237     collection.addAction("project_clean", projectClean);
1238     connect(projectClean, SIGNAL(triggered(bool)), this, SLOT(slotCleanProject()));
1239
1240     KAction* projectAdjust = new KAction(KIcon(), i18n("Adjust Profile to Current Clip"), this);
1241     collection.addAction("project_adjust_profile", projectAdjust);
1242     connect(projectAdjust, SIGNAL(triggered(bool)), m_projectList, SLOT(adjustProjectProfileToItem()));
1243
1244     KAction* monitorPlay = new KAction(KIcon("media-playback-start"), i18n("Play"), this);
1245     KShortcut playShortcut;
1246     playShortcut.setPrimary(Qt::Key_Space);
1247     playShortcut.setAlternate(Qt::Key_K);
1248     monitorPlay->setShortcut(playShortcut);
1249     collection.addAction("monitor_play", monitorPlay);
1250     connect(monitorPlay, SIGNAL(triggered(bool)), m_monitorManager, SLOT(slotPlay()));
1251
1252     KAction* monitorPause = new KAction(KIcon("media-playback-stop"), i18n("Pause"), this);
1253     collection.addAction("monitor_pause", monitorPause);
1254     connect(monitorPause, SIGNAL(triggered(bool)), m_monitorManager, SLOT(slotPause()));
1255
1256     m_playZone = new KAction(KIcon("media-playback-start"), i18n("Play Zone"), this);
1257     m_playZone->setShortcut(Qt::CTRL + Qt::Key_Space);
1258     collection.addAction("monitor_play_zone", m_playZone);
1259     connect(m_playZone, SIGNAL(triggered(bool)), m_monitorManager, SLOT(slotPlayZone()));
1260
1261     m_loopZone = new KAction(KIcon("media-playback-start"), i18n("Loop Zone"), this);
1262     m_loopZone->setShortcut(Qt::ALT + Qt::Key_Space);
1263     collection.addAction("monitor_loop_zone", m_loopZone);
1264     connect(m_loopZone, SIGNAL(triggered(bool)), m_monitorManager, SLOT(slotLoopZone()));
1265
1266     m_loopClip = new KAction(KIcon("media-playback-start"), i18n("Loop selected clip"), this);
1267     m_loopClip->setEnabled(false);
1268     collection.addAction("monitor_loop_clip", m_loopClip);
1269     connect(m_loopClip, SIGNAL(triggered(bool)), m_projectMonitor, SLOT(slotLoopClip()));
1270
1271     KAction *dvdWizard =  new KAction(KIcon("media-optical"), i18n("DVD Wizard"), this);
1272     collection.addAction("dvd_wizard", dvdWizard);
1273     connect(dvdWizard, SIGNAL(triggered(bool)), this, SLOT(slotDvdWizard()));
1274
1275     KAction *transcodeClip =  new KAction(KIcon("edit-copy"), i18n("Transcode Clips"), this);
1276     collection.addAction("transcode_clip", transcodeClip);
1277     connect(transcodeClip, SIGNAL(triggered(bool)), this, SLOT(slotTranscodeClip()));
1278
1279     KAction *archiveProject =  new KAction(KIcon("file-save"), i18n("Archive Project"), this);
1280     collection.addAction("archive_project", archiveProject);
1281     connect(archiveProject, SIGNAL(triggered(bool)), this, SLOT(slotArchiveProject()));
1282     
1283
1284     KAction *markIn = collection.addAction("mark_in");
1285     markIn->setText(i18n("Set Zone In"));
1286     markIn->setShortcut(Qt::Key_I);
1287     connect(markIn, SIGNAL(triggered(bool)), this, SLOT(slotSetInPoint()));
1288
1289     KAction *markOut = collection.addAction("mark_out");
1290     markOut->setText(i18n("Set Zone Out"));
1291     markOut->setShortcut(Qt::Key_O);
1292     connect(markOut, SIGNAL(triggered(bool)), this, SLOT(slotSetOutPoint()));
1293
1294     KAction *switchMon = collection.addAction("switch_monitor");
1295     switchMon->setText(i18n("Switch monitor"));
1296     switchMon->setShortcut(Qt::Key_T);
1297     connect(switchMon, SIGNAL(triggered(bool)), this, SLOT(slotSwitchMonitors()));
1298
1299     KAction *fullMon = collection.addAction("monitor_fullscreen");
1300     fullMon->setText(i18n("Switch monitor fullscreen"));
1301     fullMon->setIcon(KIcon("view-fullscreen"));
1302     connect(fullMon, SIGNAL(triggered(bool)), this, SLOT(slotSwitchFullscreen()));
1303
1304     KAction *insertTree = collection.addAction("insert_project_tree");
1305     insertTree->setText(i18n("Insert zone in project tree"));
1306     insertTree->setShortcut(Qt::CTRL + Qt::Key_I);
1307     connect(insertTree, SIGNAL(triggered(bool)), this, SLOT(slotInsertZoneToTree()));
1308
1309     KAction *insertTimeline = collection.addAction("insert_timeline");
1310     insertTimeline->setText(i18n("Insert zone in timeline"));
1311     insertTimeline->setShortcut(Qt::SHIFT + Qt::CTRL + Qt::Key_I);
1312     connect(insertTimeline, SIGNAL(triggered(bool)), this, SLOT(slotInsertZoneToTimeline()));
1313
1314     KAction *resizeStart =  new KAction(KIcon(), i18n("Resize Item Start"), this);
1315     collection.addAction("resize_timeline_clip_start", resizeStart);
1316     resizeStart->setShortcut(Qt::Key_1);
1317     connect(resizeStart, SIGNAL(triggered(bool)), this, SLOT(slotResizeItemStart()));
1318
1319     KAction *resizeEnd =  new KAction(KIcon(), i18n("Resize Item End"), this);
1320     collection.addAction("resize_timeline_clip_end", resizeEnd);
1321     resizeEnd->setShortcut(Qt::Key_2);
1322     connect(resizeEnd, SIGNAL(triggered(bool)), this, SLOT(slotResizeItemEnd()));
1323
1324     KAction* monitorSeekBackward = new KAction(KIcon("media-seek-backward"), i18n("Rewind"), this);
1325     monitorSeekBackward->setShortcut(Qt::Key_J);
1326     collection.addAction("monitor_seek_backward", monitorSeekBackward);
1327     connect(monitorSeekBackward, SIGNAL(triggered(bool)), m_monitorManager, SLOT(slotRewind()));
1328
1329     KAction* monitorSeekBackwardOneFrame = new KAction(KIcon("media-skip-backward"), i18n("Rewind 1 Frame"), this);
1330     monitorSeekBackwardOneFrame->setShortcut(Qt::Key_Left);
1331     collection.addAction("monitor_seek_backward-one-frame", monitorSeekBackwardOneFrame);
1332     connect(monitorSeekBackwardOneFrame, SIGNAL(triggered(bool)), m_monitorManager, SLOT(slotRewindOneFrame()));
1333
1334     KAction* monitorSeekBackwardOneSecond = new KAction(KIcon("media-skip-backward"), i18n("Rewind 1 Second"), this);
1335     monitorSeekBackwardOneSecond->setShortcut(Qt::SHIFT + Qt::Key_Left);
1336     collection.addAction("monitor_seek_backward-one-second", monitorSeekBackwardOneSecond);
1337     connect(monitorSeekBackwardOneSecond, SIGNAL(triggered(bool)), m_monitorManager, SLOT(slotRewindOneSecond()));
1338
1339     KAction* monitorSeekSnapBackward = new KAction(KIcon("media-seek-backward"), i18n("Go to Previous Snap Point"), this);
1340     monitorSeekSnapBackward->setShortcut(Qt::ALT + Qt::Key_Left);
1341     collection.addAction("monitor_seek_snap_backward", monitorSeekSnapBackward);
1342     connect(monitorSeekSnapBackward, SIGNAL(triggered(bool)), this, SLOT(slotSnapRewind()));
1343
1344     KAction* monitorSeekForward = new KAction(KIcon("media-seek-forward"), i18n("Forward"), this);
1345     monitorSeekForward->setShortcut(Qt::Key_L);
1346     collection.addAction("monitor_seek_forward", monitorSeekForward);
1347     connect(monitorSeekForward, SIGNAL(triggered(bool)), m_monitorManager, SLOT(slotForward()));
1348
1349     KAction* clipStart = new KAction(KIcon("media-seek-backward"), i18n("Go to Clip Start"), this);
1350     clipStart->setShortcut(Qt::Key_Home);
1351     collection.addAction("seek_clip_start", clipStart);
1352     connect(clipStart, SIGNAL(triggered(bool)), this, SLOT(slotClipStart()));
1353
1354     KAction* clipEnd = new KAction(KIcon("media-seek-forward"), i18n("Go to Clip End"), this);
1355     clipEnd->setShortcut(Qt::Key_End);
1356     collection.addAction("seek_clip_end", clipEnd);
1357     connect(clipEnd, SIGNAL(triggered(bool)), this, SLOT(slotClipEnd()));
1358
1359     KAction* zoneStart = new KAction(KIcon("media-seek-backward"), i18n("Go to Zone Start"), this);
1360     zoneStart->setShortcut(Qt::SHIFT + Qt::Key_I);
1361     collection.addAction("seek_zone_start", zoneStart);
1362     connect(zoneStart, SIGNAL(triggered(bool)), this, SLOT(slotZoneStart()));
1363
1364     KAction* zoneEnd = new KAction(KIcon("media-seek-forward"), i18n("Go to Zone End"), this);
1365     zoneEnd->setShortcut(Qt::SHIFT + Qt::Key_O);
1366     collection.addAction("seek_zone_end", zoneEnd);
1367     connect(zoneEnd, SIGNAL(triggered(bool)), this, SLOT(slotZoneEnd()));
1368
1369     KAction* projectStart = new KAction(KIcon("go-first"), i18n("Go to Project Start"), this);
1370     projectStart->setShortcut(Qt::CTRL + Qt::Key_Home);
1371     collection.addAction("seek_start", projectStart);
1372     connect(projectStart, SIGNAL(triggered(bool)), m_monitorManager, SLOT(slotStart()));
1373
1374     KAction* projectEnd = new KAction(KIcon("go-last"), i18n("Go to Project End"), this);
1375     projectEnd->setShortcut(Qt::CTRL + Qt::Key_End);
1376     collection.addAction("seek_end", projectEnd);
1377     connect(projectEnd, SIGNAL(triggered(bool)), m_monitorManager, SLOT(slotEnd()));
1378
1379     KAction* monitorSeekForwardOneFrame = new KAction(KIcon("media-skip-forward"), i18n("Forward 1 Frame"), this);
1380     monitorSeekForwardOneFrame->setShortcut(Qt::Key_Right);
1381     collection.addAction("monitor_seek_forward-one-frame", monitorSeekForwardOneFrame);
1382     connect(monitorSeekForwardOneFrame, SIGNAL(triggered(bool)), m_monitorManager, SLOT(slotForwardOneFrame()));
1383
1384     KAction* monitorSeekForwardOneSecond = new KAction(KIcon("media-skip-forward"), i18n("Forward 1 Second"), this);
1385     monitorSeekForwardOneSecond->setShortcut(Qt::SHIFT + Qt::Key_Right);
1386     collection.addAction("monitor_seek_forward-one-second", monitorSeekForwardOneSecond);
1387     connect(monitorSeekForwardOneSecond, SIGNAL(triggered(bool)), m_monitorManager, SLOT(slotForwardOneSecond()));
1388
1389     KAction* monitorSeekSnapForward = new KAction(KIcon("media-seek-forward"), i18n("Go to Next Snap Point"), this);
1390     monitorSeekSnapForward->setShortcut(Qt::ALT + Qt::Key_Right);
1391     collection.addAction("monitor_seek_snap_forward", monitorSeekSnapForward);
1392     connect(monitorSeekSnapForward, SIGNAL(triggered(bool)), this, SLOT(slotSnapForward()));
1393
1394     KAction* deleteItem = new KAction(KIcon("edit-delete"), i18n("Delete Selected Item"), this);
1395     deleteItem->setShortcut(Qt::Key_Delete);
1396     collection.addAction("delete_timeline_clip", deleteItem);
1397     connect(deleteItem, SIGNAL(triggered(bool)), this, SLOT(slotDeleteItem()));
1398
1399     /*KAction* editTimelineClipSpeed = new KAction(i18n("Change Clip Speed"), this);
1400     collection.addAction("change_clip_speed", editTimelineClipSpeed);
1401     editTimelineClipSpeed->setData("change_speed");
1402     connect(editTimelineClipSpeed, SIGNAL(triggered(bool)), this, SLOT(slotChangeClipSpeed()));*/
1403
1404     KAction *stickTransition = collection.addAction("auto_transition");
1405     stickTransition->setData(QString("auto"));
1406     stickTransition->setCheckable(true);
1407     stickTransition->setEnabled(false);
1408     stickTransition->setText(i18n("Automatic Transition"));
1409     connect(stickTransition, SIGNAL(triggered(bool)), this, SLOT(slotAutoTransition()));
1410
1411     KAction* groupClip = new KAction(KIcon("object-group"), i18n("Group Clips"), this);
1412     groupClip->setShortcut(Qt::CTRL + Qt::Key_G);
1413     collection.addAction("group_clip", groupClip);
1414     connect(groupClip, SIGNAL(triggered(bool)), this, SLOT(slotGroupClips()));
1415
1416     KAction* ungroupClip = new KAction(KIcon("object-ungroup"), i18n("Ungroup Clips"), this);
1417     collection.addAction("ungroup_clip", ungroupClip);
1418     ungroupClip->setShortcut(Qt::CTRL + Qt::SHIFT + Qt::Key_G);
1419     ungroupClip->setData("ungroup_clip");
1420     connect(ungroupClip, SIGNAL(triggered(bool)), this, SLOT(slotUnGroupClips()));
1421
1422     KAction* editItemDuration = new KAction(KIcon("measure"), i18n("Edit Duration"), this);
1423     collection.addAction("edit_item_duration", editItemDuration);
1424     connect(editItemDuration, SIGNAL(triggered(bool)), this, SLOT(slotEditItemDuration()));
1425
1426     KAction* clipInProjectTree = new KAction(KIcon("go-jump-definition"), i18n("Clip in Project Tree"), this);
1427     collection.addAction("clip_in_project_tree", clipInProjectTree);
1428     connect(clipInProjectTree, SIGNAL(triggered(bool)), this, SLOT(slotClipInProjectTree()));
1429
1430     /*KAction* clipToProjectTree = new KAction(KIcon("go-jump-definition"), i18n("Add Clip to Project Tree"), this);
1431     collection.addAction("clip_to_project_tree", clipToProjectTree);
1432     connect(clipToProjectTree, SIGNAL(triggered(bool)), this, SLOT(slotClipToProjectTree()));*/
1433
1434     KAction* insertOvertwrite = new KAction(KIcon(), i18n("Insert Clip Zone in Timeline (Overwrite)"), this);
1435     insertOvertwrite->setShortcut(Qt::Key_V);
1436     collection.addAction("overwrite_to_in_point", insertOvertwrite);
1437     connect(insertOvertwrite, SIGNAL(triggered(bool)), this, SLOT(slotInsertClipOverwrite()));
1438
1439     KAction* selectTimelineClip = new KAction(KIcon("edit-select"), i18n("Select Clip"), this);
1440     selectTimelineClip->setShortcut(Qt::Key_Plus);
1441     collection.addAction("select_timeline_clip", selectTimelineClip);
1442     connect(selectTimelineClip, SIGNAL(triggered(bool)), this, SLOT(slotSelectTimelineClip()));
1443
1444     KAction* deselectTimelineClip = new KAction(KIcon("edit-select"), i18n("Deselect Clip"), this);
1445     deselectTimelineClip->setShortcut(Qt::Key_Minus);
1446     collection.addAction("deselect_timeline_clip", deselectTimelineClip);
1447     connect(deselectTimelineClip, SIGNAL(triggered(bool)), this, SLOT(slotDeselectTimelineClip()));
1448
1449     KAction* selectAddTimelineClip = new KAction(KIcon("edit-select"), i18n("Add Clip To Selection"), this);
1450     selectAddTimelineClip->setShortcut(Qt::ALT + Qt::Key_Plus);
1451     collection.addAction("select_add_timeline_clip", selectAddTimelineClip);
1452     connect(selectAddTimelineClip, SIGNAL(triggered(bool)), this, SLOT(slotSelectAddTimelineClip()));
1453
1454     KAction* selectTimelineTransition = new KAction(KIcon("edit-select"), i18n("Select Transition"), this);
1455     selectTimelineTransition->setShortcut(Qt::SHIFT + Qt::Key_Plus);
1456     collection.addAction("select_timeline_transition", selectTimelineTransition);
1457     connect(selectTimelineTransition, SIGNAL(triggered(bool)), this, SLOT(slotSelectTimelineTransition()));
1458
1459     KAction* deselectTimelineTransition = new KAction(KIcon("edit-select"), i18n("Deselect Transition"), this);
1460     deselectTimelineTransition->setShortcut(Qt::SHIFT + Qt::Key_Minus);
1461     collection.addAction("deselect_timeline_transition", deselectTimelineTransition);
1462     connect(deselectTimelineTransition, SIGNAL(triggered(bool)), this, SLOT(slotDeselectTimelineTransition()));
1463
1464     KAction* selectAddTimelineTransition = new KAction(KIcon("edit-select"), i18n("Add Transition To Selection"), this);
1465     selectAddTimelineTransition->setShortcut(Qt::ALT + Qt::SHIFT + Qt::Key_Plus);
1466     collection.addAction("select_add_timeline_transition", selectAddTimelineTransition);
1467     connect(selectAddTimelineTransition, SIGNAL(triggered(bool)), this, SLOT(slotSelectAddTimelineTransition()));
1468
1469     KAction* cutTimelineClip = new KAction(KIcon("edit-cut"), i18n("Cut Clip"), this);
1470     cutTimelineClip->setShortcut(Qt::SHIFT + Qt::Key_R);
1471     collection.addAction("cut_timeline_clip", cutTimelineClip);
1472     connect(cutTimelineClip, SIGNAL(triggered(bool)), this, SLOT(slotCutTimelineClip()));
1473
1474     KAction* addClipMarker = new KAction(KIcon("bookmark-new"), i18n("Add Marker"), this);
1475     collection.addAction("add_clip_marker", addClipMarker);
1476     connect(addClipMarker, SIGNAL(triggered(bool)), this, SLOT(slotAddClipMarker()));
1477
1478     KAction* deleteClipMarker = new KAction(KIcon("edit-delete"), i18n("Delete Marker"), this);
1479     collection.addAction("delete_clip_marker", deleteClipMarker);
1480     connect(deleteClipMarker, SIGNAL(triggered(bool)), this, SLOT(slotDeleteClipMarker()));
1481
1482     KAction* deleteAllClipMarkers = new KAction(KIcon("edit-delete"), i18n("Delete All Markers"), this);
1483     collection.addAction("delete_all_clip_markers", deleteAllClipMarkers);
1484     connect(deleteAllClipMarkers, SIGNAL(triggered(bool)), this, SLOT(slotDeleteAllClipMarkers()));
1485
1486     KAction* editClipMarker = new KAction(KIcon("document-properties"), i18n("Edit Marker"), this);
1487     editClipMarker->setData(QString("edit_marker"));
1488     collection.addAction("edit_clip_marker", editClipMarker);
1489     connect(editClipMarker, SIGNAL(triggered(bool)), this, SLOT(slotEditClipMarker()));
1490
1491     KAction *addMarkerGuideQuickly = new KAction(KIcon("bookmark-new"), i18n("Add Marker/Guide quickly"), this);
1492     addMarkerGuideQuickly->setShortcut(Qt::Key_Asterisk);
1493     collection.addAction("add_marker_guide_quickly", addMarkerGuideQuickly);
1494     connect(addMarkerGuideQuickly, SIGNAL(triggered(bool)), this, SLOT(slotAddMarkerGuideQuickly()));
1495
1496     KAction* splitAudio = new KAction(KIcon("document-new"), i18n("Split Audio"), this);
1497     collection.addAction("split_audio", splitAudio);
1498     connect(splitAudio, SIGNAL(triggered(bool)), this, SLOT(slotSplitAudio()));
1499
1500     KAction* audioOnly = new KAction(KIcon("document-new"), i18n("Audio Only"), this);
1501     collection.addAction("clip_audio_only", audioOnly);
1502     audioOnly->setData("clip_audio_only");
1503     audioOnly->setCheckable(true);
1504
1505     KAction* videoOnly = new KAction(KIcon("document-new"), i18n("Video Only"), this);
1506     collection.addAction("clip_video_only", videoOnly);
1507     videoOnly->setData("clip_video_only");
1508     videoOnly->setCheckable(true);
1509
1510     KAction* audioAndVideo = new KAction(KIcon("document-new"), i18n("Audio and Video"), this);
1511     collection.addAction("clip_audio_and_video", audioAndVideo);
1512     audioAndVideo->setData("clip_audio_and_video");
1513     audioAndVideo->setCheckable(true);
1514
1515     m_clipTypeGroup = new QActionGroup(this);
1516     m_clipTypeGroup->addAction(audioOnly);
1517     m_clipTypeGroup->addAction(videoOnly);
1518     m_clipTypeGroup->addAction(audioAndVideo);
1519     connect(m_clipTypeGroup, SIGNAL(triggered(QAction *)), this, SLOT(slotUpdateClipType(QAction *)));
1520     m_clipTypeGroup->setEnabled(false);
1521
1522     KAction *insertSpace = new KAction(KIcon(), i18n("Insert Space"), this);
1523     collection.addAction("insert_space", insertSpace);
1524     connect(insertSpace, SIGNAL(triggered()), this, SLOT(slotInsertSpace()));
1525
1526     KAction *removeSpace = new KAction(KIcon(), i18n("Remove Space"), this);
1527     collection.addAction("delete_space", removeSpace);
1528     connect(removeSpace, SIGNAL(triggered()), this, SLOT(slotRemoveSpace()));
1529
1530     KAction *insertTrack = new KAction(KIcon(), i18n("Insert Track"), this);
1531     collection.addAction("insert_track", insertTrack);
1532     connect(insertTrack, SIGNAL(triggered()), this, SLOT(slotInsertTrack()));
1533
1534     KAction *deleteTrack = new KAction(KIcon(), i18n("Delete Track"), this);
1535     collection.addAction("delete_track", deleteTrack);
1536     connect(deleteTrack, SIGNAL(triggered()), this, SLOT(slotDeleteTrack()));
1537
1538     KAction *configTracks = new KAction(KIcon("configure"), i18n("Configure Tracks"), this);
1539     collection.addAction("config_tracks", configTracks);
1540     connect(configTracks, SIGNAL(triggered()), this, SLOT(slotConfigTrack()));
1541
1542     KAction *addGuide = new KAction(KIcon("document-new"), i18n("Add Guide"), this);
1543     collection.addAction("add_guide", addGuide);
1544     connect(addGuide, SIGNAL(triggered()), this, SLOT(slotAddGuide()));
1545
1546     QAction *delGuide = new KAction(KIcon("edit-delete"), i18n("Delete Guide"), this);
1547     collection.addAction("delete_guide", delGuide);
1548     connect(delGuide, SIGNAL(triggered()), this, SLOT(slotDeleteGuide()));
1549
1550     QAction *editGuide = new KAction(KIcon("document-properties"), i18n("Edit Guide"), this);
1551     collection.addAction("edit_guide", editGuide);
1552     connect(editGuide, SIGNAL(triggered()), this, SLOT(slotEditGuide()));
1553
1554     QAction *delAllGuides = new KAction(KIcon("edit-delete"), i18n("Delete All Guides"), this);
1555     collection.addAction("delete_all_guides", delAllGuides);
1556     connect(delAllGuides, SIGNAL(triggered()), this, SLOT(slotDeleteAllGuides()));
1557
1558     QAction *pasteEffects = new KAction(KIcon("edit-paste"), i18n("Paste Effects"), this);
1559     collection.addAction("paste_effects", pasteEffects);
1560     pasteEffects->setData("paste_effects");
1561     connect(pasteEffects , SIGNAL(triggered()), this, SLOT(slotPasteEffects()));
1562
1563     QAction *showTimeline = new KAction(i18n("Show Timeline"), this);
1564     collection.addAction("show_timeline", showTimeline);
1565     showTimeline->setCheckable(true);
1566     showTimeline->setChecked(true);
1567     connect(showTimeline, SIGNAL(triggered(bool)), this, SLOT(slotShowTimeline(bool)));
1568
1569     QAction *showTitleBar = new KAction(i18n("Show Title Bars"), this);
1570     collection.addAction("show_titlebars", showTitleBar);
1571     showTitleBar->setCheckable(true);
1572     connect(showTitleBar, SIGNAL(triggered(bool)), this, SLOT(slotShowTitleBars(bool)));
1573     showTitleBar->setChecked(KdenliveSettings::showtitlebars());
1574     slotShowTitleBars(KdenliveSettings::showtitlebars());
1575
1576     m_closeAction = KStandardAction::close(this,  SLOT(closeCurrentDocument()),   collection);
1577     KStandardAction::quit(this,                   SLOT(close()),                  collection);
1578     KStandardAction::open(this,                   SLOT(openFile()),               collection);
1579     m_saveAction = KStandardAction::save(this,    SLOT(saveFile()),               collection);
1580     KStandardAction::saveAs(this,                 SLOT(saveFileAs()),             collection);
1581     KStandardAction::openNew(this,                SLOT(newFile()),                collection);
1582     // TODO: make the following connection to slotEditKeys work
1583     KStandardAction::keyBindings(this,            SLOT(slotEditKeys()),           collection);
1584     KStandardAction::preferences(this,            SLOT(slotPreferences()),        collection);
1585     KStandardAction::configureNotifications(this, SLOT(configureNotifications()), collection);
1586     KStandardAction::copy(this,                   SLOT(slotCopy()),               collection);
1587     KStandardAction::paste(this,                  SLOT(slotPaste()),              collection);
1588     KStandardAction::fullScreen(this,             SLOT(slotFullScreen()), this,   collection);
1589
1590     KAction *undo = KStandardAction::undo(m_commandStack, SLOT(undo()), collection);
1591     undo->setEnabled(false);
1592     connect(m_commandStack, SIGNAL(canUndoChanged(bool)), undo, SLOT(setEnabled(bool)));
1593
1594     KAction *redo = KStandardAction::redo(m_commandStack, SLOT(redo()), collection);
1595     redo->setEnabled(false);
1596     connect(m_commandStack, SIGNAL(canRedoChanged(bool)), redo, SLOT(setEnabled(bool)));
1597
1598     /*
1599     //TODO: Add status tooltip to actions ?
1600     connect(collection, SIGNAL(actionHovered(QAction*)),
1601             this, SLOT(slotDisplayActionMessage(QAction*)));*/
1602
1603
1604     QAction *addClip = new KAction(KIcon("kdenlive-add-clip"), i18n("Add Clip"), this);
1605     collection.addAction("add_clip", addClip);
1606     connect(addClip , SIGNAL(triggered()), m_projectList, SLOT(slotAddClip()));
1607
1608     QAction *addColorClip = new KAction(KIcon("kdenlive-add-color-clip"), i18n("Add Color Clip"), this);
1609     collection.addAction("add_color_clip", addColorClip);
1610     connect(addColorClip , SIGNAL(triggered()), m_projectList, SLOT(slotAddColorClip()));
1611
1612     QAction *addSlideClip = new KAction(KIcon("kdenlive-add-slide-clip"), i18n("Add Slideshow Clip"), this);
1613     collection.addAction("add_slide_clip", addSlideClip);
1614     connect(addSlideClip , SIGNAL(triggered()), m_projectList, SLOT(slotAddSlideshowClip()));
1615
1616     QAction *addTitleClip = new KAction(KIcon("kdenlive-add-text-clip"), i18n("Add Title Clip"), this);
1617     collection.addAction("add_text_clip", addTitleClip);
1618     connect(addTitleClip , SIGNAL(triggered()), m_projectList, SLOT(slotAddTitleClip()));
1619
1620     QAction *addTitleTemplateClip = new KAction(KIcon("kdenlive-add-text-clip"), i18n("Add Template Title"), this);
1621     collection.addAction("add_text_template_clip", addTitleTemplateClip);
1622     connect(addTitleTemplateClip , SIGNAL(triggered()), m_projectList, SLOT(slotAddTitleTemplateClip()));
1623
1624     QAction *addFolderButton = new KAction(KIcon("folder-new"), i18n("Create Folder"), this);
1625     collection.addAction("add_folder", addFolderButton);
1626     connect(addFolderButton , SIGNAL(triggered()), m_projectList, SLOT(slotAddFolder()));
1627
1628     QAction *clipProperties = new KAction(KIcon("document-edit"), i18n("Clip Properties"), this);
1629     collection.addAction("clip_properties", clipProperties);
1630     clipProperties->setData("clip_properties");
1631     connect(clipProperties , SIGNAL(triggered()), m_projectList, SLOT(slotEditClip()));
1632     clipProperties->setEnabled(false);
1633
1634     QAction *openClip = new KAction(KIcon("document-open"), i18n("Edit Clip"), this);
1635     collection.addAction("edit_clip", openClip);
1636     openClip->setData("edit_clip");
1637     connect(openClip , SIGNAL(triggered()), m_projectList, SLOT(slotOpenClip()));
1638     openClip->setEnabled(false);
1639
1640     QAction *deleteClip = new KAction(KIcon("edit-delete"), i18n("Delete Clip"), this);
1641     collection.addAction("delete_clip", deleteClip);
1642     deleteClip->setData("delete_clip");
1643     connect(deleteClip , SIGNAL(triggered()), m_projectList, SLOT(slotRemoveClip()));
1644     deleteClip->setEnabled(false);
1645
1646     QAction *reloadClip = new KAction(KIcon("view-refresh"), i18n("Reload Clip"), this);
1647     collection.addAction("reload_clip", reloadClip);
1648     reloadClip->setData("reload_clip");
1649     connect(reloadClip , SIGNAL(triggered()), m_projectList, SLOT(slotReloadClip()));
1650     reloadClip->setEnabled(false);
1651
1652     QAction *proxyClip = new KAction(i18n("Proxy Clip"), this);
1653     collection.addAction("proxy_clip", proxyClip);
1654     proxyClip->setData("proxy_clip");
1655     proxyClip->setCheckable(true);
1656     proxyClip->setChecked(false);
1657     connect(proxyClip, SIGNAL(toggled(bool)), m_projectList, SLOT(slotProxyCurrentItem(bool)));
1658     
1659     QAction *stopMotion = new KAction(KIcon("image-x-generic"), i18n("Stop Motion Capture"), this);
1660     collection.addAction("stopmotion", stopMotion);
1661     connect(stopMotion , SIGNAL(triggered()), this, SLOT(slotOpenStopmotion()));
1662
1663     QMenu *addClips = new QMenu();
1664     addClips->addAction(addClip);
1665     addClips->addAction(addColorClip);
1666     addClips->addAction(addSlideClip);
1667     addClips->addAction(addTitleClip);
1668     addClips->addAction(addTitleTemplateClip);
1669     addClips->addAction(addFolderButton);
1670
1671     addClips->addAction(reloadClip);
1672     addClips->addAction(proxyClip);
1673     addClips->addAction(clipProperties);
1674     addClips->addAction(openClip);
1675     addClips->addAction(deleteClip);
1676     m_projectList->setupMenu(addClips, addClip);
1677
1678     // Setup effects and transitions actions.
1679     m_effectsActionCollection = new KActionCollection(this, KGlobal::mainComponent());
1680     //KActionCategory *transitionActions = new KActionCategory(i18n("Transitions"), m_effectsActionCollection);
1681     KActionCategory *transitionActions = new KActionCategory(i18n("Transitions"), collection);
1682     m_transitions = new KAction*[transitions.count()];
1683     for (int i = 0; i < transitions.count(); i++) {
1684         QStringList effectInfo = transitions.effectIdInfo(i);
1685         m_transitions[i] = new KAction(effectInfo.at(0), this);
1686         m_transitions[i]->setData(effectInfo);
1687         m_transitions[i]->setIconVisibleInMenu(false);
1688         QString id = effectInfo.at(2);
1689         if (id.isEmpty()) id = effectInfo.at(1);
1690         transitionActions->addAction("transition_" + id, m_transitions[i]);
1691     }
1692     //m_effectsActionCollection->readSettings();
1693
1694 }
1695
1696 void MainWindow::slotDisplayActionMessage(QAction *a)
1697 {
1698     statusBar()->showMessage(a->data().toString(), 3000);
1699 }
1700
1701 void MainWindow::loadLayouts()
1702 {
1703     QMenu *saveLayout = (QMenu*)(factory()->container("layout_save_as", this));
1704     if (m_loadLayout == NULL || saveLayout == NULL) return;
1705     KSharedConfigPtr config = KGlobal::config();
1706     KConfigGroup layoutGroup(config, "Layouts");
1707     QStringList entries = layoutGroup.keyList();
1708     QList<QAction *> loadActions = m_loadLayout->actions();
1709     QList<QAction *> saveActions = saveLayout->actions();
1710     for (int i = 1; i < 5; i++) {
1711         // Rename the layouts actions
1712         foreach(const QString & key, entries) {
1713             if (key.endsWith(QString("_%1").arg(i))) {
1714                 // Found previously saved layout
1715                 QString layoutName = key.section("_", 0, -2);
1716                 for (int j = 0; j < loadActions.count(); j++) {
1717                     if (loadActions.at(j)->data().toString().endsWith("_" + QString::number(i))) {
1718                         loadActions[j]->setText(layoutName);
1719                         loadActions[j]->setData(key);
1720                         break;
1721                     }
1722                 }
1723                 for (int j = 0; j < saveActions.count(); j++) {
1724                     if (saveActions.at(j)->data().toString().endsWith("_" + QString::number(i))) {
1725                         saveActions[j]->setText(i18n("Save as %1").arg(layoutName));
1726                         saveActions[j]->setData(key);
1727                         break;
1728                     }
1729                 }
1730             }
1731         }
1732     }
1733 }
1734
1735 void MainWindow::slotLoadLayout(QAction *action)
1736 {
1737     if (!action) return;
1738     QString layoutId = action->data().toString();
1739     if (layoutId.isEmpty()) return;
1740     KSharedConfigPtr config = KGlobal::config();
1741     KConfigGroup layouts(config, "Layouts");
1742     QByteArray state = QByteArray::fromBase64(layouts.readEntry(layoutId).toAscii());
1743     restoreState(state);
1744 }
1745
1746 void MainWindow::slotSaveLayout(QAction *action)
1747 {
1748     QString originallayoutName = action->data().toString();
1749     int layoutId = originallayoutName.section('_', -1).toInt();
1750
1751     QString layoutName = QInputDialog::getText(this, i18n("Save Layout"), i18n("Layout name:"), QLineEdit::Normal, originallayoutName.section('_', 0, -2));
1752     if (layoutName.isEmpty()) return;
1753     KSharedConfigPtr config = KGlobal::config();
1754     KConfigGroup layouts(config, "Layouts");
1755     layouts.deleteEntry(originallayoutName);
1756
1757     QByteArray st = saveState();
1758     layoutName.append("_" + QString::number(layoutId));
1759     layouts.writeEntry(layoutName, st.toBase64());
1760     loadLayouts();
1761 }
1762
1763 void MainWindow::saveOptions()
1764 {
1765     KdenliveSettings::self()->writeConfig();
1766     KSharedConfigPtr config = KGlobal::config();
1767     m_fileOpenRecent->saveEntries(KConfigGroup(config, "Recent Files"));
1768     KConfigGroup treecolumns(config, "Project Tree");
1769     treecolumns.writeEntry("columns", m_projectList->headerInfo());
1770     config->sync();
1771 }
1772
1773 void MainWindow::readOptions()
1774 {
1775     KSharedConfigPtr config = KGlobal::config();
1776     m_fileOpenRecent->loadEntries(KConfigGroup(config, "Recent Files"));
1777     KConfigGroup initialGroup(config, "version");
1778     bool upgrade = false;
1779     if (initialGroup.exists()) {
1780         if (initialGroup.readEntry("version", QString()).section(' ', 0, 0) != QString(version).section(' ', 0, 0)) {
1781             upgrade = true;
1782         }
1783
1784         if (initialGroup.readEntry("version") == "0.7") {
1785             //Add new settings from 0.7.1
1786             if (KdenliveSettings::defaultprojectfolder().isEmpty()) {
1787                 QString path = QDir::homePath() + "/kdenlive";
1788                 if (KStandardDirs::makeDir(path)  == false) {
1789                     kDebug() << "/// ERROR CREATING PROJECT FOLDER: " << path;
1790                 } else KdenliveSettings::setDefaultprojectfolder(path);
1791             }
1792         }
1793
1794     }
1795
1796     if (!initialGroup.exists() || upgrade) {
1797         // this is our first run, show Wizard
1798         Wizard *w = new Wizard(upgrade, this);
1799         if (w->exec() == QDialog::Accepted && w->isOk()) {
1800             w->adjustSettings();
1801             initialGroup.writeEntry("version", version);
1802             delete w;
1803         } else {
1804             ::exit(1);
1805         }
1806     }
1807     KConfigGroup treecolumns(config, "Project Tree");
1808     const QByteArray state = treecolumns.readEntry("columns", QByteArray());
1809     if (!state.isEmpty())
1810         m_projectList->setHeaderInfo(state);
1811 }
1812
1813 void MainWindow::slotRunWizard()
1814 {
1815     Wizard *w = new Wizard(false, this);
1816     if (w->exec() == QDialog::Accepted && w->isOk()) {
1817         w->adjustSettings();
1818     }
1819     delete w;
1820 }
1821
1822 void MainWindow::newFile(bool showProjectSettings, bool force)
1823 {
1824     if (!m_timelineArea->isEnabled() && !force)
1825         return;
1826     m_fileRevert->setEnabled(false);
1827     QString profileName = KdenliveSettings::default_profile();
1828     KUrl projectFolder = KdenliveSettings::defaultprojectfolder();
1829     QMap <QString, QString> documentProperties;
1830     QPoint projectTracks(KdenliveSettings::videotracks(), KdenliveSettings::audiotracks());
1831     if (!showProjectSettings) {
1832         if (!KdenliveSettings::activatetabs())
1833             if (!closeCurrentDocument())
1834                 return;
1835     } else {
1836         ProjectSettings *w = new ProjectSettings(NULL, QStringList(), projectTracks.x(), projectTracks.y(), KdenliveSettings::defaultprojectfolder(), false, true, this);
1837         if (w->exec() != QDialog::Accepted)
1838             return;
1839         if (!KdenliveSettings::activatetabs())
1840             if (!closeCurrentDocument())
1841                 return;
1842         if (KdenliveSettings::videothumbnails() != w->enableVideoThumbs())
1843             slotSwitchVideoThumbs();
1844         if (KdenliveSettings::audiothumbnails() != w->enableAudioThumbs())
1845             slotSwitchAudioThumbs();
1846         profileName = w->selectedProfile();
1847         projectFolder = w->selectedFolder();
1848         projectTracks = w->tracks();
1849         documentProperties.insert("enableproxy", QString::number((int) w->useProxy()));
1850         documentProperties.insert("generateproxy", QString::number((int) w->generateProxy()));
1851         documentProperties.insert("proxyminsize", QString::number(w->proxyMinSize()));
1852         documentProperties.insert("proxyparams", w->proxyParams());
1853         documentProperties.insert("proxyextension", w->proxyExtension());
1854         documentProperties.insert("generateimageproxy", QString::number((int) w->generateImageProxy()));
1855         documentProperties.insert("proxyimageminsize", QString::number(w->proxyImageMinSize()));
1856         delete w;
1857     }
1858     m_timelineArea->setEnabled(true);
1859     m_projectList->setEnabled(true);
1860     bool openBackup;
1861     KdenliveDoc *doc = new KdenliveDoc(KUrl(), projectFolder, m_commandStack, profileName, documentProperties, projectTracks, m_projectMonitor->render, m_notesWidget, &openBackup, this);
1862     doc->m_autosave = new KAutoSaveFile(KUrl(), doc);
1863     bool ok;
1864     TrackView *trackView = new TrackView(doc, &ok, this);
1865     m_timelineArea->addTab(trackView, KIcon("kdenlive"), doc->description());
1866     if (!ok) {
1867         // MLT is broken
1868         //m_timelineArea->setEnabled(false);
1869         //m_projectList->setEnabled(false);
1870         slotPreferences(6);
1871         return;
1872     }
1873     if (m_timelineArea->count() == 1) {
1874         connectDocumentInfo(doc);
1875         connectDocument(trackView, doc);
1876     } else
1877         m_timelineArea->setTabBarHidden(false);
1878     m_monitorManager->activateMonitor("clip");
1879     m_closeAction->setEnabled(m_timelineArea->count() > 1);
1880 }
1881
1882 void MainWindow::activateDocument()
1883 {
1884     if (m_timelineArea->currentWidget() == NULL || !m_timelineArea->isEnabled()) return;
1885     TrackView *currentTab = (TrackView *) m_timelineArea->currentWidget();
1886     KdenliveDoc *currentDoc = currentTab->document();
1887     connectDocumentInfo(currentDoc);
1888     connectDocument(currentTab, currentDoc);
1889 }
1890
1891 bool MainWindow::closeCurrentDocument(bool saveChanges)
1892 {
1893     QWidget *w = m_timelineArea->currentWidget();
1894     if (!w) return true;
1895     // closing current document
1896     int ix = m_timelineArea->currentIndex() + 1;
1897     if (ix == m_timelineArea->count()) ix = 0;
1898     m_timelineArea->setCurrentIndex(ix);
1899     TrackView *tabToClose = (TrackView *) w;
1900     KdenliveDoc *docToClose = tabToClose->document();
1901     if (docToClose && docToClose->isModified() && saveChanges) {
1902         QString message;
1903         if (m_activeDocument->url().fileName().isEmpty())
1904             message = i18n("Save changes to document?");
1905         else
1906             message = i18n("The project <b>\"%1\"</b> has been changed.\nDo you want to save your changes?").arg(m_activeDocument->url().fileName());
1907         switch (KMessageBox::warningYesNoCancel(this, message)) {
1908         case KMessageBox::Yes :
1909             // save document here. If saving fails, return false;
1910             if (saveFile() == false) return false;
1911             break;
1912         case KMessageBox::Cancel :
1913             return false;
1914             break;
1915         default:
1916             break;
1917         }
1918     }
1919     m_clipMonitor->slotSetClipProducer(NULL);
1920     m_projectList->slotResetProjectList();
1921     m_timelineArea->removeTab(m_timelineArea->indexOf(w));
1922     if (m_timelineArea->count() == 1) {
1923         m_timelineArea->setTabBarHidden(true);
1924         m_closeAction->setEnabled(false);
1925     }
1926     if (docToClose == m_activeDocument) {
1927         delete m_activeDocument;
1928         m_activeDocument = NULL;
1929         m_effectStack->clear();
1930         m_transitionConfig->slotTransitionItemSelected(NULL, 0, QPoint(), false);
1931     } else {
1932         delete docToClose;
1933     }
1934     if (w == m_activeTimeline) {
1935         delete m_activeTimeline;
1936         m_activeTimeline = NULL;
1937     } else {
1938         delete w;
1939     }
1940     return true;
1941 }
1942
1943 bool MainWindow::saveFileAs(const QString &outputFileName)
1944 {
1945     QString currentSceneList;
1946     m_monitorManager->stopActiveMonitor();
1947
1948     if (m_activeDocument->saveSceneList(outputFileName, m_projectMonitor->sceneList(), m_projectList->expandedFolders()) == false)
1949         return false;
1950
1951     // Save timeline thumbnails
1952     m_activeTimeline->projectView()->saveThumbnails();
1953     m_activeDocument->setUrl(KUrl(outputFileName));
1954     if (m_activeDocument->m_autosave == NULL) {
1955         m_activeDocument->m_autosave = new KAutoSaveFile(KUrl(outputFileName), this);
1956     } else m_activeDocument->m_autosave->setManagedFile(KUrl(outputFileName));
1957     setCaption(m_activeDocument->description());
1958     m_timelineArea->setTabText(m_timelineArea->currentIndex(), m_activeDocument->description());
1959     m_timelineArea->setTabToolTip(m_timelineArea->currentIndex(), m_activeDocument->url().path());
1960     m_activeDocument->setModified(false);
1961     m_fileOpenRecent->addUrl(KUrl(outputFileName));
1962     m_fileRevert->setEnabled(true);
1963     m_undoView->stack()->setClean();
1964     return true;
1965 }
1966
1967 bool MainWindow::saveFileAs()
1968 {
1969     QString outputFile = KFileDialog::getSaveFileName(m_activeDocument->projectFolder(), getMimeType(false));
1970     if (outputFile.isEmpty()) {
1971         return false;
1972     }
1973     if (QFile::exists(outputFile)) {
1974         // Show the file dialog again if the user does not want to overwrite the file
1975         if (KMessageBox::questionYesNo(this, i18n("File %1 already exists.\nDo you want to overwrite it?", outputFile)) == KMessageBox::No)
1976             return saveFileAs();
1977     }
1978     return saveFileAs(outputFile);
1979 }
1980
1981 bool MainWindow::saveFile()
1982 {
1983     if (!m_activeDocument) return true;
1984     if (m_activeDocument->url().isEmpty()) {
1985         return saveFileAs();
1986     } else {
1987         bool result = saveFileAs(m_activeDocument->url().path());
1988         m_activeDocument->m_autosave->resize(0);
1989         return result;
1990     }
1991 }
1992
1993 void MainWindow::openFile()
1994 {
1995     if (!m_startUrl.isEmpty()) {
1996         openFile(m_startUrl);
1997         m_startUrl = KUrl();
1998         return;
1999     }
2000     KUrl url = KFileDialog::getOpenUrl(KUrl("kfiledialog:///projectfolder"), getMimeType());
2001     if (url.isEmpty()) return;
2002     m_fileOpenRecent->addUrl(url);
2003     openFile(url);
2004 }
2005
2006 void MainWindow::openLastFile()
2007 {
2008     KSharedConfigPtr config = KGlobal::config();
2009     KUrl::List urls = m_fileOpenRecent->urls();
2010     //WARNING: this is buggy, we get a random url, not the last one. Bug in KRecentFileAction?
2011     if (urls.isEmpty()) newFile(false);
2012     else openFile(urls.last());
2013 }
2014
2015 void MainWindow::openFile(const KUrl &url)
2016 {
2017     // Make sure the url is a Kdenlive project file
2018     KMimeType::Ptr mime = KMimeType::findByUrl(url);
2019     if (mime.data()->is("application/x-compressed-tar")) {
2020         // Opening a compressed project file, we need to process it
2021         kDebug()<<"Opening archive, processing";
2022         ArchiveWidget *ar = new ArchiveWidget(url);
2023         if (ar->exec() == QDialog::Accepted) openFile(KUrl(ar->extractedProjectFile()));
2024         delete ar;
2025         return;
2026     }
2027     if (!url.fileName().endsWith(".kdenlive")) {
2028         // This is not a Kdenlive project file, abort loading
2029         KMessageBox::sorry(this, i18n("File %1 is not a Kdenlive project file", url.path()));
2030         return;
2031     }
2032     
2033     // Check if the document is already opened
2034     const int ct = m_timelineArea->count();
2035     bool isOpened = false;
2036     int i;
2037     for (i = 0; i < ct; i++) {
2038         TrackView *tab = (TrackView *) m_timelineArea->widget(i);
2039         KdenliveDoc *doc = tab->document();
2040         if (doc->url() == url) {
2041             isOpened = true;
2042             break;
2043         }
2044     }
2045     if (isOpened) {
2046         m_timelineArea->setCurrentIndex(i);
2047         return;
2048     }
2049
2050     if (!KdenliveSettings::activatetabs()) if (!closeCurrentDocument()) return;
2051
2052     // Check for backup file
2053     QList<KAutoSaveFile *> staleFiles = KAutoSaveFile::staleFiles(url);
2054     if (!staleFiles.isEmpty()) {
2055         if (KMessageBox::questionYesNo(this,
2056                                        i18n("Auto-saved files exist. Do you want to recover them now?"),
2057                                        i18n("File Recovery"),
2058                                        KGuiItem(i18n("Recover")), KGuiItem(i18n("Don't recover"))) == KMessageBox::Yes) {
2059             recoverFiles(staleFiles);
2060             return;
2061         } else {
2062             // remove the stale files
2063             foreach(KAutoSaveFile * stale, staleFiles) {
2064                 stale->open(QIODevice::ReadWrite);
2065                 delete stale;
2066             }
2067         }
2068     }
2069     m_messageLabel->setMessage(i18n("Opening file %1", url.path()), InformationMessage);
2070     m_messageLabel->repaint();
2071     doOpenFile(url, NULL);
2072 }
2073
2074 void MainWindow::doOpenFile(const KUrl &url, KAutoSaveFile *stale)
2075 {
2076     if (!m_timelineArea->isEnabled()) return;
2077     m_fileRevert->setEnabled(true);
2078
2079     // Recreate stopmotion widget on document change
2080     if (m_stopmotion) {
2081         delete m_stopmotion;
2082         m_stopmotion = NULL;
2083     }
2084     
2085     m_timer.start();
2086     KProgressDialog progressDialog(this, i18n("Loading project"), i18n("Loading project"));
2087     progressDialog.setAllowCancel(false);
2088     progressDialog.progressBar()->setMaximum(4);
2089     progressDialog.show();
2090     progressDialog.progressBar()->setValue(0);
2091     qApp->processEvents();
2092
2093     bool openBackup;
2094     KdenliveDoc *doc = new KdenliveDoc(url, KdenliveSettings::defaultprojectfolder(), m_commandStack, KdenliveSettings::default_profile(), QMap <QString, QString> (), QPoint(KdenliveSettings::videotracks(), KdenliveSettings::audiotracks()), m_projectMonitor->render, m_notesWidget, &openBackup, this, &progressDialog);
2095
2096     progressDialog.progressBar()->setValue(1);
2097     progressDialog.progressBar()->setMaximum(4);
2098     progressDialog.setLabelText(i18n("Loading project"));
2099     qApp->processEvents();
2100
2101     if (stale == NULL) {
2102         stale = new KAutoSaveFile(url, doc);
2103         doc->m_autosave = stale;
2104     } else {
2105         doc->m_autosave = stale;
2106         doc->setUrl(stale->managedFile());
2107         doc->setModified(true);
2108         stale->setParent(doc);
2109     }
2110     connectDocumentInfo(doc);
2111
2112     progressDialog.progressBar()->setValue(2);
2113     qApp->processEvents();
2114
2115     bool ok;
2116     TrackView *trackView = new TrackView(doc, &ok, this);
2117     connectDocument(trackView, doc);
2118     progressDialog.progressBar()->setValue(3);
2119     qApp->processEvents();
2120
2121     m_timelineArea->setCurrentIndex(m_timelineArea->addTab(trackView, KIcon("kdenlive"), doc->description()));
2122     if (!ok) {
2123         m_timelineArea->setEnabled(false);
2124         m_projectList->setEnabled(false);
2125         KMessageBox::sorry(this, i18n("Cannot open file %1.\nProject is corrupted.", url.path()));
2126         slotGotProgressInfo(QString(), -1);
2127         newFile(false, true);
2128         return;
2129     }
2130     m_timelineArea->setTabToolTip(m_timelineArea->currentIndex(), doc->url().path());
2131     trackView->setDuration(trackView->duration());
2132     trackView->projectView()->initCursorPos(m_projectMonitor->render->seekPosition().frames(doc->fps()));
2133
2134     if (m_timelineArea->count() > 1) m_timelineArea->setTabBarHidden(false);
2135     slotGotProgressInfo(QString(), -1);
2136     m_projectMonitor->adjustRulerSize(trackView->duration());
2137     m_projectMonitor->slotZoneMoved(trackView->inPoint(), trackView->outPoint());
2138     progressDialog.progressBar()->setValue(4);
2139     if (openBackup) slotOpenBackupDialog(url);
2140 }
2141
2142 void MainWindow::recoverFiles(QList<KAutoSaveFile *> staleFiles)
2143 {
2144     foreach(KAutoSaveFile * stale, staleFiles) {
2145         /*if (!stale->open(QIODevice::QIODevice::ReadOnly)) {
2146                   // show an error message; we could not steal the lockfile
2147                   // maybe another application got to the file before us?
2148                   delete stale;
2149                   continue;
2150         }*/
2151         kDebug() << "// OPENING RECOVERY: " << stale->fileName() << "\nMANAGED: " << stale->managedFile().path();
2152         // the stalefiles also contain ".lock" files so we must ignore them... bug in KAutoSaveFile?
2153         if (!stale->fileName().endsWith(".lock")) doOpenFile(KUrl(stale->fileName()), stale);
2154         else KIO::NetAccess::del(KUrl(stale->fileName()), this);
2155     }
2156 }
2157
2158 void MainWindow::parseProfiles(const QString &mltPath)
2159 {
2160     //KdenliveSettings::setDefaulttmpfolder();
2161     if (!mltPath.isEmpty()) {
2162         KdenliveSettings::setMltpath(mltPath + "/share/mlt/profiles/");
2163         KdenliveSettings::setRendererpath(mltPath + "/bin/melt");
2164     }
2165
2166     if (KdenliveSettings::mltpath().isEmpty())
2167         KdenliveSettings::setMltpath(QString(MLT_PREFIX) + QString("/share/mlt/profiles/"));
2168
2169     if (KdenliveSettings::rendererpath().isEmpty() || KdenliveSettings::rendererpath().endsWith("inigo")) {
2170         QString meltPath = QString(MLT_PREFIX) + QString("/bin/melt");
2171         if (!QFile::exists(meltPath))
2172             meltPath = KStandardDirs::findExe("melt");
2173         KdenliveSettings::setRendererpath(meltPath);
2174     }
2175
2176     if (KdenliveSettings::rendererpath().isEmpty()) {
2177         // Cannot find the MLT melt renderer, ask for location
2178         KUrlRequesterDialog *getUrl = new KUrlRequesterDialog(QString(), i18n("Cannot find the melt program required for rendering (part of MLT)"), this);
2179         if (getUrl->exec() == QDialog::Rejected) {
2180             ::exit(0);
2181         }
2182         KUrl rendererPath = getUrl->selectedUrl();
2183         delete getUrl;
2184         if (rendererPath.isEmpty()) ::exit(0);
2185         KdenliveSettings::setRendererpath(rendererPath.path());
2186     }
2187
2188     QStringList profilesFilter;
2189     profilesFilter << "*";
2190     QStringList profilesList = QDir(KdenliveSettings::mltpath()).entryList(profilesFilter, QDir::Files);
2191     if (profilesList.isEmpty()) {
2192         // Cannot find MLT path, try finding melt
2193         QString profilePath = KdenliveSettings::rendererpath();
2194         if (!profilePath.isEmpty()) {
2195             profilePath = profilePath.section('/', 0, -3);
2196             KdenliveSettings::setMltpath(profilePath + "/share/mlt/profiles/");
2197             profilesList = QDir(KdenliveSettings::mltpath()).entryList(profilesFilter, QDir::Files);
2198         }
2199         if (profilesList.isEmpty()) {
2200             // Cannot find the MLT profiles, ask for location
2201             KUrlRequesterDialog *getUrl = new KUrlRequesterDialog(KdenliveSettings::mltpath(), i18n("Cannot find your MLT profiles, please give the path"), this);
2202             getUrl->fileDialog()->setMode(KFile::Directory);
2203             if (getUrl->exec() == QDialog::Rejected) {
2204                 ::exit(0);
2205             }
2206             KUrl mltPath = getUrl->selectedUrl();
2207             delete getUrl;
2208             if (mltPath.isEmpty()) ::exit(0);
2209             KdenliveSettings::setMltpath(mltPath.path(KUrl::AddTrailingSlash));
2210             profilesList = QDir(KdenliveSettings::mltpath()).entryList(profilesFilter, QDir::Files);
2211         }
2212     }
2213
2214     kDebug() << "RESULTING MLT PATH: " << KdenliveSettings::mltpath();
2215
2216     // Parse again MLT profiles to build a list of available video formats
2217     if (profilesList.isEmpty()) parseProfiles();
2218 }
2219
2220
2221 void MainWindow::slotEditProfiles()
2222 {
2223     ProfilesDialog *w = new ProfilesDialog;
2224     if (w->exec() == QDialog::Accepted) {
2225         KdenliveSettingsDialog* d = static_cast <KdenliveSettingsDialog*>(KConfigDialog::exists("settings"));
2226         if (d) d->checkProfile();
2227     }
2228     delete w;
2229 }
2230
2231 void MainWindow::slotDetectAudioDriver()
2232 {
2233     /* WARNING: do not use this method because sometimes detects wrong driver (pulse instead of alsa),
2234     leading to no audio output, see bug #934 */
2235
2236     //decide which audio driver is really best, in some cases SDL is wrong
2237     if (KdenliveSettings::audiodrivername().isEmpty()) {
2238         QString driver;
2239         KProcess readProcess;
2240         //PulseAudio needs to be selected if it exists, the ALSA pulse pcm device is not fast enough.
2241         if (!KStandardDirs::findExe("pactl").isEmpty()) {
2242             readProcess.setOutputChannelMode(KProcess::OnlyStdoutChannel);
2243             readProcess.setProgram("pactl", QStringList() << "stat");
2244             readProcess.execute(2000); // Kill it after 2 seconds
2245
2246             QString result = QString(readProcess.readAllStandardOutput());
2247             kDebug() << "// / / / / / READING PACTL: ";
2248             kDebug() << result;
2249             if (!result.isEmpty()) {
2250                 driver = "pulse";
2251                 kDebug() << "// / / / / PULSEAUDIO DETECTED";
2252             }
2253         }
2254         //put others here
2255         KdenliveSettings::setAutoaudiodrivername(driver);
2256     }
2257 }
2258
2259 void MainWindow::slotEditProjectSettings()
2260 {
2261     QPoint p = m_activeDocument->getTracksCount();
2262     ProjectSettings *w = new ProjectSettings(m_projectList, m_activeTimeline->projectView()->extractTransitionsLumas(), p.x(), p.y(), m_activeDocument->projectFolder().path(), true, !m_activeDocument->isModified(), this);
2263     connect(w, SIGNAL(disableProxies()), this, SLOT(slotDisableProxies()));
2264
2265     if (w->exec() == QDialog::Accepted) {
2266         QString profile = w->selectedProfile();
2267         m_activeDocument->setProjectFolder(w->selectedFolder());
2268 #ifndef Q_WS_MAC
2269         m_recMonitor->slotUpdateCaptureFolder(m_activeDocument->projectFolder().path(KUrl::AddTrailingSlash));
2270 #endif
2271         if (m_renderWidget) m_renderWidget->setDocumentPath(m_activeDocument->projectFolder().path(KUrl::AddTrailingSlash));
2272         if (KdenliveSettings::videothumbnails() != w->enableVideoThumbs()) slotSwitchVideoThumbs();
2273         if (KdenliveSettings::audiothumbnails() != w->enableAudioThumbs()) slotSwitchAudioThumbs();
2274         if (m_activeDocument->profilePath() != profile) slotUpdateProjectProfile(profile);
2275         if (m_activeDocument->getDocumentProperty("proxyparams") != w->proxyParams()) {
2276             m_activeDocument->setModified();
2277             m_activeDocument->setDocumentProperty("proxyparams", w->proxyParams());
2278             if (m_activeDocument->clipManager()->clipsCount() > 0 && KMessageBox::questionYesNo(this, i18n("You have changed the proxy parameters. Do you want to recreate all proxy clips for this project?")) == KMessageBox::Yes) {
2279                 //TODO: rebuild all proxies
2280                 //m_projectList->rebuildProxies();
2281             }
2282         }
2283         if (m_activeDocument->getDocumentProperty("proxyextension") != w->proxyExtension()) {
2284             m_activeDocument->setModified();
2285             m_activeDocument->setDocumentProperty("proxyextension", w->proxyExtension());
2286         }
2287         if (m_activeDocument->getDocumentProperty("generateproxy") != QString::number((int) w->generateProxy())) {
2288             m_activeDocument->setModified();
2289             m_activeDocument->setDocumentProperty("generateproxy", QString::number((int) w->generateProxy()));
2290         }
2291         if (m_activeDocument->getDocumentProperty("proxyminsize") != QString::number(w->proxyMinSize())) {
2292             m_activeDocument->setModified();
2293             m_activeDocument->setDocumentProperty("proxyminsize", QString::number(w->proxyMinSize()));
2294         }
2295         if (m_activeDocument->getDocumentProperty("generateimageproxy") != QString::number((int) w->generateImageProxy())) {
2296             m_activeDocument->setModified();
2297             m_activeDocument->setDocumentProperty("generateimageproxy", QString::number((int) w->generateImageProxy()));
2298         }
2299         if (m_activeDocument->getDocumentProperty("proxyimageminsize") != QString::number(w->proxyImageMinSize())) {
2300             m_activeDocument->setModified();
2301             m_activeDocument->setDocumentProperty("proxyimageminsize", QString::number(w->proxyImageMinSize()));
2302         }
2303         if (QString::number((int) w->useProxy()) != m_activeDocument->getDocumentProperty("enableproxy")) {
2304             m_activeDocument->setDocumentProperty("enableproxy", QString::number((int) w->useProxy()));
2305             m_activeDocument->setModified();
2306             slotUpdateProxySettings();
2307         }
2308     }
2309     delete w;
2310 }
2311
2312 void MainWindow::slotDisableProxies()
2313 {
2314     m_activeDocument->setDocumentProperty("enableproxy", QString::number((int) false));
2315     m_activeDocument->setModified();
2316     slotUpdateProxySettings();
2317 }
2318
2319 void MainWindow::slotUpdateProjectProfile(const QString &profile)
2320 {
2321     // Recreate the stopmotion widget if profile changes
2322     if (m_stopmotion) {
2323         delete m_stopmotion;
2324         m_stopmotion = NULL;
2325     }
2326
2327     // Deselect current effect / transition
2328     m_effectStack->slotClipItemSelected(NULL, 0);
2329     m_transitionConfig->slotTransitionItemSelected(NULL, 0, QPoint(), false);
2330     m_clipMonitor->slotSetClipProducer(NULL);
2331     bool updateFps = m_activeDocument->setProfilePath(profile);
2332     KdenliveSettings::setCurrent_profile(profile);
2333     KdenliveSettings::setProject_fps(m_activeDocument->fps());
2334     setCaption(m_activeDocument->description(), m_activeDocument->isModified());
2335
2336     m_activeDocument->clipManager()->clearUnusedProducers();
2337     m_monitorManager->resetProfiles(m_activeDocument->timecode());
2338     m_transitionConfig->updateProjectFormat(m_activeDocument->mltProfile(), m_activeDocument->timecode(), m_activeDocument->tracksList());
2339     m_effectStack->updateProjectFormat(m_activeDocument->mltProfile(), m_activeDocument->timecode());
2340     m_projectList->updateProjectFormat(m_activeDocument->timecode());
2341     if (m_renderWidget) m_renderWidget->setProfile(m_activeDocument->mltProfile());
2342     m_timelineArea->setTabText(m_timelineArea->currentIndex(), m_activeDocument->description());
2343     if (updateFps) m_activeTimeline->updateProjectFps();
2344     m_activeDocument->clipManager()->clearCache();
2345     m_activeTimeline->updateProfile();
2346     m_activeDocument->setModified(true);
2347     m_commandStack->activeStack()->clear();
2348     //Update the mouse position display so it will display in DF/NDF format by default based on the project setting.
2349     slotUpdateMousePosition(0);
2350     // We need to desactivate & reactivate monitors to get a refresh
2351     //m_monitorManager->switchMonitors();
2352 }
2353
2354
2355 void MainWindow::slotRenderProject()
2356 {
2357     if (!m_renderWidget) {
2358         QString projectfolder = m_activeDocument ? m_activeDocument->projectFolder().path(KUrl::AddTrailingSlash) : KdenliveSettings::defaultprojectfolder();
2359         MltVideoProfile profile;
2360         if (m_activeDocument) profile = m_activeDocument->mltProfile();
2361         m_renderWidget = new RenderWidget(projectfolder, m_projectList->useProxy(), profile, this);
2362         connect(m_renderWidget, SIGNAL(shutdown()), this, SLOT(slotShutdown()));
2363         connect(m_renderWidget, SIGNAL(selectedRenderProfile(QMap <QString, QString>)), this, SLOT(slotSetDocumentRenderProfile(QMap <QString, QString>)));
2364         connect(m_renderWidget, SIGNAL(prepareRenderingData(bool, bool, const QString&)), this, SLOT(slotPrepareRendering(bool, bool, const QString&)));
2365         connect(m_renderWidget, SIGNAL(abortProcess(const QString &)), this, SIGNAL(abortRenderJob(const QString &)));
2366         connect(m_renderWidget, SIGNAL(openDvdWizard(const QString &, const QString &)), this, SLOT(slotDvdWizard(const QString &, const QString &)));
2367         if (m_activeDocument) {
2368             m_renderWidget->setProfile(m_activeDocument->mltProfile());
2369             m_renderWidget->setGuides(m_activeDocument->guidesXml(), m_activeDocument->projectDuration());
2370             m_renderWidget->setDocumentPath(m_activeDocument->projectFolder().path(KUrl::AddTrailingSlash));
2371             m_renderWidget->setRenderProfile(m_activeDocument->getRenderProperties());
2372         }
2373     }
2374     slotCheckRenderStatus();
2375     m_renderWidget->show();
2376     m_renderWidget->showNormal();
2377
2378     // What are the following lines supposed to do?
2379     //m_activeTimeline->tracksNumber();
2380     //m_renderWidget->enableAudio(false);
2381     //m_renderWidget->export_audio;
2382 }
2383
2384 void MainWindow::slotCheckRenderStatus()
2385 {
2386     // Make sure there are no missing clips
2387     if (m_renderWidget)
2388         m_renderWidget->missingClips(m_projectList->hasMissingClips());
2389 }
2390
2391 void MainWindow::setRenderingProgress(const QString &url, int progress)
2392 {
2393     if (m_renderWidget)
2394         m_renderWidget->setRenderJob(url, progress);
2395 }
2396
2397 void MainWindow::setRenderingFinished(const QString &url, int status, const QString &error)
2398 {
2399     if (m_renderWidget)
2400         m_renderWidget->setRenderStatus(url, status, error);
2401 }
2402
2403 void MainWindow::slotCleanProject()
2404 {
2405     if (KMessageBox::warningContinueCancel(this, i18n("This will remove all unused clips from your project."), i18n("Clean up project")) == KMessageBox::Cancel) return;
2406     m_projectList->cleanup();
2407 }
2408
2409 void MainWindow::slotUpdateMousePosition(int pos)
2410 {
2411     if (m_activeDocument)
2412         switch (m_timecodeFormat->currentIndex()) {
2413         case 0:
2414             statusBar()->changeItem(m_activeDocument->timecode().getTimecodeFromFrames(pos), ID_TIMELINE_POS);
2415             break;
2416         default:
2417             statusBar()->changeItem(QString::number(pos), ID_TIMELINE_POS);
2418         }
2419 }
2420
2421 void MainWindow::slotUpdateDocumentState(bool modified)
2422 {
2423     if (!m_activeDocument) return;
2424     setCaption(m_activeDocument->description(), modified);
2425     m_saveAction->setEnabled(modified);
2426     if (modified) {
2427         m_timelineArea->setTabTextColor(m_timelineArea->currentIndex(), palette().color(QPalette::Link));
2428         m_timelineArea->setTabIcon(m_timelineArea->currentIndex(), KIcon("document-save"));
2429     } else {
2430         m_timelineArea->setTabTextColor(m_timelineArea->currentIndex(), palette().color(QPalette::Text));
2431         m_timelineArea->setTabIcon(m_timelineArea->currentIndex(), KIcon("kdenlive"));
2432     }
2433 }
2434
2435 void MainWindow::connectDocumentInfo(KdenliveDoc *doc)
2436 {
2437     if (m_activeDocument) {
2438         if (m_activeDocument == doc) return;
2439         disconnect(m_activeDocument, SIGNAL(progressInfo(const QString &, int)), this, SLOT(slotGotProgressInfo(const QString &, int)));
2440     }
2441     connect(doc, SIGNAL(progressInfo(const QString &, int)), this, SLOT(slotGotProgressInfo(const QString &, int)));
2442 }
2443
2444 void MainWindow::connectDocument(TrackView *trackView, KdenliveDoc *doc)   //changed
2445 {
2446     //m_projectMonitor->stop();
2447     m_closeAction->setEnabled(m_timelineArea->count() > 1);
2448     kDebug() << "///////////////////   CONNECTING DOC TO PROJECT VIEW ////////////////";
2449     if (m_activeDocument) {
2450         if (m_activeDocument == doc) return;
2451         if (m_activeTimeline) {
2452             disconnect(m_projectMonitor, SIGNAL(renderPosition(int)), m_activeTimeline, SLOT(moveCursorPos(int)));
2453             disconnect(m_projectMonitor, SIGNAL(zoneUpdated(QPoint)), m_activeTimeline, SLOT(slotSetZone(QPoint)));
2454             disconnect(m_projectMonitor, SIGNAL(durationChanged(int)), m_activeTimeline, SLOT(setDuration(int)));
2455             disconnect(m_projectMonitor, SIGNAL(zoneUpdated(QPoint)), m_activeDocument, SLOT(setModified()));
2456             disconnect(m_notesWidget, SIGNAL(textChanged()), m_activeDocument, SLOT(setModified()));
2457             disconnect(m_clipMonitor, SIGNAL(zoneUpdated(QPoint)), m_activeDocument, SLOT(setModified()));
2458             disconnect(m_projectList, SIGNAL(projectModified()), m_activeDocument, SLOT(setModified()));
2459
2460             disconnect(m_projectMonitor->render, SIGNAL(refreshDocumentProducers(bool, bool)), m_activeDocument, SLOT(checkProjectClips(bool, bool)));
2461
2462             disconnect(m_activeDocument, SIGNAL(guidesUpdated()), this, SLOT(slotGuidesUpdated()));
2463             disconnect(m_activeDocument, SIGNAL(addProjectClip(DocClipBase *, bool)), m_projectList, SLOT(slotAddClip(DocClipBase *, bool)));
2464             disconnect(m_activeDocument, SIGNAL(resetProjectList()), m_projectList, SLOT(slotResetProjectList()));
2465             disconnect(m_activeDocument, SIGNAL(signalDeleteProjectClip(const QString &)), this, SLOT(slotDeleteClip(const QString &)));
2466             disconnect(m_activeDocument, SIGNAL(updateClipDisplay(const QString &)), m_projectList, SLOT(slotUpdateClip(const QString &)));
2467             disconnect(m_activeDocument, SIGNAL(selectLastAddedClip(const QString &)), m_projectList, SLOT(slotSelectClip(const QString &)));
2468             disconnect(m_activeTimeline->projectView(), SIGNAL(clipItemSelected(ClipItem*, int, bool)), m_effectStack, SLOT(slotClipItemSelected(ClipItem*, int)));
2469             disconnect(m_activeTimeline->projectView(), SIGNAL(clipItemSelected(ClipItem*, int, bool)), this, SLOT(slotActivateEffectStackView(ClipItem*, int, bool)));
2470             disconnect(m_activeTimeline->projectView(), SIGNAL(clipItemSelected(ClipItem*, int, bool)), m_projectMonitor, SLOT(slotSetSelectedClip(ClipItem*)));
2471             disconnect(m_activeTimeline->projectView(), SIGNAL(transitionItemSelected(Transition*, int, QPoint, bool)), m_transitionConfig, SLOT(slotTransitionItemSelected(Transition*, int, QPoint, bool)));
2472             disconnect(m_activeTimeline->projectView(), SIGNAL(transitionItemSelected(Transition*, int, QPoint, bool)), this, SLOT(slotActivateTransitionView(Transition *)));
2473             disconnect(m_activeTimeline->projectView(), SIGNAL(transitionItemSelected(Transition*, int, QPoint, bool)), m_projectMonitor, SLOT(slotSetSelectedClip(Transition*)));
2474             disconnect(m_activeTimeline->projectView(), SIGNAL(playMonitor()), m_projectMonitor, SLOT(slotPlay()));
2475             disconnect(m_activeTimeline->projectView(), SIGNAL(displayMessage(const QString&, MessageType)), m_messageLabel, SLOT(setMessage(const QString&, MessageType)));
2476             disconnect(m_activeTimeline->projectView(), SIGNAL(showClipFrame(DocClipBase *, QPoint, const int)), m_clipMonitor, SLOT(slotSetClipProducer(DocClipBase *, QPoint, const int)));
2477             disconnect(m_activeTimeline, SIGNAL(cursorMoved()), m_projectMonitor, SLOT(activateMonitor()));
2478             disconnect(m_activeTimeline, SIGNAL(insertTrack(int)), this, SLOT(slotInsertTrack(int)));
2479             disconnect(m_activeTimeline, SIGNAL(deleteTrack(int)), this, SLOT(slotDeleteTrack(int)));
2480             disconnect(m_activeTimeline, SIGNAL(configTrack(int)), this, SLOT(slotConfigTrack(int)));
2481             disconnect(m_activeDocument, SIGNAL(docModified(bool)), this, SLOT(slotUpdateDocumentState(bool)));
2482             disconnect(m_effectStack, SIGNAL(updateEffect(ClipItem*, int, QDomElement, QDomElement, int)), m_activeTimeline->projectView(), SLOT(slotUpdateClipEffect(ClipItem*, int, QDomElement, QDomElement, int)));
2483             disconnect(m_effectStack, SIGNAL(removeEffect(ClipItem*, int, QDomElement)), m_activeTimeline->projectView(), SLOT(slotDeleteEffect(ClipItem*, int, QDomElement)));
2484             disconnect(m_effectStack, SIGNAL(changeEffectState(ClipItem*, int, int, bool)), m_activeTimeline->projectView(), SLOT(slotChangeEffectState(ClipItem*, int, int, bool)));
2485             disconnect(m_effectStack, SIGNAL(changeEffectPosition(ClipItem*, int, int, int)), m_activeTimeline->projectView(), SLOT(slotChangeEffectPosition(ClipItem*, int, int, int)));
2486             disconnect(m_effectStack, SIGNAL(refreshEffectStack(ClipItem*)), m_activeTimeline->projectView(), SLOT(slotRefreshEffects(ClipItem*)));
2487             disconnect(m_effectStack, SIGNAL(reloadEffects()), this, SLOT(slotReloadEffects()));
2488             disconnect(m_effectStack, SIGNAL(displayMessage(const QString&, int)), this, SLOT(slotGotProgressInfo(const QString&, int)));
2489             disconnect(m_transitionConfig, SIGNAL(transitionUpdated(Transition *, QDomElement)), m_activeTimeline->projectView() , SLOT(slotTransitionUpdated(Transition *, QDomElement)));
2490             disconnect(m_transitionConfig, SIGNAL(seekTimeline(int)), m_activeTimeline->projectView() , SLOT(setCursorPos(int)));
2491             disconnect(m_activeTimeline->projectView(), SIGNAL(activateDocumentMonitor()), m_projectMonitor, SLOT(activateMonitor()));
2492             disconnect(m_activeTimeline, SIGNAL(zoneMoved(int, int)), this, SLOT(slotZoneMoved(int, int)));
2493             disconnect(m_projectList, SIGNAL(loadingIsOver()), m_activeTimeline->projectView(), SLOT(slotUpdateAllThumbs()));
2494             disconnect(m_projectList, SIGNAL(refreshClip(const QString &)), m_activeTimeline->projectView(), SLOT(slotRefreshThumbs(const QString &)));
2495             m_effectStack->clear();
2496         }
2497         //m_activeDocument->setRenderer(NULL);
2498         m_clipMonitor->stop();
2499     }
2500     KdenliveSettings::setCurrent_profile(doc->profilePath());
2501     KdenliveSettings::setProject_fps(doc->fps());
2502     m_monitorManager->resetProfiles(doc->timecode());
2503     m_clipMonitorDock->raise();
2504     m_projectList->setDocument(doc);
2505     m_transitionConfig->updateProjectFormat(doc->mltProfile(), doc->timecode(), doc->tracksList());
2506     m_effectStack->updateProjectFormat(doc->mltProfile(), doc->timecode());
2507     connect(m_projectList, SIGNAL(refreshClip(const QString &, bool)), trackView->projectView(), SLOT(slotRefreshThumbs(const QString &, bool)));
2508
2509     connect(m_projectList, SIGNAL(projectModified()), doc, SLOT(setModified()));
2510     connect(m_projectList, SIGNAL(clipNameChanged(const QString, const QString)), trackView->projectView(), SLOT(clipNameChanged(const QString, const QString)));
2511
2512     //connect(trackView, SIGNAL(cursorMoved()), m_projectMonitor, SLOT(activateMonitor()));
2513     connect(trackView, SIGNAL(insertTrack(int)), this, SLOT(slotInsertTrack(int)));
2514     connect(trackView, SIGNAL(deleteTrack(int)), this, SLOT(slotDeleteTrack(int)));
2515     connect(trackView, SIGNAL(configTrack(int)), this, SLOT(slotConfigTrack(int)));
2516     connect(trackView, SIGNAL(updateTracksInfo()), this, SLOT(slotUpdateTrackInfo()));
2517     connect(trackView, SIGNAL(mousePosition(int)), this, SLOT(slotUpdateMousePosition(int)));
2518     connect(trackView->projectView(), SIGNAL(forceClipProcessing(const QString &)), m_projectList, SLOT(slotForceProcessing(const QString &)));
2519     connect(m_projectMonitor, SIGNAL(renderPosition(int)), trackView, SLOT(moveCursorPos(int)));
2520     connect(m_projectMonitor, SIGNAL(zoneUpdated(QPoint)), trackView, SLOT(slotSetZone(QPoint)));
2521     connect(m_projectMonitor, SIGNAL(zoneUpdated(QPoint)), doc, SLOT(setModified()));
2522     connect(m_clipMonitor, SIGNAL(zoneUpdated(QPoint)), doc, SLOT(setModified()));
2523     connect(m_projectMonitor, SIGNAL(durationChanged(int)), trackView, SLOT(setDuration(int)));
2524     connect(m_projectMonitor->render, SIGNAL(refreshDocumentProducers(bool, bool)), doc, SLOT(checkProjectClips(bool, bool)));
2525
2526     connect(doc, SIGNAL(addProjectClip(DocClipBase *, bool)), m_projectList, SLOT(slotAddClip(DocClipBase *, bool)));
2527     connect(doc, SIGNAL(resetProjectList()), m_projectList, SLOT(slotResetProjectList()));
2528     connect(doc, SIGNAL(signalDeleteProjectClip(const QString &)), this, SLOT(slotDeleteClip(const QString &)));
2529     connect(doc, SIGNAL(updateClipDisplay(const QString &)), m_projectList, SLOT(slotUpdateClip(const QString &)));
2530     connect(doc, SIGNAL(selectLastAddedClip(const QString &)), m_projectList, SLOT(slotSelectClip(const QString &)));
2531
2532     connect(doc, SIGNAL(docModified(bool)), this, SLOT(slotUpdateDocumentState(bool)));
2533     connect(doc, SIGNAL(guidesUpdated()), this, SLOT(slotGuidesUpdated()));
2534     connect(doc, SIGNAL(saveTimelinePreview(const QString &)), trackView, SLOT(slotSaveTimelinePreview(const QString)));
2535     
2536     connect(m_notesWidget, SIGNAL(textChanged()), doc, SLOT(setModified()));
2537
2538     connect(trackView->projectView(), SIGNAL(clipItemSelected(ClipItem*, int, bool)), m_effectStack, SLOT(slotClipItemSelected(ClipItem*, int)));
2539     connect(trackView->projectView(), SIGNAL(updateClipMarkers(DocClipBase *)), this, SLOT(slotUpdateClipMarkers(DocClipBase*)));
2540     connect(trackView, SIGNAL(showTrackEffects(int, TrackInfo)), m_effectStack, SLOT(slotTrackItemSelected(int, TrackInfo)));
2541     connect(trackView, SIGNAL(showTrackEffects(int, TrackInfo)), this, SLOT(slotActivateEffectStackView()));
2542
2543     connect(trackView->projectView(), SIGNAL(clipItemSelected(ClipItem*, int, bool)), this, SLOT(slotActivateEffectStackView(ClipItem*, int, bool)));
2544     connect(trackView->projectView(), SIGNAL(transitionItemSelected(Transition*, int, QPoint, bool)), m_transitionConfig, SLOT(slotTransitionItemSelected(Transition*, int, QPoint, bool)));
2545     connect(trackView->projectView(), SIGNAL(transitionItemSelected(Transition*, int, QPoint, bool)), this, SLOT(slotActivateTransitionView(Transition *)));
2546     m_zoomSlider->setValue(doc->zoom().x());
2547     connect(trackView->projectView(), SIGNAL(zoomIn()), this, SLOT(slotZoomIn()));
2548     connect(trackView->projectView(), SIGNAL(zoomOut()), this, SLOT(slotZoomOut()));
2549     connect(trackView, SIGNAL(setZoom(int)), this, SLOT(slotSetZoom(int)));
2550     connect(trackView->projectView(), SIGNAL(displayMessage(const QString&, MessageType)), m_messageLabel, SLOT(setMessage(const QString&, MessageType)));
2551
2552     connect(trackView->projectView(), SIGNAL(showClipFrame(DocClipBase *, QPoint, const int)), m_clipMonitor, SLOT(slotSetClipProducer(DocClipBase *, QPoint, const int)));
2553     connect(trackView->projectView(), SIGNAL(playMonitor()), m_projectMonitor, SLOT(slotPlay()));
2554
2555     connect(trackView->projectView(), SIGNAL(clipItemSelected(ClipItem*, int, bool)), m_projectMonitor, SLOT(slotSetSelectedClip(ClipItem*)));
2556     connect(trackView->projectView(), SIGNAL(transitionItemSelected(Transition*, int, QPoint, bool)), m_projectMonitor, SLOT(slotSetSelectedClip(Transition*)));
2557
2558     connect(m_effectStack, SIGNAL(updateEffect(ClipItem*, int, QDomElement, QDomElement, int)), trackView->projectView(), SLOT(slotUpdateClipEffect(ClipItem*, int, QDomElement, QDomElement, int)));
2559     connect(m_effectStack, SIGNAL(updateClipRegion(ClipItem*, int, QString)), trackView->projectView(), SLOT(slotUpdateClipRegion(ClipItem*, int, QString)));
2560     connect(m_effectStack, SIGNAL(removeEffect(ClipItem*, int, QDomElement)), trackView->projectView(), SLOT(slotDeleteEffect(ClipItem*, int, QDomElement)));
2561     connect(m_effectStack, SIGNAL(changeEffectState(ClipItem*, int, int, bool)), trackView->projectView(), SLOT(slotChangeEffectState(ClipItem*, int, int, bool)));
2562     connect(m_effectStack, SIGNAL(changeEffectPosition(ClipItem*, int, int, int)), trackView->projectView(), SLOT(slotChangeEffectPosition(ClipItem*, int, int, int)));
2563     connect(m_effectStack, SIGNAL(refreshEffectStack(ClipItem*)), trackView->projectView(), SLOT(slotRefreshEffects(ClipItem*)));
2564     connect(m_transitionConfig, SIGNAL(transitionUpdated(Transition *, QDomElement)), trackView->projectView() , SLOT(slotTransitionUpdated(Transition *, QDomElement)));
2565     connect(m_transitionConfig, SIGNAL(seekTimeline(int)), trackView->projectView() , SLOT(setCursorPos(int)));
2566     connect(m_effectStack, SIGNAL(seekTimeline(int)), trackView->projectView() , SLOT(setCursorPos(int)));
2567     connect(m_effectStack, SIGNAL(reloadEffects()), this, SLOT(slotReloadEffects()));
2568     connect(m_effectStack, SIGNAL(displayMessage(const QString&, int)), this, SLOT(slotGotProgressInfo(const QString&, int)));
2569
2570     connect(trackView->projectView(), SIGNAL(activateDocumentMonitor()), m_projectMonitor, SLOT(activateMonitor()));
2571     connect(trackView, SIGNAL(zoneMoved(int, int)), this, SLOT(slotZoneMoved(int, int)));
2572     connect(m_projectList, SIGNAL(loadingIsOver()), trackView->projectView(), SLOT(slotUpdateAllThumbs()));
2573
2574
2575     trackView->projectView()->setContextMenu(m_timelineContextMenu, m_timelineContextClipMenu, m_timelineContextTransitionMenu, m_clipTypeGroup, (QMenu*)(factory()->container("marker_menu", this)));
2576     m_activeTimeline = trackView;
2577     if (m_renderWidget) {
2578         slotCheckRenderStatus();
2579         m_renderWidget->setProfile(doc->mltProfile());
2580         m_renderWidget->setGuides(doc->guidesXml(), doc->projectDuration());
2581         m_renderWidget->setDocumentPath(doc->projectFolder().path(KUrl::AddTrailingSlash));
2582         m_renderWidget->setRenderProfile(doc->getRenderProperties());
2583     }
2584     //doc->setRenderer(m_projectMonitor->render);
2585     m_commandStack->setActiveStack(doc->commandStack());
2586     KdenliveSettings::setProject_display_ratio(doc->dar());
2587     //doc->clipManager()->checkAudioThumbs();
2588
2589     //m_overView->setScene(trackView->projectScene());
2590     //m_overView->scale(m_overView->width() / trackView->duration(), m_overView->height() / (50 * trackView->tracksNumber()));
2591     //m_overView->fitInView(m_overView->itemAt(0, 50), Qt::KeepAspectRatio);
2592
2593     setCaption(doc->description(), doc->isModified());
2594     m_saveAction->setEnabled(doc->isModified());
2595     m_normalEditTool->setChecked(true);
2596     m_activeDocument = doc;
2597     m_activeTimeline->updateProjectFps();
2598     m_activeDocument->checkProjectClips();
2599 #ifndef Q_WS_MAC
2600     m_recMonitor->slotUpdateCaptureFolder(m_activeDocument->projectFolder().path(KUrl::AddTrailingSlash));
2601 #endif
2602     //Update the mouse position display so it will display in DF/NDF format by default based on the project setting.
2603     slotUpdateMousePosition(0);
2604     m_monitorManager->activateMonitor("clip");
2605     // set tool to select tool
2606     m_buttonSelectTool->setChecked(true);
2607 }
2608
2609 void MainWindow::slotZoneMoved(int start, int end)
2610 {
2611     m_activeDocument->setZone(start, end);
2612     m_projectMonitor->slotZoneMoved(start, end);
2613 }
2614
2615 void MainWindow::slotGuidesUpdated()
2616 {
2617     if (m_renderWidget)
2618         m_renderWidget->setGuides(m_activeDocument->guidesXml(), m_activeDocument->projectDuration());
2619 }
2620
2621 void MainWindow::slotEditKeys()
2622 {
2623     KShortcutsDialog dialog(KShortcutsEditor::AllActions, KShortcutsEditor::LetterShortcutsAllowed, this);
2624     dialog.addCollection(actionCollection(), i18nc("general keyboard shortcuts", "General"));
2625     dialog.addCollection(m_effectsActionCollection, i18nc("effects and transitions keyboard shortcuts", "Effects & Transitions"));
2626     dialog.configure();
2627 }
2628
2629 void MainWindow::slotPreferences(int page, int option)
2630 {
2631     /*
2632      * An instance of your dialog could be already created and could be
2633      * cached, in which case you want to display the cached dialog
2634      * instead of creating another one
2635      */
2636     if (m_stopmotion) m_stopmotion->slotLive(false);
2637     if (KConfigDialog::showDialog("settings")) {
2638         KdenliveSettingsDialog* d = static_cast <KdenliveSettingsDialog*>(KConfigDialog::exists("settings"));
2639         if (page != -1) d->showPage(page, option);
2640         return;
2641     }
2642
2643     // KConfigDialog didn't find an instance of this dialog, so lets
2644     // create it :
2645     
2646     // Get the mappable actions in localized form
2647     QMap<QString, QString> actions;
2648     KActionCollection* collection = actionCollection();
2649     foreach (const QString& action_name, m_action_names) {
2650         actions[collection->action(action_name)->text()] = action_name;
2651     }
2652     
2653     KdenliveSettingsDialog* dialog = new KdenliveSettingsDialog(actions, this);
2654     connect(dialog, SIGNAL(settingsChanged(const QString&)), this, SLOT(updateConfiguration()));
2655     connect(dialog, SIGNAL(doResetProfile()), m_monitorManager, SLOT(slotResetProfiles()));
2656 #ifndef Q_WS_MAC
2657     connect(dialog, SIGNAL(updateCaptureFolder()), this, SLOT(slotUpdateCaptureFolder()));
2658 #endif
2659     dialog->show();
2660     if (page != -1) dialog->showPage(page, option);
2661 }
2662
2663 void MainWindow::slotUpdateCaptureFolder()
2664 {
2665
2666 #ifndef Q_WS_MAC
2667     if (m_activeDocument) m_recMonitor->slotUpdateCaptureFolder(m_activeDocument->projectFolder().path(KUrl::AddTrailingSlash));
2668     else m_recMonitor->slotUpdateCaptureFolder(KdenliveSettings::defaultprojectfolder());
2669 #endif
2670 }
2671
2672 void MainWindow::updateConfiguration()
2673 {
2674     //TODO: we should apply settings to all projects, not only the current one
2675     if (m_activeTimeline) {
2676         m_activeTimeline->refresh();
2677         m_activeTimeline->projectView()->checkAutoScroll();
2678         m_activeTimeline->checkTrackHeight();
2679         if (m_activeDocument)
2680             m_activeDocument->clipManager()->checkAudioThumbs();
2681     }
2682     m_buttonAudioThumbs->setChecked(KdenliveSettings::audiothumbnails());
2683     m_buttonVideoThumbs->setChecked(KdenliveSettings::videothumbnails());
2684     m_buttonShowMarkers->setChecked(KdenliveSettings::showmarkers());
2685     m_buttonAutomaticSplitAudio->setChecked(KdenliveSettings::splitaudio());
2686
2687     // Update list of transcoding profiles
2688     loadTranscoders();
2689         loadStabilize();
2690 #ifdef USE_JOGSHUTTLE
2691     activateShuttleDevice();
2692 #endif
2693
2694 }
2695
2696 void MainWindow::slotSwitchSplitAudio()
2697 {
2698     KdenliveSettings::setSplitaudio(!KdenliveSettings::splitaudio());
2699     m_buttonAutomaticSplitAudio->setChecked(KdenliveSettings::splitaudio());
2700 }
2701
2702 void MainWindow::slotSwitchVideoThumbs()
2703 {
2704     KdenliveSettings::setVideothumbnails(!KdenliveSettings::videothumbnails());
2705     if (m_activeTimeline)
2706         m_activeTimeline->projectView()->slotUpdateAllThumbs();
2707     m_buttonVideoThumbs->setChecked(KdenliveSettings::videothumbnails());
2708 }
2709
2710 void MainWindow::slotSwitchAudioThumbs()
2711 {
2712     KdenliveSettings::setAudiothumbnails(!KdenliveSettings::audiothumbnails());
2713     if (m_activeTimeline) {
2714         m_activeTimeline->refresh();
2715         m_activeTimeline->projectView()->checkAutoScroll();
2716         if (m_activeDocument)
2717             m_activeDocument->clipManager()->checkAudioThumbs();
2718     }
2719     m_buttonAudioThumbs->setChecked(KdenliveSettings::audiothumbnails());
2720 }
2721
2722 void MainWindow::slotSwitchMarkersComments()
2723 {
2724     KdenliveSettings::setShowmarkers(!KdenliveSettings::showmarkers());
2725     if (m_activeTimeline)
2726         m_activeTimeline->refresh();
2727     m_buttonShowMarkers->setChecked(KdenliveSettings::showmarkers());
2728 }
2729
2730 void MainWindow::slotSwitchSnap()
2731 {
2732     KdenliveSettings::setSnaptopoints(!KdenliveSettings::snaptopoints());
2733     m_buttonSnap->setChecked(KdenliveSettings::snaptopoints());
2734 }
2735
2736
2737 void MainWindow::slotDeleteItem()
2738 {
2739     if (QApplication::focusWidget() &&
2740             QApplication::focusWidget()->parentWidget() &&
2741             QApplication::focusWidget()->parentWidget()->parentWidget() &&
2742             QApplication::focusWidget()->parentWidget()->parentWidget() == m_projectListDock) {
2743         m_projectList->slotRemoveClip();
2744
2745     } else {
2746         QWidget *widget = QApplication::focusWidget();
2747         while (widget) {
2748             if (widget == m_effectStackDock) {
2749                 m_effectStack->slotItemDel();
2750                 return;
2751             }
2752             widget = widget->parentWidget();
2753         }
2754
2755         // effect stack has no focus
2756         if (m_activeTimeline)
2757             m_activeTimeline->projectView()->deleteSelectedClips();
2758     }
2759 }
2760
2761 void MainWindow::slotUpdateClipMarkers(DocClipBase *clip)
2762 {
2763     if (m_clipMonitor->isActive())
2764         m_clipMonitor->checkOverlay();
2765     m_clipMonitor->updateMarkers(clip);
2766 }
2767
2768 void MainWindow::slotAddClipMarker()
2769 {
2770     DocClipBase *clip = NULL;
2771     GenTime pos;
2772     if (m_projectMonitor->isActive()) {
2773         if (m_activeTimeline) {
2774             ClipItem *item = m_activeTimeline->projectView()->getActiveClipUnderCursor();
2775             if (item) {
2776                 pos = GenTime((int)((m_projectMonitor->position() - item->startPos() + item->cropStart()).frames(m_activeDocument->fps()) * item->speed() + 0.5), m_activeDocument->fps());
2777                 clip = item->baseClip();
2778             }
2779         }
2780     } else {
2781         clip = m_clipMonitor->activeClip();
2782         pos = m_clipMonitor->position();
2783     }
2784     if (!clip) {
2785         m_messageLabel->setMessage(i18n("Cannot find clip to add marker"), ErrorMessage);
2786         return;
2787     }
2788     QString id = clip->getId();
2789     CommentedTime marker(pos, i18n("Marker"));
2790     MarkerDialog d(clip, marker, m_activeDocument->timecode(), i18n("Add Marker"), this);
2791     if (d.exec() == QDialog::Accepted)
2792         m_activeTimeline->projectView()->slotAddClipMarker(id, d.newMarker().time(), d.newMarker().comment());
2793 }
2794
2795 void MainWindow::slotDeleteClipMarker()
2796 {
2797     DocClipBase *clip = NULL;
2798     GenTime pos;
2799     if (m_projectMonitor->isActive()) {
2800         if (m_activeTimeline) {
2801             ClipItem *item = m_activeTimeline->projectView()->getActiveClipUnderCursor();
2802             if (item) {
2803                 pos = (m_projectMonitor->position() - item->startPos() + item->cropStart()) / item->speed();
2804                 clip = item->baseClip();
2805             }
2806         }
2807     } else {
2808         clip = m_clipMonitor->activeClip();
2809         pos = m_clipMonitor->position();
2810     }
2811     if (!clip) {
2812         m_messageLabel->setMessage(i18n("Cannot find clip to remove marker"), ErrorMessage);
2813         return;
2814     }
2815
2816     QString id = clip->getId();
2817     QString comment = clip->markerComment(pos);
2818     if (comment.isEmpty()) {
2819         m_messageLabel->setMessage(i18n("No marker found at cursor time"), ErrorMessage);
2820         return;
2821     }
2822     m_activeTimeline->projectView()->slotDeleteClipMarker(comment, id, pos);
2823 }
2824
2825 void MainWindow::slotDeleteAllClipMarkers()
2826 {
2827     DocClipBase *clip = NULL;
2828     if (m_projectMonitor->isActive()) {
2829         if (m_activeTimeline) {
2830             ClipItem *item = m_activeTimeline->projectView()->getActiveClipUnderCursor();
2831             if (item) {
2832                 clip = item->baseClip();
2833             }
2834         }
2835     } else {
2836         clip = m_clipMonitor->activeClip();
2837     }
2838     if (!clip) {
2839         m_messageLabel->setMessage(i18n("Cannot find clip to remove marker"), ErrorMessage);
2840         return;
2841     }
2842     m_activeTimeline->projectView()->slotDeleteAllClipMarkers(clip->getId());
2843 }
2844
2845 void MainWindow::slotEditClipMarker()
2846 {
2847     DocClipBase *clip = NULL;
2848     GenTime pos;
2849     if (m_projectMonitor->isActive()) {
2850         if (m_activeTimeline) {
2851             ClipItem *item = m_activeTimeline->projectView()->getActiveClipUnderCursor();
2852             if (item) {
2853                 pos = (m_projectMonitor->position() - item->startPos() + item->cropStart()) / item->speed();
2854                 clip = item->baseClip();
2855             }
2856         }
2857     } else {
2858         clip = m_clipMonitor->activeClip();
2859         pos = m_clipMonitor->position();
2860     }
2861     if (!clip) {
2862         m_messageLabel->setMessage(i18n("Cannot find clip to remove marker"), ErrorMessage);
2863         return;
2864     }
2865
2866     QString id = clip->getId();
2867     QString oldcomment = clip->markerComment(pos);
2868     if (oldcomment.isEmpty()) {
2869         m_messageLabel->setMessage(i18n("No marker found at cursor time"), ErrorMessage);
2870         return;
2871     }
2872
2873     CommentedTime marker(pos, oldcomment);
2874     MarkerDialog d(clip, marker, m_activeDocument->timecode(), i18n("Edit Marker"), this);
2875     if (d.exec() == QDialog::Accepted) {
2876         m_activeTimeline->projectView()->slotAddClipMarker(id, d.newMarker().time(), d.newMarker().comment());
2877         if (d.newMarker().time() != pos) {
2878             // remove old marker
2879             m_activeTimeline->projectView()->slotAddClipMarker(id, pos, QString());
2880         }
2881     }
2882 }
2883
2884 void MainWindow::slotAddMarkerGuideQuickly()
2885 {
2886     if (!m_activeTimeline || !m_activeDocument)
2887         return;
2888
2889     if (m_clipMonitor->isActive()) {
2890         DocClipBase *clip = m_clipMonitor->activeClip();
2891         GenTime pos = m_clipMonitor->position();
2892
2893         if (!clip) {
2894             m_messageLabel->setMessage(i18n("Cannot find clip to add marker"), ErrorMessage);
2895             return;
2896         }
2897
2898         m_activeTimeline->projectView()->slotAddClipMarker(clip->getId(), pos, m_activeDocument->timecode().getDisplayTimecode(pos, false));
2899     } else {
2900         m_activeTimeline->projectView()->slotAddGuide(false);
2901     }
2902 }
2903
2904 void MainWindow::slotAddGuide()
2905 {
2906     if (m_activeTimeline)
2907         m_activeTimeline->projectView()->slotAddGuide();
2908 }
2909
2910 void MainWindow::slotInsertSpace()
2911 {
2912     if (m_activeTimeline)
2913         m_activeTimeline->projectView()->slotInsertSpace();
2914 }
2915
2916 void MainWindow::slotRemoveSpace()
2917 {
2918     if (m_activeTimeline)
2919         m_activeTimeline->projectView()->slotRemoveSpace();
2920 }
2921
2922 void MainWindow::slotInsertTrack(int ix)
2923 {
2924     m_projectMonitor->activateMonitor();
2925     if (m_activeTimeline)
2926         m_activeTimeline->projectView()->slotInsertTrack(ix);
2927     if (m_activeDocument)
2928         m_transitionConfig->updateProjectFormat(m_activeDocument->mltProfile(), m_activeDocument->timecode(), m_activeDocument->tracksList());
2929 }
2930
2931 void MainWindow::slotDeleteTrack(int ix)
2932 {
2933     m_projectMonitor->activateMonitor();
2934     if (m_activeTimeline)
2935         m_activeTimeline->projectView()->slotDeleteTrack(ix);
2936     if (m_activeDocument)
2937         m_transitionConfig->updateProjectFormat(m_activeDocument->mltProfile(), m_activeDocument->timecode(), m_activeDocument->tracksList());
2938 }
2939
2940 void MainWindow::slotConfigTrack(int ix)
2941 {
2942     m_projectMonitor->activateMonitor();
2943     if (m_activeTimeline)
2944         m_activeTimeline->projectView()->slotConfigTracks(ix);
2945     if (m_activeDocument)
2946         m_transitionConfig->updateProjectFormat(m_activeDocument->mltProfile(), m_activeDocument->timecode(), m_activeDocument->tracksList());
2947 }
2948
2949 void MainWindow::slotEditGuide()
2950 {
2951     if (m_activeTimeline)
2952         m_activeTimeline->projectView()->slotEditGuide();
2953 }
2954
2955 void MainWindow::slotDeleteGuide()
2956 {
2957     if (m_activeTimeline)
2958         m_activeTimeline->projectView()->slotDeleteGuide();
2959 }
2960
2961 void MainWindow::slotDeleteAllGuides()
2962 {
2963     if (m_activeTimeline)
2964         m_activeTimeline->projectView()->slotDeleteAllGuides();
2965 }
2966
2967 void MainWindow::slotCutTimelineClip()
2968 {
2969     if (m_activeTimeline)
2970         m_activeTimeline->projectView()->cutSelectedClips();
2971 }
2972
2973 void MainWindow::slotInsertClipOverwrite()
2974 {
2975     if (m_activeTimeline) {
2976         QStringList data = m_clipMonitor->getZoneInfo();
2977         m_activeTimeline->projectView()->insertZoneOverwrite(data, m_activeTimeline->inPoint());
2978     }
2979 }
2980
2981 void MainWindow::slotSelectTimelineClip()
2982 {
2983     if (m_activeTimeline)
2984         m_activeTimeline->projectView()->selectClip(true);
2985 }
2986
2987 void MainWindow::slotSelectTimelineTransition()
2988 {
2989     if (m_activeTimeline)
2990         m_activeTimeline->projectView()->selectTransition(true);
2991 }
2992
2993 void MainWindow::slotDeselectTimelineClip()
2994 {
2995     if (m_activeTimeline)
2996         m_activeTimeline->projectView()->selectClip(false, true);
2997 }
2998
2999 void MainWindow::slotDeselectTimelineTransition()
3000 {
3001     if (m_activeTimeline)
3002         m_activeTimeline->projectView()->selectTransition(false, true);
3003 }
3004
3005 void MainWindow::slotSelectAddTimelineClip()
3006 {
3007     if (m_activeTimeline)
3008         m_activeTimeline->projectView()->selectClip(true, true);
3009 }
3010
3011 void MainWindow::slotSelectAddTimelineTransition()
3012 {
3013     if (m_activeTimeline)
3014         m_activeTimeline->projectView()->selectTransition(true, true);
3015 }
3016
3017 void MainWindow::slotGroupClips()
3018 {
3019     if (m_activeTimeline)
3020         m_activeTimeline->projectView()->groupClips();
3021 }
3022
3023 void MainWindow::slotUnGroupClips()
3024 {
3025     if (m_activeTimeline)
3026         m_activeTimeline->projectView()->groupClips(false);
3027 }
3028
3029 void MainWindow::slotEditItemDuration()
3030 {
3031     if (m_activeTimeline)
3032         m_activeTimeline->projectView()->editItemDuration();
3033 }
3034
3035 void MainWindow::slotAddProjectClip(KUrl url)
3036 {
3037     if (m_activeDocument)
3038         m_activeDocument->slotAddClipFile(url, QString());
3039 }
3040
3041 void MainWindow::slotAddProjectClipList(KUrl::List urls)
3042 {
3043     if (m_activeDocument)
3044         m_activeDocument->slotAddClipList(urls, QString());
3045 }
3046
3047 void MainWindow::slotAddTransition(QAction *result)
3048 {
3049     if (!result) return;
3050     QStringList info = result->data().toStringList();
3051     if (info.isEmpty()) return;
3052     QDomElement transition = transitions.getEffectByTag(info.at(1), info.at(2));
3053     if (m_activeTimeline && !transition.isNull()) {
3054         m_activeTimeline->projectView()->slotAddTransitionToSelectedClips(transition.cloneNode().toElement());
3055     }
3056 }
3057
3058 void MainWindow::slotAddVideoEffect(QAction *result)
3059 {
3060     if (!result) return;
3061     const int EFFECT_VIDEO = 1;
3062     const int EFFECT_AUDIO = 2;
3063     QStringList info = result->data().toStringList();
3064
3065     if (info.isEmpty() || info.size() < 3) return;
3066     QDomElement effect ;
3067     if (info.at(2) == QString::number((int) EFFECT_VIDEO))
3068             effect = videoEffects.getEffectByTag(info.at(0), info.at(1));
3069     else if (info.at(2) == QString::number((int) EFFECT_AUDIO))
3070             effect = audioEffects.getEffectByTag(info.at(0), info.at(1));
3071     else
3072             effect = customEffects.getEffectByTag(info.at(0), info.at(1));
3073     if (!effect.isNull()) slotAddEffect(effect);
3074     else m_messageLabel->setMessage(i18n("Cannot find effect %1 / %2").arg(info.at(0)).arg(info.at(1)), ErrorMessage);
3075 }
3076
3077
3078 void MainWindow::slotZoomIn()
3079 {
3080     m_zoomSlider->setValue(m_zoomSlider->value() - 1);
3081     slotShowZoomSliderToolTip();
3082 }
3083
3084 void MainWindow::slotZoomOut()
3085 {
3086     m_zoomSlider->setValue(m_zoomSlider->value() + 1);
3087     slotShowZoomSliderToolTip();
3088 }
3089
3090 void MainWindow::slotFitZoom()
3091 {
3092     if (m_activeTimeline)
3093         m_zoomSlider->setValue(m_activeTimeline->fitZoom());
3094 }
3095
3096 void MainWindow::slotSetZoom(int value)
3097 {
3098     value = qMax(m_zoomSlider->minimum(), value);
3099     value = qMin(m_zoomSlider->maximum(), value);
3100
3101     if (m_activeTimeline)
3102         m_activeTimeline->slotChangeZoom(value);
3103
3104     m_zoomOut->setEnabled(value < m_zoomSlider->maximum());
3105     m_zoomIn->setEnabled(value > m_zoomSlider->minimum());
3106     slotUpdateZoomSliderToolTip(value);
3107
3108     m_zoomSlider->blockSignals(true);
3109     m_zoomSlider->setValue(value);
3110     m_zoomSlider->blockSignals(false);
3111 }
3112
3113 void MainWindow::slotShowZoomSliderToolTip(int zoomlevel)
3114 {
3115     if (zoomlevel != -1)
3116         slotUpdateZoomSliderToolTip(zoomlevel);
3117
3118     QPoint global = m_zoomSlider->rect().topLeft();
3119     global.ry() += m_zoomSlider->height() / 2;
3120     QHelpEvent toolTipEvent(QEvent::ToolTip, QPoint(0, 0), m_zoomSlider->mapToGlobal(global));
3121     QApplication::sendEvent(m_zoomSlider, &toolTipEvent);
3122 }
3123
3124 void MainWindow::slotUpdateZoomSliderToolTip(int zoomlevel)
3125 {
3126     m_zoomSlider->setToolTip(i18n("Zoom Level: %1/13", (13 - zoomlevel)));
3127 }
3128
3129 void MainWindow::slotGotProgressInfo(const QString &message, int progress)
3130 {
3131     m_statusProgressBar->setValue(progress);
3132     if (progress >= 0) {
3133         if (!message.isEmpty())
3134             m_messageLabel->setMessage(message, InformationMessage);//statusLabel->setText(message);
3135         m_statusProgressBar->setVisible(true);
3136     } else if (progress == -2) {
3137         if (!message.isEmpty())
3138             m_messageLabel->setMessage(message, ErrorMessage);
3139         m_statusProgressBar->setVisible(false);
3140     } else {
3141         m_messageLabel->setMessage(QString(), DefaultMessage);
3142         m_statusProgressBar->setVisible(false);
3143     }
3144 }
3145
3146 void MainWindow::slotShowClipProperties(DocClipBase *clip)
3147 {
3148     if (clip->clipType() == TEXT) {
3149         QString titlepath = m_activeDocument->projectFolder().path(KUrl::AddTrailingSlash) + "titles/";
3150         if (!clip->getProperty("resource").isEmpty() && clip->getProperty("xmldata").isEmpty()) {
3151             // template text clip
3152
3153             // Get the list of existing templates
3154             QStringList filter;
3155             filter << "*.kdenlivetitle";
3156             QStringList templateFiles = QDir(titlepath).entryList(filter, QDir::Files);
3157
3158             QDialog *dia = new QDialog(this);
3159             Ui::TemplateClip_UI dia_ui;
3160             dia_ui.setupUi(dia);
3161             int ix = -1;
3162             const QString templatePath = clip->getProperty("resource");
3163             for (int i = 0; i < templateFiles.size(); ++i) {
3164                 dia_ui.template_list->comboBox()->addItem(templateFiles.at(i), titlepath + templateFiles.at(i));
3165                 if (templatePath == KUrl(titlepath + templateFiles.at(i)).path()) ix = i;
3166             }
3167             if (ix != -1) dia_ui.template_list->comboBox()->setCurrentIndex(ix);
3168             else dia_ui.template_list->comboBox()->insertItem(0, templatePath);
3169             dia_ui.template_list->fileDialog()->setFilter("*.kdenlivetitle");
3170             //warning: setting base directory doesn't work??
3171             KUrl startDir(titlepath);
3172             dia_ui.template_list->fileDialog()->setUrl(startDir);
3173             dia_ui.description->setText(clip->getProperty("description"));
3174             if (dia->exec() == QDialog::Accepted) {
3175                 QString textTemplate = dia_ui.template_list->comboBox()->itemData(dia_ui.template_list->comboBox()->currentIndex()).toString();
3176                 if (textTemplate.isEmpty()) textTemplate = dia_ui.template_list->comboBox()->currentText();
3177
3178                 QMap <QString, QString> newprops;
3179
3180                 if (KUrl(textTemplate).path() != templatePath) {
3181                     // The template was changed
3182                     newprops.insert("resource", textTemplate);
3183                 }
3184
3185                 if (dia_ui.description->toPlainText() != clip->getProperty("description")) {
3186                     newprops.insert("description", dia_ui.description->toPlainText());
3187                 }
3188
3189                 QString newtemplate = newprops.value("xmltemplate");
3190                 if (newtemplate.isEmpty()) newtemplate = templatePath;
3191
3192                 // template modified we need to update xmldata
3193                 QString description = newprops.value("description");
3194                 if (description.isEmpty()) description = clip->getProperty("description");
3195                 else newprops.insert("templatetext", description);
3196                 //newprops.insert("xmldata", m_projectList->generateTemplateXml(newtemplate, description).toString());
3197                 if (!newprops.isEmpty()) {
3198                     EditClipCommand *command = new EditClipCommand(m_projectList, clip->getId(), clip->properties(), newprops, true);
3199                     m_activeDocument->commandStack()->push(command);
3200                 }
3201             }
3202             delete dia;
3203             return;
3204         }
3205         QString path = clip->getProperty("resource");
3206         TitleWidget *dia_ui = new TitleWidget(KUrl(), m_activeDocument->timecode(), titlepath, m_projectMonitor->render, this);
3207         QDomDocument doc;
3208         doc.setContent(clip->getProperty("xmldata"));
3209         dia_ui->setXml(doc);
3210         if (dia_ui->exec() == QDialog::Accepted) {
3211             QMap <QString, QString> newprops;
3212             newprops.insert("xmldata", dia_ui->xml().toString());
3213             if (dia_ui->outPoint() != clip->duration().frames(m_activeDocument->fps()) - 1) {
3214                 // duration changed, we need to update duration
3215                 newprops.insert("out", QString::number(dia_ui->outPoint()));
3216                 int currentLength = QString(clip->producerProperty("length")).toInt();
3217                 if (currentLength <= dia_ui->outPoint())
3218                         newprops.insert("length", QString::number(dia_ui->outPoint() + 1));
3219                 else newprops.insert("length", clip->producerProperty("length"));
3220             }
3221             if (!path.isEmpty()) {
3222                 // we are editing an external file, asked if we want to detach from that file or save the result to that title file.
3223                 if (KMessageBox::questionYesNo(this, i18n("You are editing an external title clip (%1). Do you want to save your changes to the title file or save the changes for this project only?", path), i18n("Save Title"), KGuiItem(i18n("Save to title file")), KGuiItem(i18n("Save in project only"))) == KMessageBox::Yes) {
3224                     // save to external file
3225                     dia_ui->saveTitle(path);
3226                 } else newprops.insert("resource", QString());
3227             }
3228             EditClipCommand *command = new EditClipCommand(m_projectList, clip->getId(), clip->properties(), newprops, true);
3229             m_activeDocument->commandStack()->push(command);
3230             //m_activeTimeline->projectView()->slotUpdateClip(clip->getId());
3231             m_activeDocument->setModified(true);
3232         }
3233         delete dia_ui;
3234
3235         //m_activeDocument->editTextClip(clip->getProperty("xml"), clip->getId());
3236         return;
3237     }
3238
3239     // any type of clip but a title
3240     ClipProperties *dia = new ClipProperties(clip, m_activeDocument->timecode(), m_activeDocument->fps(), this);
3241     connect(dia, SIGNAL(addMarker(const QString &, GenTime, QString)), m_activeTimeline->projectView(), SLOT(slotAddClipMarker(const QString &, GenTime, QString)));
3242     connect(dia, SIGNAL(deleteProxy(const QString)), m_projectList, SLOT(slotDeleteProxy(const QString)));
3243     connect(dia, SIGNAL(applyNewClipProperties(const QString, QMap <QString, QString> , QMap <QString, QString> , bool, bool)), this, SLOT(slotApplyNewClipProperties(const QString, QMap <QString, QString> , QMap <QString, QString> , bool, bool)));
3244     dia->show();
3245 }
3246
3247
3248 void MainWindow::slotApplyNewClipProperties(const QString id, QMap <QString, QString> props, QMap <QString, QString> newprops, bool refresh, bool reload)
3249 {
3250     if (newprops.isEmpty()) return;
3251     EditClipCommand *command = new EditClipCommand(m_projectList, id, props, newprops, true);
3252     m_activeDocument->commandStack()->push(command);
3253     m_activeDocument->setModified();
3254
3255     if (refresh) {
3256         // update clip occurences in timeline
3257         m_activeTimeline->projectView()->slotUpdateClip(id, reload);
3258     }
3259 }
3260
3261
3262 void MainWindow::slotShowClipProperties(QList <DocClipBase *> cliplist, QMap<QString, QString> commonproperties)
3263 {
3264     ClipProperties dia(cliplist, m_activeDocument->timecode(), commonproperties, this);
3265     if (dia.exec() == QDialog::Accepted) {
3266         QUndoCommand *command = new QUndoCommand();
3267         command->setText(i18n("Edit clips"));
3268         QMap <QString, QString> newImageProps = dia.properties();
3269         // Transparency setting applies only for images
3270         QMap <QString, QString> newProps = newImageProps;
3271         newProps.remove("transparency");
3272
3273         for (int i = 0; i < cliplist.count(); i++) {
3274             DocClipBase *clip = cliplist.at(i);
3275             if (clip->clipType() == IMAGE)
3276                 new EditClipCommand(m_projectList, clip->getId(), clip->properties(), newImageProps, true, command);
3277             else 
3278                 new EditClipCommand(m_projectList, clip->getId(), clip->properties(), newProps, true, command);
3279         }
3280         m_activeDocument->commandStack()->push(command);
3281         for (int i = 0; i < cliplist.count(); i++)
3282             m_activeTimeline->projectView()->slotUpdateClip(cliplist.at(i)->getId(), dia.needsTimelineReload());
3283     }
3284 }
3285
3286 void MainWindow::customEvent(QEvent* e)
3287 {
3288     if (e->type() == QEvent::User)
3289         m_messageLabel->setMessage(static_cast <MltErrorEvent *>(e)->message(), MltError);
3290 }
3291 void MainWindow::slotActivateEffectStackView(ClipItem* item, int ix, bool raise)
3292 {
3293     Q_UNUSED(item)
3294     Q_UNUSED(ix)
3295
3296     if (raise)
3297         m_effectStack->raiseWindow(m_effectStackDock);
3298 }
3299
3300 void MainWindow::slotActivateTransitionView(Transition *t)
3301 {
3302     if (t)
3303         m_transitionConfig->raiseWindow(m_transitionConfigDock);
3304 }
3305
3306 void MainWindow::slotSnapRewind()
3307 {
3308     if (m_projectMonitor->isActive()) {
3309         if (m_activeTimeline)
3310             m_activeTimeline->projectView()->slotSeekToPreviousSnap();
3311     } else  {
3312         m_clipMonitor->slotSeekToPreviousSnap();
3313     }
3314 }
3315
3316 void MainWindow::slotSnapForward()
3317 {
3318     if (m_projectMonitor->isActive()) {
3319         if (m_activeTimeline)
3320             m_activeTimeline->projectView()->slotSeekToNextSnap();
3321     } else {
3322         m_clipMonitor->slotSeekToNextSnap();
3323     }
3324 }
3325
3326 void MainWindow::slotClipStart()
3327 {
3328     if (m_projectMonitor->isActive()) {
3329         if (m_activeTimeline)
3330             m_activeTimeline->projectView()->clipStart();
3331     }
3332 }
3333
3334 void MainWindow::slotClipEnd()
3335 {
3336     if (m_projectMonitor->isActive()) {
3337         if (m_activeTimeline)
3338             m_activeTimeline->projectView()->clipEnd();
3339     }
3340 }
3341
3342 void MainWindow::slotZoneStart()
3343 {
3344     if (m_projectMonitor->isActive())
3345         m_projectMonitor->slotZoneStart();
3346     else
3347         m_clipMonitor->slotZoneStart();
3348 }
3349
3350 void MainWindow::slotZoneEnd()
3351 {
3352     if (m_projectMonitor->isActive())
3353         m_projectMonitor->slotZoneEnd();
3354     else
3355         m_clipMonitor->slotZoneEnd();
3356 }
3357
3358 void MainWindow::slotChangeTool(QAction * action)
3359 {
3360     if (action == m_buttonSelectTool)
3361         slotSetTool(SELECTTOOL);
3362     else if (action == m_buttonRazorTool)
3363         slotSetTool(RAZORTOOL);
3364     else if (action == m_buttonSpacerTool)
3365         slotSetTool(SPACERTOOL);
3366 }
3367
3368 void MainWindow::slotChangeEdit(QAction * action)
3369 {
3370     if (!m_activeTimeline)
3371         return;
3372
3373     if (action == m_overwriteEditTool)
3374         m_activeTimeline->projectView()->setEditMode(OVERWRITEEDIT);
3375     else if (action == m_insertEditTool)
3376         m_activeTimeline->projectView()->setEditMode(INSERTEDIT);
3377     else
3378         m_activeTimeline->projectView()->setEditMode(NORMALEDIT);
3379 }
3380
3381 void MainWindow::slotSetTool(PROJECTTOOL tool)
3382 {
3383     if (m_activeDocument && m_activeTimeline) {
3384         //m_activeDocument->setTool(tool);
3385         QString message;
3386         switch (tool)  {
3387         case SPACERTOOL:
3388             message = i18n("Ctrl + click to use spacer on current track only");
3389             break;
3390         case RAZORTOOL:
3391             message = i18n("Click on a clip to cut it");
3392             break;
3393         default:
3394             message = i18n("Shift + click to create a selection rectangle, Ctrl + click to add an item to selection");
3395             break;
3396         }
3397         m_messageLabel->setMessage(message, InformationMessage);
3398         m_activeTimeline->projectView()->setTool(tool);
3399     }
3400 }
3401
3402 void MainWindow::slotCopy()
3403 {
3404     if (m_activeDocument && m_activeTimeline)
3405         m_activeTimeline->projectView()->copyClip();
3406 }
3407
3408 void MainWindow::slotPaste()
3409 {
3410     if (m_activeDocument && m_activeTimeline)
3411         m_activeTimeline->projectView()->pasteClip();
3412 }
3413
3414 void MainWindow::slotPasteEffects()
3415 {
3416     if (m_activeDocument && m_activeTimeline)
3417         m_activeTimeline->projectView()->pasteClipEffects();
3418 }
3419
3420 void MainWindow::slotFind()
3421 {
3422     if (!m_activeDocument || !m_activeTimeline) return;
3423     m_projectSearch->setEnabled(false);
3424     m_findActivated = true;
3425     m_findString.clear();
3426     m_activeTimeline->projectView()->initSearchStrings();
3427     statusBar()->showMessage(i18n("Starting -- find text as you type"));
3428     m_findTimer.start(5000);
3429     qApp->installEventFilter(this);
3430 }
3431
3432 void MainWindow::slotFindNext()
3433 {
3434     if (m_activeTimeline && m_activeTimeline->projectView()->findNextString(m_findString))
3435         statusBar()->showMessage(i18n("Found: %1", m_findString));
3436     else
3437         statusBar()->showMessage(i18n("Reached end of project"));
3438     m_findTimer.start(4000);
3439 }
3440
3441 void MainWindow::findAhead()
3442 {
3443     if (m_activeTimeline && m_activeTimeline->projectView()->findString(m_findString)) {
3444         m_projectSearchNext->setEnabled(true);
3445         statusBar()->showMessage(i18n("Found: %1", m_findString));
3446     } else {
3447         m_projectSearchNext->setEnabled(false);
3448         statusBar()->showMessage(i18n("Not found: %1", m_findString));
3449     }
3450 }
3451
3452 void MainWindow::findTimeout()
3453 {
3454     m_projectSearchNext->setEnabled(false);
3455     m_findActivated = false;
3456     m_findString.clear();
3457     statusBar()->showMessage(i18n("Find stopped"), 3000);
3458     if (m_activeTimeline) m_activeTimeline->projectView()->clearSearchStrings();
3459     m_projectSearch->setEnabled(true);
3460     removeEventFilter(this);
3461 }
3462
3463 void MainWindow::slotClipInTimeline(const QString &clipId)
3464 {
3465     if (m_activeTimeline && m_activeDocument) {
3466         QList<ItemInfo> matching = m_activeTimeline->projectView()->findId(clipId);
3467
3468         QMenu *inTimelineMenu = static_cast<QMenu*>(factory()->container("clip_in_timeline", this));
3469         inTimelineMenu->clear();
3470
3471         QList <QAction *> actionList;
3472
3473         for (int i = 0; i < matching.count(); ++i) {
3474             QString track = QString::number(matching.at(i).track);
3475             QString start = m_activeDocument->timecode().getTimecode(matching.at(i).startPos);
3476             int j = 0;
3477             QAction *a = new QAction(track + ": " + start, this);
3478             a->setData(QStringList() << track << start);
3479             connect(a, SIGNAL(triggered()), this, SLOT(slotSelectClipInTimeline()));
3480             while (j < actionList.count()) {
3481                 if (actionList.at(j)->text() > a->text()) break;
3482                 j++;
3483             }
3484             actionList.insert(j, a);
3485         }
3486         inTimelineMenu->addActions(actionList);
3487
3488         if (matching.empty())
3489             inTimelineMenu->setEnabled(false);
3490         else
3491             inTimelineMenu->setEnabled(true);
3492     }
3493 }
3494
3495 void MainWindow::slotClipInProjectTree()
3496 {
3497     if (m_activeTimeline) {
3498         const QStringList &clipIds = m_activeTimeline->projectView()->selectedClips();
3499         if (clipIds.isEmpty())
3500             return;
3501         m_projectListDock->raise();
3502         for (int i = 0; i < clipIds.count(); i++)
3503             m_projectList->selectItemById(clipIds.at(i));
3504         if (m_projectMonitor->isActive())
3505             slotSwitchMonitors();
3506     }
3507 }
3508
3509 /*void MainWindow::slotClipToProjectTree()
3510 {
3511     if (m_activeTimeline) {
3512     const QList<ClipItem *> clips =  m_activeTimeline->projectView()->selectedClipItems();
3513         if (clips.isEmpty()) return;
3514         for (int i = 0; i < clips.count(); i++) {
3515         m_projectList->slotAddXmlClip(clips.at(i)->itemXml());
3516         }
3517         //m_projectList->selectItemById(clipIds.at(i));
3518     }
3519 }*/
3520
3521 void MainWindow::slotSelectClipInTimeline()
3522 {
3523     if (m_activeTimeline) {
3524         QAction *action = qobject_cast<QAction *>(sender());
3525         QStringList data = action->data().toStringList();
3526         m_activeTimeline->projectView()->selectFound(data.at(0), data.at(1));
3527     }
3528 }
3529
3530 void MainWindow::keyPressEvent(QKeyEvent *ke)
3531 {
3532     if (m_findActivated) {
3533         if (ke->key() == Qt::Key_Backspace) {
3534             m_findString = m_findString.left(m_findString.length() - 1);
3535
3536             if (!m_findString.isEmpty())
3537                 findAhead();
3538             else
3539                 findTimeout();
3540
3541             m_findTimer.start(4000);
3542             ke->accept();
3543             return;
3544         } else if (ke->key() == Qt::Key_Escape) {
3545             findTimeout();
3546             ke->accept();
3547             return;
3548         } else if (ke->key() == Qt::Key_Space || !ke->text().trimmed().isEmpty()) {
3549             m_findString += ke->text();
3550
3551             findAhead();
3552
3553             m_findTimer.start(4000);
3554             ke->accept();
3555             return;
3556         }
3557     } else {
3558         KXmlGuiWindow::keyPressEvent(ke);
3559     }
3560 }
3561
3562
3563 /** Gets called when the window gets hidden */
3564 void MainWindow::hideEvent(QHideEvent */*event*/)
3565 {
3566     if (isMinimized() && m_monitorManager)
3567         m_monitorManager->stopActiveMonitor();
3568 }
3569
3570 bool MainWindow::eventFilter(QObject *obj, QEvent *event)
3571 {
3572     if (m_findActivated) {
3573         if (event->type() == QEvent::ShortcutOverride) {
3574             QKeyEvent* ke = (QKeyEvent*) event;
3575             if (ke->text().trimmed().isEmpty()) return false;
3576             ke->accept();
3577             return true;
3578         } else {
3579             return false;
3580         }
3581     } else {
3582         // pass the event on to the parent class
3583         return QMainWindow::eventFilter(obj, event);
3584     }
3585 }
3586
3587
3588 void MainWindow::slotSaveZone(Render *render, QPoint zone, DocClipBase *baseClip, KUrl path)
3589 {
3590     KDialog *dialog = new KDialog(this);
3591     dialog->setCaption("Save clip zone");
3592     dialog->setButtons(KDialog::Ok | KDialog::Cancel);
3593
3594     QWidget *widget = new QWidget(dialog);
3595     dialog->setMainWidget(widget);
3596
3597     QVBoxLayout *vbox = new QVBoxLayout(widget);
3598     QLabel *label1 = new QLabel(i18n("Save clip zone as:"), this);
3599     if (path.isEmpty()) {
3600         QString tmppath = m_activeDocument->projectFolder().path(KUrl::AddTrailingSlash);
3601         if (baseClip == NULL) tmppath.append("untitled.mlt");
3602         else {
3603             tmppath.append((baseClip->name().isEmpty() ? baseClip->fileURL().fileName() : baseClip->name()) + "-" + QString::number(zone.x()).rightJustified(4, '0') + ".mlt");
3604         }
3605         path = KUrl(tmppath);
3606     }
3607     KUrlRequester *url = new KUrlRequester(path, this);
3608     url->setFilter("video/mlt-playlist");
3609     QLabel *label2 = new QLabel(i18n("Description:"), this);
3610     KLineEdit *edit = new KLineEdit(this);
3611     vbox->addWidget(label1);
3612     vbox->addWidget(url);
3613     vbox->addWidget(label2);
3614     vbox->addWidget(edit);
3615     if (dialog->exec() == QDialog::Accepted) {
3616         if (QFile::exists(url->url().path())) {
3617             if (KMessageBox::questionYesNo(this, i18n("File %1 already exists.\nDo you want to overwrite it?", url->url().path())) == KMessageBox::No) {
3618                 slotSaveZone(render, zone, baseClip, url->url());
3619                 return;
3620             }
3621         }
3622         if (baseClip && !baseClip->fileURL().isEmpty()) {
3623             // create zone from clip url, so that we don't have problems with proxy clips
3624             QProcess p;
3625 #if QT_VERSION >= 0x040600
3626             QProcessEnvironment env = QProcessEnvironment::systemEnvironment();
3627             env.remove("MLT_PROFILE");
3628             p.setProcessEnvironment(env);
3629 #else
3630             QStringList env = QProcess::systemEnvironment();
3631             env << "MLT_PROFILE='\0'";
3632             p.setEnvironment(env);
3633 #endif
3634             p.start(KdenliveSettings::rendererpath(), QStringList() << baseClip->fileURL().path() << "in=" + QString::number(zone.x()) << "out=" + QString::number(zone.y()) << "-consumer" << "xml:" + url->url().path());
3635             if (!p.waitForStarted(3000)) {
3636                 KMessageBox::sorry(this, i18n("Cannot start MLT's renderer:\n%1", KdenliveSettings::rendererpath()));
3637             }
3638             else if (!p.waitForFinished(5000)) {
3639                 KMessageBox::sorry(this, i18n("Timeout while creating xml output"));
3640             }
3641         }
3642         else render->saveZone(url->url(), edit->text(), zone);
3643     }
3644
3645 }
3646
3647 void MainWindow::slotSetInPoint()
3648 {
3649     if (m_clipMonitor->isActive()) {
3650         m_clipMonitor->slotSetZoneStart();
3651     } else {
3652         m_projectMonitor->slotSetZoneStart();
3653     }
3654 }
3655
3656 void MainWindow::slotSetOutPoint()
3657 {
3658     if (m_clipMonitor->isActive()) {
3659         m_clipMonitor->slotSetZoneEnd();
3660     } else {
3661         m_projectMonitor->slotSetZoneEnd();
3662     }
3663 }
3664
3665 void MainWindow::slotResizeItemStart()
3666 {
3667     if (m_activeTimeline)
3668         m_activeTimeline->projectView()->setInPoint();
3669 }
3670
3671 void MainWindow::slotResizeItemEnd()
3672 {
3673     if (m_activeTimeline)
3674         m_activeTimeline->projectView()->setOutPoint();
3675 }
3676
3677 int MainWindow::getNewStuff(const QString &configFile)
3678 {
3679     KNS3::Entry::List entries;
3680 #if KDE_IS_VERSION(4,3,80)
3681     KNS3::DownloadDialog dialog(configFile);
3682     dialog.exec();
3683     entries = dialog.changedEntries();
3684     foreach(const KNS3::Entry & entry, entries) {
3685         if (entry.status() == KNS3::Entry::Installed)
3686             kDebug() << "// Installed files: " << entry.installedFiles();
3687     }
3688 #else
3689     KNS::Engine engine(0);
3690     if (engine.init(configFile))
3691         entries = engine.downloadDialogModal(this);
3692     foreach(KNS::Entry * entry, entries) {
3693         if (entry->status() == KNS::Entry::Installed)
3694             kDebug() << "// Installed files: " << entry->installedFiles();
3695     }
3696 #endif
3697     return entries.size();
3698 }
3699
3700 void MainWindow::slotGetNewTitleStuff()
3701 {
3702     if (getNewStuff("kdenlive_titles.knsrc") > 0)
3703         TitleWidget::refreshTitleTemplates();
3704 }
3705
3706 void MainWindow::slotGetNewLumaStuff()
3707 {
3708     if (getNewStuff("kdenlive_wipes.knsrc") > 0) {
3709         initEffects::refreshLumas();
3710         m_activeTimeline->projectView()->reloadTransitionLumas();
3711     }
3712 }
3713
3714 void MainWindow::slotGetNewRenderStuff()
3715 {
3716     if (getNewStuff("kdenlive_renderprofiles.knsrc") > 0)
3717         if (m_renderWidget)
3718             m_renderWidget->reloadProfiles();
3719 }
3720
3721 void MainWindow::slotGetNewMltProfileStuff()
3722 {
3723     if (getNewStuff("kdenlive_projectprofiles.knsrc") > 0) {
3724         // update the list of profiles in settings dialog
3725         KdenliveSettingsDialog* d = static_cast <KdenliveSettingsDialog*>(KConfigDialog::exists("settings"));
3726         if (d)
3727             d->checkProfile();
3728     }
3729 }
3730
3731 void MainWindow::slotAutoTransition()
3732 {
3733     if (m_activeTimeline)
3734         m_activeTimeline->projectView()->autoTransition();
3735 }
3736
3737 void MainWindow::slotSplitAudio()
3738 {
3739     if (m_activeTimeline)
3740         m_activeTimeline->projectView()->splitAudio();
3741 }
3742
3743 void MainWindow::slotUpdateClipType(QAction *action)
3744 {
3745     if (m_activeTimeline) {
3746         if (action->data().toString() == "clip_audio_only") m_activeTimeline->projectView()->setAudioOnly();
3747         else if (action->data().toString() == "clip_video_only") m_activeTimeline->projectView()->setVideoOnly();
3748         else m_activeTimeline->projectView()->setAudioAndVideo();
3749     }
3750 }
3751
3752 void MainWindow::slotDvdWizard(const QString &url, const QString &profile)
3753 {
3754     // We must stop the monitors since we create a new on in the dvd wizard
3755     m_clipMonitor->stop();
3756     m_projectMonitor->stop();
3757     DvdWizard w(url, profile, this);
3758     w.exec();
3759     m_projectMonitor->start();
3760 }
3761
3762 void MainWindow::slotShowTimeline(bool show)
3763 {
3764     if (show == false) {
3765         m_timelineState = saveState();
3766         centralWidget()->setHidden(true);
3767     } else {
3768         centralWidget()->setHidden(false);
3769         restoreState(m_timelineState);
3770     }
3771 }
3772
3773 void MainWindow::slotMaximizeCurrent(bool)
3774 {
3775     //TODO: is there a way to maximize current widget?
3776
3777     m_timelineState = saveState();
3778     QWidget *par = focusWidget()->parentWidget();
3779     while (par->parentWidget() && par->parentWidget() != this)
3780         par = par->parentWidget();
3781     kDebug() << "CURRENT WIDGET: " << par->objectName();
3782 }
3783
3784 void MainWindow::loadStabilize()
3785 {
3786         QMenu* stabMenu= static_cast<QMenu*>(factory()->container("stabilize", this));
3787         stabMenu->clear();
3788         Mlt::Profile profile;
3789         if (Mlt::Factory::filter(profile,(char*)"videostab")){
3790                 QAction *action=stabMenu->addAction("Videostab (vstab)");
3791                 action->setData("videostab");
3792                 connect(action,SIGNAL(triggered()), this, SLOT(slotStabilize()));
3793         }
3794         if (Mlt::Factory::filter(profile,(char*)"videostab2")){
3795                 QAction *action=stabMenu->addAction("Videostab (transcode)");
3796                 action->setData("videostab2");
3797                 connect(action,SIGNAL(triggered()), this, SLOT(slotStabilize()));
3798         }
3799
3800
3801 }
3802
3803 void MainWindow::loadTranscoders()
3804 {
3805     QMenu *transMenu = static_cast<QMenu*>(factory()->container("transcoders", this));
3806     transMenu->clear();
3807
3808     KSharedConfigPtr config = KSharedConfig::openConfig("kdenlivetranscodingrc");
3809     KConfigGroup transConfig(config, "Transcoding");
3810     // read the entries
3811     QMap< QString, QString > profiles = transConfig.entryMap();
3812     QMapIterator<QString, QString> i(profiles);
3813     while (i.hasNext()) {
3814         i.next();
3815         QStringList data = i.value().split(";", QString::SkipEmptyParts);
3816         QAction *a = transMenu->addAction(i.key());
3817         a->setData(data);
3818         if (data.count() > 1)
3819             a->setToolTip(data.at(1));
3820         connect(a, SIGNAL(triggered()), this, SLOT(slotTranscode()));
3821     }
3822 }
3823
3824 void MainWindow::slotStabilize(KUrl::List urls)
3825 {
3826         QString condition,filtername;
3827
3828         if (urls.isEmpty()) {
3829         QAction *action = qobject_cast<QAction *>(sender());
3830                 if (action){
3831                         filtername=action->data().toString();
3832                         urls = m_projectList->getConditionalUrls(condition);
3833                 }
3834     }
3835     if (urls.isEmpty()) {
3836         m_messageLabel->setMessage(i18n("No clip to transcode"), ErrorMessage);
3837         return;
3838     }
3839         Mlt::Profile profile;
3840         Mlt::Filter filter(profile,filtername.toUtf8().data());
3841         ClipStabilize *d=new ClipStabilize(urls,filtername,&filter);
3842         connect(d, SIGNAL(addClip(KUrl)), this, SLOT(slotAddProjectClip(KUrl)));
3843         d->show();
3844
3845 }
3846
3847 void MainWindow::slotTranscode(KUrl::List urls)
3848 {
3849     QString params;
3850     QString desc;
3851     QString condition;
3852     if (urls.isEmpty()) {
3853         QAction *action = qobject_cast<QAction *>(sender());
3854         QStringList data = action->data().toStringList();
3855         params = data.at(0);
3856         if (data.count() > 1) desc = data.at(1);
3857         if (data.count() > 2) condition = data.at(2);
3858         urls << m_projectList->getConditionalUrls(condition);
3859         urls.removeAll(KUrl());
3860     }
3861     if (urls.isEmpty()) {
3862         m_messageLabel->setMessage(i18n("No clip to transcode"), ErrorMessage);
3863         return;
3864     }
3865     ClipTranscode *d = new ClipTranscode(urls, params, desc);
3866     connect(d, SIGNAL(addClip(KUrl)), this, SLOT(slotAddProjectClip(KUrl)));
3867     d->show();
3868     //QProcess::startDetached("ffmpeg", parameters);
3869 }
3870
3871 void MainWindow::slotTranscodeClip()
3872 {
3873     KUrl::List urls = KFileDialog::getOpenUrls(KUrl("kfiledialog:///projectfolder"));
3874     if (urls.isEmpty()) return;
3875     slotTranscode(urls);
3876 }
3877
3878 void MainWindow::slotSetDocumentRenderProfile(QMap <QString, QString> props)
3879 {
3880     if (m_activeDocument == NULL) return;
3881     QMapIterator<QString, QString> i(props);
3882     while (i.hasNext()) {
3883         i.next();
3884         m_activeDocument->setDocumentProperty(i.key(), i.value());
3885     }
3886     m_activeDocument->setModified(true);
3887 }
3888
3889
3890 void MainWindow::slotPrepareRendering(bool scriptExport, bool zoneOnly, const QString &chapterFile)
3891 {
3892     if (m_activeDocument == NULL || m_renderWidget == NULL) return;
3893     QString scriptPath;
3894     QString playlistPath;
3895     if (scriptExport) {
3896         bool ok;
3897         QString scriptsFolder = m_activeDocument->projectFolder().path(KUrl::AddTrailingSlash) + "scripts/";
3898         QString path = m_renderWidget->getFreeScriptName();
3899         scriptPath = QInputDialog::getText(this, i18n("Create Render Script"), i18n("Script name (will be saved in: %1)", scriptsFolder), QLineEdit::Normal, KUrl(path).fileName(), &ok);
3900         if (!ok || scriptPath.isEmpty()) return;
3901         scriptPath.prepend(scriptsFolder);
3902         QFile f(scriptPath);
3903         if (f.exists()) {
3904             if (KMessageBox::warningYesNo(this, i18n("Script file already exists. Do you want to overwrite it?")) != KMessageBox::Yes)
3905                 return;
3906         }
3907         playlistPath = scriptPath + ".mlt";
3908     } else {
3909         KTemporaryFile temp;
3910         temp.setAutoRemove(false);
3911         temp.setSuffix(".mlt");
3912         temp.open();
3913         playlistPath = temp.fileName();
3914     }
3915     QString playlistContent = m_projectMonitor->sceneList();
3916     if (!chapterFile.isEmpty()) {
3917         int in = 0;
3918         int out;
3919         if (!zoneOnly) out = (int) GenTime(m_activeDocument->projectDuration()).frames(m_activeDocument->fps());
3920         else {
3921             in = m_activeTimeline->inPoint();
3922             out = m_activeTimeline->outPoint();
3923         }
3924         QDomDocument doc;
3925         QDomElement chapters = doc.createElement("chapters");
3926         chapters.setAttribute("fps", m_activeDocument->fps());
3927         doc.appendChild(chapters);
3928
3929         QDomElement guidesxml = m_activeDocument->guidesXml();
3930         QDomNodeList nodes = guidesxml.elementsByTagName("guide");
3931         for (int i = 0; i < nodes.count(); i++) {
3932             QDomElement e = nodes.item(i).toElement();
3933             if (!e.isNull()) {
3934                 QString comment = e.attribute("comment");
3935                 int time = (int) GenTime(e.attribute("time").toDouble()).frames(m_activeDocument->fps());
3936                 if (time >= in && time < out) {
3937                     if (zoneOnly) time = time - in;
3938                     QDomElement chapter = doc.createElement("chapter");
3939                     chapters.appendChild(chapter);
3940                     chapter.setAttribute("title", comment);
3941                     chapter.setAttribute("time", time);
3942                 }
3943             }
3944         }
3945         if (chapters.childNodes().count() > 0) {
3946             if (m_activeTimeline->projectView()->hasGuide(out, 0) == -1) {
3947                 // Always insert a guide in pos 0
3948                 QDomElement chapter = doc.createElement("chapter");
3949                 chapters.insertBefore(chapter, QDomNode());
3950                 chapter.setAttribute("title", i18n("Start"));
3951                 chapter.setAttribute("time", "0");
3952             }
3953             // save chapters file
3954             QFile file(chapterFile);
3955             if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
3956                 kWarning() << "//////  ERROR writing DVD CHAPTER file: " << chapterFile;
3957             } else {
3958                 file.write(doc.toString().toUtf8());
3959                 if (file.error() != QFile::NoError) {
3960                     kWarning() << "//////  ERROR writing DVD CHAPTER file: " << chapterFile;
3961                 }
3962                 file.close();
3963             }
3964         }
3965     }
3966     bool exportAudio;
3967     if (m_renderWidget->automaticAudioExport()) {
3968         exportAudio = m_activeTimeline->checkProjectAudio();
3969     } else exportAudio = m_renderWidget->selectedAudioExport();
3970
3971     // Set playlist audio volume to 100%
3972     QDomDocument doc;
3973     doc.setContent(playlistContent);
3974     QDomElement tractor = doc.documentElement().firstChildElement("tractor");
3975     if (!tractor.isNull()) {
3976         QDomNodeList props = tractor.elementsByTagName("property");
3977         for (int i = 0; i < props.count(); i++) {
3978             if (props.at(i).toElement().attribute("name") == "meta.volume") {
3979                 props.at(i).firstChild().setNodeValue("1");
3980                 break;
3981             }
3982         }
3983     }
3984     
3985     // Do we want proxy rendering
3986     if (m_projectList->useProxy() && !m_renderWidget->proxyRendering()) {
3987         QString root = doc.documentElement().attribute("root");
3988
3989         // replace proxy clips with originals
3990         QMap <QString, QString> proxies = m_projectList->getProxies();
3991
3992         QDomNodeList producers = doc.elementsByTagName("producer");
3993         QString producerResource;
3994         QString suffix;
3995         for (uint n = 0; n < producers.length(); n++) {
3996             QDomElement e = producers.item(n).toElement();
3997             producerResource = EffectsList::property(e, "resource");
3998             if (producerResource.isEmpty()) continue;
3999             if (!producerResource.startsWith("/")) {
4000                 producerResource.prepend(root + "/");
4001             }
4002             if (producerResource.contains('?')) {
4003                 // slowmotion producer
4004                 suffix = "?" + producerResource.section('?', 1);
4005                 producerResource = producerResource.section('?', 0, 0);
4006             }
4007             else suffix.clear();
4008             if (!producerResource.isEmpty()) {
4009                 if (proxies.contains(producerResource)) {
4010                     EffectsList::setProperty(e, "resource", proxies.value(producerResource) + suffix);
4011                     // We need to delete the "aspect_ratio" property because proxy clips
4012                     // sometimes have different ratio than original clips
4013                     EffectsList::removeProperty(e, "aspect_ratio");
4014                     EffectsList::removeMetaProperties(e);
4015                 }
4016             }
4017         }
4018         
4019         /*QMapIterator<QString, QString> i(proxies);
4020         while (i.hasNext()) {
4021             i.next();
4022             // Replace all keys with their values (proxy path with original path)
4023             QString key = i.key();
4024             playlistContent.replace(key, i.value());
4025             if (!root.isEmpty() && key.startsWith(root)) {
4026                 // in case the resource path in MLT playlist is relative
4027                 key.remove(0, root.count() + 1);
4028                 playlistContent.replace(key, i.value());
4029             }
4030         }*/
4031     }
4032     playlistContent = doc.toString();
4033     
4034     // Do save scenelist
4035     QFile file(playlistPath);
4036     if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
4037         m_messageLabel->setMessage(i18n("Cannot write to file %1").arg(playlistPath), ErrorMessage);
4038         return;
4039     }
4040     file.write(playlistContent.toUtf8());
4041     if (file.error() != QFile::NoError) {
4042         m_messageLabel->setMessage(i18n("Cannot write to file %1").arg(playlistPath), ErrorMessage);
4043         file.close();
4044         return;
4045     }
4046     file.close();
4047     m_renderWidget->slotExport(scriptExport, m_activeTimeline->inPoint(), m_activeTimeline->outPoint(), playlistPath, scriptPath, exportAudio);
4048 }
4049
4050 void MainWindow::slotUpdateTimecodeFormat(int ix)
4051 {
4052     KdenliveSettings::setFrametimecode(ix == 1);
4053     m_clipMonitor->updateTimecodeFormat();
4054     m_projectMonitor->updateTimecodeFormat();
4055     m_transitionConfig->updateTimecodeFormat();
4056     m_effectStack->updateTimecodeFormat();
4057     //m_activeTimeline->projectView()->clearSelection();
4058     m_activeTimeline->updateRuler();
4059 }
4060
4061 void MainWindow::slotRemoveFocus()
4062 {
4063     statusBar()->setFocus();
4064     statusBar()->clearFocus();
4065 }
4066
4067 void MainWindow::slotRevert()
4068 {
4069     if (KMessageBox::warningContinueCancel(this, i18n("This will delete all changes made since you last saved your project. Are you sure you want to continue?"), i18n("Revert to last saved version")) == KMessageBox::Cancel) return;
4070     KUrl url = m_activeDocument->url();
4071     if (closeCurrentDocument(false))
4072         doOpenFile(url, NULL);
4073 }
4074
4075
4076 void MainWindow::slotShutdown()
4077 {
4078     if (m_activeDocument) m_activeDocument->setModified(false);
4079     // Call shutdown
4080     QDBusConnectionInterface* interface = QDBusConnection::sessionBus().interface();
4081     if (interface && interface->isServiceRegistered("org.kde.ksmserver")) {
4082         QDBusInterface smserver("org.kde.ksmserver", "/KSMServer", "org.kde.KSMServerInterface");
4083         smserver.call("logout", 1, 2, 2);
4084     } else if (interface && interface->isServiceRegistered("org.gnome.SessionManager")) {
4085         QDBusInterface smserver("org.gnome.SessionManager", "/org/gnome/SessionManager", "org.gnome.SessionManager");
4086         smserver.call("Shutdown");
4087     }
4088 }
4089
4090 void MainWindow::slotUpdateTrackInfo()
4091 {
4092     if (m_activeDocument)
4093         m_transitionConfig->updateProjectFormat(m_activeDocument->mltProfile(), m_activeDocument->timecode(), m_activeDocument->tracksList());
4094 }
4095
4096 void MainWindow::slotChangePalette(QAction *action, const QString &themename)
4097 {
4098     // Load the theme file
4099     QString theme;
4100     if (action == NULL) theme = themename;
4101     else theme = action->data().toString();
4102     KdenliveSettings::setColortheme(theme);
4103     // Make palette for all widgets.
4104     QPalette plt = kapp->palette();
4105     if (theme.isEmpty()) {
4106         plt = QApplication::desktop()->palette();
4107     } else {
4108         KSharedConfigPtr config = KSharedConfig::openConfig(theme);
4109
4110 #if KDE_IS_VERSION(4,6,3)
4111         plt = KGlobalSettings::createNewApplicationPalette(config);
4112 #else
4113         // Since there was a bug in createApplicationPalette in KDE < 4.6.3 we need
4114         // to do the palette loading stuff ourselves. (https://bugs.kde.org/show_bug.cgi?id=263497)     
4115         QPalette::ColorGroup states[3] = { QPalette::Active, QPalette::Inactive,
4116                                             QPalette::Disabled };
4117         // TT thinks tooltips shouldn't use active, so we use our active colors for all states
4118         KColorScheme schemeTooltip(QPalette::Active, KColorScheme::Tooltip, config);
4119  
4120         for ( int i = 0; i < 3 ; i++ ) {
4121             QPalette::ColorGroup state = states[i];
4122             KColorScheme schemeView(state, KColorScheme::View, config);
4123             KColorScheme schemeWindow(state, KColorScheme::Window, config);
4124             KColorScheme schemeButton(state, KColorScheme::Button, config);
4125             KColorScheme schemeSelection(state, KColorScheme::Selection, config);
4126  
4127             plt.setBrush( state, QPalette::WindowText, schemeWindow.foreground() );
4128             plt.setBrush( state, QPalette::Window, schemeWindow.background() );
4129             plt.setBrush( state, QPalette::Base, schemeView.background() );
4130             plt.setBrush( state, QPalette::Text, schemeView.foreground() );
4131             plt.setBrush( state, QPalette::Button, schemeButton.background() );
4132             plt.setBrush( state, QPalette::ButtonText, schemeButton.foreground() );
4133             plt.setBrush( state, QPalette::Highlight, schemeSelection.background() );
4134             plt.setBrush( state, QPalette::HighlightedText, schemeSelection.foreground() );
4135             plt.setBrush( state, QPalette::ToolTipBase, schemeTooltip.background() );
4136             plt.setBrush( state, QPalette::ToolTipText, schemeTooltip.foreground() );
4137  
4138             plt.setColor( state, QPalette::Light, schemeWindow.shade( KColorScheme::LightShade ) );
4139             plt.setColor( state, QPalette::Midlight, schemeWindow.shade( KColorScheme::MidlightShade ) );
4140             plt.setColor( state, QPalette::Mid, schemeWindow.shade( KColorScheme::MidShade ) );
4141             plt.setColor( state, QPalette::Dark, schemeWindow.shade( KColorScheme::DarkShade ) );
4142             plt.setColor( state, QPalette::Shadow, schemeWindow.shade( KColorScheme::ShadowShade ) );
4143  
4144             plt.setBrush( state, QPalette::AlternateBase, schemeView.background( KColorScheme::AlternateBackground) );
4145             plt.setBrush( state, QPalette::Link, schemeView.foreground( KColorScheme::LinkText ) );
4146             plt.setBrush( state, QPalette::LinkVisited, schemeView.foreground( KColorScheme::VisitedText ) );
4147         }
4148 #endif
4149     }
4150
4151     kapp->setPalette(plt);
4152     const QObjectList children = statusBar()->children();
4153
4154     foreach(QObject * child, children) {
4155         if (child->isWidgetType())
4156             ((QWidget*)child)->setPalette(plt);
4157         const QObjectList subchildren = child->children();
4158         foreach(QObject * subchild, subchildren) {
4159             if (subchild->isWidgetType())
4160                 ((QWidget*)subchild)->setPalette(plt);
4161         }
4162     }
4163     if (m_activeTimeline) {
4164         m_activeTimeline->projectView()->updatePalette();
4165     }
4166 }
4167
4168
4169 QPixmap MainWindow::createSchemePreviewIcon(const KSharedConfigPtr &config)
4170 {
4171     // code taken from kdebase/workspace/kcontrol/colors/colorscm.cpp
4172     const uchar bits1[] = { 0xff, 0xff, 0xff, 0x2c, 0x16, 0x0b };
4173     const uchar bits2[] = { 0x68, 0x34, 0x1a, 0xff, 0xff, 0xff };
4174     const QSize bitsSize(24, 2);
4175     const QBitmap b1 = QBitmap::fromData(bitsSize, bits1);
4176     const QBitmap b2 = QBitmap::fromData(bitsSize, bits2);
4177
4178     QPixmap pixmap(23, 16);
4179     pixmap.fill(Qt::black); // ### use some color other than black for borders?
4180
4181     KConfigGroup group(config, "WM");
4182     QPainter p(&pixmap);
4183     KColorScheme windowScheme(QPalette::Active, KColorScheme::Window, config);
4184     p.fillRect(1,  1, 7, 7, windowScheme.background());
4185     p.fillRect(2,  2, 5, 2, QBrush(windowScheme.foreground().color(), b1));
4186
4187     KColorScheme buttonScheme(QPalette::Active, KColorScheme::Button, config);
4188     p.fillRect(8,  1, 7, 7, buttonScheme.background());
4189     p.fillRect(9,  2, 5, 2, QBrush(buttonScheme.foreground().color(), b1));
4190
4191     p.fillRect(15,  1, 7, 7, group.readEntry("activeBackground", QColor(96, 148, 207)));
4192     p.fillRect(16,  2, 5, 2, QBrush(group.readEntry("activeForeground", QColor(255, 255, 255)), b1));
4193
4194     KColorScheme viewScheme(QPalette::Active, KColorScheme::View, config);
4195     p.fillRect(1,  8, 7, 7, viewScheme.background());
4196     p.fillRect(2, 12, 5, 2, QBrush(viewScheme.foreground().color(), b2));
4197
4198     KColorScheme selectionScheme(QPalette::Active, KColorScheme::Selection, config);
4199     p.fillRect(8,  8, 7, 7, selectionScheme.background());
4200     p.fillRect(9, 12, 5, 2, QBrush(selectionScheme.foreground().color(), b2));
4201
4202     p.fillRect(15,  8, 7, 7, group.readEntry("inactiveBackground", QColor(224, 223, 222)));
4203     p.fillRect(16, 12, 5, 2, QBrush(group.readEntry("inactiveForeground", QColor(20, 19, 18)), b2));
4204
4205     p.end();
4206     return pixmap;
4207 }
4208
4209 void MainWindow::slotSwitchMonitors()
4210 {
4211     m_monitorManager->slotSwitchMonitors(!m_clipMonitor->isActive());
4212     if (m_projectMonitor->isActive()) m_activeTimeline->projectView()->setFocus();
4213     else m_projectList->focusTree();
4214 }
4215
4216 void MainWindow::slotSwitchFullscreen()
4217 {
4218     if (m_projectMonitor->isActive()) m_projectMonitor->slotSwitchFullScreen();
4219     else m_clipMonitor->slotSwitchFullScreen();
4220 }
4221
4222 void MainWindow::slotInsertZoneToTree()
4223 {
4224     if (!m_clipMonitor->isActive() || m_clipMonitor->activeClip() == NULL) return;
4225     QStringList info = m_clipMonitor->getZoneInfo();
4226     m_projectList->slotAddClipCut(info.at(0), info.at(1).toInt(), info.at(2).toInt());
4227 }
4228
4229 void MainWindow::slotInsertZoneToTimeline()
4230 {
4231     if (m_activeTimeline == NULL || m_clipMonitor->activeClip() == NULL) return;
4232     QStringList info = m_clipMonitor->getZoneInfo();
4233     m_activeTimeline->projectView()->insertClipCut(m_clipMonitor->activeClip(), info.at(1).toInt(), info.at(2).toInt());
4234 }
4235
4236
4237 void MainWindow::slotDeleteProjectClips(QStringList ids, QMap<QString, QString> folderids)
4238 {
4239     if (m_activeDocument && m_activeTimeline) {
4240         if (!ids.isEmpty()) {
4241             for (int i = 0; i < ids.size(); ++i) {
4242                 m_activeTimeline->slotDeleteClip(ids.at(i));
4243             }
4244             m_activeDocument->clipManager()->slotDeleteClips(ids);
4245         }
4246         if (!folderids.isEmpty()) m_projectList->deleteProjectFolder(folderids);
4247         m_activeDocument->setModified(true);
4248     }
4249 }
4250
4251 void MainWindow::slotShowTitleBars(bool show)
4252 {
4253     QList <QDockWidget *> docks = findChildren<QDockWidget *>();
4254     for (int i = 0; i < docks.count(); i++) {
4255         QDockWidget* dock = docks.at(i);
4256         if (show) {
4257             dock->setTitleBarWidget(0);
4258         } else {
4259             if (!dock->isFloating()) {
4260                 dock->setTitleBarWidget(new QWidget);
4261             }
4262         }
4263     }
4264     KdenliveSettings::setShowtitlebars(show);
4265 }
4266
4267 void MainWindow::slotSwitchTitles()
4268 {
4269     slotShowTitleBars(!KdenliveSettings::showtitlebars());
4270 }
4271
4272 QString MainWindow::getMimeType(bool open)
4273 {
4274     QString mimetype = "application/x-kdenlive";
4275     KMimeType::Ptr mime = KMimeType::mimeType(mimetype);
4276     if (!mime) {
4277         mimetype = "*.kdenlive";
4278         if (open) mimetype.append(" *.tar.gz");
4279     }
4280     else if (open) mimetype.append(" application/x-compressed-tar");
4281     return mimetype;
4282 }
4283
4284 void MainWindow::slotMonitorRequestRenderFrame(bool request)
4285 {
4286     if (request) {
4287         m_projectMonitor->render->sendFrameForAnalysis = true;
4288         return;
4289     } else {
4290         for (int i = 0; i < m_gfxScopesList.count(); i++) {
4291             if (m_gfxScopesList.at(i)->isVisible() && tabifiedDockWidgets(m_gfxScopesList.at(i)).isEmpty() && static_cast<AbstractGfxScopeWidget *>(m_gfxScopesList.at(i)->widget())->autoRefreshEnabled()) {
4292                 request = true;
4293                 break;
4294             }
4295         }
4296     }
4297 #ifdef DEBUG_MAINW
4298     qDebug() << "Any scope accepting new frames? " << request;
4299 #endif
4300     if (!request) {
4301         m_projectMonitor->render->sendFrameForAnalysis = false;
4302     }
4303 }
4304
4305 void MainWindow::slotUpdateGfxScopeFrameRequest()
4306 {
4307     // We need a delay to make sure widgets are hidden after a close event for example
4308     QTimer::singleShot(500, this, SLOT(slotDoUpdateGfxScopeFrameRequest()));
4309 }
4310
4311 void MainWindow::slotDoUpdateGfxScopeFrameRequest()
4312 {
4313     // Check scopes
4314     bool request = false;
4315     for (int i = 0; i < m_gfxScopesList.count(); i++) {
4316         if (!m_gfxScopesList.at(i)->widget()->visibleRegion().isEmpty() && static_cast<AbstractGfxScopeWidget *>(m_gfxScopesList.at(i)->widget())->autoRefreshEnabled()) {
4317             kDebug() << "SCOPE VISIBLE: " << static_cast<AbstractGfxScopeWidget *>(m_gfxScopesList.at(i)->widget())->widgetName();
4318             request = true;
4319             break;
4320         }
4321     }
4322     if (!request) {
4323         if (!m_projectMonitor->effectSceneDisplayed()) {
4324             m_projectMonitor->render->sendFrameForAnalysis = false;
4325         }
4326         m_clipMonitor->render->sendFrameForAnalysis = false;
4327         if (m_recMonitor)
4328             m_recMonitor->analyseFrames(false);
4329     } else {
4330         m_projectMonitor->render->sendFrameForAnalysis = true;
4331         m_clipMonitor->render->sendFrameForAnalysis = true;
4332         if (m_recMonitor)
4333             m_recMonitor->analyseFrames(true);
4334     }
4335 }
4336
4337 void MainWindow::slotUpdateAudioScopeFrameRequest()
4338 {
4339     QTimer::singleShot(500, this, SLOT(slotDoUpdateAudioScopeFrameRequest()));
4340 }
4341
4342 void MainWindow::slotDoUpdateAudioScopeFrameRequest()
4343 {
4344     bool request = false;
4345     for (int i = 0; i < m_audioScopesList.count(); i++) {
4346         if (!m_audioScopesList.at(i)->visibleRegion().isEmpty() && m_audioScopesList.at(i)->autoRefreshEnabled()) {
4347             kDebug() << "AUDIO SCOPE VISIBLE: " << m_audioScopesList.at(i)->widgetName();
4348             request = true;
4349             break;
4350         }
4351     }
4352     // Handle audio signal separately (no common interface)
4353     if (!m_audiosignal->visibleRegion().isEmpty() && m_audiosignal->monitoringEnabled()) {
4354         kDebug() << "AUDIO SCOPE VISIBLE: " << "audiosignal";
4355         request = true;
4356     }
4357 #ifdef DEBUG_MAINW
4358     qDebug() << "Scopes Requesting Audio data: " << request;
4359 #endif
4360     KdenliveSettings::setMonitor_audio(request);
4361     m_monitorManager->slotUpdateAudioMonitoring();
4362 }
4363
4364 void MainWindow::slotUpdateColorScopes()
4365 {
4366     bool request = false;
4367     kDebug()<<"// UPDATE SCOPES";
4368     for (int i = 0; i < m_gfxScopesList.count(); i++) {
4369         // Check if we need the renderer to send a new frame for update
4370         if (!m_gfxScopesList.at(i)->widget()->visibleRegion().isEmpty() && !(static_cast<AbstractGfxScopeWidget *>(m_gfxScopesList.at(i)->widget())->autoRefreshEnabled())) request = true;
4371         static_cast<AbstractGfxScopeWidget *>(m_gfxScopesList.at(i)->widget())->slotActiveMonitorChanged();
4372     }
4373     if (request && m_monitorManager->activeRenderer()) {
4374         m_monitorManager->activeRenderer()->sendFrameUpdate();
4375     }
4376 }
4377
4378 void MainWindow::slotClearColorScopes()
4379 {
4380     for (int i = 0; i < m_gfxScopesList.count(); i++) {
4381         static_cast<AbstractGfxScopeWidget *>(m_gfxScopesList.at(i)->widget())->slotClearMonitor();
4382     }
4383 }
4384
4385 void MainWindow::slotOpenStopmotion()
4386 {
4387     if (m_stopmotion == NULL) {
4388         m_stopmotion = new StopmotionWidget(m_monitorManager, m_activeDocument->projectFolder(), m_stopmotion_actions->actions(), this);
4389         connect(m_stopmotion, SIGNAL(addOrUpdateSequence(const QString &)), m_projectList, SLOT(slotAddOrUpdateSequence(const QString)));
4390         //for (int i = 0; i < m_gfxScopesList.count(); i++) {
4391             // Check if we need the renderer to send a new frame for update
4392             /*if (!m_scopesList.at(i)->widget()->visibleRegion().isEmpty() && !(static_cast<AbstractScopeWidget *>(m_scopesList.at(i)->widget())->autoRefreshEnabled())) request = true;*/
4393             //connect(m_stopmotion, SIGNAL(gotFrame(QImage)), static_cast<AbstractGfxScopeWidget *>(m_gfxScopesList.at(i)->widget()), SLOT(slotRenderZoneUpdated(QImage)));
4394             //static_cast<AbstractScopeWidget *>(m_scopesList.at(i)->widget())->slotMonitorCapture();
4395         //}
4396     }
4397     m_stopmotion->show();
4398 }
4399
4400 void MainWindow::slotDeleteClip(const QString &id)
4401 {
4402     QList <ClipProperties *> list = findChildren<ClipProperties *>();
4403     for (int i = 0; i < list.size(); ++i) {
4404         list.at(i)->disableClipId(id);
4405     }
4406     m_projectList->slotDeleteClip(id);
4407 }
4408
4409 void MainWindow::slotUpdateProxySettings()
4410 {
4411     if (m_renderWidget) m_renderWidget->updateProxyConfig(m_projectList->useProxy());
4412     if (KdenliveSettings::enableproxy())
4413         KStandardDirs::makeDir(m_activeDocument->projectFolder().path(KUrl::AddTrailingSlash) + "proxy/");
4414     m_projectList->updateProxyConfig();
4415 }
4416
4417 void MainWindow::slotInsertNotesTimecode()
4418 {
4419     int frames = m_projectMonitor->render->seekPosition().frames(m_activeDocument->fps());
4420     QString position = m_activeDocument->timecode().getTimecodeFromFrames(frames);
4421     m_notesWidget->insertHtml("<a href=\"" + QString::number(frames) + "\">" + position + "</a> ");
4422 }
4423
4424 void MainWindow::slotArchiveProject()
4425 {
4426     QList <DocClipBase*> list = m_projectList->documentClipList();
4427     QDomDocument doc = m_activeDocument->xmlSceneList(m_projectMonitor->sceneList(), m_projectList->expandedFolders());
4428     ArchiveWidget *d = new ArchiveWidget(m_activeDocument->url().fileName(), doc, list, m_activeTimeline->projectView()->extractTransitionsLumas(), this);
4429     d->exec();
4430 }
4431
4432
4433 void MainWindow::slotOpenBackupDialog(const KUrl url)
4434 {
4435     KUrl projectFile;
4436     KUrl projectFolder;
4437     QString projectId;
4438     kDebug()<<"// BACKUP URL: "<<url.path();
4439     if (!url.isEmpty()) {
4440         // we could not open the project file, guess where the backups are
4441         projectFolder = KUrl(KdenliveSettings::defaultprojectfolder());
4442         projectFile = url;
4443     }
4444     else {
4445         projectFolder = m_activeDocument->projectFolder();
4446         projectFile = m_activeDocument->url();
4447         projectId = m_activeDocument->getDocumentProperty("documentid");
4448     }
4449
4450     BackupWidget *dia = new BackupWidget(projectFile, projectFolder, projectId, this);
4451     if (dia->exec() == QDialog::Accepted) {
4452         QString requestedBackup = dia->selectedFile();
4453         m_activeDocument->backupLastSavedVersion(projectFile.path());
4454         closeCurrentDocument(false);
4455         doOpenFile(KUrl(requestedBackup), NULL);
4456         m_activeDocument->setUrl(projectFile);
4457         m_activeDocument->setModified(true);
4458         setCaption(m_activeDocument->description());
4459     }
4460     delete dia;
4461 }
4462
4463 void MainWindow::slotElapsedTime()
4464 {
4465     kDebug()<<"-----------------------------------------\n"<<"Time elapsed: "<<m_timer.elapsed()<<"\n-------------------------";
4466 }
4467
4468 #include "mainwindow.moc"
4469
4470 #ifdef DEBUG_MAINW
4471 #undef DEBUG_MAINW
4472 #endif