mainwindow.cpp 164 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
/***************************************************************************
 *   Copyright (C) 2007 by Jean-Baptiste Mardelle (jb@kdenlive.org)        *
 *                                                                         *
 *   This program is free software; you can redistribute it and/or modify  *
 *   it under the terms of the GNU General Public License as published by  *
 *   the Free Software Foundation; either version 2 of the License, or     *
 *   (at your option) any later version.                                   *
 *                                                                         *
 *   This program is distributed in the hope that it will be useful,       *
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of        *
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *
 *   GNU General Public License for more details.                          *
 *                                                                         *
 *   You should have received a copy of the GNU General Public License     *
 *   along with this program; if not, write to the                         *
 *   Free Software Foundation, Inc.,                                       *
 *   51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA          *
 ***************************************************************************/

Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
20
#include "mainwindow.h"
21
#include "assets/assetpanel.hpp"
22
#include "bin/clipcreator.hpp"
23
#include "bin/generators/generators.h"
Nicolas Carion's avatar
Nicolas Carion committed
24
#include "bin/projectclip.h"
25
26
#include "bin/projectfolder.h"
#include "bin/projectitemmodel.h"
Nicolas Carion's avatar
Nicolas Carion committed
27
#include "core.h"
28
#include "dialogs/clipcreationdialog.h"
Nicolas Carion's avatar
Nicolas Carion committed
29
30
31
#include "dialogs/kdenlivesettingsdialog.h"
#include "dialogs/renderwidget.h"
#include "dialogs/wizard.h"
32
#include "doc/docundostack.hpp"
Nicolas Carion's avatar
Nicolas Carion committed
33
#include "doc/kdenlivedoc.h"
34
#include "effects/effectlist/view/effectlistwidget.hpp"
Nicolas Carion's avatar
Nicolas Carion committed
35
36
#include "effectslist/effectbasket.h"
#include "hidetitlebars.h"
37
38
#include "jobs/jobmanager.h"
#include "jobs/scenesplitjob.hpp"
39
#include "jobs/speedjob.hpp"
Nicolas Carion's avatar
Nicolas Carion committed
40
#include "jobs/stabilizejob.hpp"
41
#include "jobs/transcodeclipjob.h"
Nicolas Carion's avatar
Nicolas Carion committed
42
43
44
#include "kdenlivesettings.h"
#include "layoutmanagement.h"
#include "library/librarywidget.h"
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
45
#include "audiomixer/mixermanager.hpp"
Nicolas Carion's avatar
Nicolas Carion committed
46
47
48
49
50
51
#include "mainwindowadaptor.h"
#include "mltconnection.h"
#include "mltcontroller/clipcontroller.h"
#include "monitor/monitor.h"
#include "monitor/monitormanager.h"
#include "monitor/scopes/audiographspectrum.h"
Nicolas Carion's avatar
linting    
Nicolas Carion committed
52
#include "profiles/profilemodel.hpp"
Vincent PINON's avatar
Vincent PINON committed
53
#include "project/cliptranscode.h"
Vincent PINON's avatar
Vincent PINON committed
54
#include "project/dialogs/archivewidget.h"
Nicolas Carion's avatar
Nicolas Carion committed
55
56
#include "project/dialogs/projectsettings.h"
#include "project/projectcommands.h"
Till Theato's avatar
Till Theato committed
57
#include "project/projectmanager.h"
Nicolas Carion's avatar
Nicolas Carion committed
58
#include "scopes/scopemanager.h"
Nicolas Carion's avatar
linting    
Nicolas Carion committed
59
#include "timeline2/view/timelinecontroller.h"
60
#include "timeline2/view/timelinetabs.hpp"
61
#include "timeline2/view/timelinewidget.h"
Nicolas Carion's avatar
Nicolas Carion committed
62
63
64
65
66
#include "titler/titlewidget.h"
#include "transitions/transitionlist/view/transitionlistwidget.hpp"
#include "transitions/transitionsrepository.hpp"
#include "utils/resourcewidget.h"
#include "utils/thememanager.h"
67

68
#include "profiles/profilerepository.hpp"
69
70
#include "widgets/progressbutton.h"
#include <config-kdenlive.h>
71

72
#include "project/dialogs/temporarydata.h"
73

74
75
76
#ifdef USE_JOGSHUTTLE
#include "jogshuttle/jogmanager.h"
#endif
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
77

78
#include <KActionCategory>
Nicolas Carion's avatar
Nicolas Carion committed
79
#include <KActionCollection>
80
#include <KActionMenu>
Nicolas Carion's avatar
Nicolas Carion committed
81
82
#include <KColorScheme>
#include <KColorSchemeManager>
83
#include <KConfigDialog>
Nicolas Carion's avatar
Nicolas Carion committed
84
85
#include <KDualAction>
#include <KEditToolBar>
86
#include <KIconTheme>
Nicolas Carion's avatar
Nicolas Carion committed
87
88
#include <KMessageBox>
#include <KNotifyConfigWidget>
89
#include <KRecentDirs>
Nicolas Carion's avatar
Nicolas Carion committed
90
91
92
93
94
#include <KShortcutsDialog>
#include <KStandardAction>
#include <KToolBar>
#include <KXMLGUIFactory>
#include <klocalizedstring.h>
95
96
#include <kns3/downloaddialog.h>
#include <kns3/knewstuffaction.h>
Nicolas Carion's avatar
Nicolas Carion committed
97
#include <ktogglefullscreenaction.h>
98

Laurent Montel's avatar
Laurent Montel committed
99
#include "kdenlive_debug.h"
Nicolas Carion's avatar
Nicolas Carion committed
100
101
102
103
#include <QAction>
#include <QFileDialog>
#include <QMenu>
#include <QMenuBar>
104
#include <QStatusBar>
Nicolas Carion's avatar
Nicolas Carion committed
105
#include <QStyleFactory>
106
#include <QUndoGroup>
Vincent PINON's avatar
Vincent PINON committed
107
#include <KConfigGroup>
108
#include <QDesktopWidget>
Vincent PINON's avatar
Vincent PINON committed
109
110
#include <QDialogButtonBox>
#include <QPushButton>
111
#include <QScreen>
Nicolas Carion's avatar
Nicolas Carion committed
112
113
#include <QStandardPaths>
#include <QVBoxLayout>
114

