4 #include "searchwidget.h" 6 #include "../core/bagplayer.h" 7 #include "../core/tracks.h" 8 #include "../core/workspace.h" 10 #include "mainwindow.h" 12 SearchWidget::SearchWidget() : QDockWidget(
"Search") {
16 QWidget *content_widget =
new QWidget();
18 auto *main_layout =
new QVBoxLayout(content_widget);
19 main_layout->setSpacing(0);
20 main_layout->setContentsMargins(0, 0, 0, 0);
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();
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();
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();
46 auto *header_view =
new HeaderView(Qt::Horizontal, content_widget);
47 QStringList _headers = {{
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);
70 main_layout->addWidget(header_view);
72 auto *result_list =
new QTreeView(content_widget);
73 main_layout->addWidget(result_list);
74 result_list->setHeaderHidden(
true);
75 result_list->setIndentation(4);
77 setWidget(content_widget);
79 class SearchResultModel :
public QAbstractTableModel {
80 QWidget *_parent =
nullptr;
81 std::shared_ptr<const SearchResults> _search_results;
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);
99 return QVariant(
"No results found");
102 return QVariant(
"Searching...");
105 if (role == Qt::TextAlignmentRole) {
106 if (!_search_results || _search_results->items.empty()) {
107 return QVariant(Qt::AlignCenter);
110 if (role == Qt::ForegroundRole) {
111 if (!_search_results || _search_results->items.empty()) {
113 _parent->palette().brush(QPalette::Disabled, QPalette::Text));
118 Qt::ItemFlags flags(
const QModelIndex &index)
const override {
119 if (!_search_results || _search_results->items.empty()) {
120 return Qt::ItemNeverHasChildren;
122 return Qt::ItemIsSelectable | Qt::ItemNeverHasChildren |
123 Qt::ItemIsEnabled | Qt::ItemIsEditable;
126 int rowCount(
const QModelIndex &parent = QModelIndex())
const override {
127 if (_search_results) {
128 return std::max(
size_t(1), _search_results->items.size());
133 int columnCount(
const QModelIndex &parent = QModelIndex())
const override {
136 void setSearchResults(
137 const std::shared_ptr<const SearchResults> &search_results) {
139 _search_results = search_results;
141 LOG_DEBUG(
"search results changed");
142 if (search_results) {
143 LOG_DEBUG(
"item count " << search_results->items.size());
146 std::shared_ptr<const SearchResults> searchResults()
const {
147 return _search_results;
151 SearchResultModel *result_model =
new SearchResultModel(result_list);
152 result_list->setModel(result_model);
154 class ItemDelegate :
public QStyledItemDelegate {
155 SearchResultModel *_model =
nullptr;
156 QObject *_parent =
nullptr;
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()) {
168 MainWindow::instance()->findAndOpenBag(
169 results->items[index.row()].branch_name);
170 if (ws->player->fileName() ==
171 results->items[index.row()].branch_name) {
173 auto span = results->items[index.row()].span;
174 ws->selection() = span;
176 ws->player->seek(span->start() + span->duration() * 0.5);
183 virtual void paint(QPainter *painter,
const QStyleOptionViewItem &option,
184 const QModelIndex &index)
const override {
186 QStyleOptionViewItem option2 = option;
187 initStyleOption(&option2, index);
190 QStyleOptionViewItem option3 = option2;
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);
200 painter->setPen(QPen(option2.palette.brush(QPalette::Text), 0));
201 painter->setFont(option2.font);
202 painter->drawText(option2.rect, option2.displayAlignment, option2.text);
208 painter->setPen(QPen(option2.palette.brush(QPalette::Text), 0));
209 painter->setFont(option2.font);
211 QTextLayout text_layout(option2.text, option2.font, painter->device());
212 text_layout.draw(painter,
213 QPoint(option2.rect.left(), option2.rect.top()));
219 result_list->setItemDelegate(
new ItemDelegate(result_model,
this));
221 result_list->setEditTriggers(QAbstractItemView::DoubleClicked |
222 QAbstractItemView::EditKeyPressed);
226 std::vector<std::string> _tokens;
230 SearchQuery(
const std::string &query) : _str(query) {
231 std::istringstream qstream(query);
235 if (!token.empty()) {
236 _tokens.push_back(token);
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(),
245 return std::tolower(a) == std::tolower(b);
252 bool empty()
const {
return _tokens.empty(); }
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();
264 if (sync->_search_exit || sync->_search_request) {
265 LOG_DEBUG(
"search aborted " << query.str());
268 SearchResultItem search_result_item;
271 auto &tracks = ws->document()->timeline()->tracks();
272 if (itrack >= tracks.size()) {
276 std::dynamic_pointer_cast<AnnotationTrack>(tracks[itrack])) {
277 auto &branches = track->branches();
278 if (ibranch >= branches.size()) {
283 auto &branch = branches[ibranch];
284 auto &spans = branch->spans();
285 if (ispan >= spans.size()) {
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;
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);
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),
314 int sig = (sync->_sort_ascending ? -1 : +1);
315 if (cmp[sync->_sort_index] != 0) {
316 return cmp[sync->_sort_index] == sig;
318 for (
auto &v : cmp) {
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) {
340 startOnMainThreadAsync([
this, result_model, search_results]() {
341 result_model->setSearchResults(search_results);
344 sync->_previous_search_results = search_results;
345 LOG_DEBUG(
"search finished " << query.str() <<
" " 346 << search_results->items.size());
349 _search_thread = std::thread([
this, sync, result_model, doSearch]() {
353 std::unique_lock<std::mutex> lock(sync->_search_mutex);
355 if (sync->_search_exit) {
358 if (sync->_search_request) {
359 sync->_search_request =
false;
360 query = SearchQuery(sync->_search_query);
363 sync->_search_condition.wait(lock);
366 std::this_thread::sleep_for(std::chrono::milliseconds(100));
367 if (sync->_search_request || sync->_search_exit) {
377 LockScope()->modified.connect(sync, [sync]() { updateSearch(sync); });
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();
389 SearchWidget::~SearchWidget() {
391 std::unique_lock<std::mutex> lock(_sync->_search_mutex);
392 _sync->_search_exit =
true;
393 _sync->_search_condition.notify_all();
395 _search_thread.join();