main.cpp 13 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
/***************************************************************************
 *   Copyright (C) 2007 by Marco Gittler (g.marco@freenet.de)              *
 *   Copyright (C) 2008 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          *
 ***************************************************************************/

21
#include "core.h"
22
#include "dialogs/splash.hpp"
Nicolas Carion's avatar
Nicolas Carion committed
23
#include "logger.hpp"
Nicolas Carion's avatar
Nicolas Carion committed
24
#include <config-kdenlive.h>
25
26

#include <mlt++/Mlt.h>
27

28
29
#include "kxmlgui_version.h"

30
#include <KAboutData>
Nicolas Carion's avatar
Nicolas Carion committed
31
#include <KConfigGroup>
Vincent Pinon's avatar
Vincent Pinon committed
32
#ifdef USE_DRMINGW
33
#include <exchndl.h>
Vincent Pinon's avatar
Vincent Pinon committed
34
#elif defined(KF5_USE_CRASH)
35
#include <KCrash>
Vincent Pinon's avatar
Vincent Pinon committed
36
#endif
Vincent Pinon's avatar
Vincent Pinon committed
37

38
#include <KIconLoader>
39
#include <KSharedConfig>
40

41
#include "definitions.h"
Laurent Montel's avatar
Laurent Montel committed
42
#include "kdenlive_debug.h"
Nicolas Carion's avatar
Nicolas Carion committed
43
#include <KDBusService>
44
#include <KIconTheme>
Vincent Pinon's avatar
Vincent Pinon committed
45
#include <QResource>
46
47
#include <QApplication>
#include <QCommandLineOption>
Nicolas Carion's avatar
Nicolas Carion committed
48
49
#include <QCommandLineParser>
#include <QDir>
50
#include <QIcon>
Nicolas Carion's avatar
Nicolas Carion committed
51
#include <QProcess>
52
#include <QQmlEngine>
Nicolas Carion's avatar
Nicolas Carion committed
53
54
#include <QUrl> //new
#include <klocalizedstring.h>
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
55

Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
56
57
58
59
#ifdef Q_OS_WIN
extern "C"
{
    // Inform the driver we could make use of the discrete gpu
60
61
    // __declspec(dllexport) DWORD NvOptimusEnablement = 0x00000001;
    // __declspec(dllexport) int AmdPowerXpressRequestHighPerformance = 1;
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
62
63
64
65
}
#endif


