mirror of
https://github.com/swift-project/pilotclient.git
synced 2026-03-22 23:05:36 +08:00
579 lines
26 KiB
C++
579 lines
26 KiB
C++
/* Copyright (C) 2013
|
|
* swift project Community / Contributors
|
|
*
|
|
* This file is part of swift project. It is subject to the license terms in the LICENSE file found in the top-level
|
|
* directory of this distribution and at http://www.swift-project.org/license.html. No part of swift project,
|
|
* including this file, may be copied, modified, propagated, or distributed except according to the terms
|
|
* contained in the LICENSE file.
|
|
*/
|
|
|
|
#include "airspace_monitor.h"
|
|
#include "blackcore/blackcorefreefunctions.h"
|
|
#include "blackmisc/project.h"
|
|
#include "blackmisc/testing.h"
|
|
#include "blackmisc/indexvariantmap.h"
|
|
|
|
namespace BlackCore
|
|
{
|
|
|
|
using namespace BlackMisc;
|
|
using namespace BlackMisc::Aviation;
|
|
using namespace BlackMisc::Audio;
|
|
using namespace BlackMisc::Network;
|
|
using namespace BlackMisc::Geo;
|
|
using namespace BlackMisc::PhysicalQuantities;
|
|
|
|
CAirspaceMonitor::CAirspaceMonitor(QObject *parent, INetwork *network, CVatsimBookingReader *bookings, CVatsimDataFileReader *dataFile)
|
|
: QObject(parent), m_network(network), m_vatsimBookingReader(bookings), m_vatsimDataFileReader(dataFile)
|
|
{
|
|
this->connect(this->m_network, &INetwork::atcPositionUpdate, this, &CAirspaceMonitor::ps_atcPositionUpdate);
|
|
this->connect(this->m_network, &INetwork::atisReplyReceived, this, &CAirspaceMonitor::ps_atisReceived);
|
|
this->connect(this->m_network, &INetwork::atisVoiceRoomReplyReceived, this, &CAirspaceMonitor::ps_atisVoiceRoomReceived);
|
|
this->connect(this->m_network, &INetwork::atisLogoffTimeReplyReceived, this, &CAirspaceMonitor::ps_atisLogoffTimeReceived);
|
|
this->connect(this->m_network, &INetwork::metarReplyReceived, this, &CAirspaceMonitor::ps_metarReceived);
|
|
this->connect(this->m_network, &INetwork::flightPlanReplyReceived, this, &CAirspaceMonitor::ps_flightPlanReceived);
|
|
this->connect(this->m_network, &INetwork::realNameReplyReceived, this, &CAirspaceMonitor::ps_realNameReplyReceived);
|
|
this->connect(this->m_network, &INetwork::icaoCodesReplyReceived, this, &CAirspaceMonitor::ps_icaoCodesReceived);
|
|
this->connect(this->m_network, &INetwork::pilotDisconnected, this, &CAirspaceMonitor::ps_pilotDisconnected);
|
|
this->connect(this->m_network, &INetwork::atcDisconnected, this, &CAirspaceMonitor::ps_atcControllerDisconnected);
|
|
this->connect(this->m_network, &INetwork::aircraftPositionUpdate, this, &CAirspaceMonitor::ps_aircraftUpdateReceived);
|
|
this->connect(this->m_network, &INetwork::frequencyReplyReceived, this, &CAirspaceMonitor::ps_frequencyReceived);
|
|
this->connect(this->m_network, &INetwork::capabilitiesReplyReceived, this, &CAirspaceMonitor::ps_capabilitiesReplyReceived);
|
|
this->connect(this->m_network, &INetwork::fsipirCustomPacketReceived, this, &CAirspaceMonitor::ps_fsipirCustomPacketReceived);
|
|
this->connect(this->m_network, &INetwork::serverReplyReceived, this, &CAirspaceMonitor::ps_serverReplyReceived);
|
|
|
|
// AutoConnection: this should also avoid race conditions by updating the bookings
|
|
this->connect(this->m_vatsimBookingReader, &CVatsimBookingReader::dataRead, this, &CAirspaceMonitor::ps_receivedBookings);
|
|
this->connect(this->m_vatsimDataFileReader, &CVatsimDataFileReader::dataRead, this, &CAirspaceMonitor::ps_receivedDataFile);
|
|
}
|
|
|
|
CFlightPlan CAirspaceMonitor::loadFlightPlanFromNetwork(const CCallsign &callsign)
|
|
{
|
|
CFlightPlan plan;
|
|
|
|
// use cache, but not for own callsign (always reload)
|
|
if (this->m_flightPlanCache.contains(callsign)) plan = this->m_flightPlanCache[callsign];
|
|
if (!plan.wasSentOrLoaded() || plan.timeDiffSentOrLoadedMs() > 30 * 1000)
|
|
{
|
|
// outdated, or not in cache at all
|
|
this->m_network->sendFlightPlanQuery(callsign);
|
|
|
|
// with this little trick we try to make an asynchronous signal / slot
|
|
// based approach a synchronous return value
|
|
QTime waitForFlightPlan = QTime::currentTime().addMSecs(1000);
|
|
while (QTime::currentTime() < waitForFlightPlan)
|
|
{
|
|
// process some other events and hope network answer is received already
|
|
QCoreApplication::processEvents(QEventLoop::AllEvents, 100);
|
|
if (m_flightPlanCache.contains(callsign))
|
|
{
|
|
plan = this->m_flightPlanCache[callsign];
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return plan;
|
|
}
|
|
|
|
CUserList CAirspaceMonitor::getUsers() const
|
|
{
|
|
CUserList users;
|
|
foreach(CAtcStation station, this->m_atcStationsOnline)
|
|
{
|
|
CUser user = station.getController();
|
|
users.push_back(user);
|
|
}
|
|
foreach(CAircraft aircraft, this->m_aircraftsInRange)
|
|
{
|
|
CUser user = aircraft.getPilot();
|
|
users.push_back(user);
|
|
}
|
|
return users;
|
|
}
|
|
|
|
CUserList CAirspaceMonitor::getUsersForCallsigns(const CCallsignList &callsigns) const
|
|
{
|
|
CUserList users;
|
|
if (callsigns.isEmpty()) return users;
|
|
CCallsignList searchList(callsigns);
|
|
|
|
// myself, which is not in the lists below
|
|
CAircraft ownAircraft = this->m_ownAircraft;
|
|
if (!ownAircraft.getCallsign().isEmpty() && searchList.contains(ownAircraft.getCallsign()))
|
|
{
|
|
searchList.remove(ownAircraft.getCallsign());
|
|
users.push_back(ownAircraft.getPilot());
|
|
}
|
|
|
|
// do aircrafts first, this will handle most callsigns
|
|
foreach(CAircraft aircraft, this->m_aircraftsInRange)
|
|
{
|
|
if (searchList.isEmpty()) break;
|
|
CCallsign callsign = aircraft.getCallsign();
|
|
if (searchList.contains(callsign))
|
|
{
|
|
CUser user = aircraft.getPilot();
|
|
users.push_back(user);
|
|
searchList.remove(callsign);
|
|
}
|
|
}
|
|
|
|
foreach(CAtcStation station, this->m_atcStationsOnline)
|
|
{
|
|
if (searchList.isEmpty()) break;
|
|
CCallsign callsign = station.getCallsign();
|
|
if (searchList.contains(callsign))
|
|
{
|
|
CUser user = station.getController();
|
|
users.push_back(user);
|
|
searchList.remove(callsign);
|
|
}
|
|
}
|
|
|
|
// we might have unresolved callsigns
|
|
// these are the ones not in range
|
|
foreach(CCallsign callsign, searchList)
|
|
{
|
|
CUserList usersByCallsign = this->m_vatsimDataFileReader->getUsersForCallsign(callsign);
|
|
if (usersByCallsign.isEmpty())
|
|
{
|
|
CUser user;
|
|
user.setCallsign(callsign);
|
|
users.push_back(user);
|
|
}
|
|
else
|
|
{
|
|
users.push_back(usersByCallsign[0]);
|
|
}
|
|
}
|
|
return users;
|
|
}
|
|
|
|
CClientList CAirspaceMonitor::getOtherClientsForCallsigns(const CCallsignList &callsigns) const
|
|
{
|
|
CClientList clients;
|
|
if (callsigns.isEmpty()) return clients;
|
|
foreach(CCallsign callsign, callsigns)
|
|
{
|
|
clients.push_back(this->m_otherClients.findBy(&CClient::getCallsign, callsign));
|
|
}
|
|
return clients;
|
|
}
|
|
|
|
BlackMisc::Aviation::CInformationMessage CAirspaceMonitor::getMetar(const BlackMisc::Aviation::CAirportIcao &airportIcaoCode)
|
|
{
|
|
CInformationMessage metar;
|
|
if (airportIcaoCode.isEmpty()) return metar;
|
|
if (this->m_metarCache.contains(airportIcaoCode)) metar = this->m_metarCache[airportIcaoCode];
|
|
if (metar.isEmpty() || metar.timeDiffReceivedMs() > 10 * 1000)
|
|
{
|
|
// outdated, or not in cache at all
|
|
this->m_network->sendMetarQuery(airportIcaoCode);
|
|
|
|
// with this little trick we try to make an asynchronous signal / slot
|
|
// based approach a synchronous return value
|
|
QTime waitForMetar = QTime::currentTime().addMSecs(1000);
|
|
while (QTime::currentTime() < waitForMetar)
|
|
{
|
|
// process some other events and hope network answer is received already
|
|
QCoreApplication::processEvents(QEventLoop::AllEvents, 100);
|
|
if (m_metarCache.contains(airportIcaoCode))
|
|
{
|
|
metar = this->m_metarCache[airportIcaoCode];
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return metar;
|
|
}
|
|
|
|
CAtcStation CAirspaceMonitor::getAtcStationForComUnit(const CComSystem &comSystem)
|
|
{
|
|
CAtcStation station;
|
|
CAtcStationList stations = this->m_atcStationsOnline.findIfComUnitTunedIn25KHz(comSystem);
|
|
if (!stations.isEmpty())
|
|
{
|
|
stations.sortBy(&CAtcStation::getDistanceToPlane);
|
|
station = stations.front();
|
|
}
|
|
return station;
|
|
}
|
|
|
|
void CAirspaceMonitor::requestDataUpdates()
|
|
{
|
|
if (!this->m_network->isConnected()) return;
|
|
|
|
foreach(CAircraft aircraft, this->m_aircraftsInRange)
|
|
{
|
|
this->m_network->sendFrequencyQuery(aircraft.getCallsign());
|
|
this->m_network->sendIcaoCodesQuery(aircraft.getCallsign());
|
|
}
|
|
}
|
|
|
|
void CAirspaceMonitor::requestAtisUpdates()
|
|
{
|
|
if (!this->m_network->isConnected()) return;
|
|
|
|
foreach(CAtcStation station, this->m_atcStationsOnline)
|
|
{
|
|
this->m_network->sendAtisQuery(station.getCallsign());
|
|
}
|
|
}
|
|
|
|
void CAirspaceMonitor::testCreateDummyOnlineAtcStations(int number)
|
|
{
|
|
this->m_atcStationsOnline.push_back(
|
|
BlackMisc::Aviation::CTesting::createAtcStations(number)
|
|
);
|
|
emit this->changedAtcStationsOnline();
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_realNameReplyReceived(const CCallsign &callsign, const QString &realname)
|
|
{
|
|
if (realname.isEmpty()) return;
|
|
CIndexVariantMap vm({CAtcStation::IndexController, CUser::IndexRealName}, realname);
|
|
this->m_atcStationsOnline.applyIf(&CAtcStation::getCallsign, callsign, vm);
|
|
this->m_atcStationsBooked.applyIf(&CAtcStation::getCallsign, callsign, vm);
|
|
|
|
vm = CIndexVariantMap({CAircraft::IndexPilot, CUser::IndexRealName}, realname);
|
|
this->m_aircraftsInRange.applyIf(&CAircraft::getCallsign, callsign, vm);
|
|
|
|
// Client
|
|
vm = CIndexVariantMap({CClient::IndexUser, CUser::IndexRealName}, realname);
|
|
this->addVoiceCapabilitiesFromDataFile(vm, callsign);
|
|
this->m_otherClients.applyIf(&CClient::getCallsign, callsign, vm);
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_capabilitiesReplyReceived(const BlackMisc::Aviation::CCallsign &callsign, quint32 flags)
|
|
{
|
|
if (callsign.isEmpty()) return;
|
|
CIndexVariantMap capabilities;
|
|
capabilities.addValue(CClient::FsdAtisCanBeReceived, (flags & INetwork::AcceptsAtisResponses));
|
|
capabilities.addValue(CClient::FsdWithInterimPositions, (flags & INetwork::SupportsInterimPosUpdates));
|
|
capabilities.addValue(CClient::FsdWithModelDescription, (flags & INetwork::SupportsModelDescriptions));
|
|
CIndexVariantMap vm(CClient::IndexCapabilities, capabilities.toQVariant());
|
|
this->addVoiceCapabilitiesFromDataFile(vm, callsign);
|
|
this->m_otherClients.applyIf(&CClient::getCallsign, callsign, vm);
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_fsipirCustomPacketReceived(const CCallsign &callsign, const QString &, const QString &, const QString &, const QString &model)
|
|
{
|
|
if (callsign.isEmpty() || model.isEmpty()) return;
|
|
|
|
// Request of other client, I can get the other's model from that
|
|
CIndexVariantMap vm({ CClient::IndexModel, CAircraftModel::IndexModelString }, QVariant(model));
|
|
vm.addValue({ CClient::IndexModel, CAircraftModel::IndexIsQueriedModelString }, QVariant(true));
|
|
if (!this->m_otherClients.contains(&CClient::getCallsign, callsign))
|
|
{
|
|
// with custom packets it can happen,
|
|
//the packet is received before any other packet
|
|
this->m_otherClients.push_back(CClient(callsign));
|
|
}
|
|
this->m_otherClients.applyIf(&CClient::getCallsign, callsign, vm);
|
|
this->sendFsipiCustomPacket(callsign); // response
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_serverReplyReceived(const CCallsign &callsign, const QString &server)
|
|
{
|
|
if (callsign.isEmpty() || server.isEmpty()) return;
|
|
CIndexVariantMap vm(CClient::IndexServer, QVariant(server));
|
|
this->m_otherClients.applyIf(&CClient::getCallsign, callsign, vm);
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_metarReceived(const QString &metarMessage)
|
|
{
|
|
if (metarMessage.length() < 10) return; // invalid
|
|
const QString icaoCode = metarMessage.left(4).toUpper();
|
|
const QString icaoCodeTower = icaoCode + "_TWR";
|
|
CCallsign callsignTower(icaoCodeTower);
|
|
CInformationMessage metar(CInformationMessage::METAR, metarMessage);
|
|
|
|
// add METAR to existing stations
|
|
CIndexVariantMap vm(CAtcStation::IndexMetar, metar.toQVariant());
|
|
this->m_atcStationsOnline.applyIf(&CAtcStation::getCallsign, callsignTower, vm);
|
|
this->m_atcStationsBooked.applyIf(&CAtcStation::getCallsign, callsignTower, vm);
|
|
this->m_metarCache.insert(icaoCode, metar);
|
|
if (this->m_atcStationsOnline.contains(&CAtcStation::getCallsign, callsignTower)) { emit this->changedAtcStationsOnline(); }
|
|
if (this->m_atcStationsBooked.contains(&CAtcStation::getCallsign, callsignTower)) { emit this->changedAtcStationsBooked(); }
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_flightPlanReceived(const CCallsign &callsign, const CFlightPlan &flightPlan)
|
|
{
|
|
CFlightPlan plan(flightPlan);
|
|
plan.setWhenLastSentOrLoaded(QDateTime::currentDateTimeUtc());
|
|
this->m_flightPlanCache.insert(callsign, plan);
|
|
}
|
|
|
|
void CAirspaceMonitor::sendFsipiCustomPacket(const CCallsign &recipientCallsign) const
|
|
{
|
|
CAircraft me = this->m_ownAircraft;
|
|
CAircraftIcao icao = me.getIcaoInfo();
|
|
QString modelString = this->m_ownAircraftModel.getModelString();
|
|
if (modelString.isEmpty()) modelString = CProject::systemNameAndVersion();
|
|
this->m_network->sendFsipiCustomPacket(recipientCallsign, icao.getAirlineDesignator(), icao.getAircraftDesignator(), icao.getAircraftCombinedType(), modelString);
|
|
}
|
|
|
|
void CAirspaceMonitor::sendFsipirCustomPacket(const CCallsign &recipientCallsign) const
|
|
{
|
|
CAircraft me = this->m_ownAircraft;
|
|
CAircraftIcao icao = me.getIcaoInfo();
|
|
QString modelString = this->m_ownAircraftModel.getModelString();
|
|
if (modelString.isEmpty()) modelString = CProject::systemNameAndVersion();
|
|
this->m_network->sendFsipirCustomPacket(recipientCallsign, icao.getAirlineDesignator(), icao.getAircraftDesignator(), icao.getAircraftCombinedType(), modelString);
|
|
}
|
|
|
|
void CAirspaceMonitor::addVoiceCapabilitiesFromDataFile(CIndexVariantMap &vm, const CCallsign &callsign)
|
|
{
|
|
Q_ASSERT(this->m_vatsimDataFileReader);
|
|
if (callsign.isEmpty()) return;
|
|
CVoiceCapabilities vc = this->m_vatsimDataFileReader->getVoiceCapabilityForCallsign(callsign);
|
|
if (!vc.isUnknown())
|
|
{
|
|
vm.addValue(CClient::IndexVoiceCapabilities, vc);
|
|
}
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_receivedBookings(const CAtcStationList &bookedStations)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
this->m_atcStationsBooked.clear();
|
|
foreach(CAtcStation bookedStation, bookedStations)
|
|
{
|
|
// complete by VATSIM data file data
|
|
this->m_vatsimDataFileReader->getAtcStations().updateFromVatsimDataFileStation(bookedStation);
|
|
// exchange booking and online data, both sides are updated
|
|
this->m_atcStationsOnline.mergeWithBooking(bookedStation);
|
|
// into list
|
|
this->m_atcStationsBooked.push_back(bookedStation);
|
|
}
|
|
emit this->changedAtcStationsBooked(); // all booked stations reloaded
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_receivedDataFile()
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
for (auto i = this->m_otherClients.begin(); i != this->m_otherClients.end(); ++i)
|
|
{
|
|
CClient client = (*i);
|
|
if (client.hasSpecifiedVoiceCapabilities()) continue;
|
|
CVoiceCapabilities vc = this->m_vatsimDataFileReader->getVoiceCapabilityForCallsign(client.getCallsign());
|
|
if (vc.isUnknown()) continue;
|
|
client.setVoiceCapabilities(vc);
|
|
(*i) = client;
|
|
}
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_atcPositionUpdate(const CCallsign &callsign, const BlackMisc::PhysicalQuantities::CFrequency &frequency, const CCoordinateGeodetic &position, const BlackMisc::PhysicalQuantities::CLength &range)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
CAtcStationList stationsWithCallsign = this->m_atcStationsOnline.findByCallsign(callsign);
|
|
if (stationsWithCallsign.isEmpty())
|
|
{
|
|
// new station
|
|
CAtcStation station;
|
|
station.setCallsign(callsign);
|
|
station.setRange(range);
|
|
station.setFrequency(frequency);
|
|
station.setPosition(position);
|
|
station.setOnline(true);
|
|
station.calculcateDistanceToPlane(this->m_ownAircraft.getPosition());
|
|
this->m_vatsimDataFileReader->getAtcStations().updateFromVatsimDataFileStation(station); // prefill
|
|
this->m_atcStationsOnline.push_back(station);
|
|
|
|
if (this->m_network->isConnected())
|
|
{
|
|
emit this->m_network->sendRealNameQuery(callsign);
|
|
emit this->m_network->sendAtisQuery(callsign); // request ATIS and voice rooms
|
|
emit this->m_network->sendServerQuery(callsign);
|
|
}
|
|
|
|
emit this->changedAtcStationsOnline();
|
|
// Remark: this->changedAtcStationOnlineConnectionStatus(station, true);
|
|
// will be sent in psFsdAtisVoiceRoomReceived
|
|
}
|
|
else
|
|
{
|
|
// update
|
|
CIndexVariantMap values;
|
|
values.addValue(CAtcStation::IndexFrequency, frequency);
|
|
values.addValue(CAtcStation::IndexPosition, position);
|
|
values.addValue(CAtcStation::IndexRange, range);
|
|
this->m_atcStationsOnline.applyIf(BlackMisc::Predicates::MemberEqual(&CAtcStation::getCallsign, callsign), values);
|
|
emit this->changedAtcStationsOnline();
|
|
}
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_atcControllerDisconnected(const CCallsign &callsign)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
if (this->m_atcStationsOnline.contains(&CAtcStation::getCallsign, callsign))
|
|
{
|
|
CAtcStation removeStation = this->m_atcStationsOnline.findByCallsign(callsign).front();
|
|
this->m_atcStationsOnline.removeIf(&CAtcStation::getCallsign, callsign);
|
|
emit this->changedAtcStationsOnline();
|
|
emit this->changedAtcStationOnlineConnectionStatus(removeStation, false);
|
|
}
|
|
|
|
// booked
|
|
this->m_atcStationsBooked.applyIf(&CAtcStation::getCallsign, callsign, CIndexVariantMap(CAtcStation::IndexIsOnline, QVariant(false)));
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_atisReceived(const CCallsign &callsign, const CInformationMessage &atisMessage)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
if (callsign.isEmpty()) return;
|
|
CIndexVariantMap vm(CAtcStation::IndexAtis, atisMessage.toQVariant());
|
|
this->m_atcStationsOnline.applyIf(&CAtcStation::getCallsign, callsign, vm);
|
|
|
|
// receiving an ATIS means station is online, update in bookings
|
|
vm.addValue(CAtcStation::IndexIsOnline, true);
|
|
this->m_atcStationsBooked.applyIf(&CAtcStation::getCallsign, callsign, vm);
|
|
if (this->m_atcStationsOnline.contains(&CAtcStation::getCallsign, callsign)) { emit this->changedAtcStationsOnline(); }
|
|
if (this->m_atcStationsBooked.contains(&CAtcStation::getCallsign, callsign)) { emit this->changedAtcStationsBooked(); }
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_atisVoiceRoomReceived(const CCallsign &callsign, const QString &url)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
QString trimmedUrl = url.trimmed();
|
|
CIndexVariantMap vm({ CAtcStation::IndexVoiceRoom, CVoiceRoom::IndexUrl }, trimmedUrl);
|
|
if (this->m_atcStationsOnline.contains(&CAtcStation::getCallsign, callsign))
|
|
{
|
|
this->m_atcStationsOnline.applyIf(&CAtcStation::getCallsign, callsign, vm);
|
|
CAtcStation station = this->m_atcStationsOnline.findFirstByCallsign(callsign);
|
|
emit this->changedAtcStationsOnline(); // single ATIS received
|
|
emit this->changedAtcStationOnlineConnectionStatus(station, true);
|
|
}
|
|
if (this->m_atcStationsBooked.contains(&CAtcStation::getCallsign, callsign))
|
|
{
|
|
// receiving a voice room means station is online, update in bookings
|
|
vm.addValue(CAtcStation::IndexIsOnline, true);
|
|
this->m_atcStationsBooked.applyIf(&CAtcStation::getCallsign, callsign, vm);
|
|
emit this->changedAtcStationsBooked(); // single ATIS received
|
|
}
|
|
|
|
// receiving voice room means ATC has voice
|
|
vm = CIndexVariantMap(CClient::IndexVoiceCapabilities, CVoiceCapabilities::fromVoiceCapabilities(CVoiceCapabilities::Voice).toQVariant());
|
|
this->m_otherClients.applyIf(&CClient::getCallsign, callsign, vm);
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_atisLogoffTimeReceived(const CCallsign &callsign, const QString &zuluTime)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
if (zuluTime.length() == 4)
|
|
{
|
|
// Logic to set logoff time
|
|
bool ok;
|
|
int h = zuluTime.left(2).toInt(&ok);
|
|
if (!ok) { return; }
|
|
int m = zuluTime.right(2).toInt(&ok);
|
|
if (!ok) { return; }
|
|
QDateTime logoffDateTime = QDateTime::currentDateTimeUtc();
|
|
logoffDateTime.setTime(QTime(h, m));
|
|
CIndexVariantMap vm(CAtcStation::IndexBookedUntil, logoffDateTime);
|
|
this->m_atcStationsOnline.applyIf(&CAtcStation::getCallsign, callsign, vm);
|
|
this->m_atcStationsBooked.applyIf(&CAtcStation::getCallsign, callsign, vm);
|
|
if (this->m_atcStationsOnline.contains(&CAtcStation::getCallsign, callsign)) { emit this->changedAtcStationsOnline(); }
|
|
if (this->m_atcStationsBooked.contains(&CAtcStation::getCallsign, callsign)) { emit this->changedAtcStationsBooked(); }
|
|
}
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_icaoCodesReceived(const CCallsign &callsign, const CAircraftIcao &icaoData)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
|
|
// update
|
|
CIndexVariantMap vm(CAircraft::IndexIcao, icaoData.toQVariant());
|
|
if (!icaoData.hasAircraftDesignator())
|
|
{
|
|
// empty so far, try to fetch from data file
|
|
qDebug() << "Empty ICAO info for " << callsign << icaoData;
|
|
CAircraftIcao icaoDataDataFile = this->m_vatsimDataFileReader->getIcaoInfo(callsign);
|
|
if (!icaoDataDataFile.hasAircraftDesignator()) return; // give up!
|
|
vm = CIndexVariantMap(CAircraft::IndexIcao, icaoData.toQVariant());
|
|
}
|
|
this->m_aircraftsInRange.applyIf(BlackMisc::Predicates::MemberEqual(&CAircraft::getCallsign, callsign), vm);
|
|
emit this->changedAircraftsInRange();
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_aircraftUpdateReceived(const CCallsign &callsign, const CAircraftSituation &situation, const CTransponder &transponder)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
|
|
CAircraftList list = this->m_aircraftsInRange.findByCallsign(callsign);
|
|
if (list.isEmpty())
|
|
{
|
|
// new aircraft
|
|
CAircraft aircraft;
|
|
aircraft.setCallsign(callsign);
|
|
aircraft.setSituation(situation);
|
|
aircraft.setTransponder(transponder);
|
|
aircraft.setCalculcatedDistanceToPosition(this->m_ownAircraft.getPosition()); // distance from myself
|
|
this->m_vatsimDataFileReader->updateWithVatsimDataFileData(aircraft);
|
|
this->m_aircraftsInRange.push_back(aircraft);
|
|
|
|
// and new client, there is a chance it has been created by
|
|
// custom packet first
|
|
if (!this->m_otherClients.contains(&CClient::getCallsign, callsign))
|
|
{
|
|
CClient c(callsign);
|
|
this->m_otherClients.push_back(c); // initial, will be filled by data later
|
|
}
|
|
|
|
if (this->m_network->isConnected())
|
|
{
|
|
// only if still connected
|
|
this->m_network->sendFrequencyQuery(callsign);
|
|
this->m_network->sendRealNameQuery(callsign);
|
|
this->m_network->sendIcaoCodesQuery(callsign);
|
|
this->m_network->sendCapabilitiesQuery(callsign);
|
|
this->m_network->sendServerQuery(callsign);
|
|
this->sendFsipirCustomPacket(callsign); // own aircraft model
|
|
}
|
|
|
|
emit this->addedAircraft(callsign, situation);
|
|
}
|
|
else
|
|
{
|
|
// update
|
|
CLength distance = this->m_ownAircraft.calculcateDistanceToPosition(situation.getPosition());
|
|
distance.switchUnit(CLengthUnit::NM());
|
|
CIndexVariantMap vm;
|
|
vm.addValue(CAircraft::IndexTransponder, transponder);
|
|
vm.addValue(CAircraft::IndexSituation, situation);
|
|
vm.addValue(CAircraft::IndexDistance, distance);
|
|
this->m_aircraftsInRange.applyIf(BlackMisc::Predicates::MemberEqual(&CAircraft::getCallsign, callsign), vm);
|
|
|
|
emit this->changedAircraftSituation(callsign, situation);
|
|
}
|
|
|
|
emit this->changedAircraftsInRange();
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_pilotDisconnected(const CCallsign &callsign)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
|
|
bool contains = this->m_aircraftsInRange.contains(&CAircraft::getCallsign, callsign);
|
|
|
|
this->m_aircraftsInRange.removeIf(&CAircraft::getCallsign, callsign);
|
|
this->m_otherClients.removeIf(&CClient::getCallsign, callsign);
|
|
|
|
if (contains)
|
|
{
|
|
emit this->removedAircraft(callsign);
|
|
}
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_frequencyReceived(const CCallsign &callsign, const CFrequency &frequency)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
|
|
// update
|
|
CIndexVariantMap vm({CAircraft::IndexCom1System, CComSystem::IndexActiveFrequency}, frequency.toQVariant());
|
|
this->m_aircraftsInRange.applyIf(BlackMisc::Predicates::MemberEqual(&CAircraft::getCallsign, callsign), vm);
|
|
emit this->changedAircraftsInRange();
|
|
}
|
|
|
|
} // namespace
|