2018-11-11 21:11:58 -05:00
|
|
|
#include "CabrilloLog.hpp"
|
|
|
|
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <utility>
|
|
|
|
#include <QString>
|
|
|
|
#include <QDateTime>
|
|
|
|
#include <QSqlDatabase>
|
|
|
|
#include <QSqlTableModel>
|
|
|
|
#include <QSqlRecord>
|
|
|
|
#include <QSqlError>
|
|
|
|
#include <QSqlQuery>
|
2018-11-12 15:10:27 -05:00
|
|
|
#include <QDataStream>
|
2018-11-11 21:11:58 -05:00
|
|
|
#include "Configuration.hpp"
|
|
|
|
#include "Bands.hpp"
|
|
|
|
#include "qt_db_helpers.hpp"
|
|
|
|
#include "pimpl_impl.hpp"
|
|
|
|
|
|
|
|
class CabrilloLog::impl final
|
|
|
|
: public QSqlTableModel
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
impl (Configuration const *);
|
|
|
|
|
2019-05-03 05:21:50 -04:00
|
|
|
QVariant data (QModelIndex const& index, int role) const
|
|
|
|
{
|
|
|
|
auto value = QSqlTableModel::data (index, role);
|
|
|
|
if (index.column () == fieldIndex ("when")
|
|
|
|
&& (Qt::DisplayRole == role || Qt::EditRole == role))
|
|
|
|
{
|
|
|
|
auto t = QDateTime::fromMSecsSinceEpoch (value.toULongLong () * 1000ull, Qt::UTC);
|
|
|
|
if (Qt::DisplayRole == role)
|
|
|
|
{
|
|
|
|
QLocale locale;
|
|
|
|
return locale.toString (t, locale.dateFormat (QLocale::ShortFormat) + " hh:mm:ss");
|
|
|
|
}
|
|
|
|
value = t;
|
|
|
|
}
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2019-01-21 17:03:33 -05:00
|
|
|
QString cabrillo_frequency_string (Radio::Frequency frequency) const;
|
|
|
|
|
2018-11-11 21:11:58 -05:00
|
|
|
Configuration const * configuration_;
|
|
|
|
QSqlQuery mutable dupe_query_;
|
|
|
|
QSqlQuery mutable export_query_;
|
|
|
|
};
|
|
|
|
|
|
|
|
CabrilloLog::impl::impl (Configuration const * configuration)
|
|
|
|
: QSqlTableModel {}
|
|
|
|
, configuration_ {configuration}
|
|
|
|
{
|
|
|
|
if (!database ().tables ().contains ("cabrillo_log"))
|
|
|
|
{
|
|
|
|
QSqlQuery query;
|
|
|
|
SQL_error_check (query, static_cast<bool (QSqlQuery::*) (QString const&)> (&QSqlQuery::exec),
|
|
|
|
"CREATE TABLE cabrillo_log ("
|
|
|
|
" id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,"
|
|
|
|
" frequency INTEGER NOT NULL,"
|
|
|
|
" \"when\" DATETIME NOT NULL,"
|
|
|
|
" call VARCHAR(20) NOT NULL,"
|
|
|
|
" exchange_sent VARCHAR(32) NOT NULL,"
|
|
|
|
" exchange_rcvd VARCHAR(32) NOT NULL,"
|
|
|
|
" band VARCHAR(6) NOT NULL"
|
|
|
|
")");
|
|
|
|
}
|
|
|
|
|
|
|
|
SQL_error_check (dupe_query_, &QSqlQuery::prepare,
|
2019-01-21 08:35:18 -05:00
|
|
|
"SELECT "
|
|
|
|
" COUNT(*) "
|
|
|
|
" FROM "
|
|
|
|
" cabrillo_log "
|
|
|
|
" WHERE "
|
|
|
|
" call = :call "
|
|
|
|
" AND band = :band");
|
2018-11-11 21:11:58 -05:00
|
|
|
|
|
|
|
SQL_error_check (export_query_, &QSqlQuery::prepare,
|
2019-01-21 08:35:18 -05:00
|
|
|
"SELECT "
|
|
|
|
" frequency"
|
|
|
|
" , \"when\""
|
|
|
|
" , exchange_sent"
|
|
|
|
" , call"
|
|
|
|
" , exchange_rcvd"
|
|
|
|
" FROM "
|
|
|
|
" cabrillo_log "
|
|
|
|
" ORDER BY "
|
|
|
|
" \"when\"");
|
2018-11-11 21:11:58 -05:00
|
|
|
|
2018-12-01 21:30:32 -05:00
|
|
|
setEditStrategy (QSqlTableModel::OnFieldChange);
|
2018-11-11 21:11:58 -05:00
|
|
|
setTable ("cabrillo_log");
|
2018-11-11 23:10:57 -05:00
|
|
|
setHeaderData (fieldIndex ("frequency"), Qt::Horizontal, tr ("Freq(kHz)"));
|
|
|
|
setHeaderData (fieldIndex ("when"), Qt::Horizontal, tr ("Date & Time(UTC)"));
|
2018-11-11 21:11:58 -05:00
|
|
|
setHeaderData (fieldIndex ("call"), Qt::Horizontal, tr ("Call"));
|
|
|
|
setHeaderData (fieldIndex ("exchange_sent"), Qt::Horizontal, tr ("Sent"));
|
|
|
|
setHeaderData (fieldIndex ("exchange_rcvd"), Qt::Horizontal, tr ("Rcvd"));
|
|
|
|
setHeaderData (fieldIndex ("band"), Qt::Horizontal, tr ("Band"));
|
2018-12-06 00:41:16 -05:00
|
|
|
|
|
|
|
// This descending order by time is important, it makes the view
|
|
|
|
// place the latest row at the top, without this the model/view
|
|
|
|
// interactions are both sluggish and unhelpful.
|
|
|
|
setSort (fieldIndex ("when"), Qt::DescendingOrder);
|
|
|
|
|
2018-11-11 21:11:58 -05:00
|
|
|
SQL_error_check (*this, &QSqlTableModel::select);
|
|
|
|
}
|
|
|
|
|
2019-01-21 17:03:33 -05:00
|
|
|
// frequency here is in kHz
|
|
|
|
QString CabrilloLog::impl::cabrillo_frequency_string (Radio::Frequency frequency) const
|
|
|
|
{
|
|
|
|
QString result;
|
|
|
|
auto band = configuration_->bands ()->find (frequency * 1000ull);
|
|
|
|
if ("1mm" == band) result = "LIGHT";
|
|
|
|
else if ("2mm" == band) result = "241G";
|
|
|
|
else if ("2.5mm" == band) result = "134G";
|
|
|
|
else if ("4mm" == band) result = "75G";
|
|
|
|
else if ("6mm" == band) result = "47G";
|
|
|
|
else if ("1.25cm" == band) result = "24G";
|
|
|
|
else if ("3cm" == band) result = "10G";
|
|
|
|
else if ("6cm" == band) result = "5.7G";
|
|
|
|
else if ("9cm" == band) result = "3.4G";
|
|
|
|
else if ("13cm" == band) result = "2.3G";
|
|
|
|
else if ("23cm" == band) result = "1.2G";
|
|
|
|
else if ("33cm" == band) result = "902";
|
|
|
|
else if ("70cm" == band) result = "432";
|
|
|
|
else if ("1.25m" == band) result = "222";
|
|
|
|
else if ("2m" == band) result = "144";
|
|
|
|
else if ("4m" == band) result = "70";
|
|
|
|
else if ("6m" == band) result = "50";
|
|
|
|
else result = QString::number (frequency);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2018-11-11 21:11:58 -05:00
|
|
|
CabrilloLog::CabrilloLog (Configuration const * configuration)
|
|
|
|
: m_ {configuration}
|
|
|
|
{
|
|
|
|
Q_ASSERT (configuration);
|
|
|
|
}
|
|
|
|
|
|
|
|
CabrilloLog::~CabrilloLog ()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2018-11-25 17:19:41 -05:00
|
|
|
QSqlTableModel * CabrilloLog::model ()
|
2018-11-11 21:11:58 -05:00
|
|
|
{
|
|
|
|
return &*m_;
|
|
|
|
}
|
|
|
|
|
2018-11-22 20:18:39 -05:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
void set_value_maybe_null (QSqlRecord& record, QString const& name, QString const& value)
|
|
|
|
{
|
|
|
|
if (value.size ())
|
|
|
|
{
|
|
|
|
record.setValue (name, value);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
record.setNull (name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CabrilloLog::add_QSO (Frequency frequency, QDateTime const& when, QString const& call
|
2018-11-11 21:11:58 -05:00
|
|
|
, QString const& exchange_sent, QString const& exchange_received)
|
|
|
|
{
|
|
|
|
auto record = m_->record ();
|
|
|
|
record.setValue ("frequency", frequency / 1000ull); // kHz
|
2018-11-22 20:18:39 -05:00
|
|
|
if (!when.isNull ())
|
|
|
|
{
|
|
|
|
record.setValue ("when", when.toMSecsSinceEpoch () / 1000ull);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
record.setNull ("when");
|
|
|
|
}
|
|
|
|
set_value_maybe_null (record, "call", call);
|
|
|
|
set_value_maybe_null (record, "exchange_sent", exchange_sent);
|
|
|
|
set_value_maybe_null (record, "exchange_rcvd", exchange_received);
|
|
|
|
set_value_maybe_null (record, "band", m_->configuration_->bands ()->find (frequency));
|
2018-12-01 21:30:32 -05:00
|
|
|
if (m_->isDirty ())
|
|
|
|
{
|
|
|
|
m_->revert (); // discard any uncommitted changes
|
|
|
|
}
|
2018-12-06 00:41:16 -05:00
|
|
|
m_->setEditStrategy (QSqlTableModel::OnManualSubmit);
|
|
|
|
ConditionalTransaction transaction {*m_};
|
2018-11-25 17:19:41 -05:00
|
|
|
auto ok = m_->insertRecord (-1, record);
|
2018-12-06 00:41:16 -05:00
|
|
|
transaction.submit ();
|
|
|
|
m_->setEditStrategy (QSqlTableModel::OnFieldChange);
|
2018-11-25 17:19:41 -05:00
|
|
|
return ok;
|
2018-11-11 21:11:58 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CabrilloLog::dupe (Frequency frequency, QString const& call) const
|
|
|
|
{
|
|
|
|
m_->dupe_query_.bindValue (":call", call);
|
|
|
|
m_->dupe_query_.bindValue (":band", m_->configuration_->bands ()->find (frequency));
|
|
|
|
SQL_error_check (m_->dupe_query_, static_cast<bool (QSqlQuery::*) ()> (&QSqlQuery::exec));
|
|
|
|
m_->dupe_query_.next ();
|
|
|
|
return m_->dupe_query_.value (0).toInt ();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CabrilloLog::reset ()
|
|
|
|
{
|
2019-01-21 08:35:18 -05:00
|
|
|
// synchronize model
|
|
|
|
while (m_->canFetchMore ()) m_->fetchMore ();
|
2018-11-11 23:10:57 -05:00
|
|
|
if (m_->rowCount ())
|
|
|
|
{
|
2018-11-25 17:19:41 -05:00
|
|
|
m_->setEditStrategy (QSqlTableModel::OnManualSubmit);
|
2018-11-11 23:10:57 -05:00
|
|
|
ConditionalTransaction transaction {*m_};
|
|
|
|
SQL_error_check (*m_, &QSqlTableModel::removeRows, 0, m_->rowCount (), QModelIndex {});
|
|
|
|
transaction.submit ();
|
2018-11-25 17:19:41 -05:00
|
|
|
m_->select (); // to refresh views
|
2018-12-01 21:30:32 -05:00
|
|
|
m_->setEditStrategy (QSqlTableModel::OnFieldChange);
|
2018-11-11 23:10:57 -05:00
|
|
|
}
|
2018-11-11 21:11:58 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void CabrilloLog::export_qsos (QTextStream& stream) const
|
|
|
|
{
|
|
|
|
SQL_error_check (m_->export_query_, static_cast<bool (QSqlQuery::*) ()> (&QSqlQuery::exec));
|
|
|
|
auto record = m_->export_query_.record ();
|
|
|
|
auto frequency_index = record.indexOf ("frequency");
|
|
|
|
auto when_index = record.indexOf ("when");
|
|
|
|
auto call_index = record.indexOf ("call");
|
|
|
|
auto sent_index = record.indexOf ("exchange_sent");
|
|
|
|
auto rcvd_index = record.indexOf ("exchange_rcvd");
|
|
|
|
while (m_->export_query_.next ())
|
|
|
|
{
|
|
|
|
auto my_call = m_->configuration_->my_callsign ();
|
|
|
|
stream << QString {"QSO: %1 DG %2 %3 %4 %5 %6\n"}
|
2019-01-21 17:03:33 -05:00
|
|
|
.arg (m_->cabrillo_frequency_string (m_->export_query_.value (frequency_index).value<Radio::Frequency> ()), 5)
|
|
|
|
.arg (QDateTime::fromMSecsSinceEpoch (m_->export_query_.value (when_index).toULongLong () * 1000ull, Qt::UTC).toString ("yyyy-MM-dd hhmm"))
|
2018-11-11 21:11:58 -05:00
|
|
|
.arg (my_call, -12)
|
|
|
|
.arg (m_->export_query_.value (sent_index).toString (), -13)
|
|
|
|
.arg (m_->export_query_.value (call_index).toString (), -12)
|
|
|
|
.arg (m_->export_query_.value (rcvd_index).toString (), -13);
|
|
|
|
}
|
|
|
|
}
|