blob: 687213cbf100dc3f7af325d3a9001f20ec2d213d (
plain)
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
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
|
/*
* SPDX-FileCopyrightText: 2019 Ismael Asensio <[email protected]>
*
* SPDX-License-Identifier: GPL-2.0-or-later
*/
#include "dolphinquery.h"
#include <QRegularExpression>
#include "config-dolphin.h"
#if HAVE_BALOO
#include <Baloo/Query>
#endif
namespace {
#if HAVE_BALOO
/** Checks if a given term in the Baloo::Query::searchString() is a special search term
* @return: the specific search token of the term, or an empty QString() if none is found
*/
QString searchTermToken(const QString& term)
{
static const QLatin1String searchTokens[] {
QLatin1String("filename:"),
QLatin1String("modified>="),
QLatin1String("rating>="),
QLatin1String("tag:"), QLatin1String("tag=")
};
for (const auto &searchToken : searchTokens) {
if (term.startsWith(searchToken)) {
return searchToken;
}
}
return QString();
}
QString stripQuotes(const QString& text)
{
if (text.length() >= 2 && text.at(0) == QLatin1Char('"')
&& text.back() == QLatin1Char('"')) {
return text.mid(1, text.size() - 2);
}
return text;
}
QStringList splitOutsideQuotes(const QString& text)
{
// Match groups on 3 possible conditions:
// - Groups with two leading quotes must close both on them (filename:""abc xyz" tuv")
// - Groups enclosed in quotes
// - Words separated by spaces
const QRegularExpression subTermsRegExp("(\\S*?\"\"[^\"]+\"[^\"]+\"+|\\S*?\"[^\"]+\"+|(?<=\\s|^)\\S+(?=\\s|$))");
auto subTermsMatchIterator = subTermsRegExp.globalMatch(text);
QStringList textParts;
while (subTermsMatchIterator.hasNext()) {
textParts << subTermsMatchIterator.next().captured(0);
}
return textParts;
}
#endif
QString trimChar(const QString& text, const QLatin1Char aChar)
{
const int start = text.startsWith(aChar) ? 1 : 0;
const int end = (text.length() > 1 && text.endsWith(aChar)) ? 1 : 0;
return text.mid(start, text.length() - start - end);
}
}
DolphinQuery DolphinQuery::fromSearchUrl(const QUrl& searchUrl)
{
DolphinQuery model;
model.m_searchUrl = searchUrl;
if (searchUrl.scheme() == QLatin1String("baloosearch")) {
model.parseBalooQuery();
} else if (searchUrl.scheme() == QLatin1String("tags")) {
// tags can contain # symbols or slashes within the Url
QString tag = trimChar(searchUrl.toString(QUrl::RemoveScheme), QLatin1Char('/'));
model.m_searchTerms << QStringLiteral("tag:%1").arg(tag);
}
return model;
}
bool DolphinQuery::supportsScheme(const QString& urlScheme)
{
static const QStringList supportedSchemes = {
QStringLiteral("baloosearch"),
QStringLiteral("tags"),
};
return supportedSchemes.contains(urlScheme);
}
void DolphinQuery::parseBalooQuery()
{
#if HAVE_BALOO
const Baloo::Query query = Baloo::Query::fromSearchUrl(m_searchUrl);
m_includeFolder = query.includeFolder();
const QStringList types = query.types();
m_fileType = types.isEmpty() ? QString() : types.first();
QStringList textParts;
QString fileName;
const QStringList subTerms = splitOutsideQuotes(query.searchString());
for (const QString& subTerm : subTerms) {
const QString token = searchTermToken(subTerm);
const QString value = stripQuotes(subTerm.mid(token.length()));
if (token == QLatin1String("filename:")) {
if (!value.isEmpty()) {
fileName = value;
m_hasFileName = true;
}
continue;
} else if (!token.isEmpty()) {
m_searchTerms << token + value;
continue;
} else if (subTerm == QLatin1String("AND") && subTerm != subTerms.at(0) && subTerm != subTerms.back()) {
continue;
} else if (!value.isEmpty()) {
textParts << value;
m_hasContentSearch = true;
}
}
if (m_hasFileName) {
if (m_hasContentSearch) {
textParts << QStringLiteral("filename:\"%1\"").arg(fileName);
} else {
textParts << fileName;
}
}
m_searchText = textParts.join(QLatin1Char(' '));
#endif
}
QUrl DolphinQuery::searchUrl() const
{
return m_searchUrl;
}
QString DolphinQuery::text() const
{
return m_searchText;
}
QString DolphinQuery::type() const
{
return m_fileType;
}
QStringList DolphinQuery::searchTerms() const
{
return m_searchTerms;
}
QString DolphinQuery::includeFolder() const
{
return m_includeFolder;
}
bool DolphinQuery::hasContentSearch() const
{
return m_hasContentSearch;
}
bool DolphinQuery::hasFileName() const
{
return m_hasFileName;
}
|