36#include "model/comm_appmsg.h"
39static const auto kUtfDegrees = wxString::FromUTF8(u8
"\u00B0");
43std::string TimeToString(
const time_t t) {
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);
50 const char* r = ctime_r(&t, buff);
51 assert(r != NULL &&
"ctime_r failed...");
52 return std::string(buff);
56std::string DegreesToString(
double degrees) {
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;
65double PosPartsToDegrees(
float degrees,
float minutes,
66 float percent_of_minute) {
67 return degrees + minutes / 60 + percent_of_minute / 6000;
73 : lat(TypeToLat(t, _lat)), lon(TypeToLong(t, _lon)), type(t) {}
76 : lat(_lat), lon(_lon), type(LatLongToType(_lat, _lon)) {};
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];
89std::string Position::TypeToStr(
const Type t)
const {
110Position::Type Position::LatLongToType(
double lat,
double lon) {
112 return lon >= 0 ? Type::NE : Type::NW;
114 return lon >= 0 ? Type::SE : Type::SW;
117double Position::TypeToLat(Type t,
double lat) {
118 return t == Type::SW || t == Type::SE ? -lat : lat;
121double Position::TypeToLong(Type t,
double lon) {
122 return t == Type::NE || t == Type::SE ? lon : -lon;
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;
136 if (parts.size() != 4) {
139 double lat = GgaPartToDouble(parts[0]);
142 else if (parts[1] !=
"N")
144 double lon = GgaPartToDouble(parts[2]);
147 else if (parts[3] !=
"E")
155std::string AppMsg::TypeToString(
const AppMsg::Type t)
const {
157 case AppMsg::Type::AisData:
160 case AppMsg::Type::BasicNavData:
161 return "basic-nav-data";
163 case AppMsg::Type::CustomMsg:
166 case AppMsg::Type::DataPrioNeeded:
167 return "data-prio-needed";
169 case AppMsg::Type::GnssFix:
172 case AppMsg::Type::GPSWatchdog:
173 return "gps-watchdog";
175 case AppMsg::Type::Undef:
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);
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;
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.