66
67
int main(int argc, char *argv[])
{
68
69
70
#ifdef USE_DRMINGW
    ExcHndlInit();
#endif
71
    // Force QDomDocument to use a deterministic XML attribute order
72
    qSetGlobalQHashSeed(0);
73
74

    Logger::init();
75
    QCoreApplication::setAttribute(Qt::AA_UseHighDpiPixmaps);
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
76
    //TODO: is it a good option ?
77
    QCoreApplication::setAttribute(Qt::AA_ShareOpenGLContexts, true);
78

79
80
#if defined(Q_OS_UNIX) && !defined(Q_OS_MAC)
    QCoreApplication::setAttribute(Qt::AA_X11InitThreads);
81
#elif defined(Q_OS_WIN)
82
83
    KSharedConfigPtr configWin = KSharedConfig::openConfig("kdenliverc");
    KConfigGroup grp1(configWin, "misc");
84
85
86
87
88
89
    if (grp1.exists()) {
        int glMode = grp1.readEntry("opengl_backend", 0);
        if (glMode > 0) {
            QCoreApplication::setAttribute((Qt::ApplicationAttribute)glMode, true);
        }
    }
90
#endif
91
    QApplication app(argc, argv);
92
    app.setApplicationName(QStringLiteral("kdenlive"));
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
93
    app.setOrganizationDomain(QStringLiteral("kde.org"));
Laurent Montel's avatar
Minor    
Laurent Montel committed
94
    app.setWindowIcon(QIcon(QStringLiteral(":/pics/kdenlive.png")));
95
    KLocalizedString::setApplicationDomain("kdenlive");
Vincent Pinon's avatar
Vincent Pinon committed
96
#ifdef Q_OS_WIN
97
98
99
100
    qputenv("KDE_FORK_SLAVES", "1");
    QString path = qApp->applicationDirPath() + QLatin1Char(';') + qgetenv("PATH");
    qputenv("PATH", path.toUtf8().constData());

Vincent Pinon's avatar
Vincent Pinon committed
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
    const QStringList themes {"/icons/breeze/breeze-icons.rcc", "/icons/breeze-dark/breeze-icons-dark.rcc"};
    for(const QString theme : themes ) {
        const QString themePath = QStandardPaths::locate(QStandardPaths::AppDataLocation, theme);
        if (!themePath.isEmpty()) {
            const QString iconSubdir = theme.left(theme.lastIndexOf('/'));
            if (QResource::registerResource(themePath, iconSubdir)) {
                if (QFileInfo::exists(QLatin1Char(':') + iconSubdir + QStringLiteral("/index.theme"))) {
                    qDebug() << "Loaded icon theme:" << theme;
                } else {
                    qWarning() << "No index.theme found in" << theme;
                    QResource::unregisterResource(themePath, iconSubdir);
                }
            } else {
                qWarning() << "Invalid rcc file" << theme;
            }
        }
    }
#endif
119
    KSharedConfigPtr config = KSharedConfig::openConfig();
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
    KConfigGroup grp(config, "unmanaged");
    if (!grp.exists()) {
        QProcessEnvironment env = QProcessEnvironment::systemEnvironment();
        if (env.contains(QStringLiteral("XDG_CURRENT_DESKTOP")) && env.value(QStringLiteral("XDG_CURRENT_DESKTOP")).toLower() == QLatin1String("kde")) {
            qCDebug(KDENLIVE_LOG) << "KDE Desktop detected, using system icons";
        } else {
            // We are not on a KDE desktop, force breeze icon theme
            // Check if breeze theme is available
            QStringList iconThemes = KIconTheme::list();
            if (iconThemes.contains(QStringLiteral("breeze"))) {
                grp.writeEntry("force_breeze", true);
                grp.writeEntry("use_dark_breeze", true);
                qCDebug(KDENLIVE_LOG) << "Non KDE Desktop detected, forcing Breeze icon theme";
            }
        }
        // Set breeze dark as default on first opening
        KConfigGroup cg(config, "UiSettings");
        cg.writeEntry("ColorScheme", "Breeze Dark");
    }
139

140
141
142
143
    // Init DBus services
    KDBusService programDBusService;
    bool forceBreeze = grp.readEntry("force_breeze", QVariant(false)).toBool();
    if (forceBreeze) {
144
145
        bool darkBreeze = grp.readEntry("use_dark_breeze", QVariant(false)).toBool();
        QIcon::setThemeName(darkBreeze ? QStringLiteral("breeze-dark") : QStringLiteral("breeze"));
146
147
    }

Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
148
    // Create KAboutData
Nicolas Carion's avatar
Nicolas Carion committed
149
    KAboutData aboutData(QByteArray("kdenlive"), i18n("Kdenlive"), KDENLIVE_VERSION, i18n("An open source video editor."), KAboutLicense::GPL,
150
                         i18n("Copyright © 2007–2019 Kdenlive authors"), i18n("Please report bugs to http://bugs.kde.org"),
151
152
                         QStringLiteral("https://kdenlive.org"));
    aboutData.addAuthor(i18n("Jean-Baptiste Mardelle"), i18n("MLT and KDE SC 4 / KF5 port, main developer and maintainer"), QStringLiteral("jb@kdenlive.org"));
Vincent Pinon's avatar
Vincent Pinon committed
153
154
    aboutData.addAuthor(i18n("Nicolas Carion"), i18n("Code re-architecture & timeline rewrite"), QStringLiteral("french.ebook.lover@gmail.com"));
    aboutData.addAuthor(i18n("Vincent Pinon"), i18n("KF5 port, Windows cross-build, bugs fixing"), QStringLiteral("vpinon@kde.org"));
155
    aboutData.addAuthor(i18n("Laurent Montel"), i18n("Bugs fixing, clean up code, optimization etc."), QStringLiteral("montel@kde.org"));
Vincent Pinon's avatar
Vincent Pinon committed
156
157
    aboutData.addAuthor(i18n("Till Theato"), i18n("Bug fixing, etc."), QStringLiteral("root@ttill.de"));
    aboutData.addAuthor(i18n("Simon A. Eugster"), i18n("Color scopes, bug fixing, etc."), QStringLiteral("simon.eu@gmail.com"));
158
159
160
161
162
163
164
    aboutData.addAuthor(i18n("Marco Gittler"), i18n("MLT transitions and effects, timeline, audio thumbs"), QStringLiteral("g.marco@freenet.de"));
    aboutData.addAuthor(i18n("Dan Dennedy"), i18n("Bug fixing, etc."), QStringLiteral("dan@dennedy.org"));
    aboutData.addAuthor(i18n("Alberto Villa"), i18n("Bug fixing, logo, etc."), QStringLiteral("avilla@FreeBSD.org"));
    aboutData.addAuthor(i18n("Jean-Michel Poure"), i18n("Rendering profiles customization"), QStringLiteral("jm@poure.com"));
    aboutData.addAuthor(i18n("Ray Lehtiniemi"), i18n("Bug fixing, etc."), QStringLiteral("rayl@mail.com"));
    aboutData.addAuthor(i18n("Steve Guilford"), i18n("Bug fixing, etc."), QStringLiteral("s.guilford@dbplugins.com"));
    aboutData.addAuthor(i18n("Jason Wood"), i18n("Original KDE 3 version author (not active anymore)"), QStringLiteral("jasonwood@blueyonder.co.uk"));
165
    aboutData.addCredit(i18n("Nara Oliveira and Farid Abdelnour | Estúdio Gunga"), i18n("Kdenlive 16.08 icon"));
166
    aboutData.setTranslator(i18n("NAME OF TRANSLATORS"), i18n("EMAIL OF TRANSLATORS"));
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
167
    aboutData.setOrganizationDomain(QByteArray("kde.org"));
Nicolas Carion's avatar
Nicolas Carion committed
168
169
    aboutData.setOtherText(
        i18n("Using:\n<a href=\"https://mltframework.org\">MLT</a> version %1\n<a href=\"https://ffmpeg.org\">FFmpeg</a> libraries", mlt_version_get_string()));
170
    aboutData.setDesktopFileName(QStringLiteral("org.kde.kdenlive"));
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
171
172

    // Register about data
173
    KAboutData::setApplicationData(aboutData);
174

175
176
177
178
    // Add rcc stored icons to the search path so that we always find our icons
    KIconLoader *loader = KIconLoader::global();
    loader->reconfigure("kdenlive", QStringList() << QStringLiteral(":/pics"));

Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
179
180
181
182
    // Set app stuff from about data
    app.setApplicationDisplayName(aboutData.displayName());
    app.setOrganizationDomain(aboutData.organizationDomain());
    app.setApplicationVersion(aboutData.version());
183
    app.setAttribute(Qt::AA_DontCreateNativeWidgetSiblings, true);
184

Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
185
186
    // Create command line parser with options
    QCommandLineParser parser;
187
    aboutData.setupCommandLine(&parser);
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
188
189
190
    parser.setApplicationDescription(aboutData.shortDescription());
    parser.addVersionOption();
    parser.addHelpOption();
191

Nicolas Carion's avatar
Nicolas Carion committed
192
193
    parser.addOption(QCommandLineOption(QStringList() << QStringLiteral("config"), i18n("Set a custom config file name"), QStringLiteral("config")));
    parser.addOption(QCommandLineOption(QStringList() << QStringLiteral("mlt-path"), i18n("Set the path for MLT environment"), QStringLiteral("mlt-path")));
194
195
    parser.addOption(QCommandLineOption(QStringList() << QStringLiteral("mlt-log"), i18n("MLT log level"), QStringLiteral("verbose/debug")));
    parser.addOption(QCommandLineOption(QStringList() << QStringLiteral("i"), i18n("Comma separated list of clips to add"), QStringLiteral("clips")));
196
    parser.addPositionalArgument(QStringLiteral("file"), i18n("Document to open"));
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
197
198
199
200
201

    // Parse command line
    parser.process(app);
    aboutData.processCommandLine(&parser);

Vincent Pinon's avatar
Vincent Pinon committed
202
203
204
#ifdef USE_DRMINGW
    ExcHndlInit();
#elif defined(KF5_USE_CRASH)
205
    KCrash::initialize();
Vincent Pinon's avatar
Vincent Pinon committed
206
#endif
207

208
    //auto splash = new Splash(&app);
209
210
    //splash->show();
    //qApp->processEvents();
211

212
213
214
215
216
    qmlRegisterUncreatableMetaObject(PlaylistState::staticMetaObject, // static meta object
                                     "com.enums",                     // import statement
                                     1, 0,                            // major and minor version of the import
                                     "ClipState",                     // name in QML
                                     "Error: only enums");
217
    qmlRegisterUncreatableMetaObject(ClipType::staticMetaObject, // static meta object
218
219
220
                                     "com.enums",                // import statement
                                     1, 0,                       // major and minor version of the import
                                     "ProducerType",             // name in QML
221
                                     "Error: only enums");
222
    if (parser.value(QStringLiteral("mlt-log")) == QStringLiteral("verbose")) {
223
        mlt_log_set_level(MLT_LOG_VERBOSE);
224
    } else if (parser.value(QStringLiteral("mlt-log")) == QStringLiteral("debug")) {
225
        mlt_log_set_level(MLT_LOG_DEBUG);
226
    }
227
    QUrl url;
Nicolas Carion's avatar
Nicolas Carion committed
228
    if (parser.positionalArguments().count() != 0) {
229
230
231
232
233
        url = QUrl::fromLocalFile(parser.positionalArguments().at(0));
        // Make sure we get an absolute URL so that we can autosave correctly
        QString currentPath = QDir::currentPath();
        QUrl startup = QUrl::fromLocalFile(currentPath.endsWith(QDir::separator()) ? currentPath : currentPath + QDir::separator());
        url = startup.resolved(url);
234
    }
235
    Core::build(!parser.value(QStringLiteral("config")).isEmpty(), parser.value(QStringLiteral("mlt-path")));
236
    pCore->initGUI(url);
237
    //delete splash;
238
    //splash->endSplash();
239
    //qApp->processEvents();
240
    int result = app.exec();
241
    Core::clean();
242

243
    if (result == EXIT_RESTART || result == EXIT_CLEAN_RESTART) {
Laurent Montel's avatar
Laurent Montel committed
244
        qCDebug(KDENLIVE_LOG) << "restarting app";
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
        if (result == EXIT_CLEAN_RESTART) {
            // Delete config file
            KSharedConfigPtr config = KSharedConfig::openConfig();
            if (config->name().contains(QLatin1String("kdenlive"))) {
                // Make sure we delete our config file
                QFile f(QStandardPaths::locate(QStandardPaths::ConfigLocation, config->name(), QStandardPaths::LocateFile));
                if (f.exists()) {
                    qDebug()<<" = = = =\nGOT Deleted file: "<<f.fileName();
                    f.remove();
                }
            }
        }
        QStringList progArgs = QString(*argv).split(QLatin1Char(' '), QString::SkipEmptyParts);
        // Remove app name
        progArgs.takeFirst();
Nicolas Carion's avatar
Nicolas Carion committed
260
        auto *restart = new QProcess;
261
        restart->start(app.applicationFilePath(), progArgs);
262
263
264
265
        restart->waitForReadyRead();
        restart->waitForFinished(1000);
        result = EXIT_SUCCESS;
    }
266
    return result;
Jean-Baptiste Mardelle's avatar
Jean-Baptiste Mardelle committed
267
}