* along with this program; if not, see <http://www.gnu.org/licenses/>.
*/
+#include <forward_list>
#include <limits>
#include <QDebug>
#include <pv/globalsettings.hpp>
#include <pv/session.hpp>
+using std::forward_list;
using std::lock_guard;
using std::make_pair;
using std::make_shared;
}
}
+void DecodeSignal::get_annotation_subset(
+ vector<pv::data::decode::Annotation> &dest,
+ uint32_t segment_id, uint64_t start_sample, uint64_t end_sample) const
+{
+ // Note: We put all vectors and lists on the heap, not the stack
+
+ const vector<Row> rows = visible_rows();
+
+ // Use forward_lists for faster merging
+ forward_list<Annotation> *all_ann_list = new forward_list<Annotation>();
+
+ for (const Row& row : rows) {
+ vector<Annotation> *ann_vector = new vector<Annotation>();
+ get_annotation_subset(*ann_vector, row, segment_id, start_sample, end_sample);
+
+ forward_list<Annotation> *ann_list =
+ new forward_list<Annotation>(ann_vector->begin(), ann_vector->end());
+ delete ann_vector;
+
+ all_ann_list->merge(*ann_list);
+ delete ann_list;
+ }
+
+ move(all_ann_list->begin(), all_ann_list->end(), back_inserter(dest));
+ delete all_ann_list;
+}
+
void DecodeSignal::save_settings(QSettings &settings) const
{
SignalBase::save_settings(settings);
}
// Add the annotation
- (*row_iter).second.emplace_annotation(pdata);
+ (*row_iter).second.emplace_annotation(pdata, &((*row_iter).first));
}
void DecodeSignal::on_capture_state_changed(int state)
QMenu *const menu = new QMenu(parent);
+ QAction *const export_all_rows =
+ new QAction(tr("Export all annotations"), this);
+ export_all_rows->setIcon(QIcon::fromTheme("document-save-as",
+ QIcon(":/icons/document-save-as.png")));
+ connect(export_all_rows, SIGNAL(triggered()), this, SLOT(on_export_all_rows()));
+ menu->addAction(export_all_rows);
+
QAction *const export_row =
new QAction(tr("Export all annotations for this row"), this);
export_row->setIcon(QIcon::fromTheme("document-save-as",
QIcon(":/icons/document-save-as.png")));
- connect(export_row, SIGNAL(triggered()),
- this, SLOT(on_export_row()));
+ connect(export_row, SIGNAL(triggered()), this, SLOT(on_export_row()));
menu->addAction(export_row);
+ menu->addSeparator();
+
+ QAction *const export_all_rows_from_here =
+ new QAction(tr("Export all annotations, starting here"), this);
+ export_all_rows_from_here->setIcon(QIcon::fromTheme("document-save-as",
+ QIcon(":/icons/document-save-as.png")));
+ connect(export_all_rows_from_here, SIGNAL(triggered()), this, SLOT(on_export_all_rows_from_here()));
+ menu->addAction(export_all_rows_from_here);
+
QAction *const export_row_from_here =
new QAction(tr("Export annotations for this row, starting here"), this);
export_row_from_here->setIcon(QIcon::fromTheme("document-save-as",
QIcon(":/icons/document-save-as.png")));
- connect(export_row_from_here, SIGNAL(triggered()),
- this, SLOT(on_export_row_from_here()));
+ connect(export_row_from_here, SIGNAL(triggered()), this, SLOT(on_export_row_from_here()));
menu->addAction(export_row_from_here);
return menu;
return selector;
}
+void DecodeTrace::export_annotations(vector<Annotation> *annotations) const
+{
+ using namespace pv::data::decode;
+
+ GlobalSettings settings;
+ const QString dir = settings.value("MainWindow/SaveDirectory").toString();
+
+ const QString file_name = QFileDialog::getSaveFileName(
+ owner_->view(), tr("Export annotations"), dir, tr("Text Files (*.txt);;All Files (*)"));
+
+ if (file_name.isEmpty())
+ return;
+
+ QString format = settings.value(GlobalSettings::Key_Dec_ExportFormat).toString();
+ const QString quote = format.contains("%q") ? "\"" : "";
+ format = format.remove("%q");
+
+ QFile file(file_name);
+ if (file.open(QIODevice::WriteOnly | QIODevice::Truncate | QIODevice::Text)) {
+ QTextStream out_stream(&file);
+
+ for (Annotation &ann : *annotations) {
+ const QString sample_range = QString("%1-%2") \
+ .arg(QString::number(ann.start_sample()), QString::number(ann.end_sample()));
+
+ const QString class_name = quote + ann.row()->class_name() + quote;
+
+ QString all_ann_text;
+ for (const QString &s : ann.annotations())
+ all_ann_text = all_ann_text + quote + s + quote + ",";
+ all_ann_text.chop(1);
+
+ const QString first_ann_text = quote + ann.annotations().front() + quote;
+
+ QString out_text = format;
+ out_text = out_text.replace("%s", sample_range);
+ out_text = out_text.replace("%d",
+ quote + QString::fromUtf8(ann.row()->decoder()->name) + quote);
+ out_text = out_text.replace("%c", class_name);
+ out_text = out_text.replace("%1", first_ann_text);
+ out_text = out_text.replace("%a", all_ann_text);
+ out_stream << out_text << '\n';
+ }
+
+ if (out_stream.status() == QTextStream::Ok)
+ return;
+ }
+
+ QMessageBox msg(owner_->view());
+ msg.setText(tr("Error"));
+ msg.setInformativeText(tr("File %1 could not be written to.").arg(file_name));
+ msg.setStandardButtons(QMessageBox::Ok);
+ msg.setIcon(QMessageBox::Warning);
+ msg.exec();
+}
+
void DecodeTrace::on_new_annotations()
{
if (!delayed_trace_updater_.isActive())
on_export_row_from_here();
}
+void DecodeTrace::on_export_all_rows()
+{
+ selected_samplepos_ = 0;
+ on_export_all_rows_from_here();
+}
+
void DecodeTrace::on_export_row_from_here()
{
using namespace pv::data::decode;
if (!selected_row_)
return;
- vector<Annotation> annotations;
+ vector<Annotation> *annotations = new vector<Annotation>();
- decode_signal_->get_annotation_subset(annotations, *selected_row_,
+ decode_signal_->get_annotation_subset(*annotations, *selected_row_,
current_segment_, selected_samplepos_, ULLONG_MAX);
- if (annotations.empty())
+ if (annotations->empty())
return;
- GlobalSettings settings;
- const QString dir = settings.value("MainWindow/SaveDirectory").toString();
-
- const QString file_name = QFileDialog::getSaveFileName(
- owner_->view(), tr("Export annotations"), dir, tr("Text Files (*.txt);;All Files (*)"));
-
- if (file_name.isEmpty())
- return;
-
- const QString format = settings.value(GlobalSettings::Key_Dec_ExportFormat).toString();
- const QString quote = format.contains("%q") ? "\"" : "";
- const QString class_name = selected_row_->class_name();
-
- QFile file(file_name);
- if (file.open(QIODevice::WriteOnly | QIODevice::Truncate | QIODevice::Text)) {
- QTextStream out_stream(&file);
-
- for (Annotation &ann : annotations) {
- const QString sample_range = QString("%1-%2").arg(ann.start_sample()).arg(
- ann.end_sample());
+ export_annotations(annotations);
+ delete annotations;
+}
- QString all_ann_text;
- for (const QString &s : ann.annotations())
- all_ann_text = all_ann_text + quote + s + quote + ",";
- all_ann_text.chop(1);
+void DecodeTrace::on_export_all_rows_from_here()
+{
+ using namespace pv::data::decode;
- const QString first_ann_text = quote + ann.annotations().front() + quote;
+ vector<Annotation> *annotations = new vector<Annotation>();
- QString out_text = format;
- out_text = out_text.replace("%s", sample_range);
- out_text = out_text.replace("%d", decode_signal_->name());
- out_text = out_text.replace("%c", class_name);
- out_text = out_text.replace("%1", first_ann_text);
- out_text = out_text.replace("%a", all_ann_text);
- out_stream << out_text << '\n';
- }
+ decode_signal_->get_annotation_subset(*annotations, current_segment_,
+ selected_samplepos_, ULLONG_MAX);
- if (out_stream.status() == QTextStream::Ok)
- return;
- }
+ if (!annotations->empty())
+ export_annotations(annotations);
- QMessageBox msg(owner_->view());
- msg.setText(tr("Error"));
- msg.setInformativeText(tr("File %1 could not be written to.").arg(file_name));
- msg.setStandardButtons(QMessageBox::Ok);
- msg.setIcon(QMessageBox::Warning);
- msg.exec();
+ delete annotations;
}
} // namespace trace