115
static const char version[] = KDENLIVE_VERSION;
Nicolas Carion's avatar
Nicolas Carion committed
116
namespace Mlt {
117
class Producer;
Laurent Montel's avatar
Laurent Montel committed
118
}
119

Laurent Montel's avatar
Laurent Montel committed
120
121
QMap<QString, QImage> MainWindow::m_lumacache;
QMap<QString, QStringList> MainWindow::m_lumaFiles;
122

123
/*static bool sortByNames(const QPair<QString, QAction *> &a, const QPair<QString, QAction*> &b)
124
125
{
    return a.first < b.first;
126
}*/
127

Yuri Chornoivan's avatar
Yuri Chornoivan committed
128
// determine the default KDE style as defined BY THE USER
129
// (as opposed to whatever style KDE considers default)
Laurent Montel's avatar
Laurent Montel committed
130
static QString defaultStyle(const char *fallback = nullptr)
131
132
133
134
135
136
{
    KSharedConfigPtr kdeGlobals = KSharedConfig::openConfig(QStringLiteral("kdeglobals"), KConfig::NoGlobals);
    KConfigGroup cg(kdeGlobals, "KDE");
    return cg.readEntry("widgetStyle", fallback);
}

Nicolas Carion's avatar
Nicolas Carion committed
137
MainWindow::MainWindow(QWidget *parent)
138
    : KXmlGuiWindow(parent)
Nicolas Carion's avatar
Nicolas Carion committed
139

140
{
141
142
}

