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