mirror of
https://github.com/swift-project/pilotclient.git
synced 2026-03-22 14:55:36 +08:00
1088 lines
46 KiB
C++
1088 lines
46 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/logmessage.h"
|
|
#include "blackmisc/blackmiscfreefunctions.h"
|
|
#include "blackmisc/propertyindexallclasses.h"
|
|
|
|
using namespace BlackMisc;
|
|
using namespace BlackMisc::Aviation;
|
|
using namespace BlackMisc::Audio;
|
|
using namespace BlackMisc::Simulation;
|
|
using namespace BlackMisc::Network;
|
|
using namespace BlackMisc::Geo;
|
|
using namespace BlackMisc::PhysicalQuantities;
|
|
|
|
namespace BlackCore
|
|
{
|
|
|
|
CAirspaceMonitor::CAirspaceMonitor(QObject *parent, BlackMisc::Simulation::IOwnAircraftProvider *ownAircraftProvider, INetwork *network, CVatsimBookingReader *bookings, CVatsimDataFileReader *dataFile)
|
|
: QObject(parent),
|
|
COwnAircraftAware(ownAircraftProvider),
|
|
m_network(network), m_vatsimBookingReader(bookings), m_vatsimDataFileReader(dataFile),
|
|
m_analyzer(new CAirspaceAnalyzer(ownAircraftProvider, this, network, this))
|
|
{
|
|
this->setObjectName("CAirspaceMonitor");
|
|
m_interimPositionUpdateTimer.setObjectName(this->objectName().append(":m_interimPositionUpdateTimer"));
|
|
|
|
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::aircraftInterimPositionUpdate, this, &CAirspaceMonitor::ps_aircraftInterimUpdateReceived);
|
|
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::customFSinnPacketReceived, this, &CAirspaceMonitor::ps_customFSinnPacketReceived);
|
|
this->connect(this->m_network, &INetwork::serverReplyReceived, this, &CAirspaceMonitor::ps_serverReplyReceived);
|
|
this->connect(this->m_network, &INetwork::aircraftConfigPacketReceived, this, &CAirspaceMonitor::ps_aircraftConfigReceived);
|
|
this->connect(&m_interimPositionUpdateTimer, &QTimer::timeout, this, &CAirspaceMonitor::ps_sendInterimPositions);
|
|
|
|
// 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);
|
|
|
|
// Analyzer
|
|
this->connect(this->m_analyzer, &CAirspaceAnalyzer::timeoutAircraft, this, &CAirspaceMonitor::ps_pilotDisconnected);
|
|
this->connect(this->m_analyzer, &CAirspaceAnalyzer::timeoutAtc, this, &CAirspaceMonitor::ps_atcControllerDisconnected);
|
|
}
|
|
|
|
CSimulatedAircraftList CAirspaceMonitor::getAircraftInRange() const
|
|
{
|
|
QReadLocker l(&m_lockAircraft);
|
|
return m_aircraftInRange;
|
|
}
|
|
|
|
CSimulatedAircraft CAirspaceMonitor::getAircraftInRangeForCallsign(const CCallsign &callsign) const
|
|
{
|
|
QReadLocker l(&m_lockAircraft);
|
|
return m_aircraftInRange.findFirstByCallsign(callsign);
|
|
}
|
|
|
|
int CAirspaceMonitor::getAircraftInRangeCount() const
|
|
{
|
|
QReadLocker l(&m_lockAircraft);
|
|
return m_aircraftInRange.size();
|
|
}
|
|
|
|
CAirspaceAircraftSnapshot CAirspaceMonitor::getLatestAirspaceAircraftSnapshot() const
|
|
{
|
|
Q_ASSERT_X(this->m_analyzer, Q_FUNC_INFO, "No analyzer");
|
|
return this->m_analyzer->getLatestAirspaceAircraftSnapshot();
|
|
}
|
|
|
|
CAircraftSituationList CAirspaceMonitor::remoteAircraftSituations(const CCallsign &callsign) const
|
|
{
|
|
QReadLocker l(&m_lockSituations);
|
|
static const CAircraftSituationList empty;
|
|
if (!m_situationsByCallsign.contains(callsign)) { return empty; }
|
|
return m_situationsByCallsign[callsign];
|
|
}
|
|
|
|
int CAirspaceMonitor::remoteAircraftSituationsCount(const CCallsign &callsign) const
|
|
{
|
|
QReadLocker l(&m_lockSituations);
|
|
if (!m_situationsByCallsign.contains(callsign)) { return -1; }
|
|
return m_situationsByCallsign[callsign].size();
|
|
}
|
|
|
|
CAircraftPartsList CAirspaceMonitor::remoteAircraftParts(const CCallsign &callsign, qint64 cutoffTimeValuesBefore) const
|
|
{
|
|
QReadLocker l(&m_lockParts);
|
|
static const CAircraftPartsList empty;
|
|
if (!m_partsByCallsign.contains(callsign)) { return empty; }
|
|
if (cutoffTimeValuesBefore < 0)
|
|
{
|
|
return m_partsByCallsign[callsign];
|
|
}
|
|
else
|
|
{
|
|
return m_partsByCallsign[callsign].findBefore(cutoffTimeValuesBefore);
|
|
}
|
|
}
|
|
|
|
bool CAirspaceMonitor::isRemoteAircraftSupportingParts(const CCallsign &callsign) const
|
|
{
|
|
QReadLocker l(&m_lockParts);
|
|
return m_aircraftSupportingParts.contains(callsign);
|
|
}
|
|
|
|
CCallsignSet CAirspaceMonitor::remoteAircraftSupportingParts() const
|
|
{
|
|
QReadLocker l(&m_lockParts);
|
|
return m_aircraftSupportingParts;
|
|
}
|
|
|
|
bool CAirspaceMonitor::connectRemoteAircraftProviderSignals(
|
|
std::function<void(const CAircraftSituation &)> situationSlot,
|
|
std::function<void(const CAircraftParts &)> partsSlot,
|
|
std::function<void(const CCallsign &)> removedAircraftSlot,
|
|
std::function<void(const CAirspaceAircraftSnapshot &)> aircraftSnapshotSlot
|
|
)
|
|
{
|
|
bool s1 = connect(this, &CAirspaceMonitor::addedAircraftSituation, situationSlot);
|
|
Q_ASSERT(s1);
|
|
bool s2 = connect(this, &CAirspaceMonitor::addedAircraftParts, partsSlot);
|
|
Q_ASSERT(s2);
|
|
bool s3 = connect(this, &CAirspaceMonitor::removedAircraft, removedAircraftSlot);
|
|
Q_ASSERT(s3);
|
|
bool s4 = connect(this->m_analyzer, &CAirspaceAnalyzer::airspaceAircraftSnapshot, aircraftSnapshotSlot);
|
|
Q_ASSERT(s4);
|
|
|
|
return s1 && s2 && s3 && s4;
|
|
}
|
|
|
|
bool CAirspaceMonitor::disconnectRemoteAircraftProviderSignals(QObject *receiver)
|
|
{
|
|
if (!receiver) { return false; }
|
|
return this->disconnect(receiver);
|
|
}
|
|
|
|
bool CAirspaceMonitor::updateAircraftEnabled(const CCallsign &callsign, bool enabledForRedering, const QString &originator)
|
|
{
|
|
Q_UNUSED(originator);
|
|
CPropertyIndexVariantMap vm(CSimulatedAircraft::IndexEnabled, CVariant::fromValue(enabledForRedering));
|
|
QWriteLocker l(&m_lockAircraft);
|
|
int c = m_aircraftInRange.applyIfCallsign(callsign, vm);
|
|
return c > 0;
|
|
}
|
|
|
|
bool CAirspaceMonitor::updateAircraftModel(const CCallsign &callsign, const CAircraftModel &model, const QString &originator)
|
|
{
|
|
Q_UNUSED(originator);
|
|
QWriteLocker l(&m_lockAircraft);
|
|
CPropertyIndexVariantMap vm(CSimulatedAircraft::IndexModel, model.toCVariant());
|
|
int c = m_aircraftInRange.applyIfCallsign(callsign, vm);
|
|
return c > 0;
|
|
}
|
|
|
|
bool CAirspaceMonitor::updateFastPositionEnabled(const CCallsign &callsign, bool enableFastPositonUpdates, const QString &originator)
|
|
{
|
|
Q_UNUSED(originator);
|
|
CPropertyIndexVariantMap vm(CSimulatedAircraft::IndexFastPositionUpdates, CVariant::fromValue(enableFastPositonUpdates));
|
|
QWriteLocker l(&m_lockAircraft);
|
|
int c = m_aircraftInRange.applyIfCallsign(callsign, vm);
|
|
return c > 0;
|
|
}
|
|
|
|
bool CAirspaceMonitor::updateAircraftRendered(const CCallsign &callsign, bool rendered, const QString &originator)
|
|
{
|
|
Q_UNUSED(originator);
|
|
CPropertyIndexVariantMap vm(CSimulatedAircraft::IndexRendered, CVariant::fromValue(rendered));
|
|
QWriteLocker l(&m_lockAircraft);
|
|
int c = m_aircraftInRange.applyIfCallsign(callsign, vm);
|
|
return c > 0;
|
|
}
|
|
|
|
void CAirspaceMonitor::updateMarkAllAsNotRendered(const QString &originator)
|
|
{
|
|
Q_UNUSED(originator);
|
|
QWriteLocker l(&m_lockAircraft);
|
|
m_aircraftInRange.markAllAsNotRendered();
|
|
}
|
|
|
|
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);
|
|
}
|
|
for (const CAircraft &aircraft : this->getAircraftInRange())
|
|
{
|
|
CUser user = aircraft.getPilot();
|
|
users.push_back(user);
|
|
}
|
|
return users;
|
|
}
|
|
|
|
CUserList CAirspaceMonitor::getUsersForCallsigns(const CCallsignSet &callsigns) const
|
|
{
|
|
CUserList users;
|
|
if (callsigns.isEmpty()) { return users; }
|
|
CCallsignSet searchList(callsigns);
|
|
|
|
// myself, which is not in the lists below
|
|
CSimulatedAircraft myAircraft(getOwnAircraft());
|
|
if (!myAircraft.getCallsign().isEmpty() && searchList.contains(myAircraft.getCallsign()))
|
|
{
|
|
searchList.remove(myAircraft.getCallsign());
|
|
users.push_back(myAircraft.getPilot());
|
|
}
|
|
|
|
// do aircraft first, this will handle most callsigns
|
|
for (const CAircraft &aircraft : this->getAircraftInRange())
|
|
{
|
|
if (searchList.isEmpty()) break;
|
|
CCallsign callsign = aircraft.getCallsign();
|
|
if (searchList.contains(callsign))
|
|
{
|
|
CUser user = aircraft.getPilot();
|
|
users.push_back(user);
|
|
searchList.remove(callsign);
|
|
}
|
|
}
|
|
|
|
for (const 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
|
|
// those are the ones not in range
|
|
for (const 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 CCallsignSet &callsigns) const
|
|
{
|
|
CClientList clients;
|
|
if (callsigns.isEmpty()) { return clients; }
|
|
clients.push_back(this->m_otherClients.findByCallsigns(callsigns));
|
|
return clients;
|
|
}
|
|
|
|
CClientList CAirspaceMonitor::getOtherClients() const
|
|
{
|
|
return m_otherClients;
|
|
}
|
|
|
|
BlackMisc::Aviation::CInformationMessage CAirspaceMonitor::getMetar(const BlackMisc::Aviation::CAirportIcaoCode &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()) { return station; }
|
|
stations.sortByDistanceToOwnAircraft();
|
|
return stations.front();
|
|
}
|
|
|
|
void CAirspaceMonitor::requestDataUpdates()
|
|
{
|
|
if (!this->m_network->isConnected()) { return; }
|
|
for (const CAircraft &aircraft : this->getAircraftInRange())
|
|
{
|
|
const CCallsign cs(aircraft.getCallsign());
|
|
this->m_network->sendFrequencyQuery(cs);
|
|
|
|
// we only query ICAO if we have none yet
|
|
// it happens sometimes with some FSD servers (e.g our testserver) a first query is skipped
|
|
if (!aircraft.hasValidAircraftDesignator())
|
|
{
|
|
this->m_network->sendIcaoCodesQuery(cs);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CAirspaceMonitor::requestAtisUpdates()
|
|
{
|
|
if (!this->m_network->isConnected()) { return; }
|
|
for (const CAtcStation &station : this->m_atcStationsOnline)
|
|
{
|
|
this->m_network->sendAtisQuery(station.getCallsign());
|
|
}
|
|
}
|
|
|
|
void CAirspaceMonitor::testCreateDummyOnlineAtcStations(int number)
|
|
{
|
|
if (number < 1) { return; }
|
|
this->m_atcStationsOnline.push_back(
|
|
BlackMisc::Aviation::CTesting::createAtcStations(number)
|
|
);
|
|
emit this->changedAtcStationsOnline();
|
|
}
|
|
|
|
void CAirspaceMonitor::testAddAircraftParts(const CAircraftParts &parts, bool incremental)
|
|
{
|
|
this->ps_aircraftConfigReceived(parts.getCallsign(), parts.toJson(), !incremental);
|
|
}
|
|
|
|
void CAirspaceMonitor::clear()
|
|
{
|
|
m_metarCache.clear();
|
|
m_flightPlanCache.clear();
|
|
m_icaoCodeCache.clear();
|
|
|
|
removeAllOnlineAtcStations();
|
|
removeAllAircraft();
|
|
removeAllOtherClients();
|
|
}
|
|
|
|
void CAirspaceMonitor::setConnected(bool connected)
|
|
{
|
|
this->m_connected = connected;
|
|
if (!connected)
|
|
{
|
|
this->clear();
|
|
}
|
|
}
|
|
|
|
void CAirspaceMonitor::enableFastPositionSending(bool enable)
|
|
{
|
|
if (enable)
|
|
{
|
|
m_interimPositionUpdateTimer.start();
|
|
}
|
|
else
|
|
{
|
|
m_interimPositionUpdateTimer.stop();
|
|
}
|
|
m_sendInterimPositions = enable;
|
|
}
|
|
|
|
void CAirspaceMonitor::gracefulShutdown()
|
|
{
|
|
if (this->m_analyzer) { this->m_analyzer->gracefulShutdown(); }
|
|
QObject::disconnect(this);
|
|
this->enableFastPositionSending(false);
|
|
}
|
|
|
|
bool CAirspaceMonitor::isFastPositionSendingEnabled() const
|
|
{
|
|
return m_sendInterimPositions;
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_realNameReplyReceived(const CCallsign &callsign, const QString &realname)
|
|
{
|
|
Q_ASSERT(this->m_vatsimDataFileReader);
|
|
if (!this->m_connected || realname.isEmpty()) { return; }
|
|
CPropertyIndexVariantMap vm({CAtcStation::IndexController, CUser::IndexRealName}, realname);
|
|
this->m_atcStationsOnline.applyIf(&CAtcStation::getCallsign, callsign, vm);
|
|
this->m_atcStationsBooked.applyIf(&CAtcStation::getCallsign, callsign, vm);
|
|
|
|
CVoiceCapabilities caps = this->m_vatsimDataFileReader->getVoiceCapabilityForCallsign(callsign);
|
|
vm = CPropertyIndexVariantMap({CAircraft::IndexPilot, CUser::IndexRealName}, realname);
|
|
vm.addValue({ CSimulatedAircraft::IndexClient, CClient::IndexUser, CUser::IndexRealName }, realname);
|
|
vm.addValue({ CSimulatedAircraft::IndexClient, CClient::IndexVoiceCapabilities }, caps);
|
|
|
|
// lock block
|
|
{
|
|
QWriteLocker l(&m_lockAircraft);
|
|
this->m_aircraftInRange.applyIf(&CAircraft::getCallsign, callsign, vm);
|
|
}
|
|
|
|
// Client
|
|
vm = CPropertyIndexVariantMap({CClient::IndexUser, CUser::IndexRealName}, realname);
|
|
vm.addValue({ CClient::IndexVoiceCapabilities }, caps);
|
|
if (!this->m_otherClients.containsCallsign(callsign)) { this->m_otherClients.push_back(CClient(callsign)); }
|
|
this->m_otherClients.applyIf(&CClient::getCallsign, callsign, vm);
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_capabilitiesReplyReceived(const BlackMisc::Aviation::CCallsign &callsign, quint32 flags)
|
|
{
|
|
if (!this->m_connected || callsign.isEmpty()) { return; }
|
|
CPropertyIndexVariantMap capabilities;
|
|
capabilities.addValue(CClient::FsdAtisCanBeReceived, (flags & INetwork::AcceptsAtisResponses));
|
|
capabilities.addValue(CClient::FsdWithInterimPositions, (flags & INetwork::SupportsInterimPosUpdates));
|
|
capabilities.addValue(CClient::FsdWithModelDescription, (flags & INetwork::SupportsModelDescriptions));
|
|
capabilities.addValue(CClient::FsdWithAircraftConfig, (flags & INetwork::SupportsAircraftConfigs));
|
|
|
|
CPropertyIndexVariantMap vm(CClient::IndexCapabilities, capabilities.toCVariant());
|
|
CVoiceCapabilities caps = m_vatsimDataFileReader->getVoiceCapabilityForCallsign(callsign);
|
|
vm.addValue({CClient::IndexVoiceCapabilities}, caps);
|
|
if (!this->m_otherClients.containsCallsign(callsign)) { this->m_otherClients.push_back(CClient(callsign)); }
|
|
this->m_otherClients.applyIf(&CClient::getCallsign, callsign, vm);
|
|
|
|
// apply same to client in aircraft
|
|
vm.prependIndex(static_cast<int>(CSimulatedAircraft::IndexClient));
|
|
// lock block
|
|
{
|
|
QWriteLocker l(&m_lockAircraft);
|
|
this->m_aircraftInRange.applyIf(&CSimulatedAircraft::getCallsign, callsign, vm);
|
|
}
|
|
if (flags & INetwork::SupportsAircraftConfigs) m_network->sendAircraftConfigQuery(callsign);
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_customFSinnPacketReceived(const CCallsign &callsign, const QString &airlineIcao, const QString &aircraftDesignator, const QString &combinedAircraftType, const QString &model)
|
|
{
|
|
if (!this->m_connected || callsign.isEmpty() || model.isEmpty()) { return; }
|
|
|
|
// Request of other client, I can get the other's model from that
|
|
CPropertyIndexVariantMap vm({ CClient::IndexModel, CAircraftModel::IndexModelString }, model);
|
|
vm.addValue({ CClient::IndexModel, CAircraftModel::IndexModelType }, static_cast<int>(CAircraftModel::TypeQueriedFromNetwork));
|
|
if (!this->m_otherClients.containsCallsign(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);
|
|
|
|
// make index from plain -> to client
|
|
vm.prependIndex(static_cast<int>(CSimulatedAircraft::IndexClient));
|
|
bool aircraftContainsCallsign;
|
|
// lock block
|
|
{
|
|
QWriteLocker l(&m_lockAircraft);
|
|
int u = this->m_aircraftInRange.applyIf(&CSimulatedAircraft::getCallsign, callsign, vm);
|
|
// if update was successful we know we have that callsign, otherwise explicit check
|
|
aircraftContainsCallsign = (u > 0) || this->m_aircraftInRange.containsCallsign(callsign);
|
|
}
|
|
|
|
// ICAO response from custom data
|
|
if (!aircraftDesignator.isEmpty())
|
|
{
|
|
CAircraftIcaoData icao(aircraftDesignator, combinedAircraftType, airlineIcao, "", ""); // from custom packet
|
|
if (aircraftContainsCallsign)
|
|
{
|
|
// we have that aircraft, set straight away
|
|
this->ps_icaoCodesReceived(callsign, icao);
|
|
}
|
|
else
|
|
{
|
|
// store in cache, we can retrieve laterxs
|
|
this->m_icaoCodeCache.insert(callsign, icao);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_serverReplyReceived(const CCallsign &callsign, const QString &server)
|
|
{
|
|
if (!this->m_connected || callsign.isEmpty() || server.isEmpty()) { return; }
|
|
if (!this->m_otherClients.containsCallsign(callsign)) { this->m_otherClients.push_back(CClient(callsign)); }
|
|
CPropertyIndexVariantMap vm(CClient::IndexServer, server);
|
|
this->m_otherClients.applyIf(&CClient::getCallsign, callsign, vm);
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_metarReceived(const QString &metarMessage)
|
|
{
|
|
if (!this->m_connected || 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
|
|
CPropertyIndexVariantMap vm(CAtcStation::IndexMetar, metar.toCVariant());
|
|
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.containsCallsign(callsignTower)) { emit this->changedAtcStationsOnline(); }
|
|
if (this->m_atcStationsBooked.containsCallsign(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::removeAllOnlineAtcStations()
|
|
{
|
|
m_atcStationsOnline.clear();
|
|
}
|
|
|
|
void CAirspaceMonitor::removeAllAircraft()
|
|
{
|
|
for (const CAircraft &aircraft : getAircraftInRange())
|
|
{
|
|
const CCallsign cs(aircraft.getCallsign());
|
|
emit removedAircraft(cs);
|
|
}
|
|
|
|
QWriteLocker l1(&m_lockParts);
|
|
QWriteLocker l2(&m_lockSituations);
|
|
m_situationsByCallsign.clear();
|
|
m_partsByCallsign.clear();
|
|
m_aircraftSupportingParts.clear();
|
|
m_aircraftInRange.clear();
|
|
m_flightPlanCache.clear();
|
|
m_icaoCodeCache.clear();
|
|
}
|
|
|
|
void CAirspaceMonitor::removeAllOtherClients()
|
|
{
|
|
m_otherClients.clear();
|
|
}
|
|
|
|
void CAirspaceMonitor::removeFromAircraftCaches(const CCallsign &callsign)
|
|
{
|
|
if (callsign.isEmpty()) { return; }
|
|
this->m_icaoCodeCache.remove(callsign);
|
|
this->m_flightPlanCache.remove(callsign);
|
|
}
|
|
|
|
void CAirspaceMonitor::fireDelayedReadyForModelMatching(const CCallsign &callsign, int trial, int delayMs)
|
|
{
|
|
BlackMisc::singleShot(delayMs, QThread::currentThread(), [ = ]()
|
|
{
|
|
this->ps_sendReadyForModelMatching(callsign, trial + 1);
|
|
});
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_receivedBookings(const CAtcStationList &bookedStations)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
if (bookedStations.isEmpty())
|
|
{
|
|
this->m_atcStationsBooked.clear();
|
|
}
|
|
else
|
|
{
|
|
CAtcStationList newBookedStations(bookedStations); // modifyable copy
|
|
for (CAtcStation &bookedStation : newBookedStations)
|
|
{
|
|
// exchange booking and online data, both sides are updated
|
|
this->m_atcStationsOnline.syncronizeWithBookedStation(bookedStation);
|
|
}
|
|
this->m_atcStationsBooked = newBookedStations;
|
|
}
|
|
emit this->changedAtcStationsBooked(); // all booked stations reloaded
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_receivedDataFile()
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
for (auto client = this->m_otherClients.begin(); client != this->m_otherClients.end(); ++client)
|
|
{
|
|
if (client->hasSpecifiedVoiceCapabilities()) { continue; } // we already have voice caps
|
|
CVoiceCapabilities vc = this->m_vatsimDataFileReader->getVoiceCapabilityForCallsign(client->getCallsign());
|
|
if (vc.isUnknown()) { continue; }
|
|
client->setVoiceCapabilities(vc);
|
|
}
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_sendReadyForModelMatching(const CCallsign &callsign, int trial)
|
|
{
|
|
// some checks for special conditions, e.g. logout -> empty list, but still signals pending
|
|
CSimulatedAircraft remoteAircraft;
|
|
{
|
|
QReadLocker l(&m_lockAircraft);
|
|
if (!this->m_connected || this->m_aircraftInRange.isEmpty()) { return; }
|
|
if (!this->m_aircraftInRange.containsCallsign(callsign)) { return; }
|
|
|
|
// build simulated aircraft and crosscheck if all data are available
|
|
remoteAircraft = CSimulatedAircraft(this->m_aircraftInRange.findFirstByCallsign(callsign));
|
|
Q_ASSERT_X(remoteAircraft.hasValidCallsign(), Q_FUNC_INFO, "Invalid callsign");
|
|
}
|
|
|
|
CClient remoteClient = this->m_otherClients.findFirstByCallsign(callsign);
|
|
remoteAircraft.setClient(remoteClient);
|
|
remoteAircraft.setModel(remoteClient.getAircraftModel());
|
|
|
|
// check if the name and ICAO query went properly through
|
|
bool dataComplete =
|
|
remoteAircraft.hasValidAircraftDesignator() &&
|
|
(!m_serverSupportsNameQuery || remoteAircraft.hasValidRealName());
|
|
|
|
if (trial < 3 && !dataComplete)
|
|
{
|
|
// allow another period for the client data to arrive, otherwise go ahead
|
|
this->fireDelayedReadyForModelMatching(callsign, trial + 1);
|
|
return;
|
|
}
|
|
|
|
Q_ASSERT(remoteAircraft.hasValidAircraftDesignator());
|
|
Q_ASSERT(!m_serverSupportsNameQuery || remoteAircraft.hasValidRealName());
|
|
emit this->readyForModelMatching(remoteAircraft);
|
|
}
|
|
|
|
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));
|
|
Q_ASSERT(CComSystem::isValidCivilAviationFrequency(frequency));
|
|
if (!this->m_connected) { return; }
|
|
CAtcStationList stationsWithCallsign = this->m_atcStationsOnline.findByCallsign(callsign);
|
|
if (stationsWithCallsign.isEmpty())
|
|
{
|
|
// new station, init with data from data file
|
|
CAtcStation station(this->m_vatsimDataFileReader->getAtcStationsForCallsign(callsign).frontOrDefault());
|
|
station.setCallsign(callsign);
|
|
station.setRange(range);
|
|
station.setFrequency(frequency);
|
|
station.setPosition(position);
|
|
station.setOnline(true);
|
|
station.calculcateDistanceAndBearingToOwnAircraft(getOwnAircraftPosition());
|
|
|
|
// sync with bookings
|
|
if (this->m_atcStationsBooked.containsCallsign(callsign))
|
|
{
|
|
CAtcStation bookedStation(this->m_atcStationsBooked.findFirstByCallsign(callsign));
|
|
station.syncronizeWithBookedStation(bookedStation);
|
|
this->m_atcStationsBooked.replaceIf(&CAtcStation::getCallsign, callsign, bookedStation);
|
|
}
|
|
|
|
this->m_atcStationsOnline.push_back(station);
|
|
|
|
// subsequent queries
|
|
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
|
|
// will be sent in psFsdAtisVoiceRoomReceived
|
|
}
|
|
else
|
|
{
|
|
// update
|
|
CPropertyIndexVariantMap vm;
|
|
vm.addValue(CAtcStation::IndexFrequency, frequency);
|
|
vm.addValue(CAtcStation::IndexPosition, position);
|
|
vm.addValue(CAtcStation::IndexRange, range);
|
|
int changed = this->m_atcStationsOnline.applyIf(&CAtcStation::getCallsign, callsign, vm, true);
|
|
if (changed > 0) { emit this->changedAtcStationsOnline(); }
|
|
}
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_atcControllerDisconnected(const CCallsign &callsign)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
|
|
this->m_otherClients.removeByCallsign(callsign);
|
|
if (this->m_atcStationsOnline.containsCallsign(callsign))
|
|
{
|
|
CAtcStation removedStation = this->m_atcStationsOnline.findFirstByCallsign(callsign);
|
|
this->m_atcStationsOnline.removeByCallsign(callsign);
|
|
emit this->changedAtcStationsOnline();
|
|
emit this->changedAtcStationOnlineConnectionStatus(removedStation, false);
|
|
}
|
|
|
|
// booked
|
|
this->m_atcStationsBooked.applyIf(&CAtcStation::getCallsign, callsign, CPropertyIndexVariantMap(CAtcStation::IndexIsOnline, CVariant::from(false)));
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_atisReceived(const CCallsign &callsign, const CInformationMessage &atisMessage)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
if (!this->m_connected || callsign.isEmpty()) return;
|
|
CPropertyIndexVariantMap vm(CAtcStation::IndexAtis, atisMessage.toCVariant());
|
|
int changedOnline = this->m_atcStationsOnline.applyIf(&CAtcStation::getCallsign, callsign, vm);
|
|
|
|
// receiving an ATIS means station is online, update in bookings
|
|
vm.addValue(CAtcStation::IndexIsOnline, true);
|
|
int changedBooked = this->m_atcStationsBooked.applyIf(&CAtcStation::getCallsign, callsign, vm);
|
|
if (changedOnline > 0) { emit this->changedAtcStationsOnline(); }
|
|
if (changedBooked > 0) { emit this->changedAtcStationsBooked(); }
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_atisVoiceRoomReceived(const CCallsign &callsign, const QString &url)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
if (!this->m_connected) { return; }
|
|
QString trimmedUrl = url.trimmed();
|
|
CPropertyIndexVariantMap vm({ CAtcStation::IndexVoiceRoom, CVoiceRoom::IndexUrl }, trimmedUrl);
|
|
int changedOnline = this->m_atcStationsOnline.applyIf(&CAtcStation::getCallsign, callsign, vm, true);
|
|
if (changedOnline < 1) { return; }
|
|
|
|
Q_ASSERT(changedOnline == 1);
|
|
CAtcStation station = this->m_atcStationsOnline.findFirstByCallsign(callsign);
|
|
emit this->changedAtcStationsOnline();
|
|
emit this->changedAtcStationOnlineConnectionStatus(station, true); // send when voice room url is available
|
|
|
|
vm.addValue(CAtcStation::IndexIsOnline, true); // with voice room ATC is online
|
|
int changedBooked = this->m_atcStationsBooked.applyIf(&CAtcStation::getCallsign, callsign, vm, true);
|
|
if (changedBooked > 0)
|
|
{
|
|
// receiving a voice room means station is online, update in bookings
|
|
emit this->changedAtcStationsBooked();
|
|
}
|
|
|
|
// receiving voice room means ATC has voice
|
|
vm = CPropertyIndexVariantMap(CClient::IndexVoiceCapabilities, CVoiceCapabilities::fromVoiceCapabilities(CVoiceCapabilities::Voice).toCVariant());
|
|
this->m_otherClients.applyIf(&CClient::getCallsign, callsign, vm, false);
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_atisLogoffTimeReceived(const CCallsign &callsign, const QString &zuluTime)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
if (!this->m_connected) { return; }
|
|
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));
|
|
|
|
CPropertyIndexVariantMap vm(CAtcStation::IndexBookedUntil, CVariant(logoffDateTime));
|
|
int changedOnline = this->m_atcStationsOnline.applyIf(&CAtcStation::getCallsign, callsign, vm, true);
|
|
int changedBooked = this->m_atcStationsBooked.applyIf(&CAtcStation::getCallsign, callsign, vm, true);
|
|
|
|
if (changedOnline > 0) { emit changedAtcStationsOnline(); }
|
|
if (changedBooked > 0) { emit changedAtcStationsBooked(); }
|
|
}
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_icaoCodesReceived(const CCallsign &callsign, const CAircraftIcaoData &icaoData)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
Q_ASSERT(!callsign.isEmpty());
|
|
if (!this->m_connected) { return; }
|
|
|
|
// update
|
|
CPropertyIndexVariantMap vm(CAircraft::IndexIcao, icaoData.toCVariant());
|
|
if (!icaoData.hasAircraftDesignator())
|
|
{
|
|
// empty so far, try to fetch from data file
|
|
CLogMessage(this).warning("Empty ICAO info for %1 %2") << callsign.toQString() << icaoData.toQString();
|
|
CAircraftIcaoData icaoDataFromDataFile = this->m_vatsimDataFileReader->getIcaoInfo(callsign);
|
|
if (!icaoDataFromDataFile.hasAircraftDesignator()) { return; } // give up!
|
|
vm = CPropertyIndexVariantMap(CAircraft::IndexIcao, icaoDataFromDataFile.toCVariant());
|
|
}
|
|
// ICAO code received when aircraft is already removed or not yet ready
|
|
// We add it to cache and use it when aircraft is created
|
|
int c;
|
|
{
|
|
QWriteLocker l(&m_lockAircraft);
|
|
if (!this->m_aircraftInRange.containsCallsign(callsign))
|
|
{
|
|
this->m_icaoCodeCache.insert(callsign, icaoData);
|
|
return;
|
|
}
|
|
|
|
// update
|
|
c = this->m_aircraftInRange.applyIfCallsign(callsign, vm);
|
|
}
|
|
if (c > 0) { ps_sendReadyForModelMatching(callsign, 1); }
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_aircraftUpdateReceived(const CAircraftSituation &situation, const CTransponder &transponder)
|
|
{
|
|
Q_ASSERT_X(BlackCore::isCurrentThreadCreatingThread(this), Q_FUNC_INFO, "Called in different thread");
|
|
if (!this->m_connected) { return; }
|
|
|
|
CCallsign callsign(situation.getCallsign());
|
|
Q_ASSERT_X(!callsign.isEmpty(), Q_FUNC_INFO, "Empty callsign");
|
|
|
|
// store situation history
|
|
this->storeAircraftSituation(situation);
|
|
emit this->addedAircraftSituation(situation);
|
|
|
|
QWriteLocker l(&m_lockAircraft);
|
|
bool exists = this->m_aircraftInRange.containsCallsign(callsign);
|
|
if (!exists)
|
|
{
|
|
// new aircraft
|
|
CSimulatedAircraft aircraft;
|
|
aircraft.setCallsign(callsign);
|
|
aircraft.setSituation(situation);
|
|
aircraft.setTransponder(transponder);
|
|
aircraft.calculcateDistanceAndBearingToOwnAircraft(getOwnAircraftPosition()); // distance from myself
|
|
|
|
// ICAO from cache if avialable
|
|
bool setIcao = false;
|
|
if (this->m_icaoCodeCache.contains(callsign))
|
|
{
|
|
CAircraftIcaoData icao = this->m_icaoCodeCache.value(callsign);
|
|
this->m_icaoCodeCache.remove(callsign);
|
|
aircraft.setIcaoInfo(icao);
|
|
setIcao = true;
|
|
}
|
|
|
|
this->m_vatsimDataFileReader->updateWithVatsimDataFileData(aircraft);
|
|
|
|
// only place where aircraft is added
|
|
this->m_aircraftInRange.push_back(aircraft);
|
|
|
|
// new client, there is a chance it has been already created by custom packet
|
|
if (!this->m_otherClients.containsCallsign(callsign))
|
|
{
|
|
CClient c(callsign);
|
|
this->m_otherClients.push_back(c); // initial, will be filled by data later
|
|
}
|
|
|
|
// only if still connected
|
|
if (this->m_network->isConnected())
|
|
{
|
|
// the order here makes some sense, as we hope to receive ICAO codes last, and everthing else already in place
|
|
|
|
// Send a custom FSinn query only if we don't have the exact model yet
|
|
CClient c = this->m_otherClients.findByCallsign(callsign).frontOrDefault();
|
|
if (c.getAircraftModel().getModelType() != CAircraftModel::TypeQueriedFromNetwork)
|
|
{
|
|
this->m_network->sendCustomFsinnQuery(callsign);
|
|
}
|
|
this->m_network->sendFrequencyQuery(callsign);
|
|
this->m_network->sendRealNameQuery(callsign);
|
|
this->m_network->sendCapabilitiesQuery(callsign);
|
|
this->m_network->sendServerQuery(callsign);
|
|
|
|
// keep as last
|
|
if (setIcao)
|
|
{
|
|
this->fireDelayedReadyForModelMatching(callsign);
|
|
}
|
|
else
|
|
{
|
|
this->m_network->sendIcaoCodesQuery(callsign);
|
|
}
|
|
emit this->addedAircraft(aircraft);
|
|
} // connected
|
|
}
|
|
else
|
|
{
|
|
// update, aircraft already exists
|
|
CLength distance = getOwnAircraft().calculateGreatCircleDistance(situation.getPosition());
|
|
distance.switchUnit(CLengthUnit::NM());
|
|
CPropertyIndexVariantMap vm;
|
|
vm.addValue(CAircraft::IndexTransponder, transponder);
|
|
vm.addValue(CAircraft::IndexSituation, situation);
|
|
vm.addValue(CAircraft::IndexDistanceToOwnAircraft, distance);
|
|
|
|
// here I expect always a changed value
|
|
this->m_aircraftInRange.applyIfCallsign(callsign, vm);
|
|
}
|
|
|
|
emit this->changedAircraftInRange();
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_aircraftInterimUpdateReceived(const CAircraftSituation &situation)
|
|
{
|
|
Q_ASSERT_X(BlackCore::isCurrentThreadCreatingThread(this), Q_FUNC_INFO, "Called in different thread");
|
|
if (!this->m_connected) { return; }
|
|
|
|
CCallsign callsign(situation.getCallsign());
|
|
Q_ASSERT_X(!callsign.isEmpty(), Q_FUNC_INFO, "Empty callsign");
|
|
|
|
// todo: Check if the timestamp is copied here as well.
|
|
|
|
// Interim packets do not have groundspeed, hence set the last known value.
|
|
// If there is no full position available yet, throw this interim position away.
|
|
CAircraftSituation iterimSituation(situation);
|
|
{
|
|
QReadLocker l(&m_lockSituations);
|
|
auto history = this->m_situationsByCallsign[callsign];
|
|
if (history.empty()) { return; } // we need one full situation
|
|
iterimSituation.setCurrentUtcTime();
|
|
iterimSituation.setGroundspeed(history.latestValue().getGroundSpeed());
|
|
}
|
|
|
|
// store situation history
|
|
this->storeAircraftSituation(iterimSituation);
|
|
emit this->addedAircraftSituation(iterimSituation);
|
|
|
|
// update aircraft
|
|
//! \todo skip aircraft updates for interim positions as for performance reasons
|
|
CLength distance = getOwnAircraft().calculateGreatCircleDistance(iterimSituation.getPosition());
|
|
distance.switchUnit(CLengthUnit::NM()); // lloks nicer
|
|
CPropertyIndexVariantMap vm;
|
|
vm.addValue(CAircraft::IndexSituation, iterimSituation);
|
|
vm.addValue(CAircraft::IndexDistanceToOwnAircraft, distance);
|
|
|
|
// here I expect always a changed value
|
|
emit this->changedAircraftInRange();
|
|
QWriteLocker l(&m_lockAircraft);
|
|
this->m_aircraftInRange.applyIfCallsign(callsign, vm);
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_pilotDisconnected(const CCallsign &callsign)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
|
|
// in case of inconsistencies I always remove here
|
|
this->m_otherClients.removeByCallsign(callsign);
|
|
this->removeFromAircraftCaches(callsign);
|
|
|
|
// block for lock scope
|
|
{
|
|
QWriteLocker l1(&m_lockParts);
|
|
QWriteLocker l2(&m_lockSituations);
|
|
m_situationsByCallsign.remove(callsign);
|
|
m_partsByCallsign.remove(callsign);
|
|
m_aircraftSupportingParts.remove(callsign);
|
|
}
|
|
|
|
bool containsCallsign;
|
|
{
|
|
QWriteLocker l(&m_lockAircraft);
|
|
containsCallsign = this->m_aircraftInRange.containsCallsign(callsign);
|
|
if (containsCallsign) { this->m_aircraftInRange.removeByCallsign(callsign); }
|
|
}
|
|
|
|
if (containsCallsign) { emit this->removedAircraft(callsign); }
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_frequencyReceived(const CCallsign &callsign, const CFrequency &frequency)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
|
|
// update
|
|
int changed;
|
|
CPropertyIndexVariantMap vm({CAircraft::IndexCom1System, CComSystem::IndexActiveFrequency}, frequency.toCVariant());
|
|
{
|
|
QWriteLocker l(&m_lockAircraft);
|
|
changed = this->m_aircraftInRange.applyIf(&CAircraft::getCallsign, callsign, vm, true);
|
|
}
|
|
if (changed > 0) { emit this->changedAircraftInRange(); }
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_aircraftConfigReceived(const BlackMisc::Aviation::CCallsign &callsign, const QJsonObject &jsonObject, bool isFull)
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
|
|
CSimulatedAircraft simAircraft(getAircraftInRangeForCallsign(callsign));
|
|
|
|
// If we are not yet synchronized, we throw away any incremental packet
|
|
if (!simAircraft.hasValidCallsign()) { return; }
|
|
if (!simAircraft.isPartsSynchronized() && !isFull) { return; }
|
|
|
|
CAircraftParts parts;
|
|
if (isFull)
|
|
{
|
|
parts.convertFromJson(jsonObject);
|
|
}
|
|
else
|
|
{
|
|
// incremental update
|
|
parts = this->remoteAircraftParts(callsign).findFirstByCallsign(callsign);
|
|
QJsonObject config = applyIncrementalObject(parts.toJson(), jsonObject);
|
|
parts.convertFromJson(config);
|
|
}
|
|
|
|
// make sure in any case right time / callsign
|
|
parts.setCurrentUtcTime();
|
|
parts.setCallsign(callsign);
|
|
|
|
// store part history (parts always absolute)
|
|
this->storeAircraftParts(parts);
|
|
emit this->addedAircraftParts(parts);
|
|
|
|
// here I expect always a changed value
|
|
QWriteLocker l(&m_lockAircraft);
|
|
this->m_aircraftInRange.setAircraftParts(callsign, parts);
|
|
}
|
|
|
|
void CAirspaceMonitor::ps_sendInterimPositions()
|
|
{
|
|
Q_ASSERT(BlackCore::isCurrentThreadCreatingThread(this));
|
|
if (!this->m_connected || !m_sendInterimPositions) { return; }
|
|
CSimulatedAircraftList aircrafts = m_aircraftInRange.findBy(&CSimulatedAircraft::fastPositionUpdates, true);
|
|
m_network->sendInterimPositions(aircrafts.getCallsigns());
|
|
}
|
|
|
|
void CAirspaceMonitor::storeAircraftSituation(const CAircraftSituation &situation)
|
|
{
|
|
const CCallsign callsign(situation.getCallsign());
|
|
Q_ASSERT_X(!callsign.isEmpty(), Q_FUNC_INFO, "empty callsign");
|
|
if (callsign.isEmpty()) { return; }
|
|
|
|
// list from new to old
|
|
QWriteLocker lock(&m_lockSituations);
|
|
CAircraftSituationList &l = this->m_situationsByCallsign[callsign];
|
|
l.push_frontMaxElements(situation, MaxSituationsPerCallsign);
|
|
|
|
// check sort order
|
|
Q_ASSERT_X(l.size() < 2 || l[0].getMSecsSinceEpoch() >= l[1].getMSecsSinceEpoch(), Q_FUNC_INFO, "wrong sort order");
|
|
}
|
|
|
|
void CAirspaceMonitor::storeAircraftParts(const CAircraftParts &parts)
|
|
{
|
|
const CCallsign callsign(parts.getCallsign());
|
|
Q_ASSERT_X(!callsign.isEmpty(), "storeAircraftParts", "empty callsign");
|
|
if (callsign.isEmpty()) { return; }
|
|
|
|
// list sorted from new to old
|
|
QWriteLocker lock(&m_lockParts);
|
|
CAircraftPartsList &l = this->m_partsByCallsign[callsign];
|
|
l.push_frontMaxElements(parts, MaxPartsPerCallsign);
|
|
|
|
if (!m_aircraftSupportingParts.contains(callsign))
|
|
{
|
|
m_aircraftSupportingParts.push_back(callsign); // mark as callsign which supports parts
|
|
}
|
|
|
|
// check sort order
|
|
Q_ASSERT_X(l.size() < 2 || l[0].getMSecsSinceEpoch() >= l[1].getMSecsSinceEpoch(), "storeAircraftParts", "wrong sort order");
|
|
}
|
|
|
|
} // namespace
|