Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qxdgdesktopportalfiledialog.cpp
Go to the documentation of this file.
1// Copyright (C) 2017-2018 Red Hat, Inc
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3// Qt-Security score:significant reason:default
4
6
7#include <private/qdesktopunixservices_p.h>
8#include <private/qguiapplication_p.h>
9#include <qpa/qplatformintegration.h>
10
11#include <QDBusConnection>
12#include <QDBusMessage>
13#include <QDBusPendingCall>
14#include <QDBusPendingCallWatcher>
15#include <QDBusPendingReply>
16#include <QDBusMetaType>
17
18#include <QEventLoop>
19#include <QFile>
20#include <QFileInfo>
21#include <QMetaType>
22#include <QMimeType>
23#include <QMimeDatabase>
24#include <QRandomGenerator>
25#include <QWindow>
26#include <QRegularExpression>
27
28QT_BEGIN_NAMESPACE
29
30using namespace Qt::StringLiterals;
31
32QDBusArgument &operator <<(QDBusArgument &arg, const QXdgDesktopPortalFileDialog::FilterCondition &filterCondition)
33{
34 arg.beginStructure();
35 arg << filterCondition.type << filterCondition.pattern;
36 arg.endStructure();
37 return arg;
38}
39
40const QDBusArgument &operator >>(const QDBusArgument &arg, QXdgDesktopPortalFileDialog::FilterCondition &filterCondition)
41{
42 uint type;
43 QString filterPattern;
44 arg.beginStructure();
45 arg >> type >> filterPattern;
46 filterCondition.type = (QXdgDesktopPortalFileDialog::ConditionType)type;
47 filterCondition.pattern = filterPattern;
48 arg.endStructure();
49
50 return arg;
51}
52
53QDBusArgument &operator <<(QDBusArgument &arg, const QXdgDesktopPortalFileDialog::Filter filter)
54{
55 arg.beginStructure();
56 arg << filter.name << filter.filterConditions;
57 arg.endStructure();
58 return arg;
59}
60
61const QDBusArgument &operator >>(const QDBusArgument &arg, QXdgDesktopPortalFileDialog::Filter &filter)
62{
63 QString name;
64 QXdgDesktopPortalFileDialog::FilterConditionList filterConditions;
65 arg.beginStructure();
66 arg >> name >> filterConditions;
67 filter.name = name;
68 filter.filterConditions = filterConditions;
69 arg.endStructure();
70
71 return arg;
72}
73
100
101QXdgDesktopPortalFileDialog::QXdgDesktopPortalFileDialog(QPlatformFileDialogHelper *nativeFileDialog, uint fileChooserPortalVersion)
104{
105 Q_D(QXdgDesktopPortalFileDialog);
106
107 if (d->nativeFileDialog) {
108 connect(d->nativeFileDialog.get(), SIGNAL(accept()), this, SIGNAL(accept()));
109 connect(d->nativeFileDialog.get(), SIGNAL(reject()), this, SIGNAL(reject()));
110 }
111
112 d->loop.connect(this, SIGNAL(accept()), SLOT(quit()));
113 d->loop.connect(this, SIGNAL(reject()), SLOT(quit()));
114}
115
119
120void QXdgDesktopPortalFileDialog::initializeDialog()
121{
122 Q_D(QXdgDesktopPortalFileDialog);
123
124 if (d->nativeFileDialog)
125 d->nativeFileDialog->setOptions(options());
126
127 if (options()->fileMode() == QFileDialogOptions::ExistingFiles)
128 d->multipleFiles = true;
129
130 if (options()->fileMode() == QFileDialogOptions::Directory || options()->fileMode() == QFileDialogOptions::DirectoryOnly)
131 d->directoryMode = true;
132
133 if (options()->isLabelExplicitlySet(QFileDialogOptions::Accept))
134 d->acceptLabel = options()->labelText(QFileDialogOptions::Accept);
135
136 if (!options()->windowTitle().isEmpty())
137 d->title = options()->windowTitle();
138
139 if (options()->acceptMode() == QFileDialogOptions::AcceptSave)
140 d->saveFile = true;
141
142 if (!options()->nameFilters().isEmpty())
143 d->nameFilters = options()->nameFilters();
144
145 if (!options()->mimeTypeFilters().isEmpty())
146 d->mimeTypesFilters = options()->mimeTypeFilters();
147
148 if (!options()->initiallySelectedMimeTypeFilter().isEmpty())
149 d->selectedMimeTypeFilter = options()->initiallySelectedMimeTypeFilter();
150
151 if (!options()->initiallySelectedNameFilter().isEmpty())
152 d->selectedNameFilter = options()->initiallySelectedNameFilter();
153
154 setDirectory(options()->initialDirectory());
155}
156
157void QXdgDesktopPortalFileDialog::openPortal(Qt::WindowFlags windowFlags, Qt::WindowModality windowModality, QWindow *parent)
158{
159 Q_D(QXdgDesktopPortalFileDialog);
160
161 QDBusMessage message = QDBusMessage::createMethodCall("org.freedesktop.portal.Desktop"_L1,
162 "/org/freedesktop/portal/desktop"_L1,
163 "org.freedesktop.portal.FileChooser"_L1,
164 d->saveFile ? "SaveFile"_L1 : "OpenFile"_L1);
165 QVariantMap options;
166 if (!d->acceptLabel.isEmpty())
167 options.insert("accept_label"_L1, d->acceptLabel);
168
169 options.insert("modal"_L1, windowModality != Qt::NonModal);
170 options.insert("multiple"_L1, d->multipleFiles);
171 options.insert("directory"_L1, d->directoryMode);
172
173 if (!d->directory.isEmpty())
174 options.insert("current_folder"_L1, QFile::encodeName(d->directory.toLocalFile()).append('\0'));
175
176 if (d->saveFile && !d->selectedFiles.isEmpty()) {
177 // current_file for the file to be pre-selected, current_name for the file name to be
178 // pre-filled current_file accepts absolute path and requires the file to exist while
179 // current_name accepts just file name
180 QFileInfo selectedFileInfo(d->selectedFiles.constFirst());
181 if (selectedFileInfo.exists())
182 options.insert("current_file"_L1,
183 QFile::encodeName(d->selectedFiles.constFirst()).append('\0'));
184 options.insert("current_name"_L1, selectedFileInfo.fileName());
185 }
186
187 // Insert filters
188 qDBusRegisterMetaType<FilterCondition>();
189 qDBusRegisterMetaType<FilterConditionList>();
190 qDBusRegisterMetaType<Filter>();
191 qDBusRegisterMetaType<FilterList>();
192
193 FilterList filterList;
194 auto selectedFilterIndex = filterList.size() - 1;
195
196 d->userVisibleToNameFilter.clear();
197
198 if (!d->mimeTypesFilters.isEmpty()) {
199 for (const QString &mimeTypefilter : d->mimeTypesFilters) {
200 QMimeDatabase mimeDatabase;
201 QMimeType mimeType = mimeDatabase.mimeTypeForName(mimeTypefilter);
202
203 // Creates e.g. (1, "image/png")
204 FilterCondition filterCondition;
205 filterCondition.type = MimeType;
206 filterCondition.pattern = mimeTypefilter;
207
208 // Creates e.g. [((1, "image/png"))]
209 FilterConditionList filterConditions;
210 filterConditions << filterCondition;
211
212 // Creates e.g. [("Images", [((1, "image/png"))])]
213 Filter filter;
214 filter.name = mimeType.comment();
215 filter.filterConditions = filterConditions;
216
217 if (filter.name.isEmpty())
218 filter.name = mimeTypefilter;
219
220 filterList << filter;
221
222 if (!d->selectedMimeTypeFilter.isEmpty() && d->selectedMimeTypeFilter == mimeTypefilter)
223 selectedFilterIndex = filterList.size() - 1;
224 }
225 } else if (!d->nameFilters.isEmpty()) {
226 for (const QString &nameFilter : d->nameFilters) {
227 // Do parsing:
228 // Supported format is ("Images (*.png *.jpg)")
229 QRegularExpression regexp(QPlatformFileDialogHelper::filterRegExp);
230 QRegularExpressionMatch match = regexp.match(nameFilter);
231 if (match.hasMatch()) {
232 QString userVisibleName = match.captured(1);
233 QStringList filterStrings = match.captured(2).split(u' ', Qt::SkipEmptyParts);
234
235 if (filterStrings.isEmpty()) {
236 qWarning() << "Filter " << userVisibleName << " is empty and will be ignored.";
237 continue;
238 }
239
240 FilterConditionList filterConditions;
241 for (const QString &filterString : filterStrings) {
242 FilterCondition filterCondition;
243 filterCondition.type = GlobalPattern;
244 filterCondition.pattern = filterString;
245 filterConditions << filterCondition;
246 }
247
248 Filter filter;
249 filter.name = userVisibleName;
250 filter.filterConditions = filterConditions;
251
252 filterList << filter;
253
254 d->userVisibleToNameFilter.insert(userVisibleName, nameFilter);
255
256 if (!d->selectedNameFilter.isEmpty() && d->selectedNameFilter == nameFilter)
257 selectedFilterIndex = filterList.size() - 1;
258 }
259 }
260 }
261
262 if (!filterList.isEmpty())
263 options.insert("filters"_L1, QVariant::fromValue(filterList));
264
265 if (selectedFilterIndex != -1)
266 options.insert("current_filter"_L1, QVariant::fromValue(filterList[selectedFilterIndex]));
267
268 options.insert("handle_token"_L1, QStringLiteral("qt%1").arg(QRandomGenerator::global()->generate()));
269
270 // TODO choices a(ssa(ss)s)
271 // List of serialized combo boxes to add to the file chooser.
272
273 auto unixServices = dynamic_cast<QDesktopUnixServices *>(
274 QGuiApplicationPrivate::platformIntegration()->services());
275 if (parent && unixServices)
276 message << unixServices->portalWindowIdentifier(parent);
277 else
278 message << QString();
279
280 message << d->title << options;
281
282 QDBusPendingCall pendingCall = QDBusConnection::sessionBus().asyncCall(message);
283 QDBusPendingCallWatcher *watcher = new QDBusPendingCallWatcher(pendingCall);
284 connect(watcher, &QDBusPendingCallWatcher::finished, this, [this, d, windowFlags, windowModality, parent] (QDBusPendingCallWatcher *watcher) {
285 QDBusPendingReply<QDBusObjectPath> reply = *watcher;
286 // Any error means the dialog is not shown and we need to fallback
287 d->failedToOpen = reply.isError();
288 if (d->failedToOpen) {
289 if (d->nativeFileDialog) {
290 d->nativeFileDialog->show(windowFlags, windowModality, parent);
291 if (d->loop.isRunning())
292 d->nativeFileDialog->exec();
293 } else {
294 Q_EMIT reject();
295 }
296 } else {
297 QDBusConnection::sessionBus().connect(nullptr,
298 reply.value().path(),
299 "org.freedesktop.portal.Request"_L1,
300 "Response"_L1,
301 this,
302 SLOT(gotResponse(uint,QVariantMap)));
303 }
304 watcher->deleteLater();
305 });
306}
307
309{
310 return false;
311}
312
313void QXdgDesktopPortalFileDialog::setDirectory(const QUrl &directory)
314{
315 Q_D(QXdgDesktopPortalFileDialog);
316
317 if (d->nativeFileDialog) {
318 d->nativeFileDialog->setOptions(options());
319 d->nativeFileDialog->setDirectory(directory);
320 }
321
322 d->directory = directory;
323}
324
326{
327 Q_D(const QXdgDesktopPortalFileDialog);
328
329 if (d->nativeFileDialog && useNativeFileDialog())
330 return d->nativeFileDialog->directory();
331
332 return d->directory;
333}
334
335void QXdgDesktopPortalFileDialog::selectFile(const QUrl &filename)
336{
337 Q_D(QXdgDesktopPortalFileDialog);
338
339 if (d->nativeFileDialog) {
340 d->nativeFileDialog->setOptions(options());
341 d->nativeFileDialog->selectFile(filename);
342 }
343
344 d->selectedFiles << filename.path();
345}
346
348{
349 Q_D(const QXdgDesktopPortalFileDialog);
350
351 if (d->nativeFileDialog && useNativeFileDialog())
352 return d->nativeFileDialog->selectedFiles();
353
354 QList<QUrl> files;
355 for (const QString &file : d->selectedFiles) {
356 files << QUrl(file);
357 }
358 return files;
359}
360
362{
363 Q_D(QXdgDesktopPortalFileDialog);
364
365 if (d->nativeFileDialog) {
366 d->nativeFileDialog->setOptions(options());
367 d->nativeFileDialog->setFilter();
368 }
369}
370
372{
373 Q_D(QXdgDesktopPortalFileDialog);
374 if (d->nativeFileDialog) {
375 d->nativeFileDialog->setOptions(options());
376 d->nativeFileDialog->selectMimeTypeFilter(filter);
377 }
378}
379
381{
382 Q_D(const QXdgDesktopPortalFileDialog);
383 return d->selectedMimeTypeFilter;
384}
385
387{
388 Q_D(QXdgDesktopPortalFileDialog);
389
390 if (d->nativeFileDialog) {
391 d->nativeFileDialog->setOptions(options());
392 d->nativeFileDialog->selectNameFilter(filter);
393 }
394}
395
397{
398 Q_D(const QXdgDesktopPortalFileDialog);
399 return d->selectedNameFilter;
400}
401
403{
404 Q_D(QXdgDesktopPortalFileDialog);
405
406 if (d->nativeFileDialog && useNativeFileDialog()) {
407 d->nativeFileDialog->exec();
408 return;
409 }
410
411 // HACK we have to avoid returning until we emit that the dialog was accepted or rejected
412 d->loop.exec();
413}
414
416{
417 Q_D(QXdgDesktopPortalFileDialog);
418
419 if (d->nativeFileDialog)
420 d->nativeFileDialog->hide();
421}
422
423bool QXdgDesktopPortalFileDialog::show(Qt::WindowFlags windowFlags, Qt::WindowModality windowModality, QWindow *parent)
424{
425 Q_D(QXdgDesktopPortalFileDialog);
426
427 initializeDialog();
428
429 if (d->nativeFileDialog && useNativeFileDialog(OpenFallback))
430 return d->nativeFileDialog->show(windowFlags, windowModality, parent);
431
432 openPortal(windowFlags, windowModality, parent);
433
434 return true;
435}
436
437void QXdgDesktopPortalFileDialog::gotResponse(uint response, const QVariantMap &results)
438{
439 Q_D(QXdgDesktopPortalFileDialog);
440
441 if (!response) {
442 if (results.contains("uris"_L1))
443 d->selectedFiles = results.value("uris"_L1).toStringList();
444
445 if (results.contains("current_filter"_L1)) {
446 const Filter selectedFilter = qdbus_cast<Filter>(results.value(QStringLiteral("current_filter")));
447 if (!selectedFilter.filterConditions.empty() && selectedFilter.filterConditions[0].type == MimeType) {
448 // s.a. QXdgDesktopPortalFileDialog::openPortal which basically does the inverse
449 d->selectedMimeTypeFilter = selectedFilter.filterConditions[0].pattern;
450 d->selectedNameFilter.clear();
451 } else {
452 d->selectedNameFilter = d->userVisibleToNameFilter.value(selectedFilter.name);
453 d->selectedMimeTypeFilter.clear();
454 }
455 }
456 Q_EMIT accept();
457 } else {
458 Q_EMIT reject();
459 }
460}
461
462bool QXdgDesktopPortalFileDialog::useNativeFileDialog(QXdgDesktopPortalFileDialog::FallbackType fallbackType) const
463{
464 Q_D(const QXdgDesktopPortalFileDialog);
465
466 if (d->failedToOpen && fallbackType != OpenFallback)
467 return true;
468
469 if (d->fileChooserPortalVersion < 3) {
470 if (options()->fileMode() == QFileDialogOptions::Directory)
471 return true;
472 else if (options()->fileMode() == QFileDialogOptions::DirectoryOnly)
473 return true;
474 }
475
476 return false;
477}
478
479QT_END_NAMESPACE
480
481#include "moc_qxdgdesktopportalfiledialog_p.cpp"
std::unique_ptr< QPlatformFileDialogHelper > nativeFileDialog
QXdgDesktopPortalFileDialogPrivate(QPlatformFileDialogHelper *nativeFileDialog, uint fileChooserPortalVersion)
void selectNameFilter(const QString &filter) override
QString selectedMimeTypeFilter() const override
bool show(Qt::WindowFlags windowFlags, Qt::WindowModality windowModality, QWindow *parent) override
void selectMimeTypeFilter(const QString &filter) override
void setDirectory(const QUrl &directory) override
void selectFile(const QUrl &filename) override
QXdgDesktopPortalFileDialog(QPlatformFileDialogHelper *nativeFileDialog=nullptr, uint fileChooserPortalVersion=0)
QList< QUrl > selectedFiles() const override
const QDBusArgument & operator>>(const QDBusArgument &argument, QXdgDBusImageVector &iconVector)
QTextStream & operator<<(QTextStream &s, QTextStreamFunction f)
QDBusArgument & operator<<(QDBusArgument &arg, const QXdgDesktopPortalFileDialog::FilterCondition &filterCondition)
const QDBusArgument & operator>>(const QDBusArgument &arg, QXdgDesktopPortalFileDialog::FilterCondition &filterCondition)