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