Kstars

schedulermodulestate.h
1/*
2 SPDX-FileCopyrightText: 2023 Wolfgang Reissenberger <sterne-jaeger@openfuture.de>
3
4 SPDX-License-Identifier: GPL-2.0-or-later
5*/
6
7#pragma once
8
9#include <QObject>
10#include <QProcess>
11#include "ekos/ekos.h"
12#include "indi/indiweather.h"
13#include "kstarsdatetime.h"
14#include "geolocation.h"
15#include "schedulertypes.h"
16#include "ekos/capture/capturetypes.h"
17#include <QDateTime>
18#include <QUrl>
19
20class SchedulerJob;
21
22namespace Ekos
23{
24
26class SchedulerJob;
27
28/**
29 * @class SchedulerState
30 * @brief The SchedulerState class holds all attributes defining the scheduler's state.
31 */
32class SchedulerModuleState : public QObject
33{
35
36 public:
37
38
39 SchedulerModuleState();
40
41 // ////////////////////////////////////////////////////////////////////
42 // Overall scheduler state
43 // ////////////////////////////////////////////////////////////////////
44 /**
45 * @brief init Set initial conditions that need to be set before starting
46 */
47 void init();
48
49 // ////////////////////////////////////////////////////////////////////
50 // profiles and scheduler jobs
51 // ////////////////////////////////////////////////////////////////////
52
53 const QString &currentProfile() const
54 {
55 return m_currentProfile;
56 }
57 /**
58 * @brief setCurrentProfile Set the current profile name.
59 * @param newName new current profile name
60 * @param signal send an update efent if true
61 */
62 void setCurrentProfile(const QString &newName, bool signal = true);
63
64 const QStringList &profiles() const
65 {
66 return m_profiles;
67 }
68 void updateProfiles(const QStringList &newProfiles);
69
70 SchedulerJob *activeJob() const
71 {
72 return m_activeJob;
73 }
74 void setActiveJob(SchedulerJob *newActiveJob);
75
76 /**
77 * @brief Return the master jobs only, slave jobs are ignored
78 */
79 QList<SchedulerJob *> leadJobs();
80
81 /**
82 * @brief Return the slave jobs only, master jobs are ignored
83 */
84 QList<SchedulerJob *> followerJobs();
85
86 QList<SchedulerJob *> &mutlableJobs()
87 {
88 return m_jobs;
89 }
90 const QList<SchedulerJob *> &jobs() const
91 {
92 return m_jobs;
93 }
94
95 void setJobs(QList<SchedulerJob *> &newJobs)
96 {
97 m_jobs = newJobs;
98 }
99
100 /**
101 * @brief updateStage Helper function that updates the stage label of the active job.
102 */
103 void updateJobStage(SchedulerJobStage stage);
104
105 /**
106 * @brief getJSONJobs get jobs in JSON format
107 * @return
108 */
109 QJsonArray getJSONJobs();
110
111
112 // ////////////////////////////////////////////////////////////////////
113 // state attributes accessors
114 // ////////////////////////////////////////////////////////////////////
115
116
117 bool dirty() const
118 {
119 return m_dirty;
120 }
121 void setDirty(bool value)
122 {
123 m_dirty = value;
124 }
125
126 // (coarse grained) execution state of the scheduler
127 const SchedulerState &schedulerState() const
128 {
129 return m_schedulerState;
130 }
131 void setSchedulerState(const SchedulerState &newState);
132
133 const StartupState &startupState() const
134 {
135 return m_startupState;
136 }
137
138 int currentPosition() const
139 {
140 return m_currentPosition;
141 }
142 void setCurrentPosition(int newCurrentPosition);
143
144 void setStartupState(StartupState state);
145
146 const QUrl &startupScriptURL() const
147 {
148 return m_startupScriptURL;
149 }
150 void setStartupScriptURL(const QUrl &newURL)
151 {
152 m_startupScriptURL = newURL;
153 }
154
155 const ShutdownState &shutdownState() const
156 {
157 return m_shutdownState;
158 }
159 void setShutdownState(ShutdownState state);
160
161 const QUrl &shutdownScriptURL() const
162 {
163 return m_shutdownScriptURL;
164 }
165 void setShutdownScriptURL(const QUrl &newShutdownScriptURL)
166 {
167 m_shutdownScriptURL = newShutdownScriptURL;
168 }
169
170 const ParkWaitState &parkWaitState() const
171 {
172 return m_parkWaitState;
173 }
174 void setParkWaitState(ParkWaitState state);
175
176 /**
177 * @brief True if the scheduler is between iterations and delaying longer than the typical update period.
178 */
179 bool currentlySleeping()
180 {
181 return iterationTimer().isActive() && timerState() == RUN_WAKEUP;
182 }
183
184 // ////////////////////////////////////////////////////////////////////
185 // job handling
186 // ////////////////////////////////////////////////////////////////////
187
188 /**
189 * @brief removeJob Remove the job from the job list at the given position.
190 * If this is the currently active job, don't remove it and return false.
191 * @return true iff removing succeeded
192 */
193 bool removeJob(const int currentRow);
194
195 /**
196 * @brief refreshSlaveLists walk through the jobs and update the slave lists
197 */
198 void refreshFollowerLists();
199
200 /**
201 * @brief walk through the job list and find the first master job
202 */
203 SchedulerJob *findLead(int position, bool upward = true);
204
205
206 // ////////////////////////////////////////////////////////////////////
207 // Controls for the preemptive shutdown feature.
208 // ////////////////////////////////////////////////////////////////////
209 // Is the scheduler shutting down until later when it will resume a job?
210 void enablePreemptiveShutdown(const QDateTime &wakeupTime);
211 void disablePreemptiveShutdown();
212 const QDateTime &preemptiveShutdownWakeupTime() const;
213 bool preemptiveShutdown() const;
214
215
216 // ////////////////////////////////////////////////////////////////////
217 // overall EKOS state
218 // ////////////////////////////////////////////////////////////////////
219 EkosState ekosState() const
220 {
221 return m_ekosState;
222 }
223 void setEkosState(EkosState state);
224 // last communication result with EKOS
225 CommunicationStatus ekosCommunicationStatus() const
226 {
227 return m_EkosCommunicationStatus;
228 }
229 void setEkosCommunicationStatus(CommunicationStatus newEkosCommunicationStatus)
230 {
231 m_EkosCommunicationStatus = newEkosCommunicationStatus;
232 }
233 // counter for failed EKOS connection attempts
234 void resetEkosConnectFailureCount(uint8_t newEkosConnectFailureCount = 0)
235 {
236 m_ekosConnectFailureCount = newEkosConnectFailureCount;
237 }
238 bool increaseEkosConnectFailureCount();
239
240 void resetParkingCapFailureCount(uint8_t value = 0)
241 {
242 m_parkingCapFailureCount = value;
243 }
244 bool increaseParkingCapFailureCount();
245 void resetParkingMountFailureCount(uint8_t value = 0)
246 {
247 m_parkingMountFailureCount = value;
248 }
249 bool increaseParkingMountFailureCount();
250 uint8_t parkingMountFailureCount() const
251 {
252 return m_parkingMountFailureCount;
253 }
254 void resetParkingDomeFailureCount(uint8_t value = 0)
255 {
256 m_parkingDomeFailureCount = value;
257 }
258 bool increaseParkingDomeFailureCount();
259
260 int indexToUse() const
261 {
262 return m_IndexToUse;
263 }
264 void setIndexToUse(int newIndexToUse)
265 {
266 m_IndexToUse = newIndexToUse;
267 }
268
269 int healpixToUse() const
270 {
271 return m_HealpixToUse;
272 }
273 void setHealpixToUse(int newHealpixToUse)
274 {
275 m_HealpixToUse = newHealpixToUse;
276 }
277
278 CapturedFramesMap &capturedFramesCount()
279 {
280 return m_CapturedFramesCount;
281 }
282
283 void setCapturedFramesCount(const CapturedFramesMap &newCapturedFramesCount)
284 {
285 m_CapturedFramesCount = newCapturedFramesCount;
286 }
287
288 void setWeatherGracePeriodActive(bool active)
289 {
290 m_WeatherGracePeriodActive = active;
291 }
292 bool weatherGracePeriodActive() const
293 {
294 return m_WeatherGracePeriodActive;
295 }
296
297 /**
298 * @brief resetFailureCounters Reset all failure counters
299 */
300 void resetFailureCounters();
301
302 // ////////////////////////////////////////////////////////////////////
303 // overall INDI state
304 // ////////////////////////////////////////////////////////////////////
305 INDIState indiState() const
306 {
307 return m_indiState;
308 }
309 void setIndiState(INDIState state);
310 // last communication result with INDI
311 CommunicationStatus indiCommunicationStatus() const
312 {
313 return m_INDICommunicationStatus;
314 }
315 void setIndiCommunicationStatus(CommunicationStatus newINDICommunicationStatus)
316 {
317 m_INDICommunicationStatus = newINDICommunicationStatus;
318 emit indiCommunicationStatusChanged(m_INDICommunicationStatus);
319 }
320 // counters for failed INDI connection attempts
321 void resetIndiConnectFailureCount(uint8_t newIndiConnectFailureCount = 0)
322 {
323 m_indiConnectFailureCount = newIndiConnectFailureCount;
324 }
325 bool increaseIndiConnectFailureCount();
326 /**
327 * @brief isINDIConnected Determines the status of the INDI connection.
328 * @return True if INDI connection is up and usable, else false.
329 */
330 bool isINDIConnected() const
331 {
332 return (indiCommunicationStatus() == Ekos::Success);
333 }
334 // ////////////////////////////////////////////////////////////////////
335 // device states
336 // ////////////////////////////////////////////////////////////////////
337 bool mountReady() const
338 {
339 return m_MountReady;
340 }
341 void setMountReady(bool readiness)
342 {
343 m_MountReady = readiness;
344 }
345 bool captureReady() const
346 {
347 return m_CaptureReady;
348 }
349 void setCaptureReady(bool readiness)
350 {
351 m_CaptureReady = readiness;
352 }
353 bool domeReady() const
354 {
355 return m_DomeReady;
356 }
357 void setDomeReady(bool readiness)
358 {
359 m_DomeReady = readiness;
360 }
361 bool capReady() const
362 {
363 return m_CapReady;
364 }
365 void setCapReady(bool readiness)
366 {
367 m_CapReady = readiness;
368 }
369
370 uint16_t captureBatch() const
371 {
372 return m_captureBatch;
373 }
374 void resetCaptureBatch()
375 {
376 m_captureBatch = 0;
377 }
378 uint16_t increaseCaptureBatch()
379 {
380 return m_captureBatch++;
381 }
382
383 uint8_t captureFailureCount() const
384 {
385 return m_captureFailureCount;
386 }
387 void resetCaptureFailureCount()
388 {
389 m_captureFailureCount = 0;
390 }
391 bool increaseCaptureFailureCount();
392
393 uint8_t focusFailureCount(const QString &trainname) const
394 {
395 return m_focusFailureCount[trainname];
396 }
397 void resetFocusFailureCount(const QString &trainname)
398 {
399 m_focusFailureCount[trainname] = 0;
400 }
401 void resetFocusFailureCount()
402 {
403 m_focusFailureCount.clear();
404 }
405 bool increaseFocusFailureCount(const QString &trainname);
406
407 bool increaseAllFocusFailureCounts();
408
409 bool autofocusCompleted(const QString &trainname) const;
410 void setAutofocusCompleted(const QString &trainname, bool value);
411 bool autofocusCompleted() const;
412
413 void resetAutofocusCompleted()
414 {
415 m_autofocusCompleted.clear();
416 }
417
418 uint8_t guideFailureCount() const
419 {
420 return m_guideFailureCount;
421 }
422 void resetGuideFailureCount()
423 {
424 m_guideFailureCount = 0;
425 }
426 bool increaseGuideFailureCount();
427
428 uint8_t alignFailureCount() const
429 {
430 return m_alignFailureCount;
431 }
432 void resetAlignFailureCount()
433 {
434 m_alignFailureCount = 0;
435 }
436 bool increaseAlignFailureCount();
437
438 int restartGuidingInterval() const
439 {
440 return m_restartGuidingInterval;
441 }
442
443 const KStarsDateTime &restartGuidingTime() const
444 {
445 return m_restartGuidingTime;
446 }
447
448 ISD::Weather::Status weatherStatus() const
449 {
450 return m_weatherStatus;
451 }
452 void setWeatherStatus(ISD::Weather::Status newWeatherStatus)
453 {
454 m_weatherStatus = newWeatherStatus;
455 }
456
457 // ////////////////////////////////////////////////////////////////////
458 // Timers and time
459 // ////////////////////////////////////////////////////////////////////
460 // Returns milliseconds since startCurrentOperationTImer() was called.
461 qint64 getCurrentOperationMsec() const;
462 // Starts the above operation timer.
463 // TODO. It would be better to make this a class and give each operation its own timer.
464 // TODO. These should be disabled once no longer relevant.
465 // These are implement with a KStarsDateTime instead of a QTimer type class
466 // so that the simulated clock can be used.
467 void startCurrentOperationTimer();
468
469 // Controls for the guiding timer, which restarts guiding after failure.
470 void cancelGuidingTimer();
471 bool isGuidingTimerActive();
472 void startGuidingTimer(int milliseconds);
473
474 /** @brief Setter used in testing to fix the local time. Otherwise getter gets from KStars instance. */
475 /** @{ */
476 static KStarsDateTime getLocalTime();
477 static void setLocalTime(KStarsDateTime *time)
478 {
479 storedLocalTime = time;
480 }
481 static bool hasLocalTime()
482 {
483 return storedLocalTime != nullptr;
484 }
485
486 /** @} */
487
488
489 // ////////////////////////////////////////////////////////////////////
490 // Astronomical calculations
491 // ////////////////////////////////////////////////////////////////////
492 /**
493 * @brief calculateDawnDusk find the next astronomical dawn and dusk after the current date and time of observation
494 */
495 static void calculateDawnDusk(QDateTime const &when, QDateTime &nDawn, QDateTime &nDusk);
496
497 /**
498 * @brief calculateDawnDusk Calculate dawn and dusk times for today
499 */
500 void calculateDawnDusk();
501
502 static QDateTime Dawn()
503 {
504 return m_Dawn;
505 }
506 static QDateTime Dusk()
507 {
508 return m_Dusk;
509 }
510 static QDateTime PreDawnDateTime()
511 {
512 return m_PreDawnDateTime;
513 }
514
515 /** @brief Setter used in testing to fix the geo location. Otherwise getter gets from KStars instance. */
516 /** @{ */
517 static const GeoLocation *getGeo();
518 static void setGeo(GeoLocation *geo)
519 {
520 storedGeo = geo;
521 }
522 static bool hasGeo();
523
524 // ////////////////////////////////////////////////////////////////////
525 // Scheduler iterations
526 // ////////////////////////////////////////////////////////////////////
527
528 // Setup the parameters for the next scheduler iteration.
529 // When milliseconds is not passed in, it uses m_UpdatePeriodMs.
530 void setupNextIteration(SchedulerTimerState nextState);
531 void setupNextIteration(SchedulerTimerState nextState, int milliseconds);
532
533 SchedulerTimerState timerState() const
534 {
535 return m_timerState;
536 }
537
538 void setTimerState(SchedulerTimerState newTimerState)
539 {
540 m_timerState = newTimerState;
541 }
542
543 QTimer &iterationTimer()
544 {
545 return m_iterationTimer;
546 }
547
548 bool iterationSetup() const
549 {
550 return m_iterationSetup;
551 }
552 void setIterationSetup(bool setup)
553 {
554 m_iterationSetup = setup;
555 }
556
557 qint64 startMSecs() const
558 {
559 return m_startMSecs;
560 }
561 void setStartMSecs(qint64 value)
562 {
563 m_startMSecs = value;
564 }
565 int increaseSchedulerIteration()
566 {
567 return ++m_schedulerIteration;
568 }
569 void resetSchedulerIteration()
570 {
571 m_schedulerIteration = 0;
572 }
573
574 int timerInterval() const
575 {
576 return m_timerInterval;
577 }
578 void setTimerInterval(int value)
579 {
580 m_timerInterval = value;
581 }
582
583 void setUpdatePeriodMs(int ms)
584 {
585 m_UpdatePeriodMs = ms;
586 }
587 int updatePeriodMs() const
588 {
589 return m_UpdatePeriodMs;
590 }
591
592 uint sequenceExecutionCounter() const
593 {
594 return m_sequenceExecutionCounter;
595 }
596 void resetSequenceExecutionCounter()
597 {
598 m_sequenceExecutionCounter = 1;
599 }
600 void increaseSequenceExecutionCounter()
601 {
602 m_sequenceExecutionCounter++;
603 }
604
605 static uint maxFailureAttempts();
606
607 QStringList &logText()
608 {
609 return m_logText;
610 }
611 QString getLogText()
612 {
613 return logText().join("\n");
614 }
615 void clearLog();
616
617 /**
618 * @brief checkRepeatSequence Check if the entire job sequence might be repeated
619 * @return true if the checkbox is set and the number of iterations is below the
620 * configured threshold
621 */
622 bool checkRepeatSequence();
623
624 void resetSolverIteration()
625 {
626 m_solverIteration = 0;
627 }
628 uint32_t increaseSolverIteration()
629 {
630 return ++m_solverIteration;
631 }
632
633 signals:
634 // ////////////////////////////////////////////////////////////////////
635 // communication with the UI
636 // ////////////////////////////////////////////////////////////////////
637 // State change of EKOS
638 void ekosStateChanged(EkosState state);
639 // State change of INDI
640 void indiStateChanged(INDIState state);
641 // High level INDI state changes
642 void indiCommunicationStatusChanged(CommunicationStatus status);
643 // overall scheduler state changed
644 void schedulerStateChanged(SchedulerState state);
645 // startup state
646 void startupStateChanged(StartupState state);
647 // shutdown state
648 void shutdownStateChanged(ShutdownState state);
649 // parking state
650 void parkWaitStateChanged(ParkWaitState state);
651 // profiles updated
652 void profilesChanged();
653 // current profile changed
654 void currentProfileChanged();
655 // new log text for the module log window
656 void newLog(const QString &text);
657 // current position in the job list changed
658 void currentPositionChanged(int pos);
659 // job stage of the current job changed
660 void jobStageChanged(SchedulerJobStage stage);
661 // night time calculation updated
662 void updateNightTime(SchedulerJob const * job = nullptr);
663
664
665 private:
666 // ////////////////////////////////////////////////////////////////////
667 // Scheduler jobs
668 // ////////////////////////////////////////////////////////////////////
669 // List of all jobs as entered by the user or file
671 // Active master job
672 SchedulerJob *m_activeJob { nullptr };
673
674 // ////////////////////////////////////////////////////////////////////
675 // state attributes
676 // ////////////////////////////////////////////////////////////////////
677 // coarse grained state describing the general execution state
678 SchedulerState m_schedulerState { SCHEDULER_IDLE };
679 // states of the scheduler startup
680 StartupState m_startupState { STARTUP_IDLE };
681 // Startup script URL
682 QUrl m_startupScriptURL;
683 // states of the scheduler shutdown
684 ShutdownState m_shutdownState { SHUTDOWN_IDLE };
685 // current position on the job list - necessary if there is no line selected in the
686 // UI, for example after deleting a row.
687 int m_currentPosition { -1 };
688 // Shutdown script URL
689 QUrl m_shutdownScriptURL;
690 // states of parking
691 ParkWaitState m_parkWaitState { PARKWAIT_IDLE };
692 // current profile
693 QString m_currentProfile;
694 // all profiles
695 QStringList m_profiles;
696 /// Store all log strings
697 QStringList m_logText;
698 // Was job modified and needs saving?
699 bool m_dirty { false };
700
701 // EKOS state describing whether EKOS is running (remember that the scheduler
702 // does not need EKOS running).
703 EkosState m_ekosState { EKOS_IDLE };
704 // Execution state of INDI
705 INDIState m_indiState { INDI_IDLE };
706 // Last communication result with EKOS and INDI
707 CommunicationStatus m_EkosCommunicationStatus { Ekos::Idle };
708 CommunicationStatus m_INDICommunicationStatus { Ekos::Idle };
709
710 // device readiness
711 bool m_MountReady { false };
712 bool m_CaptureReady { false };
713 bool m_DomeReady { false };
714 bool m_CapReady { false };
715
716 // Restricts (the internal solver) to using the index and healpix
717 // from the previous solve, if that solve was successful, when
718 // doing the pointing check. -1 means no restriction.
719 int m_IndexToUse { -1 };
720 int m_HealpixToUse { -1 };
721
722 // Check if initial autofocus is completed and do not run autofocus until
723 // there is a change is telescope position/alignment.
724 QMap <QString, bool> m_autofocusCompleted;
725
726 // Used when solving position every nth capture.
727 uint32_t m_solverIteration {0};
728
729
730 // Keep watch of weather status
731 ISD::Weather::Status m_weatherStatus { ISD::Weather::WEATHER_IDLE };
732
733 // ////////////////////////////////////////////////////////////////////
734 // counters
735 // ////////////////////////////////////////////////////////////////////
736 // count for job sequence iteration
737 uint m_sequenceExecutionCounter { 1 };
738 // Keep track of INDI connection failures
739 uint8_t m_indiConnectFailureCount { 0 };
740 // Keep track of Ekos connection failures
741 uint8_t m_ekosConnectFailureCount { 0 };
742 // failures parking dust cap
743 uint8_t m_parkingCapFailureCount { 0 };
744 // failures parking mount
745 uint8_t m_parkingMountFailureCount { 0 };
746 // failures parking dome
747 uint8_t m_parkingDomeFailureCount { 0 };
748 // How many repeated job batches did we complete thus far?
749 uint16_t m_captureBatch { 0 };
750 // Keep track of Ekos capture module failures
751 uint8_t m_captureFailureCount { 0 };
752 // Keep track of Ekos focus module failures
753 QMap <QString, uint8_t> m_focusFailureCount;
754 // Keep track of Ekos guide module failures
755 uint8_t m_guideFailureCount { 0 };
756 // Keep track of Ekos align module failures
757 uint8_t m_alignFailureCount { 0 };
758 // frames count for all signatures
759 CapturedFramesMap m_CapturedFramesCount;
760
761 // ////////////////////////////////////////////////////////////////////
762 // Scheduler iterations
763 // ////////////////////////////////////////////////////////////////////
764
765 // The type of scheduler iteration that should be run next.
766 SchedulerTimerState m_timerState { RUN_NOTHING };
767 // Variable keeping the number of millisconds the scheduler should wait
768 // after the current scheduler iteration.
769 int m_timerInterval { -1 };
770 // Whether the scheduler has been setup for the next iteration,
771 // that is, whether timerInterval and timerState have been set this iteration.
772 bool m_iterationSetup { false };
773 // The timer used to wakeup the scheduler between iterations.
774 QTimer m_iterationTimer;
775 // Counter for how many scheduler iterations have been processed.
776 int m_schedulerIteration { 0 };
777 // The time when the scheduler first started running iterations.
778 qint64 m_startMSecs { 0 };
779 // This is the time between typical scheduler iterations.
780 // The time can be modified for testing.
781 int m_UpdatePeriodMs = 1000;
782
783 // ////////////////////////////////////////////////////////////////////
784 // time and timers
785 // ////////////////////////////////////////////////////////////////////
786 // constants for current dawn and dusk
787 /// Store next dawn to calculate dark skies range
788 static QDateTime m_Dawn;
789 /// Store next dusk to calculate dark skies range
790 static QDateTime m_Dusk;
791 /// Pre-dawn is where we stop all jobs, it is a user-configurable value before Dawn.
792 static QDateTime m_PreDawnDateTime;
793 // Generic time to track timeout of current operation in progress.
794 // Used by startCurrentOperationTimer() and getCurrentOperationMsec().
795 KStarsDateTime currentOperationTime;
796 bool currentOperationTimeStarted { false };
797 // Delay for restarting the guider
798 int m_restartGuidingInterval { -1 };
799 KStarsDateTime m_restartGuidingTime;
800 // Used in testing, instead of KStars::Instance() resources
801 static KStarsDateTime *storedLocalTime;
802 // The various preemptiveShutdown states are controlled by this one variable.
803 QDateTime m_preemptiveShutdownWakeupTime;
804 // Are we in weather grace period?
805 bool m_WeatherGracePeriodActive {false};
806
807 // These are used in testing, instead of KStars::Instance() resources
808 static GeoLocation *storedGeo;
809
810};
811} // Ekos namespace
The SchedulerProcess class holds the entire business logic for controlling the execution of the EKOS ...
Contains all relevant information for specifying a location on Earth: City Name, State/Province name,...
Definition geolocation.h:28
Extension of QDateTime for KStars KStarsDateTime can represent the date/time as a Julian Day,...
The SchedulerState class holds all attributes defining the scheduler's state.
Ekos is an advanced Astrophotography tool for Linux.
Definition align.cpp:83
QMap< QString, uint16_t > CapturedFramesMap
mapping signature --> frames count
void clear()
Q_OBJECTQ_OBJECT
QString join(QChar separator) const const
bool isActive() const const
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Mar 28 2025 11:57:24 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.