/*===================================================================== QGroundControl Open Source Ground Control Station (c) 2009, 2010 QGROUNDCONTROL PROJECT This file is part of the QGROUNDCONTROL project QGROUNDCONTROL is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. QGROUNDCONTROL is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with QGROUNDCONTROL. If not, see . ======================================================================*/ /** * @file * @brief Abstract interface, represents one unmanned aerial vehicle * * @author Lorenz Meier * */ #ifndef _UASINTERFACE_H_ #define _UASINTERFACE_H_ #include #include #include #include #include #include "LinkInterface.h" #include "ProtocolInterface.h" #include "UASWaypointManager.h" class FileManager; enum BatteryType { NICD = 0, NIMH = 1, LIION = 2, LIPOLY = 3, LIFE = 4, AGZN = 5 }; ///< The type of battery used /** * @brief Interface for all robots. * * This interface is abstract and thus cannot be instantiated. It serves only as type definition. * It represents an unmanned aerial vehicle, e.g. a micro air vehicle. **/ class UASInterface : public QObject { Q_OBJECT public: virtual ~UASInterface() {} /* MANAGEMENT */ /** @brief The name of the robot **/ virtual QString getUASName() const = 0; virtual int getUASID() const = 0; ///< Get the ID of the connected UAS /** @brief The time interval the robot is switched on **/ virtual quint64 getUptime() const = 0; virtual double getLocalX() const = 0; virtual double getLocalY() const = 0; virtual double getLocalZ() const = 0; virtual bool localPositionKnown() const = 0; virtual double getLatitude() const = 0; virtual double getLongitude() const = 0; virtual double getAltitudeAMSL() const = 0; virtual double getAltitudeRelative() const = 0; virtual bool globalPositionKnown() const = 0; virtual double getRoll() const = 0; virtual double getPitch() const = 0; virtual double getYaw() const = 0; /** @brief Set the airframe of this MAV */ virtual int getAirframe() const = 0; /** @brief Get reference to the waypoint manager **/ virtual UASWaypointManager* getWaypointManager(void) = 0; virtual FileManager* getFileManager() = 0; enum Airframe { QGC_AIRFRAME_GENERIC = 0, QGC_AIRFRAME_EASYSTAR, QGC_AIRFRAME_TWINSTAR, QGC_AIRFRAME_MERLIN, QGC_AIRFRAME_CHEETAH, QGC_AIRFRAME_MIKROKOPTER, QGC_AIRFRAME_REAPER, QGC_AIRFRAME_PREDATOR, QGC_AIRFRAME_COAXIAL, QGC_AIRFRAME_PTERYX, QGC_AIRFRAME_TRICOPTER, QGC_AIRFRAME_HEXCOPTER, QGC_AIRFRAME_X8, QGC_AIRFRAME_VIPER_2_0, QGC_AIRFRAME_CAMFLYER_Q, QGC_AIRFRAME_END_OF_ENUM }; /** * @brief Get the color for this UAS * * This static function holds a color map that allows to draw a new color for each robot * * @return The next color in the color map. The map holds 20 colors and starts from the beginning * if the colors are exceeded. */ static QColor getNextColor() { /* Create color map */ static QList colors = QList() << QColor(231,72,28) << QColor(104,64,240) << QColor(203,254,121) << QColor(161,252,116) << QColor(232,33,47) << QColor(116,251,110) << QColor(234,38,107) << QColor(104,250,138) << QColor(235,43,165) << QColor(98,248,176) << QColor(236,48,221) << QColor(92,247,217) << QColor(200,54,238) << QColor(87,231,246) << QColor(151,59,239) << QColor(81,183,244) << QColor(75,133,243) << QColor(242,255,128) << QColor(230,126,23); static int nextColor = -1; if(nextColor == 18){//if at the end of the list nextColor = -1;//go back to the beginning } nextColor++; return colors[nextColor];//return the next color } /** @brief Get the type of the system (airplane, quadrotor, helicopter,..)*/ virtual int getSystemType() = 0; /** @brief Is it an airplane (or like one)?,..)*/ virtual bool isAirplane() = 0; /** @brief Get the type of the autopilot (PIXHAWK, APM, UDB, PPZ,..) */ virtual int getAutopilotType() = 0; virtual void setAutopilotType(int apType) = 0; virtual QMap getComponents() = 0; QColor getColor() { return color; } static const unsigned int WAYPOINT_RADIUS_DEFAULT_FIXED_WING = 25; static const unsigned int WAYPOINT_RADIUS_DEFAULT_ROTARY_WING = 5; enum StartCalibrationType { StartCalibrationRadio, StartCalibrationGyro, StartCalibrationMag, StartCalibrationAirspeed, StartCalibrationAccel, StartCalibrationLevel, StartCalibrationEsc, StartCalibrationCopyTrims, StartCalibrationUavcanEsc }; enum StartBusConfigType { StartBusConfigActuators, EndBusConfigActuators, }; /// Starts the specified calibration virtual void startCalibration(StartCalibrationType calType) = 0; /// Ends any current calibration virtual void stopCalibration(void) = 0; /// Starts the specified bus configuration virtual void startBusConfig(StartBusConfigType calType) = 0; /// Ends any current bus configuration virtual void stopBusConfig(void) = 0; public slots: /** @brief Executes a command **/ virtual void executeCommand(MAV_CMD command, int confirmation, float param1, float param2, float param3, float param4, float param5, float param6, float param7, int component) = 0; /** @brief Selects the airframe */ virtual void setAirframe(int airframe) = 0; /** @brief Order the robot to pair its receiver **/ virtual void pairRX(int rxType, int rxSubType) = 0; virtual void setHomePosition(double lat, double lon, double alt) = 0; /** @brief Return if this a rotary wing */ virtual bool isRotaryWing() = 0; /** @brief Return if this is a fixed wing */ virtual bool isFixedWing() = 0; /** @brief Send the full HIL state to the MAV */ #ifndef __mobile__ virtual void sendHilState(quint64 time_us, float roll, float pitch, float yaw, float rollspeed, float pitchspeed, float yawspeed, double lat, double lon, double alt, float vx, float vy, float vz, float ind_airspeed, float true_airspeed, float xacc, float yacc, float zacc) = 0; /** @brief RAW sensors for sensor HIL */ virtual void sendHilSensors(quint64 time_us, float xacc, float yacc, float zacc, float rollspeed, float pitchspeed, float yawspeed, float xmag, float ymag, float zmag, float abs_pressure, float diff_pressure, float pressure_alt, float temperature, quint32 fields_changed) = 0; /** @brief Send raw GPS for sensor HIL */ virtual void sendHilGps(quint64 time_us, double lat, double lon, double alt, int fix_type, float eph, float epv, float vel, float vn, float ve, float vd, float cog, int satellites) = 0; /** @brief Send Optical Flow sensor message for HIL, (arguments and units accoding to mavlink documentation*/ virtual void sendHilOpticalFlow(quint64 time_us, qint16 flow_x, qint16 flow_y, float flow_comp_m_x, float flow_comp_m_y, quint8 quality, float ground_distance) = 0; #endif /** @brief Send command to map a RC channel to a parameter */ virtual void sendMapRCToParam(QString param_id, float scale, float value0, quint8 param_rc_channel_index, float valueMin, float valueMax) = 0; /** @brief Send command to disable all bindings/maps between RC and parameters */ virtual void unsetRCToParameterMap() = 0; protected: QColor color; signals: /** @brief The robot state has changed */ void statusChanged(int stateFlag); /** @brief The robot state has changed * * @param uas this robot * @param status short description of status, e.g. "connected" * @param description longer textual description. Should be however limited to a short text, e.g. 200 chars. */ void statusChanged(UASInterface* uas, QString status, QString description); /** @brief A text message from the system has been received */ void textMessageReceived(int uasid, int componentid, int severity, QString text); /** * @brief Update the error count of a device * * The error count indicates how many errors occured during the use of a device. * Usually a random error from time to time is acceptable, e.g. through electromagnetic * interferences on device lines like I2C and SPI. A constantly and rapidly increasing * error count however can help to identify broken cables or misbehaving drivers. * * @param uasid System ID * @param component Name of the component, e.g. "IMU" * @param device Name of the device, e.g. "SPI0" or "I2C1" * @param count Errors occured since system startup */ void errCountChanged(int uasid, QString component, QString device, int count); /** * @brief Drop rate of communication link updated * * @param systemId id of the air system * @param receiveDrop drop rate of packets this MAV receives (sent from GCS or other MAVs) */ void dropRateChanged(int systemId, float receiveDrop); /** @brief The robot is connected **/ void connected(); /** @brief The robot is disconnected **/ void disconnected(); /** @brief A value of the robot has changed. * * Typically this is used to send lowlevel information like the battery voltage to the plotting facilities of * the groundstation. The data here should be converted to human-readable values before being passed, so ideally * SI units. * * @param uasId ID of this system * @param name name of the value, e.g. "battery voltage" * @param unit The units this variable is in as an abbreviation. For system-dependent (such as raw ADC values) use "raw", for bitfields use "bits", for true/false or on/off use "bool", for unitless values use "-". * @param value the value that changed * @param msec the timestamp of the message, in milliseconds */ void valueChanged(const int uasid, const QString& name, const QString& unit, const QVariant &value,const quint64 msecs); void parameterUpdate(int uas, int component, QString parameterName, int parameterCount, int parameterId, int type, QVariant value); /** * @brief The battery status has been updated * * @param uas sending system * @param voltage battery voltage * @param percent remaining capacity in percent * @param seconds estimated remaining flight time in seconds */ void batteryChanged(UASInterface* uas, double voltage, double current, double percent, int seconds); void batteryConsumedChanged(UASInterface* uas, double current_consumed); void statusChanged(UASInterface* uas, QString status); void thrustChanged(UASInterface*, double thrust); void heartbeat(UASInterface* uas); void attitudeChanged(UASInterface*, double roll, double pitch, double yaw, quint64 usec); void attitudeChanged(UASInterface*, int component, double roll, double pitch, double yaw, quint64 usec); void attitudeRotationRatesChanged(int uas, double rollrate, double pitchrate, double yawrate, quint64 usec); void attitudeThrustSetPointChanged(UASInterface*, float rollDesired, float pitchDesired, float yawDesired, float thrustDesired, quint64 usec); /** @brief The MAV set a new setpoint in the local (not body) NED X, Y, Z frame */ void positionSetPointsChanged(int uasid, float xDesired, float yDesired, float zDesired, float yawDesired, quint64 usec); /** @brief A user (or an autonomous mission or obstacle avoidance planner) requested to set a new setpoint */ void userPositionSetPointsChanged(int uasid, float xDesired, float yDesired, float zDesired, float yawDesired); void localPositionChanged(UASInterface*, double x, double y, double z, quint64 usec); void localPositionChanged(UASInterface*, int component, double x, double y, double z, quint64 usec); void globalPositionChanged(UASInterface*, double lat, double lon, double altAMSL, double altWGS84, quint64 usec); void altitudeChanged(UASInterface*, double altitudeAMSL, double altitudeWGS84, double altitudeRelative, double climbRate, quint64 usec); /** @brief Update the status of one satellite used for localization */ void gpsSatelliteStatusChanged(int uasid, int satid, float azimuth, float direction, float snr, bool used); // The horizontal speed (a scalar) void speedChanged(UASInterface* uas, double groundSpeed, double airSpeed, quint64 usec); // Consider adding a MAV_FRAME parameter to this; could help specifying what the 3 scalars are. void velocityChanged_NED(UASInterface*, double vx, double vy, double vz, quint64 usec); void navigationControllerErrorsChanged(UASInterface*, double altitudeError, double speedError, double xtrackError); void NavigationControllerDataChanged(UASInterface *uas, float navRoll, float navPitch, float navBearing, float targetBearing, float targetDist); void imageStarted(int imgid, int width, int height, int depth, int channels); void imageDataReceived(int imgid, const unsigned char* imageData, int length, int startIndex); /** @brief Emit the new system type */ void systemTypeSet(UASInterface* uas, unsigned int type); /** @brief Attitude control enabled/disabled */ void attitudeControlEnabled(bool enabled); /** @brief Position 2D control enabled/disabled */ void positionXYControlEnabled(bool enabled); /** @brief Altitude control enabled/disabled */ void positionZControlEnabled(bool enabled); /** @brief Heading control enabled/disabled */ void positionYawControlEnabled(bool enabled); /** @brief Optical flow status changed */ void opticalFlowStatusChanged(bool supported, bool enabled, bool ok); /** @brief Vision based localization status changed */ void visionLocalizationStatusChanged(bool supported, bool enabled, bool ok); /** @brief Infrared / Ultrasound status changed */ void distanceSensorStatusChanged(bool supported, bool enabled, bool ok); /** @brief Gyroscope status changed */ void gyroStatusChanged(bool supported, bool enabled, bool ok); /** @brief Accelerometer status changed */ void accelStatusChanged(bool supported, bool enabled, bool ok); /** @brief Magnetometer status changed */ void magSensorStatusChanged(bool supported, bool enabled, bool ok); /** @brief Barometer status changed */ void baroStatusChanged(bool supported, bool enabled, bool ok); /** @brief Differential pressure / airspeed status changed */ void airspeedStatusChanged(bool supported, bool enabled, bool ok); /** @brief Value of a remote control channel (raw) */ void remoteControlChannelRawChanged(int channelId, float raw); /** @brief Value of a remote control channel (scaled)*/ void remoteControlChannelScaledChanged(int channelId, float normalized); /** @brief Remote control RSSI changed (0% - 100%)*/ void remoteControlRSSIChanged(uint8_t rssi); /** * @brief Localization quality changed * @param fix 0: lost, 1: 2D local position hold, 2: 2D localization, 3: 3D localization */ void localizationChanged(UASInterface* uas, int fix); // ERROR AND STATUS SIGNALS /** @brief Heartbeat timed out or was regained */ void heartbeatTimeout(bool timeout, unsigned int ms); /** @brief Name of system changed */ void nameChanged(QString newName); /** @brief Core specifications have changed */ void systemSpecsChanged(int uasId); // HOME POSITION / ORIGIN CHANGES void homePositionChanged(int uas, double lat, double lon, double alt); protected: // TIMEOUT CONSTANTS static const unsigned int timeoutIntervalHeartbeat = 3500 * 1000; ///< Heartbeat timeout is 3.5 seconds }; Q_DECLARE_INTERFACE(UASInterface, "org.qgroundcontrol/1.0") #endif // _UASINTERFACE_H_