mirror of
https://github.com/swift-project/pilotclient.git
synced 2026-03-31 04:25:35 +08:00
605 lines
25 KiB
C++
605 lines
25 KiB
C++
/* Copyright (C) 2013 VATSIM Community / authors
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "network_vatlib.h"
|
|
#include <vector>
|
|
#include <exception>
|
|
#include <type_traits>
|
|
|
|
static_assert(! std::is_abstract<BlackCore::NetworkVatlib>::value, "Must implement all pure virtuals");
|
|
static_assert(! std::is_abstract<BlackCore::NetworkDummy>::value, "Must implement all pure virtuals");
|
|
|
|
//TODO just placeholders to allow this to compile
|
|
#define CLIENT_NAME_VERSION "BlackBox 0.1"
|
|
#define CLIENT_VERSION_MAJOR 0
|
|
#define CLIENT_VERSION_MINOR 1
|
|
#define CLIENT_SIMULATOR_NAME "None"
|
|
#define CLIENT_PUBLIC_ID 0
|
|
#define CLIENT_PRIVATE_KEY ""
|
|
|
|
namespace BlackCore
|
|
{
|
|
|
|
using namespace BlackMisc::PhysicalQuantities;
|
|
using namespace BlackMisc::Geo;
|
|
using namespace BlackMisc::Aviation;
|
|
using namespace BlackMisc::Network;
|
|
|
|
void exceptionDispatcher(const char* caller);
|
|
|
|
NetworkVatlib::NetworkVatlib()
|
|
: m_net(Create_Cvatlib_Network()),
|
|
m_status(Cvatlib_Network::connStatus_Idle),
|
|
m_fsdTextCodec(QTextCodec::codecForName("latin1"))
|
|
{
|
|
try
|
|
{
|
|
connect(this, SIGNAL(terminate()), this, SLOT(terminateConnection()), Qt::QueuedConnection);
|
|
|
|
Q_ASSERT_X(m_fsdTextCodec, "NetworkVatlib", "Missing default wire text encoding");
|
|
//TODO reinit m_fsdTextCodec from WireTextEncoding config setting if present
|
|
|
|
QString capabilities;
|
|
capabilities += m_net->capability_AtcInfo;
|
|
capabilities += "=1:";
|
|
capabilities += m_net->capability_InterimPos;
|
|
capabilities += "=1:";
|
|
capabilities += m_net->capability_ModelDesc;
|
|
capabilities += "=1";
|
|
|
|
m_net->CreateNetworkSession(CLIENT_NAME_VERSION, CLIENT_VERSION_MAJOR, CLIENT_VERSION_MINOR,
|
|
CLIENT_SIMULATOR_NAME, CLIENT_PUBLIC_ID, CLIENT_PRIVATE_KEY, toFSD(capabilities));
|
|
|
|
m_net->InstallOnConnectionStatusChangedEvent(onConnectionStatusChanged, this);
|
|
m_net->InstallOnTextMessageReceivedEvent(onTextMessageReceived, this);
|
|
m_net->InstallOnRadioMessageReceivedEvent(onRadioMessageReceived, this);
|
|
m_net->InstallOnPilotDisconnectedEvent(onPilotDisconnected, this);
|
|
m_net->InstallOnControllerDisconnectedEvent(onControllerDisconnected, this);
|
|
m_net->InstallOnPilotPositionUpdateEvent(onPilotPositionUpdate, this);
|
|
m_net->InstallOnInterimPilotPositionUpdateEvent(onInterimPilotPositionUpdate, this);
|
|
m_net->InstallOnAtcPositionUpdateEvent(onAtcPositionUpdate, this);
|
|
m_net->InstallOnKickedEvent(onKicked, this);
|
|
m_net->InstallOnPongEvent(onPong, this);
|
|
m_net->InstallOnMetarReceivedEvent(onMetarReceived, this);
|
|
m_net->InstallOnInfoQueryRequestReceivedEvent(onInfoQueryRequestReceived, this);
|
|
m_net->InstallOnInfoQueryReplyReceivedEvent(onInfoQueryReplyReceived, this);
|
|
m_net->InstallOnCapabilitiesReplyReceivedEvent(onCapabilitiesReplyReceived, this);
|
|
m_net->InstallOnAtisReplyReceivedEvent(onAtisReplyReceived, this);
|
|
m_net->InstallOnTemperatureDataReceivedEvent(onTemperatureDataReceived, this);
|
|
m_net->InstallOnErrorReceivedEvent(onErrorReceived, this);
|
|
m_net->InstallOnWindDataReceivedEvent(onWindDataReceived, this);
|
|
m_net->InstallOnCloudDataReceivedEvent(onCloudDataReceived, this);
|
|
m_net->InstallOnPilotInfoRequestReceivedEvent(onPilotInfoRequestReceived, this);
|
|
m_net->InstallOnPilotInfoReceivedEvent(onPilotInfoReceived, this);
|
|
|
|
m_timer.start(c_updateIntervalMillisecs, this);
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
NetworkVatlib::~NetworkVatlib()
|
|
{
|
|
m_timer.stop();
|
|
|
|
try
|
|
{
|
|
if (m_net->IsNetworkConnected())
|
|
{
|
|
m_net->LogoffAndDisconnect(0);
|
|
}
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
try
|
|
{
|
|
m_net->DestroyNetworkSession();
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
void NetworkVatlib::timerEvent(QTimerEvent *)
|
|
{
|
|
try
|
|
{
|
|
if (m_net->IsValid() && m_net->IsSessionExists())
|
|
{
|
|
m_net->DoProcessing();
|
|
}
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
QByteArray NetworkVatlib::toFSD(QString qstr) const
|
|
{
|
|
return m_fsdTextCodec->fromUnicode(qstr);
|
|
}
|
|
|
|
QByteArray NetworkVatlib::toFSD(const BlackMisc::Aviation::CCallsign &callsign) const
|
|
{
|
|
return toFSD(callsign.getStringAsSet());
|
|
}
|
|
|
|
QString NetworkVatlib::fromFSD(const char *cstr) const
|
|
{
|
|
return m_fsdTextCodec->toUnicode(cstr);
|
|
}
|
|
|
|
void exceptionDispatcher(const char *caller)
|
|
{
|
|
try
|
|
{
|
|
throw;
|
|
}
|
|
catch (const NetworkNotConnectedException &e)
|
|
{
|
|
// this could be caused by a race condition during normal operation, so not an error
|
|
qDebug() << "NetworkNotConnectedException caught in " << caller << "\n" << e.what();
|
|
}
|
|
catch (const VatlibException &e)
|
|
{
|
|
qFatal("VatlibException caught in %s\n%s", caller, e.what());
|
|
}
|
|
catch (const std::exception &e)
|
|
{
|
|
qFatal("std::exception caught in %s\n%s", caller, e.what());
|
|
}
|
|
catch (...)
|
|
{
|
|
qFatal("Unknown exception caught in %s", caller);
|
|
}
|
|
}
|
|
|
|
/********************************** * * * * * * * * * * * * * * * * * * * ************************************/
|
|
/********************************** INetwork slots ************************************/
|
|
/********************************** * * * * * * * * * * * * * * * * * * * ************************************/
|
|
|
|
void NetworkVatlib::setServer(const CServer &server)
|
|
{
|
|
Q_ASSERT_X(isDisconnected(), "NetworkVatlib", "Can't change server details while still connected");
|
|
this->m_server = server;
|
|
}
|
|
|
|
void NetworkVatlib::setCallsign(const BlackMisc::Aviation::CCallsign &callsign)
|
|
{
|
|
Q_ASSERT_X(isDisconnected(), "NetworkVatlib", "Can't change callsign while still connected");
|
|
|
|
m_callsign = toFSD(callsign);
|
|
}
|
|
|
|
void NetworkVatlib::setRealName(const QString &name)
|
|
{
|
|
Q_ASSERT_X(isDisconnected(), "NetworkVatlib", "Can't change name while still connected");
|
|
|
|
m_realname = toFSD(name);
|
|
}
|
|
|
|
void NetworkVatlib::initiateConnection()
|
|
{
|
|
Q_ASSERT_X(isDisconnected(), "NetworkVatlib", "Can't connect while still connected");
|
|
|
|
try
|
|
{
|
|
m_status = Cvatlib_Network::connStatus_Connecting; // paranoia
|
|
Cvatlib_Network::PilotConnectionInfo info;
|
|
info.callsign = m_callsign.data();
|
|
info.name = m_realname.data();
|
|
info.rating = Cvatlib_Network::pilotRating_Student; //TODO
|
|
info.sim = Cvatlib_Network::simType_XPlane; //TODO
|
|
m_net->SetPilotLoginInfo(toFSD(m_server.getAddress()).data(), m_server.getPort(),
|
|
toFSD(m_server.getUser().getId()).data(), toFSD(m_server.getUser().getPassword()).data(), info);
|
|
m_net->ConnectAndLogon();
|
|
}
|
|
catch (...)
|
|
{
|
|
m_status = Cvatlib_Network::connStatus_Idle;
|
|
exceptionDispatcher(Q_FUNC_INFO);
|
|
}
|
|
}
|
|
|
|
void NetworkVatlib::terminateConnection()
|
|
{
|
|
try
|
|
{
|
|
m_net->LogoffAndDisconnect(c_logoffTimeoutSeconds);
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
void NetworkVatlib::sendTextMessages(const BlackMisc::Network::CTextMessageList &messages)
|
|
{
|
|
if (messages.isEmpty()) return;
|
|
try
|
|
{
|
|
CTextMessageList privateMessages = messages.getPrivateMessages();
|
|
foreach(BlackMisc::Network::CTextMessage message, privateMessages)
|
|
{
|
|
if (message.getToCallsign().isEmpty()) continue;
|
|
m_net->SendPrivateTextMessage(toFSD(message.getToCallsign()), toFSD(message.getMessage()));
|
|
}
|
|
CTextMessageList radioMessages = messages.getRadioMessages();
|
|
if (radioMessages.isEmpty()) return;
|
|
foreach(BlackMisc::Network::CTextMessage message, radioMessages)
|
|
{
|
|
// I could send the same message to n frequencies in one step
|
|
// if this is really required, I need to group by message
|
|
// currently I send individual messages
|
|
std::vector<INT> freqsVec;
|
|
freqsVec.push_back(message.getFrequency().value(CFrequencyUnit::kHz()));
|
|
m_net->SendRadioTextMessage(freqsVec.size(), freqsVec.data(), toFSD(message.getMessage()));
|
|
}
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
void NetworkVatlib::sendIpQuery()
|
|
{
|
|
try
|
|
{
|
|
m_net->SendInfoQuery(Cvatlib_Network::infoQuery_IP, "SERVER");
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
void NetworkVatlib::sendFrequencyQuery(const BlackMisc::Aviation::CCallsign &callsign)
|
|
{
|
|
try
|
|
{
|
|
m_net->SendInfoQuery(Cvatlib_Network::infoQuery_Freq, toFSD(callsign));
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
void NetworkVatlib::sendServerQuery(const BlackMisc::Aviation::CCallsign &callsign)
|
|
{
|
|
try
|
|
{
|
|
m_net->SendInfoQuery(Cvatlib_Network::infoQuery_Server, toFSD(callsign));
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
void NetworkVatlib::sendAtcQuery(const BlackMisc::Aviation::CCallsign &callsign)
|
|
{
|
|
try
|
|
{
|
|
m_net->SendInfoQuery(Cvatlib_Network::infoQuery_ATC, toFSD(callsign));
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
void NetworkVatlib::sendAtisQuery(const BlackMisc::Aviation::CCallsign &callsign)
|
|
{
|
|
try
|
|
{
|
|
m_net->SendInfoQuery(Cvatlib_Network::infoQuery_ATIS, toFSD(callsign));
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
void NetworkVatlib::sendNameQuery(const BlackMisc::Aviation::CCallsign &callsign)
|
|
{
|
|
try
|
|
{
|
|
m_net->SendInfoQuery(Cvatlib_Network::infoQuery_Name, toFSD(callsign));
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
void NetworkVatlib::sendCapabilitiesQuery(const BlackMisc::Aviation::CCallsign &callsign)
|
|
{
|
|
try
|
|
{
|
|
m_net->SendInfoQuery(Cvatlib_Network::infoQuery_Capabilities, toFSD(callsign));
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
void NetworkVatlib::replyToFrequencyQuery(const BlackMisc::Aviation::CCallsign &callsign, const BlackMisc::PhysicalQuantities::CFrequency &freq)
|
|
{
|
|
try
|
|
{
|
|
m_net->ReplyToInfoQuery(Cvatlib_Network::infoQuery_Freq, toFSD(callsign), toFSD(QString::number(freq.value(CFrequencyUnit::MHz()), 'f', 3)));
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
void NetworkVatlib::replyToNameQuery(const BlackMisc::Aviation::CCallsign &callsign, const QString &realname)
|
|
{
|
|
try
|
|
{
|
|
m_net->ReplyToInfoQuery(Cvatlib_Network::infoQuery_Name, toFSD(callsign), toFSD(realname));
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
void NetworkVatlib::requestAircraftInfo(const BlackMisc::Aviation::CCallsign &callsign)
|
|
{
|
|
try
|
|
{
|
|
m_net->RequestPlaneInfo(toFSD(callsign));
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
void NetworkVatlib::sendAircraftInfo(const BlackMisc::Aviation::CCallsign &callsign, const BlackMisc::Aviation::CAircraftIcao &icao)
|
|
{
|
|
try
|
|
{
|
|
const QByteArray acTypeICAObytes = toFSD(icao.getDesignator());
|
|
const QByteArray airlineICAObytes = toFSD(icao.getAirline());
|
|
const QByteArray liverybytes = toFSD(icao.getLivery());
|
|
std::vector<const char *> keysValues;
|
|
if (!icao.getDesignator().isEmpty())
|
|
{
|
|
keysValues.push_back(m_net->acinfo_Equipment);
|
|
keysValues.push_back(acTypeICAObytes);
|
|
}
|
|
if (icao.hasAirline())
|
|
{
|
|
keysValues.push_back(m_net->acinfo_Airline);
|
|
keysValues.push_back(airlineICAObytes);
|
|
}
|
|
if (icao.hasLivery())
|
|
{
|
|
keysValues.push_back(m_net->acinfo_Livery);
|
|
keysValues.push_back(liverybytes);
|
|
}
|
|
keysValues.push_back(nullptr);
|
|
m_net->SendPlaneInfo(toFSD(callsign), keysValues.data());
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
void NetworkVatlib::ping(const BlackMisc::Aviation::CCallsign &callsign)
|
|
{
|
|
try
|
|
{
|
|
m_net->PingUser(toFSD(callsign));
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
void NetworkVatlib::requestMetar(const QString &airportICAO)
|
|
{
|
|
try
|
|
{
|
|
m_net->RequestMetar(toFSD(airportICAO));
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
void NetworkVatlib::requestWeatherData(const QString &airportICAO)
|
|
{
|
|
try
|
|
{
|
|
m_net->RequestWeatherData(toFSD(airportICAO));
|
|
}
|
|
catch (...) { exceptionDispatcher(Q_FUNC_INFO); }
|
|
}
|
|
|
|
/********************************** * * * * * * * * * * * * * * * * * * * ************************************/
|
|
/********************************** shimlib callbacks ************************************/
|
|
/********************************** * * * * * * * * * * * * * * * * * * * ************************************/
|
|
|
|
NetworkVatlib *cbvar_cast(void *cbvar)
|
|
{
|
|
return static_cast<NetworkVatlib *>(cbvar);
|
|
}
|
|
|
|
void NetworkVatlib::onConnectionStatusChanged(Cvatlib_Network *, Cvatlib_Network::connStatus, Cvatlib_Network::connStatus newStatus, void *cbvar)
|
|
{
|
|
cbvar_cast(cbvar)->m_status = newStatus;
|
|
switch (newStatus)
|
|
{
|
|
case Cvatlib_Network::connStatus_Idle: emit cbvar_cast(cbvar)->connectionStatusIdle(); break;
|
|
case Cvatlib_Network::connStatus_Connecting: emit cbvar_cast(cbvar)->connectionStatusConnecting(); break;
|
|
case Cvatlib_Network::connStatus_Connected: emit cbvar_cast(cbvar)->connectionStatusConnected(); break;
|
|
case Cvatlib_Network::connStatus_Disconnected: emit cbvar_cast(cbvar)->connectionStatusDisconnected(); break;
|
|
case Cvatlib_Network::connStatus_Error: emit cbvar_cast(cbvar)->connectionStatusError(); break;
|
|
}
|
|
}
|
|
|
|
void NetworkVatlib::onTextMessageReceived(Cvatlib_Network *, const char *from, const char *to, const char *msg, void *cbvar)
|
|
{
|
|
BlackMisc::Network::CTextMessageList messages(cbvar_cast(cbvar)->fromFSD(msg), CCallsign(cbvar_cast(cbvar)->fromFSD(from)), CCallsign(cbvar_cast(cbvar)->fromFSD(to)));
|
|
emit cbvar_cast(cbvar)->textMessagesReceived(messages);
|
|
}
|
|
|
|
void NetworkVatlib::onRadioMessageReceived(Cvatlib_Network *, const char *from, INT numFreq, INT *freqList, const char *msg, void *cbvar)
|
|
{
|
|
QList<CFrequency> frequencies;
|
|
for (int i = 0; i < numFreq; ++i)
|
|
{
|
|
frequencies.push_back(CFrequency(freqList[i], CFrequencyUnit::kHz()));
|
|
}
|
|
BlackMisc::Network::CTextMessageList messages(cbvar_cast(cbvar)->fromFSD(msg), frequencies, CCallsign(cbvar_cast(cbvar)->fromFSD(from)));
|
|
emit cbvar_cast(cbvar)->textMessagesReceived(messages);
|
|
}
|
|
|
|
void NetworkVatlib::onPilotDisconnected(Cvatlib_Network *, const char *callsign, void *cbvar)
|
|
{
|
|
emit cbvar_cast(cbvar)->pilotDisconnected(cbvar_cast(cbvar)->fromFSD(callsign));
|
|
}
|
|
|
|
void NetworkVatlib::onControllerDisconnected(Cvatlib_Network *, const char *callsign, void *cbvar)
|
|
{
|
|
emit cbvar_cast(cbvar)->atcDisconnected(cbvar_cast(cbvar)->fromFSD(callsign));
|
|
}
|
|
|
|
void NetworkVatlib::onPilotPositionUpdate(Cvatlib_Network *, const char *callsignChar , Cvatlib_Network::PilotPosUpdate pos, void *cbvar)
|
|
{
|
|
const CCallsign callsign(callsignChar);
|
|
const CAircraftSituation situation(
|
|
CCoordinateGeodetic(pos.lat, pos.lon, 0.0),
|
|
CAltitude(pos.altTrue, CAltitude::AboveGround, CLengthUnit::ft()),
|
|
CHeading(pos.heading, CHeading::True, CAngleUnit::deg()),
|
|
CAngle(pos.pitch, CAngleUnit::deg()),
|
|
CAngle(pos.bank, CAngleUnit::deg()),
|
|
CSpeed(pos.groundSpeed, CSpeedUnit::kts())
|
|
);
|
|
|
|
QString tn("transponder ");
|
|
tn.append(callsign.asString());
|
|
CTransponder::TransponderMode mode = CTransponder::ModeS;
|
|
switch (pos.xpdrMode)
|
|
{
|
|
case Cvatlib_Network::xpndrMode_Normal:
|
|
mode = CTransponder::ModeC;
|
|
break;
|
|
case Cvatlib_Network::xpndrMode_Standby:
|
|
mode = CTransponder::ModeS;
|
|
break;
|
|
case Cvatlib_Network::xpndrMode_Ident:
|
|
mode = CTransponder::StateIdent;
|
|
break;
|
|
default:
|
|
mode = CTransponder::ModeC;
|
|
break;
|
|
}
|
|
CTransponder transponder(tn, pos.xpdrCode, mode);
|
|
emit cbvar_cast(cbvar)->aircraftPositionUpdate(callsign, situation, transponder);
|
|
}
|
|
|
|
void NetworkVatlib::onInterimPilotPositionUpdate(Cvatlib_Network *, const char * /** callsign **/, Cvatlib_Network::PilotPosUpdate /** pos **/, void * /** cbvar **/)
|
|
{
|
|
//TODO
|
|
}
|
|
|
|
void NetworkVatlib::onAtcPositionUpdate(Cvatlib_Network *, const char *callsign, Cvatlib_Network::ATCPosUpdate pos, void *cbvar)
|
|
{
|
|
CFrequency freq(pos.frequency, CFrequencyUnit::kHz());
|
|
freq.switchUnit(CFrequencyUnit::MHz()); // we would not need to bother, but this makes it easier to identify
|
|
emit cbvar_cast(cbvar)->atcPositionUpdate(cbvar_cast(cbvar)->fromFSD(callsign), freq,
|
|
CCoordinateGeodetic(pos.lat, pos.lon, 0), CLength(pos.visibleRange, CLengthUnit::NM()));
|
|
}
|
|
|
|
void NetworkVatlib::onKicked(Cvatlib_Network *, const char *reason, void *cbvar)
|
|
{
|
|
emit cbvar_cast(cbvar)->kicked(cbvar_cast(cbvar)->fromFSD(reason));
|
|
}
|
|
|
|
void NetworkVatlib::onPong(Cvatlib_Network *, const char *callsign, INT elapsedTime, void *cbvar)
|
|
{
|
|
emit cbvar_cast(cbvar)->pong(cbvar_cast(cbvar)->fromFSD(callsign), CTime(elapsedTime, CTimeUnit::s()));
|
|
}
|
|
|
|
void NetworkVatlib::onMetarReceived(Cvatlib_Network *, const char *data, void *cbvar)
|
|
{
|
|
emit cbvar_cast(cbvar)->metarReceived(cbvar_cast(cbvar)->fromFSD(data));
|
|
}
|
|
|
|
void NetworkVatlib::onInfoQueryRequestReceived(Cvatlib_Network *, const char *callsign, Cvatlib_Network::infoQuery type, const char *, void *cbvar)
|
|
{
|
|
switch (type)
|
|
{
|
|
case Cvatlib_Network::infoQuery_Freq: emit cbvar_cast(cbvar)->frequencyQueryRequestReceived(cbvar_cast(cbvar)->fromFSD(callsign)); break;
|
|
case Cvatlib_Network::infoQuery_Name: emit cbvar_cast(cbvar)->nameQueryRequestReceived(cbvar_cast(cbvar)->fromFSD(callsign)); break;
|
|
}
|
|
}
|
|
|
|
void NetworkVatlib::onInfoQueryReplyReceived(Cvatlib_Network *, const char *callsign, Cvatlib_Network::infoQuery type, const char *data, const char *data2, void *cbvar)
|
|
{
|
|
switch (type)
|
|
{
|
|
case Cvatlib_Network::infoQuery_Freq: emit cbvar_cast(cbvar)->frequencyQueryReplyReceived(cbvar_cast(cbvar)->fromFSD(callsign), CFrequency(cbvar_cast(cbvar)->fromFSD(data).toFloat(), CFrequencyUnit::MHz())); break;
|
|
case Cvatlib_Network::infoQuery_Server: emit cbvar_cast(cbvar)->serverQueryReplyReceived(cbvar_cast(cbvar)->fromFSD(callsign), cbvar_cast(cbvar)->fromFSD(data)); break;
|
|
case Cvatlib_Network::infoQuery_ATC: emit cbvar_cast(cbvar)->atcQueryReplyReceived(cbvar_cast(cbvar)->fromFSD(data2), *data == 'Y'); break;
|
|
case Cvatlib_Network::infoQuery_Name: emit cbvar_cast(cbvar)->nameQueryReplyReceived(cbvar_cast(cbvar)->fromFSD(callsign), cbvar_cast(cbvar)->fromFSD(data)); break;
|
|
case Cvatlib_Network::infoQuery_IP: emit cbvar_cast(cbvar)->ipQueryReplyReceived(cbvar_cast(cbvar)->fromFSD(data)); break;
|
|
}
|
|
}
|
|
|
|
void NetworkVatlib::onCapabilitiesReplyReceived(Cvatlib_Network *net, const char *callsign, const char **keysValues, void *cbvar)
|
|
{
|
|
quint32 flags = 0;
|
|
while (*keysValues)
|
|
{
|
|
const char *key = keysValues[0];
|
|
const char *value = keysValues[1];
|
|
if (*value == '1')
|
|
{
|
|
if (key == net->capability_AtcInfo) { flags |= AcceptsAtisResponses; }
|
|
else if (key == net->capability_InterimPos) { flags |= SupportsInterimPosUpdates; }
|
|
else if (key == net->capability_ModelDesc) { flags |= SupportsModelDescriptions; }
|
|
}
|
|
keysValues += 2;
|
|
}
|
|
emit cbvar_cast(cbvar)->capabilitiesQueryReplyReceived(cbvar_cast(cbvar)->fromFSD(callsign), flags);
|
|
}
|
|
|
|
void NetworkVatlib::onAtisReplyReceived(Cvatlib_Network *, const char *callsign, Cvatlib_Network::atisLineType, const char *data, void *cbvar)
|
|
{
|
|
emit cbvar_cast(cbvar)->atisQueryReplyReceived(cbvar_cast(cbvar)->fromFSD(callsign), cbvar_cast(cbvar)->fromFSD(data));
|
|
}
|
|
|
|
void NetworkVatlib::onTemperatureDataReceived(Cvatlib_Network *, Cvatlib_Network::TempLayer /** layers **/ [4], INT /** pressure **/, void * /** cbvar **/)
|
|
{
|
|
//TODO
|
|
}
|
|
|
|
void NetworkVatlib::onErrorReceived(Cvatlib_Network *, Cvatlib_Network::error type, const char *msg, const char *data, void *cbvar)
|
|
{
|
|
switch (type)
|
|
{
|
|
case Cvatlib_Network::error_CallsignTaken: qCritical() << "The requested callsign is already taken"; goto terminate;
|
|
case Cvatlib_Network::error_CallsignInvalid: qCritical() << "The requested callsign is not valid"; goto terminate;
|
|
case Cvatlib_Network::error_CIDPasswdInvalid: qCritical() << "Wrong user ID or password"; goto terminate;
|
|
case Cvatlib_Network::error_ProtoVersion: qCritical() << "This server does not support our protocol version"; goto terminate;
|
|
case Cvatlib_Network::error_LevelTooHigh: qCritical() << "You are not authorized to use the requested pilot rating"; goto terminate;
|
|
case Cvatlib_Network::error_ServerFull: qCritical() << "The server is full"; goto terminate;
|
|
case Cvatlib_Network::error_CIDSuspended: qCritical() << "Your user account is suspended"; goto terminate;
|
|
case Cvatlib_Network::error_InvalidPosition: qCritical() << "You are not authorized to use the requested pilot rating"; goto terminate;
|
|
case Cvatlib_Network::error_SoftwareNotAuthorized: qCritical() << "This client software has not been authorized for use on this network"; goto terminate;
|
|
|
|
case Cvatlib_Network::error_Ok: break;
|
|
case Cvatlib_Network::error_Syntax: qWarning() << "Malformed packet: Syntax error: %s" << cbvar_cast(cbvar)->fromFSD(data); break;
|
|
case Cvatlib_Network::error_SourceInvalid: qDebug() << "Server: source invalid (" << cbvar_cast(cbvar)->fromFSD(data); break;
|
|
case Cvatlib_Network::error_CallsignNotExists: qDebug() << "Shim lib: " << cbvar_cast(cbvar)->fromFSD(msg) << " (" << cbvar_cast(cbvar)->fromFSD(data) << ")"; break;
|
|
case Cvatlib_Network::error_NoFP: qDebug() << "Server: no flight plan"; break;
|
|
case Cvatlib_Network::error_NoWeather: qDebug() << "Server: requested weather profile does not exist"; break;
|
|
|
|
// we have no idea what these mean
|
|
case Cvatlib_Network::error_Registered:
|
|
case Cvatlib_Network::error_InvalidControl: qWarning() << "Server: " << cbvar_cast(cbvar)->fromFSD(msg); break;
|
|
|
|
default: qFatal("VATSIM shim library: %s (error %d)", msg, type); goto terminate;
|
|
}
|
|
|
|
return;
|
|
terminate:
|
|
emit cbvar_cast(cbvar)->terminate();
|
|
}
|
|
|
|
void NetworkVatlib::onWindDataReceived(Cvatlib_Network *, Cvatlib_Network::WindLayer /** layers **/[4], void * /** cbvar **/)
|
|
{
|
|
//TODO
|
|
}
|
|
|
|
void NetworkVatlib::onCloudDataReceived(Cvatlib_Network *, Cvatlib_Network::CloudLayer /** layers **/ [2], Cvatlib_Network::StormLayer /** storm **/, float /** vis **/, void * /** cbvar **/)
|
|
{
|
|
//TODO
|
|
}
|
|
|
|
void NetworkVatlib::onPilotInfoRequestReceived(Cvatlib_Network *, const char *callsign, void *cbvar)
|
|
{
|
|
emit cbvar_cast(cbvar)->aircraftInfoRequestReceived(cbvar_cast(cbvar)->fromFSD(callsign));
|
|
}
|
|
|
|
void NetworkVatlib::onPilotInfoReceived(Cvatlib_Network *net, const char *callsign, const char **keysValues, void *cbvar)
|
|
{
|
|
BlackMisc::Aviation::CAircraftIcao icao;
|
|
while (*keysValues)
|
|
{
|
|
QString key(*keysValues);
|
|
keysValues++;
|
|
if (key == net->acinfo_Equipment) { icao.setDesignator(*keysValues); }
|
|
else if (key == net->acinfo_Airline) { icao.setAirline(*keysValues); }
|
|
else if (key == net->acinfo_Livery) { icao.setLivery(*keysValues); }
|
|
keysValues++;
|
|
}
|
|
emit cbvar_cast(cbvar)->aircraftInfoReceived(cbvar_cast(cbvar)->fromFSD(callsign), icao);
|
|
}
|
|
|
|
} // namespace
|