2014-10-31 18:17:05 +00:00
|
|
|
// Copyright (c) 2013 GitHub, Inc.
|
2014-04-25 09:49:37 +00:00
|
|
|
// Use of this source code is governed by the MIT license that can be
|
2013-05-20 13:46:43 +00:00
|
|
|
// found in the LICENSE file.
|
|
|
|
|
2014-03-16 00:30:26 +00:00
|
|
|
#include "atom/browser/ui/file_dialog.h"
|
2013-05-20 13:46:43 +00:00
|
|
|
|
|
|
|
#import <Cocoa/Cocoa.h>
|
2014-03-16 01:37:04 +00:00
|
|
|
#import <CoreServices/CoreServices.h>
|
2013-05-20 13:46:43 +00:00
|
|
|
|
2014-03-16 01:37:04 +00:00
|
|
|
#include "atom/browser/native_window.h"
|
2015-01-10 01:45:50 +00:00
|
|
|
#include "base/files/file_util.h"
|
2015-03-11 00:00:55 +00:00
|
|
|
#include "base/mac/foundation_util.h"
|
2014-08-06 05:47:44 +00:00
|
|
|
#include "base/mac/mac_util.h"
|
|
|
|
#include "base/mac/scoped_cftyperef.h"
|
2013-05-20 13:46:43 +00:00
|
|
|
#include "base/strings/sys_string_conversions.h"
|
|
|
|
|
|
|
|
namespace file_dialog {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2014-08-06 05:47:44 +00:00
|
|
|
void SetAllowedFileTypes(NSSavePanel* dialog, const Filters& filters) {
|
|
|
|
NSMutableSet* file_type_set = [NSMutableSet set];
|
|
|
|
for (size_t i = 0; i < filters.size(); ++i) {
|
|
|
|
const Filter& filter = filters[i];
|
|
|
|
for (size_t j = 0; j < filter.second.size(); ++j) {
|
2015-08-21 04:15:20 +00:00
|
|
|
// If we meet a '*' file extension, we allow all the file types and no
|
|
|
|
// need to set the specified file types.
|
|
|
|
if (filter.second[j] == "*") {
|
|
|
|
[dialog setAllowsOtherFileTypes:YES];
|
|
|
|
return;
|
|
|
|
}
|
2014-08-06 05:47:44 +00:00
|
|
|
base::ScopedCFTypeRef<CFStringRef> ext_cf(
|
|
|
|
base::SysUTF8ToCFStringRef(filter.second[j]));
|
|
|
|
[file_type_set addObject:base::mac::CFToNSCast(ext_cf.get())];
|
|
|
|
}
|
|
|
|
}
|
2015-12-30 20:38:02 +00:00
|
|
|
|
2015-12-31 10:58:16 +00:00
|
|
|
// Passing empty array to setAllowedFileTypes will cause exception.
|
2015-12-30 20:38:02 +00:00
|
|
|
NSArray* file_types = nil;
|
|
|
|
if ([file_type_set count])
|
|
|
|
file_types = [file_type_set allObjects];
|
|
|
|
|
2015-12-31 10:58:16 +00:00
|
|
|
[dialog setAllowedFileTypes:file_types];
|
2014-08-06 05:47:44 +00:00
|
|
|
}
|
|
|
|
|
2013-05-20 13:46:43 +00:00
|
|
|
void SetupDialog(NSSavePanel* dialog,
|
|
|
|
const std::string& title,
|
2016-05-06 18:10:31 +00:00
|
|
|
const std::string& button_label,
|
2014-08-06 05:47:44 +00:00
|
|
|
const base::FilePath& default_path,
|
2017-02-01 15:34:21 +00:00
|
|
|
const Filters& filters,
|
|
|
|
const std::string& message,
|
|
|
|
const std::string& name_field_label,
|
|
|
|
const bool& shows_tag_field) {
|
2013-05-20 13:46:43 +00:00
|
|
|
if (!title.empty())
|
|
|
|
[dialog setTitle:base::SysUTF8ToNSString(title)];
|
|
|
|
|
2016-05-06 18:10:31 +00:00
|
|
|
if (!button_label.empty())
|
|
|
|
[dialog setPrompt:base::SysUTF8ToNSString(button_label)];
|
|
|
|
|
2017-02-01 15:34:21 +00:00
|
|
|
if (!message.empty())
|
|
|
|
[dialog setMessage:base::SysUTF8ToNSString(message)];
|
|
|
|
|
|
|
|
if (!name_field_label.empty())
|
|
|
|
[dialog setNameFieldLabel:base::SysUTF8ToNSString(name_field_label)];
|
|
|
|
|
|
|
|
[dialog setShowsTagField:shows_tag_field];
|
|
|
|
|
2013-05-20 13:46:43 +00:00
|
|
|
NSString* default_dir = nil;
|
|
|
|
NSString* default_filename = nil;
|
|
|
|
if (!default_path.empty()) {
|
2013-12-11 07:48:19 +00:00
|
|
|
if (base::DirectoryExists(default_path)) {
|
2013-05-20 13:46:43 +00:00
|
|
|
default_dir = base::SysUTF8ToNSString(default_path.value());
|
|
|
|
} else {
|
|
|
|
default_dir = base::SysUTF8ToNSString(default_path.DirName().value());
|
|
|
|
default_filename =
|
|
|
|
base::SysUTF8ToNSString(default_path.BaseName().value());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (default_dir)
|
|
|
|
[dialog setDirectoryURL:[NSURL fileURLWithPath:default_dir]];
|
|
|
|
if (default_filename)
|
|
|
|
[dialog setNameFieldStringValue:default_filename];
|
|
|
|
|
2014-08-06 05:47:44 +00:00
|
|
|
if (filters.empty())
|
|
|
|
[dialog setAllowsOtherFileTypes:YES];
|
|
|
|
else
|
|
|
|
SetAllowedFileTypes(dialog, filters);
|
2013-05-20 13:46:43 +00:00
|
|
|
}
|
|
|
|
|
2013-09-23 11:22:36 +00:00
|
|
|
void SetupDialogForProperties(NSOpenPanel* dialog, int properties) {
|
|
|
|
[dialog setCanChooseFiles:(properties & FILE_DIALOG_OPEN_FILE)];
|
|
|
|
if (properties & FILE_DIALOG_OPEN_DIRECTORY)
|
|
|
|
[dialog setCanChooseDirectories:YES];
|
|
|
|
if (properties & FILE_DIALOG_CREATE_DIRECTORY)
|
|
|
|
[dialog setCanCreateDirectories:YES];
|
|
|
|
if (properties & FILE_DIALOG_MULTI_SELECTIONS)
|
|
|
|
[dialog setAllowsMultipleSelection:YES];
|
2016-07-11 04:30:18 +00:00
|
|
|
if (properties & FILE_DIALOG_SHOW_HIDDEN_FILES)
|
|
|
|
[dialog setShowsHiddenFiles:YES];
|
2013-09-23 11:22:36 +00:00
|
|
|
}
|
|
|
|
|
2013-09-23 11:42:07 +00:00
|
|
|
// Run modal dialog with parent window and return user's choice.
|
|
|
|
int RunModalDialog(NSSavePanel* dialog, atom::NativeWindow* parent_window) {
|
|
|
|
__block int chosen = NSFileHandlingPanelCancelButton;
|
2014-07-31 02:31:08 +00:00
|
|
|
if (!parent_window || !parent_window->GetNativeWindow()) {
|
2013-09-23 11:42:07 +00:00
|
|
|
chosen = [dialog runModal];
|
|
|
|
} else {
|
|
|
|
NSWindow* window = parent_window->GetNativeWindow();
|
|
|
|
|
|
|
|
[dialog beginSheetModalForWindow:window
|
|
|
|
completionHandler:^(NSInteger c) {
|
|
|
|
chosen = c;
|
|
|
|
[NSApp stopModal];
|
|
|
|
}];
|
|
|
|
[NSApp runModalForWindow:window];
|
|
|
|
}
|
|
|
|
|
|
|
|
return chosen;
|
|
|
|
}
|
|
|
|
|
2013-09-23 11:22:36 +00:00
|
|
|
void ReadDialogPaths(NSOpenPanel* dialog, std::vector<base::FilePath>* paths) {
|
|
|
|
NSArray* urls = [dialog URLs];
|
|
|
|
for (NSURL* url in urls)
|
|
|
|
if ([url isFileURL])
|
|
|
|
paths->push_back(base::FilePath(base::SysNSStringToUTF8([url path])));
|
|
|
|
}
|
|
|
|
|
2013-05-20 13:46:43 +00:00
|
|
|
} // namespace
|
|
|
|
|
2013-09-23 08:27:22 +00:00
|
|
|
bool ShowOpenDialog(atom::NativeWindow* parent_window,
|
|
|
|
const std::string& title,
|
2016-05-06 18:10:31 +00:00
|
|
|
const std::string& button_label,
|
2013-05-20 13:46:43 +00:00
|
|
|
const base::FilePath& default_path,
|
2014-08-06 04:44:02 +00:00
|
|
|
const Filters& filters,
|
2013-05-20 13:46:43 +00:00
|
|
|
int properties,
|
2017-02-01 16:01:01 +00:00
|
|
|
const std::string& message,
|
2013-05-20 13:46:43 +00:00
|
|
|
std::vector<base::FilePath>* paths) {
|
|
|
|
DCHECK(paths);
|
|
|
|
NSOpenPanel* dialog = [NSOpenPanel openPanel];
|
|
|
|
|
2017-02-09 13:47:26 +00:00
|
|
|
// NSOpenPanel does not support name_field_label and shows_tag_field
|
|
|
|
SetupDialog(dialog, title, button_label, default_path, filters, message, "", false);
|
2013-09-23 11:22:36 +00:00
|
|
|
SetupDialogForProperties(dialog, properties);
|
2013-05-20 13:46:43 +00:00
|
|
|
|
2013-09-23 11:42:07 +00:00
|
|
|
int chosen = RunModalDialog(dialog, parent_window);
|
2013-09-23 08:27:22 +00:00
|
|
|
if (chosen == NSFileHandlingPanelCancelButton)
|
2013-05-20 13:46:43 +00:00
|
|
|
return false;
|
|
|
|
|
2013-09-23 11:22:36 +00:00
|
|
|
ReadDialogPaths(dialog, paths);
|
2013-05-20 13:46:43 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-09-23 11:22:36 +00:00
|
|
|
void ShowOpenDialog(atom::NativeWindow* parent_window,
|
|
|
|
const std::string& title,
|
2016-05-06 18:10:31 +00:00
|
|
|
const std::string& button_label,
|
2013-09-23 11:22:36 +00:00
|
|
|
const base::FilePath& default_path,
|
2014-08-06 04:44:02 +00:00
|
|
|
const Filters& filters,
|
2013-09-23 11:22:36 +00:00
|
|
|
int properties,
|
2017-02-01 16:01:01 +00:00
|
|
|
const std::string& message,
|
2013-09-23 11:22:36 +00:00
|
|
|
const OpenDialogCallback& c) {
|
|
|
|
NSOpenPanel* dialog = [NSOpenPanel openPanel];
|
|
|
|
|
2017-02-09 13:47:26 +00:00
|
|
|
// NSOpenPanel does not support name_field_label and shows_tag_field
|
|
|
|
SetupDialog(dialog, title, button_label, default_path, filters, message, "", false);
|
2013-09-23 11:22:36 +00:00
|
|
|
SetupDialogForProperties(dialog, properties);
|
|
|
|
|
|
|
|
// Duplicate the callback object here since c is a reference and gcd would
|
|
|
|
// only store the pointer, by duplication we can force gcd to store a copy.
|
|
|
|
__block OpenDialogCallback callback = c;
|
|
|
|
|
|
|
|
NSWindow* window = parent_window ? parent_window->GetNativeWindow() : NULL;
|
|
|
|
[dialog beginSheetModalForWindow:window
|
|
|
|
completionHandler:^(NSInteger chosen) {
|
|
|
|
if (chosen == NSFileHandlingPanelCancelButton) {
|
|
|
|
callback.Run(false, std::vector<base::FilePath>());
|
|
|
|
} else {
|
|
|
|
std::vector<base::FilePath> paths;
|
|
|
|
ReadDialogPaths(dialog, &paths);
|
|
|
|
callback.Run(true, paths);
|
|
|
|
}
|
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
2013-09-23 11:36:52 +00:00
|
|
|
bool ShowSaveDialog(atom::NativeWindow* parent_window,
|
2013-05-20 13:46:43 +00:00
|
|
|
const std::string& title,
|
2016-05-06 18:10:31 +00:00
|
|
|
const std::string& button_label,
|
2013-05-20 13:46:43 +00:00
|
|
|
const base::FilePath& default_path,
|
2014-08-06 04:44:02 +00:00
|
|
|
const Filters& filters,
|
2017-02-01 15:34:21 +00:00
|
|
|
const std::string& message,
|
|
|
|
const std::string& name_field_label,
|
2017-02-09 13:33:33 +00:00
|
|
|
bool shows_tag_field,
|
2013-05-20 13:46:43 +00:00
|
|
|
base::FilePath* path) {
|
|
|
|
DCHECK(path);
|
|
|
|
NSSavePanel* dialog = [NSSavePanel savePanel];
|
|
|
|
|
2017-02-01 15:34:21 +00:00
|
|
|
SetupDialog(dialog, title, button_label, default_path, filters, message,
|
|
|
|
name_field_label, shows_tag_field);
|
2013-05-20 13:46:43 +00:00
|
|
|
|
2013-09-23 11:42:07 +00:00
|
|
|
int chosen = RunModalDialog(dialog, parent_window);
|
2013-09-23 11:36:52 +00:00
|
|
|
if (chosen == NSFileHandlingPanelCancelButton || ![[dialog URL] isFileURL])
|
|
|
|
return false;
|
2013-05-20 13:46:43 +00:00
|
|
|
|
2013-09-23 11:36:52 +00:00
|
|
|
*path = base::FilePath(base::SysNSStringToUTF8([[dialog URL] path]));
|
|
|
|
return true;
|
2013-05-20 13:46:43 +00:00
|
|
|
}
|
|
|
|
|
2013-09-23 12:08:32 +00:00
|
|
|
void ShowSaveDialog(atom::NativeWindow* parent_window,
|
|
|
|
const std::string& title,
|
2016-05-06 18:10:31 +00:00
|
|
|
const std::string& button_label,
|
2013-09-23 12:08:32 +00:00
|
|
|
const base::FilePath& default_path,
|
2014-08-06 04:44:02 +00:00
|
|
|
const Filters& filters,
|
2017-02-01 15:34:21 +00:00
|
|
|
const std::string& message,
|
|
|
|
const std::string& name_field_label,
|
2017-02-09 13:33:33 +00:00
|
|
|
bool shows_tag_field,
|
2013-09-23 12:08:32 +00:00
|
|
|
const SaveDialogCallback& c) {
|
|
|
|
NSSavePanel* dialog = [NSSavePanel savePanel];
|
|
|
|
|
2017-02-01 15:34:21 +00:00
|
|
|
SetupDialog(dialog, title, button_label, default_path, filters, message,
|
|
|
|
name_field_label, shows_tag_field);
|
2016-08-16 23:08:01 +00:00
|
|
|
[dialog setCanSelectHiddenExtension:YES];
|
2013-09-23 12:08:32 +00:00
|
|
|
|
|
|
|
__block SaveDialogCallback callback = c;
|
|
|
|
|
|
|
|
NSWindow* window = parent_window ? parent_window->GetNativeWindow() : NULL;
|
|
|
|
[dialog beginSheetModalForWindow:window
|
|
|
|
completionHandler:^(NSInteger chosen) {
|
|
|
|
if (chosen == NSFileHandlingPanelCancelButton) {
|
|
|
|
callback.Run(false, base::FilePath());
|
|
|
|
} else {
|
|
|
|
std::string path = base::SysNSStringToUTF8([[dialog URL] path]);
|
|
|
|
callback.Run(true, base::FilePath(path));
|
|
|
|
}
|
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
2013-05-20 13:46:43 +00:00
|
|
|
} // namespace file_dialog
|