TAMSVIZ
Visualization and annotation tool for ROS
searchwidget.cpp
1 // TAMSVIZ
2 // (c) 2020 Philipp Ruppel
3 
4 #include "searchwidget.h"
5 
6 #include "../core/bagplayer.h"
7 #include "../core/tracks.h"
8 #include "../core/workspace.h"
9 
10 #include "mainwindow.h"
11 
12 SearchWidget::SearchWidget() : QDockWidget("Search") {
13 
14  auto sync = _sync;
15 
16  QWidget *content_widget = new QWidget();
17 
18  auto *main_layout = new QVBoxLayout(content_widget);
19  main_layout->setSpacing(0);
20  main_layout->setContentsMargins(0, 0, 0, 0);
21 
22  auto *search_box = new QLineEdit(content_widget);
23  main_layout->addWidget(search_box);
24  search_box->setPlaceholderText("Search annotations...");
25  connect(search_box, &QLineEdit::textEdited, this,
26  [sync](const QString &text) {
27  std::unique_lock<std::mutex> lock(sync->_search_mutex);
28  sync->_search_query = text.toStdString();
29  sync->_search_request = true;
30  sync->_search_condition.notify_all();
31  });
32 
33  struct HeaderView : QHeaderView {
34  HeaderView(Qt::Orientation orientation, QWidget *parent = nullptr)
35  : QHeaderView(orientation, parent) {}
36  virtual QSize sizeHint() const override {
37  QSize size = QHeaderView::sizeHint();
38  return size;
39  }
40  virtual int sizeHintForRow(int row) const override { return 0; }
41  virtual bool hasHeightForWidth() const override { return true; }
42  virtual int heightForWidth(int w) const override {
43  return sizeHint().height();
44  }
45  };
46  auto *header_view = new HeaderView(Qt::Horizontal, content_widget);
47  QStringList _headers = {{
48  "Bag",
49  "Track",
50  "Span",
51  "Start",
52  }};
53  auto *header_model =
54  new QStandardItemModel(1, _headers.size(), content_widget);
55  header_model->setHorizontalHeaderLabels(_headers);
56  header_view->setModel(header_model);
57  header_view->setSectionResizeMode(QHeaderView::Stretch);
58  header_view->setSortIndicatorShown(true);
59  header_view->setSectionsClickable(true);
60  header_view->setDefaultAlignment(Qt::AlignLeft | Qt::AlignVCenter);
61  header_view->setSizeIncrement(QSize(1, 1));
62  header_view->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
63  header_view->setMinimumSize(0, 0);
64  connect(header_view, &QHeaderView::sortIndicatorChanged, this,
65  [this](int logicalIndex, Qt::SortOrder order) {
66  _sync->_sort_index = logicalIndex;
67  _sync->_sort_ascending = (order != Qt::AscendingOrder);
68  updateSearch(_sync);
69  });
70  main_layout->addWidget(header_view);
71 
72  auto *result_list = new QTreeView(content_widget);
73  main_layout->addWidget(result_list);
74  result_list->setHeaderHidden(true);
75  result_list->setIndentation(4);
76 
77  setWidget(content_widget);
78 
79  class SearchResultModel : public QAbstractTableModel {
80  QWidget *_parent = nullptr;
81  std::shared_ptr<const SearchResults> _search_results;
82 
83  public:
84  SearchResultModel(QWidget *parent)
85  : QAbstractTableModel(parent), _parent(parent) {}
86  QVariant data(const QModelIndex &index, int role) const override {
87  if (role == Qt::DisplayRole) {
88  if (_search_results) {
89  if (index.column() == 0 &&
90  index.row() < _search_results->items.size()) {
91  auto &item = _search_results->items[index.row()];
92  std::string text = item.span_label +
93  (item.span_label.empty() ? "" : " - ") +
94  item.track_label + "\n " + item.branch_name +
95  "\n " + std::to_string(item.start) + " " +
96  std::to_string(item.duration);
97  return QString::fromStdString(text);
98  } else {
99  return QVariant("No results found");
100  }
101  } else {
102  return QVariant("Searching...");
103  }
104  }
105  if (role == Qt::TextAlignmentRole) {
106  if (!_search_results || _search_results->items.empty()) {
107  return QVariant(Qt::AlignCenter);
108  }
109  }
110  if (role == Qt::ForegroundRole) {
111  if (!_search_results || _search_results->items.empty()) {
112  return QVariant(
113  _parent->palette().brush(QPalette::Disabled, QPalette::Text));
114  }
115  }
116  return QVariant();
117  }
118  Qt::ItemFlags flags(const QModelIndex &index) const override {
119  if (!_search_results || _search_results->items.empty()) {
120  return Qt::ItemNeverHasChildren;
121  } else {
122  return Qt::ItemIsSelectable | Qt::ItemNeverHasChildren |
123  Qt::ItemIsEnabled | Qt::ItemIsEditable;
124  }
125  }
126  int rowCount(const QModelIndex &parent = QModelIndex()) const override {
127  if (_search_results) {
128  return std::max(size_t(1), _search_results->items.size());
129  } else {
130  return 1;
131  }
132  }
133  int columnCount(const QModelIndex &parent = QModelIndex()) const override {
134  return 1;
135  }
136  void setSearchResults(
137  const std::shared_ptr<const SearchResults> &search_results) {
138  beginResetModel();
139  _search_results = search_results;
140  endResetModel();
141  LOG_DEBUG("search results changed");
142  if (search_results) {
143  LOG_DEBUG("item count " << search_results->items.size());
144  }
145  }
146  std::shared_ptr<const SearchResults> searchResults() const {
147  return _search_results;
148  }
149  };
150 
151  SearchResultModel *result_model = new SearchResultModel(result_list);
152  result_list->setModel(result_model);
153 
154  class ItemDelegate : public QStyledItemDelegate {
155  SearchResultModel *_model = nullptr;
156  QObject *_parent = nullptr;
157 
158  public:
159  ItemDelegate(SearchResultModel *model, QObject *parent)
160  : QStyledItemDelegate(parent), _model(model), _parent(parent) {}
161  virtual QWidget *createEditor(QWidget *parent,
162  const QStyleOptionViewItem &option,
163  const QModelIndex &index) const override {
164  LOG_DEBUG("open row " << index.row());
165  if (auto results = _model->searchResults()) {
166  if (index.row() < results->items.size()) {
167  LockScope ws;
168  MainWindow::instance()->findAndOpenBag(
169  results->items[index.row()].branch_name);
170  if (ws->player->fileName() ==
171  results->items[index.row()].branch_name) {
172  ActionScope ws("Select");
173  auto span = results->items[index.row()].span;
174  ws->selection() = span;
175  if (span) {
176  ws->player->seek(span->start() + span->duration() * 0.5);
177  }
178  }
179  }
180  }
181  return nullptr;
182  }
183  virtual void paint(QPainter *painter, const QStyleOptionViewItem &option,
184  const QModelIndex &index) const override {
185 
186  QStyleOptionViewItem option2 = option;
187  initStyleOption(&option2, index);
188 
189  {
190  QStyleOptionViewItem option3 = option2;
191  option3.text = "";
192  option3.state |= QStyle::State_Enabled;
193  option3.state |= QStyle::State_Active;
194  QStyle *style = QApplication::style();
195  style->drawControl(QStyle::CE_ItemViewItem, &option3, painter, nullptr);
196  }
197 
198  if (1) {
199  painter->save();
200  painter->setPen(QPen(option2.palette.brush(QPalette::Text), 0));
201  painter->setFont(option2.font);
202  painter->drawText(option2.rect, option2.displayAlignment, option2.text);
203  painter->restore();
204  }
205 
206  if (0) {
207  painter->save();
208  painter->setPen(QPen(option2.palette.brush(QPalette::Text), 0));
209  painter->setFont(option2.font);
210 
211  QTextLayout text_layout(option2.text, option2.font, painter->device());
212  text_layout.draw(painter,
213  QPoint(option2.rect.left(), option2.rect.top()));
214 
215  painter->restore();
216  }
217  }
218  };
219  result_list->setItemDelegate(new ItemDelegate(result_model, this));
220 
221  result_list->setEditTriggers(QAbstractItemView::DoubleClicked |
222  QAbstractItemView::EditKeyPressed);
223 
224  class SearchQuery {
225  std::string _str;
226  std::vector<std::string> _tokens;
227 
228  public:
229  SearchQuery() {}
230  SearchQuery(const std::string &query) : _str(query) {
231  std::istringstream qstream(query);
232  while (qstream) {
233  std::string token;
234  qstream >> token;
235  if (!token.empty()) {
236  _tokens.push_back(token);
237  }
238  }
239  }
240  const std::string &str() const { return _str; }
241  bool match(const std::string &label) const {
242  for (auto &token : _tokens) {
243  if (std::search(label.begin(), label.end(), token.begin(), token.end(),
244  [](char a, char b) {
245  return std::tolower(a) == std::tolower(b);
246  }) == label.end()) {
247  return false;
248  }
249  }
250  return true;
251  }
252  bool empty() const { return _tokens.empty(); }
253  };
254 
255  auto doSearch = [this, sync, result_model](const SearchQuery &query) {
256  LOG_DEBUG("start search " << query.str());
257  auto search_results = std::make_shared<SearchResults>();
258  search_results->query_empty = query.empty();
259  {
260  size_t itrack = 0;
261  size_t ibranch = 0;
262  size_t ispan = 0;
263  while (true) {
264  if (sync->_search_exit || sync->_search_request) {
265  LOG_DEBUG("search aborted " << query.str());
266  return;
267  }
268  SearchResultItem search_result_item;
269  {
270  LockScope ws;
271  auto &tracks = ws->document()->timeline()->tracks();
272  if (itrack >= tracks.size()) {
273  break;
274  }
275  if (auto track =
276  std::dynamic_pointer_cast<AnnotationTrack>(tracks[itrack])) {
277  auto &branches = track->branches();
278  if (ibranch >= branches.size()) {
279  itrack++;
280  ibranch = 0;
281  continue;
282  }
283  auto &branch = branches[ibranch];
284  auto &spans = branch->spans();
285  if (ispan >= spans.size()) {
286  ibranch++;
287  ispan = 0;
288  continue;
289  }
290  auto &span = spans[ispan];
291  search_result_item.branch_name = branch->name();
292  search_result_item.track_label = track->label();
293  search_result_item.span_label = span->label();
294  search_result_item.start = span->start();
295  search_result_item.duration = span->duration();
296  search_result_item.span = span;
297  }
298  }
299  if (query.match(search_result_item.track_label) ||
300  query.match(search_result_item.span_label)) {
301  search_results->items.push_back(search_result_item);
302  }
303  ispan++;
304  }
305  }
306  std::sort(search_results->items.begin(), search_results->items.end(),
307  [sync](const SearchResultItem &a, const SearchResultItem &b) {
308  std::array<int, 4> cmp = {{
309  compareValues(a.branch_name, b.branch_name),
310  compareValues(a.track_label, b.track_label),
311  compareValues(a.span_label, b.span_label),
312  compareValues(a.start, b.start),
313  }};
314  int sig = (sync->_sort_ascending ? -1 : +1);
315  if (cmp[sync->_sort_index] != 0) {
316  return cmp[sync->_sort_index] == sig;
317  }
318  for (auto &v : cmp) {
319  if (v != 0) {
320  return v == sig;
321  }
322  }
323  return false;
324  });
325  bool changed = false;
326  if (sync->_previous_search_results &&
327  (sync->_previous_search_results->items.size() ==
328  search_results->items.size())) {
329  for (size_t i = 0; i < search_results->items.size(); i++) {
330  if (sync->_previous_search_results->items[i].span !=
331  search_results->items[i].span) {
332  changed = true;
333  break;
334  }
335  }
336  } else {
337  changed = true;
338  }
339  if (changed) {
340  startOnMainThreadAsync([this, result_model, search_results]() {
341  result_model->setSearchResults(search_results);
342  });
343  }
344  sync->_previous_search_results = search_results;
345  LOG_DEBUG("search finished " << query.str() << " "
346  << search_results->items.size());
347  };
348 
349  _search_thread = std::thread([this, sync, result_model, doSearch]() {
350  while (true) {
351  SearchQuery query;
352  {
353  std::unique_lock<std::mutex> lock(sync->_search_mutex);
354  while (true) {
355  if (sync->_search_exit) {
356  return;
357  }
358  if (sync->_search_request) {
359  sync->_search_request = false;
360  query = SearchQuery(sync->_search_query);
361  break;
362  }
363  sync->_search_condition.wait(lock);
364  }
365  }
366  std::this_thread::sleep_for(std::chrono::milliseconds(100));
367  if (sync->_search_request || sync->_search_exit) {
368  continue;
369  }
370  doSearch(query);
371  }
372  });
373 
374  updateSearch(sync);
375 
376  {
377  LockScope()->modified.connect(sync, [sync]() { updateSearch(sync); });
378  }
379 }
380 
381 void SearchWidget::updateSearch(const std::shared_ptr<SearchSync> &sync) {
382  std::unique_lock<std::mutex> lock(sync->_search_mutex);
383  if (sync->_visible) {
384  sync->_search_request = true;
385  sync->_search_condition.notify_all();
386  }
387 }
388 
389 SearchWidget::~SearchWidget() {
390  {
391  std::unique_lock<std::mutex> lock(_sync->_search_mutex);
392  _sync->_search_exit = true;
393  _sync->_search_condition.notify_all();
394  }
395  _search_thread.join();
396 }