From 21085db4e36c10cad4d8f8658a98a8ce146f90dd Mon Sep 17 00:00:00 2001 From: "trop[bot]" <37223003+trop[bot]@users.noreply.github.com> Date: Tue, 11 Jun 2024 10:48:58 -0500 Subject: [PATCH] fix: multiple selection in `//shell_dialogs` Portal/KDE implementations (#42426) * fix: multiple selection in //shell_dialogs portal implementation Co-authored-by: Shelley Vohr * fix: allow multiple directory selection in KDE implementation Co-authored-by: Shelley Vohr * chore: update patches --------- Co-authored-by: trop[bot] <37223003+trop[bot]@users.noreply.github.com> Co-authored-by: Shelley Vohr Co-authored-by: PatchUp <73610968+patchup[bot]@users.noreply.github.com> --- ...ing_dialog_features_to_shell_dialogs.patch | 193 ++++++++++++++++-- shell/browser/ui/file_dialog_linux.cc | 5 +- 2 files changed, 179 insertions(+), 19 deletions(-) diff --git a/patches/chromium/feat_add_support_for_missing_dialog_features_to_shell_dialogs.patch b/patches/chromium/feat_add_support_for_missing_dialog_features_to_shell_dialogs.patch index 860e930fe1fe..5de26fa96cf9 100644 --- a/patches/chromium/feat_add_support_for_missing_dialog_features_to_shell_dialogs.patch +++ b/patches/chromium/feat_add_support_for_missing_dialog_features_to_shell_dialogs.patch @@ -186,24 +186,10 @@ index 698f97b23f851c02af037880585c82d4494da63f..0a3b701a701289a2124214e7421a6383 } diff --git a/ui/gtk/select_file_dialog_linux_gtk.h b/ui/gtk/select_file_dialog_linux_gtk.h -index 53ae15f14c45ee72abdae172fc4555c9e4b3ff9a..af181afd9db1351cd886ba24dd651c7bf2f8a716 100644 +index 53ae15f14c45ee72abdae172fc4555c9e4b3ff9a..ee19c3f399a1d060d5e9bd0dc5f1b3828381e8df 100644 --- a/ui/gtk/select_file_dialog_linux_gtk.h +++ b/ui/gtk/select_file_dialog_linux_gtk.h -@@ -15,6 +15,13 @@ - - namespace gtk { - -+struct ExtraSettings { -+ std::string button_label; -+ bool show_overwrite_confirmation = true; -+ bool show_hidden = false; -+ bool allow_multiple_selection = false; -+}; -+ - // Implementation of SelectFileDialog that shows a Gtk common dialog for - // choosing a file or folder. This acts as a modal dialog. - class SelectFileDialogLinuxGtk : public ui::SelectFileDialogLinux, -@@ -90,19 +97,23 @@ class SelectFileDialogLinuxGtk : public ui::SelectFileDialogLinux, +@@ -90,19 +90,23 @@ class SelectFileDialogLinuxGtk : public ui::SelectFileDialogLinux, GtkWidget* CreateSelectFolderDialog(Type type, const std::string& title, const base::FilePath& default_path, @@ -231,7 +217,7 @@ index 53ae15f14c45ee72abdae172fc4555c9e4b3ff9a..af181afd9db1351cd886ba24dd651c7b // Removes and returns the |params| associated with |dialog| from // |params_map_|. -@@ -121,7 +132,8 @@ class SelectFileDialogLinuxGtk : public ui::SelectFileDialogLinux, +@@ -121,7 +125,8 @@ class SelectFileDialogLinuxGtk : public ui::SelectFileDialogLinux, // Common function for CreateFileOpenDialog and CreateMultiFileOpenDialog. GtkWidget* CreateFileOpenHelper(const std::string& title, const base::FilePath& default_path, @@ -241,3 +227,176 @@ index 53ae15f14c45ee72abdae172fc4555c9e4b3ff9a..af181afd9db1351cd886ba24dd651c7b // Callback for when the user responds to a Save As or Open File dialog. void OnSelectSingleFileDialogResponse(GtkWidget* dialog, int response_id); +diff --git a/ui/shell_dialogs/select_file_dialog_linux.h b/ui/shell_dialogs/select_file_dialog_linux.h +index 20ad001988831afca73315c577f90c824a36e282..57a8d35ace583eaafb526f70935d21c0f8fd1078 100644 +--- a/ui/shell_dialogs/select_file_dialog_linux.h ++++ b/ui/shell_dialogs/select_file_dialog_linux.h +@@ -26,6 +26,13 @@ class SHELL_DIALOGS_EXPORT SelectFileDialogLinux : public SelectFileDialog { + SelectFileDialogLinux(const SelectFileDialogLinux&) = delete; + SelectFileDialogLinux& operator=(const SelectFileDialogLinux&) = delete; + ++ struct ExtraSettings { ++ std::string button_label; ++ bool show_overwrite_confirmation = true; ++ bool show_hidden = false; ++ bool allow_multiple_selection = false; ++ }; ++ + // Returns true if the SelectFileDialog class returned by + // NewSelectFileDialogImplKDE will actually work. + static bool CheckKDEDialogWorksOnUIThread(std::string& kdialog_version); +diff --git a/ui/shell_dialogs/select_file_dialog_linux_kde.cc b/ui/shell_dialogs/select_file_dialog_linux_kde.cc +index 796e98cd42a5c6087da6cdf1d7bff4248113aeab..bcf43ab96bcb426fde6362dd0da4421758854449 100644 +--- a/ui/shell_dialogs/select_file_dialog_linux_kde.cc ++++ b/ui/shell_dialogs/select_file_dialog_linux_kde.cc +@@ -479,6 +479,9 @@ void SelectFileDialogLinuxKde::CreateSelectFolderDialog( + int title_message_id = (type == SELECT_UPLOAD_FOLDER) + ? IDS_SELECT_UPLOAD_FOLDER_DIALOG_TITLE + : IDS_SELECT_FOLDER_DIALOG_TITLE; ++ ExtraSettings extra_settings; ++ if (params) ++ extra_settings = *(static_cast(params)); + pipe_task_runner_->PostTaskAndReplyWithResult( + FROM_HERE, + base::BindOnce( +@@ -486,7 +489,7 @@ void SelectFileDialogLinuxKde::CreateSelectFolderDialog( + KDialogParams( + "--getexistingdirectory", GetTitle(title, title_message_id), + default_path.empty() ? *last_opened_path() : default_path, parent, +- false, false)), ++ false, extra_settings.allow_multiple_selection)), + base::BindOnce( + &SelectFileDialogLinuxKde::OnSelectSingleFolderDialogResponse, this, + parent, params)); +diff --git a/ui/shell_dialogs/select_file_dialog_linux_portal.cc b/ui/shell_dialogs/select_file_dialog_linux_portal.cc +index decba61300c21f7f5d070b24c23ff2e08b06d161..ae6e76186b6db9d0d32d51baaaeafa6106225c0f 100644 +--- a/ui/shell_dialogs/select_file_dialog_linux_portal.cc ++++ b/ui/shell_dialogs/select_file_dialog_linux_portal.cc +@@ -218,6 +218,10 @@ void SelectFileDialogLinuxPortal::SelectFileImpl( + info_->main_task_runner = base::SequencedTaskRunner::GetCurrentDefault(); + listener_params_ = params; + ++ ExtraSettings extra_settings; ++ if (params) ++ extra_settings = *(static_cast(params)); ++ + if (owning_window) { + if (auto* root = owning_window->GetRootWindow()) { + if (auto* host = root->GetNativeWindowProperty( +@@ -245,7 +249,7 @@ void SelectFileDialogLinuxPortal::SelectFileImpl( + host_->GetAcceleratedWidget(), + base::BindOnce( + &SelectFileDialogLinuxPortal::SelectFileImplWithParentHandle, +- this, title, default_path, filter_set, default_extension))) { ++ this, title, default_path, filter_set, default_extension, extra_settings))) { + // Return early to skip the fallback below. + return; + } else { +@@ -255,7 +259,7 @@ void SelectFileDialogLinuxPortal::SelectFileImpl( + + // No parent, so just use a blank parent handle. + SelectFileImplWithParentHandle(title, default_path, filter_set, +- default_extension, ""); ++ default_extension, extra_settings, ""); + } + + bool SelectFileDialogLinuxPortal::HasMultipleFileTypeChoicesImpl() { +@@ -452,6 +456,7 @@ void SelectFileDialogLinuxPortal::SelectFileImplWithParentHandle( + base::FilePath default_path, + PortalFilterSet filter_set, + base::FilePath::StringType default_extension, ++ const ExtraSettings& settings, + std::string parent_handle) { + bool default_path_exists = CallDirectoryExistsOnUIThread(default_path); + dbus_thread_linux::GetTaskRunner()->PostTask( +@@ -460,7 +465,7 @@ void SelectFileDialogLinuxPortal::SelectFileImplWithParentHandle( + &SelectFileDialogLinuxPortal::DialogInfo::SelectFileImplOnBusThread, + info_, std::move(title), std::move(default_path), default_path_exists, + std::move(filter_set), std::move(default_extension), +- std::move(parent_handle))); ++ std::move(parent_handle), std::move(settings))); + } + + void SelectFileDialogLinuxPortal::DialogInfo::SelectFileImplOnBusThread( +@@ -469,7 +474,8 @@ void SelectFileDialogLinuxPortal::DialogInfo::SelectFileImplOnBusThread( + const bool default_path_exists, + PortalFilterSet filter_set, + base::FilePath::StringType default_extension, +- std::string parent_handle) { ++ std::string parent_handle, ++ const ExtraSettings& settings) { + DCHECK(dbus_thread_linux::GetTaskRunner()->RunsTasksInCurrentSequence()); + dbus::Bus* bus = AcquireBusOnBusThread(); + if (!bus->Connect()) +@@ -515,7 +521,7 @@ void SelectFileDialogLinuxPortal::DialogInfo::SelectFileImplOnBusThread( + base::StringPrintf("handle_%d", handle_token_counter_++); + + AppendOptions(&writer, response_handle_token, default_path, +- default_path_exists, filter_set); ++ default_path_exists, filter_set, settings); + + // The sender part of the handle object contains the D-Bus connection name + // without the prefix colon and with all dots replaced with underscores. +@@ -545,7 +551,8 @@ void SelectFileDialogLinuxPortal::DialogInfo::AppendOptions( + const std::string& response_handle_token, + const base::FilePath& default_path, + const bool default_path_exists, +- const SelectFileDialogLinuxPortal::PortalFilterSet& filter_set) { ++ const SelectFileDialogLinuxPortal::PortalFilterSet& filter_set, ++ const ExtraSettings& settings) { + dbus::MessageWriter options_writer(nullptr); + writer->OpenArray("{sv}", &options_writer); + +@@ -553,8 +560,10 @@ void SelectFileDialogLinuxPortal::DialogInfo::AppendOptions( + response_handle_token); + + if (type == SelectFileDialog::Type::SELECT_UPLOAD_FOLDER) { +- AppendStringOption(&options_writer, kFileChooserOptionAcceptLabel, +- l10n_util::GetStringUTF8( ++ const std::string accept_label = settings.button_label.empty() ++ ? kFileChooserOptionAcceptLabel ++ : settings.button_label; ++ AppendStringOption(&options_writer, accept_label, l10n_util::GetStringUTF8( + IDS_SELECT_UPLOAD_FOLDER_DIALOG_UPLOAD_BUTTON)); + } + +@@ -562,6 +571,7 @@ void SelectFileDialogLinuxPortal::DialogInfo::AppendOptions( + type == SelectFileDialog::Type::SELECT_UPLOAD_FOLDER || + type == SelectFileDialog::Type::SELECT_EXISTING_FOLDER) { + AppendBoolOption(&options_writer, kFileChooserOptionDirectory, true); ++ AppendBoolOption(&options_writer, kFileChooserOptionMultiple, settings.allow_multiple_selection); + } else if (type == SelectFileDialog::Type::SELECT_OPEN_MULTI_FILE) { + AppendBoolOption(&options_writer, kFileChooserOptionMultiple, true); + } +diff --git a/ui/shell_dialogs/select_file_dialog_linux_portal.h b/ui/shell_dialogs/select_file_dialog_linux_portal.h +index c487f7da19e2d05696a8eb72f2fa3e12972149f3..02a40c571570974dcc61e1b1f7ed95fbfc2bedf2 100644 +--- a/ui/shell_dialogs/select_file_dialog_linux_portal.h ++++ b/ui/shell_dialogs/select_file_dialog_linux_portal.h +@@ -115,7 +115,8 @@ class SelectFileDialogLinuxPortal : public SelectFileDialogLinux { + const bool default_path_exists, + PortalFilterSet filter_set, + base::FilePath::StringType default_extension, +- std::string parent_handle); ++ std::string parent_handle, ++ const ExtraSettings& settings); + Type type; + // The task runner the SelectFileImpl method was called on. + scoped_refptr main_task_runner; +@@ -143,7 +144,8 @@ class SelectFileDialogLinuxPortal : public SelectFileDialogLinux { + const std::string& response_handle_token, + const base::FilePath& default_path, + const bool derfault_path_exists, +- const PortalFilterSet& filter_set); ++ const PortalFilterSet& filter_set, ++ const ExtraSettings& settings); + void AppendFilterStruct(dbus::MessageWriter* writer, + const PortalFilter& filter); + std::vector ConvertUrisToPaths( +@@ -190,6 +192,7 @@ class SelectFileDialogLinuxPortal : public SelectFileDialogLinux { + base::FilePath default_path, + PortalFilterSet filter_set, + base::FilePath::StringType default_extension, ++ const ExtraSettings& settings, + std::string parent_handle); + + void DialogCreatedOnMainThread(); diff --git a/shell/browser/ui/file_dialog_linux.cc b/shell/browser/ui/file_dialog_linux.cc index 336825a8660b..b24a869043aa 100644 --- a/shell/browser/ui/file_dialog_linux.cc +++ b/shell/browser/ui/file_dialog_linux.cc @@ -67,8 +67,9 @@ class FileChooserDialog : public ui::SelectFileDialog::Listener { ~FileChooserDialog() override = default; - gtk::ExtraSettings GetExtraSettings(const DialogSettings& settings) { - gtk::ExtraSettings extra; + ui::SelectFileDialogLinux::ExtraSettings GetExtraSettings( + const DialogSettings& settings) { + ui::SelectFileDialogLinux::ExtraSettings extra; extra.button_label = settings.button_label; extra.show_overwrite_confirmation = settings.properties & SAVE_DIALOG_SHOW_OVERWRITE_CONFIRMATION;