31#include "model/autopilot_output.h"
32#include "model/comm_drv_n2k_serial.h"
34#include "model/comm_n0183_output.h"
35#include "model/comm_vars.h"
36#include "model/config_vars.h"
37#include "model/georef.h"
38#include "model/nmea_ctx_factory.h"
39#include "model/own_ship.h"
40#include "model/routeman.h"
42#include "N2kMessages.h"
44#include "observable_globvar.h"
47#include "androidUTIL.h"
50bool UpdateAutopilotN0183(
Routeman &routeman) {
51 NMEA0183 nmea0183 = routeman.GetNMEA0183();
52 RoutePoint *pActivePoint = routeman.GetpActivePoint();
56 if ((g_maxWPNameLength >= 3) && (g_maxWPNameLength <= 32))
57 maxName = g_maxWPNameLength;
61 double r_Sog(0.0), r_Cog(0.0);
62 if (!std::isnan(gSog)) r_Sog = gSog;
63 if (!std::isnan(gCog)) r_Cog = gCog;
67 nmea0183.TalkerID =
"EC";
69 nmea0183.Rmb.IsDataValid = bGPSValid ? NTrue : NFalse;
70 nmea0183.Rmb.CrossTrackError = routeman.GetCurrentXTEToActivePoint();
71 nmea0183.Rmb.DirectionToSteer = routeman.GetXTEDir() < 0 ? Left : Right;
72 nmea0183.Rmb.RangeToDestinationNauticalMiles =
73 routeman.GetCurrentRngToActivePoint();
74 nmea0183.Rmb.BearingToDestinationDegreesTrue =
75 routeman.GetCurrentBrgToActivePoint();
77 if (pActivePoint->m_lat < 0.)
78 nmea0183.Rmb.DestinationPosition.Latitude.Set(-pActivePoint->m_lat,
"S");
80 nmea0183.Rmb.DestinationPosition.Latitude.Set(pActivePoint->m_lat,
"N");
82 if (pActivePoint->m_lon < 0.)
83 nmea0183.Rmb.DestinationPosition.Longitude.Set(-pActivePoint->m_lon,
"W");
85 nmea0183.Rmb.DestinationPosition.Longitude.Set(pActivePoint->m_lon,
"E");
87 nmea0183.Rmb.DestinationClosingVelocityKnots =
89 cos((r_Cog - routeman.GetCurrentBrgToActivePoint()) * PI / 180.0);
90 nmea0183.Rmb.IsArrivalCircleEntered =
91 routeman.GetArrival() ? NTrue : NFalse;
92 nmea0183.Rmb.FAAModeIndicator = bGPSValid ?
"A" :
"N";
97 nmea0183.Rmb.To = pActivePoint->GetName().Truncate(wp_len);
99 routeman.GetpActiveRouteSegmentBeginPoint()->GetName().Truncate(
101 nmea0183.Rmb.Write(snt);
103 }
while (snt.Sentence.size() > 82 && wp_len > 0);
105 BroadcastNMEA0183Message(snt.Sentence, routeman.GetNmeaLog(),
106 routeman.GetMessageSentEventVar());
111 nmea0183.TalkerID = _T(
"EC");
114 nmea0183.Rmc.IsDataValid = NTrue;
115 if (!bGPSValid) nmea0183.Rmc.IsDataValid = NFalse;
118 nmea0183.Rmc.Position.Latitude.Set(-gLat, _T(
"S"));
120 nmea0183.Rmc.Position.Latitude.Set(gLat, _T(
"N"));
123 nmea0183.Rmc.Position.Longitude.Set(-gLon, _T(
"W"));
125 nmea0183.Rmc.Position.Longitude.Set(gLon, _T(
"E"));
127 nmea0183.Rmc.SpeedOverGroundKnots = r_Sog;
128 nmea0183.Rmc.TrackMadeGoodDegreesTrue = r_Cog;
130 if (!std::isnan(gVar)) {
132 nmea0183.Rmc.MagneticVariation = -gVar;
133 nmea0183.Rmc.MagneticVariationDirection = West;
135 nmea0183.Rmc.MagneticVariation = gVar;
136 nmea0183.Rmc.MagneticVariationDirection = East;
139 nmea0183.Rmc.MagneticVariation =
143 if (!gRmcTime.IsEmpty() && !gRmcDate.IsEmpty()) {
144 nmea0183.Rmc.UTCTime = gRmcTime;
145 nmea0183.Rmc.Date = gRmcDate;
147 wxDateTime now = wxDateTime::Now();
148 wxDateTime utc = now.ToUTC();
149 wxString time = utc.Format(_T(
"%H%M%S"));
150 nmea0183.Rmc.UTCTime = time;
151 wxString date = utc.Format(_T(
"%d%m%y"));
152 nmea0183.Rmc.Date = date;
155 nmea0183.Rmc.FAAModeIndicator =
"A";
156 if (!bGPSValid) nmea0183.Rmc.FAAModeIndicator =
"N";
158 nmea0183.Rmc.Write(snt);
160 BroadcastNMEA0183Message(snt.Sentence, routeman.GetNmeaLog(),
161 routeman.GetMessageSentEventVar());
166 nmea0183.TalkerID = _T(
"EC");
170 nmea0183.Apb.IsLoranBlinkOK =
172 if (!bGPSValid) nmea0183.Apb.IsLoranBlinkOK = NFalse;
174 nmea0183.Apb.IsLoranCCycleLockOK = NTrue;
175 if (!bGPSValid) nmea0183.Apb.IsLoranCCycleLockOK = NFalse;
177 nmea0183.Apb.CrossTrackErrorMagnitude =
178 routeman.GetCurrentXTEToActivePoint();
180 if (routeman.GetXTEDir() < 0)
181 nmea0183.Apb.DirectionToSteer = Left;
183 nmea0183.Apb.DirectionToSteer = Right;
185 nmea0183.Apb.CrossTrackUnits = _T(
"N");
187 if (routeman.GetArrival())
188 nmea0183.Apb.IsArrivalCircleEntered = NTrue;
190 nmea0183.Apb.IsArrivalCircleEntered = NFalse;
194 nmea0183.Apb.IsPerpendicular = NFalse;
196 nmea0183.Apb.To = pActivePoint->GetName().Truncate(maxName);
199 DistanceBearingMercator(pActivePoint->m_lat, pActivePoint->m_lon,
200 routeman.GetpActiveRouteSegmentBeginPoint()->m_lat,
201 routeman.GetpActiveRouteSegmentBeginPoint()->m_lon,
204 if (g_bMagneticAPB && !std::isnan(gVar)) {
206 ((brg1 - gVar) >= 0.) ? (brg1 - gVar) : (brg1 - gVar + 360.);
207 double bapm = ((routeman.GetCurrentBrgToActivePoint() - gVar) >= 0.)
208 ? (routeman.GetCurrentBrgToActivePoint() - gVar)
209 : (routeman.GetCurrentBrgToActivePoint() - gVar + 360.);
211 nmea0183.Apb.BearingOriginToDestination = brg1m;
212 nmea0183.Apb.BearingOriginToDestinationUnits = _T(
"M");
214 nmea0183.Apb.BearingPresentPositionToDestination = bapm;
215 nmea0183.Apb.BearingPresentPositionToDestinationUnits = _T(
"M");
217 nmea0183.Apb.HeadingToSteer = bapm;
218 nmea0183.Apb.HeadingToSteerUnits = _T(
"M");
220 nmea0183.Apb.BearingOriginToDestination = brg1;
221 nmea0183.Apb.BearingOriginToDestinationUnits = _T(
"T");
223 nmea0183.Apb.BearingPresentPositionToDestination =
224 routeman.GetCurrentBrgToActivePoint();
225 nmea0183.Apb.BearingPresentPositionToDestinationUnits = _T(
"T");
227 nmea0183.Apb.HeadingToSteer = routeman.GetCurrentBrgToActivePoint();
228 nmea0183.Apb.HeadingToSteerUnits = _T(
"T");
231 nmea0183.Apb.Write(snt);
232 BroadcastNMEA0183Message(snt.Sentence, routeman.GetNmeaLog(),
233 routeman.GetMessageSentEventVar());
238 nmea0183.TalkerID = _T(
"EC");
242 nmea0183.Xte.IsLoranBlinkOK =
244 if (!bGPSValid) nmea0183.Xte.IsLoranBlinkOK = NFalse;
246 nmea0183.Xte.IsLoranCCycleLockOK = NTrue;
247 if (!bGPSValid) nmea0183.Xte.IsLoranCCycleLockOK = NFalse;
249 nmea0183.Xte.CrossTrackErrorDistance =
250 routeman.GetCurrentXTEToActivePoint();
252 if (routeman.GetXTEDir() < 0)
253 nmea0183.Xte.DirectionToSteer = Left;
255 nmea0183.Xte.DirectionToSteer = Right;
257 nmea0183.Xte.CrossTrackUnits = _T(
"N");
259 nmea0183.Xte.Write(snt);
260 BroadcastNMEA0183Message(snt.Sentence, routeman.GetNmeaLog(),
261 routeman.GetMessageSentEventVar());
267bool UpdateAutopilotN2K(
Routeman &routeman) {
268 bool fail_any =
false;
271 auto ®istry = CommDriverRegistry::GetInstance();
272 const std::vector<DriverPtr> &drivers = registry.GetDrivers();
275 for (
auto key : routeman.GetOutpuDriverArray()) {
276 for (
auto &d : drivers) {
277 if (d->Key() == key) {
278 std::unordered_map<std::string, std::string> attributes =
280 auto protocol_it = attributes.find(
"protocol");
281 if (protocol_it != attributes.end()) {
282 std::string protocol = protocol_it->second;
284 if (protocol ==
"nmea2000") {
291 if (!found)
return false;
296 drv_serial->AddTxPGN(129283);
297 drv_serial->AddTxPGN(129284);
298 drv_serial->AddTxPGN(129285);
301 fail_any |= !SendPGN129285(routeman, found);
302 fail_any |= !SendPGN129284(routeman, found);
303 fail_any |= !SendPGN129283(routeman, found);
305 return (fail_any == 0);
309 bool fail_any =
false;
313 if ((g_maxWPNameLength >= 3) && (g_maxWPNameLength <= 32))
314 maxName = g_maxWPNameLength;
319 char route_name[] =
"Route";
320 SetN2kPGN129285(msg129285, 0, 0, 0, N2kdir_forward, 0, route_name);
323 RoutePoint *pLegBeginPoint = routeman.GetpActiveRouteSegmentBeginPoint();
324 wxString start_point_name = pLegBeginPoint->GetName().Truncate(maxName);
325 std::string sname = start_point_name.ToStdString();
326 char *s = (
char *)sname.c_str();
328 fail_any |= !AppendN2kPGN129285(msg129285, 0, s, pLegBeginPoint->m_lat,
329 pLegBeginPoint->m_lon);
331 RoutePoint *pActivePoint = routeman.GetpActivePoint();
332 wxString destination_name = pActivePoint->GetName().Truncate(maxName);
333 std::string dname = destination_name.ToStdString();
334 char *d = (
char *)dname.c_str();
335 fail_any |= !AppendN2kPGN129285(msg129285, 1, d, pActivePoint->m_lat,
336 pActivePoint->m_lon);
338 if (fail_any)
return false;
340 auto dest_addr = std::make_shared<const NavAddr2000>(driver->
iface, 255);
341 std::vector<uint8_t> payload;
342 for (
int i = 0; i < msg129285.DataLen; i++)
343 payload.push_back(msg129285.Data[i]);
345 std::make_shared<const Nmea2000Msg>(129285, payload, dest_addr, 6);
346 fail_any |= !driver->SendMessage(PGN129285, dest_addr);
348 return (fail_any == 0);
352 bool fail_any =
false;
354 RoutePoint *pActivePoint = routeman.GetpActivePoint();
358 if (!std::isnan(gCog) && !std::isnan(gSog)) {
359 double brg = routeman.GetCurrentBrgToActivePoint();
360 vmg = gSog * cos((brg - gCog) * PI / 180.);
362 wxTimeSpan tttg_span;
363 wxDateTime arrival_time = wxDateTime::Now();
365 double tttg_sec = (routeman.GetCurrentRngToActivePoint() / gSog) * 3600;
366 tttg_span = wxTimeSpan::Seconds((
long)tttg_sec);
367 arrival_time += tttg_span;
369 double time_days_1979 = arrival_time.GetTicks() / (3600. * 24.);
373 double eta_time_days;
374 double eta_time_seconds = modf(time_days_1979, &eta_time_days);
375 int16_t eta_time_days_16 =
static_cast<uint16_t
>(eta_time_days);
380 routeman.GetCurrentRngToActivePoint() * 1852.,
387 routeman.GetCurrentSegmentCourse() * PI /
389 routeman.GetCurrentBrgToActivePoint() * PI /
397 auto dest_addr = std::make_shared<const NavAddr2000>(driver->
iface, 255);
398 std::vector<uint8_t> payload;
399 for (
int i = 0; i < msg129284.DataLen; i++)
400 payload.push_back(msg129284.Data[i]);
402 std::make_shared<const Nmea2000Msg>(129284, payload, dest_addr, 6);
403 fail_any |= !driver->SendMessage(PGN129284, dest_addr);
405 return (fail_any == 0);
409 bool fail_any =
false;
411 RoutePoint *pActivePoint = routeman.GetpActivePoint();
413 SetN2kPGN129283(msg129283, 0,
416 routeman.GetCurrentXTEToActivePoint()
419 auto dest_addr = std::make_shared<const NavAddr2000>(driver->
iface, 255);
420 std::vector<uint8_t> payload;
421 for (
int i = 0; i < msg129283.DataLen; i++)
422 payload.push_back(msg129283.Data[i]);
424 std::make_shared<const Nmea2000Msg>(129283, payload, dest_addr, 6);
425 fail_any |= !driver->SendMessage(PGN129283, dest_addr);
427 return (fail_any == 0);
Common interface for all drivers.
const std::string iface
Physical device for 0183, else a unique string.
Represents a waypoint or mark within the navigation system.
Driver registration container, a singleton.
const std::unordered_map< std::string, std::string > GetAttributes(DriverHandle handle)
Query a specific driver for attributes.