143
void MainWindow::init()
144
{
145
    QString desktopStyle = QApplication::style()->objectName();
146
147
148
149
150
    // Load themes
    auto themeManager = new ThemeManager(actionCollection());
    actionCollection()->addAction(QStringLiteral("themes_menu"), themeManager);
    connect(themeManager, &ThemeManager::themeChanged, this, &MainWindow::slotThemeChanged);

151
152
153
154
    if (!KdenliveSettings::widgetstyle().isEmpty() && QString::compare(desktopStyle, KdenliveSettings::widgetstyle(), Qt::CaseInsensitive) != 0) {
        // User wants a custom widget style, init
        doChangeStyle();
    }
Nicolas Carion's avatar
Nicolas Carion committed
155

156
    // Widget themes for non KDE users
Laurent Montel's avatar
Laurent Montel committed
157
    KActionMenu *stylesAction = new KActionMenu(i18n("Style"), this);
Nicolas Carion's avatar
Nicolas Carion committed
158
    auto *stylesGroup = new QActionGroup(stylesAction);
159

160
161
    // GTK theme does not work well with Kdenlive, and does not support color theming, so avoid it
    QStringList availableStyles = QStyleFactory::keys();
162
163
    if (KdenliveSettings::widgetstyle().isEmpty()) {
        // First run
164
        QStringList incompatibleStyles = {QStringLiteral("GTK+"), QStringLiteral("windowsvista"), QStringLiteral("Windows")};
165

166
        if (incompatibleStyles.contains(desktopStyle, Qt::CaseInsensitive)) {
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
167
            if (availableStyles.contains(QStringLiteral("breeze"), Qt::CaseInsensitive)) {
168
169
                // Auto switch to Breeze theme
                KdenliveSettings::setWidgetstyle(QStringLiteral("Breeze"));
170
		QApplication::setStyle(QStyleFactory::create(QStringLiteral("Breeze")));
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
171
            } else if (availableStyles.contains(QStringLiteral("fusion"), Qt::CaseInsensitive)) {
172
                KdenliveSettings::setWidgetstyle(QStringLiteral("Fusion"));
173
		QApplication::setStyle(QStyleFactory::create(QStringLiteral("Fusion")));
174
            }
175
176
        } else {
            KdenliveSettings::setWidgetstyle(QStringLiteral("Default"));
177
178
179
        }
    }

Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
180
181
    // Add default style action
    QAction *defaultStyle = new QAction(i18n("Default"), stylesGroup);
182
    defaultStyle->setData(QStringLiteral("Default"));
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
183
184
    defaultStyle->setCheckable(true);
    stylesAction->addAction(defaultStyle);
185
    if (KdenliveSettings::widgetstyle() == QLatin1String("Default") || KdenliveSettings::widgetstyle().isEmpty()) {
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
186
187
188
        defaultStyle->setChecked(true);
    }

Nicolas Carion's avatar
Nicolas Carion committed
189
    for (const QString &style : availableStyles) {
Nicolas Carion's avatar
Nicolas Carion committed
190
        auto *a = new QAction(style, stylesGroup);
191
192
        a->setCheckable(true);
        a->setData(style);
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
193
194
195
        if (KdenliveSettings::widgetstyle() == style) {
            a->setChecked(true);
        }
196
197
198
        stylesAction->addAction(a);
    }
    connect(stylesGroup, &QActionGroup::triggered, this, &MainWindow::slotChangeStyle);
Nicolas Carion's avatar
Nicolas Carion committed
199
    // QIcon::setThemeSearchPaths(QStringList() <<QStringLiteral(":/icons/"));
200
201

    new RenderingAdaptor(this);
202
    QString defaultProfile = KdenliveSettings::default_profile();
203
    pCore->setCurrentProfile(defaultProfile.isEmpty() ? ProjectManager::getDefaultProjectFormat() : defaultProfile);
204
    m_commandStack = new QUndoGroup();
205
206

    // If using a custom profile, make sure the file exists or fallback to default
207
208
    QString currentProfilePath = pCore->getCurrentProfile()->path();
    if (currentProfilePath.startsWith(QLatin1Char('/')) && !QFile::exists(currentProfilePath)) {
209
        KMessageBox::sorry(this, i18n("Cannot find your default profile, switching to ATSC 1080p 25"));
210
        pCore->setCurrentProfile(QStringLiteral("atsc_1080p_25"));
Laurent Montel's avatar
Laurent Montel committed
211
        KdenliveSettings::setDefault_profile(QStringLiteral("atsc_1080p_25"));
212
    }
213

214
    m_gpuAllowed = EffectsRepository::get()->hasInternalEffect(QStringLiteral("glsl.manager"));
215

216
    m_shortcutRemoveFocus = new QShortcut(QKeySequence(QStringLiteral("Esc")), this);
Laurent Montel's avatar
Laurent Montel committed
217
    connect(m_shortcutRemoveFocus, &QShortcut::activated, this, &MainWindow::slotRemoveFocus);
Simon Eugster's avatar
Simon Eugster committed
218

219
    /// Add Widgets
220
    setDockOptions(dockOptions() | QMainWindow::AllowNestedDocks | QMainWindow::AllowTabbedDocks);
221
    setDockOptions(dockOptions() | QMainWindow::GroupedDragging);
Nicolas Carion's avatar
Nicolas Carion committed
222
    setTabPosition(Qt::AllDockWidgetAreas, (QTabWidget::TabPosition)KdenliveSettings::tabposition());
223
    m_timelineToolBar = toolBar(QStringLiteral("timelineToolBar"));
224
    m_timelineToolBarContainer = new QWidget(this);
Nicolas Carion's avatar
Nicolas Carion committed
225
    auto *ctnLay = new QVBoxLayout;
226
227
    ctnLay->setSpacing(0);
    ctnLay->setContentsMargins(0, 0, 0, 0);
228
229
    m_timelineToolBarContainer->setLayout(ctnLay);
    ctnLay->addWidget(m_timelineToolBar);
230
231
232
233
    KSharedConfigPtr config = KSharedConfig::openConfig();
    KConfigGroup mainConfig(config, QStringLiteral("MainWindow"));
    KConfigGroup tbGroup(&mainConfig, QStringLiteral("Toolbar timelineToolBar"));
    m_timelineToolBar->applySettings(tbGroup);
234
235
236
237
238
    QFrame *fr = new QFrame(this);
    fr->setFrameShape(QFrame::HLine);
    fr->setMaximumHeight(1);
    fr->setLineWidth(1);
    ctnLay->addWidget(fr);
239
    setCentralWidget(m_timelineToolBarContainer);
240
    setupActions();
241

Laurent Montel's avatar
Laurent Montel committed
242
    QDockWidget *libraryDock = addDock(i18n("Library"), QStringLiteral("library"), pCore->library());
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
243
    QDockWidget *mixerDock = addDock(i18n("Audio Mixer"), QStringLiteral("mixer"), pCore->mixer());
244
245
246
    QAction *showMixer = new QAction(QIcon::fromTheme(QStringLiteral("adjustlevels")), i18n("Audio Mixer"), this);
    showMixer->setCheckable(true);
    addAction(QStringLiteral("audiomixer_button"), showMixer);
247
248
249
250
    connect(mixerDock, &QDockWidget::visibilityChanged, [&, showMixer](bool visible) {
        pCore->mixer()->connectMixer(visible);
        showMixer->setChecked(visible);
    });
251
252
253
254
255
256
257
    connect(showMixer, &QAction::triggered, [&, mixerDock]() {
        if (mixerDock->isVisible()) {
            mixerDock->close();
        } else {
            mixerDock->show();
        }
    });
Simon Eugster's avatar
Simon Eugster committed
258

Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
259
    m_clipMonitor = new Monitor(Kdenlive::ClipMonitor, pCore->monitorManager(), this);
260
    pCore->bin()->setMonitor(m_clipMonitor);
261
262
    connect(m_clipMonitor, &Monitor::addMarker, this, &MainWindow::slotAddMarkerGuideQuickly);
    connect(m_clipMonitor, &Monitor::deleteMarker, this, &MainWindow::slotDeleteClipMarker);
263
264
    connect(m_clipMonitor, &Monitor::seekToPreviousSnap, this, &MainWindow::slotSnapRewind);
    connect(m_clipMonitor, &Monitor::seekToNextSnap, this, &MainWindow::slotSnapForward);
265

266
    connect(pCore->bin(), &Bin::findInTimeline, this, &MainWindow::slotClipInTimeline, Qt::DirectConnection);
267
268
    connect(pCore->bin(), &Bin::setupTargets, this, [&] (bool hasVideo, QList <int> audioStreams) {
            getCurrentTimeline()->controller()->setTargetTracks(hasVideo, audioStreams);
269
270
        }
    );
271

Nicolas Carion's avatar
Nicolas Carion committed
272
    // TODO deprecated, replace with Bin methods if necessary
273
    /*connect(m_projectList, SIGNAL(loadingIsOver()), this, SLOT(slotElapsedTime()));
274
    connect(m_projectList, SIGNAL(updateRenderStatus()), this, SLOT(slotCheckRenderStatus()));
Laurent Montel's avatar
Laurent Montel committed
275
    connect(m_projectList, SIGNAL(updateProfile(QString)), this, SLOT(slotUpdateProjectProfile(QString)));
276
    connect(m_projectList, SIGNAL(refreshClip(QString,bool)), pCore->monitorManager(), SLOT(slotRefreshCurrentMonitor(QString)));
277
    connect(m_clipMonitor, SIGNAL(zoneUpdated(QPoint)), m_projectList, SLOT(slotUpdateClipCut(QPoint)));*/
278

Laurent Montel's avatar
Laurent Montel committed
279
    connect(m_clipMonitor, &Monitor::passKeyPress, this, &MainWindow::triggerKey);
280

Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
281
    m_projectMonitor = new Monitor(Kdenlive::ProjectMonitor, pCore->monitorManager(), this);
Laurent Montel's avatar
Laurent Montel committed
282
    connect(m_projectMonitor, &Monitor::passKeyPress, this, &MainWindow::triggerKey);
283
    connect(m_projectMonitor, &Monitor::addMarker, this, &MainWindow::slotAddMarkerGuideQuickly);
284
    connect(m_projectMonitor, &Monitor::deleteMarker, this, &MainWindow::slotDeleteGuide);
285
286
    connect(m_projectMonitor, &Monitor::seekToPreviousSnap, this, &MainWindow::slotSnapRewind);
    connect(m_projectMonitor, &Monitor::seekToNextSnap, this, &MainWindow::slotSnapForward);
287
    connect(m_loopClip, &QAction::triggered, m_projectMonitor, &Monitor::slotLoopClip);
288

289
    pCore->monitorManager()->initMonitors(m_clipMonitor, m_projectMonitor);
290
    connect(m_clipMonitor, &Monitor::addMasterEffect, pCore->bin(), &Bin::slotAddEffect);
291

292
293
    m_timelineTabs = new TimelineTabs(this);
    ctnLay->addWidget(m_timelineTabs);
294

295
296
297
298
299
300
301
    // Screen grab widget
    QWidget *grabWidget = new QWidget(this);
    QVBoxLayout *grabLayout = new QVBoxLayout;
    grabWidget->setLayout(grabLayout);
    QToolBar *recToolbar = new QToolBar(grabWidget);
    grabLayout->addWidget(recToolbar);
    grabLayout->addStretch(10);
302
    // Check number of monitors for FFmpeg screen capture
Vincent Pinon's avatar
Vincent Pinon committed
303
    int screens = QApplication::screens().count();
304
305
306
307
308
309
310
311
312
313
    if (screens > 1) {
        QComboBox *screenCombo = new QComboBox(recToolbar);
        for (int ix = 0; ix < screens; ix++) {
            screenCombo->addItem(i18n("Monitor %1", ix));
        }
        connect(screenCombo, static_cast<void (QComboBox::*)(int)>(&QComboBox::currentIndexChanged), m_clipMonitor, &Monitor::slotSetScreen);
        recToolbar->addWidget(screenCombo);
        // Update screen grab monitor choice in case we changed from fullscreen
        screenCombo->setEnabled(KdenliveSettings::grab_capture_type() == 0);
    }
314
315
316
317
318
319
320
321
322
    QAction *recAction = m_clipMonitor->recAction();
    addAction(QStringLiteral("screengrab_record"), recAction);
    recToolbar->addAction(recAction);
    QAction *recConfig = new QAction(QIcon::fromTheme(QStringLiteral("configure")), i18n("Configure Recording"), this);
    recToolbar->addAction(recConfig);
    connect(recConfig, &QAction::triggered, [&]() {
        pCore->showConfigDialog(4, 0);
    });
    QDockWidget *screenGrabDock = addDock(i18n("Screen Grab"), QStringLiteral("screengrab"), grabWidget);
323

324
325
    // Audio spectrum scope
    m_audioSpectrum = new AudioGraphSpectrum(pCore->monitorManager());
Laurent Montel's avatar
Laurent Montel committed
326
    QDockWidget *spectrumDock = addDock(i18n("Audio Spectrum"), QStringLiteral("audiospectrum"), m_audioSpectrum);
327
    // Close library and audiospectrum on first run
328
    screenGrabDock->close();
329
330
331
    libraryDock->close();
    spectrumDock->close();

332
    m_projectBinDock = addDock(i18n("Project Bin"), QStringLiteral("project_bin"), pCore->bin());
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
333

334
    m_assetPanel = new AssetPanel(this);
335
    m_effectStackDock = addDock(i18n("Properties"), QStringLiteral("effect_stack"), m_assetPanel);
336
337
    connect(m_assetPanel, &AssetPanel::doSplitEffect, m_projectMonitor, &Monitor::slotSwitchCompare);
    connect(m_assetPanel, &AssetPanel::doSplitBinEffect, m_clipMonitor, &Monitor::slotSwitchCompare);
338
339
340
341
    connect(m_assetPanel, &AssetPanel::switchCurrentComposition, [&](int cid, const QString &compositionId) {
        getMainTimeline()->controller()->getModel()->switchComposition(cid, compositionId);
    });

342
    connect(m_timelineTabs, &TimelineTabs::showTransitionModel, m_assetPanel, &AssetPanel::showTransition);
343
344
345
    connect(m_timelineTabs, &TimelineTabs::showTransitionModel, [&] () {
        m_effectStackDock->raise();
    });
346
    connect(m_timelineTabs, &TimelineTabs::showItemEffectStack, m_assetPanel, &AssetPanel::showEffectStack);
347
348
349
350
    connect(m_timelineTabs, &TimelineTabs::showItemEffectStack, [&] () {
        m_effectStackDock->raise();
    });

351
    connect(m_timelineTabs, &TimelineTabs::updateZoom, this, &MainWindow::updateZoomSlider);
352
    connect(pCore->bin(), &Bin::requestShowEffectStack, m_assetPanel, &AssetPanel::showEffectStack);
353
    connect(pCore->bin(), &Bin::requestShowEffectStack, [&] () {
354
355
        // Don't raise effect stack on clip bin in case it is docked with bin or clip monitor
        // m_effectStackDock->raise();
356
    });
357
    connect(this, &MainWindow::clearAssetPanel, m_assetPanel, &AssetPanel::clearAssetPanel);
358
359
360
    connect(m_assetPanel, &AssetPanel::seekToPos, [this](int pos) {
        ObjectId oId = m_assetPanel->effectStackOwner();
        switch (oId.first) {
Nicolas Carion's avatar
Nicolas Carion committed
361
362
363
        case ObjectType::TimelineTrack:
        case ObjectType::TimelineClip:
        case ObjectType::TimelineComposition:
364
        case ObjectType::Master:
Nicolas Carion's avatar
Nicolas Carion committed
365
366
367
368
369
370
371
372
            getCurrentTimeline()->controller()->setPosition(pos);
            break;
        case ObjectType::BinClip:
            m_clipMonitor->requestSeek(pos);
            break;
        default:
            qDebug() << "ERROR unhandled object type";
            break;
373
374
        }
    });
375

376
377
    m_effectList2 = new EffectListWidget(this);
    connect(m_effectList2, &EffectListWidget::activateAsset, pCore->projectManager(), &ProjectManager::activateAsset);
378
    connect(m_assetPanel, &AssetPanel::reloadEffect, m_effectList2, &EffectListWidget::reloadCustomEffect);
379
    m_effectListDock = addDock(i18n("Effects"), QStringLiteral("effect_list"), m_effectList2);
Simon Eugster's avatar
Simon Eugster committed
380

381
    m_transitionList2 = new TransitionListWidget(this);
382
    m_transitionListDock = addDock(i18n("Compositions"), QStringLiteral("transition_list"), m_transitionList2);
383

Simon Eugster's avatar
Simon Eugster committed
384
    // Add monitors here to keep them at the right of the window
385
386
    m_clipMonitorDock = addDock(i18n("Clip Monitor"), QStringLiteral("clip_monitor"), m_clipMonitor);
    m_projectMonitorDock = addDock(i18n("Project Monitor"), QStringLiteral("project_monitor"), m_projectMonitor);
Simon Eugster's avatar
Simon Eugster committed
387

388
    m_undoView = new QUndoView();
389
    m_undoView->setCleanIcon(QIcon::fromTheme(QStringLiteral("edit-clear")));
390
391
    m_undoView->setEmptyLabel(i18n("Clean"));
    m_undoView->setGroup(m_commandStack);
392
    m_undoViewDock = addDock(i18n("Undo History"), QStringLiteral("undo_history"), m_undoView);
393

394
    // Color and icon theme stuff
Laurent Montel's avatar
Laurent Montel committed
395
    connect(m_commandStack, &QUndoGroup::cleanChanged, m_saveAction, &QAction::setDisabled);
396
    addAction(QStringLiteral("styles_menu"), stylesAction);
Simon Eugster's avatar
Simon Eugster committed
397

398
399
400
401
402
    QAction *iconAction = new QAction(i18n("Force Breeze Icon Theme"), this);
    iconAction->setCheckable(true);
    iconAction->setChecked(KdenliveSettings::force_breeze());
    addAction(QStringLiteral("force_icon_theme"), iconAction);
    connect(iconAction, &QAction::triggered, this, &MainWindow::forceIconSet);
403

Simon Eugster's avatar
Simon Eugster committed
404
405
406
407
    // Close non-general docks for the initial layout
    // only show important ones
    m_undoViewDock->close();

408
    /// Tabify Widgets
409
    tabifyDockWidget(m_transitionListDock, m_effectListDock);
410
    tabifyDockWidget(m_effectStackDock, pCore->bin()->clipPropertiesDock());
Nicolas Carion's avatar
Nicolas Carion committed
411
    // tabifyDockWidget(m_effectListDock, m_effectStackDock);
412

413
    tabifyDockWidget(m_clipMonitorDock, m_projectMonitorDock);
414
    bool firstRun = readOptions();
Simon Eugster's avatar
Simon Eugster committed
415

416
    // Build effects menu
417
    m_effectsMenu = new QMenu(i18n("Add Effect"), this);
418
    m_effectActions = new KActionCategory(i18n("Effects"), actionCollection());
419
    m_effectList2->reloadEffectMenu(m_effectsMenu, m_effectActions);
420

421
422
423
    m_transitionsMenu = new QMenu(i18n("Add Transition"), this);
    m_transitionActions = new KActionCategory(i18n("Transitions"), actionCollection());

Nicolas Carion's avatar
Nicolas Carion committed
424
    auto *scmanager = new ScopeManager(this);
425

Till Theato's avatar
Till Theato committed
426
    new LayoutManagement(this);
427
    new HideTitleBars(this);
428
429
    m_extraFactory = new KXMLGUIClient(this);
    buildDynamicActions();
430

431
    // Create Effect Basket (dropdown list of favorites)
432
433
    m_effectBasket = new EffectBasket(this);
    connect(m_effectBasket, &EffectBasket::activateAsset, pCore->projectManager(), &ProjectManager::activateAsset);
434
    connect(m_effectList2, &EffectListWidget::reloadFavorites, m_effectBasket, &EffectBasket::slotReloadBasket);
Nicolas Carion's avatar
Nicolas Carion committed
435
    auto *widgetlist = new QWidgetAction(this);
436
    widgetlist->setDefaultWidget(m_effectBasket);
Nicolas Carion's avatar
Nicolas Carion committed
437
    // widgetlist->setText(i18n("Favorite Effects"));
438
    widgetlist->setToolTip(i18n("Favorite Effects"));
439
    widgetlist->setIcon(QIcon::fromTheme(QStringLiteral("favorite")));
Nicolas Carion's avatar
Nicolas Carion committed
440
    auto *menu = new QMenu(this);
441
442
    menu->addAction(widgetlist);

Nicolas Carion's avatar
Nicolas Carion committed
443
    auto *basketButton = new QToolButton(this);
444
    basketButton->setMenu(menu);
445
    basketButton->setToolButtonStyle(toolBar()->toolButtonStyle());
446
447
    basketButton->setDefaultAction(widgetlist);
    basketButton->setPopupMode(QToolButton::InstantPopup);
Nicolas Carion's avatar
Nicolas Carion committed
448
    // basketButton->setText(i18n("Favorite Effects"));
449
    basketButton->setToolTip(i18n("Favorite Effects"));
450
    basketButton->setIcon(QIcon::fromTheme(QStringLiteral("favorite")));
451

Nicolas Carion's avatar
Nicolas Carion committed
452
    auto *toolButtonAction = new QWidgetAction(this);
453
    toolButtonAction->setText(i18n("Favorite Effects"));
454
    toolButtonAction->setIcon(QIcon::fromTheme(QStringLiteral("favorite")));
455
    toolButtonAction->setDefaultWidget(basketButton);
456
    addAction(QStringLiteral("favorite_effects"), toolButtonAction);
Laurent Montel's avatar
Laurent Montel committed
457
    connect(toolButtonAction, &QAction::triggered, basketButton, &QToolButton::showMenu);
458
    connect(m_effectBasket, &EffectBasket::activateAsset, menu, &QMenu::close);
459

460
461
    // Render button
    ProgressButton *timelineRender = new ProgressButton(i18n("Render"), 100, this);
Nicolas Carion's avatar
Nicolas Carion committed
462
    auto *tlrMenu = new QMenu(this);
463
464
    timelineRender->setMenu(tlrMenu);
    connect(this, &MainWindow::setRenderProgress, timelineRender, &ProgressButton::setProgress);
Nicolas Carion's avatar
Nicolas Carion committed
465
    auto *renderButtonAction = new QWidgetAction(this);
466
    renderButtonAction->setText(i18n("Render Button"));
467
    renderButtonAction->setIcon(QIcon::fromTheme(QStringLiteral("media-record")));
468
469
470
    renderButtonAction->setDefaultWidget(timelineRender);
    addAction(QStringLiteral("project_render_button"), renderButtonAction);

471
    // Timeline preview button
472
    ProgressButton *timelinePreview = new ProgressButton(i18n("Rendering preview"), 1000, this);
Nicolas Carion's avatar
Nicolas Carion committed
473
    auto *tlMenu = new QMenu(this);
474
    timelinePreview->setMenu(tlMenu);
475
    connect(this, &MainWindow::setPreviewProgress, timelinePreview, &ProgressButton::setProgress);
Nicolas Carion's avatar
Nicolas Carion committed
476
    auto *previewButtonAction = new QWidgetAction(this);
477
    previewButtonAction->setText(i18n("Timeline Preview"));
478
    previewButtonAction->setIcon(QIcon::fromTheme(QStringLiteral("preview-render-on")));
479
    previewButtonAction->setDefaultWidget(timelinePreview);
480
    addAction(QStringLiteral("timeline_preview_button"), previewButtonAction);
481

482
    setupGUI(KXmlGuiWindow::ToolBar | KXmlGuiWindow::StatusBar | KXmlGuiWindow::Save | KXmlGuiWindow::Create);
483
    if (firstRun) {
484
        if (QScreen *current = QApplication::primaryScreen()) {
485
486
487
488
489
490
491
            if (current->availableSize().height() < 1000) {
                resize(current->availableSize());
            } else {
                resize(current->availableSize() / 1.5);
            }
        }
    }
492
493
494
    updateActionsToolTip();
    m_timelineToolBar->setToolButtonStyle(Qt::ToolButtonFollowStyle);
    m_timelineToolBar->setProperty("otherToolbar", true);
495
496
    timelinePreview->setToolButtonStyle(m_timelineToolBar->toolButtonStyle());
    connect(m_timelineToolBar, &QToolBar::toolButtonStyleChanged, timelinePreview, &ProgressButton::setToolButtonStyle);
497

498
    timelineRender->setToolButtonStyle(toolBar()->toolButtonStyle());
499
500
    /*ScriptingPart* sp = new ScriptingPart(this, QStringList());
    guiFactory()->addClient(sp);*/
501

502
    loadGenerators();
503
    loadDockActions();
504
    loadClipActions();
505
506
507
508
509
510
511
512
513
514
515
    QMenu *openGLMenu = static_cast<QMenu *>(factory()->container(QStringLiteral("qt_opengl"), this));
#if defined(Q_OS_WIN)
    connect(openGLMenu, &QMenu::triggered, [&](QAction *ac) {
        KdenliveSettings::setOpengl_backend(ac->data().toInt());
        if (KMessageBox::questionYesNo(this, i18n("Kdenlive needs to be restarted to change this setting. Do you want to proceed?")) != KMessageBox::Yes) {
            return;
        }
        slotRestart(false);
    });
#else
    if (openGLMenu) {
516
        openGLMenu->menuAction()->setVisible(false);;
517
518
    }
#endif
519
    // Connect monitor overlay info menu.
Laurent Montel's avatar
Laurent Montel committed
520
    QMenu *monitorOverlay = static_cast<QMenu *>(factory()->container(QStringLiteral("monitor_config_overlay"), this));
Laurent Montel's avatar
Laurent Montel committed
521
    connect(monitorOverlay, &QMenu::triggered, this, &MainWindow::slotSwitchMonitorOverlay);
522

Nicolas Carion's avatar
Nicolas Carion committed
523
524
525
526
    m_projectMonitor->setupMenu(static_cast<QMenu *>(factory()->container(QStringLiteral("monitor_go"), this)), monitorOverlay, m_playZone, m_loopZone, nullptr,
                                m_loopClip);
    m_clipMonitor->setupMenu(static_cast<QMenu *>(factory()->container(QStringLiteral("monitor_go"), this)), monitorOverlay, m_playZone, m_loopZone,
                             static_cast<QMenu *>(factory()->container(QStringLiteral("marker_menu"), this)));
527

Laurent Montel's avatar
Laurent Montel committed
528
    QMenu *clipInTimeline = static_cast<QMenu *>(factory()->container(QStringLiteral("clip_in_timeline"), this));
529
    clipInTimeline->setIcon(QIcon::fromTheme(QStringLiteral("go-jump")));
530
    pCore->bin()->setupGeneratorMenu();
Marco Gittler's avatar
Marco Gittler committed
531

Laurent Montel's avatar
Laurent Montel committed
532
    connect(pCore->monitorManager(), &MonitorManager::updateOverlayInfos, this, &MainWindow::slotUpdateMonitorOverlays);
533

534
    // Setup and fill effects and transitions menus.
Laurent Montel's avatar
Laurent Montel committed
535
    QMenu *m = static_cast<QMenu *>(factory()->container(QStringLiteral("video_effects_menu"), this));
536
537
    connect(m, &QMenu::triggered, this, &MainWindow::slotAddEffect);
    connect(m_effectsMenu, &QMenu::triggered, this, &MainWindow::slotAddEffect);
Laurent Montel's avatar
Laurent Montel committed
538
    connect(m_transitionsMenu, &QMenu::triggered, this, &MainWindow::slotAddTransition);
539

540
541
    m_timelineContextMenu = new QMenu(this);

542
543
    m_timelineContextMenu->addAction(actionCollection()->action(QStringLiteral("insert_space")));
    m_timelineContextMenu->addAction(actionCollection()->action(QStringLiteral("delete_space")));
544
    m_timelineContextMenu->addAction(actionCollection()->action(QStringLiteral("delete_space_all_tracks")));
545
546
    m_timelineContextMenu->addAction(actionCollection()->action(KStandardAction::name(KStandardAction::Paste)));

547
    // QMenu *markersMenu = static_cast<QMenu *>(factory()->container(QStringLiteral("marker_menu"), this));
548
549
550
551
552

    /*m_timelineClipActions->addMenu(markersMenu);
    m_timelineClipActions->addSeparator();
    m_timelineClipActions->addMenu(m_transitionsMenu);
    m_timelineClipActions->addMenu(m_effectsMenu);*/
Marco Gittler's avatar
Marco Gittler committed
553

554
    slotConnectMonitors();
555

556
    m_timelineToolBar->setToolButtonStyle(Qt::ToolButtonIconOnly);
557
    // TODO: let user select timeline toolbar toolbutton style
Nicolas Carion's avatar
Nicolas Carion committed
558
    // connect(toolBar(), &QToolBar::iconSizeChanged, m_timelineToolBar, &QToolBar::setToolButtonStyle);
559
560
    m_timelineToolBar->setContextMenuPolicy(Qt::CustomContextMenu);
    connect(m_timelineToolBar, &QWidget::customContextMenuRequested, this, &MainWindow::showTimelineToolbarMenu);
561
562

    QAction *prevRender = actionCollection()->action(QStringLiteral("prerender_timeline_zone"));
563
564
    QAction *stopPrevRender = actionCollection()->action(QStringLiteral("stop_prerender_timeline"));
    tlMenu->addAction(stopPrevRender);
565
566
    tlMenu->addAction(actionCollection()->action(QStringLiteral("set_render_timeline_zone")));
    tlMenu->addAction(actionCollection()->action(QStringLiteral("unset_render_timeline_zone")));
567
    tlMenu->addAction(actionCollection()->action(QStringLiteral("clear_render_timeline_zone")));
568
569

    // Automatic timeline preview action
570
    QAction *autoRender = new QAction(QIcon::fromTheme(QStringLiteral("view-refresh")), i18n("Automatic Preview"), this);
571
572
573
574
    autoRender->setCheckable(true);
    autoRender->setChecked(KdenliveSettings::autopreview());
    connect(autoRender, &QAction::triggered, this, &MainWindow::slotToggleAutoPreview);
    tlMenu->addAction(autoRender);
575
    tlMenu->addSeparator();
576
    tlMenu->addAction(actionCollection()->action(QStringLiteral("disable_preview")));
577
    tlMenu->addAction(actionCollection()->action(QStringLiteral("manage_cache")));
578
    timelinePreview->defineDefaultAction(prevRender, stopPrevRender);
579
    timelinePreview->setAutoRaise(true);
580

581
582
    QAction *showRender = actionCollection()->action(QStringLiteral("project_render"));
    tlrMenu->addAction(showRender);
583
    tlrMenu->addAction(actionCollection()->action(QStringLiteral("stop_project_render")));
584
    timelineRender->defineDefaultAction(showRender, showRender);
585
586
    timelineRender->setAutoRaise(true);

587
    // Populate encoding profiles
588
    KConfig conf(QStringLiteral("encodingprofiles.rc"), KConfig::CascadeConfig, QStandardPaths::AppDataLocation);
589
    /*KConfig conf(QStringLiteral("encodingprofiles.rc"), KConfig::CascadeConfig, QStandardPaths::AppDataLocation);
590
    if (KdenliveSettings::proxyparams().isEmpty() || KdenliveSettings::proxyextension().isEmpty()) {
591
        KConfigGroup group(&conf, "proxy");
Nicolas Carion's avatar
Nicolas Carion committed
592
        QMap<QString, QString> values = group.entryMap();
593
594
595
        QMapIterator<QString, QString> i(values);
        if (i.hasNext()) {
            i.next();
596
597
598
            QString proxystring = i.value();
            KdenliveSettings::setProxyparams(proxystring.section(QLatin1Char(';'), 0, 0));
            KdenliveSettings::setProxyextension(proxystring.section(QLatin1Char(';'), 1, 1));
599
        }
600
    }*/
601
    if (KdenliveSettings::v4l_parameters().isEmpty() || KdenliveSettings::v4l_extension().isEmpty()) {
602
        KConfigGroup group(&conf, "video4linux");
Nicolas Carion's avatar
Nicolas Carion committed
603
        QMap<QString, QString> values = group.entryMap();
604
605
606
        QMapIterator<QString, QString> i(values);
        if (i.hasNext()) {
            i.next();
607
608
609
            QString v4lstring = i.value();
            KdenliveSettings::setV4l_parameters(v4lstring.section(QLatin1Char(';'), 0, 0));
            KdenliveSettings::setV4l_extension(v4lstring.section(QLatin1Char(';'), 1, 1));
610
611
        }
    }
612
613
    if (KdenliveSettings::grab_parameters().isEmpty() || KdenliveSettings::grab_extension().isEmpty()) {
        KConfigGroup group(&conf, "screengrab");
Nicolas Carion's avatar
Nicolas Carion committed
614
        QMap<QString, QString> values = group.entryMap();
615
616
617
        QMapIterator<QString, QString> i(values);
        if (i.hasNext()) {
            i.next();
618
619
620
            QString grabstring = i.value();
            KdenliveSettings::setGrab_parameters(grabstring.section(QLatin1Char(';'), 0, 0));
            KdenliveSettings::setGrab_extension(grabstring.section(QLatin1Char(';'), 1, 1));
621
622
        }
    }
623
    if (KdenliveSettings::decklink_parameters().isEmpty() || KdenliveSettings::decklink_extension().isEmpty()) {
624
        KConfigGroup group(&conf, "decklink");
Nicolas Carion's avatar
Nicolas Carion committed
625
        QMap<QString, QString> values = group.entryMap();
626
627
628
        QMapIterator<QString, QString> i(values);
        if (i.hasNext()) {
            i.next();
629
630
631
            QString decklinkstring = i.value();
            KdenliveSettings::setDecklink_parameters(decklinkstring.section(QLatin1Char(';'), 0, 0));
            KdenliveSettings::setDecklink_extension(decklinkstring.section(QLatin1Char(';'), 1, 1));
632
633
        }
    }
Vincent Pinon's avatar
Vincent Pinon committed
634
635
636
    if (!QDir(KdenliveSettings::currenttmpfolder()).isReadable())
        KdenliveSettings::setCurrenttmpfolder(QStandardPaths::writableLocation(QStandardPaths::TempLocation));

Laurent Montel's avatar
Laurent Montel committed
637
    QTimer::singleShot(0, this, &MainWindow::GUISetupDone);
638

639
640
641
#ifdef USE_JOGSHUTTLE
    new JogManager(this);
#endif
642
    scmanager->slotCheckActiveScopes();
Nicolas Carion's avatar
format    
Nicolas Carion committed
643
    // m_messageLabel->setMessage(QStringLiteral("This is a beta version. Always backup your data"), MltError);
644
}
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
645

