2014-10-31 18:17:05 +00:00
|
|
|
// 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
|
2014-02-14 14:07:23 +00:00
|
|
|
// found in the LICENSE file.
|
|
|
|
|
2014-03-16 00:30:26 +00:00
|
|
|
#include "atom/browser/ui/file_dialog.h"
|
2014-02-14 14:07:23 +00:00
|
|
|
|
2014-07-04 16:00:54 +00:00
|
|
|
#include <gdk/gdk.h>
|
|
|
|
#include <gdk/gdkx.h>
|
|
|
|
#include <gtk/gtk.h>
|
|
|
|
|
2014-07-21 13:48:23 +00:00
|
|
|
// This conflicts with mate::Converter,
|
|
|
|
#undef True
|
|
|
|
#undef False
|
|
|
|
// and V8.
|
|
|
|
#undef None
|
|
|
|
|
2014-03-16 01:37:04 +00:00
|
|
|
#include "atom/browser/native_window.h"
|
2014-02-14 14:07:23 +00:00
|
|
|
#include "base/callback.h"
|
2015-01-10 01:45:50 +00:00
|
|
|
#include "base/files/file_util.h"
|
2014-08-06 06:49:02 +00:00
|
|
|
#include "base/strings/string_util.h"
|
2014-07-04 16:00:54 +00:00
|
|
|
#include "chrome/browser/ui/libgtk2ui/gtk2_signal.h"
|
|
|
|
#include "ui/aura/window.h"
|
|
|
|
#include "ui/aura/window_tree_host.h"
|
2014-10-01 09:02:00 +00:00
|
|
|
#include "ui/views/widget/desktop_aura/x11_desktop_handler.h"
|
2014-02-14 14:07:23 +00:00
|
|
|
|
|
|
|
namespace file_dialog {
|
|
|
|
|
2014-03-13 04:58:27 +00:00
|
|
|
namespace {
|
|
|
|
|
2014-07-04 16:00:54 +00:00
|
|
|
const char kAuraTransientParent[] = "aura-transient-parent";
|
|
|
|
|
|
|
|
void SetGtkTransientForAura(GtkWidget* dialog, aura::Window* parent) {
|
|
|
|
if (!parent || !parent->GetHost())
|
|
|
|
return;
|
|
|
|
|
|
|
|
gtk_widget_realize(dialog);
|
|
|
|
GdkWindow* gdk_window = gtk_widget_get_window(dialog);
|
|
|
|
|
|
|
|
// TODO(erg): Check to make sure we're using X11 if wayland or some other
|
|
|
|
// display server ever happens. Otherwise, this will crash.
|
|
|
|
XSetTransientForHint(GDK_WINDOW_XDISPLAY(gdk_window),
|
|
|
|
GDK_WINDOW_XID(gdk_window),
|
|
|
|
parent->GetHost()->GetAcceleratedWidget());
|
|
|
|
|
|
|
|
// We also set the |parent| as a property of |dialog|, so that we can unlink
|
|
|
|
// the two later.
|
|
|
|
g_object_set_data(G_OBJECT(dialog), kAuraTransientParent, parent);
|
|
|
|
}
|
|
|
|
|
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) {
|
|
|
|
return EndsWith(file_info->filename, *file_extension, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Deletes |data| when gtk_file_filter_add_custom() is done with it.
|
|
|
|
void OnFileFilterDataDestroyed(std::string* file_extension) {
|
|
|
|
delete file_extension;
|
|
|
|
}
|
|
|
|
|
2014-03-13 04:58:27 +00:00
|
|
|
class FileChooserDialog {
|
|
|
|
public:
|
|
|
|
FileChooserDialog(GtkFileChooserAction action,
|
|
|
|
atom::NativeWindow* parent_window,
|
|
|
|
const std::string& title,
|
2014-08-06 06:49:02 +00:00
|
|
|
const base::FilePath& default_path,
|
|
|
|
const Filters& filters)
|
2014-06-24 13:52:06 +00:00
|
|
|
: dialog_scope_(new atom::NativeWindow::DialogScope(parent_window)) {
|
2014-03-13 05:58:53 +00:00
|
|
|
const char* confirm_text = GTK_STOCK_OK;
|
|
|
|
if (action == GTK_FILE_CHOOSER_ACTION_SAVE)
|
|
|
|
confirm_text = GTK_STOCK_SAVE;
|
|
|
|
else if (action == GTK_FILE_CHOOSER_ACTION_OPEN)
|
|
|
|
confirm_text = GTK_STOCK_OPEN;
|
|
|
|
|
2014-03-13 04:58:27 +00:00
|
|
|
dialog_ = gtk_file_chooser_dialog_new(
|
|
|
|
title.c_str(),
|
2014-07-04 16:00:54 +00:00
|
|
|
NULL,
|
2014-03-13 04:58:27 +00:00
|
|
|
action,
|
|
|
|
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
|
2014-03-13 05:58:53 +00:00
|
|
|
confirm_text, GTK_RESPONSE_ACCEPT,
|
2014-03-13 04:58:27 +00:00
|
|
|
NULL);
|
2014-07-04 16:00:54 +00:00
|
|
|
if (parent_window) {
|
|
|
|
gfx::NativeWindow window = parent_window->GetNativeWindow();
|
|
|
|
SetGtkTransientForAura(dialog_, window);
|
|
|
|
}
|
2014-03-13 05:03:38 +00:00
|
|
|
|
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
|
|
|
|
2014-03-13 05:03:38 +00:00
|
|
|
gtk_window_set_modal(GTK_WINDOW(dialog_), TRUE);
|
2014-03-13 05:10:13 +00:00
|
|
|
|
|
|
|
if (!default_path.empty()) {
|
2015-06-04 02:08:16 +00:00
|
|
|
if (base::DirectoryExists(default_path)) {
|
2014-03-13 05:10:13 +00:00
|
|
|
gtk_file_chooser_set_current_folder(GTK_FILE_CHOOSER(dialog_),
|
|
|
|
default_path.value().c_str());
|
2015-06-04 02:08:16 +00:00
|
|
|
} else {
|
|
|
|
gtk_file_chooser_set_current_folder(GTK_FILE_CHOOSER(dialog_),
|
|
|
|
default_path.DirName().value().c_str());
|
|
|
|
gtk_file_chooser_set_current_name(GTK_FILE_CHOOSER(dialog_),
|
|
|
|
default_path.BaseName().value().c_str());
|
|
|
|
}
|
2014-03-13 05:10:13 +00:00
|
|
|
}
|
2014-08-06 06:49:02 +00:00
|
|
|
|
|
|
|
if (!filters.empty())
|
|
|
|
AddFilters(filters);
|
2014-03-13 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual ~FileChooserDialog() {
|
|
|
|
gtk_widget_destroy(dialog_);
|
|
|
|
}
|
|
|
|
|
2014-03-13 05:58:53 +00:00
|
|
|
void RunAsynchronous() {
|
2014-03-13 04:58:27 +00:00
|
|
|
g_signal_connect(dialog_, "delete-event",
|
|
|
|
G_CALLBACK(gtk_widget_hide_on_delete), NULL);
|
|
|
|
g_signal_connect(dialog_, "response",
|
2014-03-13 05:58:53 +00:00
|
|
|
G_CALLBACK(OnFileDialogResponseThunk), this);
|
2014-03-13 04:58:27 +00:00
|
|
|
gtk_widget_show_all(dialog_);
|
2014-10-01 09:02:00 +00:00
|
|
|
|
|
|
|
// We need to call gtk_window_present after making the widgets visible to
|
|
|
|
// make sure window gets correctly raised and gets focus.
|
|
|
|
int time = views::X11DesktopHandler::get()->wm_user_time_ms();
|
|
|
|
gtk_window_present_with_time(GTK_WINDOW(dialog_), time);
|
2014-03-13 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
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();
|
|
|
|
}
|
|
|
|
|
2014-03-13 05:17:03 +00:00
|
|
|
base::FilePath GetFileName() const {
|
|
|
|
gchar* filename = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(dialog_));
|
2014-03-13 05:22:49 +00:00
|
|
|
base::FilePath path(filename);
|
|
|
|
g_free(filename);
|
|
|
|
return path;
|
2014-03-13 05:17:03 +00:00
|
|
|
}
|
|
|
|
|
2014-03-13 05:58:53 +00:00
|
|
|
std::vector<base::FilePath> GetFileNames() const {
|
|
|
|
std::vector<base::FilePath> paths;
|
|
|
|
GSList* filenames = gtk_file_chooser_get_filenames(
|
|
|
|
GTK_FILE_CHOOSER(dialog_));
|
|
|
|
for (GSList* iter = filenames; iter != NULL; iter = g_slist_next(iter)) {
|
|
|
|
base::FilePath path(static_cast<char*>(iter->data));
|
|
|
|
g_free(iter->data);
|
|
|
|
paths.push_back(path);
|
|
|
|
}
|
|
|
|
g_slist_free(filenames);
|
|
|
|
return paths;
|
|
|
|
}
|
|
|
|
|
|
|
|
CHROMEGTK_CALLBACK_1(FileChooserDialog, void, OnFileDialogResponse, int);
|
2014-03-13 05:17:03 +00:00
|
|
|
|
|
|
|
GtkWidget* dialog() const { return dialog_; }
|
2014-03-13 04:58:27 +00:00
|
|
|
|
|
|
|
private:
|
2014-08-06 06:49:02 +00:00
|
|
|
void AddFilters(const Filters& filters);
|
|
|
|
|
2014-03-13 04:58:27 +00:00
|
|
|
GtkWidget* dialog_;
|
|
|
|
|
|
|
|
SaveDialogCallback save_callback_;
|
|
|
|
OpenDialogCallback open_callback_;
|
|
|
|
|
2014-06-24 13:52:06 +00:00
|
|
|
scoped_ptr<atom::NativeWindow::DialogScope> dialog_scope_;
|
|
|
|
|
2014-03-13 04:58:27 +00:00
|
|
|
DISALLOW_COPY_AND_ASSIGN(FileChooserDialog);
|
|
|
|
};
|
|
|
|
|
2014-03-13 05:58:53 +00:00
|
|
|
void FileChooserDialog::OnFileDialogResponse(GtkWidget* widget, int response) {
|
2014-03-13 04:58:27 +00:00
|
|
|
gtk_widget_hide_all(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>());
|
|
|
|
}
|
2014-03-13 04:58:27 +00:00
|
|
|
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) {
|
|
|
|
scoped_ptr<std::string> file_extension(
|
|
|
|
new std::string("." + filter.second[j]));
|
|
|
|
gtk_file_filter_add_custom(
|
|
|
|
gtk_filter,
|
|
|
|
GTK_FILE_FILTER_FILENAME,
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-13 04:58:27 +00:00
|
|
|
} // namespace
|
|
|
|
|
2014-02-14 14:07:23 +00:00
|
|
|
bool ShowOpenDialog(atom::NativeWindow* parent_window,
|
|
|
|
const std::string& title,
|
|
|
|
const base::FilePath& default_path,
|
2014-08-06 04:44:02 +00:00
|
|
|
const Filters& filters,
|
2014-02-14 14:07:23 +00:00
|
|
|
int properties,
|
|
|
|
std::vector<base::FilePath>* paths) {
|
2014-03-13 06:01:34 +00:00
|
|
|
GtkFileChooserAction action = GTK_FILE_CHOOSER_ACTION_OPEN;
|
|
|
|
if (properties & FILE_DIALOG_OPEN_DIRECTORY)
|
|
|
|
action = GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER;
|
2014-08-06 06:49:02 +00:00
|
|
|
FileChooserDialog open_dialog(action, parent_window, title, default_path,
|
|
|
|
filters);
|
2014-03-13 06:01:34 +00:00
|
|
|
if (properties & FILE_DIALOG_MULTI_SELECTIONS)
|
|
|
|
gtk_file_chooser_set_select_multiple(GTK_FILE_CHOOSER(open_dialog.dialog()),
|
|
|
|
TRUE);
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
2014-02-14 14:07:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ShowOpenDialog(atom::NativeWindow* parent_window,
|
|
|
|
const std::string& title,
|
|
|
|
const base::FilePath& default_path,
|
2014-08-06 04:44:02 +00:00
|
|
|
const Filters& filters,
|
2014-02-14 14:07:23 +00:00
|
|
|
int properties,
|
|
|
|
const OpenDialogCallback& callback) {
|
2014-03-13 05:58:53 +00:00
|
|
|
GtkFileChooserAction action = GTK_FILE_CHOOSER_ACTION_OPEN;
|
|
|
|
if (properties & FILE_DIALOG_OPEN_DIRECTORY)
|
|
|
|
action = GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER;
|
|
|
|
FileChooserDialog* open_dialog = new FileChooserDialog(
|
2014-08-06 06:49:02 +00:00
|
|
|
action, parent_window, title, default_path, filters);
|
2014-03-13 05:58:53 +00:00
|
|
|
if (properties & FILE_DIALOG_MULTI_SELECTIONS)
|
|
|
|
gtk_file_chooser_set_select_multiple(
|
|
|
|
GTK_FILE_CHOOSER(open_dialog->dialog()), TRUE);
|
|
|
|
|
|
|
|
open_dialog->RunOpenAsynchronous(callback);
|
2014-02-14 14:07:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ShowSaveDialog(atom::NativeWindow* parent_window,
|
|
|
|
const std::string& title,
|
|
|
|
const base::FilePath& default_path,
|
2014-08-06 04:44:02 +00:00
|
|
|
const Filters& filters,
|
2014-02-14 14:07:23 +00:00
|
|
|
base::FilePath* path) {
|
2014-08-06 06:49:02 +00:00
|
|
|
FileChooserDialog save_dialog(GTK_FILE_CHOOSER_ACTION_SAVE, parent_window,
|
|
|
|
title, default_path, filters);
|
2014-03-13 05:17:03 +00:00
|
|
|
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;
|
|
|
|
}
|
2014-02-14 14:07:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ShowSaveDialog(atom::NativeWindow* parent_window,
|
|
|
|
const std::string& title,
|
|
|
|
const base::FilePath& default_path,
|
2014-08-06 04:44:02 +00:00
|
|
|
const Filters& filters,
|
2014-02-14 14:07:23 +00:00
|
|
|
const SaveDialogCallback& callback) {
|
2014-03-13 05:58:53 +00:00
|
|
|
FileChooserDialog* save_dialog = new FileChooserDialog(
|
2014-08-06 06:49:02 +00:00
|
|
|
GTK_FILE_CHOOSER_ACTION_SAVE, parent_window, title, default_path,
|
|
|
|
filters);
|
2014-03-13 05:58:53 +00:00
|
|
|
save_dialog->RunSaveAsynchronous(callback);
|
2014-02-14 14:07:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace file_dialog
|