Merge pull request #7182 from electron/shell-return-values

Return booleans from more shell APIs
This commit is contained in:
Cheng Zhao 2016-09-13 16:46:05 +09:00 committed by GitHub
commit b13bab6fae
5 changed files with 57 additions and 53 deletions

View file

@ -21,11 +21,11 @@ namespace platform_util {
// Show the given file in a file manager. If possible, select the file. // Show the given file in a file manager. If possible, select the file.
// Must be called from the UI thread. // Must be called from the UI thread.
void ShowItemInFolder(const base::FilePath& full_path); bool ShowItemInFolder(const base::FilePath& full_path);
// Open the given file in the desktop's default manner. // Open the given file in the desktop's default manner.
// Must be called from the UI thread. // Must be called from the UI thread.
void OpenItem(const base::FilePath& full_path); bool OpenItem(const base::FilePath& full_path);
// Open the given external protocol URL in the desktop's default manner. // Open the given external protocol URL in the desktop's default manner.
// (For example, mailto: URLs in the default mail user agent.) // (For example, mailto: URLs in the default mail user agent.)

View file

@ -59,16 +59,16 @@ namespace platform_util {
// TODO(estade): It would be nice to be able to select the file in the file // TODO(estade): It would be nice to be able to select the file in the file
// manager, but that probably requires extending xdg-open. For now just // manager, but that probably requires extending xdg-open. For now just
// show the folder. // show the folder.
void ShowItemInFolder(const base::FilePath& full_path) { bool ShowItemInFolder(const base::FilePath& full_path) {
base::FilePath dir = full_path.DirName(); base::FilePath dir = full_path.DirName();
if (!base::DirectoryExists(dir)) if (!base::DirectoryExists(dir))
return; return false;
XDGOpen(dir.value(), true); return XDGOpen(dir.value(), true);
} }
void OpenItem(const base::FilePath& full_path) { bool OpenItem(const base::FilePath& full_path) {
XDGOpen(full_path.value(), true); return XDGOpen(full_path.value(), true);
} }
bool OpenExternal(const GURL& url, bool activate) { bool OpenExternal(const GURL& url, bool activate) {

View file

@ -18,7 +18,7 @@
namespace platform_util { namespace platform_util {
void ShowItemInFolder(const base::FilePath& path) { bool ShowItemInFolder(const base::FilePath& path) {
// The API only takes absolute path. // The API only takes absolute path.
base::FilePath full_path = base::FilePath full_path =
path.IsAbsolute() ? path : base::MakeAbsoluteFilePath(path); path.IsAbsolute() ? path : base::MakeAbsoluteFilePath(path);
@ -26,8 +26,11 @@ void ShowItemInFolder(const base::FilePath& path) {
DCHECK([NSThread isMainThread]); DCHECK([NSThread isMainThread]);
NSString* path_string = base::SysUTF8ToNSString(full_path.value()); NSString* path_string = base::SysUTF8ToNSString(full_path.value());
if (!path_string || ![[NSWorkspace sharedWorkspace] selectFile:path_string if (!path_string || ![[NSWorkspace sharedWorkspace] selectFile:path_string
inFileViewerRootedAtPath:@""]) inFileViewerRootedAtPath:@""]) {
LOG(WARNING) << "NSWorkspace failed to select file " << full_path.value(); LOG(WARNING) << "NSWorkspace failed to select file " << full_path.value();
return false;
}
return true;
} }
// This function opens a file. This doesn't use LaunchServices or NSWorkspace // This function opens a file. This doesn't use LaunchServices or NSWorkspace
@ -37,11 +40,11 @@ void ShowItemInFolder(const base::FilePath& path) {
// 2. Silent no-op for unassociated file types: http://crbug.com/50263 // 2. Silent no-op for unassociated file types: http://crbug.com/50263
// Instead, an AppleEvent is constructed to tell the Finder to open the // Instead, an AppleEvent is constructed to tell the Finder to open the
// document. // document.
void OpenItem(const base::FilePath& full_path) { bool OpenItem(const base::FilePath& full_path) {
DCHECK([NSThread isMainThread]); DCHECK([NSThread isMainThread]);
NSString* path_string = base::SysUTF8ToNSString(full_path.value()); NSString* path_string = base::SysUTF8ToNSString(full_path.value());
if (!path_string) if (!path_string)
return; return false;
// Create the target of this AppleEvent, the Finder. // Create the target of this AppleEvent, the Finder.
base::mac::ScopedAEDesc<AEAddressDesc> address; base::mac::ScopedAEDesc<AEAddressDesc> address;
@ -52,7 +55,7 @@ void OpenItem(const base::FilePath& full_path) {
address.OutPointer()); // result address.OutPointer()); // result
if (status != noErr) { if (status != noErr) {
OSSTATUS_LOG(WARNING, status) << "Could not create OpenItem() AE target"; OSSTATUS_LOG(WARNING, status) << "Could not create OpenItem() AE target";
return; return false;
} }
// Build the AppleEvent data structure that instructs Finder to open files. // Build the AppleEvent data structure that instructs Finder to open files.
@ -65,7 +68,7 @@ void OpenItem(const base::FilePath& full_path) {
theEvent.OutPointer()); // result theEvent.OutPointer()); // result
if (status != noErr) { if (status != noErr) {
OSSTATUS_LOG(WARNING, status) << "Could not create OpenItem() AE event"; OSSTATUS_LOG(WARNING, status) << "Could not create OpenItem() AE event";
return; return false;
} }
// Create the list of files (only ever one) to open. // Create the list of files (only ever one) to open.
@ -76,7 +79,7 @@ void OpenItem(const base::FilePath& full_path) {
fileList.OutPointer()); // resultList fileList.OutPointer()); // resultList
if (status != noErr) { if (status != noErr) {
OSSTATUS_LOG(WARNING, status) << "Could not create OpenItem() AE file list"; OSSTATUS_LOG(WARNING, status) << "Could not create OpenItem() AE file list";
return; return false;
} }
// Add the single path to the file list. C-style cast to avoid both a // Add the single path to the file list. C-style cast to avoid both a
@ -92,11 +95,11 @@ void OpenItem(const base::FilePath& full_path) {
if (status != noErr) { if (status != noErr) {
OSSTATUS_LOG(WARNING, status) OSSTATUS_LOG(WARNING, status)
<< "Could not add file path to AE list in OpenItem()"; << "Could not add file path to AE list in OpenItem()";
return; return false;
} }
} else { } else {
LOG(WARNING) << "Could not get FSRef for path URL in OpenItem()"; LOG(WARNING) << "Could not get FSRef for path URL in OpenItem()";
return; return false;
} }
// Attach the file list to the AppleEvent. // Attach the file list to the AppleEvent.
@ -106,7 +109,7 @@ void OpenItem(const base::FilePath& full_path) {
if (status != noErr) { if (status != noErr) {
OSSTATUS_LOG(WARNING, status) OSSTATUS_LOG(WARNING, status)
<< "Could not put the AE file list the path in OpenItem()"; << "Could not put the AE file list the path in OpenItem()";
return; return false;
} }
// Send the actual event. Do not care about the reply. // Send the actual event. Do not care about the reply.
@ -122,6 +125,7 @@ void OpenItem(const base::FilePath& full_path) {
OSSTATUS_LOG(WARNING, status) OSSTATUS_LOG(WARNING, status)
<< "Could not send AE to Finder in OpenItem()"; << "Could not send AE to Finder in OpenItem()";
} }
return status == noErr;
} }
bool OpenExternal(const GURL& url, bool activate) { bool OpenExternal(const GURL& url, bool activate) {

View file

@ -203,15 +203,15 @@ HRESULT DeleteFileProgressSink::ResumeTimer() {
namespace platform_util { namespace platform_util {
void ShowItemInFolder(const base::FilePath& full_path) { bool ShowItemInFolder(const base::FilePath& full_path) {
base::win::ScopedCOMInitializer com_initializer; base::win::ScopedCOMInitializer com_initializer;
if (!com_initializer.succeeded()) if (!com_initializer.succeeded())
return; return false;
base::FilePath dir = full_path.DirName().AsEndingWithSeparator(); base::FilePath dir = full_path.DirName().AsEndingWithSeparator();
// ParseDisplayName will fail if the directory is "C:", it must be "C:\\". // ParseDisplayName will fail if the directory is "C:", it must be "C:\\".
if (dir.empty()) if (dir.empty())
return; return false;
typedef HRESULT (WINAPI *SHOpenFolderAndSelectItemsFuncPtr)( typedef HRESULT (WINAPI *SHOpenFolderAndSelectItemsFuncPtr)(
PCIDLIST_ABSOLUTE pidl_Folder, PCIDLIST_ABSOLUTE pidl_Folder,
@ -232,29 +232,27 @@ void ShowItemInFolder(const base::FilePath& full_path) {
HMODULE shell32_base = GetModuleHandle(L"shell32.dll"); HMODULE shell32_base = GetModuleHandle(L"shell32.dll");
if (!shell32_base) { if (!shell32_base) {
NOTREACHED() << " " << __FUNCTION__ << "(): Can't open shell32.dll"; NOTREACHED() << " " << __FUNCTION__ << "(): Can't open shell32.dll";
return; return false;
} }
open_folder_and_select_itemsPtr = open_folder_and_select_itemsPtr =
reinterpret_cast<SHOpenFolderAndSelectItemsFuncPtr> reinterpret_cast<SHOpenFolderAndSelectItemsFuncPtr>
(GetProcAddress(shell32_base, "SHOpenFolderAndSelectItems")); (GetProcAddress(shell32_base, "SHOpenFolderAndSelectItems"));
} }
if (!open_folder_and_select_itemsPtr) { if (!open_folder_and_select_itemsPtr) {
ui::win::OpenFolderViaShell(dir); return ui::win::OpenFolderViaShell(dir);
return;
} }
base::win::ScopedComPtr<IShellFolder> desktop; base::win::ScopedComPtr<IShellFolder> desktop;
HRESULT hr = SHGetDesktopFolder(desktop.Receive()); HRESULT hr = SHGetDesktopFolder(desktop.Receive());
if (FAILED(hr)) if (FAILED(hr))
return; return false;
base::win::ScopedCoMem<ITEMIDLIST> dir_item; base::win::ScopedCoMem<ITEMIDLIST> dir_item;
hr = desktop->ParseDisplayName(NULL, NULL, hr = desktop->ParseDisplayName(NULL, NULL,
const_cast<wchar_t *>(dir.value().c_str()), const_cast<wchar_t *>(dir.value().c_str()),
NULL, &dir_item, NULL); NULL, &dir_item, NULL);
if (FAILED(hr)) { if (FAILED(hr)) {
ui::win::OpenFolderViaShell(dir); return ui::win::OpenFolderViaShell(dir);
return;
} }
base::win::ScopedCoMem<ITEMIDLIST> file_item; base::win::ScopedCoMem<ITEMIDLIST> file_item;
@ -262,44 +260,43 @@ void ShowItemInFolder(const base::FilePath& full_path) {
const_cast<wchar_t *>(full_path.value().c_str()), const_cast<wchar_t *>(full_path.value().c_str()),
NULL, &file_item, NULL); NULL, &file_item, NULL);
if (FAILED(hr)) { if (FAILED(hr)) {
ui::win::OpenFolderViaShell(dir); return ui::win::OpenFolderViaShell(dir);
return;
} }
const ITEMIDLIST* highlight[] = { file_item }; const ITEMIDLIST* highlight[] = { file_item };
hr = (*open_folder_and_select_itemsPtr)(dir_item, arraysize(highlight), hr = (*open_folder_and_select_itemsPtr)(dir_item, arraysize(highlight),
highlight, NULL); highlight, NULL);
if (!FAILED(hr))
return true;
if (FAILED(hr)) { // On some systems, the above call mysteriously fails with "file not
// On some systems, the above call mysteriously fails with "file not // found" even though the file is there. In these cases, ShellExecute()
// found" even though the file is there. In these cases, ShellExecute() // seems to work as a fallback (although it won't select the file).
// seems to work as a fallback (although it won't select the file). if (hr == ERROR_FILE_NOT_FOUND) {
if (hr == ERROR_FILE_NOT_FOUND) { return ui::win::OpenFolderViaShell(dir);
ui::win::OpenFolderViaShell(dir); } else {
} else { LPTSTR message = NULL;
LPTSTR message = NULL; DWORD message_length = FormatMessage(
DWORD message_length = FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM,
FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM, 0, hr, 0, reinterpret_cast<LPTSTR>(&message), 0, NULL);
0, hr, 0, reinterpret_cast<LPTSTR>(&message), 0, NULL); LOG(WARNING) << " " << __FUNCTION__
LOG(WARNING) << " " << __FUNCTION__ << "(): Can't open full_path = \""
<< "(): Can't open full_path = \"" << full_path.value() << "\""
<< full_path.value() << "\"" << " hr = " << hr
<< " hr = " << hr << " " << reinterpret_cast<LPTSTR>(&message);
<< " " << reinterpret_cast<LPTSTR>(&message); if (message)
if (message) LocalFree(message);
LocalFree(message);
ui::win::OpenFolderViaShell(dir); return ui::win::OpenFolderViaShell(dir);
}
} }
} }
void OpenItem(const base::FilePath& full_path) { bool OpenItem(const base::FilePath& full_path) {
if (base::DirectoryExists(full_path)) if (base::DirectoryExists(full_path))
ui::win::OpenFolderViaShell(full_path); return ui::win::OpenFolderViaShell(full_path);
else else
ui::win::OpenFileViaShell(full_path); return ui::win::OpenFileViaShell(full_path);
} }
bool OpenExternal(const base::string16& url, bool activate) { bool OpenExternal(const base::string16& url, bool activate) {

View file

@ -20,13 +20,15 @@ The `shell` module has the following methods:
* `fullPath` String * `fullPath` String
Show the given file in a file manager. If possible, select the file. Show the given file in a file manager. If possible, select the file. Returns
true if the item was successfully shown, false otherwise.
### `shell.openItem(fullPath)` ### `shell.openItem(fullPath)`
* `fullPath` String * `fullPath` String
Open the given file in the desktop's default manner. Open the given file in the desktop's default manner. Returns true if the item
was successfully opened, false otherwise.
### `shell.openExternal(url[, options])` ### `shell.openExternal(url[, options])`
@ -44,6 +46,7 @@ application was available to open the URL, false otherwise.
* `fullPath` String * `fullPath` String
Move the given file to trash and returns a boolean status for the operation. Move the given file to trash and returns a boolean status for the operation.
Returns true if the item was successfully moved to the trash, false otherwise.
### `shell.beep()` ### `shell.beep()`