mirror of
https://github.com/swift-project/pilotclient.git
synced 2026-04-01 13:36:48 +08:00
Deals with a special topic: While frequencies at VATSIM/flightsim are usually set in 25kHz spacing, some ATC stations (radar) require 8.33kHz tuning. The now made extension handles this and allows to listen to a voice room even if the corresponding frequency is not exactly tuned in. Example: EKDK_CTR 135,270 but needs to be tuned in as 135,275
124 lines
4.9 KiB
C++
124 lines
4.9 KiB
C++
/* Copyright (C) 2013 VATSIM Community / contributors
|
|
* 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 "blackmisc/aviocomsystem.h"
|
|
|
|
using namespace BlackMisc::PhysicalQuantities;
|
|
|
|
namespace BlackMisc
|
|
{
|
|
namespace Aviation
|
|
{
|
|
/*
|
|
* Valid values?
|
|
*/
|
|
bool CComSystem::validValues() const
|
|
{
|
|
if (this->isDefaultValue()) return true; // special case
|
|
return
|
|
(CComSystem::isValidCivilAviationFrequency(this->getFrequencyActive()) ||
|
|
CComSystem::isValidMilitaryFrequency(this->getFrequencyActive())) &&
|
|
(CComSystem::isValidCivilAviationFrequency(this->getFrequencyStandby()) ||
|
|
CComSystem::isValidMilitaryFrequency(this->getFrequencyStandby()));
|
|
}
|
|
|
|
/*
|
|
* Validate
|
|
*/
|
|
bool CComSystem::validate(bool strict) const
|
|
{
|
|
if (this->isDefaultValue()) return true;
|
|
bool valid = this->validValues();
|
|
if (!strict) return valid;
|
|
Q_ASSERT_X(valid, "CComSystem::validate", "illegal values");
|
|
if (!valid) throw std::range_error("Illegal values in CComSystem::validate");
|
|
return true;
|
|
}
|
|
|
|
void CComSystem::setFrequencyActiveMHz(double frequencyMHz)
|
|
{
|
|
CFrequency f(frequencyMHz, CFrequencyUnit::MHz());
|
|
if (f == this->getFrequencyActive()) return; // save all the comparisons / rounding
|
|
CComSystem::roundToChannelSpacing(f, this->m_channelSpacing);
|
|
this->CModulator::setFrequencyActive(f);
|
|
this->validate(true);
|
|
}
|
|
|
|
void CComSystem::setFrequencyStandbyMHz(double frequencyMHz)
|
|
{
|
|
CFrequency f(frequencyMHz, CFrequencyUnit::MHz());
|
|
if (f == this->getFrequencyStandby()) return; // save all the comparisons / rounding
|
|
CComSystem::roundToChannelSpacing(f, this->m_channelSpacing);
|
|
this->CModulator::setFrequencyStandby(f);
|
|
this->validate(true);
|
|
}
|
|
|
|
/*
|
|
* Round to channel spacing
|
|
*/
|
|
void CComSystem::roundToChannelSpacing(PhysicalQuantities::CFrequency &frequency, ChannelSpacing channelSpacing)
|
|
{
|
|
double channelSpacingKHz = CComSystem::channelSpacingToFrequencyKHz(channelSpacing);
|
|
double f = frequency.valueRounded(CFrequencyUnit::kHz(), 0);
|
|
quint32 d = static_cast<quint32>(f / channelSpacingKHz);
|
|
frequency.switchUnit(CFrequencyUnit::MHz());
|
|
double f0 = frequency.valueRounded(CFrequencyUnit::MHz(), 3);
|
|
double f1 = CMath::round(d * (channelSpacingKHz / 1000.0), 3);
|
|
double f2 = CMath::round((d + 1) * (channelSpacingKHz / 1000.0), 3);
|
|
bool down = qAbs(f1 - f0) < qAbs(f2 - f0); // which is the closest value
|
|
frequency.setCurrentUnitValue(down ? f1 : f2);
|
|
}
|
|
|
|
/*
|
|
* Within channel spacing
|
|
*/
|
|
bool CComSystem::isWithinChannelSpacing(const CFrequency &setFrequency, const CFrequency &compareFrequency, CComSystem::ChannelSpacing channelSpacing)
|
|
{
|
|
if (setFrequency == compareFrequency) return true; // shortcut for many of such comparisons
|
|
double channelSpacingKHz = 0.5 * CComSystem::channelSpacingToFrequencyKHz(channelSpacing);
|
|
double compareFrequencyKHz = compareFrequency.value(CFrequencyUnit::kHz());
|
|
double setFrequencyKHz = setFrequency.value(CFrequencyUnit::kHz());
|
|
return (setFrequencyKHz - channelSpacingKHz < compareFrequencyKHz) &&
|
|
(setFrequencyKHz + channelSpacingKHz > compareFrequencyKHz);
|
|
}
|
|
|
|
/*
|
|
* Helper, give me number for channels spacing
|
|
*/
|
|
double CComSystem::channelSpacingToFrequencyKHz(ChannelSpacing channelSpacing)
|
|
{
|
|
switch (channelSpacing)
|
|
{
|
|
case ChannelSpacing50KHz: return 50.0;
|
|
case ChannelSpacing25KHz: return 25.0;
|
|
case ChannelSpacing8_33KHz: return 25.0 / 3.0;
|
|
default: qFatal("Wrong channel spacing"); return 0.0; // return just supressing compiler warning
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Marshall
|
|
*/
|
|
void CComSystem::marshallToDbus(QDBusArgument &argument) const
|
|
{
|
|
CModulator::marshallToDbus(argument);
|
|
argument << static_cast<uint>(this->m_channelSpacing);
|
|
}
|
|
|
|
/*
|
|
* Unmarshall
|
|
*/
|
|
void CComSystem::unmarshallFromDbus(const QDBusArgument &argument)
|
|
{
|
|
CModulator::unmarshallFromDbus(argument);
|
|
uint cs;
|
|
argument >> cs;
|
|
this->m_channelSpacing = static_cast<ChannelSpacing>(cs);
|
|
}
|
|
|
|
|
|
} // namespace
|
|
}
|