File indexing completed on 2025-10-19 04:39:58
0001 /* 0002 SPDX-FileCopyrightText: 2014 Till Theato <root@ttill.de> 0003 SPDX-License-Identifier: GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL 0004 */ 0005 0006 #pragma once 0007 0008 #include "definitions.h" 0009 #include "jobs/taskmanager.h" 0010 #include "kdenlivecore_export.h" 0011 #include "undohelper.hpp" 0012 #include "utils/timecode.h" 0013 #include <KSharedDataCache> 0014 #include <QColor> 0015 #include <QMutex> 0016 #include <QObject> 0017 #include <QPoint> 0018 #include <QTextEdit> 0019 #include <QThreadPool> 0020 #include <QUrl> 0021 #include <knewstuff_version.h> 0022 #include <memory> 0023 #include <unordered_set> 0024 0025 #include <mlt++/MltProfile.h> 0026 #include <mlt++/MltPlaylist.h> 0027 0028 class Bin; 0029 class DocUndoStack; 0030 class EffectStackModel; 0031 class KdenliveDoc; 0032 class LibraryWidget; 0033 class MainWindow; 0034 class MediaCapture; 0035 class MediaBrowser; 0036 class MixerManager; 0037 class Monitor; 0038 class MonitorManager; 0039 class ProfileModel; 0040 class ProjectItemModel; 0041 class ProjectManager; 0042 class SubtitleEdit; 0043 class SubtitleModel; 0044 class TextBasedEdit; 0045 class GuidesList; 0046 class TimeRemap; 0047 0048 namespace Mlt { 0049 class Repository; 0050 class Producer; 0051 } // namespace Mlt 0052 0053 #define EXIT_RESTART (42) 0054 #define EXIT_CLEAN_RESTART (43) 0055 #define pCore Core::self() 0056 0057 /** 0058 * @class Core 0059 * @brief Singleton that provides access to the different parts of Kdenlive 0060 * 0061 * Needs to be initialize before any widgets are created in MainWindow. 0062 * Plugins should be loaded after the widget setup. 0063 */ 0064 class /*KDENLIVECORE_EXPORT*/ Core : public QObject 0065 { 0066 Q_OBJECT 0067 0068 public: 0069 Core(const Core &) = delete; 0070 Core &operator=(const Core &) = delete; 0071 Core(Core &&) = delete; 0072 Core &operator=(Core &&) = delete; 0073 0074 ~Core() override; 0075 0076 /** 0077 * @brief Setup the basics of the application, in particular the connection 0078 * with Mlt 0079 * @param MltPath (optional) path to MLT environment 0080 */ 0081 static bool build(const QString &packageType, bool testMode = false); 0082 0083 void initHeadless(const QUrl &url); 0084 0085 /** 0086 * @brief Init the GUI part of the app and show the main window 0087 * @param inSandbox does the app run in a sanbox? If yes, we use App path to deduce 0088 * other binaries paths (melt, ffmpeg, etc) 0089 * @param MltPath 0090 * @param Url (optional) file to open 0091 * If Url is present, it will be opened, otherwise, if openlastproject is 0092 * set, latest project will be opened. If no file is open after trying this, 0093 * a default new file will be created. 0094 * @param clipsToLoad 0095 */ 0096 void initGUI(bool inSandbox, const QString &MltPath, const QUrl &Url, const QString &clipsToLoad = QString()); 0097 0098 /** @brief Returns a pointer to the singleton object. */ 0099 static std::unique_ptr<Core> &self(); 0100 0101 /** @brief Delete the global core instance */ 0102 static void clean(); 0103 0104 /** @brief Returns a pointer to the main window. */ 0105 MainWindow *window(); 0106 0107 /** @brief Open a file using an external app. */ 0108 QString openExternalApp(QString appPath, const QStringList args); 0109 0110 /** @brief Returns a pointer to the project manager. */ 0111 ProjectManager *projectManager(); 0112 /** @brief Returns a pointer to the current project. */ 0113 KdenliveDoc *currentDoc(); 0114 /** @brief Returns project's timecode. */ 0115 Timecode timecode() const; 0116 /** @brief Returns a pointer to the monitor manager. */ 0117 MonitorManager *monitorManager(); 0118 /** @brief Returns a pointer to the media browser widget. */ 0119 MediaBrowser *mediaBrowser(); 0120 /** @brief Returns a pointer to the view of the project bin. */ 0121 Bin *bin(); 0122 /** @brief Returns a pointer to the view of the active bin (or main bin on no focus). */ 0123 Bin *activeBin(); 0124 /** @brief Select a clip in the Bin from its id. */ 0125 void selectBinClip(const QString &id, bool activateMonitor = true, int frame = -1, const QPoint &zone = QPoint()); 0126 /** @brief Selects an item in the current timeline (clip, composition, subtitle). */ 0127 void selectTimelineItem(int id); 0128 /** @brief Returns a pointer to the model of the project bin. */ 0129 std::shared_ptr<ProjectItemModel> projectItemModel(); 0130 /** @brief Returns a pointer to the library. */ 0131 LibraryWidget *library(); 0132 /** @brief Returns a pointer to the subtitle edit. */ 0133 SubtitleEdit *subtitleWidget(); 0134 /** @brief Returns a pointer to the text based editing widget. */ 0135 TextBasedEdit *textEditWidget(); 0136 /** @brief Returns a pointer to the guides list widget. */ 0137 GuidesList *guidesList(); 0138 /** @brief Returns a pointer to the time remapping widget. */ 0139 TimeRemap *timeRemapWidget(); 0140 /** @brief Returns true if clip displayed in remap widget is the bin clip with id clipId. */ 0141 bool currentRemap(const QString &clipId); 0142 /** @brief Returns a pointer to the audio mixer. */ 0143 MixerManager *mixer(); 0144 ToolType::ProjectTool activeTool(); 0145 0146 /** @brief Returns a pointer to MLT's repository */ 0147 std::unique_ptr<Mlt::Repository> &getMltRepository(); 0148 0149 /** @brief Returns a pointer to the current profile */ 0150 std::unique_ptr<ProfileModel> &getCurrentProfile() const; 0151 const QString &getCurrentProfilePath() const; 0152 0153 /** @brief Define the active profile 0154 * @returns true if profile exists, false if not found 0155 */ 0156 bool setCurrentProfile(const QString profilePath); 0157 /** @brief Returns Sample Aspect Ratio of current profile */ 0158 double getCurrentSar() const; 0159 /** @brief Returns Display Aspect Ratio of current profile */ 0160 double getCurrentDar() const; 0161 0162 /** @brief Returns frame rate of current profile */ 0163 double getCurrentFps() const; 0164 0165 /** @brief Returns the frame size (width x height) of current profile */ 0166 QSize getCurrentFrameSize() const; 0167 /** @brief Returns the frame display size (width x height) of current profile */ 0168 QSize getCurrentFrameDisplaySize() const; 0169 /** @brief Request project monitor refresh, and restore previously active monitor */ 0170 void refreshProjectMonitorOnce(); 0171 /** @brief Request project monitor refresh if current position is inside range*/ 0172 void refreshProjectRange(QPair<int, int> range); 0173 /** @brief Request project monitor refresh if referenced item is under cursor */ 0174 void refreshProjectItem(const ObjectId &id); 0175 /** @brief Returns a reference to a monitor (clip or project monitor) */ 0176 Monitor *getMonitor(int id); 0177 /** @brief Seek a monitor to position */ 0178 void seekMonitor(int id, int position); 0179 /** @brief Returns timeline's active track info (position and tag) */ 0180 QPair <int,QString> currentTrackInfo() const; 0181 /** @brief This function must be called whenever the profile used changes */ 0182 void profileChanged(); 0183 0184 /** @brief Create and push and undo object based on the corresponding functions 0185 Note that if you class permits and requires it, you should use the macro PUSH_UNDO instead*/ 0186 void pushUndo(const Fun &undo, const Fun &redo, const QString &text); 0187 void pushUndo(QUndoCommand *command); 0188 /** @brief display a user info/warning message in statusbar */ 0189 void displayMessage(const QString &message, MessageType type, int timeout = -1); 0190 /** @brief display timeline selection info in statusbar */ 0191 void displaySelectionMessage(const QString &message); 0192 /** @brief Clear asset view if itemId is displayed. */ 0193 void clearAssetPanel(int itemId); 0194 /** @brief Returns the effectstack of a given bin clip. */ 0195 std::shared_ptr<EffectStackModel> getItemEffectStack(const QUuid &uuid, int itemType, int itemId); 0196 int getItemPosition(const ObjectId &id); 0197 /** @brief Get item in point. */ 0198 int getItemIn(const ObjectId &id); 0199 /** @brief Get item in point, possibly from another timeline. */ 0200 int getItemIn(const QUuid &uuid, const ObjectId &id); 0201 int getItemTrack(const ObjectId &id); 0202 int getItemDuration(const ObjectId &id); 0203 QSize getItemFrameSize(const ObjectId &id); 0204 /** @brief Returns the capabilities of a clip: AudioOnly, VideoOnly or Disabled if both are allowed */ 0205 PlaylistState::ClipState getItemState(const ObjectId &id); 0206 /** @brief Get a list of video track names with indexes */ 0207 QMap<int, QString> getTrackNames(bool videoOnly); 0208 /** @brief Returns the composition A track (MLT index / Track id) */ 0209 QPair<int, int> getCompositionATrack(int cid) const; 0210 void setCompositionATrack(int cid, int aTrack); 0211 /** @brief Return true if composition's a_track is automatic (no forced track) 0212 */ 0213 bool compositionAutoTrack(int cid) const; 0214 std::shared_ptr<DocUndoStack> undoStack(); 0215 double getClipSpeed(int id) const; 0216 /** @brief Mark an item as invalid for timeline preview */ 0217 void invalidateItem(ObjectId itemId); 0218 void invalidateRange(QPair<int, int>range); 0219 void prepareShutdown(); 0220 void finishShutdown(); 0221 /** the keyframe model changed (effect added, deleted, active effect changed), inform timeline */ 0222 void updateItemKeyframes(ObjectId id); 0223 /** A fade for clip id changed, update timeline */ 0224 void updateItemModel(ObjectId id, const QString &service); 0225 /** Show / hide keyframes for a timeline clip */ 0226 void showClipKeyframes(ObjectId id, bool enable); 0227 Mlt::Profile &thumbProfile(); 0228 /** @brief Returns the current project duration */ 0229 int projectDuration() const; 0230 /** @brief Returns true if current project has some rendered timeline preview */ 0231 bool hasTimelinePreview() const; 0232 /** @brief Returns monitor position */ 0233 int getMonitorPosition(Kdenlive::MonitorId id = Kdenlive::ProjectMonitor) const; 0234 /** @brief Handles audio and video capture **/ 0235 void startMediaCapture(int tid, bool, bool); 0236 void stopMediaCapture(int tid, bool, bool); 0237 QStringList getAudioCaptureDevices(); 0238 int getMediaCaptureState(); 0239 bool isMediaMonitoring() const; 0240 bool isMediaCapturing() const; 0241 MediaCapture *getAudioDevice(); 0242 /** @brief Returns Project Folder name for capture output location */ 0243 QString getProjectFolderName(bool folderForAudio = false); 0244 /** @brief Returns a timeline clip's bin id */ 0245 QString getTimelineClipBinId(int cid); 0246 /** @brief Returns all track ids in timeline */ 0247 std::unordered_set<QString> getAllTimelineTracksId(); 0248 /** @brief Returns a frame duration from a timecode */ 0249 int getDurationFromString(const QString &time); 0250 /** @brief An error occurred within a filter, inform user */ 0251 void processInvalidFilter(const QString &service, const QString &id, const QString &message); 0252 /** @brief Update current project's tags */ 0253 void updateProjectTags(int previousCount, const QMap <int, QStringList> &tags); 0254 /** @brief Returns the project profile */ 0255 Mlt::Profile &getProjectProfile(); 0256 /** @brief Returns the consumer profile, that will be scaled 0257 * according to preview settings. Should only be used on the consumer */ 0258 Mlt::Profile &getMonitorProfile(); 0259 /** @brief Returns a copy of current timeline's master playlist */ 0260 std::unique_ptr<Mlt::Producer> getMasterProducerInstance(); 0261 /** @brief Returns a copy of a track's playlist */ 0262 std::unique_ptr<Mlt::Producer> getTrackProducerInstance(int tid); 0263 /** @brief Returns the undo stack index (position). */ 0264 int undoIndex() const; 0265 /** @brief Enable / disable monitor multitrack view. Returns false if multitrack was not previously enabled */ 0266 bool enableMultiTrack(bool enable); 0267 /** @brief Returns number of audio channels for this project. */ 0268 int audioChannels(); 0269 /** @brief Add guides in the project. */ 0270 void addGuides(const QList <int> &guides); 0271 /** @brief Temporarily un/plug a list of clips in timeline. */ 0272 void temporaryUnplug(const QList<int> &clipIds, bool hide); 0273 /** @brief Transcode a video file. */ 0274 void transcodeFile(const QString &url); 0275 /** @brief Display key binding info in statusbar. */ 0276 void setWidgetKeyBinding(const QString &mess = QString()); 0277 KSharedDataCache audioThumbCache; 0278 /* @brief The thread job pool for clip jobs, allowing to set a max number of concurrent jobs */ 0279 TaskManager taskManager; 0280 /** @brief The number of clip load jobs changed */ 0281 void loadingClips(int, bool allowInterrupt = false); 0282 /** @brief Resize current mix item */ 0283 void resizeMix(int cid, int duration, MixAlignment align, int leftFrames = -1); 0284 /** @brief Get Mix cut pos (the duration of the mix on the right clip) */ 0285 int getMixCutPos(const ObjectId &) const; 0286 /** @brief Get alignment info for a mix item */ 0287 MixAlignment getMixAlign(const ObjectId &) const; 0288 /** @brief Closing current document, do some cleanup */ 0289 void cleanup(); 0290 /** @brief Clear clip reference in timeremap widget */ 0291 void clearTimeRemap(); 0292 /** @brief Create the dock widgets */ 0293 void buildDocks(); 0294 #if KNEWSTUFF_VERSION < QT_VERSION_CHECK(5, 98, 0) 0295 /** @brief Instantiates a "Get Hot New Stuff" dialog. 0296 * @param configFile configuration file for KNewStuff 0297 * @return number of installed items */ 0298 int getNewStuff(const QString &configFile); 0299 #endif 0300 /** @brief Get the frame size of the clip above a composition */ 0301 const QSize getCompositionSizeOnTrack(const ObjectId &id); 0302 void loadTimelinePreview(const QUuid uuid, const QString &chunks, const QString &dirty, bool enablePreview, Mlt::Playlist &playlist); 0303 /** @brief Returns true if the audio mixer widget is visible */ 0304 bool audioMixerVisible{false}; 0305 QString packageType() { return m_packageType; }; 0306 /** @brief Start / stop audio capture */ 0307 void switchCapture(); 0308 /** @brief Get the uuid of currently active timeline */ 0309 const QUuid currentTimelineId() const; 0310 /** @brief Update a sequence AV info (has audio/video) */ 0311 void updateSequenceAVType(const QUuid &uuid, int tracksCount); 0312 /** @brief A list of markers type categories {marker type, {color, category name}} */ 0313 struct MarkerCategory 0314 { 0315 QColor color; 0316 QString displayName; 0317 }; 0318 QMap<int, MarkerCategory> markerTypes; 0319 0320 private: 0321 explicit Core(const QString &packageType); 0322 static std::unique_ptr<Core> m_self; 0323 0324 /** @brief Makes sure Qt's locale and system locale settings match. */ 0325 void initLocale(); 0326 0327 MainWindow *m_mainWindow{nullptr}; 0328 ProjectManager *m_projectManager{nullptr}; 0329 MonitorManager *m_monitorManager{nullptr}; 0330 std::shared_ptr<ProjectItemModel> m_projectItemModel; 0331 LibraryWidget *m_library{nullptr}; 0332 SubtitleEdit *m_subtitleWidget{nullptr}; 0333 TextBasedEdit *m_textEditWidget{nullptr}; 0334 GuidesList *m_guidesList{nullptr}; 0335 TimeRemap *m_timeRemapWidget{nullptr}; 0336 MixerManager *m_mixerWidget{nullptr}; 0337 MediaBrowser *m_mediaBrowser{nullptr}; 0338 0339 /** @brief Current project's profile path */ 0340 QString m_currentProfile; 0341 0342 QString m_profile; 0343 QString m_packageType; 0344 Timecode m_timecode; 0345 Mlt::Profile m_thumbProfile; 0346 /** @brief Mlt profile used in the consumer 's monitors */ 0347 Mlt::Profile m_monitorProfile; 0348 /** @brief Mlt profile used to build the project's clips */ 0349 Mlt::Profile m_projectProfile; 0350 bool m_guiConstructed = false; 0351 /** @brief Check that the profile is valid (width is a multiple of 8 and height a multiple of 2 */ 0352 void checkProfileValidity(); 0353 std::unique_ptr<MediaCapture> m_capture; 0354 QUrl m_mediaCaptureFile; 0355 void resetThumbProfile(); 0356 0357 public Q_SLOTS: 0358 /** @brief Trigger (launch) an action by its actionCollection name */ 0359 void triggerAction(const QString &name); 0360 /** @brief Get an action's descriptive text by its actionCollection name */ 0361 const QString actionText(const QString &name); 0362 /** @brief Add an action to the app's actionCollection */ 0363 void addActionToCollection(const QString &name, QAction *action); 0364 /** @brief display a user info/warning message in the project bin */ 0365 void displayBinMessage(const QString &text, int type, const QList<QAction *> &actions = QList<QAction *>(), bool showClose = false, BinMessage::BinCategory messageCategory = BinMessage::BinCategory::NoMessage); 0366 void displayBinLogMessage(const QString &text, int type, const QString logInfo); 0367 /** @brief Create small thumbnails for luma used in compositions */ 0368 void buildLumaThumbs(const QStringList &values); 0369 /** @brief Try to find a display name for the given filename. 0370 * This is especially helpful for mlt's dynamically created luma files without thumb (luma01.pgm, luma02.pgm,...), 0371 * but also for others as it makes the visible name translatable. 0372 * @return The name that fits to the filename or if none is found the filename it self 0373 */ 0374 const QString nameForLumaFile(const QString &filename); 0375 /** @brief Set current project modified. */ 0376 void setDocumentModified(); 0377 /** @brief Show currently selected effect zone in timeline ruler. */ 0378 void showEffectZone(ObjectId id, QPair <int, int>inOut, bool checked); 0379 void updateMasterZones(); 0380 /** @brief Open the proxies test dialog. */ 0381 void testProxies(); 0382 /** @brief Refresh the monitor profile when project profile changes. */ 0383 void updateMonitorProfile(); 0384 /** @brief Add a new Bin Widget. */ 0385 void addBin(const QString &id = QString()); 0386 /** @brief Transcode a bin clip video. */ 0387 void transcodeFriendlyFile(const QString &binId, bool checkProfile); 0388 /** @brief Reset audio monitoring volume and channels. */ 0389 void resetAudioMonitoring(); 0390 /** @brief Start audio recording (after countdown). */ 0391 void startRecording(); 0392 /** @brief Show or hide track head audio rec controls. */ 0393 void monitorAudio(int tid, bool monitor); 0394 0395 Q_SIGNALS: 0396 void coreIsReady(); 0397 void updateLibraryPath(); 0398 //void updateMonitorProfile(); 0399 /** @brief Call config dialog on a selected page / tab */ 0400 void showConfigDialog(Kdenlive::ConfigPage, int); 0401 void finalizeRecording(const QString &captureFile); 0402 void autoScrollChanged(); 0403 /** @brief Update the message about the current loading progress */ 0404 void loadingMessageNewStage(const QString &message, int max = -1); 0405 /** @brief Increase the progress of the loading message by 1 */ 0406 void loadingMessageIncrease(); 0407 /** @brief Hide the message about the loading progress */ 0408 void loadingMessageHide(); 0409 /** @brief Opening finished, close splash screen */ 0410 void closeSplash(); 0411 /** @brief Trigger an update of the speech models list */ 0412 void voskModelUpdate(const QStringList models); 0413 /** @brief Update current effect zone */ 0414 void updateEffectZone(const QPoint p, bool withUndo); 0415 /** @brief The effect stask is about to be deleted, disconnect everything */ 0416 void disconnectEffectStack(); 0417 /** @brief Add a time remap effect to clip and show keyframes dialog */ 0418 void remapClip(int cid); 0419 /** @brief A monitor property changed, check if we need to reset */ 0420 void monitorProfileUpdated(); 0421 /** @brief Color theme changed, process refresh */ 0422 void updatePalette(); 0423 /** @brief Emitted when a clip is resized (to handle clip monitor inserted zones) */ 0424 void clipInstanceResized(const QString &binId); 0425 /** @brief Contains the project audio levels */ 0426 void audioLevelsAvailable(const QVector<double>& levels); 0427 /** @brief A frame was displayed in monitor, update audio mixer */ 0428 void updateMixerLevels(int pos); 0429 /** @brief Audio recording was started or stopped*/ 0430 void switchTimelineRecord(bool on); 0431 /** @brief Launch audio recording on track tid*/ 0432 void recordAudio(int tid, bool record); 0433 /** @brief Inform widgets that the project profile (and possibly fps) changed */ 0434 void updateProjectTimecode(); 0435 /** @brief Visible guide categories changed, reload snaps in timeline */ 0436 void refreshActiveGuides(); 0437 /** @brief The default marker category was changed, update guides list button */ 0438 void updateDefaultMarkerCategory(); 0439 /** @brief The default speech engine was changed */ 0440 void speechEngineChanged(); 0441 /** @brief Auto fit track height property changed, adjust size */ 0442 void autoTrackHeight(bool); 0443 /** @brief The number of missing clips in the project changed, inform user when rendering */ 0444 void gotMissingClipsCount(int total, int used); 0445 /** @brief Tell the current progress task to stop */ 0446 void stopProgressTask(); 0447 /** @brief A project clip was deleted */ 0448 void binClipDeleted(int cid); 0449 };