/*===================================================================== PIXHAWK Micro Air Vehicle Flying Robotics Toolkit (c) 2009 PIXHAWK PROJECT <http://pixhawk.ethz.ch> This file is part of the PIXHAWK project PIXHAWK 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. PIXHAWK 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 PIXHAWK. If not, see <http://www.gnu.org/licenses/>. ======================================================================*/ /** * @file * @brief Helper functions * * @author Lorenz Meier <mavteam@student.ethz.ch> * */ #ifndef _MG_H_ #define _MG_H_ #include <QDateTime> #include <QDebug> #include <QDir> #include <QThread> #include <cmath> namespace MG { const static int MAX_FLIGHT_TIME = 60 * 60 * 24 * 21; class VERSION { public: static const int MAJOR = 1; static const int MINOR = 01; }; class SYSTEM { public: static const int ID = 127; static const int COMPID = 0; static int getID() { return SYSTEM::ID; } }; class SLEEP : public QThread { public: /** * @brief Set a thread to sleep for seconds * @param s time in seconds to sleep **/ static void sleep(unsigned long s) { QThread::sleep(s); } /** * @brief Set a thread to sleep for milliseconds * @param ms time in milliseconds to sleep **/ static void msleep(unsigned long ms) { QThread::msleep(ms); } /** * @brief Set a thread to sleep for microseconds * @param us time in microseconds to sleep **/ static void usleep(unsigned long us) { QThread::usleep(us); } }; class UNITS { public: /** The distance units supported for display by the groundstation **/ enum DistanceUnit { METER, CENTIMETER, MILLIMETER, INCH, FEET, MILE }; /** * @brief Convert a distance in meters into another distance unit system * * @param in The distance to convert * @param newUnit The new unit to convert to * * @return The converted distance */ static double convertFromMeter(double in, DistanceUnit newUnit) { double result = in; // Conversion table in meter static const double inInch = 39.3700787; static const double inFeet = 3.2808399; static const double inMile = 0.000000621371192; static const double inCentimeter = 100; static const double inMillimeter = 1000; switch (newUnit) { case METER: result = in; break; case CENTIMETER: result = in * inCentimeter; break; case MILLIMETER: result = in * inMillimeter; break; case INCH: result = in * inInch; break; case FEET: result = in * inFeet; break; case MILE: result = in * inMile; break; } return result; } /** * @brief Convert between two distance units * * This convenience function allows to convert between arbitrary distance units * * @param in The input distance * @param inputUnit The input unit * @param outputUnit The output unit * * @return The converted distance */ static double convert(double in, DistanceUnit inputUnit, DistanceUnit outputUnit) { double meters = convertToMeter(in, inputUnit); return convertFromMeter(meters, outputUnit); } /** * @brief Convert a distance to the meter unit * * @param in The distance to convert * @param inputUnit The unit the distance is represented in * * @return The converted distance */ static double convertToMeter(double in, DistanceUnit inputUnit) { double result = in; // Conversion table in meter static const double inInch = 39.3700787; static const double inFeet = 3.2808399; static const double inMile = 0.000000621371192; static const double inCentimeter = 100; static const double inMillimeter = 1000; // Don't convert if new unit is same unit switch (inputUnit) { case METER: result = in; break; case CENTIMETER: result = in / inCentimeter; break; case MILLIMETER: result = in / inMillimeter; break; case INCH: result = in / inInch; break; case FEET: result = in / inFeet; break; case MILE: result = in / inMile; break; } return result; } }; class DISPLAY { public: DISPLAY() { // Initialize static class display with notebook display default value //pixelSize = 0.224f; //setPixelSize(0.224f); } ~DISPLAY() { } /** * @brief Get the size of a single pixel * * This value can be used to generate user interfaces with * a size in physical units, for example a gauge which is * always 50.8 mm (2") in diameter, regardless of the screen. * * @return The horizontal and vertical size of a pixel-square */ static double getPixelSize() { return 0.224f; } /** * @brief Set the size of a single pixel * * @param size The horizontal and vertical size of a pixel-square */ static void setPixelSize(double size) { pixelSize = size; } /** * @brief Set the size of a single pixel * * This method calculates the pixel size from the vertical and horizontal * resolution and the screen diameter. The diameter is in mm (as this unit * is a SI unit). One inch = 25.4 mm * * @param horizontalResolution The horizontal screen resolution, e.g. 1280. * @param verticalResolution The vertical screen resolution, e.g. 800. * @param screenDiameter The screen diameter in mm, e.g. 13.3" = 338 mm. */ static void setPixelSize(int horizontalResolution, int verticalResolution, double screenDiameter) { pixelSize = screenDiameter / sqrt(static_cast<float>(horizontalResolution*horizontalResolution + verticalResolution*verticalResolution)); } private: /** The size of a single pixel **/ static double pixelSize; }; class STAT { /** The time interval for the last few moments in milliseconds **/ static const int SHORT_TERM_INTERVAL = 300; /** The time interval for the last moment in milliseconds **/ static const int CURRENT_INTERVAL = 50; }; class TIME { public: //static const QString ICONDIR = "./icons"; /** * @brief Convenience method to get the milliseconds time stamp for now * * The timestamp is created at the instant of calling this method. It is * defined as the number of milliseconds since unix epoch, which is * 1.1.1970, 00:00 UTC. * * @return The number of milliseconds elapsed since unix epoch * @deprecated Will the replaced by time helper class **/ static quint64 getGroundTimeNow() { QDateTime time = QDateTime::currentDateTime(); time = time.toUTC(); /* Return seconds and milliseconds, in milliseconds unit */ quint64 milliseconds = time.toTime_t() * static_cast<quint64>(1000); return static_cast<quint64>(milliseconds + time.time().msec()); } /** * @brief Convenience method to get the milliseconds time stamp for now * * The timestamp is created at the instant of calling this method. It is * defined as the number of milliseconds since unix epoch, which is * 1.1.1970, 00:00 UTC. * * @return The number of milliseconds elapsed since unix epoch * @deprecated Will the replaced by time helper class **/ static quint64 getGroundTimeNowUsecs() { QDateTime time = QDateTime::currentDateTime(); time = time.toUTC(); /* Return seconds and milliseconds, in milliseconds unit */ quint64 microseconds = time.toTime_t() * static_cast<quint64>(1000000); return static_cast<quint64>(microseconds + (time.time().msec()*1000)); } /*tatic quint64 getMissionTimeUsecs() { ; }*/ /** * Convert milliseconds to an QDateTime object. This method converts the amount of * milliseconds since 1.1.1970, 00:00 UTC (unix epoch) to a QDateTime date object. * * @param msecs The milliseconds since unix epoch (in Qt unsigned 64bit integer type quint64) * @return The QDateTime object set to corresponding date and time * @deprecated Will the replaced by time helper class **/ static QDateTime msecToQDateTime(quint64 msecs) { QDateTime time = QDateTime(); /* Set date and time depending on the seconds since unix epoch, * integer division truncates the milliseconds */ time.setTime_t(msecs / 1000); /* Add the milliseconds, modulo returns the milliseconds part */ return time.addMSecs(msecs % 1000); } }; class DIR { public: /** * @brief Get the current support file directory. * * The support files are files like icons or fonts and are typically found in the * same directory as the main executable. * * @return The absolute path of the directory **/ static QString getSupportFilesDirectory() { // Checks if the software is executed in the development environment QString path = QDir::current().absolutePath(); QDir currentDir = QDir::current(); if (QDir::current().dirName().toLower() == "debug") { // Debug folder of development environment path.append("/.."); } else if (QDir::current().dirName().toLower() == "release") { // Release folder of development environment path.append("/.."); } else if (QDir::current().dirName().toLower() == "bin") { // Release folder of development environment path.append("/.."); } else if (QDir::current().dirName().toLower() == "macos") { // Mac application bundle in development environment path.append("/../../../../.."); } // Check if we are still in a development folder if(currentDir.cdUp()) { if(currentDir.dirName().toLower() == "build") { path.append("/.."); } } //TODO The Mac application bundle in distribution is not yet included here //qDebug() << "MG::supportfilesdirectory" << path; return path; } /** * @brief Get the current icon directory. * * The icon directory is typically a subdirectory of the main directory, * but depends on the platform. For example in OS X it is part of the bundle. * * @return The absolute path of the icon directory **/ static QString getIconDirectory() { return MG::DIR::getSupportFilesDirectory() + "/files/images/"; } }; } #endif // _MG_H_