# HG changeset patch # User Chris Cannam # Date 1547044912 0 # Node ID 0930a27ebea2900493df44300251d68f4e619eaf # Parent 79032214f79d2dcca1ea0d3b677bf484d2763dde Support changing the separator (if more than one plausible one is found) and repopulating the dialog diff -r 79032214f79d -r 0930a27ebea2 widgets/CSVFormatDialog.cpp --- a/widgets/CSVFormatDialog.cpp Wed Dec 19 10:40:17 2018 +0000 +++ b/widgets/CSVFormatDialog.cpp Wed Jan 09 14:41:52 2019 +0000 @@ -34,16 +34,49 @@ #include "base/Debug.h" -CSVFormatDialog::CSVFormatDialog(QWidget *parent, CSVFormat format, +CSVFormatDialog::CSVFormatDialog(QWidget *parent, + CSVFormat format, int maxDisplayCols) : QDialog(parent), + m_csvFilePath(""), + m_referenceSampleRate(0), m_format(format), m_maxDisplayCols(maxDisplayCols), m_fuzzyColumn(-1) { + init(); +} + +CSVFormatDialog::CSVFormatDialog(QWidget *parent, + QString csvFilePath, + sv_samplerate_t referenceSampleRate, + int maxDisplayCols) : + QDialog(parent), + m_csvFilePath(csvFilePath), + m_referenceSampleRate(referenceSampleRate), + m_maxDisplayCols(maxDisplayCols), + m_fuzzyColumn(-1) +{ + m_format = CSVFormat(csvFilePath); + m_format.setSampleRate(referenceSampleRate); + init(); +} + +CSVFormatDialog::~CSVFormatDialog() +{ +} + +static int sampleRates[] = { + 8000, 11025, 12000, 22050, 24000, 32000, + 44100, 48000, 88200, 96000, 176400, 192000 +}; + +void +CSVFormatDialog::init() +{ setModal(true); setWindowTitle(tr("Select Data Format")); - + QGridLayout *layout = new QGridLayout; int row = 0; @@ -52,6 +85,105 @@ (new QLabel(tr("Please select the correct data format for this file.")), row++, 0, 1, 4); + m_exampleFrame = nullptr; + m_exampleFrameRow = row++; + + std::set plausible = m_format.getPlausibleSeparators(); + SVDEBUG << "Have " << plausible.size() << " plausible separator(s)" << endl; + + if (m_csvFilePath != "" && plausible.size() > 1) { + // can only update when separator changed if we still have a + // file to refer to + layout->addWidget(new QLabel(tr("Column separator:")), row, 0); + m_separatorCombo = new QComboBox; + for (QChar c: plausible) { + m_separatorCombo->addItem(QString(c)); + if (c == m_format.getSeparator()) { + m_separatorCombo->setCurrentIndex(m_separatorCombo->count()-1); + } + } + m_separatorCombo->setEditable(false); + + layout->addWidget(m_separatorCombo, row++, 1); + connect(m_separatorCombo, SIGNAL(activated(QString)), + this, SLOT(separatorChanged(QString))); + } + + layout->addWidget(new QLabel(tr("Timing is specified:")), row, 0); + + m_timingTypeCombo = new QComboBox; + + m_timingLabels = { + { TimingExplicitSeconds, tr("Explicitly, in seconds") }, + { TimingExplicitMsec, tr("Explicitly, in milliseconds") }, + { TimingExplicitSamples, tr("Explicitly, in audio sample frames") }, + { TimingImplicit, tr("Implicitly: rows are equally spaced in time") } + }; + + for (auto &l: m_timingLabels) { + m_timingTypeCombo->addItem(l.second); + } + + layout->addWidget(m_timingTypeCombo, row++, 1, 1, 2); + + connect(m_timingTypeCombo, SIGNAL(activated(int)), + this, SLOT(timingTypeChanged(int))); + + m_sampleRateLabel = new QLabel(tr("Audio sample rate (Hz):")); + layout->addWidget(m_sampleRateLabel, row, 0); + + m_sampleRateCombo = new QComboBox; + for (int i = 0; i < int(sizeof(sampleRates) / sizeof(sampleRates[0])); ++i) { + m_sampleRateCombo->addItem(QString("%1").arg(sampleRates[i])); + } + m_sampleRateCombo->setEditable(true); + + layout->addWidget(m_sampleRateCombo, row++, 1); + connect(m_sampleRateCombo, SIGNAL(activated(QString)), + this, SLOT(sampleRateChanged(QString))); + connect(m_sampleRateCombo, SIGNAL(editTextChanged(QString)), + this, SLOT(sampleRateChanged(QString))); + + m_windowSizeLabel = new QLabel(tr("Frame increment between rows:")); + layout->addWidget(m_windowSizeLabel, row, 0); + + m_windowSizeCombo = new QComboBox; + for (int i = 0; i <= 16; ++i) { + int value = 1 << i; + m_windowSizeCombo->addItem(QString("%1").arg(value)); + } + m_windowSizeCombo->setEditable(true); + + layout->addWidget(m_windowSizeCombo, row++, 1); + connect(m_windowSizeCombo, SIGNAL(activated(QString)), + this, SLOT(windowSizeChanged(QString))); + connect(m_windowSizeCombo, SIGNAL(editTextChanged(QString)), + this, SLOT(windowSizeChanged(QString))); + + m_modelLabel = new QLabel; + QFont f(m_modelLabel->font()); + f.setItalic(true); + m_modelLabel->setFont(f); + layout->addWidget(m_modelLabel, row++, 0, 1, 4); + + QDialogButtonBox *bb = new QDialogButtonBox(QDialogButtonBox::Ok | + QDialogButtonBox::Cancel); + layout->addWidget(bb, row++, 0, 1, 4); + connect(bb, SIGNAL(accepted()), this, SLOT(accept())); + connect(bb, SIGNAL(rejected()), this, SLOT(reject())); + + setLayout(layout); + + repopulate(); +} + +void +CSVFormatDialog::repopulate() +{ + SVCERR << "CSVFormatDialog::repopulate()" << endl; + + QGridLayout *layout = qobject_cast(this->layout()); + QFrame *exampleFrame = new QFrame; exampleFrame->setFrameStyle(QFrame::StyledPanel | QFrame::Sunken); exampleFrame->setLineWidth(2); @@ -66,9 +198,11 @@ QFont fp; fp.setPointSize(int(floor(fp.pointSize() * 0.9))); - int columns = format.getColumnCount(); + int columns = m_format.getColumnCount(); QList example = m_format.getExample(); + m_columnPurposeCombos.clear(); + for (int i = 0; i < columns; ++i) { QComboBox *cpc = new QComboBox; @@ -115,29 +249,14 @@ } } - layout->addWidget(exampleFrame, row, 0, 1, 4); + if (m_exampleFrame) { + delete m_exampleFrame; + } + m_exampleFrame = exampleFrame; + + layout->addWidget(exampleFrame, m_exampleFrameRow, 0, 1, 4); layout->setColumnStretch(3, 10); - layout->setRowStretch(row++, 10); - - layout->addWidget(new QLabel(tr("Timing is specified:")), row, 0); - - m_timingTypeCombo = new QComboBox; - - m_timingLabels = { - { TimingExplicitSeconds, tr("Explicitly, in seconds") }, - { TimingExplicitMsec, tr("Explicitly, in milliseconds") }, - { TimingExplicitSamples, tr("Explicitly, in audio sample frames") }, - { TimingImplicit, tr("Implicitly: rows are equally spaced in time") } - }; - - for (auto &l: m_timingLabels) { - m_timingTypeCombo->addItem(l.second); - } - - layout->addWidget(m_timingTypeCombo, row++, 1, 1, 2); - - connect(m_timingTypeCombo, SIGNAL(activated(int)), - this, SLOT(timingTypeChanged(int))); + layout->setRowStretch(m_exampleFrameRow, 10); m_initialTimingOption = TimingImplicit; if (m_format.getTimingType() == CSVFormat::ExplicitTiming) { @@ -153,70 +272,23 @@ } } m_timingTypeCombo->setCurrentIndex(int(m_initialTimingOption)); - - m_sampleRateLabel = new QLabel(tr("Audio sample rate (Hz):")); - layout->addWidget(m_sampleRateLabel, row, 0); - int sampleRates[] = { - 8000, 11025, 12000, 22050, 24000, 32000, - 44100, 48000, 88200, 96000, 176400, 192000 - }; - - m_sampleRateCombo = new QComboBox; for (int i = 0; i < int(sizeof(sampleRates) / sizeof(sampleRates[0])); ++i) { - m_sampleRateCombo->addItem(QString("%1").arg(sampleRates[i])); if (sampleRates[i] == m_format.getSampleRate()) { m_sampleRateCombo->setCurrentIndex(i); } } - m_sampleRateCombo->setEditable(true); - - layout->addWidget(m_sampleRateCombo, row++, 1); - connect(m_sampleRateCombo, SIGNAL(activated(QString)), - this, SLOT(sampleRateChanged(QString))); - connect(m_sampleRateCombo, SIGNAL(editTextChanged(QString)), - this, SLOT(sampleRateChanged(QString))); - - m_windowSizeLabel = new QLabel(tr("Frame increment between rows:")); - layout->addWidget(m_windowSizeLabel, row, 0); - m_windowSizeCombo = new QComboBox; for (int i = 0; i <= 16; ++i) { int value = 1 << i; - m_windowSizeCombo->addItem(QString("%1").arg(value)); if (value == int(m_format.getWindowSize())) { m_windowSizeCombo->setCurrentIndex(i); } } - m_windowSizeCombo->setEditable(true); - - layout->addWidget(m_windowSizeCombo, row++, 1); - connect(m_windowSizeCombo, SIGNAL(activated(QString)), - this, SLOT(windowSizeChanged(QString))); - connect(m_windowSizeCombo, SIGNAL(editTextChanged(QString)), - this, SLOT(windowSizeChanged(QString))); - - m_modelLabel = new QLabel; - QFont f(m_modelLabel->font()); - f.setItalic(true); - m_modelLabel->setFont(f); - layout->addWidget(m_modelLabel, row++, 0, 1, 4); - - QDialogButtonBox *bb = new QDialogButtonBox(QDialogButtonBox::Ok | - QDialogButtonBox::Cancel); - layout->addWidget(bb, row++, 0, 1, 4); - connect(bb, SIGNAL(accepted()), this, SLOT(accept())); - connect(bb, SIGNAL(rejected()), this, SLOT(reject())); - - setLayout(layout); timingTypeChanged(m_timingTypeCombo->currentIndex()); } -CSVFormatDialog::~CSVFormatDialog() -{ -} - CSVFormat CSVFormatDialog::getFormat() const { @@ -315,6 +387,19 @@ } void +CSVFormatDialog::separatorChanged(QString sep) +{ + if (sep == "" || m_csvFilePath == "") { + return; + } + + m_format.setSeparator(sep[0]); + m_format.guessFormatFor(m_csvFilePath); + + repopulate(); +} + +void CSVFormatDialog::timingTypeChanged(int type) { // Update any column purpose combos diff -r 79032214f79d -r 0930a27ebea2 widgets/CSVFormatDialog.h --- a/widgets/CSVFormatDialog.h Wed Dec 19 10:40:17 2018 +0000 +++ b/widgets/CSVFormatDialog.h Wed Jan 09 14:41:52 2019 +0000 @@ -21,6 +21,7 @@ class QTableWidget; class QComboBox; class QLabel; +class QFrame; #include @@ -31,12 +32,19 @@ public: CSVFormatDialog(QWidget *parent, CSVFormat initialFormat, - int maxDisplayCols = 5); + int maxDisplayCols); + + CSVFormatDialog(QWidget *parent, + QString csvFilePath, // to guess format of + sv_samplerate_t referenceSampleRate, + int maxDisplayCols); + ~CSVFormatDialog(); CSVFormat getFormat() const; protected slots: + void separatorChanged(QString); void timingTypeChanged(int type); void sampleRateChanged(QString); void windowSizeChanged(QString); @@ -46,6 +54,8 @@ void updateModelLabel(); protected: + QString m_csvFilePath; + sv_samplerate_t m_referenceSampleRate; CSVFormat m_format; int m_maxDisplayCols; @@ -58,11 +68,17 @@ std::map m_timingLabels; TimingOption m_initialTimingOption; + void init(); + void repopulate(); void columnPurposeChangedForAnnotationType(QComboBox *, int purpose); void updateComboVisibility(); void applyStartTimePurpose(); void removeStartTimePurpose(); + + QFrame *m_exampleFrame; + int m_exampleFrameRow; + QComboBox *m_separatorCombo; QComboBox *m_timingTypeCombo; QLabel *m_sampleRateLabel; QComboBox *m_sampleRateCombo;