10#include <qeventloop.h>
17#include <private/qguiapplication_p.h>
18#include <private/qdesktopunixservices_p.h>
19#include <qpa/qplatformintegration.h>
20#include <qpa/qplatformfontdatabase.h>
25#include <pango/pango.h>
27#if QT_CONFIG(xlib) && defined(GDK_WINDOWING_X11)
31#ifdef GDK_WINDOWING_WAYLAND
32#include <gdk/gdkwayland.h>
40#define PREVIEW_WIDTH 256
41#define PREVIEW_HEIGHT 512
45using namespace Qt::StringLiterals;
50 QGtk3Dialog(GtkWidget *gtkWidget, QPlatformDialogHelper *helper);
56 bool show(Qt::WindowFlags flags, Qt::WindowModality modality, QWindow *parent);
60 static void onResponse(QPlatformDialogHelper *helper,
int response);
64 QPlatformDialogHelper *helper;
65 Qt::WindowModality modality;
69 : gtkWidget(gtkWidget)
72 g_signal_connect_swapped(G_OBJECT(gtkWidget),
"response", G_CALLBACK(onResponse), helper);
73 g_signal_connect(G_OBJECT(gtkWidget),
"delete-event", G_CALLBACK(gtk_widget_hide_on_delete), NULL);
78 gtk_clipboard_store(gtk_clipboard_get(GDK_SELECTION_CLIPBOARD));
79 gtk_widget_destroy(gtkWidget);
84 return GTK_DIALOG(gtkWidget);
89 if (modality == Qt::ApplicationModal) {
95 loop.connect(helper, SIGNAL(accept()), SLOT(quit()));
96 loop.connect(helper, SIGNAL(reject()), SLOT(quit()));
101bool QGtk3Dialog::show(Qt::WindowFlags flags, Qt::WindowModality modality, QWindow *parent)
104 this->modality = modality;
106 gtk_widget_realize(gtkWidget);
108 GdkWindow *gdkWindow = gtk_widget_get_window(gtkWidget);
111#if defined(GDK_WINDOWING_WAYLAND) && GTK_CHECK_VERSION(3
, 22
, 0
)
112 }
else if (GDK_IS_WAYLAND_WINDOW(gdkWindow)) {
113 const auto unixServices =
dynamic_cast<QDesktopUnixServices *>(
114 QGuiApplicationPrivate::platformIntegration()->services());
116 const auto handle = unixServices->portalWindowIdentifier(parent);
117 if (handle.startsWith(
"wayland:"_L1)) {
118 auto handleBa = handle.sliced(8).toUtf8();
119 gdk_wayland_window_set_transient_for_exported(gdkWindow, handleBa.data());
123#if QT_CONFIG(xlib) && defined(GDK_WINDOWING_X11)
124 }
else if (GDK_IS_X11_WINDOW(gdkWindow)) {
125 GdkDisplay *gdkDisplay = gdk_window_get_display(gdkWindow);
126 XSetTransientForHint(gdk_x11_display_get_xdisplay(gdkDisplay),
127 gdk_x11_window_get_xid(gdkWindow),
133 if (modality != Qt::NonModal) {
134 gdk_window_set_modal_hint(gdkWindow,
true);
137 gtk_widget_show(gtkWidget);
138 gdk_window_focus(gdkWindow, GDK_CURRENT_TIME);
144 gtk_widget_hide(gtkWidget);
147void QGtk3Dialog::onResponse(QPlatformDialogHelper *helper,
int response)
149 if (response == GTK_RESPONSE_OK)
150 emit helper->accept();
152 emit helper->reject();
157 d.reset(
new QGtk3Dialog(gtk_color_chooser_dialog_new(
"",
nullptr),
this));
158 g_signal_connect_swapped(d->gtkDialog(),
"notify::rgba", G_CALLBACK(onColorChanged),
this);
168 return d->show(flags, modality, parent);
184 if (color.alpha() < 255)
185 gtk_color_chooser_set_use_alpha(GTK_COLOR_CHOOSER(gtkDialog),
true);
187 gdkColor.red = color.redF();
188 gdkColor.green = color.greenF();
189 gdkColor.blue = color.blueF();
190 gdkColor.alpha = color.alphaF();
191 gtk_color_chooser_set_rgba(GTK_COLOR_CHOOSER(gtkDialog), &gdkColor);
198 gtk_color_chooser_get_rgba(GTK_COLOR_CHOOSER(gtkDialog), &gdkColor);
199 return QColor::fromRgbF(gdkColor.red, gdkColor.green, gdkColor.blue, gdkColor.alpha);
204 emit dialog->currentColorChanged(dialog->currentColor());
210 gtk_window_set_title(GTK_WINDOW(gtkDialog), qUtf8Printable(options()->windowTitle()));
212 gtk_color_chooser_set_use_alpha(GTK_COLOR_CHOOSER(gtkDialog), options()->testOption(QColorDialogOptions::ShowAlphaChannel));
217 d.reset(
new QGtk3Dialog(gtk_file_chooser_dialog_new(
"",
nullptr,
218 GTK_FILE_CHOOSER_ACTION_OPEN,
219 qUtf8Printable(QGtk3Theme::defaultStandardButtonText(QPlatformDialogHelper::Cancel)), GTK_RESPONSE_CANCEL,
220 qUtf8Printable(QGtk3Theme::defaultStandardButtonText(QPlatformDialogHelper::Ok)), GTK_RESPONSE_OK,
223 g_signal_connect(GTK_FILE_CHOOSER(d->gtkDialog()),
"selection-changed", G_CALLBACK(onSelectionChanged),
this);
224 g_signal_connect_swapped(GTK_FILE_CHOOSER(d->gtkDialog()),
"current-folder-changed", G_CALLBACK(onCurrentFolderChanged),
this);
225 g_signal_connect_swapped(GTK_FILE_CHOOSER(d->gtkDialog()),
"notify::filter", G_CALLBACK(onFilterChanged),
this);
227 previewWidget = gtk_image_new();
228 g_signal_connect(G_OBJECT(d->gtkDialog()),
"update-preview", G_CALLBACK(onUpdatePreview),
this);
229 gtk_file_chooser_set_preview_widget(GTK_FILE_CHOOSER(d->gtkDialog()), previewWidget);
242 return d->show(flags, modality, parent);
256 _selection = selectedFiles();
269 gtk_file_chooser_set_current_folder(GTK_FILE_CHOOSER(gtkDialog), qUtf8Printable(directory.toLocalFile()));
281 gchar *folder = gtk_file_chooser_get_current_folder(GTK_FILE_CHOOSER(gtkDialog));
283 ret = QString::fromUtf8(folder);
286 return QUrl::fromLocalFile(ret);
291 setFileChooserAction();
292 selectFileInternal(filename);
298 if (options()->acceptMode() == QFileDialogOptions::AcceptSave) {
300 gtk_file_chooser_set_current_folder(GTK_FILE_CHOOSER(gtkDialog), qUtf8Printable(fi.path()));
301 gtk_file_chooser_set_current_name(GTK_FILE_CHOOSER(gtkDialog), qUtf8Printable(fi.fileName()));
303 gtk_file_chooser_select_filename(GTK_FILE_CHOOSER(gtkDialog), qUtf8Printable(filename.toLocalFile()));
311 if (!_selection.isEmpty())
314 QList<QUrl> selection;
316 GSList *filenames = gtk_file_chooser_get_filenames(GTK_FILE_CHOOSER(gtkDialog));
317 for (GSList *it = filenames; it; it = it->next)
318 selection += QUrl::fromLocalFile(QString::fromUtf8((
const char*)it->data));
319 g_slist_free(filenames);
333 gtk_file_chooser_set_filter(GTK_FILE_CHOOSER(gtkDialog), gtkFilter);
340 GtkFileFilter *gtkFilter = gtk_file_chooser_get_filter(GTK_FILE_CHOOSER(gtkDialog));
341 return _filterNames.value(gtkFilter);
347 gchar *filename = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(gtkDialog));
349 selection = QString::fromUtf8(filename);
352 emit helper->currentChanged(QUrl::fromLocalFile(selection));
357 emit dialog->directoryEntered(dialog->directory());
362 emit dialog->filterSelected(dialog->selectedNameFilter());
367 gchar *filename = gtk_file_chooser_get_preview_filename(GTK_FILE_CHOOSER(gtkDialog));
369 gtk_file_chooser_set_preview_widget_active(GTK_FILE_CHOOSER(gtkDialog),
false);
376 if (!fileinfo.exists() || !fileinfo.isFile()) {
378 gtk_file_chooser_set_preview_widget_active(GTK_FILE_CHOOSER(gtkDialog),
false);
386 gtk_image_set_from_pixbuf(GTK_IMAGE(helper->previewWidget), pixbuf);
387 g_object_unref(pixbuf);
389 gtk_file_chooser_set_preview_widget_active(GTK_FILE_CHOOSER(gtkDialog), pixbuf ?
true :
false);
394 switch (options->fileMode()) {
395 case QFileDialogOptions::AnyFile:
396 case QFileDialogOptions::ExistingFile:
397 case QFileDialogOptions::ExistingFiles:
398 if (options->acceptMode() == QFileDialogOptions::AcceptOpen)
399 return GTK_FILE_CHOOSER_ACTION_OPEN;
401 return GTK_FILE_CHOOSER_ACTION_SAVE;
402 case QFileDialogOptions::Directory:
403 case QFileDialogOptions::DirectoryOnly:
405 if (options->acceptMode() == QFileDialogOptions::AcceptOpen)
406 return GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER;
408 return GTK_FILE_CHOOSER_ACTION_CREATE_FOLDER;
416 const GtkFileChooserAction action = gtkFileChooserAction(options());
417 gtk_file_chooser_set_action(GTK_FILE_CHOOSER(gtkDialog), action);
423 const QSharedPointer<QFileDialogOptions> &opts = options();
425 gtk_window_set_title(GTK_WINDOW(gtkDialog), qUtf8Printable(opts->windowTitle()));
426 gtk_file_chooser_set_local_only(GTK_FILE_CHOOSER(gtkDialog),
true);
428 setFileChooserAction();
430 const bool selectMultiple = opts->fileMode() == QFileDialogOptions::ExistingFiles;
431 gtk_file_chooser_set_select_multiple(GTK_FILE_CHOOSER(gtkDialog), selectMultiple);
433 const bool confirmOverwrite = !opts->testOption(QFileDialogOptions::DontConfirmOverwrite);
434 gtk_file_chooser_set_do_overwrite_confirmation(GTK_FILE_CHOOSER(gtkDialog), confirmOverwrite);
436 const bool readOnly = opts->testOption(QFileDialogOptions::ReadOnly);
437 gtk_file_chooser_set_create_folders(GTK_FILE_CHOOSER(gtkDialog), !readOnly);
439 const QStringList nameFilters = opts->nameFilters();
440 if (!nameFilters.isEmpty())
441 setNameFilters(nameFilters);
443 if (opts->initialDirectory().isLocalFile())
444 setDirectory(opts->initialDirectory());
446 const auto initiallySelected = opts->initiallySelectedFiles();
447 for (
const QUrl &filename : initiallySelected)
448 selectFileInternal(filename);
450 const QString initialNameFilter = opts->initiallySelectedNameFilter();
451 if (!initialNameFilter.isEmpty())
452 selectNameFilter(initialNameFilter);
454 GtkWidget *acceptButton = gtk_dialog_get_widget_for_response(gtkDialog, GTK_RESPONSE_OK);
456 if (opts->isLabelExplicitlySet(QFileDialogOptions::Accept))
457 gtk_button_set_label(GTK_BUTTON(acceptButton), qUtf8Printable(opts->labelText(QFileDialogOptions::Accept)));
458 else if (opts->acceptMode() == QFileDialogOptions::AcceptOpen)
459 gtk_button_set_label(GTK_BUTTON(acceptButton), qUtf8Printable(QGtk3Theme::defaultStandardButtonText(QPlatformDialogHelper::Open)));
461 gtk_button_set_label(GTK_BUTTON(acceptButton), qUtf8Printable(QGtk3Theme::defaultStandardButtonText(QPlatformDialogHelper::Save)));
464 GtkWidget *rejectButton = gtk_dialog_get_widget_for_response(gtkDialog, GTK_RESPONSE_CANCEL);
466 if (opts->isLabelExplicitlySet(QFileDialogOptions::Reject))
467 gtk_button_set_label(GTK_BUTTON(rejectButton), qUtf8Printable(opts->labelText(QFileDialogOptions::Reject)));
469 gtk_button_set_label(GTK_BUTTON(rejectButton), qUtf8Printable(QGtk3Theme::defaultStandardButtonText(QPlatformDialogHelper::Cancel)));
476 foreach (GtkFileFilter *filter, _filters)
477 gtk_file_chooser_remove_filter(GTK_FILE_CHOOSER(gtkDialog), filter);
480 _filterNames.clear();
482 for (
const QString &filter : filters) {
483 GtkFileFilter *gtkFilter = gtk_file_filter_new();
484 const QString name = filter.left(filter.indexOf(u'('));
485 const QStringList extensions = cleanFilterList(filter);
487 gtk_file_filter_set_name(gtkFilter, qUtf8Printable(name.isEmpty() ? extensions.join(
", "_L1) : name));
488 for (
const QString &ext : extensions)
489 gtk_file_filter_add_pattern(gtkFilter, qUtf8Printable(ext));
491 gtk_file_chooser_add_filter(GTK_FILE_CHOOSER(gtkDialog), gtkFilter);
493 _filters.insert(filter, gtkFilter);
494 _filterNames.insert(gtkFilter, filter);
500 d.reset(
new QGtk3Dialog(gtk_font_chooser_dialog_new(
"",
nullptr),
this));
501 g_signal_connect_swapped(d->gtkDialog(),
"notify::font", G_CALLBACK(onFontChanged),
this);
511 return d->show(flags, modality, parent);
526 PangoFontDescription *desc = pango_font_description_new();
527 pango_font_description_set_size(desc, (font.pointSizeF() > 0.0 ? font.pointSizeF() : QFontInfo(font).pointSizeF()) * PANGO_SCALE);
528 pango_font_description_set_family(desc, qUtf8Printable(QFontInfo(font).family()));
530 int weight = font.weight();
531 if (weight >= QFont::Black)
532 pango_font_description_set_weight(desc, PANGO_WEIGHT_HEAVY);
533 else if (weight >= QFont::ExtraBold)
534 pango_font_description_set_weight(desc, PANGO_WEIGHT_ULTRABOLD);
535 else if (weight >= QFont::Bold)
536 pango_font_description_set_weight(desc, PANGO_WEIGHT_BOLD);
537 else if (weight >= QFont::DemiBold)
538 pango_font_description_set_weight(desc, PANGO_WEIGHT_SEMIBOLD);
539 else if (weight >= QFont::Medium)
540 pango_font_description_set_weight(desc, PANGO_WEIGHT_MEDIUM);
541 else if (weight >= QFont::Normal)
542 pango_font_description_set_weight(desc, PANGO_WEIGHT_NORMAL);
543 else if (weight >= QFont::Light)
544 pango_font_description_set_weight(desc, PANGO_WEIGHT_LIGHT);
545 else if (weight >= QFont::ExtraLight)
546 pango_font_description_set_weight(desc, PANGO_WEIGHT_ULTRALIGHT);
548 pango_font_description_set_weight(desc, PANGO_WEIGHT_THIN);
550 int style = font.style();
551 if (style == QFont::StyleItalic)
552 pango_font_description_set_style(desc, PANGO_STYLE_ITALIC);
553 else if (style == QFont::StyleOblique)
554 pango_font_description_set_style(desc, PANGO_STYLE_OBLIQUE);
556 pango_font_description_set_style(desc, PANGO_STYLE_NORMAL);
558 char *str = pango_font_description_to_string(desc);
559 QString name = QString::fromUtf8(str);
560 pango_font_description_free(desc);
568 PangoFontDescription *desc = pango_font_description_from_string(qUtf8Printable(name));
569 font.setPointSizeF(
static_cast<
float>(pango_font_description_get_size(desc)) / PANGO_SCALE);
571 QString family = QString::fromUtf8(pango_font_description_get_family(desc));
572 if (!family.isEmpty())
573 font.setFamilies(QStringList{family});
575 font.setWeight(QFont::Weight(pango_font_description_get_weight(desc)));
577 PangoStyle style = pango_font_description_get_style(desc);
578 if (style == PANGO_STYLE_ITALIC)
579 font.setStyle(QFont::StyleItalic);
580 else if (style == PANGO_STYLE_OBLIQUE)
581 font.setStyle(QFont::StyleOblique);
583 font.setStyle(QFont::StyleNormal);
585 pango_font_description_free(desc);
591 GtkFontChooser *gtkDialog = GTK_FONT_CHOOSER(d->gtkDialog());
592 gtk_font_chooser_set_font(gtkDialog, qUtf8Printable(qt_fontToString(font)));
597 GtkFontChooser *gtkDialog = GTK_FONT_CHOOSER(d->gtkDialog());
598 gchar *name = gtk_font_chooser_get_font(gtkDialog);
599 QFont font = qt_fontFromString(QString::fromUtf8(name));
606 emit dialog->currentFontChanged(dialog->currentFont());
612 const QSharedPointer<QFontDialogOptions> &opts = options();
614 gtk_window_set_title(GTK_WINDOW(gtkDialog), qUtf8Printable(opts->windowTitle()));
619#include "moc_qgtk3dialoghelpers.cpp"
void setCurrentColor(const QColor &color) override
QColor currentColor() const override
~QGtk3ColorDialogHelper()
static void onResponse(QPlatformDialogHelper *helper, int response)
QGtk3Dialog(GtkWidget *gtkWidget, QPlatformDialogHelper *helper)
GtkDialog * gtkDialog() const
bool show(Qt::WindowFlags flags, Qt::WindowModality modality, QWindow *parent)
void setDirectory(const QUrl &directory) override
void setFilter() override
QString selectedNameFilter() const override
void selectNameFilter(const QString &filter) override
void selectFile(const QUrl &filename) override
QList< QUrl > selectedFiles() const override
QUrl directory() const override
bool defaultNameFilterDisables() const override
void setCurrentFont(const QFont &font) override
QFont currentFont() const override
static GtkFileChooserAction gtkFileChooserAction(const QSharedPointer< QFileDialogOptions > &options)
static QFont qt_fontFromString(const QString &name)
static QString qt_fontToString(const QFont &font)
struct _GtkDialog GtkDialog
struct _GtkFileFilter GtkFileFilter