diff --git a/Configuration.cpp b/Configuration.cpp index 66c67176d..a2adce827 100644 --- a/Configuration.cpp +++ b/Configuration.cpp @@ -567,6 +567,13 @@ private: QString opCall_; QString udp_server_name_; port_type udp_server_port_; +// QString n1mm_server_name () const; + QString n1mm_server_name_; + port_type n1mm_server_port_; + bool broadcast_to_n1mm_; +// port_type n1mm_server_port () const; +// bool valid_n1mm_info () const; +// bool broadcast_to_n1mm() const; bool accept_udp_requests_; bool udpWindowToFront_; bool udpWindowRestore_; @@ -662,6 +669,9 @@ QString Configuration::opCall() const {return m_->opCall_;} QString Configuration::udp_server_name () const {return m_->udp_server_name_;} auto Configuration::udp_server_port () const -> port_type {return m_->udp_server_port_;} bool Configuration::accept_udp_requests () const {return m_->accept_udp_requests_;} +QString Configuration::n1mm_server_name () const {return m_->n1mm_server_name_;} +auto Configuration::n1mm_server_port () const -> port_type {return m_->n1mm_server_port_;} +bool Configuration::broadcast_to_n1mm () const {return m_->broadcast_to_n1mm_;} bool Configuration::udpWindowToFront () const {return m_->udpWindowToFront_;} bool Configuration::udpWindowRestore () const {return m_->udpWindowRestore_;} Bands * Configuration::bands () {return &m_->bands_;} @@ -772,6 +782,15 @@ void Configuration::sync_transceiver (bool force_signal, bool enforce_mode_and_s } } +bool Configuration::valid_n1mm_info () const +{ + // do very rudimentary checking on the n1mm server name and port number. + // + auto server_name = m_->n1mm_server_name_; + auto port_number = m_->n1mm_server_port_; + return(!(server_name.trimmed().isEmpty() || port_number == 0)); +} + namespace { #if defined (Q_OS_MAC) @@ -907,6 +926,9 @@ Configuration::impl::impl (Configuration * self, QDir const& temp_directory, ui_->udp_server_port_spin_box->setMinimum (1); ui_->udp_server_port_spin_box->setMaximum (std::numeric_limits::max ()); + ui_->n1mm_server_port_spin_box->setMinimum (1); + ui_->n1mm_server_port_spin_box->setMaximum (std::numeric_limits::max ()); + // // assign ids to radio buttons // @@ -1145,6 +1167,9 @@ void Configuration::impl::initialize_models () ui_->udp_server_line_edit->setText (udp_server_name_); ui_->udp_server_port_spin_box->setValue (udp_server_port_); ui_->accept_udp_requests_check_box->setChecked (accept_udp_requests_); + ui_->n1mm_server_name_line_edit->setText (n1mm_server_name_); + ui_->n1mm_server_port_spin_box->setValue (n1mm_server_port_); + ui_->enable_n1mm_broadcast_check_box->setChecked (broadcast_to_n1mm_); ui_->udpWindowToFront->setChecked(udpWindowToFront_); ui_->udpWindowRestore->setChecked(udpWindowRestore_); ui_->calibration_intercept_spin_box->setValue (calibration_.intercept); @@ -1350,6 +1375,9 @@ void Configuration::impl::read_settings () opCall_ = settings_->value ("OpCall", "").toString (); udp_server_name_ = settings_->value ("UDPServer", "127.0.0.1").toString (); udp_server_port_ = settings_->value ("UDPServerPort", 2237).toUInt (); + n1mm_server_name_ = settings_->value ("N1MMServer", "127.0.0.1").toString (); + n1mm_server_port_ = settings_->value ("N1MMServerPort", 2333).toUInt (); + broadcast_to_n1mm_ = settings_->value ("BroadcastToN1MM", false).toBool (); accept_udp_requests_ = settings_->value ("AcceptUDPRequests", false).toBool (); udpWindowToFront_ = settings_->value ("udpWindowToFront",false).toBool (); udpWindowRestore_ = settings_->value ("udpWindowRestore",false).toBool (); @@ -1449,6 +1477,9 @@ void Configuration::impl::write_settings () settings_->setValue ("OpCall", opCall_); settings_->setValue ("UDPServer", udp_server_name_); settings_->setValue ("UDPServerPort", udp_server_port_); + settings_->setValue ("N1MMServer", n1mm_server_name_); + settings_->setValue ("N1MMServerPort", n1mm_server_port_); + settings_->setValue ("BroadcastToN1MM", broadcast_to_n1mm_); settings_->setValue ("AcceptUDPRequests", accept_udp_requests_); settings_->setValue ("udpWindowToFront", udpWindowToFront_); settings_->setValue ("udpWindowRestore", udpWindowRestore_); @@ -1864,6 +1895,12 @@ void Configuration::impl::accept () } accept_udp_requests_ = ui_->accept_udp_requests_check_box->isChecked (); + auto new_n1mm_server = ui_->n1mm_server_name_line_edit->text (); + n1mm_server_name_ = new_n1mm_server; + auto new_n1mm_port = ui_->n1mm_server_port_spin_box->value (); + n1mm_server_port_ = new_n1mm_port; + broadcast_to_n1mm_ = ui_->enable_n1mm_broadcast_check_box->isChecked (); + udpWindowToFront_ = ui_->udpWindowToFront->isChecked (); udpWindowRestore_ = ui_->udpWindowRestore->isChecked (); diff --git a/Configuration.hpp b/Configuration.hpp index 4a876c6d9..2b5df527a 100644 --- a/Configuration.hpp +++ b/Configuration.hpp @@ -139,6 +139,10 @@ public: QString opCall() const; QString udp_server_name () const; port_type udp_server_port () const; + QString n1mm_server_name () const; + port_type n1mm_server_port () const; + bool valid_n1mm_info () const; + bool broadcast_to_n1mm() const; bool accept_udp_requests () const; bool udpWindowToFront () const; bool udpWindowRestore () const; diff --git a/Configuration.ui b/Configuration.ui index ab01a37cc..ca9443730 100644 --- a/Configuration.ui +++ b/Configuration.ui @@ -1652,7 +1652,11 @@ QListView::item:hover { - + + + <html><head/><body><p>The callsign of the operator, if different from the station callsign.</p></body></html> + + @@ -1821,6 +1825,60 @@ for assessing propagation and system performance. + + + + N1MM Logger+ Broadcasts + + + + QFormLayout::AllNonFixedFieldsGrow + + + + + <html><head/><body><p>When checked, WSJT-X will broadcast a logged contact in ADIF format to the configured hostname and port. </p></body></html> + + + Enable logged contact ADIF broadcast + + + + + + + + + <html><head/><body><p>N1MM Server name or IP address:</p></body></html> + + + + + + + <html><head/><body><p>Optional host name of N1MM Logger+ program to receive ADIF UDP broadcasts. This is usually 'localhost' or ip address 127.0.0.1</p><p>Formats:</p><ul style="margin-top: 0px; margin-bottom: 0px; margin-left: 0px; margin-right: 0px; -qt-list-indent: 1;"><li style=" margin-top:12px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">hostname</li><li style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">IPv4 address</li><li style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">IPv6 address</li><li style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">IPv4 multicast group address</li><li style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">IPv6 multicast group address</li></ul><p>Clearing this field will disable broadcasting of ADIF information via UDP.</p></body></html> + + + + + + + <html><head/><body><p>N1MM Server port number:</p></body></html> + + + + + + + <html><head/><body><p>Enter the port number that WSJT-X should use for UDP broadcasts of ADIF log information. For N1MM Logger+, this value should be 2333. If this is zero, no updates will be broadcast.</p></body></html> + + + + + + + + @@ -2688,12 +2746,12 @@ soundcard changes - - - - + + + + diff --git a/logbook/adif.cpp b/logbook/adif.cpp index a62cd62f1..3184ed3a4 100644 --- a/logbook/adif.cpp +++ b/logbook/adif.cpp @@ -14,7 +14,7 @@ void ADIF::init(QString const& filename) { _filename = filename; - _data.clear(); + _data.clear(); } @@ -165,13 +165,47 @@ QList ADIF::getCallList() const ++i; } return p; -} +} + + int ADIF::getCount() const { return _data.size(); } +QString ADIF::QSOToADIF(QString const& hisCall, QString const& hisGrid, QString const& mode, QString const& rptSent, QString const& rptRcvd, QDateTime const& dateTimeOn, QDateTime const& dateTimeOff, QString const& band, + QString const& comments, QString const& name, QString const& strDialFreq, QString const& m_myCall, QString const& m_myGrid, QString const& m_txPower) +{ + QString t; + t = "" + hisCall; + t += " " + hisGrid; + t += " " + mode; + t += " " + rptSent; + t += " " + rptRcvd; + t += " " + dateTimeOn.date().toString("yyyyMMdd"); + t += " " + dateTimeOn.time().toString("hhmmss"); + t += " " + dateTimeOff.date().toString("yyyyMMdd"); + t += " " + dateTimeOff.time().toString("hhmmss"); + t += " " + band; + t += " " + strDialFreq; + t += " " + + m_myCall; + t += " " + + m_myGrid; + if (m_txPower != "") + t += " " + m_txPower; + if (comments != "") + t += " " + comments; + if (name != "") + t += " " + name; + t += " "; + return t; +} + // open ADIF file and append the QSO details. Return true on success bool ADIF::addQSOToFile(QString const& hisCall, QString const& hisGrid, QString const& mode, QString const& rptSent, QString const& rptRcvd, QDateTime const& dateTimeOn, QDateTime const& dateTimeOff, QString const& band, @@ -187,30 +221,8 @@ bool ADIF::addQSOToFile(QString const& hisCall, QString const& hisGrid, QString out << "WSJT-X ADIF Export" << endl; // new file QString t; - t="" + hisCall; - t+=" " + hisGrid; - t+=" " + mode; - t+=" " + rptSent; - t+=" " + rptRcvd; - t+=" " + dateTimeOn.date ().toString ("yyyyMMdd"); - t+=" " + dateTimeOn.time ().toString ("hhmmss"); - t+=" " + dateTimeOff.date ().toString ("yyyyMMdd"); - t+=" " + dateTimeOff.time ().toString ("hhmmss"); - t+=" " + band; - t+=" " + strDialFreq; - t+=" " + - m_myCall; - t+=" " + - m_myGrid; - if(m_txPower!="") t+= " " + m_txPower; - if(comments!="") t+=" " + comments; - if(name!="") t+=" " + name; - if(operator_call!="") t+=" " + operator_call; - t+=" "; + t = QSOToADIF(hisCall,hisGrid,mode,rptSent,rptRcvd,dateTimeOn,dateTimeOff, + band,comments,name,strDialFreq,m_myCall,m_myGrid,m_txPower); out << t << endl; f2.close(); } diff --git a/logbook/adif.h b/logbook/adif.h index 925c9a7d7..f19712355 100644 --- a/logbook/adif.h +++ b/logbook/adif.h @@ -32,7 +32,11 @@ class ADIF bool addQSOToFile(QString const& hisCall, QString const& hisGrid, QString const& mode, QString const& rptSent, QString const& rptRcvd, QDateTime const& dateTimeOn, QDateTime const& dateTimeOff, QString const& band, QString const& comments, QString const& name, QString const& strDialFreq, QString const& m_myCall, QString const& m_myGrid, QString const& m_txPower,QString const& operator_call); - private: + QString QSOToADIF(QString const& hisCall, QString const& hisGrid, QString const& mode, QString const& rptSent, QString const& rptRcvd, QDateTime const& dateTimeOn, QDateTime const& dateTimeOff, QString const& band, + QString const& comments, QString const& name, QString const& strDialFreq, QString const& m_myCall, QString const& m_myGrid, QString const& m_txPower); + + +private: struct QSO { QString call,band,mode,date; diff --git a/logqso.cpp b/logqso.cpp index a9631d905..9118a2f63 100644 --- a/logqso.cpp +++ b/logqso.cpp @@ -5,6 +5,7 @@ #include #include #include +#include #include "logbook/adif.h" #include "MessageBox.hpp" @@ -85,6 +86,7 @@ void LogQSO::initLogQSO(QString const& hisCall, QString const& hisGrid, QString ui->band->setText (m_config->bands ()->find (dialFreq)); ui->loggedOperator->setText(opCall); show (); + QTimer::singleShot(700, this, SLOT(accept())); } void LogQSO::accept() @@ -118,6 +120,20 @@ void LogQSO::accept() tr ("Cannot open \"%1\"").arg (adifilePath)); } + // Log to N1MM Logger + if (m_config->broadcast_to_n1mm() && m_config->valid_n1mm_info()) { + QString adif = adifile.QSOToADIF(hisCall,hisGrid,mode,rptSent,rptRcvd,m_dateTimeOn,m_dateTimeOff,band,comments,name,strDialFreq,m_myCall,m_myGrid,m_txPower); + const QHostAddress n1mmhost = QHostAddress(m_config->n1mm_server_name()); + QByteArray qba_adif = adif.toLatin1(); + QUdpSocket _sock; + + auto rzult = _sock.writeDatagram ( qba_adif, n1mmhost, quint16(m_config->n1mm_server_port())); + if (rzult == -1) { + MessageBox::warning_message (this, tr ("Error sending log to N1MM"), + tr ("Write returned \"%1\"").arg (errno)); + } + } + //Log this QSO to file "wsjtx.log" static QFile f {QDir {QStandardPaths::writableLocation (QStandardPaths::DataLocation)}.absoluteFilePath ("wsjtx.log")}; if(!f.open(QIODevice::Text | QIODevice::Append)) {