41#include <wx/datetime.h>
45#include <wx/textfile.h>
47#include <wx/tokenzr.h>
48#include <wx/filename.h>
52#include "rapidjson/document.h"
53#include "rapidjson/writer.h"
54#include "rapidjson/stringbuffer.h"
56#include "model/ais_decoder.h"
58#include "model/meteo_points.h"
59#include "model/ais_target_data.h"
61#include "model/config_vars.h"
62#include "model/geodesic.h"
63#include "model/georef.h"
64#include "model/idents.h"
65#include "model/multiplexer.h"
66#include "model/navutil_base.h"
67#include "model/own_ship.h"
68#include "model/route_point.h"
69#include "model/select.h"
70#include "SoundFactory.h"
71#include "model/track.h"
75static const long long lNaN = 0xfff8000000000000;
76#define NAN (*(double *)&lNaN)
79wxEvtHandler *g_pais_alert_dialog_active;
83bool g_bUseOnlyConfirmedAISName;
84wxString GetShipNameFromFile(
int);
85wxString AISTargetNameFileName;
86bool isBuoyMmsi(
const int);
109EVT_TIMER(TIMER_AIS1, AisDecoder::OnTimerAIS)
110EVT_TIMER(TIMER_DSC, AisDecoder::OnTimerDSC)
113static const
double ms_to_knot_factor = 1.9438444924406;
119static
bool b_firstrx;
120static
int first_rx_ticks;
122static
double arpa_ref_hdg = NAN;
124static inline
double GeodesicRadToDeg(
double rads) {
125 return rads * 180.0 / M_PI;
128static inline double MS2KNOTS(
double ms) {
return ms * 1.9438444924406; }
130int AisMeteoNewMmsi(
int,
int,
int,
int,
int);
134 AIS_Target_Name_Hash *AISTargetNamesC,
135 AIS_Target_Name_Hash *AISTargetNamesNC,
long mmsi);
138 : m_signalk_selfid(
""), m_callbacks(callbacks) {
140 AISTargetNamesC =
new AIS_Target_Name_Hash;
141 AISTargetNamesNC =
new AIS_Target_Name_Hash;
143 if (g_benableAISNameCache) {
144 if (wxFileName::FileExists(AISTargetNameFileName)) {
146 if (infile.Open(AISTargetNameFileName)) {
147 AIS_Target_Name_Hash *HashFile = AISTargetNamesNC;
148 wxString line = infile.GetFirstLine();
149 while (!infile.Eof()) {
150 if (line.IsSameAs(wxT(
"+++==Confirmed Entry's==+++")))
151 HashFile = AISTargetNamesC;
153 if (line.IsSameAs(wxT(
"+++==Non Confirmed Entry's==+++")))
154 HashFile = AISTargetNamesNC;
156 wxStringTokenizer tokenizer(line, _T(
","));
157 int mmsi = wxAtoi(tokenizer.GetNextToken());
158 wxString name = tokenizer.GetNextToken().Trim();
159 (*HashFile)[mmsi] = name;
162 line = infile.GetNextLine();
169 BuildERIShipTypeHash();
171 g_pais_alert_dialog_active =
nullptr;
172 m_bAIS_Audio_Alert_On =
false;
176 m_bAIS_AlertPlaying =
false;
178 TimerAIS.SetOwner(
this, TIMER_AIS1);
179 TimerAIS.Start(TIMER_AIS_MSEC, wxTIMER_CONTINUOUS);
181 m_ptentative_dsctarget = NULL;
182 m_dsc_timer.SetOwner(
this, TIMER_DSC);
192AisDecoder::~AisDecoder(
void) {
201 if (outfile.Open(AISTargetNameFileName)) {
203 content = wxT(
"+++==Confirmed Entry's==+++");
204 AIS_Target_Name_Hash::iterator it;
205 for (it = AISTargetNamesC->begin(); it != AISTargetNamesC->end(); ++it) {
206 content.append(_T(
"\r\n"));
207 content.append(wxString::Format(wxT(
"%i"), it->first));
208 content.append(_T(
",")).append(it->second);
210 content.append(_T(
"\r\n"));
211 content.append(_T(
"+++==Non Confirmed Entry's==+++"));
212 for (it = AISTargetNamesNC->begin(); it != AISTargetNamesNC->end(); ++it) {
213 content.append(_T(
"\r\n"));
214 content.append(wxString::Format(wxT(
"%i"), it->first));
215 content.append(_T(
",")).append(it->second);
217 outfile.Write(content);
221 AISTargetNamesC->clear();
222 delete AISTargetNamesC;
223 AISTargetNamesNC->clear();
224 delete AISTargetNamesNC;
229 m_AIS_Audio_Alert_Timer.Stop();
234 "First message[1, 2] ticks: %d Last Message [1,2]ticks %d Difference: "
236 first_rx_ticks, rx_ticks, rx_ticks - first_rx_ticks);
240bool IsTargetOnTheIgnoreList(
const int &mmsi) {
242 for (
unsigned int i = 0; i < g_MMSI_Props_Array.GetCount(); i++) {
243 if (mmsi == g_MMSI_Props_Array[i]->MMSI) {
245 if (props->m_bignore) {
253void AisDecoder::InitCommListeners(
void) {
256 auto &msgbus = NavMsgBus::GetInstance();
261 listener_N0183_VDM.
Listen(n0183_msg_VDM,
this, EVT_N0183_VDM);
264 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
265 HandleN0183_AIS(n0183_msg);
270 listener_N0183_FRPOS.
Listen(n0183_msg_FRPOS,
this, EVT_N0183_FRPOS);
274 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
275 HandleN0183_AIS(n0183_msg);
280 listener_N0183_CDDSC.
Listen(n0183_msg_CDDSC,
this, EVT_N0183_CDDSC);
283 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
284 HandleN0183_AIS(n0183_msg);
289 listener_N0183_CDDSE.
Listen(n0183_msg_CDDSE,
this, EVT_N0183_CDDSE);
292 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
293 HandleN0183_AIS(n0183_msg);
298 listener_N0183_TLL.
Listen(n0183_msg_TLL,
this, EVT_N0183_TLL);
302 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
303 HandleN0183_AIS(n0183_msg);
308 listener_N0183_TTM.
Listen(n0183_msg_ttm,
this, EVT_N0183_TTM);
311 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
312 HandleN0183_AIS(n0183_msg);
317 listener_N0183_OSD.
Listen(n0183_msg_OSD,
this, EVT_N0183_OSD);
320 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
321 HandleN0183_AIS(n0183_msg);
326 listener_N0183_WPL.
Listen(n0183_msg_WPL,
this, EVT_N0183_WPL);
329 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
330 HandleN0183_AIS(n0183_msg);
335 listener_SignalK.
Listen(sk_msg,
this, EVT_SIGNALK);
337 HandleSignalK(UnpackEvtPointer<SignalkMsg>(ev));
342 Nmea2000Msg n2k_msg_129038(
static_cast<uint64_t
>(129038));
343 listener_N2K_129038.
Listen(n2k_msg_129038,
this, EVT_N2K_129038);
345 HandleN2K_129038(UnpackEvtPointer<Nmea2000Msg>(ev));
350 Nmea2000Msg n2k_msg_129039(
static_cast<uint64_t
>(129039));
351 listener_N2K_129039.
Listen(n2k_msg_129039,
this, EVT_N2K_129039);
353 HandleN2K_129039(UnpackEvtPointer<Nmea2000Msg>(ev));
358 Nmea2000Msg n2k_msg_129041(
static_cast<uint64_t
>(129041));
359 listener_N2K_129041.
Listen(n2k_msg_129041,
this, EVT_N2K_129041);
361 HandleN2K_129041(UnpackEvtPointer<Nmea2000Msg>(ev));
366 Nmea2000Msg n2k_msg_129794(
static_cast<uint64_t
>(129794));
367 listener_N2K_129794.
Listen(n2k_msg_129794,
this, EVT_N2K_129794);
369 HandleN2K_129794(UnpackEvtPointer<Nmea2000Msg>(ev));
374 Nmea2000Msg n2k_msg_129809(
static_cast<uint64_t
>(129809));
375 listener_N2K_129809.
Listen(n2k_msg_129809,
this, EVT_N2K_129809);
377 HandleN2K_129809(UnpackEvtPointer<Nmea2000Msg>(ev));
382 Nmea2000Msg n2k_msg_129810(
static_cast<uint64_t
>(129810));
383 listener_N2K_129810.
Listen(n2k_msg_129810,
this, EVT_N2K_129810);
385 HandleN2K_129810(UnpackEvtPointer<Nmea2000Msg>(ev));
390 Nmea2000Msg n2k_msg_129793(
static_cast<uint64_t
>(129793));
391 listener_N2K_129793.
Listen(n2k_msg_129793,
this, EVT_N2K_129793);
393 HandleN2K_129793(UnpackEvtPointer<Nmea2000Msg>(ev));
397bool AisDecoder::HandleN0183_AIS(std::shared_ptr<const Nmea0183Msg> n0183_msg) {
398 std::string str = n0183_msg->payload;
399 wxString sentence(str.c_str());
400 DecodeN0183(sentence);
405bool AisDecoder::HandleN2K_129038(std::shared_ptr<const Nmea2000Msg> n2k_msg) {
406 std::vector<unsigned char> v = n2k_msg->payload;
409 tN2kAISRepeat Repeat;
420 tN2kAISNavStatus NavStat = N2kaisns_Under_Way_Motoring;
421 tN2kAISTransceiverInformation AISTransceiverInformation;
423 if (ParseN2kPGN129038(v, MessageID, Repeat, UserID, Latitude, Longitude,
424 Accuracy, RAIM, Seconds, COG, SOG, Heading, ROT,
425 NavStat, AISTransceiverInformation)) {
428 if (mmsi == g_OwnShipmmsi || IsTargetOnTheIgnoreList(mmsi))
return false;
432 long mmsi_long = mmsi;
433 std::shared_ptr<AisTargetData> pTargetData = 0;
434 bool bnewtarget =
false;
436 auto it = AISTargetList.find(mmsi);
437 if (it == AISTargetList.end())
439 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
443 pTargetData = it->second;
446 wxDateTime now = wxDateTime::Now();
450 pTargetData->MMSI = mmsi;
451 pTargetData->MID = MessageID;
452 pTargetData->MMSI = mmsi;
453 pTargetData->Class = AIS_CLASS_A;
455 if (97 == pTargetData->MMSI / 10000000) {
456 pTargetData->Class = AIS_SART;
458 pTargetData->StaticReportTicks = now.GetTicks();
460 pTargetData->NavStatus = (ais_nav_status)NavStat;
461 if (!N2kIsNA(SOG)) pTargetData->SOG = MS2KNOTS(SOG);
462 if (!N2kIsNA(COG)) pTargetData->COG = GeodesicRadToDeg(COG);
463 if (!N2kIsNA(Heading)) pTargetData->HDG = GeodesicRadToDeg(Heading);
464 if (!N2kIsNA(Longitude)) pTargetData->Lon = Longitude;
465 if (!N2kIsNA(Latitude)) pTargetData->Lat = Latitude;
467 pTargetData->ROTAIS = ROT;
469 double rot_dir = 1.0;
481 pTargetData->b_OwnShip =
482 AISTransceiverInformation ==
483 tN2kAISTransceiverInformation::N2kaisown_information_not_broadcast;
484 pTargetData->b_active =
true;
485 pTargetData->b_lost =
false;
486 pTargetData->b_positionOnceValid =
true;
487 pTargetData->LastPositionReportTicks = pTargetData->PositionReportTicks;
488 pTargetData->PositionReportTicks = now.GetTicks();
490 pSelectAIS->DeleteSelectablePoint((
void *)(
long)mmsi, SELTYPE_AISTARGET);
491 CommitAISTarget(pTargetData,
"",
true, bnewtarget);
500bool AisDecoder::HandleN2K_129039(std::shared_ptr<const Nmea2000Msg> n2k_msg) {
501 std::vector<unsigned char> v = n2k_msg->payload;
516 tN2kAISRepeat Repeat;
526 tN2kAISNavStatus NavStat = N2kaisns_Under_Way_Motoring;
527 tN2kAISTransceiverInformation AISTransceiverInformation;
529 bool DSC, Band, Msg22, State, Display;
532 if (ParseN2kPGN129039(v, MessageID, Repeat, UserID, Latitude, Longitude,
533 Accuracy, RAIM, Seconds, COG, SOG,
534 AISTransceiverInformation, Heading, Unit, Display, DSC,
535 Band, Msg22, Mode, State)) {
538 if (mmsi == g_OwnShipmmsi || IsTargetOnTheIgnoreList(mmsi))
return false;
542 long mmsi_long = mmsi;
543 std::shared_ptr<AisTargetData> pTargetData = 0;
544 bool bnewtarget =
false;
546 auto it = AISTargetList.find(mmsi);
547 if (it == AISTargetList.end())
549 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
553 pTargetData = it->second;
556 wxDateTime now = wxDateTime::Now();
560 pTargetData->MMSI = mmsi;
561 pTargetData->MID = MessageID;
562 if (!isBuoyMmsi(mmsi))
563 pTargetData->Class = AIS_CLASS_B;
565 pTargetData->Class = AIS_BUOY;
567 pTargetData->NavStatus = UNDEFINED;
568 if (!N2kIsNA(SOG)) pTargetData->SOG = MS2KNOTS(SOG);
569 if (!N2kIsNA(COG)) pTargetData->COG = GeodesicRadToDeg(COG);
570 if (!N2kIsNA(Heading)) pTargetData->HDG = GeodesicRadToDeg(Heading);
571 if (!N2kIsNA(Longitude)) pTargetData->Lon = Longitude;
572 if (!N2kIsNA(Latitude)) pTargetData->Lat = Latitude;
574 pTargetData->b_positionOnceValid =
true;
575 pTargetData->b_active =
true;
576 pTargetData->b_lost =
false;
577 pTargetData->LastPositionReportTicks = pTargetData->PositionReportTicks;
578 pTargetData->PositionReportTicks = now.GetTicks();
579 pTargetData->b_OwnShip =
580 AISTransceiverInformation ==
581 tN2kAISTransceiverInformation::N2kaisown_information_not_broadcast;
583 pSelectAIS->DeleteSelectablePoint((
void *)(
long)mmsi, SELTYPE_AISTARGET);
584 CommitAISTarget(pTargetData,
"",
true, bnewtarget);
594bool AisDecoder::HandleN2K_129041(std::shared_ptr<const Nmea2000Msg> n2k_msg) {
595 std::vector<unsigned char> v = n2k_msg->payload;
597 tN2kAISAtoNReportData data;
600 struct tN2kAISAtoNReportData {
602 tN2kAISRepeat Repeat;
611 double PositionReferenceStarboard ;
612 double PositionReferenceTrueNorth;
613 tN2kAISAtoNType AtoNType;
614 bool OffPositionIndicator;
615 bool VirtualAtoNFlag;
616 bool AssignedModeFlag;
617 tN2kGNSStype GNSSType;
619 tN2kAISTransceiverInformation AISTransceiverInformation;
620 char AtoNName[34 + 1];
623 if (ParseN2kPGN129041(v, data)) {
624 int mmsi = data.UserID;
626 if (mmsi == g_OwnShipmmsi || IsTargetOnTheIgnoreList(mmsi))
return false;
630 long mmsi_long = mmsi;
631 std::shared_ptr<AisTargetData> pTargetData = 0;
632 bool bnewtarget =
false;
634 auto it = AISTargetList.find(mmsi);
635 if (it == AISTargetList.end())
637 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
641 pTargetData = it->second;
645 pTargetData->MMSI = mmsi;
647 wxDateTime now = wxDateTime::Now();
650 int offpos = data.OffPositionIndicator;
651 int virt = data.VirtualAtoNFlag;
654 pTargetData->NavStatus = ATON_VIRTUAL;
656 pTargetData->NavStatus = ATON_REAL;
658 pTargetData->m_utc_sec = data.Seconds;
660 if (pTargetData->m_utc_sec <= 59) {
661 pTargetData->NavStatus += 1;
662 if (offpos) pTargetData->NavStatus += 1;
665 data.AtoNName[34] = 0;
666 strncpy(pTargetData->ShipName, data.AtoNName, SHIP_NAME_LEN - 1);
667 pTargetData->ShipName[
sizeof(pTargetData->ShipName) - 1] =
'\0';
668 pTargetData->b_nameValid =
true;
669 pTargetData->MID = 124;
671 pTargetData->ShipType = data.AtoNType;
672 pTargetData->Class = AIS_ATON;
674 if (!N2kIsNA(data.Longitude)) pTargetData->Lon = data.Longitude;
675 if (!N2kIsNA(data.Latitude)) pTargetData->Lat = data.Latitude;
676 pTargetData->b_positionDoubtful =
false;
677 pTargetData->b_positionOnceValid =
true;
678 pTargetData->LastPositionReportTicks = pTargetData->PositionReportTicks;
679 pTargetData->PositionReportTicks = now.GetTicks();
683 pSelectAIS->DeleteSelectablePoint((
void *)(
long)mmsi, SELTYPE_AISTARGET);
684 CommitAISTarget(pTargetData,
"",
true, bnewtarget);
686 touch_state.Notify();
693bool AisDecoder::HandleN2K_129794(std::shared_ptr<const Nmea2000Msg> n2k_msg) {
694 std::vector<unsigned char> v = n2k_msg->payload;
697 tN2kAISRepeat Repeat;
701 char Name[SHIP_NAME_LEN];
710 char Destination[DESTINATION_LEN];
711 tN2kAISVersion AISversion;
712 tN2kGNSStype GNSStype;
714 tN2kAISTranceiverInfo AISinfo;
716 if (ParseN2kPGN129794(v, MessageID, Repeat, UserID, IMOnumber, Callsign, Name,
717 VesselType, Length, Beam, PosRefStbd, PosRefBow,
718 ETAdate, ETAtime, Draught, Destination, AISversion,
719 GNSStype, DTE, AISinfo)) {
722 if (mmsi == g_OwnShipmmsi || IsTargetOnTheIgnoreList(mmsi))
return false;
726 long mmsi_long = mmsi;
727 std::shared_ptr<AisTargetData> pTargetData = 0;
728 bool bnewtarget =
false;
730 auto it = AISTargetList.find(mmsi);
731 if (it == AISTargetList.end())
733 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
737 pTargetData = it->second;
741 pTargetData->MMSI = mmsi;
742 strncpy(pTargetData->ShipName, Name, SHIP_NAME_LEN - 1);
743 pTargetData->ShipName[
sizeof(pTargetData->ShipName) - 1] =
'\0';
744 Name[
sizeof(Name) - 1] = 0;
745 pTargetData->b_nameValid =
true;
746 pTargetData->MID = 124;
748 pTargetData->b_OwnShip =
750 tN2kAISTranceiverInfo::N2kaisti_Own_information_not_broadcast;
752 pTargetData->DimA = PosRefBow;
753 pTargetData->DimB = Length - PosRefBow;
754 pTargetData->DimC = Beam - PosRefStbd;
755 pTargetData->DimD = PosRefStbd;
756 pTargetData->Draft = Draught;
757 pTargetData->IMO = IMOnumber;
758 strncpy(pTargetData->CallSign, Callsign, CALL_SIGN_LEN - 1);
759 pTargetData->CallSign[
sizeof(pTargetData->CallSign) - 1] =
'\0';
760 pTargetData->ShipType = (
unsigned char)VesselType;
761 strncpy(pTargetData->Destination, Destination, DESTINATION_LEN - 1);
762 pTargetData->Destination[
sizeof(pTargetData->Destination) - 1] =
'\0';
763 Destination[
sizeof(Destination) - 1] = 0;
765 if (!N2kIsNA(ETAdate) && !N2kIsNA(ETAtime)) {
766 long secs = (ETAdate * 24 * 3600) + wxRound(ETAtime);
767 wxDateTime t((time_t)secs);
769 wxDateTime tz = t.ToUTC();
770 pTargetData->ETA_Mo = tz.GetMonth() + 1;
771 pTargetData->ETA_Day = tz.GetDay();
772 pTargetData->ETA_Hr = tz.GetHour();
773 pTargetData->ETA_Min = tz.GetMinute();
777 pSelectAIS->DeleteSelectablePoint((
void *)(
long)mmsi, SELTYPE_AISTARGET);
778 CommitAISTarget(pTargetData,
"",
true, bnewtarget);
786bool AisDecoder::HandleN2K_129809(std::shared_ptr<const Nmea2000Msg> n2k_msg) {
787 std::vector<unsigned char> v = n2k_msg->payload;
790 tN2kAISRepeat Repeat;
794 if (ParseN2kPGN129809(v, MessageID, Repeat, UserID, Name)) {
797 if (mmsi == g_OwnShipmmsi || IsTargetOnTheIgnoreList(mmsi))
return false;
801 long mmsi_long = mmsi;
802 std::shared_ptr<AisTargetData> pTargetData = 0;
803 bool bnewtarget =
false;
805 auto it = AISTargetList.find(mmsi);
806 if (it == AISTargetList.end())
808 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
812 pTargetData = it->second;
816 pTargetData->MMSI = mmsi;
817 Name[
sizeof(Name) - 1] = 0;
818 strncpy(pTargetData->ShipName, Name, SHIP_NAME_LEN - 1);
819 pTargetData->b_nameValid =
true;
820 pTargetData->MID = 124;
822 pSelectAIS->DeleteSelectablePoint((
void *)(
long)mmsi, SELTYPE_AISTARGET);
823 CommitAISTarget(pTargetData,
"",
true, bnewtarget);
833bool AisDecoder::HandleN2K_129810(std::shared_ptr<const Nmea2000Msg> n2k_msg) {
834 std::vector<unsigned char> v = n2k_msg->payload;
837 tN2kAISRepeat Repeat;
846 uint32_t MothershipID;
848 if (ParseN2kPGN129810(v, MessageID, Repeat, UserID, VesselType, Vendor,
849 Callsign, Length, Beam, PosRefStbd, PosRefBow,
853 if (mmsi == g_OwnShipmmsi || IsTargetOnTheIgnoreList(mmsi))
return false;
857 long mmsi_long = mmsi;
858 std::shared_ptr<AisTargetData> pTargetData = 0;
859 bool bnewtarget =
false;
861 auto it = AISTargetList.find(mmsi);
862 if (it == AISTargetList.end())
864 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
868 pTargetData = it->second;
872 pTargetData->MMSI = mmsi;
873 pTargetData->DimA = PosRefBow;
874 pTargetData->DimB = Length - PosRefBow;
875 pTargetData->DimC = Beam - PosRefStbd;
876 pTargetData->DimD = PosRefStbd;
877 strncpy(pTargetData->CallSign, Callsign, CALL_SIGN_LEN - 1);
878 pTargetData->CallSign[
sizeof(pTargetData->CallSign) - 1] =
'\0';
879 pTargetData->ShipType = (
unsigned char)VesselType;
881 pSelectAIS->DeleteSelectablePoint((
void *)(long)mmsi, SELTYPE_AISTARGET);
882 CommitAISTarget(pTargetData,
"",
true, bnewtarget);
891bool AisDecoder::HandleN2K_129793(std::shared_ptr<const Nmea2000Msg> n2k_msg) {
892 std::vector<unsigned char> v = n2k_msg->payload;
895 tN2kAISRepeat Repeat;
899 unsigned int SecondsSinceMidnight;
900 unsigned int DaysSinceEpoch;
902 if (ParseN2kPGN129793(v, MessageID, Repeat, UserID, Longitude, Latitude,
903 SecondsSinceMidnight, DaysSinceEpoch)) {
904 wxDateTime now = wxDateTime::Now();
910 long mmsi_long = mmsi;
911 std::shared_ptr<AisTargetData> pTargetData = 0;
912 bool bnewtarget =
false;
914 auto it = AISTargetList.find(mmsi);
915 if (it == AISTargetList.end())
917 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
921 pTargetData = it->second;
925 pTargetData->MMSI = mmsi;
926 pTargetData->Class = AIS_BASE;
928 if (!N2kIsNA(Longitude)) pTargetData->Lon = Longitude;
929 if (!N2kIsNA(Latitude)) pTargetData->Lat = Latitude;
930 pTargetData->b_positionDoubtful =
false;
931 pTargetData->b_positionOnceValid =
true;
932 pTargetData->LastPositionReportTicks = pTargetData->PositionReportTicks;
933 pTargetData->PositionReportTicks = now.GetTicks();
937 pSelectAIS->DeleteSelectablePoint((
void *)(
long)mmsi, SELTYPE_AISTARGET);
938 CommitAISTarget(pTargetData,
"",
true, bnewtarget);
954void AisDecoder::BuildERIShipTypeHash(
void) {
955 make_hash_ERI(8000, _(
"Vessel, type unknown"));
956 make_hash_ERI(8150, _(
"Freightbarge"));
957 make_hash_ERI(8160, _(
"Tankbarge"));
958 make_hash_ERI(8163, _(
"Tankbarge, dry cargo as if liquid (e.g. cement)"));
959 make_hash_ERI(8450, _(
"Service vessel, police patrol, port service"));
960 make_hash_ERI(8430, _(
"Pushboat, single"));
961 make_hash_ERI(8510, _(
"Object, not otherwise specified"));
962 make_hash_ERI(8470, _(
"Object, towed, not otherwise specified"));
963 make_hash_ERI(8490, _(
"Bunkership"));
964 make_hash_ERI(8010, _(
"Motor freighter"));
965 make_hash_ERI(8020, _(
"Motor tanker"));
966 make_hash_ERI(8021, _(
"Motor tanker, liquid cargo, type N"));
967 make_hash_ERI(8022, _(
"Motor tanker, liquid cargo, type C"));
968 make_hash_ERI(8023, _(
"Motor tanker, dry cargo as if liquid (e.g. cement)"));
969 make_hash_ERI(8030, _(
"Container vessel"));
970 make_hash_ERI(8040, _(
"Gas tanker"));
971 make_hash_ERI(8050, _(
"Motor freighter, tug"));
972 make_hash_ERI(8060, _(
"Motor tanker, tug"));
973 make_hash_ERI(8070, _(
"Motor freighter with one or more ships alongside"));
974 make_hash_ERI(8080, _(
"Motor freighter with tanker"));
975 make_hash_ERI(8090, _(
"Motor freighter pushing one or more freighters"));
976 make_hash_ERI(8100, _(
"Motor freighter pushing at least one tank-ship"));
977 make_hash_ERI(8110, _(
"Tug, freighter"));
978 make_hash_ERI(8120, _(
"Tug, tanker"));
979 make_hash_ERI(8130, _(
"Tug freighter, coupled"));
980 make_hash_ERI(8140, _(
"Tug, freighter/tanker, coupled"));
981 make_hash_ERI(8161, _(
"Tankbarge, liquid cargo, type N"));
982 make_hash_ERI(8162, _(
"Tankbarge, liquid cargo, type C"));
983 make_hash_ERI(8170, _(
"Freightbarge with containers"));
984 make_hash_ERI(8180, _(
"Tankbarge, gas"));
985 make_hash_ERI(8210, _(
"Pushtow, one cargo barge"));
986 make_hash_ERI(8220, _(
"Pushtow, two cargo barges"));
987 make_hash_ERI(8230, _(
"Pushtow, three cargo barges"));
988 make_hash_ERI(8240, _(
"Pushtow, four cargo barges"));
989 make_hash_ERI(8250, _(
"Pushtow, five cargo barges"));
990 make_hash_ERI(8260, _(
"Pushtow, six cargo barges"));
991 make_hash_ERI(8270, _(
"Pushtow, seven cargo barges"));
992 make_hash_ERI(8280, _(
"Pushtow, eight cargo barges"));
993 make_hash_ERI(8290, _(
"Pushtow, nine or more barges"));
994 make_hash_ERI(8310, _(
"Pushtow, one tank/gas barge"));
996 _(
"Pushtow, two barges at least one tanker or gas barge"));
998 _(
"Pushtow, three barges at least one tanker or gas barge"));
1000 _(
"Pushtow, four barges at least one tanker or gas barge"));
1002 _(
"Pushtow, five barges at least one tanker or gas barge"));
1004 _(
"Pushtow, six barges at least one tanker or gas barge"));
1006 _(
"Pushtow, seven barges at least one tanker or gas barge"));
1008 _(
"Pushtow, eight barges at least one tanker or gas barge"));
1010 8390, _(
"Pushtow, nine or more barges at least one tanker or gas barge"));
1011 make_hash_ERI(8400, _(
"Tug, single"));
1012 make_hash_ERI(8410, _(
"Tug, one or more tows"));
1013 make_hash_ERI(8420, _(
"Tug, assisting a vessel or linked combination"));
1014 make_hash_ERI(8430, _(
"Pushboat, single"));
1015 make_hash_ERI(8440, _(
"Passenger ship, ferry, cruise ship, red cross ship"));
1016 make_hash_ERI(8441, _(
"Ferry"));
1017 make_hash_ERI(8442, _(
"Red cross ship"));
1018 make_hash_ERI(8443, _(
"Cruise ship"));
1019 make_hash_ERI(8444, _(
"Passenger ship without accommodation"));
1020 make_hash_ERI(8460, _(
"Vessel, work maintenance craft, floating derrick, "
1021 "cable-ship, buoy-ship, dredge"));
1022 make_hash_ERI(8480, _(
"Fishing boat"));
1023 make_hash_ERI(8500, _(
"Barge, tanker, chemical"));
1024 make_hash_ERI(1500, _(
"General cargo Vessel maritime"));
1025 make_hash_ERI(1510, _(
"Unit carrier maritime"));
1026 make_hash_ERI(1520, _(
"Bulk carrier maritime"));
1027 make_hash_ERI(1530, _(
"Tanker"));
1028 make_hash_ERI(1540, _(
"Liquified gas tanker"));
1029 make_hash_ERI(1850, _(
"Pleasure craft, longer than 20 metres"));
1030 make_hash_ERI(1900, _(
"Fast ship"));
1031 make_hash_ERI(1910, _(
"Hydrofoil"));
1037void AisDecoder::HandleSignalK(std::shared_ptr<const SignalkMsg> sK_msg) {
1038 rapidjson::Document root;
1040 root.Parse(sK_msg->raw_message);
1042 if (root.HasParseError())
return;
1044 if (root.HasMember(
"self")) {
1050 if (m_signalk_selfid.IsEmpty()) {
1055 int meteo_SiteID = 0;
1056 if (root.HasMember(
"context") && root[
"context"].IsString()) {
1057 wxString context = root[
"context"].GetString();
1058 if (context == m_signalk_selfid) {
1060 wxLogMessage(_T(
"** Ignore context own ship.."));
1064 wxString mmsi_string;
1065 if (context.StartsWith(_T(
"vessels.urn:mrn:imo:mmsi:"), &mmsi_string) ||
1066 context.StartsWith(_T(
"atons.urn:mrn:imo:mmsi:"), &mmsi_string) ||
1067 context.StartsWith(_T(
"aircraft.urn:mrn:imo:mmsi:"), &mmsi_string)) {
1070 if (mmsi_string.ToLong(&mmsi)) {
1075 }
else if (context.StartsWith(_T(
"meteo.urn:mrn:imo:mmsi:"),
1078 origin_mmsi = wxAtoi(wxString(mmsi_string).BeforeFirst(
':'));
1079 meteo_SiteID = wxAtoi(
'1' + wxString(mmsi_string).AfterFirst(
':'));
1082 int meteo_mmsi = AisMeteoNewMmsi(origin_mmsi, 0, 0, 999, meteo_SiteID);
1093 if (g_pMUX && g_pMUX->IsLogActive()) {
1095 logmsg.Printf(
"AIS :MMSI: %ld", mmsi);
1096 std::string source = sK_msg->source->to_string();
1101 if (IsTargetOnTheIgnoreList(mmsi))
return;
1104 if (mmsi == g_OwnShipmmsi)
return;
1109 writer.
Write(root, dbg);
1111 wxString msg( _T(
"AisDecoder::OnEvtSignalK: ") );
1115 std::shared_ptr<AisTargetData> pTargetData = 0;
1116 std::shared_ptr<AisTargetData> pStaleTarget = NULL;
1117 bool bnewtarget =
false;
1118 int last_report_ticks;
1120 getAISTarget(mmsi, pTargetData, pStaleTarget, bnewtarget, last_report_ticks,
1123 pTargetData->MMSI = mmsi;
1124 getMmsiProperties(pTargetData);
1125 if (root.HasMember(
"updates") && root[
"updates"].IsArray()) {
1126 for (rapidjson::Value::ConstValueIterator itr = root[
"updates"].Begin();
1127 itr != root[
"updates"].End(); ++itr) {
1128 handleUpdate(pTargetData, bnewtarget, *itr);
1133 if (97 == mmsi / 10000000) {
1134 pTargetData->Class = AIS_SART;
1135 }
else if (1994 == mmsi / 100000) {
1137 pTargetData->Class = AIS_METEO;
1138 pTargetData->met_data.original_mmsi = origin_mmsi;
1139 pTargetData->met_data.stationID = meteo_SiteID;
1142 wxString met_name = pTargetData->ShipName;
1143 if (met_name.Find(
"METEO") == wxNOT_FOUND) {
1146 s_id << meteo_SiteID;
1147 id1 = wxAtoi(s_id.Mid(1, 3));
1148 id2 = wxAtoi(s_id.Mid(4, 3));
1149 met_name =
"METEO ";
1150 met_name << wxString::Format(
"%03d", (id1 + id2)).Right(3);
1151 strncpy(pTargetData->ShipName, met_name, SHIP_NAME_LEN - 1);
1153 pTargetData->b_nameValid =
true;
1154 pTargetData->MID = 123;
1155 pTargetData->COG = -1.;
1156 pTargetData->HDG = 511;
1157 pTargetData->SOG = -1.;
1158 pTargetData->b_NoTrack =
true;
1159 pTargetData->b_show_track =
false;
1161 pTargetData->b_OwnShip =
false;
1162 AISTargetList[pTargetData->MMSI] = pTargetData;
1166void AisDecoder::handleUpdate(std::shared_ptr<AisTargetData> pTargetData,
1167 bool bnewtarget,
const rapidjson::Value &update) {
1168 wxString sfixtime =
"";
1170 if (update.HasMember(
"timestamp")) {
1171 sfixtime = update[
"timestamp"].GetString();
1173 if (update.HasMember(
"values") && update[
"values"].IsArray()) {
1174 for (rapidjson::Value::ConstValueIterator itr = update[
"values"].Begin();
1175 itr != update[
"values"].End(); ++itr) {
1176 updateItem(pTargetData, bnewtarget, *itr, sfixtime);
1179 wxDateTime now = wxDateTime::Now();
1180 pTargetData->m_utc_hour = now.ToUTC().GetHour();
1181 pTargetData->m_utc_min = now.ToUTC().GetMinute();
1182 pTargetData->m_utc_sec = now.ToUTC().GetSecond();
1184 pTargetData->b_active =
true;
1185 pTargetData->b_lost =
false;
1187 if (pTargetData->b_positionOnceValid) {
1188 long mmsi_long = pTargetData->MMSI;
1190 pSelectAIS->AddSelectablePoint(pTargetData->Lat, pTargetData->Lon,
1191 (
void *)mmsi_long, SELTYPE_AISTARGET);
1192 pSel->SetUserData(pTargetData->MMSI);
1194 UpdateOneCPA(pTargetData.get());
1195 if (pTargetData->b_show_track) UpdateOneTrack(pTargetData.get());
1198void AisDecoder::updateItem(std::shared_ptr<AisTargetData> pTargetData,
1199 bool bnewtarget,
const rapidjson::Value &item,
1200 wxString &sfixtime)
const {
1201 if (item.HasMember(
"path") && item.HasMember(
"value")) {
1202 const wxString &update_path = item[
"path"].GetString();
1203 if (update_path == _T(
"navigation.position")) {
1204 if (item[
"value"].HasMember(
"latitude") &&
1205 item[
"value"].HasMember(
"longitude")) {
1206 wxDateTime now = wxDateTime::Now();
1208 double lat = item[
"value"][
"latitude"].GetDouble();
1209 double lon = item[
"value"][
"longitude"].GetDouble();
1210 pTargetData->LastPositionReportTicks = pTargetData->PositionReportTicks;
1211 pTargetData->PositionReportTicks = now.GetTicks();
1212 pTargetData->StaticReportTicks = now.GetTicks();
1213 pTargetData->Lat = lat;
1214 pTargetData->Lon = lon;
1215 pTargetData->b_positionOnceValid =
true;
1216 pTargetData->b_positionDoubtful =
false;
1223 }
else if (update_path == _T(
"navigation.speedOverGround") &&
1224 item[
"value"].IsNumber()) {
1225 pTargetData->SOG = item[
"value"].GetDouble() * ms_to_knot_factor;
1226 }
else if (update_path == _T(
"navigation.courseOverGroundTrue") &&
1227 item[
"value"].IsNumber()) {
1228 pTargetData->COG = GEODESIC_RAD2DEG(item[
"value"].GetDouble());
1229 }
else if (update_path == _T(
"navigation.headingTrue") &&
1230 item[
"value"].IsNumber()) {
1231 pTargetData->HDG = GEODESIC_RAD2DEG(item[
"value"].GetDouble());
1232 }
else if (update_path == _T(
"navigation.rateOfTurn") &&
1233 item[
"value"].IsNumber()) {
1234 pTargetData->ROTAIS = 4.733 * sqrt(item[
"value"].GetDouble());
1235 }
else if (update_path == _T(
"design.aisShipType")) {
1236 if (item[
"value"].HasMember(
"id")) {
1237 if (!pTargetData->b_isDSCtarget) {
1238 pTargetData->ShipType = item[
"value"][
"id"].GetUint();
1241 }
else if (update_path == _T(
"atonType")) {
1242 if (item[
"value"].HasMember(
"id")) {
1243 pTargetData->ShipType = item[
"value"][
"id"].GetUint();
1245 }
else if (update_path == _T(
"virtual")) {
1246 if (item[
"value"].GetBool()) {
1247 pTargetData->NavStatus = ATON_VIRTUAL;
1249 pTargetData->NavStatus = ATON_REAL;
1251 }
else if (update_path == _T(
"offPosition")) {
1252 if (item[
"value"].GetBool()) {
1253 if (ATON_REAL == pTargetData->NavStatus) {
1254 pTargetData->NavStatus = ATON_REAL_OFFPOSITION;
1255 }
else if (ATON_VIRTUAL == pTargetData->NavStatus) {
1256 pTargetData->NavStatus = ATON_VIRTUAL_OFFPOSITION;
1259 }
else if (update_path == _T(
"design.draft")) {
1260 if (item[
"value"].HasMember(
"maximum") &&
1261 item[
"value"][
"maximum"].IsNumber()) {
1262 pTargetData->Draft = item[
"value"][
"maximum"].GetDouble();
1263 pTargetData->Euro_Draft = item[
"value"][
"maximum"].GetDouble();
1265 if (item[
"value"].HasMember(
"current") &&
1266 item[
"value"][
"current"].IsNumber()) {
1267 double draft = item[
"value"][
"current"].GetDouble();
1269 pTargetData->Draft = draft;
1270 pTargetData->Euro_Draft = draft;
1273 }
else if (update_path == _T(
"design.length")) {
1274 if (pTargetData->DimB == 0) {
1275 if (item[
"value"].HasMember(
"overall")) {
1276 if (item[
"value"][
"overall"].IsNumber()) {
1277 pTargetData->Euro_Length = item[
"value"][
"overall"].GetDouble();
1278 pTargetData->DimA = item[
"value"][
"overall"].GetDouble();
1280 pTargetData->DimB = 0;
1283 }
else if (update_path == _T(
"sensors.ais.class")) {
1284 wxString aisclass = item[
"value"].GetString();
1285 if (aisclass == _T(
"A")) {
1286 if (!pTargetData->b_isDSCtarget) pTargetData->Class = AIS_CLASS_A;
1287 }
else if (aisclass == _T(
"B")) {
1288 if (!pTargetData->b_isDSCtarget) {
1289 if (!isBuoyMmsi(pTargetData->MMSI))
1290 pTargetData->Class = AIS_CLASS_B;
1292 pTargetData->Class = AIS_BUOY;
1295 pTargetData->NavStatus = UNDEFINED;
1297 }
else if (aisclass == _T(
"BASE")) {
1298 pTargetData->Class = AIS_BASE;
1299 }
else if (aisclass == _T(
"ATON")) {
1300 pTargetData->Class = AIS_ATON;
1302 }
else if (update_path == _T(
"sensors.ais.fromBow")) {
1303 if (pTargetData->DimB == 0 && pTargetData->DimA != 0) {
1304 int length = pTargetData->DimA;
1305 if (item[
"value"].IsNumber()) {
1306 pTargetData->DimA = item[
"value"].GetDouble();
1307 pTargetData->DimB = length - item[
"value"].GetDouble();
1310 }
else if (update_path == _T(
"design.beam")) {
1311 if (pTargetData->DimD == 0) {
1312 if (item[
"value"].IsNumber()) {
1313 pTargetData->Euro_Beam = item[
"value"].GetDouble();
1314 pTargetData->DimC = item[
"value"].GetDouble();
1316 pTargetData->DimD = 0;
1318 }
else if (update_path == _T(
"sensors.ais.fromCenter")) {
1319 if (pTargetData->DimD == 0 && pTargetData->DimC != 0) {
1320 int beam = pTargetData->DimC;
1321 int center = beam / 2;
1322 if (item[
"value"].IsNumber()) {
1325 pTargetData->DimC = center + item[
"value"].GetDouble();
1326 pTargetData->DimD = beam - pTargetData->DimC;
1329 }
else if (update_path == _T(
"navigation.state")) {
1330 wxString state = item[
"value"].GetString();
1331 if (state == _T(
"motoring")) {
1332 pTargetData->NavStatus = UNDERWAY_USING_ENGINE;
1333 }
else if (state == _T(
"anchored")) {
1334 pTargetData->NavStatus = AT_ANCHOR;
1335 }
else if (state == _T(
"not under command")) {
1336 pTargetData->NavStatus = NOT_UNDER_COMMAND;
1337 }
else if (state == _T(
"restricted manouverability")) {
1338 pTargetData->NavStatus = RESTRICTED_MANOEUVRABILITY;
1339 }
else if (state == _T(
"constrained by draft")) {
1340 pTargetData->NavStatus = CONSTRAINED_BY_DRAFT;
1341 }
else if (state == _T(
"moored")) {
1342 pTargetData->NavStatus = MOORED;
1343 }
else if (state == _T(
"aground")) {
1344 pTargetData->NavStatus = AGROUND;
1345 }
else if (state == _T(
"fishing")) {
1346 pTargetData->NavStatus = FISHING;
1347 }
else if (state == _T(
"sailing")) {
1348 pTargetData->NavStatus = UNDERWAY_SAILING;
1349 }
else if (state == _T(
"hazardous material high speed")) {
1350 pTargetData->NavStatus = HSC;
1351 }
else if (state == _T(
"hazardous material wing in ground")) {
1352 pTargetData->NavStatus = WIG;
1353 }
else if (state == _T(
"ais-sart")) {
1354 pTargetData->NavStatus = RESERVED_14;
1356 pTargetData->NavStatus = UNDEFINED;
1358 }
else if (update_path == _T(
"navigation.destination.commonName")) {
1359 const wxString &destination = item[
"value"].GetString();
1360 pTargetData->Destination[0] =
'\0';
1361 strncpy(pTargetData->Destination, destination.c_str(),
1362 DESTINATION_LEN - 1);
1363 }
else if (update_path == _T(
"navigation.specialManeuver")) {
1364 if (strcmp(
"not available", item[
"value"].GetString()) != 0 &&
1365 pTargetData->IMO < 1) {
1366 const wxString &bluesign = item[
"value"].GetString();
1367 if (_T(
"not engaged") == bluesign) {
1368 pTargetData->blue_paddle = 1;
1370 if (_T(
"engaged") == bluesign) {
1371 pTargetData->blue_paddle = 2;
1373 pTargetData->b_blue_paddle =
1374 pTargetData->blue_paddle == 2 ? true :
false;
1376 }
else if (update_path == _T(
"sensors.ais.designatedAreaCode")) {
1377 if (item[
"value"].GetInt() == 200) {
1378 pTargetData->b_hasInlandDac =
true;
1380 }
else if (update_path == _T(
"sensors.ais.functionalId")) {
1381 if (item[
"value"].GetInt() == 10 && pTargetData->b_hasInlandDac) {
1383 pTargetData->b_isEuroInland =
true;
1387 }
else if (update_path ==
"environment.date") {
1388 wxString issued = item[
"value"].GetString();
1392 ParseGPXDateTime(tz, issued);
1393 pTargetData->met_data.day = tz.GetDay();
1394 pTargetData->met_data.hour = tz.GetHour();
1395 pTargetData->met_data.minute = tz.GetMinute();
1397 }
else if (update_path ==
"environment.wind.averageSpeed" &&
1398 item[
"value"].IsNumber()) {
1399 pTargetData->met_data.wind_kn = MS2KNOTS(item[
"value"].GetDouble());
1400 }
else if (update_path ==
"environment.wind.gust" &&
1401 item[
"value"].IsNumber()) {
1402 pTargetData->met_data.wind_gust_kn = MS2KNOTS(item[
"value"].GetDouble());
1403 }
else if (update_path ==
"environment.wind.directionTrue" &&
1404 item[
"value"].IsNumber()) {
1405 pTargetData->met_data.wind_dir =
1406 GEODESIC_RAD2DEG(item[
"value"].GetDouble());
1407 }
else if (update_path ==
"environment.wind.gustDirectionTrue" &&
1408 item[
"value"].IsNumber()) {
1409 pTargetData->met_data.wind_gust_dir =
1410 GEODESIC_RAD2DEG(item[
"value"].GetDouble());
1411 }
else if (update_path ==
"environment.outside.temperature" &&
1412 item[
"value"].IsNumber()) {
1413 pTargetData->met_data.air_temp = KelvinToC(item[
"value"].GetDouble());
1414 }
else if (update_path ==
"environment.outside.relativeHumidity" &&
1415 item[
"value"].IsNumber()) {
1416 pTargetData->met_data.rel_humid = item[
"value"].GetDouble();
1417 }
else if (update_path ==
"environment.outside.dewPointTemperature" &&
1418 item[
"value"].IsNumber()) {
1419 pTargetData->met_data.dew_point = KelvinToC(item[
"value"].GetDouble());
1420 }
else if (update_path ==
"environment.outside.pressure" &&
1421 item[
"value"].IsNumber()) {
1422 pTargetData->met_data.airpress =
1423 static_cast<int>(item[
"value"].GetDouble() / 100);
1424 }
else if (update_path ==
"environment.water.level" &&
1425 item[
"value"].IsNumber()) {
1426 pTargetData->met_data.water_lev_dev = item[
"value"].GetDouble();
1427 }
else if (update_path ==
"environment.water.current.drift" &&
1428 item[
"value"].IsNumber()) {
1429 pTargetData->met_data.current = MS2KNOTS(item[
"value"].GetDouble());
1430 }
else if (update_path ==
"environment.water.current.set" &&
1431 item[
"value"].IsNumber()) {
1432 pTargetData->met_data.curr_dir =
1433 GEODESIC_RAD2DEG(item[
"value"].GetDouble());
1434 }
else if (update_path ==
"environment.water.levelTendencyValue" &&
1435 item[
"value"].IsNumber()) {
1436 pTargetData->met_data.water_lev_trend =
1437 static_cast<int>(item[
"value"].GetDouble());
1438 }
else if (update_path ==
"environment.water.levelTendency") {
1440 }
else if (update_path ==
"environment.water.waves.significantHeight" &&
1441 item[
"value"].IsNumber()) {
1442 pTargetData->met_data.wave_height = item[
"value"].GetDouble();
1443 }
else if (update_path ==
"environment.water.waves.period" &&
1444 item[
"value"].IsNumber()) {
1445 pTargetData->met_data.wave_period =
1446 static_cast<int>(item[
"value"].GetDouble());
1447 }
else if (update_path ==
"environment.water.waves.directionTrue" &&
1448 item[
"value"].IsNumber()) {
1449 pTargetData->met_data.wave_dir =
1450 GEODESIC_RAD2DEG(item[
"value"].GetDouble());
1451 }
else if (update_path ==
"environment.water.swell.height" &&
1452 item[
"value"].IsNumber()) {
1453 pTargetData->met_data.swell_height = item[
"value"].GetDouble();
1454 }
else if (update_path ==
"environment.water.swell.period" &&
1455 item[
"value"].IsNumber()) {
1456 pTargetData->met_data.swell_per =
1457 static_cast<int>(item[
"value"].GetDouble());
1458 }
else if (update_path ==
"environment.water.swell.directionTrue" &&
1459 item[
"value"].IsNumber()) {
1460 pTargetData->met_data.swell_dir =
1461 GEODESIC_RAD2DEG(item[
"value"].GetDouble());
1462 }
else if (update_path ==
"environment.water.temperature" &&
1463 item[
"value"].IsNumber()) {
1464 pTargetData->met_data.water_temp = KelvinToC(item[
"value"].GetDouble());
1465 }
else if (update_path ==
"environment.water.salinity" &&
1466 item[
"value"].IsNumber()) {
1467 pTargetData->met_data.salinity = item[
"value"].GetDouble();
1468 }
else if (update_path ==
"environment.water.ice") {
1470 }
else if (update_path ==
"environment.water.iceValue" &&
1471 item[
"value"].IsNumber()) {
1472 pTargetData->met_data.ice =
static_cast<int>(item[
"value"].GetDouble());
1473 }
else if (update_path ==
"environment.water.seaStateValue" &&
1474 item[
"value"].IsNumber()) {
1475 pTargetData->met_data.seastate =
1476 static_cast<int>(item[
"value"].GetDouble());
1477 }
else if (update_path ==
"environment.water.seaState") {
1479 }
else if (update_path ==
"environment.outside.precipitation") {
1481 }
else if (update_path ==
"environment.outside.precipitationValue" &&
1482 item[
"value"].IsNumber()) {
1483 pTargetData->met_data.precipitation =
1484 static_cast<int>(item[
"value"].GetDouble());
1485 }
else if (update_path ==
"environment.outside.pressureTendencyValue" &&
1486 item[
"value"].IsNumber()) {
1487 pTargetData->met_data.airpress_tend =
1488 static_cast<int>(item[
"value"].GetDouble());
1489 }
else if (update_path ==
"environment.outside.pressureTendency") {
1491 }
else if (update_path ==
"environment.outside.horizontalVisibility" &&
1492 item[
"value"].IsNumber()) {
1493 pTargetData->met_data.hor_vis =
1494 GEODESIC_METERS2NM(item[
"value"].GetDouble());
1495 }
else if (update_path ==
1496 "environment.outside.horizontalVisibility.overRange") {
1497 pTargetData->met_data.hor_vis_GT = item[
"value"].GetBool();
1498 }
else if (update_path == _T(
"")) {
1499 if (item[
"value"].HasMember(
"name")) {
1500 const wxString &name = item[
"value"][
"name"].GetString();
1501 strncpy(pTargetData->ShipName, name.c_str(), SHIP_NAME_LEN - 1);
1502 pTargetData->b_nameValid =
true;
1503 pTargetData->MID = 123;
1504 }
else if (item[
"value"].HasMember(
"registrations")) {
1505 const wxString &imo = item[
"value"][
"registrations"][
"imo"].GetString();
1506 pTargetData->IMO = wxAtoi(imo.Right(7));
1507 }
else if (item[
"value"].HasMember(
"communication")) {
1508 const wxString &callsign =
1509 item[
"value"][
"communication"][
"callsignVhf"].GetString();
1510 strncpy(pTargetData->CallSign, callsign.c_str(), 7);
1512 if (item[
"value"].HasMember(
"mmsi") &&
1513 1994 != (pTargetData->MMSI) / 100000) {
1515 wxString tmp = item[
"value"][
"mmsi"].GetString();
1516 if (tmp.ToLong(&mmsi)) {
1517 pTargetData->MMSI = mmsi;
1519 if (97 == mmsi / 10000000) {
1520 pTargetData->Class = AIS_SART;
1522 if (111 == mmsi / 1000000) {
1523 pTargetData->b_SarAircraftPosnReport =
true;
1526 AISshipNameCache(pTargetData.get(), AISTargetNamesC, AISTargetNamesNC,
1531 wxLogMessage(wxString::Format(
1532 _T(
"** AisDecoder::updateItem: unhandled path %s"), update_path));
1534 rapidjson::StringBuffer buffer;
1535 rapidjson::Writer<rapidjson::StringBuffer> writer(buffer);
1536 item.Accept(writer);
1537 wxString msg(_T(
"update: "));
1538 msg.append(buffer.GetString());
1548AisError AisDecoder::DecodeSingleVDO(
const wxString &str,
GenericPosDatEx *pos,
1549 wxString *accumulator) {
1551 if (str.Len() > 128)
return AIS_NMEAVDX_TOO_LONG;
1553 if (!NMEACheckSumOK(str))
return AIS_NMEAVDX_CHECKSUM_BAD;
1555 if (!pos)
return AIS_GENERIC_ERROR;
1557 if (!accumulator)
return AIS_GENERIC_ERROR;
1560 if (!str.Mid(1, 5).IsSameAs(_T(
"AIVDO")))
return AIS_GENERIC_ERROR;
1563 wxStringTokenizer tkz(str, _T(
","));
1566 token = tkz.GetNextToken();
1568 token = tkz.GetNextToken();
1569 int nsentences = atoi(token.mb_str());
1571 token = tkz.GetNextToken();
1572 int isentence = atoi(token.mb_str());
1574 token = tkz.GetNextToken();
1575 token = tkz.GetNextToken();
1577 wxString string_to_parse;
1578 string_to_parse.Clear();
1591 if ((1 == nsentences) && (1 == isentence)) {
1592 string_to_parse = tkz.GetNextToken();
1595 else if (nsentences > 1) {
1596 if (1 == isentence) {
1597 *accumulator = tkz.GetNextToken();
1601 accumulator->Append(tkz.GetNextToken());
1604 if (isentence == nsentences) {
1605 string_to_parse = *accumulator;
1609 if (string_to_parse.IsEmpty() &&
1611 return AIS_INCOMPLETE_MULTIPART;
1620 auto TargetData = AisTargetDataMaker::GetInstance().GetTargetData();
1622 bool bdecode_result = Parse_VDXBitstring(&strbit, TargetData);
1624 if (bdecode_result) {
1625 switch (TargetData->MID) {
1630 if (!TargetData->b_positionDoubtful) {
1631 pos->
kLat = TargetData->Lat;
1632 pos->
kLon = TargetData->Lon;
1638 if (TargetData->COG == 360.0)
1641 pos->
kCog = TargetData->COG;
1643 if (TargetData->SOG > 102.2)
1646 pos->
kSog = TargetData->SOG;
1648 if ((
int)TargetData->HDG == 511)
1651 pos->
kHdt = TargetData->HDG;
1659 return AIS_GENERIC_ERROR;
1664 return AIS_GENERIC_ERROR;
1672AisError AisDecoder::DecodeN0183(
const wxString &str) {
1673 AisError ret = AIS_GENERIC_ERROR;
1674 wxString string_to_parse;
1676 double gpsg_lat, gpsg_lon, gpsg_mins, gpsg_degs;
1677 double gpsg_cog, gpsg_sog, gpsg_utc_time;
1678 int gpsg_utc_hour = 0;
1679 int gpsg_utc_min = 0;
1680 int gpsg_utc_sec = 0;
1681 char gpsg_name_str[21];
1684 bool bdecode_result =
false;
1691 long arpa_tgt_num = 0;
1692 double arpa_sog = 0.;
1693 double arpa_cog = 0.;
1694 double arpa_lat = 0.;
1695 double arpa_lon = 0.;
1696 double arpa_dist = 0.;
1697 double arpa_brg = 0.;
1698 wxString arpa_brgunit;
1699 wxString arpa_status;
1700 wxString arpa_distunit;
1701 wxString arpa_cogunit;
1702 wxString arpa_reftarget;
1703 double arpa_mins, arpa_degs;
1704 double arpa_utc_time;
1705 int arpa_utc_hour = 0;
1706 int arpa_utc_min = 0;
1707 int arpa_utc_sec = 0;
1708 char arpa_name_str[21];
1709 bool arpa_lost =
true;
1710 bool arpa_nottracked =
false;
1712 double aprs_lat = 0.;
1713 double aprs_lon = 0.;
1714 char aprs_name_str[21];
1715 double aprs_mins, aprs_degs;
1717 std::shared_ptr<AisTargetData> pTargetData = 0;
1718 std::shared_ptr<AisTargetData> pStaleTarget = NULL;
1719 bool bnewtarget =
false;
1720 int last_report_ticks;
1724 if (str.Len() > 128)
return AIS_NMEAVDX_TOO_LONG;
1726 if (!NMEACheckSumOK(str)) {
1727 return AIS_NMEAVDX_CHECKSUM_BAD;
1729 if (str.Mid(1, 2).IsSameAs(_T(
"CD"))) {
1732 }
else if (str.Mid(3, 3).IsSameAs(_T(
"TTM"))) {
1736 wxString string(str);
1737 wxStringTokenizer tkz(
string, _T(
",*"));
1740 token = tkz.GetNextToken();
1741 token = tkz.GetNextToken();
1742 token.ToLong(&arpa_tgt_num);
1743 token = tkz.GetNextToken();
1744 token.ToDouble(&arpa_dist);
1745 token = tkz.GetNextToken();
1746 token.ToDouble(&arpa_brg);
1747 arpa_brgunit = tkz.GetNextToken();
1748 if (arpa_brgunit == _T(
"R")) {
1749 if (std::isnan(arpa_ref_hdg)) {
1750 if (!std::isnan(gHdt))
1755 arpa_brg += arpa_ref_hdg;
1756 if (arpa_brg >= 360.) arpa_brg -= 360.;
1758 token = tkz.GetNextToken();
1759 token.ToDouble(&arpa_sog);
1760 token = tkz.GetNextToken();
1761 token.ToDouble(&arpa_cog);
1762 arpa_cogunit = tkz.GetNextToken();
1763 if (arpa_cogunit == _T(
"R")) {
1764 if (std::isnan(arpa_ref_hdg)) {
1765 if (!std::isnan(gHdt))
1770 arpa_cog += arpa_ref_hdg;
1771 if (arpa_cog >= 360.) arpa_cog -= 360.;
1773 token = tkz.GetNextToken();
1774 token = tkz.GetNextToken();
1776 arpa_distunit = tkz.GetNextToken();
1777 token = tkz.GetNextToken();
1778 if (token == wxEmptyString)
1779 token = wxString::Format(_T(
"ARPA %ld"), arpa_tgt_num);
1780 int len = wxMin(token.Length(), 20);
1781 strncpy(arpa_name_str, token.mb_str(), len);
1782 arpa_name_str[len] = 0;
1783 arpa_status = tkz.GetNextToken();
1784 if (arpa_status != _T(
"L" )) {
1786 }
else if (arpa_status != wxEmptyString)
1787 arpa_nottracked =
true;
1788 arpa_reftarget = tkz.GetNextToken();
1789 if (tkz.HasMoreTokens()) {
1790 token = tkz.GetNextToken();
1791 token.ToDouble(&arpa_utc_time);
1792 arpa_utc_hour = (int)(arpa_utc_time / 10000.0);
1793 arpa_utc_min = (int)(arpa_utc_time / 100.0) - arpa_utc_hour * 100;
1795 (int)arpa_utc_time - arpa_utc_hour * 10000 - arpa_utc_min * 100;
1797 arpa_utc_hour = wxDateTime::Now().ToUTC().GetHour();
1798 arpa_utc_min = wxDateTime::Now().ToUTC().GetMinute();
1799 arpa_utc_sec = wxDateTime::Now().ToUTC().GetSecond();
1802 if (arpa_distunit == _T(
"K")) {
1803 arpa_dist = fromUsrDistance(arpa_dist, DISTANCE_KM, g_iDistanceFormat);
1804 arpa_sog = fromUsrSpeed(arpa_sog, SPEED_KMH, g_iSpeedFormat);
1805 }
else if (arpa_distunit == _T(
"S")) {
1806 arpa_dist = fromUsrDistance(arpa_dist, DISTANCE_MI, g_iDistanceFormat);
1807 arpa_sog = fromUsrSpeed(arpa_sog, SPEED_MPH, g_iSpeedFormat);
1810 mmsi = arpa_mmsi = 199200000 + arpa_tgt_num;
1814 }
else if (str.Mid(3, 3).IsSameAs(_T(
"TLL"))) {
1817 wxString aprs_tll_str;
1818 wxString string(str);
1819 wxStringTokenizer tkz(
string, _T(
",*"));
1822 aprs_tll_str = tkz.GetNextToken();
1823 token = tkz.GetNextToken();
1824 token.ToLong(&arpa_tgt_num);
1825 token = tkz.GetNextToken();
1826 token.ToDouble(&arpa_lat);
1827 arpa_degs = (int)(arpa_lat / 100.0);
1828 arpa_mins = arpa_lat - arpa_degs * 100.0;
1829 arpa_lat = arpa_degs + arpa_mins / 60.0;
1830 token = tkz.GetNextToken();
1831 if (token.Mid(0, 1).Contains(_T(
"S")) ==
true ||
1832 token.Mid(0, 1).Contains(_T(
"s")) ==
true)
1833 arpa_lat = 0. - arpa_lat;
1834 token = tkz.GetNextToken();
1835 token.ToDouble(&arpa_lon);
1836 arpa_degs = (int)(arpa_lon / 100.0);
1837 arpa_mins = arpa_lon - arpa_degs * 100.0;
1838 arpa_lon = arpa_degs + arpa_mins / 60.0;
1839 token = tkz.GetNextToken();
1840 if (token.Mid(0, 1).Contains(_T(
"W")) ==
true ||
1841 token.Mid(0, 1).Contains(_T(
"w")) ==
true)
1842 arpa_lon = 0. - arpa_lon;
1843 token = tkz.GetNextToken();
1844 if (token == wxEmptyString)
1845 token = wxString::Format(_T(
"ARPA %d"), arpa_tgt_num);
1846 int len = wxMin(token.Length(), 20);
1847 strncpy(arpa_name_str, token.mb_str(), len);
1848 arpa_name_str[len] = 0;
1849 token = tkz.GetNextToken();
1850 token.ToDouble(&arpa_utc_time);
1851 arpa_utc_hour = (int)(arpa_utc_time / 10000.0);
1852 arpa_utc_min = (int)(arpa_utc_time / 100.0) - arpa_utc_hour * 100;
1854 (int)arpa_utc_time - arpa_utc_hour * 10000 - arpa_utc_min * 100;
1859 if (arpa_status != _T(
"L"))
1861 else if (arpa_status != wxEmptyString)
1862 arpa_nottracked =
true;
1863 arpa_reftarget = tkz.GetNextToken();
1870 }
else if (str.Mid(3, 3).IsSameAs(_T(
"OSD"))) {
1872 wxString string(str);
1873 wxStringTokenizer tkz(
string, _T(
",*"));
1876 token = tkz.GetNextToken();
1877 token = tkz.GetNextToken();
1878 token.ToDouble(&arpa_ref_hdg);
1888 }
else if (g_bWplUsePosition && str.Mid(3, 3).IsSameAs(_T(
"WPL"))) {
1890 wxString string(str);
1891 wxStringTokenizer tkz(
string, _T(
",*"));
1894 token = tkz.GetNextToken();
1895 token = tkz.GetNextToken();
1896 token.ToDouble(&aprs_lat);
1897 aprs_degs = (int)(aprs_lat / 100.0);
1898 aprs_mins = aprs_lat - aprs_degs * 100.0;
1899 aprs_lat = aprs_degs + aprs_mins / 60.0;
1900 token = tkz.GetNextToken();
1901 if (token.Mid(0, 1).Contains(_T(
"S")) ==
true ||
1902 token.Mid(0, 1).Contains(_T(
"s")) ==
true)
1903 aprs_lat = 0. - aprs_lat;
1904 token = tkz.GetNextToken();
1905 token.ToDouble(&aprs_lon);
1906 aprs_degs = (int)(aprs_lon / 100.0);
1907 aprs_mins = aprs_lon - aprs_degs * 100.0;
1908 aprs_lon = aprs_degs + aprs_mins / 60.0;
1909 token = tkz.GetNextToken();
1910 if (token.Mid(0, 1).Contains(_T(
"W")) ==
true ||
1911 token.Mid(0, 1).Contains(_T(
"w")) ==
true)
1912 aprs_lon = 0. - aprs_lon;
1913 token = tkz.GetNextToken();
1914 int len = wxMin(token.Length(), 20);
1915 strncpy(aprs_name_str, token.mb_str(), len + 1);
1916 if (0 == g_WplAction) {
1918 aprs_name_str[len] = 0;
1919 for (i = 0; i < len; i++) {
1921 hash += (int)(aprs_name_str[i]);
1922 while (hash >= 100000) hash = hash / 100000;
1924 mmsi = aprs_mmsi = 199300000 + hash;
1928 }
else if (1 == g_WplAction) {
1930 aprs_name_str, wxEmptyString,
false);
1932 InsertWpt(pWP,
true);
1934 }
else if (str.Mid(1, 5).IsSameAs(_T(
"FRPOS"))) {
1938 wxString string(str);
1939 wxStringTokenizer tkz(
string, _T(
",*"));
1942 token = tkz.GetNextToken();
1944 token = tkz.GetNextToken();
1945 token.ToDouble(&gpsg_lat);
1946 gpsg_degs = (int)(gpsg_lat / 100.0);
1947 gpsg_mins = gpsg_lat - gpsg_degs * 100.0;
1948 gpsg_lat = gpsg_degs + gpsg_mins / 60.0;
1950 token = tkz.GetNextToken();
1951 if (token.Mid(0, 1).Contains(_T(
"S")) ==
true ||
1952 token.Mid(0, 1).Contains(_T(
"s")) ==
true)
1953 gpsg_lat = 0. - gpsg_lat;
1955 token = tkz.GetNextToken();
1956 token.ToDouble(&gpsg_lon);
1957 gpsg_degs = (int)(gpsg_lon / 100.0);
1958 gpsg_mins = gpsg_lon - gpsg_degs * 100.0;
1959 gpsg_lon = gpsg_degs + gpsg_mins / 60.0;
1961 token = tkz.GetNextToken();
1962 if (token.Mid(0, 1).Contains(_T(
"W")) ==
true ||
1963 token.Mid(0, 1).Contains(_T(
"w")) ==
true)
1964 gpsg_lon = 0. - gpsg_lon;
1966 token = tkz.GetNextToken();
1969 token = tkz.GetNextToken();
1970 token.ToDouble(&gpsg_sog);
1972 token = tkz.GetNextToken();
1973 token.ToDouble(&gpsg_cog);
1975 token = tkz.GetNextToken();
1978 token = tkz.GetNextToken();
1979 token.ToDouble(&gpsg_utc_time);
1980 gpsg_utc_hour = (int)(gpsg_utc_time / 10000.0);
1981 gpsg_utc_min = (int)(gpsg_utc_time / 100.0) - gpsg_utc_hour * 100;
1983 (int)gpsg_utc_time - gpsg_utc_hour * 10000 - gpsg_utc_min * 100;
1987 token = tkz.GetNextToken();
1988 int i, len, hash = 0;
1989 len = wxMin(wxStrlen(token), 20);
1990 strncpy(gpsg_name_str, token.mb_str(), len);
1991 gpsg_name_str[len] = 0;
1992 for (i = 0; i < len; i++) {
1994 hash += (int)(token[i]);
1995 while (hash >= 100000) hash = hash / 100000;
1998 gpsg_mmsi = 199000000 + hash;
2000 }
else if (!str.Mid(3, 2).IsSameAs(_T(
"VD"))) {
2001 return AIS_NMEAVDX_BAD;
2007 wxString string(str);
2008 wxStringTokenizer tkz(
string, _T(
","));
2011 token = tkz.GetNextToken();
2013 token = tkz.GetNextToken();
2014 nsentences = atoi(token.mb_str());
2016 token = tkz.GetNextToken();
2017 isentence = atoi(token.mb_str());
2019 token = tkz.GetNextToken();
2020 long lsequence_id = 0;
2021 token.ToLong(&lsequence_id);
2023 token = tkz.GetNextToken();
2025 token.ToLong(&lchannel);
2028 string_to_parse.Clear();
2032 if ((1 == nsentences) && (1 == isentence)) {
2033 string_to_parse = tkz.GetNextToken();
2036 else if (nsentences > 1) {
2037 if (1 == isentence) {
2038 sentence_accumulator = tkz.GetNextToken();
2042 sentence_accumulator += tkz.GetNextToken();
2045 if (isentence == nsentences) {
2046 string_to_parse = sentence_accumulator;
2050 if (mmsi || (!string_to_parse.IsEmpty() &&
2051 (string_to_parse.Len() < AIS_MAX_MESSAGE_LEN))) {
2053 wxCharBuffer abuf = string_to_parse.ToUTF8();
2055 return AIS_GENERIC_ERROR;
2060 if (!mmsi) mmsi = strbit.GetInt(9, 30);
2061 long mmsi_long = mmsi;
2064 int origin_mmsi = 0;
2065 int messID = strbit.GetInt(1, 6);
2066 int dac = strbit.GetInt(41, 10);
2067 int fi = strbit.GetInt(51, 6);
2069 int met_lon, met_lat;
2070 if (dac == 001 && fi == 31) {
2072 met_lon = strbit.GetInt(57, 25);
2073 met_lat = strbit.GetInt(82, 24);
2074 mmsi = AisMeteoNewMmsi(mmsi, met_lat, met_lon, 25, 0);
2077 }
else if (dac == 367 && fi == 33) {
2079 const int size = strbit.GetBitCount();
2080 if (size < 168)
return AIS_GENERIC_ERROR;
2081 const int startb = 56;
2082 const int slot_size = 112;
2083 const int extra_bits = (size - startb) % slot_size;
2084 if (extra_bits > 0)
return AIS_GENERIC_ERROR;
2086 int mes_type = strbit.GetInt(57, 4);
2087 int site_ID = strbit.GetInt(77, 7);
2088 if (mes_type == 0) {
2090 met_lon = strbit.GetInt(90, 28);
2091 met_lat = strbit.GetInt(118, 27);
2092 mmsi = AisMeteoNewMmsi(mmsi, met_lat, met_lon, 28, site_ID);
2097 int x_mmsi = AisMeteoNewMmsi(mmsi, 91, 181, 0, site_ID);
2103 return AIS_GENERIC_ERROR;
2109 if (mmsi == g_OwnShipmmsi)
return AIS_GENERIC_ERROR;
2112 auto it = AISTargetList.find(mmsi);
2113 if (it == AISTargetList.end())
2115 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
2120 pTargetData->MMSI = mmsi;
2121 pTargetData->met_data.original_mmsi = origin_mmsi;
2124 pTargetData = it->second;
2127 pStaleTarget = pTargetData;
2129 pTargetData->MMSI = mmsi;
2130 pTargetData->met_data.original_mmsi = origin_mmsi;
2133 for (
unsigned int i = 0; i < g_MMSI_Props_Array.GetCount(); i++) {
2135 if (mmsi == props->MMSI) {
2137 if (TRACKTYPE_NEVER == props->TrackType) {
2138 pTargetData->b_show_track =
false;
2139 }
else if (TRACKTYPE_ALWAYS == props->TrackType) {
2140 pTargetData->b_show_track =
true;
2145 if (props->m_bignore)
return AIS_NoError;
2148 else if (props->m_bVDM) {
2150 if (str.Mid(3, 9).IsSameAs(wxT(
"VDM,1,1,,"))) {
2151 int message_ID = strbit.GetInt(1, 6);
2154 if ((message_ID <= 3) || (message_ID == 18)) {
2156 pTargetData->b_OwnShip =
true;
2158 wxString aivdostr = str;
2159 aivdostr.replace(1, 5,
"AIVDO");
2160 unsigned char calculated_checksum = 0;
2161 wxString::iterator i;
2162 for (i = aivdostr.begin() + 1; i != aivdostr.end() && *i !=
'*';
2164 calculated_checksum ^=
static_cast<unsigned char>(*i);
2167 if (i <= aivdostr.end() - 3)
2170 wxString::Format(_(
"%02X"), calculated_checksum));
2172 gps_watchdog_timeout_ticks =
2175 std::string full_sentence = aivdostr.ToStdString();
2176 std::string identifier(
"AIVDO");
2180 auto address = std::make_shared<NavAddr0183>(
"virtual");
2181 auto msg = std::make_shared<const Nmea0183Msg>(
2182 identifier, full_sentence, address);
2183 auto msg_all = std::make_shared<const Nmea0183Msg>(*msg,
"ALL");
2185 auto &msgbus = NavMsgBus::GetInstance();
2187 msgbus.Notify(std::move(msg));
2188 msgbus.Notify(std::move(msg_all));
2198 wxDateTime now = wxDateTime::Now();
2202 last_report_ticks = pStaleTarget->PositionReportTicks;
2204 last_report_ticks = now.GetTicks();
2208 pSelectAIS->DeleteSelectablePoint((
void *)mmsi_long, SELTYPE_AISTARGET);
2212 pTargetData->LastPositionReportTicks = pTargetData->PositionReportTicks;
2213 pTargetData->PositionReportTicks = now.GetTicks();
2214 pTargetData->StaticReportTicks = now.GetTicks();
2215 pTargetData->m_utc_hour = gpsg_utc_hour;
2216 pTargetData->m_utc_min = gpsg_utc_min;
2217 pTargetData->m_utc_sec = gpsg_utc_sec;
2218 pTargetData->m_date_string = gpsg_date;
2219 pTargetData->MMSI = gpsg_mmsi;
2220 pTargetData->NavStatus = 0;
2221 pTargetData->Lat = gpsg_lat;
2222 pTargetData->Lon = gpsg_lon;
2223 pTargetData->b_positionOnceValid =
true;
2224 pTargetData->COG = gpsg_cog;
2225 pTargetData->SOG = gpsg_sog;
2226 pTargetData->ShipType = 52;
2227 pTargetData->Class = AIS_GPSG_BUDDY;
2228 memcpy(pTargetData->ShipName, gpsg_name_str,
sizeof(gpsg_name_str));
2229 pTargetData->b_nameValid =
true;
2230 pTargetData->b_active =
true;
2231 pTargetData->b_lost =
false;
2233 bdecode_result =
true;
2234 }
else if (arpa_mmsi) {
2235 pTargetData->m_utc_hour = arpa_utc_hour;
2236 pTargetData->m_utc_min = arpa_utc_min;
2237 pTargetData->m_utc_sec = arpa_utc_sec;
2238 pTargetData->MMSI = arpa_mmsi;
2239 pTargetData->NavStatus = 15;
2240 if (str.Mid(3, 3).IsSameAs(_T(
"TLL"))) {
2243 (now.GetTicks() - pTargetData->PositionReportTicks);
2244 if (age_of_last > 0) {
2245 ll_gc_ll_reverse(pTargetData->Lat, pTargetData->Lon, arpa_lat,
2246 arpa_lon, &pTargetData->COG, &pTargetData->SOG);
2247 pTargetData->SOG = pTargetData->SOG * 3600 / age_of_last;
2250 pTargetData->Lat = arpa_lat;
2251 pTargetData->Lon = arpa_lon;
2252 }
else if (str.Mid(3, 3).IsSameAs(_T(
"TTM"))) {
2253 if (arpa_dist != 0.)
2254 ll_gc_ll(gLat, gLon, arpa_brg, arpa_dist, &pTargetData->Lat,
2258 pTargetData->COG = arpa_cog;
2259 pTargetData->SOG = arpa_sog;
2261 pTargetData->LastPositionReportTicks = pTargetData->PositionReportTicks;
2262 pTargetData->PositionReportTicks = now.GetTicks();
2263 pTargetData->StaticReportTicks = now.GetTicks();
2264 pTargetData->b_positionOnceValid =
true;
2265 pTargetData->ShipType = 55;
2266 pTargetData->Class = AIS_ARPA;
2268 memcpy(pTargetData->ShipName, arpa_name_str,
sizeof(arpa_name_str));
2269 if (arpa_status != _T(
"Q"))
2270 pTargetData->b_nameValid =
true;
2272 pTargetData->b_nameValid =
false;
2273 pTargetData->b_active = !arpa_lost;
2274 pTargetData->b_lost = arpa_nottracked;
2276 bdecode_result =
true;
2277 }
else if (aprs_mmsi) {
2278 pTargetData->m_utc_hour = now.GetHour();
2279 pTargetData->m_utc_min = now.GetMinute();
2280 pTargetData->m_utc_sec = now.GetSecond();
2281 pTargetData->MMSI = aprs_mmsi;
2282 pTargetData->NavStatus = 15;
2284 int age_of_last = (now.GetTicks() - pTargetData->PositionReportTicks);
2285 if (age_of_last > 0) {
2286 ll_gc_ll_reverse(pTargetData->Lat, pTargetData->Lon, aprs_lat,
2287 aprs_lon, &pTargetData->COG, &pTargetData->SOG);
2288 pTargetData->SOG = pTargetData->SOG * 3600 / age_of_last;
2291 pTargetData->LastPositionReportTicks = pTargetData->PositionReportTicks;
2292 pTargetData->PositionReportTicks = now.GetTicks();
2293 pTargetData->StaticReportTicks = now.GetTicks();
2294 pTargetData->Lat = aprs_lat;
2295 pTargetData->Lon = aprs_lon;
2296 pTargetData->b_positionOnceValid =
true;
2297 pTargetData->ShipType = 56;
2298 pTargetData->Class = AIS_APRS;
2299 memcpy(pTargetData->ShipName, aprs_name_str,
sizeof(aprs_name_str));
2300 pTargetData->b_nameValid =
true;
2301 pTargetData->b_active =
true;
2302 pTargetData->b_lost =
false;
2304 bdecode_result =
true;
2308 Parse_VDXBitstring(&strbit, pTargetData);
2312 getMmsiProperties(pTargetData);
2315 pTargetData->RecentPeriod =
2316 pTargetData->PositionReportTicks - last_report_ticks;
2326 CommitAISTarget(pTargetData, str, bdecode_result, bnewtarget);
2331 if ((n_msgs % 10000) == 0)
2332 printf(
"n_msgs %10d m_n_targets: %6d n_msg1: %10d n_msg5+24: %10d \n",
2333 n_msgs, m_n_targets, n_msg1, n_msg5 + n_msg24);
2339void AisDecoder::CommitAISTarget(std::shared_ptr<AisTargetData> pTargetData,
2340 const wxString &str,
bool message_valid,
2342 m_pLatestTargetData = pTargetData;
2344 if (!str.IsEmpty()) {
2345 if (str.Mid(3, 3).IsSameAs(_T(
"VDO")))
2346 pTargetData->b_OwnShip =
true;
2348 pTargetData->b_OwnShip =
false;
2351 if (!pTargetData->b_OwnShip) {
2353 if (0 == m_persistent_tracks.count(pTargetData->MMSI)) {
2355 pTargetData->b_PersistTrack =
false;
2357 for (
unsigned int i = 0; i < g_MMSI_Props_Array.GetCount(); i++) {
2358 if (pTargetData->MMSI == g_MMSI_Props_Array[i]->MMSI) {
2360 pTargetData->b_mPropPersistTrack = props->m_bPersistentTrack;
2367 pTargetData->b_NoTrack =
false;
2372 if (message_valid) {
2374 if (pTargetData->MMSI) {
2375 AISshipNameCache(pTargetData.get(), AISTargetNamesC, AISTargetNamesNC,
2378 AISTargetList[pTargetData->MMSI] =
2381 if (!pTargetData->area_notices.empty()) {
2382 auto it = AIS_AreaNotice_Sources.find(pTargetData->MMSI);
2383 if (it == AIS_AreaNotice_Sources.end())
2384 AIS_AreaNotice_Sources[pTargetData->MMSI] = pTargetData;
2389 if (!pTargetData->b_OwnShip) {
2390 if (pTargetData->b_positionOnceValid) {
2391 long mmsi_long = pTargetData->MMSI;
2392 SelectItem *pSel = pSelectAIS->AddSelectablePoint(
2393 pTargetData->Lat, pTargetData->Lon, (
void *)mmsi_long,
2395 pSel->SetUserData(pTargetData->MMSI);
2399 UpdateOneCPA(pTargetData.get());
2402 if (pTargetData->b_show_track) UpdateOneTrack(pTargetData.get());
2415 if (!pTargetData->b_OwnShip) {
2416 if (pTargetData->b_positionOnceValid) {
2417 long mmsi_long = pTargetData->MMSI;
2418 SelectItem *pSel = pSelectAIS->AddSelectablePoint(
2419 pTargetData->Lat, pTargetData->Lon, (
void *)mmsi_long,
2421 pSel->SetUserData(pTargetData->MMSI);
2428void AisDecoder::getAISTarget(
long mmsi,
2429 std::shared_ptr<AisTargetData> &pTargetData,
2430 std::shared_ptr<AisTargetData> &pStaleTarget,
2431 bool &bnewtarget,
int &last_report_ticks,
2433 now = wxDateTime::Now();
2434 auto it = AISTargetList.find(mmsi);
2435 if (it == AISTargetList.end())
2437 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
2441 pTargetData = it->second;
2442 pStaleTarget = pTargetData;
2449 last_report_ticks = pStaleTarget->PositionReportTicks;
2451 last_report_ticks = now.GetTicks();
2455 pSelectAIS->DeleteSelectablePoint((
void *)mmsi, SELTYPE_AISTARGET);
2458void AisDecoder::getMmsiProperties(
2459 std::shared_ptr<AisTargetData> &pTargetData) {
2460 for (
unsigned int i = 0; i < g_MMSI_Props_Array.GetCount(); i++) {
2461 if (pTargetData->MMSI == g_MMSI_Props_Array[i]->MMSI) {
2463 pTargetData->b_isFollower = props->m_bFollower;
2464 pTargetData->b_mPropPersistTrack = props->m_bPersistentTrack;
2465 if (TRACKTYPE_NEVER == props->TrackType) {
2466 pTargetData->b_show_track =
false;
2467 }
else if (TRACKTYPE_ALWAYS == props->TrackType) {
2468 pTargetData->b_show_track =
true;
2475std::shared_ptr<AisTargetData> AisDecoder::ProcessDSx(
const wxString &str,
2477 double dsc_lat = 0.;
2478 double dsc_lon = 0.;
2479 double dsc_mins, dsc_degs, dsc_tmp, dsc_addr;
2481 double dse_lat = 0.;
2482 double dse_lon = 0.;
2483 long dsc_fmt, dsc_quadrant, dsc_cat, dsc_nature;
2486 int dsc_tx_mmsi = 0;
2488 double dse_cog = 0.;
2489 double dse_sog = 0.;
2490 wxString dse_shipName = wxEmptyString;
2495 std::shared_ptr<AisTargetData> pTargetData = NULL;
2499 wxString string(str);
2500 wxStringTokenizer tkz(
string, _T(
",*"));
2503 token = tkz.GetNextToken();
2505 if (str.Mid(3, 3).IsSameAs(_T(
"DSC"))) {
2506 m_dsc_last_string = str;
2508 token = tkz.GetNextToken();
2512 token = tkz.GetNextToken();
2514 if (dsc_fmt == 12 || dsc_fmt == 16) {
2515 dsc_mmsi = wxAtoi(token.Mid(0, 9));
2517 token.ToDouble(&dsc_addr);
2518 dsc_mmsi = 0 - (int)(dsc_addr / 10);
2521 token = tkz.GetNextToken();
2522 token.ToLong(&dsc_cat);
2524 token = tkz.GetNextToken();
2525 if (!token.IsSameAs(wxEmptyString)) {
2526 token.ToLong(&dsc_nature);
2530 token = tkz.GetNextToken();
2532 token = tkz.GetNextToken();
2533 token.ToDouble(&dsc_tmp);
2535 token = tkz.GetNextToken();
2536 token = tkz.GetNextToken();
2537 if (dsc_fmt == 16 && dsc_cat == 12 && !token.IsSameAs(wxEmptyString)) {
2539 dsc_tx_mmsi = dsc_mmsi;
2540 dsc_mmsi = wxAtoi(token.Mid(0, 9));
2542 token = tkz.GetNextToken();
2543 if (dsc_fmt == 16 && dsc_cat == 12) {
2544 if (!token.IsSameAs(wxEmptyString)) {
2545 token.ToLong(&dsc_nature);
2549 token = tkz.GetNextToken();
2550 token = tkz.GetNextToken();
2552 dsc_quadrant = (int)(dsc_tmp / 1000000000.0);
2554 if (dsc_quadrant > 3)
2557 dsc_lat = (int)(dsc_tmp / 100000.0);
2558 dsc_lon = dsc_tmp - dsc_lat * 100000.0;
2559 dsc_lat = dsc_lat - dsc_quadrant * 10000;
2560 dsc_degs = (int)(dsc_lat / 100.0);
2561 dsc_mins = dsc_lat - dsc_degs * 100.0;
2562 dsc_lat = dsc_degs + dsc_mins / 60.0;
2564 dsc_degs = (int)(dsc_lon / 100.0);
2565 dsc_mins = dsc_lon - dsc_degs * 100.0;
2566 dsc_lon = dsc_degs + dsc_mins / 60.0;
2567 switch (dsc_quadrant) {
2583 if (dsc_fmt != 02) mmsi = (int)dsc_mmsi;
2585 }
else if (str.Mid(3, 3).IsSameAs(_T(
"DSE"))) {
2586 token = tkz.GetNextToken();
2587 token = tkz.GetNextToken();
2588 token = tkz.GetNextToken();
2589 token = tkz.GetNextToken();
2590 dse_mmsi = wxAtoi(token.Mid(
2596 token = tkz.GetNextToken();
2599 token.ToDouble(&dse_tmp);
2600 dse_lat = (int)(dse_tmp / 10000.0);
2601 dse_lon = (int)(dse_tmp - dse_lat * 10000.0);
2602 dse_lat = dse_lat / 600000.0;
2603 dse_lon = dse_lon / 600000.0;
2606 while (tkz.HasMoreTokens()) {
2607 dseSymbol = tkz.GetNextToken();
2608 token = tkz.GetNextToken();
2609 if (dseSymbol.IsSameAs(_T(
"00"))) {
2610 token.ToDouble(&dse_tmp);
2611 dse_lat = (int)(dse_tmp / 10000.0);
2612 dse_lon = (int)(dse_tmp - dse_lat * 10000.0);
2613 dse_lat = dse_lat / 600000.0;
2614 dse_lon = dse_lon / 600000.0;
2615 }
else if (dseSymbol.IsSameAs(_T(
"01"))) {
2616 }
else if (dseSymbol.IsSameAs(_T(
"02"))) {
2617 token.ToDouble(&dse_tmp);
2618 dse_sog = dse_tmp / 10.0;
2619 }
else if (dseSymbol.IsSameAs(_T(
"03"))) {
2620 token.ToDouble(&dse_tmp);
2621 dse_cog = dse_tmp / 10.0;
2622 }
else if (dseSymbol.IsSameAs(_T(
"04"))) {
2623 dse_shipName = DecodeDSEExpansionCharacters(token);
2624 }
else if (dseSymbol.IsSameAs(_T(
"05"))) {
2625 }
else if (dseSymbol.IsSameAs(_T(
"06"))) {
2628 mmsi = abs((
int)dse_mmsi);
2632 wxDateTime now = wxDateTime::Now();
2634 int last_report_ticks = now.GetTicks();
2637 auto it = AISTargetList.find(mmsi);
2638 std::shared_ptr<AisTargetData> pStaleTarget = NULL;
2639 if (it == AISTargetList.end()) {
2641 pStaleTarget = it->second;
2642 last_report_ticks = pStaleTarget->PositionReportTicks;
2648 m_ptentative_dsctarget = AisTargetDataMaker::GetInstance().GetTargetData();
2650 m_ptentative_dsctarget->PositionReportTicks = now.GetTicks();
2651 m_ptentative_dsctarget->StaticReportTicks = now.GetTicks();
2653 m_ptentative_dsctarget->MMSI = mmsi;
2654 m_ptentative_dsctarget->NavStatus =
2656 m_ptentative_dsctarget->Lat = dsc_lat;
2657 m_ptentative_dsctarget->Lon = dsc_lon;
2658 m_ptentative_dsctarget->b_positionOnceValid =
true;
2659 m_ptentative_dsctarget->COG = 0;
2660 m_ptentative_dsctarget->SOG = 0;
2661 m_ptentative_dsctarget->ShipType = dsc_fmt;
2662 m_ptentative_dsctarget->Class = AIS_DSC;
2663 m_ptentative_dsctarget->b_isDSCtarget =
true;
2664 m_ptentative_dsctarget->b_nameValid =
true;
2665 if (dsc_fmt == 12 || (dsc_fmt == 16 && dsc_cat == 12)) {
2666 snprintf(m_ptentative_dsctarget->ShipName, SHIP_NAME_LEN,
"DISTRESS %d",
2668 m_ptentative_dsctarget->m_dscNature = int(dsc_nature);
2669 m_ptentative_dsctarget->m_dscTXmmsi = dsc_tx_mmsi;
2671 snprintf(m_ptentative_dsctarget->ShipName, SHIP_NAME_LEN,
"POSITION %d",
2675 m_ptentative_dsctarget->b_active =
true;
2676 m_ptentative_dsctarget->b_lost =
false;
2677 m_ptentative_dsctarget->RecentPeriod =
2678 m_ptentative_dsctarget->PositionReportTicks - last_report_ticks;
2681 if (!b_take_dsc) m_dsc_timer.Start(1000, wxTIMER_ONE_SHOT);
2686 if (dse_mmsi || b_take_dsc) {
2687 if (m_ptentative_dsctarget) {
2693 m_ptentative_dsctarget->Lat =
2694 m_ptentative_dsctarget->Lat +
2695 ((m_ptentative_dsctarget->Lat) >= 0 ? dse_lat : -dse_lat);
2696 m_ptentative_dsctarget->Lon =
2697 m_ptentative_dsctarget->Lon +
2698 ((m_ptentative_dsctarget->Lon) >= 0 ? dse_lon : -dse_lon);
2699 if (dse_shipName.length() > 0) {
2700 memset(m_ptentative_dsctarget->ShipName,
'\0', SHIP_NAME_LEN);
2701 snprintf(m_ptentative_dsctarget->ShipName, dse_shipName.length(),
2702 "%s", dse_shipName.ToAscii().data());
2704 m_ptentative_dsctarget->COG = dse_cog;
2705 m_ptentative_dsctarget->SOG = dse_sog;
2709 m_ptentative_dsctarget->RecentPeriod =
2710 m_ptentative_dsctarget->PositionReportTicks - last_report_ticks;
2715 auto it = AISTargetList.find(mmsi);
2716 if (it == AISTargetList.end()) {
2717 pTargetData = m_ptentative_dsctarget;
2719 pTargetData = it->second;
2720 std::vector<AISTargetTrackPoint> ptrack =
2721 std::move(pTargetData->m_ptrack);
2722 pTargetData->CloneFrom(
2723 m_ptentative_dsctarget
2726 pTargetData->m_ptrack =
2733 m_ptentative_dsctarget = NULL;
2735 m_pLatestTargetData = pTargetData;
2737 AISTargetList[pTargetData->MMSI] =
2740 long mmsi_long = pTargetData->MMSI;
2744 pSelectAIS->DeleteSelectablePoint((
void *)mmsi_long, SELTYPE_AISTARGET);
2747 pSelectAIS->AddSelectablePoint(pTargetData->Lat, pTargetData->Lon,
2748 (
void *)mmsi_long, SELTYPE_AISTARGET);
2749 pSel->SetUserData(pTargetData->MMSI);
2752 UpdateOneCPA(pTargetData.get());
2755 if (pTargetData->b_show_track) UpdateOneTrack(pTargetData.get());
2763wxString AisDecoder::DecodeDSEExpansionCharacters(wxString dseData) {
2765 char lookupTable[] = {
'0',
'1',
'2',
'3',
'4',
'5',
'6',
'7',
'8',
'9',
' ',
2766 'A',
'B',
'C',
'D',
'E',
'F',
'G',
'H',
'I',
'J',
'K',
2767 'L',
'M',
'N',
'O',
'P',
'Q',
'R',
'S',
'T',
'U',
'V',
2768 'W',
'X',
'Y',
'Z',
'.',
',',
'-',
'/',
' '};
2770 for (
size_t i = 0; i < dseData.length(); i += 2) {
2771 result.append(1, lookupTable[strtol(dseData.Mid(i, 2).data(), NULL, 10)]);
2780 std::shared_ptr<AisTargetData> ptd) {
2781 bool parse_result =
false;
2782 bool b_posn_report =
false;
2784 wxDateTime now = wxDateTime::Now();
2786 int message_ID = bstr->
GetInt(1, 6);
2787 ptd->MID = message_ID;
2790 int met_mmsi = ptd->MMSI;
2793 ptd->MMSI = bstr->
GetInt(9, 30);
2795 switch (message_ID) {
2801 ptd->NavStatus = bstr->
GetInt(39, 4);
2802 ptd->SOG = 0.1 * (bstr->
GetInt(51, 10));
2804 int lon = bstr->
GetInt(62, 28);
2805 if (lon & 0x08000000)
2807 double lon_tentative = lon / 600000.;
2809 int lat = bstr->
GetInt(90, 27);
2810 if (lat & 0x04000000)
2812 double lat_tentative = lat / 600000.;
2814 if ((lon_tentative <= 180.) &&
2818 ptd->Lon = lon_tentative;
2819 ptd->Lat = lat_tentative;
2820 ptd->b_positionDoubtful =
false;
2821 ptd->b_positionOnceValid =
true;
2822 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
2823 ptd->PositionReportTicks = now.GetTicks();
2825 ptd->b_positionDoubtful =
true;
2828 ptd->COG = 0.1 * (bstr->
GetInt(117, 12));
2829 ptd->HDG = 1.0 * (bstr->
GetInt(129, 9));
2831 ptd->ROTAIS = bstr->
GetInt(43, 8);
2832 double rot_dir = 1.0;
2834 if (ptd->ROTAIS == 128)
2836 else if ((ptd->ROTAIS & 0x80) == 0x80) {
2837 ptd->ROTAIS = ptd->ROTAIS - 256;
2841 ptd->ROTIND = wxRound(rot_dir * pow((((
double)ptd->ROTAIS) / 4.733),
2844 ptd->m_utc_sec = bstr->
GetInt(138, 6);
2846 if ((1 == message_ID) ||
2849 ptd->SyncState = bstr->
GetInt(151, 2);
2850 ptd->SlotTO = bstr->
GetInt(153, 2);
2851 if ((ptd->SlotTO == 1) && (ptd->SyncState == 0))
2853 ptd->m_utc_hour = bstr->
GetInt(155, 5);
2855 ptd->m_utc_min = bstr->
GetInt(160, 7);
2857 if ((ptd->m_utc_hour < 24) && (ptd->m_utc_min < 60) &&
2858 (ptd->m_utc_sec < 60)) {
2859 wxDateTime rx_time(ptd->m_utc_hour, ptd->m_utc_min, ptd->m_utc_sec);
2860 rx_ticks = rx_time.GetTicks();
2862 first_rx_ticks = rx_ticks;
2870 ptd->blue_paddle = bstr->
GetInt(144, 2);
2871 ptd->b_blue_paddle = (ptd->blue_paddle == 2);
2873 if (!ptd->b_isDSCtarget) ptd->Class = AIS_CLASS_A;
2876 int mmsi_start = ptd->MMSI / 10000000;
2878 if (mmsi_start == 97) {
2879 ptd->Class = AIS_SART;
2880 ptd->StaticReportTicks =
2894 parse_result =
true;
2895 b_posn_report =
true;
2904 ptd->SOG = 0.1 * (bstr->
GetInt(47, 10));
2906 int lon = bstr->
GetInt(58, 28);
2907 if (lon & 0x08000000)
2909 double lon_tentative = lon / 600000.;
2911 int lat = bstr->
GetInt(86, 27);
2912 if (lat & 0x04000000)
2914 double lat_tentative = lat / 600000.;
2916 if ((lon_tentative <= 180.) &&
2920 ptd->Lon = lon_tentative;
2921 ptd->Lat = lat_tentative;
2922 ptd->b_positionDoubtful =
false;
2923 ptd->b_positionOnceValid =
true;
2924 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
2925 ptd->PositionReportTicks = now.GetTicks();
2927 ptd->b_positionDoubtful =
true;
2929 ptd->COG = 0.1 * (bstr->
GetInt(113, 12));
2930 ptd->HDG = 1.0 * (bstr->
GetInt(125, 9));
2932 ptd->m_utc_sec = bstr->
GetInt(134, 6);
2934 if (!ptd->b_isDSCtarget) {
2935 if (!isBuoyMmsi(ptd->MMSI))
2936 ptd->Class = AIS_CLASS_B;
2938 ptd->Class = AIS_BUOY;
2940 parse_result =
true;
2941 b_posn_report =
true;
2949 ptd->SOG = 0.1 * (bstr->
GetInt(47, 10));
2950 int lon = bstr->
GetInt(58, 28);
2951 if (lon & 0x08000000)
2953 double lon_tentative = lon / 600000.;
2955 int lat = bstr->
GetInt(86, 27);
2956 if (lat & 0x04000000)
2958 double lat_tentative = lat / 600000.;
2960 if ((lon_tentative <= 180.) &&
2964 ptd->Lon = lon_tentative;
2965 ptd->Lat = lat_tentative;
2966 ptd->b_positionDoubtful =
false;
2967 ptd->b_positionOnceValid =
true;
2968 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
2969 ptd->PositionReportTicks = now.GetTicks();
2971 ptd->b_positionDoubtful =
true;
2973 ptd->COG = 0.1 * (bstr->
GetInt(113, 12));
2974 ptd->HDG = 1.0 * (bstr->
GetInt(125, 9));
2975 ptd->m_utc_sec = bstr->
GetInt(134, 6);
2977 bstr->GetStr(144, 120, &ptd->ShipName[0], SHIP_NAME_LEN);
2978 ptd->b_nameValid =
true;
2979 if (!ptd->b_isDSCtarget) {
2980 ptd->ShipType = (
unsigned char)bstr->
GetInt(264, 8);
2982 ptd->DimA = bstr->
GetInt(272, 9);
2983 ptd->DimB = bstr->
GetInt(281, 9);
2984 ptd->DimC = bstr->
GetInt(290, 6);
2985 ptd->DimD = bstr->
GetInt(296, 6);
2987 if (!ptd->b_isDSCtarget) {
2989 if (!isBuoyMmsi(ptd->MMSI))
2990 ptd->Class = AIS_CLASS_B;
2992 ptd->Class = AIS_BUOY;
2994 parse_result =
true;
2995 b_posn_report =
true;
3008 int bitCorrection = 10;
3009 int resolution = 10;
3012 double lon_tentative = 181.;
3013 double lat_tentative = 91.;
3016 printf(
"AIS Message 27 - received:\r\n");
3017 printf(
"MMSI : %i\r\n", ptd->MMSI);
3023 if (!ptd->b_isDSCtarget) ptd->Class = AIS_CLASS_A;
3025 ptd->NavStatus = bstr->
GetInt(39, 4);
3027 int lon = bstr->
GetInt(45, 18);
3028 int lat = bstr->
GetInt(63, 17);
3030 lat_tentative = lat;
3031 lon_tentative = lon;
3034 if (lat >= (0x4000000 >> bitCorrection)) {
3035 lat_tentative = (0x8000000 >> bitCorrection) - lat;
3036 lat_tentative *= -1;
3040 if (lon >= (0x8000000 >> bitCorrection)) {
3041 lon_tentative = (0x10000000 >> bitCorrection) - lon;
3042 lon_tentative *= -1;
3046 lat_tentative = lat_tentative / resolution / 60.0;
3047 lon_tentative = lon_tentative / resolution / 60.0;
3050 printf(
"Latitude : %f\r\n", lat_tentative);
3051 printf(
"Longitude : %f\r\n", lon_tentative);
3055 int positionLatency = bstr->
GetInt(95, 1);
3057 if ((lon_tentative <= 180.) &&
3061 ptd->Lon = lon_tentative;
3062 ptd->Lat = lat_tentative;
3063 ptd->b_positionDoubtful =
false;
3064 ptd->b_positionOnceValid =
true;
3065 if (positionLatency == 0) {
3068 printf(
"Low latency position report.\r\n");
3070 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
3071 ptd->PositionReportTicks = now.GetTicks();
3074 ptd->b_positionDoubtful =
true;
3076 ptd->SOG = 1.0 * (bstr->
GetInt(80, 6));
3077 ptd->COG = 1.0 * (bstr->
GetInt(85, 9));
3079 b_posn_report =
true;
3080 parse_result =
true;
3086 if (!ptd->b_isDSCtarget) ptd->Class = AIS_CLASS_A;
3092 int AIS_version_indicator = bstr->
GetInt(39, 2);
3093 if (AIS_version_indicator < 4) {
3094 ptd->IMO = bstr->
GetInt(41, 30);
3096 bstr->GetStr(71, 42, &ptd->CallSign[0], 7);
3097 bstr->GetStr(113, 120, &ptd->ShipName[0], SHIP_NAME_LEN);
3098 ptd->b_nameValid =
true;
3099 if (!ptd->b_isDSCtarget) {
3100 ptd->ShipType = (
unsigned char)bstr->
GetInt(233, 8);
3103 ptd->DimA = bstr->
GetInt(241, 9);
3104 ptd->DimB = bstr->
GetInt(250, 9);
3105 ptd->DimC = bstr->
GetInt(259, 6);
3106 ptd->DimD = bstr->
GetInt(265, 6);
3108 ptd->ETA_Mo = bstr->
GetInt(275, 4);
3109 ptd->ETA_Day = bstr->
GetInt(279, 5);
3110 ptd->ETA_Hr = bstr->
GetInt(284, 5);
3111 ptd->ETA_Min = bstr->
GetInt(289, 6);
3113 ptd->Draft = (double)(bstr->
GetInt(295, 8)) / 10.0;
3115 bstr->GetStr(303, 120, &ptd->Destination[0], DESTINATION_LEN - 1);
3117 ptd->StaticReportTicks = now.GetTicks();
3119 parse_result =
true;
3126 int part_number = bstr->
GetInt(39, 2);
3127 if (0 == part_number) {
3128 bstr->GetStr(41, 120, &ptd->ShipName[0], SHIP_NAME_LEN);
3129 ptd->b_nameValid =
true;
3130 parse_result =
true;
3132 }
else if (1 == part_number) {
3133 if (!ptd->b_isDSCtarget) {
3134 ptd->ShipType = (
unsigned char)bstr->
GetInt(41, 8);
3136 bstr->GetStr(91, 42, &ptd->CallSign[0], 7);
3138 ptd->DimA = bstr->
GetInt(133, 9);
3139 ptd->DimB = bstr->
GetInt(142, 9);
3140 ptd->DimC = bstr->
GetInt(151, 6);
3141 ptd->DimD = bstr->
GetInt(157, 6);
3142 parse_result =
true;
3148 ptd->Class = AIS_BASE;
3150 ptd->m_utc_hour = bstr->
GetInt(62, 5);
3151 ptd->m_utc_min = bstr->
GetInt(67, 6);
3152 ptd->m_utc_sec = bstr->
GetInt(73, 6);
3154 int lon = bstr->
GetInt(80, 28);
3155 if (lon & 0x08000000)
3157 double lon_tentative = lon / 600000.;
3159 int lat = bstr->
GetInt(108, 27);
3160 if (lat & 0x04000000)
3162 double lat_tentative = lat / 600000.;
3164 if ((lon_tentative <= 180.) &&
3168 ptd->Lon = lon_tentative;
3169 ptd->Lat = lat_tentative;
3170 ptd->b_positionDoubtful =
false;
3171 ptd->b_positionOnceValid =
true;
3172 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
3173 ptd->PositionReportTicks = now.GetTicks();
3175 ptd->b_positionDoubtful =
true;
3181 parse_result =
true;
3182 b_posn_report =
true;
3188 ptd->SOG = bstr->
GetInt(51, 10);
3190 int lon = bstr->
GetInt(62, 28);
3191 if (lon & 0x08000000)
3193 double lon_tentative = lon / 600000.;
3195 int lat = bstr->
GetInt(90, 27);
3196 if (lat & 0x04000000)
3198 double lat_tentative = lat / 600000.;
3200 if ((lon_tentative <= 180.) &&
3204 ptd->Lon = lon_tentative;
3205 ptd->Lat = lat_tentative;
3206 ptd->b_positionDoubtful =
false;
3207 ptd->b_positionOnceValid =
true;
3208 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
3209 ptd->PositionReportTicks = now.GetTicks();
3211 ptd->b_positionDoubtful =
true;
3214 ptd->COG = 0.1 * (bstr->
GetInt(117, 12));
3216 int alt_tent = bstr->
GetInt(39, 12);
3217 ptd->altitude = alt_tent;
3219 ptd->b_SarAircraftPosnReport =
true;
3221 parse_result =
true;
3222 b_posn_report =
true;
3228 ptd->ShipType = (
unsigned char)bstr->
GetInt(39, 5);
3234 ptd->DimA = bstr->
GetInt(220, 9);
3235 ptd->DimB = bstr->
GetInt(229, 9);
3236 ptd->DimC = bstr->
GetInt(238, 6);
3237 ptd->DimD = bstr->
GetInt(244, 6);
3240 ptd->m_utc_sec = bstr->
GetInt(254, 6);
3242 int offpos = bstr->
GetInt(260, 1);
3243 int virt = bstr->
GetInt(270, 1);
3246 ptd->NavStatus = ATON_VIRTUAL;
3248 ptd->NavStatus = ATON_REAL;
3249 if (ptd->m_utc_sec <= 59 ) {
3250 ptd->NavStatus += 1;
3251 if (offpos) ptd->NavStatus += 1;
3254 bstr->GetStr(44, 120, &ptd->ShipName[0], SHIP_NAME_LEN);
3257 if (bstr->GetBitCount() > 276) {
3258 int nx = ((bstr->GetBitCount() - 272) / 6) * 6;
3259 bstr->GetStr(273, nx, &ptd->ShipNameExtension[0], 14);
3260 ptd->ShipNameExtension[14] = 0;
3262 ptd->ShipNameExtension[0] = 0;
3265 ptd->b_nameValid =
true;
3267 parse_result =
true;
3269 ptd->Class = AIS_ATON;
3271 int lon = bstr->
GetInt(165, 28);
3273 if (lon & 0x08000000)
3275 double lon_tentative = lon / 600000.;
3277 int lat = bstr->
GetInt(193, 27);
3279 if (lat & 0x04000000)
3281 double lat_tentative = lat / 600000.;
3283 if ((lon_tentative <= 180.) &&
3287 ptd->Lon = lon_tentative;
3288 ptd->Lat = lat_tentative;
3289 ptd->b_positionDoubtful =
false;
3290 ptd->b_positionOnceValid =
true;
3291 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
3292 ptd->PositionReportTicks = now.GetTicks();
3294 ptd->b_positionDoubtful =
true;
3296 b_posn_report =
true;
3301 int dac = bstr->
GetInt(41, 10);
3302 int fi = bstr->
GetInt(51, 6);
3308 ptd->b_isEuroInland =
true;
3310 bstr->GetStr(57, 48, &ptd->Euro_VIN[0], 8);
3311 ptd->Euro_Length = ((double)bstr->
GetInt(105, 13)) / 10.0;
3312 ptd->Euro_Beam = ((double)bstr->
GetInt(118, 10)) / 10.0;
3313 ptd->UN_shiptype = bstr->
GetInt(128, 14);
3314 ptd->Euro_Draft = ((double)bstr->
GetInt(145, 11)) / 100.0;
3315 parse_result =
true;
3318 if (dac == 1 || dac == 366)
3322 if (bstr->GetBitCount() >= 111) {
3324 an.link_id = bstr->
GetInt(57, 10);
3325 an.notice_type = bstr->
GetInt(67, 7);
3326 an.month = bstr->
GetInt(74, 4);
3327 an.day = bstr->
GetInt(78, 5);
3328 an.hour = bstr->
GetInt(83, 5);
3329 an.minute = bstr->
GetInt(88, 6);
3330 an.duration_minutes = bstr->
GetInt(94, 18);
3332 wxDateTime now = wxDateTime::Now();
3335 an.start_time.Set(an.day, wxDateTime::Month(an.month - 1),
3336 now.GetYear(), an.hour, an.minute);
3341 if (an.start_time > now + wxTimeSpan::Hours(48))
3342 an.start_time.Set(an.day, wxDateTime::Month(an.month - 1),
3343 now.GetYear() - 1, an.hour, an.minute);
3346 an.start_time + wxTimeSpan::Minutes(an.duration_minutes);
3351 if (an.expiry_time < now - wxTimeSpan::Hours(24)) {
3352 an.start_time.Set(an.day, wxDateTime::Month(an.month - 1),
3353 now.GetYear() + 1, an.hour, an.minute);
3355 an.start_time + wxTimeSpan::Minutes(an.duration_minutes);
3360 int subarea_len = 87;
3363 float pos_scale = 60000.0;
3371 pos_scale = 600000.0;
3376 int subarea_count = (bstr->GetBitCount() - 111) / subarea_len;
3377 for (
int i = 0; i < subarea_count; ++i) {
3378 int base = 111 + i * subarea_len;
3380 sa.shape = bstr->
GetInt(base + 1, 3);
3381 int scale_factor = 1;
3382 if (sa.shape == AIS8_001_22_SHAPE_TEXT) {
3385 bstr->GetStr(base + 4, subarea_len - 3, t, 14);
3386 sa.text = wxString(t, wxConvUTF8);
3388 int scale_multipliers[4] = {1, 10, 100, 1000};
3389 scale_factor = scale_multipliers[bstr->
GetInt(base + 4, 2)];
3391 case AIS8_001_22_SHAPE_SECTOR:
3392 sa.left_bound_deg = bstr->
GetInt(
3393 base + 6 + lon_len + lat_len + prec_size + 12, 9);
3394 sa.right_bound_deg = bstr->
GetInt(
3395 base + 6 + lon_len + lat_len + prec_size + 12 + 9, 9);
3396 case AIS8_001_22_SHAPE_CIRCLE:
3398 bstr->
GetInt(base + 6 + lon_len + lat_len + 3, 12) *
3401 case AIS8_001_22_SHAPE_RECT:
3403 bstr->
GetInt(base + 6, lon_len,
true) / pos_scale;
3405 bstr->
GetInt(base + 6 + lon_len, lat_len,
true) /
3408 bstr->
GetInt(base + 6 + lon_len + lat_len + prec_size,
3413 base + 6 + lon_len + lat_len + prec_size + 8, 8) *
3415 sa.orient_deg = bstr->
GetInt(
3416 base + 6 + lon_len + lat_len + prec_size + 8 + 8, 9);
3418 case AIS8_001_22_SHAPE_POLYLINE:
3419 case AIS8_001_22_SHAPE_POLYGON:
3420 for (
int i = 0; i < 4; ++i) {
3421 sa.angles[i] = bstr->
GetInt(base + 6 + i * 20, 10) * 0.5;
3423 bstr->
GetInt(base + 16 + i * 20, 10) * scale_factor;
3427 an.sub_areas.push_back(sa);
3429 ptd->area_notices[an.link_id] = an;
3430 parse_result =
true;
3436 if (bstr->GetBitCount() >= 360) {
3438 if (met_mmsi != 666) ptd->MMSI = met_mmsi;
3441 double lon_tentative = 181.;
3442 double lat_tentative = 91.;
3444 int lon = bstr->
GetInt(57, 25);
3445 int lat = bstr->
GetInt(82, 24);
3447 if (lon & 0x01000000)
3449 lon_tentative = lon / 60000.;
3451 if (lat & 0x00800000)
3453 lat_tentative = lat / 60000.;
3455 ptd->Lon = lon_tentative;
3456 ptd->Lat = lat_tentative;
3459 wxString x = ptd->ShipName;
3460 if (x.Find(
"METEO") == wxNOT_FOUND) {
3462 wxString slat = wxString::Format(
"%0.3f", lat_tentative);
3463 wxString slon = wxString::Format(
"%0.3f", lon_tentative);
3464 slat.ToDouble(&id1);
3465 slon.ToDouble(&id2);
3466 wxString nameID =
"METEO ";
3467 nameID << wxString::Format(
"%0.3f", abs(id1) + abs(id2)).Right(3);
3468 strncpy(ptd->ShipName, nameID, SHIP_NAME_LEN - 1);
3471 ptd->met_data.pos_acc = bstr->
GetInt(106, 1);
3472 ptd->met_data.day = bstr->
GetInt(107, 5);
3473 ptd->met_data.hour = bstr->
GetInt(112, 5);
3474 ptd->met_data.minute = bstr->
GetInt(117, 6);
3475 ptd->met_data.wind_kn = bstr->
GetInt(123, 7);
3476 ptd->met_data.wind_gust_kn = bstr->
GetInt(130, 7);
3477 ptd->met_data.wind_dir = bstr->
GetInt(137, 9);
3478 ptd->met_data.wind_gust_dir = bstr->
GetInt(146, 9);
3480 int tmp = bstr->
GetInt(155, 11);
3481 if (tmp & 0x00000400)
3483 ptd->met_data.air_temp = tmp / 10.;
3484 ptd->met_data.rel_humid = bstr->
GetInt(166, 7);
3485 int dew = bstr->
GetInt(173, 10);
3486 if (dew & 0x00000200)
3488 ptd->met_data.dew_point = dew / 10.;
3493 ptd->met_data.airpress = bstr->
GetInt(183, 9) + 799;
3494 ptd->met_data.airpress_tend = bstr->
GetInt(192, 2);
3496 int horVis = bstr->
GetInt(194, 8);
3497 if (horVis & 0x80u) {
3499 ptd->met_data.hor_vis_GT =
true;
3501 ptd->met_data.hor_vis_GT =
false;
3503 ptd->met_data.hor_vis = horVis / 10.0;
3505 ptd->met_data.water_lev_dev = (bstr->
GetInt(202, 12) / 100.) - 10.;
3506 ptd->met_data.water_lev_trend = bstr->
GetInt(214, 2);
3507 ptd->met_data.current = bstr->
GetInt(216, 8) / 10.;
3508 ptd->met_data.curr_dir = bstr->
GetInt(224, 9);
3509 ptd->met_data.wave_height = bstr->
GetInt(277, 8) / 10.;
3510 ptd->met_data.wave_period = bstr->
GetInt(285, 6);
3511 ptd->met_data.wave_dir = bstr->
GetInt(291, 9);
3512 ptd->met_data.swell_height = bstr->
GetInt(300, 8) / 10;
3513 ptd->met_data.swell_per = bstr->
GetInt(308, 6);
3514 ptd->met_data.swell_dir = bstr->
GetInt(314, 9);
3515 ptd->met_data.seastate = bstr->
GetInt(323, 4);
3517 int wt = bstr->
GetInt(327, 10);
3518 if (wt & 0x00000200)
3520 ptd->met_data.water_temp = wt / 10.;
3522 ptd->met_data.precipitation = bstr->
GetInt(337, 3);
3523 ptd->met_data.salinity = bstr->
GetInt(340, 9) / 10.;
3524 ptd->met_data.ice = bstr->
GetInt(349, 2);
3526 ptd->Class = AIS_METEO;
3530 ptd->b_NoTrack =
true;
3531 ptd->b_show_track =
false;
3532 ptd->b_positionDoubtful =
false;
3533 ptd->b_positionOnceValid =
true;
3534 b_posn_report =
true;
3535 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
3536 ptd->PositionReportTicks = now.GetTicks();
3537 ptd->b_nameValid =
true;
3539 parse_result =
true;
3545 if (dac == 367 && fi == 33) {
3549 const int size = bstr->GetBitCount();
3552 if (met_mmsi != 666) ptd->MMSI = met_mmsi;
3553 const int startbits = 56;
3554 const int slotsize = 112;
3555 const int slots_count = (size - startbits) / slotsize;
3557 for (
int slot = 0; slot < slots_count; slot++) {
3558 slotbit = slot * slotsize;
3559 int type = bstr->
GetInt(slotbit + 57, 4);
3560 ptd->met_data.hour = bstr->
GetInt(slotbit + 66, 5);
3561 ptd->met_data.minute = bstr->
GetInt(slotbit + 71, 6);
3562 int Site_ID = bstr->
GetInt(slotbit + 77, 7);
3565 if (!ptd->b_nameValid) {
3566 wxString nameID =
"METEO Site: ";
3568 strncpy(ptd->ShipName, nameID, SHIP_NAME_LEN - 1);
3569 ptd->b_nameValid =
true;
3573 int lon = bstr->
GetInt(slotbit + 90, 28);
3574 if (lon & 0x08000000)
3576 ptd->Lon = lon / 600000.;
3578 int lat = bstr->
GetInt(slotbit + 118, 27);
3579 if (lat & 0x04000000)
3581 ptd->Lat = lat / 600000.;
3582 ptd->b_positionOnceValid =
true;
3584 }
else if (type == 1) {
3585 bstr->GetStr(slotbit + 84, 84, &ptd->ShipName[0], SHIP_NAME_LEN);
3586 ptd->b_nameValid =
true;
3588 }
else if (type == 2) {
3590 int descr = bstr->
GetInt(slotbit + 116, 3);
3591 if (descr == 1 || descr == 2) {
3592 ptd->met_data.wind_kn = bstr->
GetInt(slotbit + 84, 7);
3593 ptd->met_data.wind_gust_kn = bstr->
GetInt(slotbit + 91, 7);
3594 ptd->met_data.wind_dir = bstr->
GetInt(slotbit + 98, 9);
3595 ptd->met_data.wind_gust_dir = bstr->
GetInt(slotbit + 107, 9);
3598 }
else if (type == 3) {
3600 int descr = bstr->
GetInt(slotbit + 108, 3);
3601 if (descr == 1 || descr == 2) {
3602 int wltype = bstr->
GetInt(slotbit + 84, 1);
3603 int wl = bstr->
GetInt(slotbit + 85, 16);
3604 if (wl & 0x00004000)
3608 ptd->met_data.water_level = wl / 100.;
3610 ptd->met_data.water_lev_dev = wl / 100.;
3612 ptd->met_data.water_lev_trend = bstr->
GetInt(slotbit + 101, 2);
3613 ptd->met_data.vertical_ref = bstr->
GetInt(slotbit + 103, 5);
3615 }
else if (type == 6) {
3616 int readbearing = bstr->
GetInt(slotbit + 84, 9);
3617 int readdistance = bstr->
GetInt(slotbit + 93, 9);
3618 ptd->met_data.current = bstr->
GetInt(slotbit + 102, 8) / 10.0;
3619 ptd->met_data.curr_dir = bstr->
GetInt(slotbit + 110, 9);
3620 int readLevel = bstr->
GetInt(slotbit + 119, 9);
3622 }
else if (type == 7) {
3624 bstr->
GetInt(slotbit + 111, 3);
3625 if (swell_descr == 1 || swell_descr == 2) {
3626 ptd->met_data.swell_height =
3627 bstr->
GetInt(slotbit + 84, 8) / 10.0;
3628 ptd->met_data.swell_per = bstr->
GetInt(slotbit + 92, 6);
3629 ptd->met_data.swell_dir = bstr->
GetInt(slotbit + 98, 9);
3631 ptd->met_data.seastate = bstr->
GetInt(slotbit + 107, 4);
3632 int wt_descr = bstr->
GetInt(slotbit + 131, 3);
3633 if (wt_descr == 1 || wt_descr == 2)
3634 ptd->met_data.water_temp =
3635 bstr->
GetInt(slotbit + 114, 10) / 10. - 10.;
3637 int wawe_descr = bstr->
GetInt(slotbit + 157, 3);
3638 if (wawe_descr == 1 || wawe_descr == 2) {
3639 ptd->met_data.wave_height =
3640 bstr->
GetInt(slotbit + 134, 8) / 10.0;
3641 ptd->met_data.wave_period = bstr->
GetInt(slotbit + 142, 6);
3642 ptd->met_data.wave_dir = bstr->
GetInt(slotbit + 148, 9);
3644 ptd->met_data.salinity = bstr->
GetInt(slotbit + 160, 9 / 10.0);
3646 }
else if (type == 8) {
3647 ptd->met_data.water_temp =
3648 bstr->
GetInt(slotbit + 84, 10) / 10.0 - 10.0;
3649 ptd->met_data.salinity = bstr->
GetInt(slotbit + 120, 9) / 10.0;
3651 }
else if (type == 9) {
3652 int tmp = bstr->
GetInt(slotbit + 84, 11);
3653 if (tmp & 0x00000400)
3655 ptd->met_data.air_temp = tmp / 10.;
3656 int pp, precip = bstr->
GetInt(slotbit + 98, 2);
3667 ptd->met_data.precipitation = pp;
3668 ptd->met_data.hor_vis = bstr->
GetInt(slotbit + 100, 8) / 10.0;
3669 ptd->met_data.dew_point =
3670 bstr->
GetInt(slotbit + 108, 10) / 10.0 - 20.0;
3671 ptd->met_data.airpress = bstr->
GetInt(slotbit + 121, 9) + 799;
3672 ptd->met_data.airpress_tend = bstr->
GetInt(slotbit + 130, 2);
3673 ptd->met_data.salinity = bstr->
GetInt(slotbit + 135, 9) / 10.0;
3675 }
else if (type == 11) {
3677 int descr = bstr->
GetInt(slotbit + 113, 3);
3678 if (descr == 1 || descr == 2) {
3679 ptd->met_data.wind_kn = bstr->
GetInt(slotbit + 84, 7);
3680 ptd->met_data.wind_gust_kn = bstr->
GetInt(slotbit + 91, 7);
3681 ptd->met_data.wind_dir = bstr->
GetInt(slotbit + 98, 9);
3686 if (ptd->b_positionOnceValid) {
3687 ptd->Class = AIS_METEO;
3691 ptd->b_NoTrack =
true;
3692 ptd->b_show_track =
false;
3693 ptd->b_positionDoubtful =
false;
3694 b_posn_report =
true;
3695 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
3696 ptd->PositionReportTicks = now.GetTicks();
3697 ptd->b_nameValid =
true;
3699 parse_result =
true;
3709 char msg_14_text[968];
3710 if (bstr->GetBitCount() > 40) {
3711 int nx = ((bstr->GetBitCount() - 40) / 6) * 6;
3712 int nd = bstr->GetStr(41, nx, msg_14_text, 968);
3714 nd = wxMin(nd, 967);
3715 msg_14_text[nd] = 0;
3716 ptd->MSG_14_text = wxString(msg_14_text, wxConvUTF8);
3718 parse_result =
true;
3736 if (b_posn_report) ptd->b_lost =
false;
3738 if (
true == parse_result) {
3740 if (!ptd->b_active && !ptd->b_positionDoubtful && b_posn_report)
3741 ptd->b_active =
true;
3744 return parse_result;
3747bool AisDecoder::NMEACheckSumOK(
const wxString &str_in) {
3748 unsigned char checksum_value = 0;
3749 int sentence_hex_sum;
3751 wxCharBuffer buf = str_in.ToUTF8();
3752 if (!buf.data())
return false;
3754 char str_ascii[AIS_MAX_MESSAGE_LEN + 1];
3755 strncpy(str_ascii, buf.data(), AIS_MAX_MESSAGE_LEN);
3756 str_ascii[AIS_MAX_MESSAGE_LEN] =
'\0';
3758 int string_length = strlen(str_ascii);
3760 int payload_length = 0;
3761 while ((payload_length < string_length) &&
3762 (str_ascii[payload_length] !=
'*'))
3765 if (payload_length == string_length)
3770 while (index < payload_length) {
3771 checksum_value ^= str_ascii[index];
3775 if (string_length > 4) {
3777 scanstr[0] = str_ascii[payload_length + 1];
3778 scanstr[1] = str_ascii[payload_length + 2];
3780 sscanf(scanstr,
"%2x", &sentence_hex_sum);
3782 if (sentence_hex_sum == checksum_value)
return true;
3788void AisDecoder::UpdateAllCPA(
void) {
3790 for (
const auto &it : GetTargetList()) {
3791 std::shared_ptr<AisTargetData> td = it.second;
3793 if (NULL != td) UpdateOneCPA(td.get());
3797void AisDecoder::UpdateAllTracks(
void) {
3799 for (
const auto &it : GetTargetList()) {
3800 std::shared_ptr<AisTargetData> td = it.second;
3802 if (NULL != td) UpdateOneTrack(td.get());
3808 if (!ptarget->b_positionOnceValid)
return;
3810 if (ptarget->m_ptrack.size() > 0) {
3812 if (fabs(LastTrackpoint.m_lat - ptarget->Lat) > .1 ||
3813 fabs(LastTrackpoint.m_lon - ptarget->Lon) > .1) {
3816 ptarget->m_ptrack.pop_back();
3817 ptarget->b_positionDoubtful =
true;
3824 if ((ptarget->PositionReportTicks - ptarget->LastPositionReportTicks) > 2) {
3827 ptrackpoint.m_lat = ptarget->Lat;
3828 ptrackpoint.m_lon = ptarget->Lon;
3829 ptrackpoint.m_time = wxDateTime::Now().GetTicks();
3831 ptarget->m_ptrack.push_back(ptrackpoint);
3833 if (ptarget->b_PersistTrack || ptarget->b_mPropPersistTrack) {
3835 if (0 == m_persistent_tracks.count(ptarget->MMSI)) {
3837 t->SetName(wxString::Format(
3838 _T(
"AIS %s (%u) %s %s"), ptarget->GetFullName().c_str(),
3839 ptarget->MMSI, wxDateTime::Now().FormatISODate().c_str(),
3840 wxDateTime::Now().FormatISOTime().c_str()));
3841 g_TrackList.push_back(t);
3843 m_persistent_tracks[ptarget->MMSI] = t;
3845 t = m_persistent_tracks[ptarget->MMSI];
3848 vector2D point(ptrackpoint.m_lon, ptrackpoint.m_lat);
3850 t->AddNewPoint(point, wxDateTime(ptrackpoint.m_time).ToUTC());
3853 pSelect->AddSelectableTrackSegment(tp->m_lat, tp->m_lon, tp1->m_lat,
3854 tp1->m_lon, tp, tp1, t);
3864 wxDateTime::Now().GetTicks() - (time_t)(g_AISShowTracks_Mins * 60);
3866 ptarget->m_ptrack.erase(
3867 std::remove_if(ptarget->m_ptrack.begin(), ptarget->m_ptrack.end(),
3869 return track.m_time < test_time;
3871 ptarget->m_ptrack.end());
3876void AisDecoder::DeletePersistentTrack(
Track *track) {
3877 for (std::map<int, Track *>::iterator iterator = m_persistent_tracks.begin();
3878 iterator != m_persistent_tracks.end(); iterator++) {
3879 if (iterator->second == track) {
3880 int mmsi = iterator->first;
3881 m_persistent_tracks.erase(iterator);
3883 if (0 == m_persistent_tracks.count(mmsi)) {
3884 for (
unsigned int i = 0; i < g_MMSI_Props_Array.GetCount(); i++) {
3885 if (mmsi == g_MMSI_Props_Array[i]->MMSI) {
3887 if (props->m_bPersistentTrack) {
3891 std::shared_ptr<AisTargetData> td =
3892 Get_Target_Data_From_MMSI(mmsi);
3894 props->m_bPersistentTrack =
false;
3895 td->b_mPropPersistTrack =
false;
3897 if (!m_callbacks.confirm_stop_track()) {
3898 props->m_bPersistentTrack =
true;
3910void AisDecoder::UpdateAllAlarms(
void) {
3911 m_bGeneralAlert =
false;
3914 for (
const auto &it : GetTargetList()) {
3915 std::shared_ptr<AisTargetData> td = it.second;
3919 if (!m_bGeneralAlert) {
3921 if ((td->CPA < g_CPAWarn_NM) && (td->TCPA > 0) &&
3922 (td->Class != AIS_ATON) && (td->Class != AIS_BASE))
3923 m_bGeneralAlert =
true;
3926 if (g_bAIS_CPA_Alert_Suppress_Moored && (td->SOG <= g_ShowMoored_Kts))
3927 m_bGeneralAlert =
false;
3930 if ((g_bCPAMax) && (td->Range_NM > g_CPAMax_NM))
3931 m_bGeneralAlert =
false;
3934 if ((g_bTCPA_Max) && (td->TCPA > g_TCPA_Max)) m_bGeneralAlert =
false;
3937 if (td->Class == AIS_SART && td->NavStatus == 14)
3938 m_bGeneralAlert =
true;
3941 if ((td->Class == AIS_DSC) &&
3942 ((td->ShipType == 12) || (td->ShipType == 16)))
3943 m_bGeneralAlert =
true;
3946 ais_alert_type this_alarm = AIS_NO_ALERT;
3949 if (td->Class == AIS_SART && td->NavStatus == 14)
3950 this_alarm = AIS_ALERT_SET;
3953 if ((td->Class == AIS_DSC) &&
3954 ((td->ShipType == 12) || (td->ShipType == 16)))
3955 this_alarm = AIS_ALERT_SET;
3957 if (g_bCPAWarn && td->b_active && td->b_positionOnceValid &&
3958 (td->Class != AIS_SART) && (td->Class != AIS_DSC)) {
3961 if ((g_bHideMoored) && (td->SOG <= g_ShowMoored_Kts)) {
3962 td->n_alert_state = AIS_NO_ALERT;
3968 if (g_bAIS_CPA_Alert_Suppress_Moored &&
3969 (td->SOG <= g_ShowMoored_Kts)) {
3970 td->n_alert_state = AIS_NO_ALERT;
3976 if (td->Range_NM > g_CPAMax_NM) {
3977 td->n_alert_state = AIS_NO_ALERT;
3982 if ((td->CPA < g_CPAWarn_NM) && (td->TCPA > 0) &&
3983 (td->Class != AIS_ATON) && (td->Class != AIS_BASE) &&
3984 (td->Class != AIS_METEO)) {
3986 if (td->TCPA < g_TCPA_Max) {
3987 if (td->b_isFollower)
3988 this_alarm = AIS_ALERT_NO_DIALOG_SET;
3990 this_alarm = AIS_ALERT_SET;
3993 if (td->b_isFollower)
3994 this_alarm = AIS_ALERT_NO_DIALOG_SET;
3996 this_alarm = AIS_ALERT_SET;
4004 if (g_bAIS_ACK_Timeout || (td->Class == AIS_SART) ||
4005 ((td->Class == AIS_DSC) &&
4006 ((td->ShipType == 12) || (td->ShipType == 16)))) {
4007 if (td->b_in_ack_timeout) {
4008 wxTimeSpan delta = wxDateTime::Now() - td->m_ack_time;
4009 if (delta.GetMinutes() > g_AckTimeout_Mins)
4010 td->b_in_ack_timeout =
false;
4016 if (td->b_in_ack_timeout) {
4017 if (this_alarm == AIS_NO_ALERT) td->b_in_ack_timeout =
false;
4021 td->n_alert_state = this_alarm;
4027 ptarget->Range_NM = -1.;
4036 DistanceBearingMercator(ptarget->Lat, ptarget->Lon, gLat, gLon, &brg, &dist);
4037 ptarget->Range_NM = dist;
4040 if (dist <= 1e-5) ptarget->Brg = -1.0;
4042 if (!ptarget->b_positionOnceValid || !bGPSValid) {
4043 ptarget->bCPA_Valid =
false;
4047 if (ptarget->Class == AIS_METEO) {
4048 ptarget->bCPA_Valid =
false;
4057 if (ptarget->b_OwnShip) {
4059 ptarget->TCPA = -100;
4060 ptarget->bCPA_Valid =
false;
4064 double cpa_calc_ownship_cog = gCog;
4065 double cpa_calc_target_cog = ptarget->COG;
4068 if (std::isnan(gSog) || (gSog > 102.2)) {
4069 ptarget->bCPA_Valid =
false;
4074 if (std::isnan(gCog) || gCog == 360.0) {
4076 cpa_calc_ownship_cog =
4080 ptarget->bCPA_Valid =
false;
4086 if (ptarget->COG == 360.0) {
4087 if (ptarget->SOG > 102.2) {
4088 ptarget->bCPA_Valid =
false;
4090 }
else if (ptarget->SOG < .01)
4091 cpa_calc_target_cog =
4095 ptarget->bCPA_Valid =
false;
4101 double v0 = gSog * 1852.;
4102 double v1 = ptarget->SOG * 1852.;
4104 if ((v0 < 1e-6) && (v1 < 1e-6)) {
4108 ptarget->bCPA_Valid =
false;
4115 double east1 = (ptarget->Lon - gLon) * 60 * 1852;
4116 double north1 = (ptarget->Lat - gLat) * 60 * 1852;
4118 double east = east1 * (cos(gLat * PI / 180.));
4120 double north = north1;
4123 double cosa = cos((90. - cpa_calc_ownship_cog) * PI / 180.);
4124 double sina = sin((90. - cpa_calc_ownship_cog) * PI / 180.);
4125 double cosb = cos((90. - cpa_calc_target_cog) * PI / 180.);
4126 double sinb = sin((90. - cpa_calc_target_cog) * PI / 180.);
4129 double fc = (v0 * cosa) - (v1 * cosb);
4130 double fs = (v0 * sina) - (v1 * sinb);
4132 double d = (fc * fc) + (fs * fs);
4140 tcpa = ((fc * east) + (fs * north)) / d;
4143 ptarget->TCPA = tcpa * 60.;
4148 double OwnshipLatCPA, OwnshipLonCPA, TargetLatCPA, TargetLonCPA;
4150 ll_gc_ll(gLat, gLon, cpa_calc_ownship_cog, gSog * tcpa, &OwnshipLatCPA,
4152 ll_gc_ll(ptarget->Lat, ptarget->Lon, cpa_calc_target_cog,
4153 ptarget->SOG * tcpa, &TargetLatCPA, &TargetLonCPA);
4156 ptarget->CPA = DistGreatCircle(OwnshipLatCPA, OwnshipLonCPA, TargetLatCPA,
4159 ptarget->bCPA_Valid =
true;
4161 if (ptarget->TCPA < 0) ptarget->bCPA_Valid =
false;
4165void AisDecoder::OnTimerDSC(wxTimerEvent &event) {
4168 if (m_ptentative_dsctarget) {
4169 ProcessDSx(m_dsc_last_string,
true);
4173void AisDecoder::OnTimerAIS(wxTimerEvent &event) {
4178 wxDateTime now = wxDateTime::Now();
4181 std::unordered_map<int, std::shared_ptr<AisTargetData>> ¤t_targets =
4184 auto it = current_targets.begin();
4185 std::vector<int> remove_array;
4187 while (it != current_targets.end()) {
4188 if (it->second == NULL)
4190 current_targets.erase(it);
4195 std::shared_ptr<AisTargetData> xtd = it->second;
4197 int target_posn_age = now.GetTicks() - xtd->PositionReportTicks;
4198 int target_static_age = now.GetTicks() - xtd->StaticReportTicks;
4210 double removelost_Mins = fmax(g_RemoveLost_Mins, g_MarkLost_Mins);
4212 if (g_bInlandEcdis && (xtd->Class != AIS_ARPA)) {
4213 double iECD_LostTimeOut = 0.0;
4217 if (xtd->Class == AIS_CLASS_B) {
4218 if ((xtd->NavStatus == MOORED) || (xtd->NavStatus == AT_ANCHOR))
4219 iECD_LostTimeOut = 18 * 60;
4221 iECD_LostTimeOut = 180;
4223 if (xtd->Class == AIS_CLASS_A) {
4224 if ((xtd->NavStatus == MOORED) || (xtd->NavStatus == AT_ANCHOR)) {
4226 iECD_LostTimeOut = 18 * 60;
4228 iECD_LostTimeOut = 60;
4230 iECD_LostTimeOut = 60;
4233 if ((target_posn_age > iECD_LostTimeOut) &&
4234 (xtd->Class != AIS_GPSG_BUDDY))
4235 xtd->b_active =
false;
4237 removelost_Mins = (2 * iECD_LostTimeOut) / 60.;
4238 }
else if (g_bMarkLost) {
4239 if ((target_posn_age > g_MarkLost_Mins * 60) &&
4240 (xtd->Class != AIS_GPSG_BUDDY))
4241 xtd->b_active =
false;
4244 if (xtd->Class == AIS_SART || xtd->Class == AIS_METEO)
4245 removelost_Mins = 18.0;
4249 if (g_bRemoveLost || g_bInlandEcdis) {
4251 (xtd->Class == AIS_ARPA &&
4253 if (((target_posn_age > removelost_Mins * 60) &&
4254 (xtd->Class != AIS_GPSG_BUDDY)) ||
4259 xtd->b_positionOnceValid =
false;
4267 long mmsi_long = xtd->MMSI;
4268 pSelectAIS->DeleteSelectablePoint((
void *)mmsi_long, SELTYPE_AISTARGET);
4273 if (target_static_age > removelost_Mins * 60 * 3 || b_arpalost) {
4274 xtd->b_removed =
true;
4276 remove_array.push_back(xtd->MMSI);
4283 for (
unsigned int i = 0; i < g_MMSI_Props_Array.GetCount(); i++) {
4285 if (xtd->MMSI == props->MMSI) {
4286 if (props->m_bignore) {
4287 remove_array.push_back(xtd->MMSI);
4288 xtd->b_removed =
true;
4296 if (xtd->MMSI == g_OwnShipmmsi) {
4297 remove_array.push_back(xtd->MMSI);
4298 xtd->b_removed =
true;
4306 for (
unsigned int i = 0; i < remove_array.size(); i++) {
4307 auto itd = current_targets.find(remove_array[i]);
4308 if (itd != current_targets.end()) {
4309 std::shared_ptr<AisTargetData> td = itd->second;
4310 current_targets.erase(itd);
4319 m_bSuppressed =
false;
4320 if (g_bAIS_CPA_Alert_Suppress_Moored || g_bHideMoored ||
4321 (g_bShowScaled && g_bAllowShowScaled))
4322 m_bSuppressed =
true;
4324 m_bAIS_Audio_Alert_On =
false;
4333 std::shared_ptr<AisTargetData> palert_target = NULL;
4334 int audioType = AISAUDIO_NONE;
4336 if (!g_pais_alert_dialog_active) {
4337 pAISMOBRoute = NULL;
4338 double tcpa_min = 1e6;
4339 double sart_range = 1e6;
4340 std::shared_ptr<AisTargetData> palert_target_cpa = NULL;
4341 std::shared_ptr<AisTargetData> palert_target_sart = NULL;
4342 std::shared_ptr<AisTargetData> palert_target_dsc = NULL;
4344 for (it = current_targets.begin(); it != current_targets.end(); ++it) {
4345 std::shared_ptr<AisTargetData> td = it->second;
4347 if ((td->Class != AIS_SART) && (td->Class != AIS_DSC)) {
4348 if (g_bAIS_CPA_Alert && td->b_active) {
4349 if ((AIS_ALERT_SET == td->n_alert_state) && !td->b_in_ack_timeout) {
4350 if (td->TCPA < tcpa_min) {
4351 tcpa_min = td->TCPA;
4352 palert_target_cpa = td;
4356 }
else if ((td->Class == AIS_DSC) &&
4357 ((td->ShipType == 12) || (td->ShipType == 16))) {
4359 if ((AIS_ALERT_SET == td->n_alert_state) && !td->b_in_ack_timeout) {
4360 palert_target_dsc = td;
4363 td->b_isDSCtarget =
false;
4368 else if (td->Class == AIS_SART) {
4370 if ((AIS_ALERT_SET == td->n_alert_state) && !td->b_in_ack_timeout) {
4371 if (td->Range_NM < sart_range) {
4372 tcpa_min = sart_range;
4373 palert_target_sart = td;
4383 palert_target = palert_target_cpa;
4384 if (palert_target) audioType = AISAUDIO_CPA;
4386 if (palert_target_sart) {
4387 palert_target = palert_target_sart;
4388 audioType = AISAUDIO_SART;
4391 if (palert_target_dsc) {
4392 palert_target = palert_target_dsc;
4393 audioType = AISAUDIO_DSC;
4397 palert_target = Get_Target_Data_From_MMSI(m_callbacks.get_target_mmsi());
4402 TimerAIS.Start(TIMER_AIS_MSEC, wxTIMER_CONTINUOUS);
4405std::shared_ptr<AisTargetData> AisDecoder::Get_Target_Data_From_MMSI(
int mmsi) {
4406 if (AISTargetList.find(mmsi) == AISTargetList.end())
4409 return AISTargetList[mmsi];
4412ArrayOfMmsiProperties g_MMSI_Props_Array;
4416MmsiProperties::MmsiProperties(wxString &spec) {
4418 wxStringTokenizer tkz(spec, _T(
";"));
4421 s = tkz.GetNextToken();
4426 s = tkz.GetNextToken();
4428 if (s.Upper() == _T(
"ALWAYS"))
4429 TrackType = TRACKTYPE_ALWAYS;
4430 else if (s.Upper() == _T(
"NEVER"))
4431 TrackType = TRACKTYPE_NEVER;
4434 s = tkz.GetNextToken();
4436 if (s.Upper() == _T(
"IGNORE")) m_bignore =
true;
4439 s = tkz.GetNextToken();
4441 if (s.Upper() == _T(
"MOB")) m_bMOB =
true;
4444 s = tkz.GetNextToken();
4446 if (s.Upper() == _T(
"VDM")) m_bVDM =
true;
4449 s = tkz.GetNextToken();
4451 if (s.Upper() == _T(
"FOLLOWER")) m_bFollower =
true;
4454 s = tkz.GetNextToken();
4456 if (s.Upper() == _T(
"PERSIST")) m_bPersistentTrack =
true;
4459 s = tkz.GetNextToken();
4461 m_ShipName = s.Upper();
4465MmsiProperties::~MmsiProperties() {}
4467void MmsiProperties::Init(
void) {
4469 TrackType = TRACKTYPE_DEFAULT;
4473 m_bFollower =
false;
4474 m_bPersistentTrack =
false;
4475 m_ShipName = wxEmptyString;
4478wxString MmsiProperties::Serialize(
void) {
4482 sMMSI.Printf(_T(
"%d"), MMSI);
4486 if (TRACKTYPE_ALWAYS == TrackType)
4487 sMMSI << _T(
"always");
4488 else if (TRACKTYPE_NEVER == TrackType)
4489 sMMSI << _T(
"never");
4494 sMMSI << _T(
"ignore");
4509 sMMSI << _T(
"Follower");
4513 if (m_bPersistentTrack) {
4514 sMMSI << _T(
"PERSIST");
4518 if (m_ShipName == wxEmptyString) {
4519 m_ShipName = GetShipNameFromFile(MMSI);
4521 sMMSI << m_ShipName;
4526 AIS_Target_Name_Hash *AISTargetNamesC,
4527 AIS_Target_Name_Hash *AISTargetNamesNC,
long mmsi) {
4528 if (g_benableAISNameCache) {
4529 wxString ship_name = wxEmptyString;
4532 if (!pTargetData->b_nameValid) {
4533 AIS_Target_Name_Hash::iterator it = AISTargetNamesC->find(mmsi);
4534 if (it != AISTargetNamesC->end()) {
4535 ship_name = (*AISTargetNamesC)[mmsi].Left(20);
4536 strncpy(pTargetData->ShipName, ship_name.mb_str(),
4537 ship_name.length() + 1);
4538 pTargetData->b_nameValid =
true;
4539 pTargetData->b_nameFromCache =
true;
4540 }
else if (!g_bUseOnlyConfirmedAISName) {
4541 it = AISTargetNamesNC->find(mmsi);
4542 if (it != AISTargetNamesNC->end()) {
4543 ship_name = (*AISTargetNamesNC)[mmsi].Left(20);
4544 strncpy(pTargetData->ShipName, ship_name.mb_str(),
4545 ship_name.length() + 1);
4546 pTargetData->b_nameValid =
true;
4547 pTargetData->b_nameFromCache =
true;
4552 else if ((pTargetData->MID == 5) || (pTargetData->MID == 24) ||
4553 (pTargetData->MID == 19) ||
4554 (pTargetData->MID == 123) ||
4555 (pTargetData->MID == 124)) {
4557 pTargetData->b_nameFromCache =
false;
4558 ship_name = trimAISField(pTargetData->ShipName);
4559 AIS_Target_Name_Hash::iterator itC = AISTargetNamesC->find(mmsi);
4560 AIS_Target_Name_Hash::iterator itNC = AISTargetNamesNC->find(mmsi);
4562 AISTargetNamesC->end()) {
4563 if ((*AISTargetNamesC)[mmsi] ==
4565 if (itNC != AISTargetNamesNC->end()) {
4567 AISTargetNamesNC->erase(itNC);
4570 if (itNC != AISTargetNamesNC->end()) {
4572 if ((*AISTargetNamesNC)[mmsi] ==
4575 (*AISTargetNamesC)[mmsi] = ship_name;
4577 AISTargetNamesNC->erase(itNC);
4580 (*AISTargetNamesNC)[mmsi] = ship_name;
4582 if (g_bUseOnlyConfirmedAISName)
4583 strncpy(pTargetData->ShipName, (*AISTargetNamesC)[mmsi].mb_str(),
4584 (*AISTargetNamesC)[mmsi].Left(20).Length() + 1);
4586 (*AISTargetNamesNC)[mmsi] = ship_name;
4591 AISTargetNamesNC->end()) {
4592 if ((*AISTargetNamesNC)[mmsi] ==
4595 (*AISTargetNamesC)[mmsi] = ship_name;
4597 AISTargetNamesNC->erase(itNC);
4599 (*AISTargetNamesNC)[mmsi] = ship_name;
4602 (*AISTargetNamesNC)[mmsi] = ship_name;
4604 if (g_bUseOnlyConfirmedAISName) {
4605 pTargetData->ShipName[SHIP_NAME_LEN - 1] =
'\0';
4606 strncpy(pTargetData->ShipName,
"Unknown ",
4614wxString GetShipNameFromFile(
int nmmsi) {
4615 wxString name = wxEmptyString;
4616 if (g_benableAISNameCache) {
4617 std::ifstream infile(AISTargetNameFileName.mb_str());
4620 while (getline(infile, line)) {
4621 wxStringTokenizer tokenizer(wxString::FromUTF8(line.c_str()), _T(
","));
4622 if (nmmsi == wxAtoi(tokenizer.GetNextToken())) {
4623 name = tokenizer.GetNextToken().Trim();
4626 tokenizer.GetNextToken();
4635int AisMeteoNewMmsi(
int orig_mmsi,
int m_lat,
int m_lon,
int lon_bits = 0,
4639 if ((!lon_bits || lon_bits == 999) && siteID) {
4642 auto &points = AisMeteoPoints::GetInstance().GetPoints();
4643 if (points.size()) {
4644 for (
const auto &point : points) {
4646 if (siteID == point.siteID && orig_mmsi == point.orig_mmsi) {
4648 new_mmsi = point.mmsi;
4654 if (!found && !lon_bits) {
4659 double lon_tentative = 181.;
4660 double lat_tentative = 91.;
4662 if (lon_bits == 25) {
4663 if (m_lon & 0x01000000)
4664 m_lon |= 0xFE000000;
4665 lon_tentative = m_lon / 60000.;
4667 if (m_lat & 0x00800000)
4668 m_lat |= 0xFF000000;
4669 lat_tentative = m_lat / 60000.;
4671 }
else if (lon_bits == 28) {
4672 if (m_lon & 0x08000000)
4673 m_lon |= 0xf0000000;
4674 lon_tentative = m_lon / 600000.;
4676 if (m_lat & 0x04000000)
4677 m_lat |= 0xf8000000;
4678 lat_tentative = m_lat / 600000.;
4683 wxString slon = wxString::Format(
"%0.3f", lon_tentative);
4684 wxString slat = wxString::Format(
"%0.3f", lat_tentative);
4692 static int nextMeteommsi = 199400000;
4693 auto &points = AisMeteoPoints::GetInstance().GetPoints();
4695 if (lon_bits != 999 && points.size()) {
4696 wxString t_lat, t_lon;
4697 for (
const auto &point : points) {
4699 if (slat.IsSameAs(point.lat) && slon.IsSameAs(point.lon)) {
4701 new_mmsi = point.mmsi;
4711 AisMeteoPoint(nextMeteommsi, slat, slon, siteID, orig_mmsi));
4712 new_mmsi = nextMeteommsi;
4717bool isBuoyMmsi(
const int msi) {
4722 int mid = msi / 1000000;
4723 if ((mid > 200 && mid < 800) || mid >= 970) {
Global state for AIS decoder.
int GetInt(int sp, int len, bool signed_flag=false)
sp is starting bit, 1-based
EventVar plugin_msg
A JSON message should be sent.
EventVar new_track
Notified on new track creation.
EventVar info_update
Notified when AIS user dialogs should update.
EventVar touch_state
Notified when gFrame->TouchAISActive() should be invoked.
Add a new point to the list of Meteo stations.
const void Notify()
Notify all listeners, no data supplied.
void LogInputMessage(const wxString &msg, const wxString &stream_name, bool b_filter, bool b_error=false, const wxString error_msg=wxEmptyString)
Logs an input message with context information.
A regular Nmea0183 message.
See: https://github.com/OpenCPN/OpenCPN/issues/2729#issuecomment-1179506343.
void Listen(const std::string &key, wxEvtHandler *listener, wxEventType evt)
Set object to send wxEventType ev to listener on changes in key.
Custom event class for OpenCPN's notification system.
std::shared_ptr< const void > GetSharedPtr() const
Gets the event's payload data.
Represents a waypoint or mark within the navigation system.
bool m_bIsolatedMark
Flag indicating if the waypoint is a standalone mark.
A parsed SignalK message over ipv4.
Represents a single point in a track.
Represents a track, which is a series of connected track points.
The JSON document writer.
void Write(const wxJSONValue &value, wxString &str)
Write the JSONvalue object to a JSON text.
Raw messages layer, supports sending and recieving navmsg messages.
A generic position and navigation data structure.
double kCog
Course over ground in degrees.
double kHdt
True heading in degrees.
double kHdm
Magnetic heading in degrees.
double kLat
Latitude in decimal degrees.
double kSog
Speed over ground in knots.
double kVar
Magnetic variation in degrees.
double kLon
Longitude in decimal degrees.