7#include "schedulerjob.h"
10#include "artificialhorizoncomponent.h"
11#include "kstarsdata.h"
12#include "skymapcomposite.h"
15#include "schedulermodulestate.h"
16#include "schedulerutils.h"
18#include "ksnotification.h"
20#include <ekos_scheduler_debug.h>
22#define BAD_SCORE -1000
23#define MIN_ALTITUDE 15.0
27GeoLocation *SchedulerJob::storedGeo =
nullptr;
28KStarsDateTime *SchedulerJob::storedLocalTime =
nullptr;
29ArtificialHorizon *SchedulerJob::storedHorizon =
nullptr;
52 return QString(
"????");
61 case SCHEDSTAGE_SLEWING:
63 case SCHEDSTAGE_SLEW_COMPLETE:
64 return "SLEW_COMPLETE";
65 case SCHEDSTAGE_FOCUSING:
67 case SCHEDSTAGE_FOCUS_COMPLETE:
68 return "FOCUS_COMPLETE";
69 case SCHEDSTAGE_ALIGNING:
71 case SCHEDSTAGE_ALIGN_COMPLETE:
72 return "ALIGN_COMPLETE";
73 case SCHEDSTAGE_RESLEWING:
75 case SCHEDSTAGE_RESLEWING_COMPLETE:
76 return "RESLEWING_COMPLETE";
77 case SCHEDSTAGE_POSTALIGN_FOCUSING:
78 return "POSTALIGN_FOCUSING";
79 case SCHEDSTAGE_POSTALIGN_FOCUSING_COMPLETE:
80 return "POSTALIGN_FOCUSING_COMPLETE";
81 case SCHEDSTAGE_GUIDING:
83 case SCHEDSTAGE_GUIDING_COMPLETE:
84 return "GUIDING_COMPLETE";
85 case SCHEDSTAGE_CAPTURING:
87 case SCHEDSTAGE_COMPLETE:
90 return QString(
"????");
93QString SchedulerJob::jobStartupConditionString(
StartupCondition condition)
const
100 return QString(
"AT %1").arg(getStartAtTime().
toString(
"MM/dd hh:mm"));
102 return QString(
"????");
109 case FINISH_SEQUENCE:
116 return QString(
"AT %1").arg(getFinishAtTime().
toString(
"MM/dd hh:mm"));
118 return QString(
"????");
121SchedulerJob::SchedulerJob()
123 if (KStarsData::Instance() !=
nullptr)
128SchedulerJob::SchedulerJob(KSMoon *moonPtr)
133void SchedulerJob::setName(
const QString &value)
138void SchedulerJob::setGroup(
const QString &value)
145void SchedulerJob::setCompletedIterations(
int value)
147 completedIterations = value;
148 if (completionCondition == FINISH_REPEAT)
149 setRepeatsRemaining(getRepeatsRequired() - completedIterations);
152KStarsDateTime SchedulerJob::getLocalTime()
154 return Ekos::SchedulerModuleState::getLocalTime();
157ArtificialHorizon
const *SchedulerJob::getHorizon()
160 return storedHorizon;
161 if (KStarsData::Instance() ==
nullptr || KStarsData::Instance()->skyComposite() ==
nullptr
162 || KStarsData::Instance()->skyComposite()->artificialHorizon() ==
nullptr)
164 return &KStarsData::Instance()->
skyComposite()->artificialHorizon()->getHorizon();
169 startupCondition = value;
172 if (value == START_ASAP)
173 startupTime = QDateTime();
176 setEstimatedTime(estimatedTime);
179 SchedulerModuleState::calculateDawnDusk(startupTime, nextDawn, nextDusk);
182void SchedulerJob::setStartupTime(
const QDateTime &value,
bool refreshDawnDusk)
188 startupCondition = START_AT;
190 startupCondition = fileStartupCondition;
193 altitudeAtStartup = SchedulerUtils::findAltitude(getTargetCoords(), startupTime, &settingAtStartup);
196 setEstimatedTime(estimatedTime);
199 for (
auto follower : followerJobs())
200 follower->setStartupTime(value,
false);
204 SchedulerModuleState::calculateDawnDusk(startupTime, nextDawn, nextDusk);
207void SchedulerJob::setSequenceFile(
const QUrl &value)
209 sequenceFile = value;
212void SchedulerJob::setFITSFile(
const QUrl &value)
217void SchedulerJob::setMinAltitude(
const double &value)
222bool SchedulerJob::hasAltitudeConstraint()
const
224 return hasMinAltitude() ||
225 (getEnforceArtificialHorizon() && (getHorizon() !=
nullptr) && getHorizon()->altitudeConstraintsExist()) ||
226 (Options::enableAltitudeLimits() &&
227 (Options::minimumAltLimit() > 0 ||
228 Options::maximumAltLimit() < 90));
231void SchedulerJob::setMinMoonSeparation(
const double &value)
233 minMoonSeparation = value;
236void SchedulerJob::setMaxMoonAltitude(
const double &value)
238 maxMoonAltitude = value;
241void SchedulerJob::setStopTime(
const QDateTime &value)
248 altitudeAtStop = SchedulerUtils::findAltitude(getTargetCoords(), stopTime, &settingAtStop);
251 for (
auto follower : followerJobs())
254 if (follower->getStartupTime().isValid() && value.
isValid()
255 && (follower->getEstimatedTime() < 0 || follower->getEstimatedTime() > getEstimatedTime()))
256 follower->setEstimatedTime(getEstimatedTime());
261void SchedulerJob::setFinishAtTime(
const QDateTime &value)
263 setStopTime(QDateTime());
268 setCompletionCondition(FINISH_AT);
269 finishAtTime = value;
270 setEstimatedTime(-1);
273 else if (FINISH_LOOP == completionCondition)
275 finishAtTime = QDateTime();
276 setEstimatedTime(-1);
279 else if (startupTime.isValid())
281 finishAtTime = startupTime.addSecs(estimatedTime);
284 else setEstimatedTime(estimatedTime);
288 Q_ASSERT_X(finishAtTime.isValid() ?
289 (FINISH_AT == completionCondition || FINISH_REPEAT == completionCondition || FINISH_SEQUENCE == completionCondition) :
290 FINISH_LOOP == completionCondition,
291 __FUNCTION__,
"Valid completion time implies job is FINISH_AT/REPEAT/SEQUENCE, else job is FINISH_LOOP.");
296 completionCondition = value;
299 switch (completionCondition)
302 setFinishAtTime(QDateTime());
305 if (0 < getRepeatsRequired())
306 setRepeatsRequired(0);
309 case FINISH_SEQUENCE:
310 if (1 != getRepeatsRequired())
311 setRepeatsRequired(1);
315 if (0 == getRepeatsRequired())
316 setRepeatsRequired(1);
324void SchedulerJob::setStepPipeline(
const StepPipeline &value)
326 stepPipeline = value;
332 stateTime = getLocalTime();
338 lastErrorTime = getLocalTime();
339 KSNotification::event(QLatin1String(
"EkosSchedulerJobFail"),
i18n(
"Ekos job failed (%1)", getName()),
340 KSNotification::Scheduler,
341 KSNotification::Alert);
346 setStartupCondition(fileStartupCondition);
347 setStartupTime(startAtTime);
348 setEstimatedTime(-1);
352 lastAbortTime = getLocalTime();
353 setStartupCondition(fileStartupCondition);
362 setFollowerState(value, force);
367 for (
auto follower : followerJobs())
378 follower->setState(value);
382 follower->setState(follower->getCompletionCondition() == FINISH_LOOP
388 follower->setState(value);
394void SchedulerJob::updateSharedFollowerAttributes()
397 for (
auto follower : followerJobs())
399 follower->setStartupTime(getStartupTime(),
false);
400 follower->setStartAtTime(getStartAtTime());
401 follower->setFollowerState(getState(),
true);
406void SchedulerJob::setSequenceCount(
const int count)
408 sequenceCount = count;
411void SchedulerJob::setCompletedCount(
const int count)
413 completedCount = count;
424 fileStartupCondition = value;
427void SchedulerJob::setStartAtTime(
const QDateTime &value)
432void SchedulerJob::setEstimatedTime(
const int64_t &value)
443 if (START_ASAP != fileStartupCondition && FINISH_AT == completionCondition)
445 estimatedTime = startupTime.secsTo(finishAtTime);
448 else if (FINISH_AT != completionCondition && FINISH_LOOP != completionCondition)
450 estimatedTime = value;
453 else estimatedTime = value;
456void SchedulerJob::setInSequenceFocus(
bool value)
458 inSequenceFocus = value;
461void SchedulerJob::setEnforceTwilight(
bool value)
463 enforceTwilight = value;
464 SchedulerModuleState::calculateDawnDusk(startupTime, nextDawn, nextDusk);
467void SchedulerJob::setEnforceArtificialHorizon(
bool value)
469 enforceArtificialHorizon = value;
472void SchedulerJob::setLightFramesRequired(
bool value)
474 lightFramesRequired = value;
477void SchedulerJob::setCalibrationMountPark(
bool value)
479 m_CalibrationMountPark = value;
482void SchedulerJob::setRepeatsRequired(
const uint16_t &value)
484 repeatsRequired = value;
487 if (1 < repeatsRequired)
489 if (FINISH_REPEAT != completionCondition)
490 setCompletionCondition(FINISH_REPEAT);
492 else if (0 < repeatsRequired)
494 if (FINISH_SEQUENCE != completionCondition)
495 setCompletionCondition(FINISH_SEQUENCE);
499 if (FINISH_LOOP != completionCondition)
500 setCompletionCondition(FINISH_LOOP);
504void SchedulerJob::setRepeatsRemaining(
const uint16_t &value)
506 repeatsRemaining = value;
511 capturedFramesMap = value;
514void SchedulerJob::setTargetCoords(
const dms &ra,
const dms &dec,
double djd)
516 targetCoords.setRA0(ra);
517 targetCoords.setDec0(dec);
519 targetCoords.apparentCoord(
static_cast<long double>(J2000), djd);
522void SchedulerJob::setPositionAngle(
double value)
524 m_PositionAngle = value;
527void SchedulerJob::reset()
530 stage = SCHEDSTAGE_IDLE;
531 stateTime = getLocalTime();
532 lastAbortTime = QDateTime();
533 lastErrorTime = QDateTime();
535 startupCondition = fileStartupCondition;
536 startupTime = fileStartupCondition == START_AT ? startAtTime : QDateTime();
539 SchedulerModuleState::calculateDawnDusk(startupTime, nextDawn, nextDusk);
541 stopTime = QDateTime();
545 repeatsRemaining = repeatsRequired;
546 completedIterations = 0;
552bool SchedulerJob::decreasingAltitudeOrder(SchedulerJob
const *job1, SchedulerJob
const *job2, QDateTime
const &when)
554 bool A_is_setting = job1->settingAtStartup;
555 double const altA = when.
isValid() ?
556 SchedulerUtils::findAltitude(job1->getTargetCoords(), when, &A_is_setting) :
557 job1->altitudeAtStartup;
559 bool B_is_setting = job2->settingAtStartup;
560 double const altB = when.
isValid() ?
561 SchedulerUtils::findAltitude(job2->getTargetCoords(), when, &B_is_setting) :
562 job2->altitudeAtStartup;
565 if (A_is_setting && !B_is_setting)
567 else if (!A_is_setting && B_is_setting)
571 return (A_is_setting && B_is_setting) ? altA < altB : altB < altA;
574bool SchedulerJob::satisfiesAltitudeConstraint(
double azimuth,
double altitude, QString *altitudeReason,
575 double *margin)
const
577 if (m_LeadJob !=
nullptr)
578 return m_LeadJob->satisfiesAltitudeConstraint(azimuth, altitude, altitudeReason, margin);
579 if (margin) *margin = 90;
580 if (Options::enableAltitudeLimits())
584 *margin = std::min(fabs(altitude - Options::minimumAltLimit()),
585 fabs(altitude - Options::maximumAltLimit()));
586 if (altitude < Options::minimumAltLimit() || altitude > Options::maximumAltLimit())
588 if (altitudeReason !=
nullptr)
590 if (altitude < Options::minimumAltLimit())
591 *altitudeReason = QString(
"altitude %1 < mount altitude limit %2")
592 .
arg(altitude, 0,
'f', 1).
arg(Options::minimumAltLimit(), 0,
'f', 1);
594 *altitudeReason = QString(
"altitude %1 > mount altitude limit %2")
595 .
arg(altitude, 0,
'f', 1).
arg(Options::maximumAltLimit(), 0,
'f', 1);
601 const double minAltitude = getMinAltitude();
603 *margin = std::min(*margin, fabs(minAltitude - altitude));
606 if (altitude < minAltitude)
608 if (altitudeReason !=
nullptr)
609 *altitudeReason = QString(
"altitude %1 < minAltitude %2").
arg(altitude, 0,
'f', 1).
arg(getMinAltitude(), 0,
'f', 1);
613 if (getHorizon() !=
nullptr && enforceArtificialHorizon)
615 double horizonMargin;
616 bool ok = getHorizon()->isAltitudeOK(azimuth, altitude, altitudeReason, &horizonMargin);
617 if (margin && horizonMargin >= 0 && horizonMargin < *margin)
618 *margin = horizonMargin;
625bool SchedulerJob::moonConstraintsOK(QDateTime
const &when, QString *reason,
double *margin)
const
630 if ((moon ==
nullptr) || ((getMinMoonSeparation() <= 0) && (getMaxMoonAltitude() >= 90)))
634 KStarsDateTime ltWhen(when.
isValid() ?
635 Qt::UTC == when.
timeSpec() ? SchedulerModuleState::getGeo()->UTtoLT(KStarsDateTime(when)) : when :
639 SkyPoint
const target = getTargetCoords();
645 KSNumbers numbers(ltWhen.djd());
648 CachingDms LST = SchedulerModuleState::getGeo()->GSTtoLST(SchedulerModuleState::getGeo()->LTtoUT(ltWhen).gst());
649 moon->updateCoords(&numbers,
true, SchedulerModuleState::getGeo()->lat(), &LST,
true);
650 moon->EquatorialToHorizontal(&LST, SchedulerModuleState::getGeo()->lat());
652 bool separationOK =
true;
653 if (getMinMoonSeparation() > 0)
655 const double val = moon->angularDistanceTo(&o).Degrees() - getMinMoonSeparation();
656 separationOK = val >= 0;
661 bool altitudeOK =
true;
662 if (getMaxMoonAltitude() < 90)
664 const double val = moon->alt().Degrees() - getMaxMoonAltitude();
665 altitudeOK = val <= 0;
667 *margin = std::min(*margin, fabs(val));
670 bool result = separationOK && altitudeOK;
673 if (reason !=
nullptr && !result)
675 if (!separationOK && !altitudeOK)
676 *reason = QString(
"moon separation and altitude");
677 else if (!separationOK)
678 *reason = QString(
"moon separation");
679 else if (!altitudeOK)
680 *reason = QString(
"moon altitude");
685QDateTime SchedulerJob::calculateNextTime(QDateTime
const &when,
bool checkIfConstraintsAreMet,
int increment,
686 QString *reason,
bool runningJob,
const QDateTime &until)
const
690 KStarsDateTime ltWhen(when.
isValid() ?
691 Qt::UTC == when.
timeSpec() ? SchedulerModuleState::getGeo()->UTtoLT(KStarsDateTime(when)) : when :
695 SkyPoint
const target = getTargetCoords();
701 KStarsDateTime
const ut = SchedulerModuleState::getGeo()->LTtoUT(ltWhen);
703 unsigned int maxMinute = 1e8;
704 if (!runningJob && until.
isValid())
705 maxMinute = when.
secsTo(until) / 60;
707 if (maxMinute > 24 * 60)
710 unsigned int nextAltCheck = 0;
713 constexpr int MAX_SKIP = 30;
716 for (
unsigned int minute = 0; minute < maxMinute; minute += increment)
718 KStarsDateTime
const ltOffset(ltWhen.addSecs(minute * 60));
721 QDateTime nextSuccess;
722 if (getEnforceTwilight() && !runsDuringAstronomicalNightTime(ltOffset, &nextSuccess))
724 if (checkIfConstraintsAreMet)
729 const int minutesToSuccess = ltOffset.secsTo(nextSuccess) / 60 - increment;
730 if (minutesToSuccess > 0)
731 minute += minutesToSuccess;
739 if (reason) *reason =
"twilight";
744 if (minute >= nextAltCheck)
747 const bool altAndMoonOK = checkAltitudeAndMoon(o, ltOffset, reason, &margin);
748 bool done = ((checkIfConstraintsAreMet && altAndMoonOK) || (!checkIfConstraintsAreMet && !altAndMoonOK));
757 if (minute - skipStart <= MAX_SKIP)
759 for (
unsigned int min = skipStart; min < minute; min += increment)
761 KStarsDateTime
const lt(ltWhen.addSecs(min * 60));
763 const bool checkOK = checkAltitudeAndMoon(o, lt, reason,
nullptr);
764 if ((checkIfConstraintsAreMet && checkOK) || (!checkIfConstraintsAreMet && !checkOK))
780 skipStart = minute + increment;
781 nextAltCheck = minute + MAX_SKIP;
786 skipStart = minute + increment;
787 nextAltCheck = minute + 16;
792 skipStart = minute + increment;
799bool SchedulerJob::checkAltitudeAndMoon(SkyObject o,
const KStarsDateTime <Offset, QString *reason,
double *margin)
const
802 KSNumbers numbers(ltOffset.
djd());
806 CachingDms
const LST = SchedulerModuleState::getGeo()->GSTtoLST(SchedulerModuleState::getGeo()->LTtoUT(ltOffset).gst());
811 bool const altitudeOK = satisfiesAltitudeConstraint(azimuth, altitude, reason, margin);
816 bool moonConstraint = moonConstraintsOK(ltOffset, reason, &moonMargin);
818 *margin = std::min(*margin, moonMargin);
819 return moonConstraint;
824bool SchedulerJob::runsDuringAstronomicalNightTime(
const QDateTime &time,
825 QDateTime *nextPossibleSuccess)
const
827 if (m_LeadJob !=
nullptr)
828 return m_LeadJob->runsDuringAstronomicalNightTime(time, nextPossibleSuccess);
835 static QDateTime previousMinDawnDusk, previousTime;
836 static GeoLocation
const *previousGeo =
nullptr;
837 static bool previousAnswer;
838 static double previousPreDawnTime = 0;
839 static QDateTime nextSuccess;
842 static std::mutex nightTimeMutex;
843 const std::lock_guard<std::mutex> lock(nightTimeMutex);
847 time >= previousTime && time < previousMinDawnDusk &&
848 SchedulerModuleState::getGeo() == previousGeo &&
849 Options::preDawnTime() == previousPreDawnTime)
851 if (!previousAnswer && nextPossibleSuccess !=
nullptr)
852 *nextPossibleSuccess = nextSuccess;
853 return previousAnswer;
857 previousAnswer = runsDuringAstronomicalNightTimeInternal(time, &previousMinDawnDusk, &nextSuccess);
859 previousGeo = SchedulerModuleState::getGeo();
860 previousPreDawnTime = Options::preDawnTime();
861 if (!previousAnswer && nextPossibleSuccess !=
nullptr)
862 *nextPossibleSuccess = nextSuccess;
863 return previousAnswer;
868bool SchedulerJob::runsDuringAstronomicalNightTimeInternal(
const QDateTime &time, QDateTime *minDawnDusk,
869 QDateTime *nextPossibleSuccess)
const
871 if (m_LeadJob !=
nullptr)
872 return m_LeadJob->runsDuringAstronomicalNightTimeInternal(time, minDawnDusk, nextPossibleSuccess);
875 QDateTime nDawn = nextDawn, nDusk = nextDusk;
879 SchedulerModuleState::calculateDawnDusk(time, nDawn, nDusk);
888 QDateTime
const earlyDawn = nDawn.
addSecs(-60.0 * abs(Options::preDawnTime()));
890 *minDawnDusk = earlyDawn < nDusk ? earlyDawn : nDusk;
894 bool result = nDawn < nDusk && t <= earlyDawn;
897 if (nextPossibleSuccess !=
nullptr)
899 if (result) *nextPossibleSuccess = QDateTime();
900 else *nextPossibleSuccess = nDusk;
906void SchedulerJob::setInitialFilter(
const QString &value)
908 m_InitialFilter = value;
911const QString &SchedulerJob::getInitialFilter()
const
913 return m_InitialFilter;
916bool SchedulerJob::StartTimeCache::check(
const QDateTime &from,
const QDateTime &until,
917 QDateTime *result, QDateTime *newFrom)
const
922 foreach (
const StartTimeComputation &computation, startComputations)
924 if (from >= computation.from &&
925 (!computation.until.isValid() || from < computation.until) &&
926 (!computation.result.isValid() || from < computation.result))
928 if (computation.result.isValid() || until <= computation.until)
931 *result = computation.result;
932 *newFrom = QDateTime();
938 *result = QDateTime();
939 *newFrom = computation.until;
947void SchedulerJob::StartTimeCache::clear()
const
949 startComputations.
clear();
952void SchedulerJob::StartTimeCache::add(
const QDateTime &from,
const QDateTime &until,
const QDateTime &result)
const
955 if (startComputations.size() > 10)
956 startComputations.clear();
961 endTime = from.
addSecs(24 * 3600);
964 QDateTime oneDay = from.
addSecs(24 * 3600);
971 StartTimeComputation c;
975 startComputations.push_back(c);
979QDateTime SchedulerJob::getNextPossibleStartTime(
const QDateTime &when,
int increment,
bool runningJob,
980 const QDateTime &until)
const
983 when.
isValid() ? (
Qt::UTC == when.
timeSpec() ? SchedulerModuleState::getGeo()->UTtoLT(KStarsDateTime(when)) : when)
989 if (!runningJob && START_AT == getFileStartupCondition())
991 int secondsFromNow = ltWhen.secsTo(getStartAtTime());
992 if (secondsFromNow < -500)
995 ltWhen = secondsFromNow > 0 ? getStartAtTime() : ltWhen;
999 if (getCompletionCondition() == FINISH_AT)
1001 const QDateTime &t = getFinishAtTime();
1002 if (t.
isValid() && t < ltWhen)
1007 return calculateNextTime(ltWhen,
true, increment,
nullptr, runningJob, until);
1010 QDateTime result, newFrom;
1011 if (startTimeCache.check(ltWhen, until, &result, &newFrom))
1018 result = calculateNextTime(ltWhen,
true, increment,
nullptr, runningJob, until);
1020 startTimeCache.add(ltWhen, until, result);
1026QDateTime SchedulerJob::getNextEndTime(
const QDateTime &start,
int increment, QString *reason,
const QDateTime &until)
const
1029 start.
isValid() ? (
Qt::UTC == start.
timeSpec() ? SchedulerModuleState::getGeo()->UTtoLT(KStarsDateTime(start)) : start)
1035 if (START_AT == getFileStartupCondition())
1037 if (getStartAtTime().secsTo(ltStart) < -120)
1041 if (reason) *reason =
"before start-at time";
1048 if (getCompletionCondition() == FINISH_AT)
1050 const QDateTime &t = getFinishAtTime();
1051 if (t.
isValid() && t < ltStart)
1053 if (reason) *reason =
"end-at time";
1056 auto result = calculateNextTime(ltStart,
false, increment, reason,
false, until);
1057 if (!result.
isValid() || result.
secsTo(getFinishAtTime()) < 0)
1059 if (reason) *reason =
"end-at time";
1060 return getFinishAtTime();
1065 return calculateNextTime(ltStart,
false, increment, reason,
false, until);
1071QString progressLineLabel(CCDFrameType frameType,
const QMap<SequenceJob::PropertyID, QVariant> &properties,
1074 QString jobTargetName =
properties[SequenceJob::SJ_TargetName].toString();
1075 auto exposure =
properties[SequenceJob::SJ_Exposure].toDouble();
1078 int precisionRequired = 0;
1079 double fraction = exposure - fabs(exposure);
1080 if (fraction > .0001)
1082 precisionRequired = 1;
1083 fraction = fraction * 10;
1084 fraction = fraction - fabs(fraction);
1085 if (fraction > .0001)
1087 precisionRequired = 2;
1088 fraction = fraction * 10;
1089 fraction = fraction - fabs(fraction);
1090 if (fraction > .0001)
1091 precisionRequired = 3;
1094 if (precisionRequired == 0)
1095 label += QString(
"%1s").
arg(
static_cast<int>(exposure));
1097 label += QString(
"%1s").
arg(exposure, 0,
'f', precisionRequired);
1099 if (
properties.contains(SequenceJob::SJ_Filter))
1101 auto filterType =
properties[SequenceJob::SJ_Filter].toString();
1103 label += filterType;
1111 else if (frameType != FRAME_LIGHT)
1114 label += (char)frameType;
1120QString progressLine(
const SchedulerJob::JobProgress &progress)
1122 QString
label = progressLineLabel(progress.type, progress.properties, progress.isDarkFlat).
append(
":");
1124 const double seconds = progress.numCompleted * progress.properties[SequenceJob::SJ_Exposure].
toDouble();
1128 else if (seconds < 60)
1129 timeStr = QString(
"%1 %2").
arg(
static_cast<int>(seconds)).
arg(
i18n(
"seconds"));
1130 else if (seconds < 60 * 60)
1131 timeStr = QString(
"%1 %2").
arg(seconds / 60.0, 0,
'f', 1).
arg(
i18n(
"minutes"));
1133 timeStr = QString(
"%1 %3").
arg(seconds / 3600.0, 0,
'f', 1).
arg(
i18n(
"hours"));
1137 return QString(
"%1\t%2 %3 %4")
1138 .
arg(label, -12,
' ')
1139 .
arg(progress.numCompleted, 4)
1145const QString SchedulerJob::getProgressSummary()
const
1148 for (
const auto &p : m_Progress)
1150 summary.
append(progressLine(p));
1156QJsonObject SchedulerJob::toJson()
const
1158 bool is_setting =
false;
1159 double const alt = SchedulerUtils::findAltitude(getTargetCoords(), QDateTime(), &is_setting);
1164 {
"pa", m_PositionAngle},
1165 {
"targetRA", getTargetCoords().ra0().Hours()},
1166 {
"targetDEC", getTargetCoords().dec0().Degrees()},
1169 {
"sequenceCount", sequenceCount},
1170 {
"completedCount", completedCount},
1171 {
"minAltitude", minAltitude},
1172 {
"minMoonSeparation", minMoonSeparation},
1173 {
"maxMoonAltitude", maxMoonAltitude},
1174 {
"repeatsRequired", repeatsRequired},
1175 {
"repeatsRemaining", repeatsRemaining},
1176 {
"inSequenceFocus", inSequenceFocus},
1177 {
"startupTime", startupTime.isValid() ? startupTime.toString() :
"--"},
1178 {
"completionTime", finishAtTime.isValid() ? finishAtTime.toString() :
"--"},
1180 {
"altitudeFormatted", m_AltitudeFormatted},
1181 {
"startupFormatted", m_StartupFormatted},
1182 {
"endFormatted", m_EndFormatted},
1183 {
"sequence", sequenceFile.toString() },
SkyMapComposite * skyComposite()
SkyObject * findByName(const QString &name, bool exact=true) override
Search the children of this SkyMapComposite for a SkyObject whose name matches the argument.
const CachingDms & ra0() const
virtual void updateCoordsNow(const KSNumbers *num)
updateCoordsNow Shortcut for updateCoords( const KSNumbers *num, false, nullptr, nullptr,...
void EquatorialToHorizontal(const CachingDms *LST, const CachingDms *lat)
Determine the (Altitude, Azimuth) coordinates of the SkyPoint from its (RA, Dec) coordinates,...
void setRA0(dms r)
Sets RA0, the catalog Right Ascension.
const CachingDms & dec0() const
void setDec0(dms d)
Sets Dec0, the catalog Declination.
const double & Degrees() const
QString i18n(const char *text, const TYPE &arg...)
char * toString(const EngineQuery &query)
Ekos is an advanced Astrophotography tool for Linux.
SchedulerJobStage
Running stages of a SchedulerJob.
StartupCondition
Conditions under which a SchedulerJob may start.
SchedulerJobStatus
States of a SchedulerJob.
@ SCHEDJOB_ABORTED
Job encountered a transitory issue while processing, and will be rescheduled.
@ SCHEDJOB_INVALID
Job has an incorrect configuration, and cannot proceed.
@ SCHEDJOB_ERROR
Job encountered a fatal issue while processing, and must be reset manually.
@ SCHEDJOB_COMPLETE
Job finished all required captures.
@ SCHEDJOB_EVALUATION
Job is being evaluated.
@ SCHEDJOB_SCHEDULED
Job was evaluated, and has a schedule.
@ SCHEDJOB_BUSY
Job is being processed.
@ SCHEDJOB_IDLE
Job was just created, and is not evaluated yet.
QMap< QString, uint16_t > CapturedFramesMap
mapping signature --> frames count
CompletionCondition
Conditions under which a SchedulerJob may complete.
QString label(StandardShortcut id)
QDateTime addSecs(qint64 s) const const
bool isValid() const const
qint64 secsTo(const QDateTime &other) const const
void setTimeZone(const QTimeZone &toZone)
Qt::TimeSpec timeSpec() const const
QString & append(QChar ch)
QString arg(Args &&... args) const const
qsizetype size() const const
double toDouble(bool *ok) const const