Merge pull request #7182 from electron/shell-return-values
Return booleans from more shell APIs
This commit is contained in:
commit
b13bab6fae
5 changed files with 57 additions and 53 deletions
|
@ -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.)
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
|
@ -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()`
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue