2023-06-12 05:21:52 +02:00
|
|
|
#include <TinyGPS++.h>
|
2023-07-30 23:12:50 +02:00
|
|
|
#include <WiFi.h>
|
2023-06-06 18:19:56 +02:00
|
|
|
#include "configuration.h"
|
2023-06-08 06:58:10 +02:00
|
|
|
#include "gps_utils.h"
|
2023-06-06 18:19:56 +02:00
|
|
|
|
|
|
|
|
extern Configuration Config;
|
2023-07-30 23:12:50 +02:00
|
|
|
extern WiFiClient espClient;
|
2023-06-20 01:44:55 +02:00
|
|
|
String distance;
|
2023-06-06 18:19:56 +02:00
|
|
|
|
2024-02-25 16:00:44 +01:00
|
|
|
|
2023-06-06 18:19:56 +02:00
|
|
|
namespace GPS_Utils {
|
|
|
|
|
|
2024-04-09 17:08:23 +02:00
|
|
|
char *ax25_base91enc(char *s, uint8_t n, uint32_t v) {
|
|
|
|
|
for(s += n, *s = '\0'; n; n--) {
|
|
|
|
|
*(--s) = v % 91 + 33;
|
|
|
|
|
v /= 91;
|
2024-02-24 14:09:05 +01:00
|
|
|
}
|
2024-04-09 17:08:23 +02:00
|
|
|
return(s);
|
2024-02-24 14:09:05 +01:00
|
|
|
}
|
2023-06-06 18:19:56 +02:00
|
|
|
|
2024-04-09 17:08:23 +02:00
|
|
|
String encodeGPS(float latitude, float longitude, String overlay, String symbol) {
|
|
|
|
|
String encodedData = overlay;
|
|
|
|
|
uint32_t aprs_lat, aprs_lon;
|
|
|
|
|
aprs_lat = 900000000 - latitude * 10000000;
|
|
|
|
|
aprs_lat = aprs_lat / 26 - aprs_lat / 2710 + aprs_lat / 15384615;
|
|
|
|
|
aprs_lon = 900000000 + longitude * 10000000 / 2;
|
|
|
|
|
aprs_lon = aprs_lon / 26 - aprs_lon / 2710 + aprs_lon / 15384615;
|
|
|
|
|
|
|
|
|
|
String Ns, Ew, helper;
|
|
|
|
|
if(latitude < 0) { Ns = "S"; } else { Ns = "N"; }
|
|
|
|
|
if(latitude < 0) { latitude= -latitude; }
|
|
|
|
|
|
|
|
|
|
if(longitude < 0) { Ew = "W"; } else { Ew = "E"; }
|
|
|
|
|
if(longitude < 0) { longitude= -longitude; }
|
|
|
|
|
|
|
|
|
|
char helper_base91[] = {"0000\0"};
|
|
|
|
|
int i;
|
|
|
|
|
ax25_base91enc(helper_base91, 4, aprs_lat);
|
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
|
encodedData += helper_base91[i];
|
2024-02-24 14:09:05 +01:00
|
|
|
}
|
2024-04-09 17:08:23 +02:00
|
|
|
ax25_base91enc(helper_base91, 4, aprs_lon);
|
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
|
encodedData += helper_base91[i];
|
2024-02-24 14:09:05 +01:00
|
|
|
}
|
2024-04-09 17:08:23 +02:00
|
|
|
encodedData += symbol + " x" + "\x47";
|
|
|
|
|
return encodedData;
|
2023-09-21 07:08:10 +02:00
|
|
|
}
|
2024-02-24 14:09:05 +01:00
|
|
|
|
|
|
|
|
String generateBeacon() {
|
2024-03-07 17:46:38 +01:00
|
|
|
String beaconPacket = Config.callsign + ">APLRG1," + Config.beacon.path;
|
|
|
|
|
|
|
|
|
|
if (Config.aprs_is.active && Config.digi.mode == 0) { // If APRSIS enabled and Digi disabled
|
|
|
|
|
beaconPacket += ",qAC";
|
2024-04-09 17:08:23 +02:00
|
|
|
}
|
|
|
|
|
return beaconPacket + ":!" + encodeGPS(Config.beacon.latitude, Config.beacon.longitude, Config.beacon.overlay, Config.beacon.symbol);;
|
2023-06-09 01:31:27 +02:00
|
|
|
}
|
2024-01-03 02:12:10 +01:00
|
|
|
|
2024-02-24 14:09:05 +01:00
|
|
|
String generateiGateLoRaBeacon() {
|
2024-04-09 17:08:23 +02:00
|
|
|
return Config.callsign + ">APLRG1," + Config.beacon.path + ":!" + encodeGPS(Config.beacon.latitude, Config.beacon.longitude, Config.beacon.overlay, Config.beacon.symbol);
|
2023-07-31 05:53:59 +02:00
|
|
|
}
|
2023-06-06 18:19:56 +02:00
|
|
|
|
2024-02-24 14:09:05 +01:00
|
|
|
double calculateDistanceTo(double latitude, double longitude) {
|
2024-03-07 17:46:38 +01:00
|
|
|
return TinyGPSPlus::distanceBetween(Config.beacon.latitude,Config.beacon.longitude, latitude, longitude) / 1000.0;
|
2024-01-12 03:22:58 +01:00
|
|
|
}
|
2023-06-12 05:21:52 +02:00
|
|
|
|
2024-02-24 14:09:05 +01:00
|
|
|
String decodeEncodedGPS(String packet) {
|
|
|
|
|
String GPSPacket = packet.substring(packet.indexOf(":!")+3);
|
|
|
|
|
String encodedLatitude = GPSPacket.substring(0,4);
|
|
|
|
|
String encodedLongtitude = GPSPacket.substring(4,8);
|
2023-06-12 05:21:52 +02:00
|
|
|
|
2024-02-24 14:09:05 +01:00
|
|
|
int Y1 = int(encodedLatitude[0]);
|
|
|
|
|
int Y2 = int(encodedLatitude[1]);
|
|
|
|
|
int Y3 = int(encodedLatitude[2]);
|
|
|
|
|
int Y4 = int(encodedLatitude[3]);
|
|
|
|
|
float decodedLatitude = 90.0 - ((((Y1-33) * pow(91,3)) + ((Y2-33) * pow(91,2)) + ((Y3-33) * 91) + Y4-33) / 380926.0);
|
|
|
|
|
|
|
|
|
|
int X1 = int(encodedLongtitude[0]);
|
|
|
|
|
int X2 = int(encodedLongtitude[1]);
|
|
|
|
|
int X3 = int(encodedLongtitude[2]);
|
|
|
|
|
int X4 = int(encodedLongtitude[3]);
|
|
|
|
|
float decodedLongitude = -180.0 + ((((X1-33) * pow(91,3)) + ((X2-33) * pow(91,2)) + ((X3-33) * 91) + X4-33) / 190463.0);
|
|
|
|
|
distance = String(calculateDistanceTo(decodedLatitude, decodedLongitude),1);
|
|
|
|
|
return String(decodedLatitude,5) + "N / " + String(decodedLongitude,5) + "E / " + distance + "km";
|
2024-01-03 02:12:10 +01:00
|
|
|
}
|
2023-06-12 05:21:52 +02:00
|
|
|
|
2024-02-24 14:09:05 +01:00
|
|
|
String getReceivedGPS(String packet) {
|
|
|
|
|
String infoGPS;
|
|
|
|
|
if (packet.indexOf(":!") > 10) {
|
|
|
|
|
infoGPS = packet.substring(packet.indexOf(":!")+2);
|
|
|
|
|
} else if (packet.indexOf(":=") > 10) {
|
|
|
|
|
infoGPS = packet.substring(packet.indexOf(":=")+2);
|
|
|
|
|
}
|
|
|
|
|
String Latitude = infoGPS.substring(0,8);
|
|
|
|
|
String Longitude = infoGPS.substring(9,18);
|
2024-01-03 02:12:10 +01:00
|
|
|
|
2024-02-24 14:09:05 +01:00
|
|
|
float convertedLatitude, convertedLongitude;
|
|
|
|
|
String firstLatPart = Latitude.substring(0,2);
|
|
|
|
|
String secondLatPart = Latitude.substring(2,4);
|
|
|
|
|
String thirdLatPart = Latitude.substring(Latitude.indexOf(".")+1,Latitude.indexOf(".")+3);
|
|
|
|
|
String firstLngPart = Longitude.substring(0,3);
|
|
|
|
|
String secondLngPart = Longitude.substring(3,5);
|
|
|
|
|
String thirdLngPart = Longitude.substring(Longitude.indexOf(".")+1,Longitude.indexOf(".")+3);
|
|
|
|
|
convertedLatitude = firstLatPart.toFloat() + (secondLatPart.toFloat()/60) + (thirdLatPart.toFloat()/(60*100));
|
|
|
|
|
convertedLongitude = firstLngPart.toFloat() + (secondLngPart.toFloat()/60) + (thirdLngPart.toFloat()/(60*100));
|
|
|
|
|
|
|
|
|
|
String LatSign = String(Latitude[7]);
|
|
|
|
|
String LngSign = String(Longitude[8]);
|
|
|
|
|
if (LatSign == "S") {
|
|
|
|
|
convertedLatitude = -convertedLatitude;
|
|
|
|
|
}
|
|
|
|
|
if (LngSign == "W") {
|
|
|
|
|
convertedLongitude = -convertedLongitude;
|
|
|
|
|
}
|
|
|
|
|
distance = String(calculateDistanceTo(convertedLatitude, convertedLongitude),1);
|
|
|
|
|
return String(convertedLatitude,5) + "N / " + String(convertedLongitude,5) + "E / " + distance + "km";
|
2024-01-03 02:12:10 +01:00
|
|
|
}
|
2024-02-24 14:09:05 +01:00
|
|
|
|
|
|
|
|
String getDistance(String packet) {
|
2024-03-27 19:28:27 +01:00
|
|
|
uint8_t encodedBytePosition = 0;
|
2024-02-24 14:09:05 +01:00
|
|
|
if (packet.indexOf(":!") > 10) {
|
|
|
|
|
encodedBytePosition = packet.indexOf(":!") + 14;
|
|
|
|
|
}
|
|
|
|
|
if (packet.indexOf(":=") > 10) {
|
|
|
|
|
encodedBytePosition = packet.indexOf(":=") + 14;
|
|
|
|
|
}
|
|
|
|
|
if (encodedBytePosition != 0) {
|
|
|
|
|
if (String(packet[encodedBytePosition]) == "G" || String(packet[encodedBytePosition]) == "Q" || String(packet[encodedBytePosition]) == "[" || String(packet[encodedBytePosition]) == "H") {
|
|
|
|
|
return decodeEncodedGPS(packet);
|
|
|
|
|
} else {
|
|
|
|
|
return getReceivedGPS(packet);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
return " _ / _ / _ ";
|
|
|
|
|
}
|
2023-08-15 17:04:54 +02:00
|
|
|
}
|
2023-06-12 05:21:52 +02:00
|
|
|
|
2023-06-06 18:19:56 +02:00
|
|
|
}
|