2023-05-01 20:56:52 +00:00
|
|
|
#include <QDebug>
|
|
|
|
#include "logcategories.h"
|
|
|
|
|
|
|
|
#include "memories.h"
|
|
|
|
#include "ui_memories.h"
|
|
|
|
|
|
|
|
memories::memories(rigCapabilities rigCaps, QWidget *parent) :
|
|
|
|
QDialog(parent),
|
|
|
|
rigCaps(rigCaps),
|
|
|
|
ui(new Ui::memories)
|
|
|
|
{
|
|
|
|
ui->setupUi(this);
|
|
|
|
ui->table->setColumnCount(totalColumns);
|
|
|
|
QStringList headers;
|
2023-05-02 20:18:41 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
columnRecall=0,
|
|
|
|
columnNum,
|
2023-05-04 20:55:40 +00:00
|
|
|
columnScan,
|
2023-05-02 20:18:41 +00:00
|
|
|
columnName,
|
|
|
|
columnVFO,
|
|
|
|
columnFrequency,
|
|
|
|
columnMode,
|
|
|
|
columnFilter,
|
|
|
|
columnData,
|
|
|
|
columnDuplex,
|
|
|
|
columnToneMode,
|
|
|
|
columnDSQL,
|
|
|
|
columnTone,
|
|
|
|
columnTSQL,
|
|
|
|
columnDTCS,
|
|
|
|
columnDTCSPolarity,
|
|
|
|
columnOffset,
|
|
|
|
columnUR,
|
|
|
|
columnR1,
|
|
|
|
columnR2,
|
2023-05-04 20:55:40 +00:00
|
|
|
columnFrequencyB,
|
|
|
|
columnModeB,
|
|
|
|
columnFilterB,
|
|
|
|
columnDataB,
|
|
|
|
columnToneModeB,
|
|
|
|
columnDSQLB,
|
|
|
|
columnToneB,
|
|
|
|
columnTSQLB,
|
|
|
|
columnDTCSPolarityB,
|
|
|
|
columnDTCSB,
|
|
|
|
columnDVSquelchB,
|
|
|
|
columnURB,
|
|
|
|
columnR1B,
|
|
|
|
columnR2B,
|
|
|
|
|
2023-05-02 20:18:41 +00:00
|
|
|
*/
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
headers << "" << "Num" << "Scan" << "Name" << "VFO" << "Freq" << "Mode" << "Data" << "Filter" << "Duplex" << "Tn Mode" << "Tone" << "TSQL" <<
|
|
|
|
"DTCS" << "DTCS Pol" << "DSQL" << "DV Sql" << "Offset" << "UR" << "R1" << "R2" << "Freq B" << "Mode B" << "Data B" << "Filter B" <<
|
|
|
|
"Tn Mode B" << "Tone B" << "TSQL B" << "DTCSP B" << "DTCS B" << "DSQL B" << "DV Sql B" << "UR B" << "R1 B" << "R2 B";
|
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
ui->table->setHorizontalHeaderLabels(headers);
|
2023-05-02 20:18:41 +00:00
|
|
|
|
2023-05-03 12:38:06 +00:00
|
|
|
ui->groupLabel->hide();
|
|
|
|
ui->group->hide();
|
|
|
|
ui->vfoMode->hide();
|
|
|
|
ui->memoryMode->hide();
|
2023-05-02 20:18:41 +00:00
|
|
|
|
|
|
|
ui->group->blockSignals(true);
|
|
|
|
for (int i=1;i<=rigCaps.memGroups;i++) {
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->group->addItem(QString("Group %0").arg(i,2,10,QChar('0')),i);
|
2023-05-01 20:56:52 +00:00
|
|
|
}
|
2023-05-03 18:08:20 +00:00
|
|
|
if (rigCaps.commands.contains(funcSatelliteMode)){
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->group->addItem("Satellite",200);
|
2023-05-03 18:08:20 +00:00
|
|
|
}
|
2023-05-02 20:18:41 +00:00
|
|
|
ui->group->setCurrentIndex(-1);
|
|
|
|
ui->group->blockSignals(false);
|
2023-05-01 20:56:52 +00:00
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
scan = QStringList({"OFF","*1","*2","*3"});
|
2023-05-02 20:18:41 +00:00
|
|
|
|
2023-05-03 12:52:15 +00:00
|
|
|
dataModes = QStringList({"OFF","DATA1"});
|
2023-05-02 20:18:41 +00:00
|
|
|
if (rigCaps.commands.contains(funcDATA2Mod))
|
|
|
|
dataModes.append("DATA2");
|
|
|
|
if (rigCaps.commands.contains(funcDATA3Mod))
|
|
|
|
dataModes.append("DATA3");
|
|
|
|
|
2023-05-03 12:52:15 +00:00
|
|
|
filters = QStringList({"FIL1","FIL2","FIL3"});
|
2023-05-02 20:18:41 +00:00
|
|
|
|
2023-05-03 12:52:15 +00:00
|
|
|
duplexModes = QStringList({"OFF","DUP-","DUP+","RPS"});
|
2023-05-02 20:18:41 +00:00
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
toneModes = QStringList({"OFF","TONE","TSQL"});
|
2023-05-02 20:18:41 +00:00
|
|
|
if (rigCaps.commands.contains(funcRepeaterDTCS))
|
|
|
|
toneModes.append("DTCS");
|
|
|
|
|
2023-05-03 12:52:15 +00:00
|
|
|
tones = QStringList({"67.0","69.3","71.9","74.4","77.0","79.7","82.5","85.4","88.5","91.5","94.8","97.4","100.0","103.5","107.2","110.9","114.8","118.8","123.0","127.3","131.8","136.5",
|
2023-05-02 20:18:41 +00:00
|
|
|
"141.3","146.2","151.4","156.7","159.8","162.2","165.5","167.9","171.3","173.8","177.3","179.9","183.5","186.2","189.9","192.8","196.6","199.5","203.5","206.5",
|
2023-05-03 12:52:15 +00:00
|
|
|
"210.7","218.1","225.7","229.1","233.6","241.8","250.3","254.1"});
|
2023-05-02 20:18:41 +00:00
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
dtcs = QStringList({"023","025","026","031","032","036","043","047","051","053","054","065","071","072","073","074","114","115","116","122","125","131","132","134","143","145","152","155","156","162","165",
|
|
|
|
"172","174","205","212","223","225","226","243","244","245","246","251","252","255","261","263","265","266","271","274","306","311","315","325","331","332","343","346",
|
|
|
|
"351","356","364","365","371","411","412","413","423","431","432","445","446","452","454","455","462","464","465","466","503","506","516","523","526","532","546","565",
|
|
|
|
"606","612","624","627","631","632","654","662","664","703","712","723","731","732","734","743","754"});
|
2023-05-01 20:56:52 +00:00
|
|
|
if (rigCaps.commands.contains(funcVFOEqualAB)) {
|
2023-05-03 12:52:15 +00:00
|
|
|
VFO = QStringList({"VFOA","VFOB"});
|
2023-05-01 20:56:52 +00:00
|
|
|
} else if (rigCaps.commands.contains(funcVFOEqualMS)) {
|
2023-05-04 20:55:40 +00:00
|
|
|
VFO = QStringList({"MAIN","SUB"});
|
2023-05-01 20:56:52 +00:00
|
|
|
}
|
2023-05-02 20:18:41 +00:00
|
|
|
|
2023-05-03 12:52:15 +00:00
|
|
|
dsql = QStringList({"OFF","DSQL","CSQL"});
|
2023-05-04 20:55:40 +00:00
|
|
|
dtcsp = QStringList({"BOTH N","N/R","R/N","BOTH R"});
|
2023-05-01 20:56:52 +00:00
|
|
|
|
|
|
|
foreach (auto mode, rigCaps.modes){
|
|
|
|
modes.append(mode.name);
|
|
|
|
}
|
|
|
|
|
2023-05-02 20:18:41 +00:00
|
|
|
numEditor = new tableEditor(QRegularExpression("[0-9]{0,3}"),ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnNum, numEditor);
|
|
|
|
|
|
|
|
if (rigCaps.memGroups>1)
|
2023-05-03 12:38:06 +00:00
|
|
|
nameEditor = new tableEditor(QRegularExpression("[0-9A-Za-z\\/\\ ]{0,16}$"),ui->table);
|
2023-05-02 20:18:41 +00:00
|
|
|
else
|
2023-05-03 12:38:06 +00:00
|
|
|
nameEditor = new tableEditor(QRegularExpression("[0-9A-Za-z\\/\\ ]{0,10}$"),ui->table);
|
2023-05-01 20:56:52 +00:00
|
|
|
|
2023-05-02 20:18:41 +00:00
|
|
|
ui->table->setItemDelegateForColumn(columnName, nameEditor);
|
2023-05-01 20:56:52 +00:00
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
scanList = new tableCombobox(createModel(scanModel, scan),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnScan, scanList);
|
2023-05-01 20:56:52 +00:00
|
|
|
|
|
|
|
vfoList = new tableCombobox(createModel(vfoModel, VFO),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnVFO, vfoList);
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
freqEditor = new tableEditor(QRegularExpression("[0-9.]{0,10}"),ui->table);
|
2023-05-02 20:18:41 +00:00
|
|
|
ui->table->setItemDelegateForColumn(columnFrequency, freqEditor);
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
freqEditorB = new tableEditor(QRegularExpression("[0-9.]{0,10}"),ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnFrequencyB, freqEditorB);
|
2023-05-02 20:18:41 +00:00
|
|
|
|
|
|
|
modesList = new tableCombobox(createModel(modesModel, modes),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnMode, modesList);
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
modesListB = new tableCombobox(createModel(modesModelB, modes),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnModeB, modesListB);
|
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
dataList = new tableCombobox(createModel(dataModel, dataModes),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnData, dataList);
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
dataListB = new tableCombobox(createModel(dataModelB, dataModes),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnDataB, dataListB);
|
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
filterList = new tableCombobox(createModel(filterModel, filters),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnFilter, filterList);
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
filterListB = new tableCombobox(createModel(filterModelB, filters),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnFilterB, filterListB);
|
|
|
|
|
2023-05-02 20:18:41 +00:00
|
|
|
duplexList = new tableCombobox(createModel(duplexModel, duplexModes),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnDuplex, duplexList);
|
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
toneModesList = new tableCombobox(createModel(toneModesModel, toneModes),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnToneMode, toneModesList);
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
toneModesListB = new tableCombobox(createModel(toneModesModelB, toneModes),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnToneModeB, toneModesListB);
|
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
tonesList = new tableCombobox(createModel(tonesModel, tones),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnTone, tonesList);
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
tonesListB = new tableCombobox(createModel(tonesModelB, tones),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnToneB, tonesListB);
|
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
tsqlList = new tableCombobox(createModel(tsqlModel, tones),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnTSQL, tsqlList);
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
tsqlListB = new tableCombobox(createModel(tsqlModelB, tones),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnTSQLB, tsqlListB);
|
|
|
|
|
2023-05-02 20:18:41 +00:00
|
|
|
dsqlList = new tableCombobox(createModel(dsqlModel, dsql),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnDSQL, dsqlList);
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
dsqlListB = new tableCombobox(createModel(dsqlModelB, dsql),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnDSQLB, dsqlListB);
|
2023-05-02 20:18:41 +00:00
|
|
|
|
|
|
|
dtcspList = new tableCombobox(createModel(dtcspModel, dtcsp),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnDTCSPolarity, dtcspList);
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
dtcsList = new tableCombobox(createModel(dtcsModel, dtcs),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnDTCS, dtcsList);
|
|
|
|
|
|
|
|
dtcspListB = new tableCombobox(createModel(dtcspModelB, dtcsp),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnDTCSPolarityB, dtcspListB);
|
|
|
|
|
|
|
|
dtcsListB = new tableCombobox(createModel(dtcsModel, dtcs),false,ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnDTCSB, dtcsList);
|
|
|
|
|
2023-05-02 20:18:41 +00:00
|
|
|
offsetEditor = new tableEditor(QRegularExpression("[0-9]{0,7}"),ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnOffset, offsetEditor);
|
|
|
|
|
|
|
|
dvsqlEditor = new tableEditor(QRegularExpression("[0-9]{0,2}"),ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnDVSquelch, dvsqlEditor);
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
dvsqlEditorB = new tableEditor(QRegularExpression("[0-9]{0,2}"),ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnDVSquelchB, dvsqlEditorB);
|
|
|
|
|
2023-05-03 12:38:06 +00:00
|
|
|
urEditor = new tableEditor(QRegularExpression("[0-9A-Z\\/\\ ]{0,8}$"),ui->table);
|
2023-05-02 20:18:41 +00:00
|
|
|
ui->table->setItemDelegateForColumn(columnUR, urEditor);
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
urEditorB = new tableEditor(QRegularExpression("[0-9A-Z\\/\\ ]{0,8}$"),ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnURB, urEditorB);
|
|
|
|
|
2023-05-03 12:38:06 +00:00
|
|
|
r1Editor = new tableEditor(QRegularExpression("[0-9A-Z\\/\\ ]{0,8}$"),ui->table);
|
2023-05-02 20:18:41 +00:00
|
|
|
ui->table->setItemDelegateForColumn(columnR1, r1Editor);
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
r1EditorB = new tableEditor(QRegularExpression("[0-9A-Z\\/\\ ]{0,8}$"),ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnR1B, r1EditorB);
|
|
|
|
|
2023-05-03 12:38:06 +00:00
|
|
|
r2Editor = new tableEditor(QRegularExpression("[0-9A-Z\\/\\ ]{0,8}$"),ui->table);
|
2023-05-02 20:18:41 +00:00
|
|
|
ui->table->setItemDelegateForColumn(columnR2, r2Editor);
|
2023-05-01 20:56:52 +00:00
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
r2EditorB = new tableEditor(QRegularExpression("[0-9A-Z\\/\\ ]{0,8}$"),ui->table);
|
|
|
|
ui->table->setItemDelegateForColumn(columnR2B, r2EditorB);
|
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
connect(ui->table,SIGNAL(rowAdded(int)),this,SLOT(rowAdded(int)));
|
2023-05-02 20:18:41 +00:00
|
|
|
connect(ui->table,SIGNAL(rowDeleted(quint32)),this,SLOT(rowDeleted(quint32)));
|
2023-05-04 20:55:40 +00:00
|
|
|
connect(&timeoutTimer, SIGNAL(timeout()), this, SLOT(timeout()));
|
2023-05-02 20:18:41 +00:00
|
|
|
|
|
|
|
ui->table->sortByColumn(0,Qt::AscendingOrder);
|
2023-05-01 20:56:52 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
void memories::populate()
|
|
|
|
{
|
2023-05-02 20:18:41 +00:00
|
|
|
ui->group->setCurrentIndex(0);
|
2023-05-01 20:56:52 +00:00
|
|
|
}
|
|
|
|
memories::~memories()
|
|
|
|
{
|
|
|
|
qInfo() << "Deleting memories table";
|
|
|
|
ui->table->clear();
|
|
|
|
delete ui;
|
|
|
|
}
|
|
|
|
|
|
|
|
void memories::rowAdded(int row)
|
|
|
|
{
|
|
|
|
// Find the next available memory number:
|
|
|
|
qInfo() << "Row Added" << row;
|
2023-05-04 20:55:40 +00:00
|
|
|
|
|
|
|
quint32 num=1;
|
|
|
|
|
|
|
|
/* This feels unnecessarily complicated, but here we:
|
|
|
|
* Set the memory number to 1
|
|
|
|
* create a vector of all current memory numbers (ignoring any that are NULL)
|
|
|
|
* If empty use 1 as the memory number
|
|
|
|
* if not, sort it into numerical order
|
|
|
|
* check if there is a gap (use that if there is)
|
|
|
|
* If not, go back to previous, check we haven't exceeded the total allowed and if not add 1 to previous
|
|
|
|
* If no empty memories available, log a warning and quit
|
|
|
|
*/
|
|
|
|
|
|
|
|
std::vector<int> rows;
|
2023-05-01 20:56:52 +00:00
|
|
|
ui->table->blockSignals(true);
|
|
|
|
for (int i=0;i < ui->table->rowCount();i++)
|
|
|
|
{
|
2023-05-04 20:55:40 +00:00
|
|
|
if (ui->table->item(i,columnNum) != NULL)
|
|
|
|
{
|
|
|
|
rows.push_back(ui->table->item(i,columnNum)->text().toInt());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rows.size()!=0)
|
|
|
|
{
|
|
|
|
std::sort(rows.begin(),rows.end());
|
|
|
|
auto i = std::adjacent_find( rows.begin(), rows.end(), [](int l, int r){return l+1<r;} );
|
|
|
|
if (i == rows.end())
|
2023-05-01 20:56:52 +00:00
|
|
|
{
|
2023-05-04 20:55:40 +00:00
|
|
|
// No gaps found so work on highest value found
|
|
|
|
i--;
|
|
|
|
if ((ui->group->currentData().toInt() != 200 && *i < rigCaps.memories-1) || (ui->group->currentData().toInt() == 200 && *i < rigCaps.satMemories-1))
|
|
|
|
{
|
|
|
|
num = 1+ *i;
|
|
|
|
} else {
|
|
|
|
qWarning() << "Sorry no free memory slots found, please delete one first";
|
|
|
|
ui->table->removeRow(row);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
num = 1+ *i;
|
|
|
|
}
|
2023-05-01 20:56:52 +00:00
|
|
|
}
|
2023-05-04 20:55:40 +00:00
|
|
|
|
|
|
|
QPushButton* recall = new QPushButton("Recall");
|
|
|
|
ui->table->setCellWidget(row,columnRecall,recall);
|
|
|
|
connect(recall, &QPushButton::clicked, this,
|
|
|
|
[=]() { qInfo() << "Recalling" << num; emit recallMemory(num);});
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnNum),QString::number(num).rightJustified(3,'0'));
|
|
|
|
// Set default values (where possible) for all other values:
|
|
|
|
if (ui->table->item(row,columnScan) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnScan),scan[0]);
|
|
|
|
if (ui->table->item(row,columnVFO) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnVFO),VFO[0]);
|
|
|
|
if (ui->table->item(row,columnData) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnData),dataModes[0]);
|
|
|
|
if (ui->table->item(row,columnDuplex) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnDuplex),duplexModes[0]);
|
|
|
|
if (ui->table->item(row,columnToneMode) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnToneMode),toneModes[0]);
|
|
|
|
if (ui->table->item(row,columnDSQL) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnDSQL),dsql[0]);
|
|
|
|
if (ui->table->item(row,columnTone) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnTone),tones[0]);
|
|
|
|
if (ui->table->item(row,columnTSQL) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnTSQL),tones[0]);
|
|
|
|
if (ui->table->item(row,columnDTCSPolarity) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnDTCSPolarity),dtcsp[0]);
|
|
|
|
if (ui->table->item(row,columnDTCS) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnDTCS),dtcs[0]);
|
|
|
|
if (ui->table->item(row,columnDVSquelch) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnDVSquelch),"00");
|
|
|
|
if (ui->table->item(row,columnOffset) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnOffset),"0");
|
|
|
|
if (ui->table->item(row,columnUR) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnUR)," ");
|
|
|
|
if (ui->table->item(row,columnR1) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnR1)," ");
|
|
|
|
if (ui->table->item(row,columnR2) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnR2)," ");
|
|
|
|
if (ui->table->item(row,columnDataB) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnDataB),dataModes[0]);
|
|
|
|
if (ui->table->item(row,columnToneModeB) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnToneModeB),toneModes[0]);
|
|
|
|
if (ui->table->item(row,columnToneB) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnToneB),tones[0]);
|
|
|
|
if (ui->table->item(row,columnTSQLB) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnTSQLB),tones[0]);
|
|
|
|
if (ui->table->item(row,columnDTCSPolarityB) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnDTCSPolarityB),dtcsp[0]);
|
|
|
|
if (ui->table->item(row,columnDTCSB) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnDTCSB),dtcs[0]);
|
|
|
|
if (ui->table->item(row,columnDVSquelchB) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnDVSquelchB),"00");
|
|
|
|
if (ui->table->item(row,columnURB) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnURB)," ");
|
|
|
|
if (ui->table->item(row,columnR1B) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnR1B)," ");
|
|
|
|
if (ui->table->item(row,columnR2B) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnR2B)," ");
|
|
|
|
|
|
|
|
ui->table->item(row,columnNum)->setFlags(ui->table->item(row,columnNum)->flags() | Qt::ItemIsEditable);
|
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
ui->table->blockSignals(false);
|
|
|
|
}
|
|
|
|
|
2023-05-02 20:18:41 +00:00
|
|
|
void memories::rowDeleted(quint32 mem)
|
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
{
|
|
|
|
if (mem && mem <= rigCaps.memories) {
|
|
|
|
qInfo() << "Mem Deleted" << mem;
|
|
|
|
emit clearMemory(mem);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void memories::on_table_cellChanged(int row, int col)
|
|
|
|
{
|
2023-05-03 18:08:20 +00:00
|
|
|
// If the import is updating a hidden column, ignore it.
|
|
|
|
if (ui->table->isColumnHidden(col))
|
|
|
|
return;
|
2023-05-01 20:56:52 +00:00
|
|
|
|
|
|
|
if (row != currentRow || currentMemory == Q_NULLPTR)
|
|
|
|
{
|
|
|
|
if (currentMemory != Q_NULLPTR)
|
|
|
|
{
|
|
|
|
delete currentMemory;
|
|
|
|
currentMemory = Q_NULLPTR;
|
|
|
|
}
|
|
|
|
currentMemory = new memoryType();
|
|
|
|
currentRow = row;
|
|
|
|
}
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
currentMemory->group = ui->group->currentData().toInt();
|
2023-05-01 20:56:52 +00:00
|
|
|
currentMemory->channel = (ui->table->item(row,columnNum) == NULL) ? 0 : ui->table->item(row,columnNum)->text().toInt();
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
if (currentMemory->group==200) {
|
|
|
|
currentMemory->sat=true;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (quint8 f=0;f<scan.size();f++)
|
2023-05-01 20:56:52 +00:00
|
|
|
{
|
2023-05-04 20:55:40 +00:00
|
|
|
if ((ui->table->item(row,columnScan) == NULL) ? false : ui->table->item(row,columnScan)->text().toUpper()==scan[f].toUpper()) {
|
|
|
|
currentMemory->scan=f;
|
2023-05-01 20:56:52 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (quint8 f=0;f<VFO.size();f++)
|
|
|
|
{
|
2023-05-04 20:55:40 +00:00
|
|
|
if ((ui->table->item(row,columnVFO) == NULL) ? false : ui->table->item(row,columnVFO)->text().toUpper()==VFO[f].toUpper()) {
|
2023-05-01 20:56:52 +00:00
|
|
|
currentMemory->frequency.VFO=(selVFO_t)f;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
currentMemory->frequency.Hz = (ui->table->item(row,columnFrequency) == NULL) ? 0 : quint64(ui->table->item(row,columnFrequency)->text().toDouble()*1000000.0);
|
|
|
|
currentMemory->frequencyB.Hz = (ui->table->item(row,columnFrequencyB) == NULL) ? 0 : quint64(ui->table->item(row,columnFrequencyB)->text().toDouble()*1000000.0);
|
2023-05-01 20:56:52 +00:00
|
|
|
|
|
|
|
foreach (auto m, rigCaps.modes){
|
|
|
|
if ((ui->table->item(row,columnMode) == NULL) ? false : ui->table->item(row,columnMode)->text()==m.name) {
|
|
|
|
currentMemory->mode=m.mk;
|
2023-05-04 20:55:40 +00:00
|
|
|
}
|
|
|
|
if ((ui->table->item(row,columnModeB) == NULL) ? false : ui->table->item(row,columnModeB)->text()==m.name) {
|
|
|
|
currentMemory->modeB=m.mk;
|
2023-05-01 20:56:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (quint8 f=0;f<dataModes.size();f++)
|
|
|
|
{
|
|
|
|
if ((ui->table->item(row,columnData) == NULL) ? false : ui->table->item(row,columnData)->text()==dataModes[f]) {
|
|
|
|
currentMemory->datamode=f;
|
2023-05-04 20:55:40 +00:00
|
|
|
}
|
|
|
|
if ((ui->table->item(row,columnDataB) == NULL) ? false : ui->table->item(row,columnDataB)->text()==dataModes[f]) {
|
|
|
|
currentMemory->datamodeB=f;
|
2023-05-01 20:56:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (quint8 f=0;f<filters.size();f++)
|
|
|
|
{
|
|
|
|
if ((ui->table->item(row,columnFilter) == NULL) ? false : ui->table->item(row,columnFilter)->text()==filters[f]) {
|
2023-05-02 20:18:41 +00:00
|
|
|
currentMemory->filter=f+1; // Filters start at 1
|
2023-05-04 20:55:40 +00:00
|
|
|
}
|
|
|
|
if ((ui->table->item(row,columnFilterB) == NULL) ? false : ui->table->item(row,columnFilterB)->text()==filters[f]) {
|
|
|
|
currentMemory->filterB=f+1; // Filters start at 1
|
2023-05-02 20:18:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (quint8 f=0;f<duplexModes.size();f++)
|
|
|
|
{
|
|
|
|
if ((ui->table->item(row,columnDuplex) == NULL) ? false : ui->table->item(row,columnDuplex)->text()==duplexModes[f]) {
|
|
|
|
currentMemory->duplex=f;
|
2023-05-01 20:56:52 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (quint8 f=0;f<toneModes.size();f++)
|
|
|
|
{
|
2023-05-04 20:55:40 +00:00
|
|
|
if ((ui->table->item(row,columnToneMode) == NULL) ? false : ui->table->item(row,columnToneMode)->text().toUpper()==toneModes[f].toUpper()) {
|
2023-05-01 20:56:52 +00:00
|
|
|
currentMemory->tonemode=f;
|
2023-05-04 20:55:40 +00:00
|
|
|
}
|
|
|
|
if ((ui->table->item(row,columnToneModeB) == NULL) ? false : ui->table->item(row,columnToneModeB)->text().toUpper()==toneModes[f].toUpper()) {
|
|
|
|
currentMemory->tonemodeB=f;
|
2023-05-01 20:56:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-02 20:18:41 +00:00
|
|
|
for (quint8 f=0;f<dsql.size();f++)
|
|
|
|
{
|
|
|
|
if ((ui->table->item(row,columnDSQL) == NULL) ? false : ui->table->item(row,columnDSQL)->text()==dsql[f]) {
|
|
|
|
currentMemory->dsql=f;
|
2023-05-04 20:55:40 +00:00
|
|
|
}
|
|
|
|
if ((ui->table->item(row,columnDSQLB) == NULL) ? false : ui->table->item(row,columnDSQLB)->text()==dsql[f]) {
|
|
|
|
currentMemory->dsqlB=f;
|
2023-05-02 20:18:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
currentMemory->tone = (ui->table->item(row,columnTone) == NULL) ? 670 : int(ui->table->item(row,columnTone)->text().toFloat()*10.0);
|
2023-05-04 20:55:40 +00:00
|
|
|
currentMemory->toneB = (ui->table->item(row,columnToneB) == NULL) ? 670 : int(ui->table->item(row,columnToneB)->text().toFloat()*10.0);
|
2023-05-01 20:56:52 +00:00
|
|
|
|
|
|
|
currentMemory->tsql = (ui->table->item(row,columnTSQL) == NULL) ? 670 : int(ui->table->item(row,columnTSQL)->text().toFloat()*10.0);
|
2023-05-04 20:55:40 +00:00
|
|
|
currentMemory->tsqlB = (ui->table->item(row,columnTSQLB) == NULL) ? 670 : int(ui->table->item(row,columnTSQLB)->text().toFloat()*10.0);
|
2023-05-01 20:56:52 +00:00
|
|
|
|
2023-05-02 20:18:41 +00:00
|
|
|
currentMemory->dtcs = (ui->table->item(row,columnDTCS) == NULL) ? 23 : int(ui->table->item(row,columnDTCS)->text().toUInt());
|
2023-05-04 20:55:40 +00:00
|
|
|
currentMemory->dtcsB = (ui->table->item(row,columnDTCSB) == NULL) ? 23 : int(ui->table->item(row,columnDTCSB)->text().toUInt());
|
2023-05-02 20:18:41 +00:00
|
|
|
|
|
|
|
for (quint8 f=0;f<dtcsp.size();f++)
|
|
|
|
{
|
2023-05-04 20:55:40 +00:00
|
|
|
if ((ui->table->item(row,columnDTCSPolarity) == NULL) ? false : ui->table->item(row,columnDTCSPolarity)->text().toUpper()==dtcsp[f].toUpper()) {
|
2023-05-02 20:18:41 +00:00
|
|
|
currentMemory->dtcsp=f;
|
2023-05-04 20:55:40 +00:00
|
|
|
}
|
|
|
|
if ((ui->table->item(row,columnDTCSPolarityB) == NULL) ? false : ui->table->item(row,columnDTCSPolarityB)->text().toUpper()==dtcsp[f].toUpper()) {
|
|
|
|
currentMemory->dtcspB=f;
|
2023-05-02 20:18:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
currentMemory->dvsql = (ui->table->item(row,columnDVSquelch) == NULL) ? 0 : int(ui->table->item(row,columnDVSquelch)->text().toUInt());
|
2023-05-04 20:55:40 +00:00
|
|
|
currentMemory->dvsqlB = (ui->table->item(row,columnDVSquelchB) == NULL) ? 0 : int(ui->table->item(row,columnDVSquelchB)->text().toUInt());
|
|
|
|
|
2023-05-02 20:18:41 +00:00
|
|
|
currentMemory->duplexOffset.Hz = (ui->table->item(row,columnOffset) == NULL) ? 0 : int(ui->table->item(row,columnOffset)->text().toULongLong());
|
|
|
|
currentMemory->duplexOffset.MHzDouble=currentMemory->duplexOffset.Hz/1000000.0;
|
|
|
|
currentMemory->duplexOffset.VFO=selVFO_t::activeVFO;
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
memcpy(currentMemory->UR,((ui->table->item(row,columnUR) == NULL) ? "" : ui->table->item(row,columnUR)->text()).trimmed().toStdString().c_str(),8);
|
|
|
|
memcpy(currentMemory->URB,((ui->table->item(row,columnURB) == NULL) ? "" : ui->table->item(row,columnURB)->text()).trimmed().toStdString().c_str(),8);
|
|
|
|
|
|
|
|
memcpy(currentMemory->R1,((ui->table->item(row,columnR1) == NULL) ? "" : ui->table->item(row,columnR1)->text()).trimmed().toStdString().c_str(),8);
|
|
|
|
memcpy(currentMemory->R1B,((ui->table->item(row,columnR1B) == NULL) ? "" : ui->table->item(row,columnR1B)->text()).trimmed().toStdString().c_str(),8);
|
2023-05-02 20:18:41 +00:00
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
memcpy(currentMemory->R2,((ui->table->item(row,columnR2) == NULL) ? "" : ui->table->item(row,columnR2)->text()).trimmed().toStdString().c_str(),8);
|
|
|
|
memcpy(currentMemory->R2B,((ui->table->item(row,columnR2B) == NULL) ? "" : ui->table->item(row,columnR2B)->text()).trimmed().toStdString().c_str(),8);
|
|
|
|
|
|
|
|
memcpy(currentMemory->name,((ui->table->item(row,columnName) == NULL) ? "" : ui->table->item(row,columnName)->text()).trimmed().toStdString().c_str(),16);
|
2023-05-02 20:18:41 +00:00
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
// Only write the memory if ALL values are non-null
|
|
|
|
bool write=true;
|
|
|
|
for (int f=1; f<ui->table->columnCount();f++)
|
|
|
|
{
|
2023-05-03 18:08:20 +00:00
|
|
|
if (!ui->table->isColumnHidden(f) && ui->table->item(row,f) == NULL)
|
2023-05-01 20:56:52 +00:00
|
|
|
write=false;
|
|
|
|
}
|
|
|
|
if (write) {
|
|
|
|
emit setMemory(*currentMemory);
|
|
|
|
// Set number to not be editable once written. Not sure why but this crashes?
|
|
|
|
//ui->table->item(row,columnNum)->setFlags(ui->table->item(row,columnNum)->flags() & (~Qt::ItemIsEditable));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-02 20:18:41 +00:00
|
|
|
void memories::on_group_currentIndexChanged(int index)
|
|
|
|
{
|
2023-05-04 20:55:40 +00:00
|
|
|
timeoutTimer.start(MEMORY_TIMEOUT);
|
|
|
|
|
2023-05-02 20:18:41 +00:00
|
|
|
ui->group->setEnabled(false);
|
|
|
|
ui->table->blockSignals(true);
|
|
|
|
ui->table->setRowCount(0);
|
|
|
|
ui->table->blockSignals(false);
|
2023-05-03 18:08:20 +00:00
|
|
|
|
|
|
|
// Hide all columns except recall
|
|
|
|
for (int i=1;i<ui->table->columnCount();i++)
|
|
|
|
{
|
|
|
|
ui->table->hideColumn(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
QVector<memParserFormat> parser;
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
if (ui->group->currentData().toInt() == 200) {
|
|
|
|
emit setSatelliteMode(true);
|
2023-05-03 18:08:20 +00:00
|
|
|
parser = rigCaps.satParser;
|
2023-05-03 18:19:58 +00:00
|
|
|
} else {
|
2023-05-04 20:55:40 +00:00
|
|
|
emit setSatelliteMode(false);
|
2023-05-03 18:19:58 +00:00
|
|
|
parser = rigCaps.memParser;
|
2023-05-03 18:08:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
foreach (auto parse, parser) {
|
|
|
|
switch (parse.spec)
|
|
|
|
{
|
|
|
|
case 'a':
|
|
|
|
ui->groupLabel->show();
|
|
|
|
ui->group->show();
|
|
|
|
ui->vfoMode->show();
|
|
|
|
ui->memoryMode->show();
|
|
|
|
break;
|
|
|
|
case 'b':
|
|
|
|
ui->table->showColumn(columnNum);
|
|
|
|
break;
|
|
|
|
case 'c':
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->showColumn(columnScan);
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
|
|
|
case 'd':
|
|
|
|
ui->table->showColumn(columnFrequency);
|
|
|
|
break;
|
2023-05-04 20:55:40 +00:00
|
|
|
case 'D':
|
|
|
|
ui->table->showColumn(columnFrequencyB);
|
|
|
|
break;
|
2023-05-03 18:08:20 +00:00
|
|
|
case 'e':
|
|
|
|
ui->table->showColumn(columnMode);
|
|
|
|
break;
|
2023-05-04 20:55:40 +00:00
|
|
|
case 'E':
|
|
|
|
ui->table->showColumn(columnModeB);
|
|
|
|
break;
|
2023-05-03 18:08:20 +00:00
|
|
|
case 'f':
|
|
|
|
ui->table->showColumn(columnFilter);
|
|
|
|
break;
|
2023-05-04 20:55:40 +00:00
|
|
|
case 'F':
|
|
|
|
ui->table->showColumn(columnFilterB);
|
|
|
|
break;
|
2023-05-03 18:08:20 +00:00
|
|
|
case 'g':
|
|
|
|
ui->table->showColumn(columnData);
|
|
|
|
break;
|
2023-05-04 20:55:40 +00:00
|
|
|
case 'G':
|
|
|
|
ui->table->showColumn(columnDataB);
|
|
|
|
break;
|
2023-05-03 18:08:20 +00:00
|
|
|
case 'h':
|
|
|
|
ui->table->showColumn(columnDuplex);
|
|
|
|
ui->table->showColumn(columnToneMode);
|
|
|
|
break;
|
|
|
|
case 'i':
|
|
|
|
ui->table->showColumn(columnData);
|
|
|
|
ui->table->showColumn(columnToneMode);
|
|
|
|
break;
|
|
|
|
case 'j':
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->showColumn(columnToneMode);
|
|
|
|
break;
|
|
|
|
case 'J':
|
|
|
|
ui->table->showColumn(columnToneModeB);
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
|
|
|
case 'k':
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->showColumn(columnDSQL);
|
|
|
|
break;
|
|
|
|
case 'K':
|
|
|
|
ui->table->showColumn(columnDSQLB);
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
|
|
|
case 'l':
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->showColumn(columnTone);
|
|
|
|
break;
|
|
|
|
case 'L':
|
|
|
|
ui->table->showColumn(columnToneB);
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
|
|
|
case 'm':
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->showColumn(columnTSQL);
|
|
|
|
break;
|
|
|
|
case 'M':
|
|
|
|
ui->table->showColumn(columnTSQLB);
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
|
|
|
case 'n':
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->showColumn(columnDTCSPolarity);
|
|
|
|
break;
|
|
|
|
case 'N':
|
|
|
|
ui->table->showColumn(columnDTCSPolarityB);
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
|
|
|
case 'o':
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->showColumn(columnDTCS);
|
|
|
|
break;
|
|
|
|
case 'O':
|
|
|
|
ui->table->showColumn(columnDTCSB);
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
|
|
|
case 'p':
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->showColumn(columnDVSquelch);
|
|
|
|
break;
|
|
|
|
case 'P':
|
|
|
|
ui->table->showColumn(columnDVSquelchB);
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
|
|
|
case 'q':
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->showColumn(columnOffset);
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
|
|
|
case 'r':
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->showColumn(columnUR);
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
2023-05-04 20:55:40 +00:00
|
|
|
case 'R':
|
|
|
|
ui->table->showColumn(columnURB);
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
2023-05-04 20:55:40 +00:00
|
|
|
case 's':
|
|
|
|
ui->table->showColumn(columnR1);
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
2023-05-04 20:55:40 +00:00
|
|
|
case 'S':
|
|
|
|
ui->table->showColumn(columnR1B);
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
2023-05-04 20:55:40 +00:00
|
|
|
case 't':
|
|
|
|
ui->table->showColumn(columnR2);
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
2023-05-04 20:55:40 +00:00
|
|
|
case 'T':
|
|
|
|
ui->table->showColumn(columnR2B);
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
2023-05-04 20:55:40 +00:00
|
|
|
case 'u':
|
|
|
|
ui->table->showColumn(columnName);
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
if (ui->group->currentData().toInt()==200) {
|
|
|
|
ui->table->horizontalHeader()->setSectionResizeMode(QHeaderView::ResizeToContents);
|
|
|
|
} else
|
2023-05-02 20:18:41 +00:00
|
|
|
{
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->horizontalHeader()->setSectionResizeMode(QHeaderView::Stretch);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ui->group->currentData().toInt() == 200) {
|
|
|
|
for (quint16 m=1;m<=rigCaps.satMemories;m++)
|
|
|
|
{
|
|
|
|
emit getSatMemory(quint32(m & 0xffff));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (quint16 m=1;m<=rigCaps.memories;m++)
|
|
|
|
{
|
|
|
|
if (rigCaps.memGroups>1)
|
|
|
|
emit getMemory(quint32((index+1)<<16) | (m & 0xffff));
|
|
|
|
else
|
|
|
|
emit getMemory(quint32(m & 0xffff));
|
|
|
|
}
|
2023-05-02 20:18:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void memories::on_vfoMode_clicked()
|
|
|
|
{
|
|
|
|
emit vfoMode();
|
|
|
|
}
|
|
|
|
|
|
|
|
void memories::on_memoryMode_clicked()
|
|
|
|
{
|
|
|
|
emit memoryMode();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
void memories::receiveMemory(memoryType mem)
|
|
|
|
{
|
2023-05-04 20:55:40 +00:00
|
|
|
timeoutTimer.start(MEMORY_TIMEOUT);
|
|
|
|
|
|
|
|
if (mem.scan < 4) {
|
2023-05-01 20:56:52 +00:00
|
|
|
ui->table->blockSignals(true);
|
2023-05-04 20:55:40 +00:00
|
|
|
qInfo(logRig()) << "Received memory" << mem.channel << "Scan" << mem.scan << "Name" << mem.name << "Freq" << mem.frequency.Hz << "Mode" << mem.mode;
|
2023-05-01 20:56:52 +00:00
|
|
|
|
|
|
|
int row=-1;
|
|
|
|
for (int n = 0; n<ui->table->rowCount();n++)
|
|
|
|
{
|
|
|
|
if (ui->table->item(n,0) != NULL && ui->table->item(n,0)->text().toInt() == mem.channel) {
|
|
|
|
row = n;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (row == -1) {
|
|
|
|
ui->table->insertRow(ui->table->rowCount());
|
|
|
|
row=ui->table->rowCount()-1;
|
|
|
|
}
|
|
|
|
|
|
|
|
QPushButton* recall = new QPushButton("Recall");
|
2023-05-02 20:18:41 +00:00
|
|
|
ui->table->setCellWidget(row,columnRecall,recall);
|
2023-05-01 20:56:52 +00:00
|
|
|
connect(recall, &QPushButton::clicked, this,
|
2023-05-04 20:55:40 +00:00
|
|
|
[=]() { qInfo() << "Recalling" << mem.channel; emit recallMemory(mem.channel);});
|
2023-05-01 20:56:52 +00:00
|
|
|
|
2023-05-02 20:18:41 +00:00
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnNum),QString::number(mem.channel & 0xffff).rightJustified(3,'0'));
|
2023-05-01 20:56:52 +00:00
|
|
|
ui->table->item(row,columnNum)->setFlags(ui->table->item(row,columnNum)->flags() & (~Qt::ItemIsEditable));
|
2023-05-02 20:18:41 +00:00
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
// Set every combobox to default value first:
|
|
|
|
if (ui->table->item(row,columnScan) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnScan),scan[0]);
|
|
|
|
if (ui->table->item(row,columnVFO) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnVFO),VFO[0]);
|
|
|
|
if (ui->table->item(row,columnData) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnData),dataModes[0]);
|
|
|
|
if (ui->table->item(row,columnDuplex) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnDuplex),duplexModes[0]);
|
|
|
|
if (ui->table->item(row,columnToneMode) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnToneMode),toneModes[0]);
|
|
|
|
if (ui->table->item(row,columnDSQL) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnDSQL),dsql[0]);
|
|
|
|
if (ui->table->item(row,columnTone) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnTone),tones[0]);
|
|
|
|
if (ui->table->item(row,columnTSQL) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnTSQL),tones[0]);
|
|
|
|
if (ui->table->item(row,columnDTCSPolarity) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnDTCSPolarity),dtcsp[0]);
|
|
|
|
if (ui->table->item(row,columnDTCS) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnDTCS),dtcs[0]);
|
|
|
|
if (ui->table->item(row,columnDataB) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnDataB),dataModes[0]);
|
|
|
|
if (ui->table->item(row,columnToneModeB) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnToneModeB),toneModes[0]);
|
|
|
|
if (ui->table->item(row,columnToneB) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnToneB),tones[0]);
|
|
|
|
if (ui->table->item(row,columnTSQLB) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnTSQLB),tones[0]);
|
|
|
|
if (ui->table->item(row,columnDTCSPolarityB) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnDTCSPolarityB),dtcsp[0]);
|
|
|
|
if (ui->table->item(row,columnDTCSB) == NULL) ui->table->model()->setData(ui->table->model()->index(row,columnDTCSB),dtcs[0]);
|
|
|
|
|
|
|
|
for (int f=0;f<scan.size();f++)
|
2023-05-01 20:56:52 +00:00
|
|
|
{
|
2023-05-04 20:55:40 +00:00
|
|
|
if (f==mem.scan) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnScan),scan[f]);
|
2023-05-01 20:56:52 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2023-05-02 20:18:41 +00:00
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
for (int f=0;f<VFO.size();f++)
|
|
|
|
{
|
|
|
|
if (f==mem.frequency.VFO) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnVFO),VFO[f]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2023-05-02 20:18:41 +00:00
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnFrequency),QString::number(double(mem.frequency.Hz/1000000.0),'f',3));
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnFrequencyB),QString::number(double(mem.frequencyB.Hz/1000000.0),'f',3));
|
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
foreach (auto mode, rigCaps.modes){
|
|
|
|
if (mode.reg == mem.mode) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnMode),mode.name);
|
2023-05-04 20:55:40 +00:00
|
|
|
}
|
|
|
|
if (mode.reg == mem.modeB) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnModeB),mode.name);
|
2023-05-01 20:56:52 +00:00
|
|
|
}
|
|
|
|
}
|
2023-05-02 20:18:41 +00:00
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
for (int f=0;f<dataModes.size();f++)
|
|
|
|
{
|
|
|
|
if (f==mem.datamode) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnData),dataModes[f]);
|
2023-05-04 20:55:40 +00:00
|
|
|
}
|
|
|
|
if (f==mem.datamodeB) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnDataB),dataModes[f]);
|
2023-05-01 20:56:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int f=0;f<filters.size();f++)
|
|
|
|
{
|
|
|
|
if (f==mem.filter-1) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnFilter),filters[f]);
|
2023-05-04 20:55:40 +00:00
|
|
|
}
|
|
|
|
if (f==mem.filterB-1) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnFilterB),filters[f]);
|
2023-05-01 20:56:52 +00:00
|
|
|
}
|
|
|
|
}
|
2023-05-02 20:18:41 +00:00
|
|
|
|
|
|
|
for (int f=0;f<duplexModes.size();f++)
|
|
|
|
{
|
|
|
|
if (f==mem.duplex) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnDuplex),duplexModes[f]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
for (int f=0;f<toneModes.size();f++)
|
|
|
|
{
|
|
|
|
if (f==mem.tonemode) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnToneMode),toneModes[f]);
|
2023-05-04 20:55:40 +00:00
|
|
|
}
|
|
|
|
if (f==mem.tonemodeB) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnToneModeB),toneModes[f]);
|
2023-05-01 20:56:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-02 20:18:41 +00:00
|
|
|
for (int f=0;f<dsql.size();f++)
|
|
|
|
{
|
|
|
|
if (f==mem.dsql) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnDSQL),dsql[f]);
|
2023-05-04 20:55:40 +00:00
|
|
|
}
|
|
|
|
if (f==mem.dsqlB) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnDSQLB),dsql[f]);
|
2023-05-02 20:18:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnTone),QString::number((float)mem.tone/10,'f',1));
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnToneB),QString::number((float)mem.toneB/10,'f',1));
|
2023-05-02 20:18:41 +00:00
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnTSQL),QString::number((float)mem.tsql/10,'f',1));
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnTSQLB),QString::number((float)mem.tsqlB/10,'f',1));
|
2023-05-02 20:18:41 +00:00
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnDVSquelch),QString::number(mem.dvsql).rightJustified(2,'0'));
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnDVSquelchB),QString::number(mem.dvsqlB).rightJustified(2,'0'));
|
2023-05-02 20:18:41 +00:00
|
|
|
|
|
|
|
for (int f=0;f<dtcsp.size();f++)
|
|
|
|
{
|
|
|
|
if (f==mem.dtcsp) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnDTCSPolarity),dtcsp[f]);
|
2023-05-04 20:55:40 +00:00
|
|
|
}
|
|
|
|
if (f==mem.dtcspB) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnDTCSPolarityB),dtcsp[f]);
|
2023-05-02 20:18:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
for (int f=0;f<dtcs.size();f++)
|
|
|
|
{
|
|
|
|
if (dtcs[f].toInt()==mem.dtcs) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnDTCS),dtcs[f]);
|
|
|
|
}
|
|
|
|
if (dtcs[f].toInt()==mem.dtcsB) {
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnDTCSB),dtcs[f]);
|
|
|
|
}
|
|
|
|
}
|
2023-05-02 20:18:41 +00:00
|
|
|
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnOffset),QString::number(mem.duplexOffset.Hz));
|
|
|
|
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnUR),QString(mem.UR));
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnURB),QString(mem.URB));
|
2023-05-02 20:18:41 +00:00
|
|
|
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnR1),QString(mem.R1));
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnR1B),QString(mem.R1B));
|
2023-05-02 20:18:41 +00:00
|
|
|
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnR2),QString(mem.R2));
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnR2B),QString(mem.R2B));
|
2023-05-02 20:18:41 +00:00
|
|
|
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(row,columnName),QString(mem.name));
|
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
ui->table->blockSignals(false);
|
|
|
|
}
|
2023-05-02 20:18:41 +00:00
|
|
|
|
|
|
|
// If this is the last channel, re-enable the group combo.
|
2023-05-04 20:55:40 +00:00
|
|
|
if (mem.channel == rigCaps.memories || (mem.sat && mem.channel == rigCaps.satMemories)) {
|
|
|
|
timeoutTimer.stop();
|
2023-05-02 20:18:41 +00:00
|
|
|
ui->group->setEnabled(true);
|
|
|
|
}
|
2023-05-01 20:56:52 +00:00
|
|
|
}
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
void memories::timeout()
|
|
|
|
{
|
|
|
|
QMessageBox::information(this,"Timeout", "Timeout receiving memories, check rig connection", QMessageBox::Ok);
|
|
|
|
timeoutTimer.stop();
|
|
|
|
ui->group->setEnabled(true);
|
|
|
|
}
|
|
|
|
|
2023-05-01 20:56:52 +00:00
|
|
|
QStandardItemModel* memories::createModel(QStandardItemModel* model, QStringList strings)
|
|
|
|
{
|
|
|
|
model = new QStandardItemModel();
|
|
|
|
|
|
|
|
for (int i=0; i < strings.size();i++)
|
|
|
|
{
|
|
|
|
QStandardItem *itemName = new QStandardItem(strings[i]);
|
|
|
|
QStandardItem *itemId = new QStandardItem(i);
|
|
|
|
|
|
|
|
QList<QStandardItem*> row;
|
|
|
|
row << itemName << itemId;
|
|
|
|
|
|
|
|
model->appendRow(row);
|
|
|
|
}
|
|
|
|
|
|
|
|
return model;
|
|
|
|
}
|
2023-05-03 18:08:20 +00:00
|
|
|
|
|
|
|
|
|
|
|
void memories::on_csvImport_clicked()
|
|
|
|
{
|
|
|
|
QString file = QFileDialog::getOpenFileName(this,"Select import filename","","CSV Files (*.csv)");
|
|
|
|
|
|
|
|
if (!file.isEmpty())
|
|
|
|
{
|
|
|
|
ui->table->sortByColumn(0,Qt::AscendingOrder); // Force natural order
|
|
|
|
|
|
|
|
QFile data(file);
|
|
|
|
if (!data.open(QIODevice::ReadOnly)) {
|
2023-05-04 20:55:40 +00:00
|
|
|
qInfo() << "Couldn't open file for .csv import" << data.errorString();
|
2023-05-03 18:08:20 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QTextStream input(&data);
|
|
|
|
QStringList row;
|
|
|
|
int rows=0;
|
|
|
|
while (readCSVRow(input, &row)) {
|
|
|
|
qInfo() << row;
|
|
|
|
if (!rows++)
|
|
|
|
continue; // Skip the first row
|
|
|
|
|
2023-05-04 20:55:40 +00:00
|
|
|
int rownum = -1;
|
2023-05-03 18:08:20 +00:00
|
|
|
for(int i=0;i<ui->table->rowCount();i++) {
|
|
|
|
if (ui->table->item(i, 1)->text().toInt() == row[0].toInt()) {
|
2023-05-04 20:55:40 +00:00
|
|
|
rownum=i;
|
2023-05-03 18:08:20 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2023-05-04 20:55:40 +00:00
|
|
|
if (rownum == -1)
|
2023-05-03 18:08:20 +00:00
|
|
|
{
|
|
|
|
// We need to add a new row
|
2023-05-04 20:55:40 +00:00
|
|
|
rownum = ui->table->rowCount();
|
|
|
|
ui->table->insertRow(rownum);
|
2023-05-03 18:08:20 +00:00
|
|
|
QPushButton* recall = new QPushButton("Recall");
|
2023-05-04 20:55:40 +00:00
|
|
|
ui->table->setCellWidget(rownum,columnRecall,recall);
|
2023-05-03 18:08:20 +00:00
|
|
|
connect(recall, &QPushButton::clicked, this,
|
2023-05-04 20:55:40 +00:00
|
|
|
[=]() { qInfo() << "Recalling" << row[0].toInt(); emit recallMemory(row[0].toInt());});
|
2023-05-03 18:08:20 +00:00
|
|
|
}
|
2023-05-04 20:55:40 +00:00
|
|
|
// rownum is now the row we need to work on.
|
|
|
|
|
|
|
|
int colnum=1;
|
|
|
|
|
|
|
|
ui->table->blockSignals(true);
|
2023-05-03 18:08:20 +00:00
|
|
|
for (int i=0; i<row.size();i++)
|
|
|
|
{
|
2023-05-04 20:55:40 +00:00
|
|
|
while (colnum < ui->table->columnCount() && ui->table->isColumnHidden(colnum)) {
|
|
|
|
colnum++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Stop blocking signals for last column (should force it to be sent to rig)
|
|
|
|
if (i == row.size()-1)
|
|
|
|
ui->table->blockSignals(false);
|
|
|
|
|
|
|
|
if (colnum < ui->table->columnCount())
|
|
|
|
{
|
|
|
|
QString data = QString(row[i]).trimmed();
|
|
|
|
switch (colnum)
|
|
|
|
{
|
|
|
|
// special cases:
|
|
|
|
case columnFrequency:
|
|
|
|
case columnFrequencyB:
|
|
|
|
data= QString::number(data.toDouble(),'f',3);
|
|
|
|
break;
|
|
|
|
case columnDTCS:
|
|
|
|
case columnDTCSB:
|
|
|
|
data=QString::number(data.toInt()).rightJustified(3,'0');
|
|
|
|
break;
|
|
|
|
case columnDVSquelch:
|
|
|
|
case columnDVSquelchB:
|
|
|
|
data=QString::number(data.toInt()).rightJustified(2,'0');
|
|
|
|
break;
|
|
|
|
case columnTone:
|
|
|
|
case columnToneB:
|
|
|
|
case columnTSQL:
|
|
|
|
case columnTSQLB:
|
|
|
|
if (data.endsWith("Hz")) data=data.mid(0,data.length()-2);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ui->table->model()->setData(ui->table->model()->index(rownum,colnum),data);
|
|
|
|
colnum++;
|
2023-05-03 18:08:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void memories::on_csvExport_clicked()
|
|
|
|
{
|
|
|
|
QString file = QFileDialog::getSaveFileName(this,"Select export filename","","CSV Files (*.csv)");
|
|
|
|
|
|
|
|
if (!file.isEmpty())
|
|
|
|
{
|
|
|
|
ui->table->sortByColumn(0,Qt::AscendingOrder); // Force natural order
|
|
|
|
|
|
|
|
QFile data(file);
|
|
|
|
if (data.open(QFile::WriteOnly | QIODevice::Truncate)) {
|
|
|
|
QTextStream output(&data);
|
2023-05-04 20:55:40 +00:00
|
|
|
for(int i=1;i<ui->table->columnCount();i++) {
|
|
|
|
if (!ui->table->isColumnHidden(i))
|
|
|
|
{
|
|
|
|
output << "\"" << ui->table->horizontalHeaderItem(i)->text() << "\"";
|
|
|
|
if (i < ui->table->columnCount()-1)
|
|
|
|
output << ",";
|
|
|
|
else
|
|
|
|
output << "\n";
|
|
|
|
}
|
2023-05-03 18:08:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for(int i=0;i<ui->table->rowCount();i++) {
|
|
|
|
for(int j=1;j<ui->table->columnCount();j++) {
|
2023-05-04 20:55:40 +00:00
|
|
|
if (!ui->table->isColumnHidden(j))
|
|
|
|
{
|
|
|
|
output << "\"" << ((ui->table->item(i,j) == NULL) ? "" : ui->table->item(i,j)->text().trimmed()) << "\"";
|
|
|
|
if (j < ui->table->columnCount()-1)
|
|
|
|
output << ",";
|
|
|
|
else
|
|
|
|
output << "\n";
|
|
|
|
}
|
2023-05-03 18:08:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
data.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Public Domain CSV parser from user iamantony
|
|
|
|
// https://www.appsloveworld.com/cplus/100/37/parsing-through-a-csv-file-in-qt
|
|
|
|
bool memories::readCSVRow(QTextStream &in, QStringList *row) {
|
|
|
|
|
|
|
|
static const int delta[][5] = {
|
|
|
|
// , " \n ? eof
|
|
|
|
{ 1, 2, -1, 0, -1 }, // 0: parsing (store char)
|
|
|
|
{ 1, 2, -1, 0, -1 }, // 1: parsing (store column)
|
|
|
|
{ 3, 4, 3, 3, -2 }, // 2: quote entered (no-op)
|
|
|
|
{ 3, 4, 3, 3, -2 }, // 3: parsing inside quotes (store char)
|
|
|
|
{ 1, 3, -1, 0, -1 }, // 4: quote exited (no-op)
|
|
|
|
// -1: end of row, store column, success
|
|
|
|
// -2: eof inside quotes
|
|
|
|
};
|
|
|
|
|
|
|
|
row->clear();
|
|
|
|
|
|
|
|
if (in.atEnd())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
int state = 0, t;
|
|
|
|
char ch;
|
|
|
|
QString cell;
|
|
|
|
|
|
|
|
while (state >= 0) {
|
|
|
|
|
|
|
|
if (in.atEnd())
|
|
|
|
t = 4;
|
|
|
|
else {
|
|
|
|
in >> ch;
|
|
|
|
if (ch == ',') t = 0;
|
|
|
|
else if (ch == '\"') t = 1;
|
|
|
|
else if (ch == '\n') t = 2;
|
|
|
|
else t = 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
state = delta[state][t];
|
|
|
|
|
|
|
|
switch (state) {
|
|
|
|
case 0:
|
|
|
|
case 3:
|
|
|
|
cell += ch;
|
|
|
|
break;
|
|
|
|
case -1:
|
|
|
|
case 1:
|
|
|
|
row->append(cell);
|
|
|
|
cell = "";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (state == -2) {
|
|
|
|
qInfo() << "End-of-file found while inside quotes.";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|