OpenCPN Partial API docs
Loading...
Searching...
No Matches
comm_appmsg.cpp
1/***************************************************************************
2 *
3 * Project: OpenCPN
4 * Purpose: Implement comm_appmsg.h -- Decoded application messages.
5 * Author: David Register, Alec Leamas
6 *
7 ***************************************************************************
8 * Copyright (C) 2022 by David Register, Alec Leamas *
9 * *
10 * This program is free software; you can redistribute it and/or modify *
11 * it under the terms of the GNU General Public License as published by *
12 * the Free Software Foundation; either version 2 of the License, or *
13 * (at your option) any later version. *
14 * *
15 * This program is distributed in the hope that it will be useful, *
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
18 * GNU General Public License for more details. *
19 * *
20 * You should have received a copy of the GNU General Public License *
21 * along with this program; if not, write to the *
22 * Free Software Foundation, Inc., *
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
24 **************************************************************************/
25
26// For compilers that support precompilation, includes "wx.h".
27#include <wx/wxprec.h>
28
29#ifndef WX_PRECOMP
30#include <wx/wx.h>
31#endif // precompiled headers
32
33#include <sstream>
34#include <iomanip>
35
36#include "model/comm_appmsg.h"
37#include "model/ocpn_utils.h"
38
39static const auto kUtfDegrees = wxString::FromUTF8(u8"\u00B0");
40
41/* Free functions. */
42
43std::string TimeToString(const time_t t) {
44 char buff[30];
45#ifdef _MSC_VER
46 errno_t e = ctime_s(buff, sizeof(buff), &t);
47 assert(e == 0 && "Huh? ctime_s returned an error");
48 return std::string(buff);
49#else
50 const char* r = ctime_r(&t, buff);
51 assert(r != NULL && "ctime_r failed...");
52 return std::string(buff);
53#endif
54}
55
56std::string DegreesToString(double degrees) {
57 using namespace std;
58 std::stringstream buf;
59 buf << setw(2) << static_cast<int>(trunc(degrees)) << kUtfDegrees
60 << static_cast<int>(trunc(degrees * 100)) % 100 << "," << setw(2)
61 << (static_cast<int>(trunc(degrees * 10000)) % 10000) % 100;
62 return buf.str();
63}
64
65double PosPartsToDegrees(float degrees, float minutes,
66 float percent_of_minute) {
67 return degrees + minutes / 60 + percent_of_minute / 6000;
68}
69
70/* Position implementation */
71
72Position::Position(double _lat, double _lon, Type t)
73 : lat(TypeToLat(t, _lat)), lon(TypeToLong(t, _lon)), type(t) {}
74
75Position::Position(double _lat, double _lon)
76 : lat(_lat), lon(_lon), type(LatLongToType(_lat, _lon)) {};
77
78Position::Position() : lat(0), lon(0), type(Type::Undef) {};
79
80std::string Position::to_string() const {
81 std::stringstream buf;
82 const std::string NE(TypeToStr(type));
83 auto lat_s = DegreesToString(abs(lat));
84 auto lon_s = DegreesToString(abs(lon));
85 buf << lat_s << NE[0] << " " << lon_s << NE[1];
86 return buf.str();
87}
88
89std::string Position::TypeToStr(const Type t) const {
90 switch (t) {
91 case Type::NE:
92 return "NE";
93 break;
94 case Type::NW:
95 return "NW";
96 break;
97 case Type::SE:
98 return "SE";
99 break;
100 case Type::SW:
101 return "SW";
102 break;
103 case Type::Undef:
104 return "Undefined";
105 break;
106 }
107 return "??"; // Not reached, but compiler complains.
108}
109
110Position::Type Position::LatLongToType(double lat, double lon) {
111 if (lat >= 0)
112 return lon >= 0 ? Type::NE : Type::NW;
113 else
114 return lon >= 0 ? Type::SE : Type::SW;
115}
116
117double Position::TypeToLat(Type t, double lat) {
118 return t == Type::SW || t == Type::SE ? -lat : lat;
119}
120
121double Position::TypeToLong(Type t, double lon) {
122 return t == Type::NE || t == Type::SE ? lon : -lon;
123}
124
126static double GgaPartToDouble(const std::string& s) {
127 size_t dotpos = s.find('.');
128 if (dotpos < 2) return nan("");
129 auto degrees = s.substr(0, dotpos - 2);
130 auto minutes = s.substr(dotpos - 2);
131 return std::stod(degrees) + std::stod(minutes) / 60;
132}
133
134Position Position::ParseGGA(const std::string gga) {
135 auto parts = ocpn::split(gga.c_str(), ",");
136 if (parts.size() != 4) {
137 return Position();
138 }
139 double lat = GgaPartToDouble(parts[0]);
140 if (parts[1] == "S")
141 lat = -lat;
142 else if (parts[1] != "N")
143 lat = nan("");
144 double lon = GgaPartToDouble(parts[2]);
145 if (parts[3] == "W")
146 lon = -lon;
147 else if (parts[3] != "E")
148 lon = nan("");
149
150 return lat != nan("") && lon != nan("") ? Position(lat, lon) : Position();
151}
152
153/* Appmsg implementation */
154
155std::string AppMsg::TypeToString(const AppMsg::Type t) const {
156 switch (t) {
157 case AppMsg::Type::AisData:
158 return "ais-data";
159 break;
160 case AppMsg::Type::BasicNavData:
161 return "basic-nav-data";
162 break;
163 case AppMsg::Type::CustomMsg:
164 return "custom-msg";
165 break;
166 case AppMsg::Type::DataPrioNeeded:
167 return "data-prio-needed";
168 break;
169 case AppMsg::Type::GnssFix:
170 return "gnss-fix";
171 break;
172 case AppMsg::Type::GPSWatchdog:
173 return "gps-watchdog";
174 break;
175 case AppMsg::Type::Undef:
176 return "??";
177 break;
178 }
179 return "????"; // Not reached, for the compiler.
180}
181
182std::string BasicNavDataMsg::to_string() const {
183 std::stringstream ss;
184 ss << AppMsg::to_string() << " pos: " << pos.to_string() << " sog: " << sog
185 << " cog: " << cog << " var: " << var << " hdt: " << hdt
186 << " vflag: " << vflag << " set_time: " << TimeToString(set_time.tv_sec);
187 return ss.str();
188}
189std::string AisData::to_string() const {
190 std::stringstream ss;
191 ss << TimeToString(time) << " " << pos.to_string() << " sog: " << sog
192 << " cog: " << cog << " heading: " << heading
193 << " rate of turn: " << rate_of_turn << "type: 0x" << std::hex << type
194 << " name " << name << " callsign " << callsign << " dest: " << dest
195 << " length " << std::dec << length << " beam: " << beam
196 << " draft: " << draft << " status: 0x" << std::hex << status;
197 return ss.str();
198}
Position()
Construct a (0,0) position, type == Undef.
static Position ParseGGA(const std::string gga)
Parse a GGA string like "5800.588,N,01145.776,E" as present in GGA and other n0183 messages.
std::string to_string() const
Return utf string like 65°25,11N 21°12,01E.
std::vector< std::string > split(const char *token_string, const std::string &delimiter)
Return vector of items in s separated by delimiter.
Miscellaneous utilities, many of which string related.