646
void MainWindow::slotThemeChanged(const QString &name)
647
{
648
    KSharedConfigPtr config = KSharedConfig::openConfig(name);
649
    QPalette plt = KColorScheme::createApplicationPalette(config);
650
    // qApp->setPalette(plt);
651
    // Required for qml palette change
652
    QGuiApplication::setPalette(plt);
653
654
655
656

    QColor background = plt.window().color();
    bool useDarkIcons = background.value() < 100;

657
658
    if (m_assetPanel) {
        m_assetPanel->updatePalette();
659
    }
660
661
662
663
664
665
666
667
    if (m_effectList2) {
        // Trigger a repaint to have icons adapted
        m_effectList2->reset();
    }
    if (m_transitionList2) {
        // Trigger a repaint to have icons adapted
        m_transitionList2->reset();
    }
Laurent Montel's avatar
Laurent Montel committed
668
669
670
671
672
673
    if (m_clipMonitor) {
        m_clipMonitor->setPalette(plt);
    }
    if (m_projectMonitor) {
        m_projectMonitor->setPalette(plt);
    }
674
675
    if (m_timelineTabs) {
        m_timelineTabs->setPalette(plt);
676
        getMainTimeline()->controller()->resetView();
677
678
679
    }
    if (m_audioSpectrum) {
        m_audioSpectrum->refreshPixmap();
680
    }
681

682
683
684
685
686
687
688
689
    KSharedConfigPtr kconfig = KSharedConfig::openConfig();
    KConfigGroup initialGroup(kconfig, "version");
    if (initialGroup.exists() && KdenliveSettings::force_breeze() && useDarkIcons != KdenliveSettings::use_dark_breeze()) {
        // We need to reload icon theme
        QIcon::setThemeName(useDarkIcons ? QStringLiteral("breeze-dark") : QStringLiteral("breeze"));
        KdenliveSettings::setUse_dark_breeze(useDarkIcons);
    }

Vincent Pinon's avatar
Vincent Pinon committed
690
#if (KXMLGUI_VERSION < QT_VERSION_CHECK(5, 23, 0)) || defined(Q_OS_WIN)
691
    // Not required anymore with auto colored icons since KF5 5.23
692
    if (m_themeInitialized && useDarkIcons != m_isDarkTheme) {
Vincent Pinon's avatar
Vincent Pinon committed
693
        QIcon::setThemeName(useDarkIcons ? QStringLiteral("breeze-dark") : QStringLiteral("breeze"));
694
695
696
        if (pCore->bin()) {
            pCore->bin()->refreshIcons();
        }
Laurent Montel's avatar
Laurent Montel committed
697
698
699
700
701
702
703
704
705
        if (m_clipMonitor) {
            m_clipMonitor->refreshIcons();
        }
        if (m_projectMonitor) {
            m_projectMonitor->refreshIcons();
        }
        if (pCore->monitorManager()) {
            pCore->monitorManager()->refreshIcons();
        }
706

Nicolas Carion's avatar
Nicolas Carion committed
707
        for (QAction *action : actionCollection()->actions()) {
708
            QIcon icon = action->icon();
Laurent Montel's avatar
Laurent Montel committed
709
710
711
            if (icon.isNull()) {
                continue;
            }
712
            QString iconName = icon.name();
713
            QIcon newIcon = QIcon::fromTheme(iconName);
Laurent Montel's avatar
Laurent Montel committed
714
715
716
            if (newIcon.isNull()) {
                continue;
            }
717
718
719
720
721
            action->setIcon(newIcon);
        }
    }
    m_themeInitialized = true;
    m_isDarkTheme = useDarkIcons;
722
#endif
723
724
}

