electron/atom/browser/ui/file_dialog_gtk.cc

303 lines
10 KiB
C++
Raw Normal View History

// Copyright (c) 2014 GitHub, Inc.
2014-04-25 09:49:37 +00:00
// Use of this source code is governed by the MIT license that can be
// found in the LICENSE file.
2014-03-16 00:30:26 +00:00
#include "atom/browser/ui/file_dialog.h"
2018-02-10 01:29:32 +00:00
#include <glib/gi18n.h> // _() macro
#include "atom/browser/native_window_views.h"
#include "atom/browser/unresponsive_suppressor.h"
#include "base/callback.h"
#include "base/files/file_util.h"
2014-08-06 06:49:02 +00:00
#include "base/strings/string_util.h"
2017-01-26 10:55:19 +00:00
#include "chrome/browser/ui/libgtkui/gtk_util.h"
#include "ui/base/glib/glib_signal.h"
#include "ui/views/widget/desktop_aura/x11_desktop_handler.h"
namespace file_dialog {
DialogSettings::DialogSettings() = default;
DialogSettings::DialogSettings(const DialogSettings&) = default;
DialogSettings::~DialogSettings() = default;
namespace {
static const int kPreviewWidth = 256;
static const int kPreviewHeight = 512;
2014-08-06 06:49:02 +00:00
// Makes sure that .jpg also shows .JPG.
gboolean FileFilterCaseInsensitive(const GtkFileFilterInfo* file_info,
std::string* file_extension) {
2015-08-21 03:06:29 +00:00
// Makes .* file extension matches all file types.
if (*file_extension == ".*")
return true;
2018-02-15 21:20:55 +00:00
return base::EndsWith(file_info->filename, *file_extension,
base::CompareCase::INSENSITIVE_ASCII);
2014-08-06 06:49:02 +00:00
}
// Deletes |data| when gtk_file_filter_add_custom() is done with it.
void OnFileFilterDataDestroyed(std::string* file_extension) {
delete file_extension;
}
class FileChooserDialog {
public:
2018-02-15 21:20:55 +00:00
FileChooserDialog(GtkFileChooserAction action, const DialogSettings& settings)
2017-02-08 01:32:58 +00:00
: parent_(static_cast<atom::NativeWindowViews*>(settings.parent_window)),
filters_(settings.filters) {
2018-02-09 15:57:39 +00:00
const char* confirm_text = _("_OK");
2016-05-15 06:02:27 +00:00
2017-02-08 01:32:58 +00:00
if (!settings.button_label.empty())
confirm_text = settings.button_label.c_str();
2016-05-15 06:02:27 +00:00
else if (action == GTK_FILE_CHOOSER_ACTION_SAVE)
2018-02-09 15:57:39 +00:00
confirm_text = _("_Save");
2014-03-13 05:58:53 +00:00
else if (action == GTK_FILE_CHOOSER_ACTION_OPEN)
2018-02-09 15:57:39 +00:00
confirm_text = _("_Open");
2014-03-13 05:58:53 +00:00
dialog_ = gtk_file_chooser_dialog_new(
2018-02-15 21:20:55 +00:00
settings.title.c_str(), NULL, action, _("_Cancel"), GTK_RESPONSE_CANCEL,
confirm_text, GTK_RESPONSE_ACCEPT, NULL);
if (parent_) {
parent_->SetEnabled(false);
2017-01-26 10:55:19 +00:00
libgtkui::SetGtkTransientForAura(dialog_, parent_->GetNativeWindow());
gtk_window_set_modal(GTK_WINDOW(dialog_), TRUE);
}
2014-03-13 05:20:43 +00:00
if (action == GTK_FILE_CHOOSER_ACTION_SAVE)
gtk_file_chooser_set_do_overwrite_confirmation(GTK_FILE_CHOOSER(dialog_),
TRUE);
2014-03-13 05:27:01 +00:00
if (action != GTK_FILE_CHOOSER_ACTION_OPEN)
gtk_file_chooser_set_create_folders(GTK_FILE_CHOOSER(dialog_), TRUE);
2014-03-13 05:20:43 +00:00
2017-02-08 01:32:58 +00:00
if (!settings.default_path.empty()) {
if (base::DirectoryExists(settings.default_path)) {
2018-02-15 21:20:55 +00:00
gtk_file_chooser_set_current_folder(
GTK_FILE_CHOOSER(dialog_), settings.default_path.value().c_str());
} else {
if (settings.default_path.IsAbsolute()) {
gtk_file_chooser_set_current_folder(
GTK_FILE_CHOOSER(dialog_),
settings.default_path.DirName().value().c_str());
}
2018-02-15 21:20:55 +00:00
gtk_file_chooser_set_current_name(
GTK_FILE_CHOOSER(dialog_),
2017-02-08 01:32:58 +00:00
settings.default_path.BaseName().value().c_str());
}
2014-03-13 05:10:13 +00:00
}
2014-08-06 06:49:02 +00:00
2017-02-08 01:32:58 +00:00
if (!settings.filters.empty())
AddFilters(settings.filters);
preview_ = gtk_image_new();
g_signal_connect(dialog_, "update-preview",
G_CALLBACK(OnUpdatePreviewThunk), this);
gtk_file_chooser_set_preview_widget(GTK_FILE_CHOOSER(dialog_), preview_);
}
~FileChooserDialog() {
gtk_widget_destroy(dialog_);
if (parent_)
parent_->SetEnabled(true);
}
void SetupProperties(int properties) {
const auto hasProp = [properties](FileDialogProperty prop) {
return gboolean((properties & prop) != 0);
};
auto* file_chooser = GTK_FILE_CHOOSER(dialog());
gtk_file_chooser_set_select_multiple(file_chooser,
hasProp(FILE_DIALOG_MULTI_SELECTIONS));
gtk_file_chooser_set_show_hidden(file_chooser,
hasProp(FILE_DIALOG_SHOW_HIDDEN_FILES));
}
2014-03-13 05:58:53 +00:00
void RunAsynchronous() {
g_signal_connect(dialog_, "delete-event",
G_CALLBACK(gtk_widget_hide_on_delete), NULL);
2018-02-15 21:20:55 +00:00
g_signal_connect(dialog_, "response", G_CALLBACK(OnFileDialogResponseThunk),
this);
gtk_widget_show_all(dialog_);
// We need to call gtk_window_present after making the widgets visible to
// make sure window gets correctly raised and gets focus.
2017-01-26 10:55:19 +00:00
int time = ui::X11EventSource::GetInstance()->GetTimestamp();
gtk_window_present_with_time(GTK_WINDOW(dialog_), time);
}
2014-03-13 05:58:53 +00:00
void RunSaveAsynchronous(const SaveDialogCallback& callback) {
save_callback_ = callback;
RunAsynchronous();
}
void RunOpenAsynchronous(const OpenDialogCallback& callback) {
open_callback_ = callback;
RunAsynchronous();
}
base::FilePath GetFileName() const {
gchar* filename = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(dialog_));
const base::FilePath path(filename);
g_free(filename);
return path;
}
2014-03-13 05:58:53 +00:00
std::vector<base::FilePath> GetFileNames() const {
std::vector<base::FilePath> paths;
auto* filenames = gtk_file_chooser_get_filenames(GTK_FILE_CHOOSER(dialog_));
for (auto* iter = filenames; iter != NULL; iter = iter->next) {
auto* filename = static_cast<char*>(iter->data);
paths.emplace_back(filename);
g_free(filename);
2014-03-13 05:58:53 +00:00
}
g_slist_free(filenames);
return paths;
}
CHROMEG_CALLBACK_1(FileChooserDialog,
void,
OnFileDialogResponse,
GtkWidget*,
int);
GtkWidget* dialog() const { return dialog_; }
private:
2014-08-06 06:49:02 +00:00
void AddFilters(const Filters& filters);
atom::NativeWindowViews* parent_;
atom::UnresponsiveSuppressor unresponsive_suppressor_;
2015-07-07 07:45:13 +00:00
GtkWidget* dialog_;
GtkWidget* preview_;
Filters filters_;
SaveDialogCallback save_callback_;
OpenDialogCallback open_callback_;
// Callback for when we update the preview for the selection.
CHROMEG_CALLBACK_0(FileChooserDialog, void, OnUpdatePreview, GtkWidget*);
DISALLOW_COPY_AND_ASSIGN(FileChooserDialog);
};
2014-03-13 05:58:53 +00:00
void FileChooserDialog::OnFileDialogResponse(GtkWidget* widget, int response) {
gtk_widget_hide(dialog_);
2014-03-13 05:58:53 +00:00
if (!save_callback_.is_null()) {
if (response == GTK_RESPONSE_ACCEPT)
save_callback_.Run(true, GetFileName());
else
save_callback_.Run(false, base::FilePath());
} else if (!open_callback_.is_null()) {
if (response == GTK_RESPONSE_ACCEPT)
open_callback_.Run(true, GetFileNames());
else
open_callback_.Run(false, std::vector<base::FilePath>());
}
delete this;
}
2014-08-06 06:49:02 +00:00
void FileChooserDialog::AddFilters(const Filters& filters) {
for (size_t i = 0; i < filters.size(); ++i) {
const Filter& filter = filters[i];
GtkFileFilter* gtk_filter = gtk_file_filter_new();
for (size_t j = 0; j < filter.second.size(); ++j) {
auto file_extension =
std::make_unique<std::string>("." + filter.second[j]);
2014-08-06 06:49:02 +00:00
gtk_file_filter_add_custom(
2018-02-15 21:20:55 +00:00
gtk_filter, GTK_FILE_FILTER_FILENAME,
2014-08-06 06:49:02 +00:00
reinterpret_cast<GtkFileFilterFunc>(FileFilterCaseInsensitive),
file_extension.release(),
reinterpret_cast<GDestroyNotify>(OnFileFilterDataDestroyed));
}
gtk_file_filter_set_name(gtk_filter, filter.first.c_str());
gtk_file_chooser_add_filter(GTK_FILE_CHOOSER(dialog_), gtk_filter);
}
}
void FileChooserDialog::OnUpdatePreview(GtkWidget* chooser) {
gchar* filename =
gtk_file_chooser_get_preview_filename(GTK_FILE_CHOOSER(chooser));
if (!filename) {
gtk_file_chooser_set_preview_widget_active(GTK_FILE_CHOOSER(chooser),
FALSE);
return;
}
// Don't attempt to open anything which isn't a regular file. If a named pipe,
// this may hang. See https://crbug.com/534754.
struct stat stat_buf;
if (stat(filename, &stat_buf) != 0 || !S_ISREG(stat_buf.st_mode)) {
g_free(filename);
gtk_file_chooser_set_preview_widget_active(GTK_FILE_CHOOSER(chooser),
FALSE);
return;
}
// This will preserve the image's aspect ratio.
GdkPixbuf* pixbuf = gdk_pixbuf_new_from_file_at_size(filename, kPreviewWidth,
kPreviewHeight, nullptr);
g_free(filename);
if (pixbuf) {
gtk_image_set_from_pixbuf(GTK_IMAGE(preview_), pixbuf);
g_object_unref(pixbuf);
}
gtk_file_chooser_set_preview_widget_active(GTK_FILE_CHOOSER(chooser),
pixbuf ? TRUE : FALSE);
}
} // namespace
2017-02-08 01:32:58 +00:00
bool ShowOpenDialog(const DialogSettings& settings,
std::vector<base::FilePath>* paths) {
GtkFileChooserAction action = GTK_FILE_CHOOSER_ACTION_OPEN;
2017-02-08 01:32:58 +00:00
if (settings.properties & FILE_DIALOG_OPEN_DIRECTORY)
action = GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER;
2017-02-08 01:32:58 +00:00
FileChooserDialog open_dialog(action, settings);
open_dialog.SetupProperties(settings.properties);
gtk_widget_show_all(open_dialog.dialog());
int response = gtk_dialog_run(GTK_DIALOG(open_dialog.dialog()));
if (response == GTK_RESPONSE_ACCEPT) {
*paths = open_dialog.GetFileNames();
return true;
} else {
return false;
}
}
2017-02-08 01:32:58 +00:00
void ShowOpenDialog(const DialogSettings& settings,
const OpenDialogCallback& callback) {
2014-03-13 05:58:53 +00:00
GtkFileChooserAction action = GTK_FILE_CHOOSER_ACTION_OPEN;
2017-02-08 01:32:58 +00:00
if (settings.properties & FILE_DIALOG_OPEN_DIRECTORY)
2014-03-13 05:58:53 +00:00
action = GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER;
2017-02-08 01:32:58 +00:00
FileChooserDialog* open_dialog = new FileChooserDialog(action, settings);
open_dialog->SetupProperties(settings.properties);
2014-03-13 05:58:53 +00:00
open_dialog->RunOpenAsynchronous(callback);
}
2018-02-15 21:20:55 +00:00
bool ShowSaveDialog(const DialogSettings& settings, base::FilePath* path) {
2017-02-08 01:32:58 +00:00
FileChooserDialog save_dialog(GTK_FILE_CHOOSER_ACTION_SAVE, settings);
gtk_widget_show_all(save_dialog.dialog());
int response = gtk_dialog_run(GTK_DIALOG(save_dialog.dialog()));
if (response == GTK_RESPONSE_ACCEPT) {
*path = save_dialog.GetFileName();
return true;
} else {
return false;
}
}
2017-02-08 01:32:58 +00:00
void ShowSaveDialog(const DialogSettings& settings,
const SaveDialogCallback& callback) {
2018-02-15 21:20:55 +00:00
FileChooserDialog* save_dialog =
new FileChooserDialog(GTK_FILE_CHOOSER_ACTION_SAVE, settings);
2014-03-13 05:58:53 +00:00
save_dialog->RunSaveAsynchronous(callback);
}
} // namespace file_dialog