1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
|
// Copyright (C) 2018 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
#include "bookmarkfilter.h"
#include "bookmark.h"
#include "bookmarkmanager.h"
#include "texteditortr.h"
#include <utils/algorithm.h>
using namespace Core;
using namespace Utils;
namespace TextEditor::Internal {
BookmarkFilter::BookmarkFilter(BookmarkManager *manager)
: m_manager(manager)
{
setId("Bookmarks");
setDisplayName(Tr::tr("Bookmarks"));
setDescription(Tr::tr("Locates bookmarks. Filter by file name, by the text on the line of the "
"bookmark, or by the bookmark's note text."));
setPriority(Medium);
setDefaultShortcutString("b");
}
LocatorMatcherTasks BookmarkFilter::matchers()
{
using namespace Tasking;
Storage<LocatorStorage> storage;
const auto onSetup = [this, storage] { storage->reportOutput(match(storage->input())); };
return {{Sync(onSetup), storage}};
}
LocatorFilterEntries BookmarkFilter::match(const QString &input) const
{
if (m_manager->rowCount() == 0)
return {};
const auto match = [this](const QString &name, BookmarkManager::Roles role) {
return m_manager->match(m_manager->index(0, 0), role, name, -1,
Qt::MatchContains | Qt::MatchWrap);
};
const int colonIndex = input.lastIndexOf(':');
QModelIndexList fileNameLineNumberMatches;
if (colonIndex >= 0) {
// Filter by "fileName:lineNumber" pattern
const QString fileName = input.left(colonIndex);
const QString lineNumber = input.mid(colonIndex + 1);
fileNameLineNumberMatches = match(fileName, BookmarkManager::Filename);
fileNameLineNumberMatches = Utils::filtered(
fileNameLineNumberMatches, [lineNumber](const QModelIndex &index) {
return index.data(BookmarkManager::LineNumber).toString().contains(lineNumber);
});
}
const QModelIndexList matches = filteredUnique(fileNameLineNumberMatches
+ match(input, BookmarkManager::Filename)
+ match(input, BookmarkManager::LineNumber)
+ match(input, BookmarkManager::Note)
+ match(input, BookmarkManager::LineText));
LocatorFilterEntries entries;
for (const QModelIndex &idx : matches) {
const Bookmark *bookmark = m_manager->bookmarkForIndex(idx);
const QString filename = bookmark->filePath().fileName();
LocatorFilterEntry entry;
entry.displayName = QString("%1:%2").arg(filename).arg(bookmark->lineNumber());
// TODO: according to QModelIndex docs, we shouldn't store model indexes:
// Model indexes should be used immediately and then discarded.
// You should not rely on indexes to remain valid after calling model functions
// that change the structure of the model or delete items.
entry.acceptor = [manager = m_manager, idx] {
if (Bookmark *bookmark = manager->bookmarkForIndex(idx))
manager->gotoBookmark(bookmark);
return AcceptResult();
};
if (!bookmark->note().isEmpty())
entry.extraInfo = bookmark->note();
else if (!bookmark->lineText().isEmpty())
entry.extraInfo = bookmark->lineText();
else
entry.extraInfo = bookmark->filePath().toString();
int highlightIndex = entry.displayName.indexOf(input, 0, Qt::CaseInsensitive);
if (highlightIndex >= 0) {
entry.highlightInfo = {highlightIndex, int(input.length())};
} else {
highlightIndex = entry.extraInfo.indexOf(input, 0, Qt::CaseInsensitive);
if (highlightIndex >= 0) {
entry.highlightInfo = {highlightIndex, int(input.length()),
LocatorFilterEntry::HighlightInfo::ExtraInfo};
} else if (colonIndex >= 0) {
const QString fileName = input.left(colonIndex);
const QString lineNumber = input.mid(colonIndex + 1);
highlightIndex = entry.displayName.indexOf(fileName, 0,
Qt::CaseInsensitive);
if (highlightIndex >= 0) {
entry.highlightInfo = {highlightIndex, int(fileName.length())};
highlightIndex = entry.displayName.indexOf(
lineNumber, highlightIndex, Qt::CaseInsensitive);
if (highlightIndex >= 0) {
entry.highlightInfo.startsDisplay += highlightIndex;
entry.highlightInfo.lengthsDisplay += lineNumber.length();
}
}
}
}
entry.displayIcon = bookmark->icon();
entries.append(entry);
}
return entries;
}
} // TextEditor::Internal
|