/* Copyright (C) 2013 VATSIM Community / contributors * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "blackmisc/coordinategeodetic.h" #include "blackmisc/blackmiscfreefunctions.h" #include "mathematics.h" #include using namespace BlackMisc::PhysicalQuantities; using namespace BlackMisc::Math; namespace BlackMisc { namespace Geo { /* * String for converter */ QString CCoordinateGeodetic::convertToQString(bool i18n) const { QString s = "Geodetic: {%1, %2, %3}"; return s.arg(this->m_latitude.valueRoundedWithUnit(6, i18n)).arg(this->m_longitude.valueRoundedWithUnit(6, i18n)).arg(this->m_geodeticHeight.valueRoundedWithUnit(6, i18n)); } /* * metaTypeId */ int CCoordinateGeodetic::getMetaTypeId() const { return qMetaTypeId(); } /* * is a */ bool CCoordinateGeodetic::isA(int metaTypeId) const { if (metaTypeId == qMetaTypeId()) { return true; } return this->CValueObject::isA(metaTypeId); } /* * Compare */ int CCoordinateGeodetic::compareImpl(const CValueObject &otherBase) const { const auto &other = static_cast(otherBase); return compare(TupleConverter::toTuple(*this), TupleConverter::toTuple(other)); } /* * Marshall to Dbus */ void CCoordinateGeodetic::marshallToDbus(QDBusArgument &argument) const { argument << TupleConverter::toTuple(*this); } /* * Unmarshall from Dbus */ void CCoordinateGeodetic::unmarshallFromDbus(const QDBusArgument &argument) { argument >> TupleConverter::toTuple(*this); } /* * Same coordinate */ bool CCoordinateGeodetic::operator ==(const CCoordinateGeodetic &other) const { if (this == &other) return true; return TupleConverter::toTuple(*this) == TupleConverter::toTuple(other); } /* * Unequal? */ bool CCoordinateGeodetic::operator !=(const CCoordinateGeodetic &other) const { return !((*this) == other); } /* * Register metadata */ void CCoordinateGeodetic::registerMetadata() { qRegisterMetaType(); qDBusRegisterMetaType(); } /* * Hash */ uint CCoordinateGeodetic::getValueHash() const { return qHash(TupleConverter::toTuple(*this)); } /* * To JSON */ QJsonObject CCoordinateGeodetic::toJson() const { return BlackMisc::serializeJson(CCoordinateGeodetic::jsonMembers(), TupleConverter::toTuple(*this)); } /* * To JSON */ void CCoordinateGeodetic::fromJson(const QJsonObject &json) { BlackMisc::deserializeJson(json, CCoordinateGeodetic::jsonMembers(), TupleConverter::toTuple(*this)); } /* * Members */ const QStringList &CCoordinateGeodetic::jsonMembers() { return TupleConverter::jsonMembers(); } /* * From WGS84 coordinates */ CCoordinateGeodetic CCoordinateGeodetic::fromWgs84(const QString &latitudeWgs84, const QString &longitudeWgs84, const CLength height) { CLatitude lat = CLatitude::fromWgs84(latitudeWgs84); CLongitude lon = CLongitude::fromWgs84(longitudeWgs84); return CCoordinateGeodetic(lat, lon, height); } /* * Great circle distance */ PhysicalQuantities::CLength greatCircleDistance(const ICoordinateGeodetic &coordinate1, const ICoordinateGeodetic &coordinate2) { // same coordinate results in 0 distance if (coordinate1.latitude() == coordinate2.latitude() && coordinate1.longitude() == coordinate2.longitude()) { return CLength(0, CLengthUnit::NM()); } // first, prelimary distance calculation // http://www.geodatasource.com/developers/c double dist; double lon1rad = coordinate1.longitude().value(CAngleUnit::rad()); double lon2rad = coordinate2.longitude().value(CAngleUnit::rad()); double lat1rad = coordinate1.latitude().value(CAngleUnit::rad()); double lat2rad = coordinate2.latitude().value(CAngleUnit::rad()); double theta = lon1rad - lon2rad; dist = qSin(lat1rad) * qSin(lat2rad) + qCos(lat1rad) * qCos(lat2rad) * cos(theta); dist = qAcos(dist); dist = CMath::rad2deg(dist); dist = dist * 60; // dist in NM return CLength(qAbs(dist), CLengthUnit::NM()); } PhysicalQuantities::CAngle initialBearing(const ICoordinateGeodetic &coordinate1, const ICoordinateGeodetic &coordinate2) { // same coordinate results in 0 distance if (coordinate1.latitude() == coordinate2.latitude() && coordinate1.longitude() == coordinate2.longitude()) { return CAngle(0, CAngleUnit::deg()); } // http://www.yourhomenow.com/house/haversine.html double lon1rad = coordinate1.longitude().value(CAngleUnit::rad()); double lon2rad = coordinate2.longitude().value(CAngleUnit::rad()); double lat1rad = coordinate1.latitude().value(CAngleUnit::rad()); double lat2rad = coordinate2.latitude().value(CAngleUnit::rad()); double dLon = lon1rad - lon2rad; double y = qSin(dLon) * qCos(lat2rad); double x = qCos(lat1rad) * qSin(lat2rad) - qSin(lat1rad) * qCos(lat2rad) * qCos(dLon); double bearing = qAtan2(y, x); bearing = CMath::rad2deg(bearing); // now in deg bearing = CMath::normalizeDegrees(bearing); // normalize return CAngle(bearing, CAngleUnit::deg()); } /* * Great circle distance */ CLength ICoordinateGeodetic::greatCircleDistance(const ICoordinateGeodetic &otherCoordinate) { return Geo::greatCircleDistance((*this), otherCoordinate); } /* * Initial bearing */ CAngle ICoordinateGeodetic::initialBearing(const ICoordinateGeodetic &otherCoordinate) { return Geo::initialBearing((*this), otherCoordinate); } /* * My index */ bool ICoordinateGeodetic::indexInRange(int index) { return index >= static_cast(IndexLatitude) && index <= static_cast(IndexLongitudeAsString); } /* * Property by index */ QVariant ICoordinateGeodetic::propertyByIndex(int index) const { switch (index) { case IndexLatitude: return this->latitude().toQVariant(); case IndexLongitude: return this->longitude().toQVariant(); case IndexLatitudeAsString: return QVariant(this->latitudeAsString()); case IndexLongitudeAsString: return QVariant(this->longitudeAsString()); default: break; } Q_ASSERT_X(false, "ICoordinateGeodetic", "index unknown"); QString m = QString("no property, index ").append(QString::number(index)); return QVariant::fromValue(m); } /* * Property by index */ QVariant CCoordinateGeodetic::propertyByIndex(int index) const { if (ICoordinateGeodetic::indexInRange(index)) { return ICoordinateGeodetic::propertyByIndex(index); } switch (index) { case IndexGeodeticHeight: return this->m_geodeticHeight.toQVariant(); break; default: break; } Q_ASSERT_X(false, "CCoordinateGeodetic", "index unknown"); QString m = QString("no property, index ").append(QString::number(index)); return QVariant::fromValue(m); } /* * Set property as index */ void CCoordinateGeodetic::setPropertyByIndex(const QVariant &variant, int index) { switch (index) { case IndexGeodeticHeight: this->setGeodeticHeight(variant.value()); break; case IndexLatitude: this->setLatitude(variant.value()); break; case IndexLongitude: this->setLongitude(variant.value()); break; case IndexLatitudeAsString: this->setLatitude(CLatitude::fromWgs84(variant.toString())); break; case IndexLongitudeAsString: this->setLongitude(CLongitude::fromWgs84(variant.toString())); break; default: Q_ASSERT_X(false, "CCoordinateGeodetic", "index unknown (setter)"); break; } } } // namespace } // namespace