725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
void MainWindow::updateActionsToolTip()
{
    // Add shortcut to action tooltips
    QList<KActionCollection *> collections = KActionCollection::allCollections();
    for (int i = 0; i < collections.count(); ++i) {
        KActionCollection *coll = collections.at(i);
        for (QAction *tempAction : coll->actions()) {
            // find the shortcut pattern and delete (note the preceding space in the RegEx)
            QString strippedTooltip = tempAction->toolTip().remove(QRegExp(QStringLiteral("\\s\\(.*\\)")));
            // append shortcut if it exists for action
            if (tempAction->shortcut() == QKeySequence(0)) {
                tempAction->setToolTip(strippedTooltip);
            } else {
                tempAction->setToolTip(strippedTooltip + QStringLiteral(" (") + tempAction->shortcut().toString() + QLatin1Char(')'));
            }
            connect(tempAction, &QAction::changed, this, &MainWindow::updateAction);
        }
    }
}

void MainWindow::updateAction()
{
Nicolas Carion's avatar
Nicolas Carion committed
747
    auto *action = qobject_cast<QAction *>(sender());
748
749
750
751
752
    QString toolTip = KLocalizedString::removeAcceleratorMarker(action->toolTip());
    QString strippedTooltip = toolTip.remove(QRegExp(QStringLiteral("\\s\\(.*\\)")));
    action->setToolTip(i18nc("@info:tooltip Tooltip of toolbar button", "%1 (%2)", strippedTooltip, action->shortcut().toString()));
}

753
MainWindow::~MainWindow()
754
{
755
    pCore->prepareShutdown();
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
756
    m_timelineTabs->disconnectTimeline(getMainTimeline());
757
    delete m_audioSpectrum;
Laurent Montel's avatar
Laurent Montel committed
758
759
760
761
762
763
    if (m_projectMonitor) {
        m_projectMonitor->stop();
    }
    if (m_clipMonitor) {
        m_clipMonitor->stop();
    }
764
    ClipController::