7#include <private/qdesktopunixservices_p.h>
8#include <private/qguiapplication_p.h>
9#include <qpa/qplatformintegration.h>
11#include <QDBusConnection>
12#include <QDBusMessage>
13#include <QDBusPendingCall>
14#include <QDBusPendingCallWatcher>
15#include <QDBusPendingReply>
16#include <QDBusMetaType>
23#include <QMimeDatabase>
24#include <QRandomGenerator>
26#include <QRegularExpression>
30using namespace Qt::StringLiterals;
35 arg << filterCondition
.type << filterCondition.pattern;
43 QString filterPattern;
45 arg >> type >> filterPattern;
47 filterCondition.pattern = filterPattern;
56 arg << filter.name << filter.filterConditions;
64 QXdgDesktopPortalFileDialog::FilterConditionList filterConditions;
66 arg >> name >> filterConditions;
68 filter.filterConditions = filterConditions;
105 Q_D(QXdgDesktopPortalFileDialog);
107 if (d->nativeFileDialog) {
108 connect(d->nativeFileDialog.get(), SIGNAL(accept()),
this, SIGNAL(accept()));
109 connect(d->nativeFileDialog.get(), SIGNAL(reject()),
this, SIGNAL(reject()));
112 d->loop.connect(
this, SIGNAL(accept()), SLOT(quit()));
113 d->loop.connect(
this, SIGNAL(reject()), SLOT(quit()));
122 Q_D(QXdgDesktopPortalFileDialog);
124 if (d->nativeFileDialog)
125 d->nativeFileDialog->setOptions(options());
127 if (options()->fileMode() == QFileDialogOptions::ExistingFiles)
128 d->multipleFiles =
true;
130 if (options()->fileMode() == QFileDialogOptions::Directory || options()->fileMode() == QFileDialogOptions::DirectoryOnly)
131 d->directoryMode =
true;
133 if (options()->isLabelExplicitlySet(QFileDialogOptions::Accept))
134 d->acceptLabel = options()->labelText(QFileDialogOptions::Accept);
136 if (!options()->windowTitle().isEmpty())
137 d->title = options()->windowTitle();
139 if (options()->acceptMode() == QFileDialogOptions::AcceptSave)
142 if (!options()->nameFilters().isEmpty())
143 d->nameFilters = options()->nameFilters();
145 if (!options()->mimeTypeFilters().isEmpty())
146 d->mimeTypesFilters = options()->mimeTypeFilters();
148 if (!options()->initiallySelectedMimeTypeFilter().isEmpty())
149 d->selectedMimeTypeFilter = options()->initiallySelectedMimeTypeFilter();
151 if (!options()->initiallySelectedNameFilter().isEmpty())
152 d->selectedNameFilter = options()->initiallySelectedNameFilter();
154 setDirectory(options()->initialDirectory());
159 Q_D(QXdgDesktopPortalFileDialog);
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);
166 if (!d->acceptLabel.isEmpty())
167 options.insert(
"accept_label"_L1, d->acceptLabel);
169 options.insert(
"modal"_L1, windowModality != Qt::NonModal);
170 options.insert(
"multiple"_L1, d->multipleFiles);
171 options.insert(
"directory"_L1, d->directoryMode);
173 if (!d->directory.isEmpty())
174 options.insert(
"current_folder"_L1, QFile::encodeName(d->directory.toLocalFile()).append(
'\0'));
176 if (d->saveFile && !d->selectedFiles.isEmpty()) {
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());
189 qDBusRegisterMetaType<FilterConditionList>();
190 qDBusRegisterMetaType<Filter>();
191 qDBusRegisterMetaType<FilterList>();
193 FilterList filterList;
194 auto selectedFilterIndex = filterList.size() - 1;
196 d->userVisibleToNameFilter.clear();
198 if (!d->mimeTypesFilters.isEmpty()) {
199 for (
const QString &mimeTypefilter : d->mimeTypesFilters) {
200 QMimeDatabase mimeDatabase;
201 QMimeType mimeType = mimeDatabase.mimeTypeForName(mimeTypefilter);
204 FilterCondition filterCondition;
205 filterCondition.type = MimeType;
206 filterCondition.pattern = mimeTypefilter;
209 FilterConditionList filterConditions;
210 filterConditions << filterCondition;
214 filter.name = mimeType.comment();
215 filter.filterConditions = filterConditions;
217 if (filter.name.isEmpty())
218 filter.name = mimeTypefilter;
220 filterList << filter;
222 if (!d->selectedMimeTypeFilter.isEmpty() && d->selectedMimeTypeFilter == mimeTypefilter)
223 selectedFilterIndex = filterList.size() - 1;
225 }
else if (!d->nameFilters.isEmpty()) {
226 for (
const QString &nameFilter : d->nameFilters) {
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);
235 if (filterStrings.isEmpty()) {
236 qWarning() <<
"Filter " << userVisibleName <<
" is empty and will be ignored.";
240 FilterConditionList filterConditions;
241 for (
const QString &filterString : filterStrings) {
242 FilterCondition filterCondition;
243 filterCondition.type = GlobalPattern;
244 filterCondition.pattern = filterString;
245 filterConditions << filterCondition;
249 filter.name = userVisibleName;
250 filter.filterConditions = filterConditions;
252 filterList << filter;
254 d->userVisibleToNameFilter.insert(userVisibleName, nameFilter);
256 if (!d->selectedNameFilter.isEmpty() && d->selectedNameFilter == nameFilter)
257 selectedFilterIndex = filterList.size() - 1;
262 if (!filterList.isEmpty())
263 options.insert(
"filters"_L1, QVariant::fromValue(filterList));
265 if (selectedFilterIndex != -1)
266 options.insert(
"current_filter"_L1, QVariant::fromValue(filterList[selectedFilterIndex]));
268 options.insert(
"handle_token"_L1, QStringLiteral(
"qt%1").arg(QRandomGenerator::global()->generate()));
273 auto unixServices =
dynamic_cast<QDesktopUnixServices *>(
274 QGuiApplicationPrivate::platformIntegration()->services());
275 if (parent && unixServices)
276 message << unixServices->portalWindowIdentifier(parent);
278 message << QString();
280 message << d->title << options;
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;
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();
297 QDBusConnection::sessionBus().connect(
nullptr,
298 reply.value().path(),
299 "org.freedesktop.portal.Request"_L1,
302 SLOT(gotResponse(uint,QVariantMap)));
304 watcher->deleteLater();
315 Q_D(QXdgDesktopPortalFileDialog);
317 if (d->nativeFileDialog) {
318 d->nativeFileDialog->setOptions(options());
319 d->nativeFileDialog->setDirectory(directory);
322 d->directory = directory;
327 Q_D(
const QXdgDesktopPortalFileDialog);
329 if (d->nativeFileDialog && useNativeFileDialog())
330 return d->nativeFileDialog->directory();
337 Q_D(QXdgDesktopPortalFileDialog);
339 if (d->nativeFileDialog) {
340 d->nativeFileDialog->setOptions(options());
341 d->nativeFileDialog->selectFile(filename);
344 d->selectedFiles << filename.path();
349 Q_D(
const QXdgDesktopPortalFileDialog);
351 if (d->nativeFileDialog && useNativeFileDialog())
352 return d->nativeFileDialog->selectedFiles();
355 for (
const QString &file : d->selectedFiles) {
363 Q_D(QXdgDesktopPortalFileDialog);
365 if (d->nativeFileDialog) {
366 d->nativeFileDialog->setOptions(options());
367 d->nativeFileDialog->setFilter();
373 Q_D(QXdgDesktopPortalFileDialog);
374 if (d->nativeFileDialog) {
375 d->nativeFileDialog->setOptions(options());
376 d->nativeFileDialog->selectMimeTypeFilter(filter);
382 Q_D(
const QXdgDesktopPortalFileDialog);
383 return d->selectedMimeTypeFilter;
388 Q_D(QXdgDesktopPortalFileDialog);
390 if (d->nativeFileDialog) {
391 d->nativeFileDialog->setOptions(options());
392 d->nativeFileDialog->selectNameFilter(filter);
398 Q_D(
const QXdgDesktopPortalFileDialog);
399 return d->selectedNameFilter;
404 Q_D(QXdgDesktopPortalFileDialog);
406 if (d->nativeFileDialog && useNativeFileDialog()) {
407 d->nativeFileDialog->exec();
417 Q_D(QXdgDesktopPortalFileDialog);
419 if (d->nativeFileDialog)
420 d->nativeFileDialog->hide();
425 Q_D(QXdgDesktopPortalFileDialog);
429 if (d->nativeFileDialog && useNativeFileDialog(OpenFallback))
430 return d->nativeFileDialog->show(windowFlags, windowModality, parent);
432 openPortal(windowFlags, windowModality, parent);
439 Q_D(QXdgDesktopPortalFileDialog);
442 if (results.contains(
"uris"_L1))
443 d->selectedFiles = results.value(
"uris"_L1).toStringList();
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) {
449 d->selectedMimeTypeFilter = selectedFilter.filterConditions[0].pattern;
450 d->selectedNameFilter.clear();
452 d->selectedNameFilter = d->userVisibleToNameFilter.value(selectedFilter.name);
453 d->selectedMimeTypeFilter.clear();
464 Q_D(
const QXdgDesktopPortalFileDialog);
466 if (d->failedToOpen && fallbackType != OpenFallback)
469 if (d->fileChooserPortalVersion < 3) {
470 if (options()->fileMode() == QFileDialogOptions::Directory)
472 else if (options()->fileMode() == QFileDialogOptions::DirectoryOnly)
481#include "moc_qxdgdesktopportalfiledialog_p.cpp"
QMap< QString, QString > userVisibleToNameFilter
std::unique_ptr< QPlatformFileDialogHelper > nativeFileDialog
QStringList selectedFiles
uint fileChooserPortalVersion
QString selectedMimeTypeFilter
QXdgDesktopPortalFileDialogPrivate(QPlatformFileDialogHelper *nativeFileDialog, uint fileChooserPortalVersion)
QString selectedNameFilter
QStringList mimeTypesFilters
QString selectedNameFilter() const override
void selectNameFilter(const QString &filter) override
QString selectedMimeTypeFilter() const override
void setFilter() override
bool defaultNameFilterDisables() const override
bool show(Qt::WindowFlags windowFlags, Qt::WindowModality windowModality, QWindow *parent) override
void selectMimeTypeFilter(const QString &filter) override
~QXdgDesktopPortalFileDialog()
void setDirectory(const QUrl &directory) override
void selectFile(const QUrl &filename) override
QUrl directory() const 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)