Kstars

align.h
1/*
2 SPDX-FileCopyrightText: 2013 Jasem Mutlaq <mutlaqja@ikarustech.com>
3 SPDX-FileCopyrightText: 2013-2021 Jasem Mutlaq <mutlaqja@ikarustech.com>
4 SPDX-FileCopyrightText: 2018-2020 Robert Lancaster <rlancaste@gmail.com>
5 SPDX-FileCopyrightText: 2019-2021 Hy Murveit <hy@murveit.com>
6
7 SPDX-License-Identifier: GPL-2.0-or-later
8*/
9
10#pragma once
11
12#include "ui_align.h"
13#include "ekos/ekos.h"
14#include "indi/indicamera.h"
15#include "indi/indistd.h"
16#include "indi/indimount.h"
17#include "skypoint.h"
18
19#include <QTime>
20#include <QTimer>
21#include <QElapsedTimer>
22#include <KConfigDialog>
23
24#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
25#include <QtDBus/qtdbusglobal.h>
26#elif QT_VERSION >= QT_VERSION_CHECK(5, 8, 0)
27#include <qtdbusglobal.h>
28#else
29#include <qdbusmacros.h>
30#endif
31
32#include <stellarsolver.h>
33#include <memory>
34
36
37class AlignView;
38class FITSViewer;
39class FOV;
40class StarObject;
41class ProfileInfo;
42class RotatorSettings;
43
44namespace Ekos
45{
47class DarkProcessor;
48class FilterManager;
50class OpsAstrometry;
51class OpsAlign;
52class StellarSolverProfileEditor;
53class OpsPrograms;
54class OpsASTAP;
55class OpsAstrometryIndexFiles;
56class MountModel;
58class ManualRotator;
59
60/**
61 *@class Align
62 *@short Align class handles plate-solving and polar alignment measurement and correction using astrometry.net
63 * The align class employs StellarSolver library for local solvers and supports remote INDI-based solver.
64 * StellarSolver supports internal and external solvers (Astrometry.net, ASTAP, Online Astrometry).
65 * If an image is solved successfully, the image central J2000 RA & DE coordinates along with pixel scale, rotation, and partiy are
66 * reported back.
67 * Index files management is supported with ability to download astrometry.net files. The user may select and edit different solver
68 * profiles that provide settings to control both extraction and solving profiles in detail. Manual and automatic field rotation
69 * is supported in order to align the solved images to a particular orientation in the sky. The manual rotation assistant is an interactive
70 * tool that helps the user to arrive at the desired framing.
71 * Align module provide Polar Align Helper tool which enables easy-to-follow polar alignment procedure given wide FOVs (> 1.5 degrees)
72 * Legacy polar aligment is deprecated.
73 *@author Jasem Mutlaq
74 *@version 2.0
75 */
76class Align : public QWidget, public Ui::Align
77{
79 Q_CLASSINFO("D-Bus Interface", "org.kde.kstars.Ekos.Align")
80 Q_PROPERTY(Ekos::AlignState status READ status NOTIFY newStatus)
81 Q_PROPERTY(QStringList logText READ logText NOTIFY newLog)
82 Q_PROPERTY(QString opticalTrain READ opticalTrain WRITE setOpticalTrain)
83 Q_PROPERTY(QString camera READ camera)
84 Q_PROPERTY(QString filterWheel READ filterWheel)
85 Q_PROPERTY(QString filter READ filter WRITE setFilter)
86 Q_PROPERTY(double exposure READ exposure WRITE setExposure)
87 Q_PROPERTY(double targetPositionAngle MEMBER m_TargetPositionAngle)
88 Q_PROPERTY(QList<double> fov READ fov)
89 Q_PROPERTY(QList<double> cameraInfo READ cameraInfo)
90 Q_PROPERTY(QList<double> telescopeInfo READ telescopeInfo)
91 //Q_PROPERTY(QString solverArguments READ solverArguments WRITE setSolverArguments)
92
93 public:
94 explicit Align(const QSharedPointer<ProfileInfo> &activeProfile);
95 virtual ~Align() override;
96
97 typedef enum { GOTO_SYNC, GOTO_SLEW, GOTO_NOTHING } GotoMode;
98 typedef enum { SOLVER_LOCAL, SOLVER_REMOTE } SolverMode;
99 typedef enum
100 {
101 ALIGN_RESULT_SUCCESS,
102 ALIGN_RESULT_WARNING,
103 ALIGN_RESULT_FAILED
104 } AlignResult;
105
106 typedef enum
107 {
108 BLIND_IDLE,
109 BLIND_ENGAGNED,
110 BLIND_USED
111 } BlindState;
112
113 /** @defgroup AlignDBusInterface Ekos DBus Interface - Align Module
114 * Ekos::Align interface provides advanced scripting capabilities to solve images using online or offline astrometry.net
115 */
116
117 /*@{*/
118
119 /** DBUS interface function.
120 * Select CCD
121 * @param device CCD device name
122 * @return Returns true if device if found and selected, false otherwise.
123 */
124 Q_SCRIPTABLE QString camera();
125
126 /** DBUS interface function.
127 * select the filter device from the available filter drivers. The filter device can be the same as the CCD driver if the filter functionality was embedded within the driver.
128 * @param device The filter device name
129 * @return Returns true if filter device is found and set, false otherwise.
130 */
131 Q_SCRIPTABLE QString filterWheel();
132
133 /** DBUS interface function.
134 * select the filter from the available filters.
135 * @param filter The filter name
136 * @return Returns true if filter is found and set, false otherwise.
137 */
138 Q_SCRIPTABLE bool setFilter(const QString &filter);
139 Q_SCRIPTABLE QString filter();
140
141 /** DBUS interface function.
142 * Start the plate-solving process given the passed image file.
143 * @param filename Name of image file to solve. FITS and JPG/JPG/TIFF formats are accepted.
144 * @param isGenerated Set to true if filename is generated from a CCD capture operation. If the file is loaded from any storage or network media, pass false.
145 * @return Returns true if device if found and selected, false otherwise.
146 */
147 Q_SCRIPTABLE Q_NOREPLY void startSolving();
148
149 /** DBUS interface function.
150 * Select Solver Action after successfully solving an image.
151 * @param mode 0 for Sync, 1 for Slew To Target, 2 for Nothing (just display solution results)
152 */
153 Q_SCRIPTABLE Q_NOREPLY void setSolverAction(int mode);
154
155 /** DBUS interface function.
156 * Returns the solver's solution results
157 * @return Returns array of doubles. First item is RA in degrees. Second item is DEC in degrees.
158 */
159 Q_SCRIPTABLE QList<double> getSolutionResult();
160
161 /** DBUS interface function.
162 * Returns the solver's current status
163 * @return Returns solver status (Ekos::AlignState)
164 */
166 {
167 return state;
168 }
169
170 /** DBUS interface function.
171 * @return Returns State of load slew procedure. Idle if not started. Busy if in progress. Ok if complete. Alert if procedure failed.
172 */
173 Q_SCRIPTABLE int getLoadAndSlewStatus()
174 {
175 return m_SolveFromFile;
176 }
177
178 /** DBUS interface function.
179 * Sets the exposure of the selected CCD device.
180 * @param value Exposure value in seconds
181 */
182 Q_SCRIPTABLE Q_NOREPLY void setExposure(double value);
183 Q_SCRIPTABLE double exposure()
184 {
185 return alignExposure->value();
186 }
187
188 /** DBUS interface function.
189 * Get currently active camera info in this order:
190 * width, height, pixel_size_x, pixel_size_y
191 */
192 Q_SCRIPTABLE QList<double> cameraInfo();
193
194 /** DBUS interface function.
195 * Get current active telescope info in this order:
196 * focal length, aperture
197 */
199
200 /** @}*/
201
202 /**
203 * @brief Add Camera to the list of available Cameras.
204 * @param device pointer to camera device.
205 * @return True if added successfully, false if duplicate or failed to add.
206 */
207 bool setCamera(ISD::Camera *device);
208
209 /**
210 * @brief addFilterWheel Add new filter wheel filter device.
211 * @param device pointer to filter device.
212 * @return True if added successfully, false if duplicate or failed to add.
213 */
214 bool setFilterWheel(ISD::FilterWheel *device);
215
216 /**
217 * @brief Add new mount
218 * @param device pointer to mount device.
219 * @return True if added successfully, false if duplicate or failed to add.
220 */
221 bool setMount(ISD::Mount *device);
222
223 /**
224 * @brief Add new Dome
225 * @param device pointer to dome device.
226 * @return True if added successfully, false if duplicate or failed to add.
227 */
228 bool setDome(ISD::Dome *device);
229
230 /**
231 * @brief Add new Rotator
232 * @param device pointer to rotator device.
233 */
234 void setRotator(ISD::Rotator *device);
235
236 void removeDevice(const QSharedPointer<ISD::GenericDevice> &device);
237
238 /**
239 * @brief setAstrometryDevice
240 * @param newAstrometry
241 */
243
244 /**
245 * @brief CCD information is updated, sync them.
246 */
247 void syncCameraInfo();
248
249 /**
250 * @brief syncCCDControls Update camera controls like gain, offset, ISO..etc.
251 */
252 void syncCameraControls();
253
254 /**
255 * @brief Generate arguments we pass to the remote solver.
256 */
258
259 /**
260 * @brief Does our parser exist in the system?
261 */
262 bool isParserOK();
263
264 // Log
265 QStringList logText()
266 {
267 return m_LogText;
268 }
269 QString getLogText()
270 {
271 return m_LogText.join("\n");
272 }
273 Q_SCRIPTABLE void clearLog();
274
275 /**
276 * @brief getFOVScale Returns calculated FOV values
277 * @param fov_w FOV width in arcmins
278 * @param fov_h FOV height in arcmins
279 * @param fov_scale FOV scale in arcsec per pixel
280 */
281 void getFOVScale(double &fov_w, double &fov_h, double &fov_scale);
282 QList<double> fov();
283
284 /**
285 * @brief getCalculatedFOVScale Get calculated FOV scales from the current CCD+Telescope combination.
286 * @param fov_w return calculated fov width in arcminutes
287 * @param fov_h return calculated fov height in arcminutes
288 * @param fov_scale return calculated fov pixcale in arcsecs per pixel.
289 * @note This is NOT the same as effective FOV which is the measured FOV from astrometry. It is the
290 * theoretical FOV from calculated values.
291 */
292 void getCalculatedFOVScale(double &fov_w, double &fov_h, double &fov_scale);
293
294 void setupFilterManager();
295 void setupPlot();
296 void setupSolutionTable();
297 void setupOptions();
298
299 /**
300 * @brief Sync the telescope to the solved alignment coordinate.
301 */
302 void Sync();
303
304 /**
305 * @brief Slew the telescope to the solved alignment coordinate.
306 */
307 void Slew();
308
309 /**
310 * @brief Sync the telescope to the solved alignment coordinate, and then slew to the target coordinate.
311 */
312 void SlewToTarget();
313
314 /**
315 * @brief getStellarSolverProfiles
316 * @return list of StellarSolver profile names
317 */
319
320 GotoMode currentGOTOMode() const
321 {
322 return m_CurrentGotoMode;
323 }
324
325 /**
326 * @brief generateOptions Generate astrometry.net option given the supplied map
327 * @param optionsMap List of key=value pairs for all astrometry.net options
328 * @return String List of valid astrometry.net options
329 */
330 static QStringList generateRemoteOptions(const QVariantMap &optionsMap);
331 static void generateFOVBounds(double fov_h, QString &fov_low, QString &fov_high, double tolerance = 0.05);
332
333 // access to the mount model UI, required for testing
334 MountModel * mountModel() const
335 {
336 return m_MountModel;
337 }
338
339 PolarAlignmentAssistant *polarAlignmentAssistant() const
340 {
341 return m_PolarAlignmentAssistant;
342 }
343
344 bool wcsSynced() const
345 {
346 return m_wcsSynced;
347 }
348
349 /**
350 * @brief Process updated device properties
351 * @param prop INDI Property
352 */
353 void updateProperty(INDI::Property prop);
354
355
356 /**
357 * @brief Check CCD and make sure information is updated and FOV is re-calculated.
358 * @param CCDNum By default, we check the already selected CCD in the dropdown menu. If CCDNum is specified, the check is made against this specific CCD in the dropdown menu. CCDNum is the index of the CCD in the dropdown menu.
359 */
360 void checkCamera();
361
362 /**
363 * @brief Check Filter and make sure information is updated accordingly.
364 * @param filterNum By default, we check the already selected filter in the dropdown menu. If filterNum is specified, the check is made against this specific filter in the dropdown menu.
365 * filterNum is the index of the filter in the dropdown menu.
366 */
367 void checkFilter();
368
369 /**
370 * @brief checkCameraExposureProgress Track the progress of CCD exposure
371 * @param targetChip Target chip under exposure
372 * @param remaining how many seconds remaining
373 * @param state status of exposure
374 */
375 void checkCameraExposureProgress(ISD::CameraChip *targetChip, double remaining, IPState state);
376 /**
377 * @brief Process new FITS received from CCD.
378 * @param bp pointer to blob property
379 */
380 void processData(const QSharedPointer<FITSData> &data);
381
382 /** DBUS interface function.
383 * Loads an image (FITS, RAW, or JPG/PNG) and solve its coordinates, then it slews to the solved coordinates and an image is captured and solved to ensure
384 * the telescope is pointing to the same coordinates of the image.
385 * @param image buffer to image data.
386 * @param extension image extension (e.g. cr2, jpg, fits,..etc).
387 */
388 bool loadAndSlew(const QByteArray &image, const QString &extension);
389
390 /** \addtogroup AlignDBusInterface
391 * @{
392 */
393
394 /**
395 * @brief Stop aligning
396 * @param mode stop mode (abort or suspend)
397 */
398 void stop(Ekos::AlignState mode);
399
400 /** DBUS interface function.
401 * Aborts the solving operation, handle outside of the align module.
402 */
403 Q_SCRIPTABLE Q_NOREPLY void abort()
404 {
406 }
407
408 /**
409 * @brief Suspend aligning, recovery handled by the align module itself.
410 */
411 void suspend()
412 {
414 }
415
416 /** DBUS interface function.
417 * Select the solver mode
418 * @param type Set solver type. 0 LOCAL, 1 REMOTE (requires remote astrometry driver to be activated)
419 */
420 Q_SCRIPTABLE Q_NOREPLY void setSolverMode(int mode);
421
422 /** DBUS interface function.
423 * Capture and solve an image using the astrometry.net engine
424 * @return Returns true if the procedure started successful, false otherwise (return true, not false, when retrying!)
425 */
426 Q_SCRIPTABLE bool captureAndSolve(bool initialCall = true);
427
428 /** DBUS interface function.
429 * Loads an image (FITS, RAW, or JPG/PNG) and solve its coordinates, then it slews to the solved coordinates and an image is captured and solved to ensure
430 * the telescope is pointing to the same coordinates of the image.
431 * @param fileURL URL to the image to solve
432 */
433 Q_SCRIPTABLE bool loadAndSlew(QString fileURL = QString());
434
435 /** DBUS interface function.
436 * Sets the target coordinates that the solver compares the solution coordinates to.
437 * By default, the target coordinates are those of the current mount when the capture and
438 * solve operation is started. In case of SYNC, only the error between the solution and target
439 * coordinates is calculated. When Slew to Target is selected, the mount would be slewed afterwards to
440 * this target coordinate.
441 * @param ra0 J2000 Right Ascension in hours.
442 * @param de0 J2000 Declination in degrees.
443 */
444 Q_SCRIPTABLE Q_NOREPLY void setTargetCoords(double ra0, double de0);
445
446 /**
447 * @brief getTargetCoords QList of target coordinates.
448 * @return First value is J2000 RA in hours. Second value is J2000 DE in degrees.
449 */
450 Q_SCRIPTABLE QList<double> getTargetCoords();
451
452
453 /**
454 * @brief Set the alignment target where the mount is expected to point at.
455 * @param targetCoord exact coordinates of the target position.
456 */
457 void setTarget(const SkyPoint &targetCoord);
458
459 /**
460 * @brief Set the coordinates that the mount reports as its position
461 * @param position current mount position
462 */
463 void setTelescopeCoordinates(const SkyPoint &position)
464 {
465 m_TelescopeCoord = position;
466 }
467
468 Q_SCRIPTABLE Q_NOREPLY void setTargetPositionAngle(double value);
469
470 /** DBUS interface function.
471 * Sets the binning of the selected CCD device.
472 * @param binIndex Index of binning value. Default values range from 0 (binning 1x1) to 3 (binning 4x4)
473 */
474 Q_SCRIPTABLE Q_NOREPLY void setBinningIndex(int binIndex);
475
476 /** @}*/
477
478 /**
479 * @brief Solver finished successfully, process the data and execute the required actions depending on the mode.
480 * @param orientation Orientation of image in degrees (East of North)
481 * @param ra Center RA in solved image, degrees.
482 * @param dec Center DEC in solved image, degrees.
483 * @param pixscale Image scale is arcsec/pixel
484 * @param eastToTheRight When the image is rotated, so that North is up, East would be to the right.
485 */
486 void solverFinished(double orientation, double ra, double dec, double pixscale, bool eastToTheRight);
487
488 void solverComplete();
489
490 /**
491 * @brief Process solver failure.
492 */
493 void solverFailed();
494
495 /**
496 * @brief We received new telescope info, process them and update FOV.
497 */
498 bool syncTelescopeInfo();
499
500 void setFocusStatus(Ekos::FocusState state);
501
502 // Log
503 void appendLogText(const QString &);
504
505 // Capture
506 void setCaptureComplete();
507
508 // Update Capture Module status
509 void setCaptureStatus(Ekos::CaptureState newState);
510 // Update Mount module status
511 void setMountStatus(ISD::Mount::Status newState);
512
513 void zoomAlignView();
514 void setAlignZoom(double scale);
515
516 // Manual Rotator Dialog
517 void toggleManualRotator(bool toggled);
518
519 /**
520 * @brief checkIfRotationRequired Check whether we need to perform an ALIGN_ROTATING action, whether manual or automatic.
521 * @return True if rotation is required as per the settings, false is not required.
522 */
524
525 // Settings
526 QVariantMap getAllSettings() const;
527 void setAllSettings(const QVariantMap &settings);
528
529 /**
530 * @brief settleSettings Run this function after timeout from debounce timer to update database
531 * and emit settingsChanged signal. This is required so we don't overload output.
532 */
533 void settleSettings();
534
535 // Trains
536 QString opticalTrain() const
537 {
538 return opticalTrainCombo->currentText();
539 }
540 void setOpticalTrain(const QString &value)
541 {
542 opticalTrainCombo->setCurrentText(value);
543 }
544
545 Ekos::OpsAlign *getAlignOptionsModule()
546 {
547 return opsAlign;
548 }
549
550 private slots:
551 // Solver timeout
552 void checkAlignmentTimeout();
553 void setAlignTableResult(AlignResult result);
554
555 // External View
556 void showFITSViewer();
557 void toggleAlignWidgetFullScreen();
558
559 /**
560 * @brief prepareCapture Set common settings for capture for align module
561 * @param targetChip target Chip
562 */
563 void prepareCapture(ISD::CameraChip *targetChip);
564
565 //Solutions Display slots
566 void buildTarget();
567 void handlePointTooltip(QMouseEvent *event);
568 void handleVerticalPlotSizeChange();
569 void handleHorizontalPlotSizeChange();
570 void selectSolutionTableRow(int row, int column);
571 void slotClearAllSolutionPoints();
572 void slotRemoveSolutionPoint();
573 void slotAutoScaleGraph();
574
575 // Model
576 void slotMountModel();
577
578 // Capture Timeout
579 void processCaptureTimeout();
580
581 protected slots:
582 /**
583 * @brief After a solver process is completed successfully, sync, slew to target, or do nothing as set by the user.
584 */
585 void executeGOTO();
586
587 /**
588 * @brief refreshAlignOptions is called when settings are updated in OpsAlign.
589 */
590 void refreshAlignOptions();
591
592 void processPAHStage(int stage);
593
594 signals:
595 void newLog(const QString &text);
596 void newStatus(Ekos::AlignState state);
597 void newPAAStage(int stage);
598 void newSolution(const QVariantMap &solution);
599
600 // This is sent when we load an image in the view
601 void newImage(const QSharedPointer<FITSView> &view);
602 // This is sent when the pixmap is updated within the view
603 void newFrame(const QSharedPointer<FITSView> &view);
604 // Send new solver results
605 void newSolverResults(double orientation, double ra, double dec, double pixscale);
606
607 // Train changed
608 void trainChanged();
609
610 // Settings
611 void settingsUpdated(const QVariantMap &settings);
612
613 // Manual Rotator
614 void manualRotatorChanged(double currentPA, double targetPA, double pixscale);
615
616 // Astrometry index files progress
617 void newDownloadProgress(QString info);
618
619 private:
620
621 void setupOpticalTrainManager();
622 void refreshOpticalTrain();
623
624 ////////////////////////////////////////////////////////////////////
625 /// Settings
626 ////////////////////////////////////////////////////////////////////
627
628 /**
629 * @brief Connect GUI elements to sync settings once updated.
630 */
631 void connectSettings();
632 /**
633 * @brief Stop updating settings when GUI elements are updated.
634 */
635 void disconnectSettings();
636 /**
637 * @brief loadSettings Load setting from Options and set them accordingly.
638 */
639 void loadGlobalSettings();
640
641 /**
642 * @brief syncSettings When checkboxes, comboboxes, or spin boxes are updated, save their values in the
643 * global and per-train settings.
644 */
645 void syncSettings();
646
647 /**
648 * @brief syncControl Sync setting to widget. The value depends on the widget type.
649 * @param settings Map of all settings
650 * @param key name of widget to sync
651 * @param widget pointer of widget to set
652 * @return True if sync successful, false otherwise
653 */
654 bool syncControl(const QVariantMap &settings, const QString &key, QWidget * widget);
655
656 void setState (AlignState value);
657 /**
658 * @brief Retrieve the align status indicator
659 */
660 QProgressIndicator *getProgressStatus();
661
662 /**
663 * @brief Stop the progress animation in the solution table
664 */
665 void stopProgressAnimation();
666
667 void exportSolutionPoints();
668
669 /**
670 * @brief Calculate Field of View of CCD+Telescope combination that we need to pass to astrometry.net solver.
671 */
672 void calculateFOV();
673
674 /**
675 * @brief calculateEffectiveFocalLength Calculate Focal Length purely form astrometric data.
676 */
677 void calculateEffectiveFocalLength(double newFOVW);
678
679 /**
680 * @brief calculateAlignTargetDiff Find the difference between aligned vs. target coordinates and update
681 * the GUI accordingly.
682 */
683 void calculateAlignTargetDiff();
684
685 /**
686 * @brief Get formatted RA & DEC coordinates compatible with astrometry.net format.
687 * @param ra Right ascension
688 * @param dec Declination
689 * @param ra_str will contain the formatted RA string
690 * @param dec_str will contain the formatted DEC string
691 */
692 void getFormattedCoords(double ra, double dec, QString &ra_str, QString &dec_str);
693
694 uint8_t getSolverDownsample(uint16_t binnedW);
695
696 /**
697 * @brief setWCSEnabled enables/disables World Coordinate System settings in the CCD driver.
698 * @param enable true to enable WCS, false to disable.
699 */
700 void setWCSEnabled(bool enable);
701
702 void resizeEvent(QResizeEvent *event) override;
703
704 KPageWidgetItem *m_IndexFilesPage;
705 QString savedOptionsProfiles;
706
707 /**
708 * @brief React when a mount motion has been detected
709 */
710 void handleMountMotion();
711
712 /**
713 * @brief Continue aligning according to the current mount status
714 */
715 void handleMountStatus();
716
717 /**
718 * @brief initPolarAlignmentAssistant Initialize Polar Alignment Asssistant Tool
719 */
720 void setupPolarAlignmentAssistant();
721
722 void setupRotatorControl();
723
724 /**
725 * @brief initManualRotator Initialize Manual Rotator Tool
726 */
727 void setupManualRotator();
728
729 /**
730 * @brief initDarkProcessor Initialize Dark Processor
731 */
732 void setuptDarkProcessor();
733
734 bool matchPAHStage(uint32_t stage);
735
736
737 // Effective FOV
738
739 /**
740 * @brief getEffectiveFOV Search database for effective FOV that matches the current profile and settings
741 * @return Variant Map containing effect FOV data or empty variant map if none found
742 */
743 QVariantMap getEffectiveFOV();
744 void saveNewEffectiveFOV(double newFOVW, double newFOVH);
745 QList<QVariantMap> effectiveFOVs;
746 void syncFOV();
747
748 // We are using calculated FOV now until a more accurate effective FOV is found.
749 bool m_EffectiveFOVPending { false };
750 /// Which chip should we invoke in the current CCD?
751 bool useGuideHead { false };
752 /// Can the mount sync its coordinates to those set by Ekos?
753 bool canSync { false };
754 // m_SolveFromFile is true we load an image and solve it, no capture is done.
755 bool m_SolveFromFile { false };
756 // Target Position Angle of solver Load&Slew image to be used for rotator if necessary
757 double m_TargetPositionAngle { std::numeric_limits<double>::quiet_NaN() };
758 // Target Pierside of solver Load&Slew image to be used
759 ISD::Mount::PierSide m_TargetPierside = ISD::Mount::PIER_UNKNOWN;
760 double currentRotatorPA { -1 };
761 /// Solver iterations count
762 uint8_t solverIterations { 0 };
763 /// Was solving with scale off used?
764 BlindState useBlindScale {BLIND_IDLE};
765 /// Was solving with position off used?
766 BlindState useBlindPosition {BLIND_IDLE};
767
768 // FOV
769 double m_CameraPixelWidth { -1 };
770 double m_CameraPixelHeight { -1 };
771 uint16_t m_CameraWidth { 0 };
772 uint16_t m_CameraHeight { 0 };
773
774 double m_FocalLength {-1};
775 double m_Aperture {-1};
776 double m_FocalRatio {-1};
777 double m_Reducer = {-1};
778
779 double m_FOVWidth { 0 };
780 double m_FOVHeight { 0 };
781 double m_FOVPixelScale { 0 };
782
783 // Keep raw rotator angle
784 double sRawAngle { INVALID_VALUE };
785 // Keep track of solver results
786 double sOrientation { INVALID_VALUE };
787 double sRA { INVALID_VALUE };
788 double sDEC { INVALID_VALUE };
789
790 /// Solver alignment coordinates
791 SkyPoint m_AlignCoord;
792 /// Target coordinates the mount will slew to
793 SkyPoint m_TargetCoord;
794 /// Final coordinates we want to reach in case of differential align
795 SkyPoint m_DestinationCoord;
796 /// Current telescope coordinates
797 SkyPoint m_TelescopeCoord;
798 /// Difference between solution and target coordinate
799 double m_TargetDiffTotal { 1e6 };
800 double m_TargetDiffRA { 1e6 };
801 double m_TargetDiffDE { 1e6 };
802
803 /// Progress icon if the solver is running
804 std::unique_ptr<QProgressIndicator> pi;
805
806 /// Keep track of how long the solver is running
807 QElapsedTimer solverTimer;
808
809 // The StellarSolver
810 std::unique_ptr<StellarSolver> m_StellarSolver;
811 // StellarSolver Profiles
812 QList<SSolver::Parameters> m_StellarSolverProfiles;
813
814 /// Have we slewed?
815 bool m_wasSlewStarted { false };
816 // Above flag only stays false for 10s after slew start.
817 QElapsedTimer slewStartTimer;
818 bool didSlewStart();
819 // Only wait this many milliseconds for slew to start.
820 // Otherwise assume it has begun.
821 static constexpr int MAX_WAIT_FOR_SLEW_START_MSEC = 10000;
822
823 // Online and Offline parsers
824 AstrometryParser* parser { nullptr };
825 std::unique_ptr<RemoteAstrometryParser> remoteParser;
826 QSharedPointer<ISD::GenericDevice> m_RemoteParserDevice;
827
828 // Pointers to our devices
829 ISD::Mount *m_Mount { nullptr };
830 ISD::Dome *m_Dome { nullptr };
831 ISD::Camera *m_Camera { nullptr };
832 ISD::Rotator *m_Rotator { nullptr };
833 ISD::FilterWheel *m_FilterWheel { nullptr };
834
835 int currentFilterPosition { -1 };
836 /// True if we need to change filter position and wait for result before continuing capture
837 bool filterPositionPending { false };
838
839 /// Keep track of solver FOV to be plotted in the skymap after each successful solve operation
840 std::shared_ptr<FOV> solverFOV;
841 std::shared_ptr<FOV> sensorFOV;
842
843 /// WCS
844 bool m_wcsSynced { false };
845
846 /// Log
847 QStringList m_LogText;
848
849 /// Issue counters
850 uint8_t m_CaptureTimeoutCounter { 0 };
851 uint8_t m_CaptureErrorCounter { 0 };
852 uint8_t m_SlewErrorCounter { 0 };
853 bool m_resetCaptureTimeoutCounter = false;
854
855 QTimer m_CaptureTimer;
856
857 // State
858 AlignState state { ALIGN_IDLE };
859 FocusState m_FocusState { FOCUS_IDLE };
860 CaptureState m_CaptureState { CAPTURE_IDLE };
861
862 // Track which upload mode the CCD is set to. If set to UPLOAD_LOCAL, then we need to switch it to UPLOAD_CLIENT in order to do focusing, and then switch it back to UPLOAD_LOCAL
863 ISD::Camera::UploadMode rememberUploadMode { ISD::Camera::UPLOAD_CLIENT };
864
865 GotoMode m_CurrentGotoMode;
866
867 QString dirPath;
868
869 // Timer
870 QTimer m_AlignTimer;
871 QTimer m_DebounceTimer;
872
873 // Align Frame
874 QSharedPointer<AlignView> m_AlignView;
875
876 // FITS Viewer in case user want to display in it instead of internal view
877 QSharedPointer<FITSViewer> fv;
878
879 QUrl alignURL;
880 QUrl alignURLPath;
881
882 // keep track of autoWSC
883 bool rememberAutoWCS { false };
884 bool rememberSolverWCS { false };
885
886 // move rotator
887 bool RotatorGOTO { false };
888
889 // Align slew
890 bool targetAccuracyNotMet { false };
891
892 // Astrometry Options
893 OpsAstrometry *opsAstrometry { nullptr };
894 OpsAlign *opsAlign { nullptr };
895 OpsPrograms *opsPrograms { nullptr };
896 OpsAstrometryIndexFiles *opsAstrometryIndexFiles { nullptr };
897 OpsASTAP *opsASTAP { nullptr };
898 StellarSolverProfileEditor *optionsProfileEditor { nullptr };
899
900 // Drawing
901 QCPCurve *centralTarget { nullptr };
902 QCPCurve *yellowTarget { nullptr };
903 QCPCurve *redTarget { nullptr };
904 QCPCurve *concentricRings { nullptr };
905
906 // Telescope Settings
907 double m_EffectiveFocalLength = -1;
908 bool m_isRateSynced = false;
909
910 // CCD Exposure Looping
911 bool m_RememberCameraFastExposure = { false };
912
913 // Controls
914 double alignGainSpecialValue {INVALID_VALUE};
915 double TargetCustomGainValue {-1};
916
917 // Data
918 QSharedPointer<FITSData> m_ImageData;
919
920 // Active Profile
921 QSharedPointer<ProfileInfo> m_ActiveProfile;
922
923 // Threshold to notify settle time is 3 seconds
924 static constexpr uint16_t DELAY_THRESHOLD_NOTIFY { 3000 };
925
926 // Mount Model
927 // N.B. We do not need to use "smart pointer" here as the object memroy
928 // is taken care of by the Qt framework.
929 MountModel *m_MountModel {nullptr};
930 PolarAlignmentAssistant *m_PolarAlignmentAssistant {nullptr};
931 ManualRotator *m_ManualRotator {nullptr};
932
933 // Dark Processor
934 QPointer<DarkProcessor> m_DarkProcessor;
935
936 // Filter Manager
937 QSharedPointer<FilterManager> m_FilterManager;
938
939 // Rotator Control
940 QSharedPointer<RotatorSettings> m_RotatorControlPanel;
941 int m_RotatorTimeFrame = 0;
942 bool m_estimateRotatorTimeFrame = false;
943
944 // Settings
945 QVariantMap m_Settings;
946 QVariantMap m_GlobalSettings;
947
948 bool m_UsedScale = false;
949 bool m_UsedPosition = false;
950 double m_ScaleUsed = 0;
951 double m_RAUsed = 0;
952 double m_DECUsed = 0;
953};
954}
void Slew()
Slew the telescope to the solved alignment coordinate.
Definition align.cpp:2954
static QStringList generateRemoteOptions(const QVariantMap &optionsMap)
generateOptions Generate astrometry.net option given the supplied map
Definition align.cpp:1193
void checkCameraExposureProgress(ISD::CameraChip *targetChip, double remaining, IPState state)
checkCameraExposureProgress Track the progress of CCD exposure
Definition align.cpp:3287
QStringList generateRemoteArgs(const QSharedPointer< FITSData > &imageData)
Generate arguments we pass to the remote solver.
Definition align.cpp:1283
Q_SCRIPTABLE bool setFilter(const QString &filter)
DBUS interface function.
Definition align.cpp:3172
void syncCameraInfo()
CCD information is updated, sync them.
Definition align.cpp:861
bool setCamera(ISD::Camera *device)
Add Camera to the list of available Cameras.
Definition align.cpp:642
Q_SCRIPTABLE QList< double > cameraInfo()
DBUS interface function.
bool setDome(ISD::Dome *device)
Add new Dome.
Definition align.cpp:749
bool setMount(ISD::Mount *device)
Add new mount.
Definition align.cpp:692
Q_SCRIPTABLE int getLoadAndSlewStatus()
DBUS interface function.
Definition align.h:173
void syncCameraControls()
syncCCDControls Update camera controls like gain, offset, ISO..etc.
Definition align.cpp:912
void checkCamera()
Check CCD and make sure information is updated and FOV is re-calculated.
Definition align.cpp:604
Q_SCRIPTABLE Ekos::AlignState status()
DBUS interface function.
Definition align.h:165
bool isParserOK()
Does our parser exist in the system?
Definition align.cpp:530
Q_SCRIPTABLE QList< double > getSolutionResult()
DBUS interface function.
Definition align.cpp:2620
Q_SCRIPTABLE QList< double > telescopeInfo()
DBUS interface function.
QStringList getStellarSolverProfiles()
getStellarSolverProfiles
Definition align.cpp:3927
Q_SCRIPTABLE Q_NOREPLY void startSolving()
DBUS interface function.
Definition align.cpp:1831
void refreshAlignOptions()
refreshAlignOptions is called when settings are updated in OpsAlign.
Definition align.cpp:3484
void getFOVScale(double &fov_w, double &fov_h, double &fov_scale)
getFOVScale Returns calculated FOV values
Definition align.cpp:971
void executeGOTO()
After a solver process is completed successfully, sync, slew to target, or do nothing as set by the u...
Definition align.cpp:2914
bool loadAndSlew(const QByteArray &image, const QString &extension)
DBUS interface function.
Definition align.cpp:3085
void setAstrometryDevice(const QSharedPointer< ISD::GenericDevice > &device)
setAstrometryDevice
Definition align.cpp:3471
void solverFinished(double orientation, double ra, double dec, double pixscale, bool eastToTheRight)
Solver finished successfully, process the data and execute the required actions depending on the mode...
Definition align.cpp:2094
bool checkIfRotationRequired()
checkIfRotationRequired Check whether we need to perform an ALIGN_ROTATING action,...
Definition align.cpp:2455
Q_SCRIPTABLE QString camera()
DBUS interface function.
bool setFilterWheel(ISD::FilterWheel *device)
addFilterWheel Add new filter wheel filter device.
Definition align.cpp:3129
void Sync()
Sync the telescope to the solved alignment coordinate.
Definition align.cpp:2936
void checkFilter()
Check Filter and make sure information is updated accordingly.
Definition align.cpp:3189
void setRotator(ISD::Rotator *device)
Add new Rotator.
Definition align.cpp:3223
Q_SCRIPTABLE QString filterWheel()
DBUS interface function.
void processData(const QSharedPointer< FITSData > &data)
Process new FITS received from CCD.
Definition align.cpp:1710
void getCalculatedFOVScale(double &fov_w, double &fov_h, double &fov_scale)
getCalculatedFOVScale Get calculated FOV scales from the current CCD+Telescope combination.
Definition align.cpp:1005
Q_SCRIPTABLE Q_NOREPLY void setExposure(double value)
DBUS interface function.
Definition align.cpp:3109
void settleSettings()
settleSettings Run this function after timeout from debounce timer to update database and emit settin...
Definition align.cpp:4275
Q_SCRIPTABLE Q_NOREPLY void setSolverAction(int mode)
DBUS interface function.
Definition align.cpp:1825
bool syncTelescopeInfo()
We received new telescope info, process them and update FOV.
Definition align.cpp:818
void updateProperty(INDI::Property prop)
Process updated device properties.
Definition align.cpp:2645
void solverFailed()
Process solver failure.
Definition align.cpp:2374
void SlewToTarget()
Sync the telescope to the solved alignment coordinate, and then slew to the target coordinate.
Definition align.cpp:2983
AstrometryParser is an interface for online and offline astrometry parsers.
The DarkProcessor class.
The PolarAlignmentAssistant class.
RemoteAstrometryParser invokes the remote astrometry.net solver in the remote CCD driver to solve the...
Primary window to view monochrome and color FITS images.
Definition fitsviewer.h:54
A simple class encapsulating a Field-of-View symbol.
Definition fov.h:28
Camera class controls an INDI Camera device.
Definition indicamera.h:45
Class handles control of INDI dome devices.
Definition indidome.h:25
device handle controlling Mounts.
Definition indimount.h:29
Rotator class handles control of INDI Rotator devices.
Definition indirotator.h:20
The QProgressIndicator class lets an application display a progress indicator to show that a long tas...
The sky coordinates of a point in the sky.
Definition skypoint.h:45
This is a subclass of SkyObject.
Definition starobject.h:33
void setTarget(const SkyPoint &targetCoord)
Set the alignment target where the mount is expected to point at.
Definition align.cpp:3820
Q_SCRIPTABLE Q_NOREPLY void setSolverMode(int mode)
DBUS interface function.
Definition align.cpp:561
void stop(Ekos::AlignState mode)
Stop aligning.
Definition align.cpp:2534
void setTelescopeCoordinates(const SkyPoint &position)
Set the coordinates that the mount reports as its position.
Definition align.h:463
Q_SCRIPTABLE Q_NOREPLY void setTargetCoords(double ra0, double de0)
DBUS interface function.
Definition align.cpp:3811
Q_SCRIPTABLE Q_NOREPLY void setBinningIndex(int binIndex)
DBUS interface function.
Definition align.cpp:3114
Q_SCRIPTABLE bool captureAndSolve(bool initialCall=true)
DBUS interface function.
Definition align.cpp:1416
void suspend()
Suspend aligning, recovery handled by the align module itself.
Definition align.h:411
Q_SCRIPTABLE QList< double > getTargetCoords()
getTargetCoords QList of target coordinates.
Definition align.cpp:3827
Q_SCRIPTABLE Q_NOREPLY void abort()
DBUS interface function.
Definition align.h:403
Ekos is an advanced Astrophotography tool for Linux.
Definition align.cpp:83
AlignState
Definition ekos.h:145
@ ALIGN_ABORTED
Alignment aborted by user or agent.
Definition ekos.h:149
@ ALIGN_IDLE
No ongoing operations.
Definition ekos.h:146
@ ALIGN_SUSPENDED
Alignment operations suspended.
Definition ekos.h:155
CaptureState
Capture states.
Definition ekos.h:92
@ CAPTURE_IDLE
Definition ekos.h:93
Q_CLASSINFO(Name, Value)
Q_OBJECTQ_OBJECT
Q_PROPERTY(...)
virtual bool event(QEvent *e)
QString join(QChar separator) const const
QWidget(QWidget *parent, Qt::WindowFlags f)
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Mar 28 2025 11:57:23 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.