feat: app.moveToApplicationsFolder conflict handling (#18916)
Resolves #18805. We want to keep default move conflict handling behavior in that it's still what most users would expect, but there exist edge cases in which users may not want to be forced into that behavior. This thus introduces an optional conflict handler that allows developers access to more granular move actions. They could now allow the user to choose whether to delete an existing app in favor of the current one being moved, or whether to quit the current app and focus on the existing one should it both exist and be running. I added a fair amount of new documentation outlining this behavior, but if there are things users may benefit from seeing examples of or nuances that should be added please leave feedback!
This commit is contained in:
parent
0db6789210
commit
f6a29707b6
4 changed files with 86 additions and 9 deletions
|
@ -1223,7 +1223,11 @@ This method can only be called before app is ready.
|
||||||
Returns `Boolean` - Whether the application is currently running from the
|
Returns `Boolean` - Whether the application is currently running from the
|
||||||
systems Application folder. Use in combination with `app.moveToApplicationsFolder()`
|
systems Application folder. Use in combination with `app.moveToApplicationsFolder()`
|
||||||
|
|
||||||
### `app.moveToApplicationsFolder()` _macOS_
|
### `app.moveToApplicationsFolder([options])` _macOS_
|
||||||
|
|
||||||
|
* `options` Object (optional)
|
||||||
|
* `conflictHandler` Function<Boolean> (optional) - A handler for potential conflict in move failure.
|
||||||
|
* `conflictType` String - the type of move conflict encountered by the handler; can be `exists` or `existsAndRunning`, where `exists` means that an app of the same name is present in the Applications directory and `existsAndRunning` means both that it exists and that it's presently running.
|
||||||
|
|
||||||
Returns `Boolean` - Whether the move was successful. Please note that if
|
Returns `Boolean` - Whether the move was successful. Please note that if
|
||||||
the move is successful, your application will quit and relaunch.
|
the move is successful, your application will quit and relaunch.
|
||||||
|
@ -1236,7 +1240,28 @@ the user to confirm the operation, you may do so using the
|
||||||
move to fail. For instance if the user cancels the authorization dialog, this
|
move to fail. For instance if the user cancels the authorization dialog, this
|
||||||
method returns false. If we fail to perform the copy, then this method will
|
method returns false. If we fail to perform the copy, then this method will
|
||||||
throw an error. The message in the error should be informative and tell
|
throw an error. The message in the error should be informative and tell
|
||||||
you exactly what went wrong
|
you exactly what went wrong.
|
||||||
|
|
||||||
|
By default, if an app of the same name as the one being moved exists in the Applications directory and is _not_ running, the existing app will be trashed and the active app moved into its place. If it _is_ running, the pre-existing running app will assume focus and the the previously active app will quit itself. This behavior can be changed by providing the optional conflict handler, where the boolean returned by the handler determines whether or not the move conflict is resolved with default behavior. i.e. returning `false` will ensure no further action is taken, returning `true` will result in the default behavior and the method continuing.
|
||||||
|
|
||||||
|
For example:
|
||||||
|
|
||||||
|
```js
|
||||||
|
app.moveToApplicationsFolder({
|
||||||
|
conflictHandler: (conflictType) => {
|
||||||
|
if (conflictType === 'exists') {
|
||||||
|
return dialog.showMessageBoxSync({
|
||||||
|
type: 'question',
|
||||||
|
buttons: ['Halt Move', 'Continue Move'],
|
||||||
|
defaultId: 0,
|
||||||
|
message: 'An app of this name already exists'
|
||||||
|
}) === 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
|
Would mean that if an app already exists in the user directory, if the user chooses to 'Continue Move' then the function would continue with its default behavior and the existing app will be trashed and the active app moved into its place.
|
||||||
|
|
||||||
## Properties
|
## Properties
|
||||||
|
|
||||||
|
|
|
@ -1430,7 +1430,6 @@ void App::BuildPrototype(v8::Isolate* isolate,
|
||||||
base::BindRepeating(&Browser::ResignCurrentActivity, browser))
|
base::BindRepeating(&Browser::ResignCurrentActivity, browser))
|
||||||
.SetMethod("updateCurrentActivity",
|
.SetMethod("updateCurrentActivity",
|
||||||
base::BindRepeating(&Browser::UpdateCurrentActivity, browser))
|
base::BindRepeating(&Browser::UpdateCurrentActivity, browser))
|
||||||
// TODO(juturu): Remove in 2.0, deprecate before then with warnings
|
|
||||||
.SetMethod("moveToApplicationsFolder", &App::MoveToApplicationsFolder)
|
.SetMethod("moveToApplicationsFolder", &App::MoveToApplicationsFolder)
|
||||||
.SetMethod("isInApplicationsFolder", &App::IsInApplicationsFolder)
|
.SetMethod("isInApplicationsFolder", &App::IsInApplicationsFolder)
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -11,6 +11,9 @@
|
||||||
|
|
||||||
namespace electron {
|
namespace electron {
|
||||||
|
|
||||||
|
// Possible bundle movement conflicts
|
||||||
|
enum class BundlerMoverConflictType { EXISTS, EXISTS_AND_RUNNING };
|
||||||
|
|
||||||
namespace ui {
|
namespace ui {
|
||||||
|
|
||||||
namespace cocoa {
|
namespace cocoa {
|
||||||
|
@ -21,6 +24,8 @@ class AtomBundleMover {
|
||||||
static bool IsCurrentAppInApplicationsFolder();
|
static bool IsCurrentAppInApplicationsFolder();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
static bool ShouldContinueMove(BundlerMoverConflictType type,
|
||||||
|
mate::Arguments* args);
|
||||||
static bool IsInApplicationsFolder(NSString* bundlePath);
|
static bool IsInApplicationsFolder(NSString* bundlePath);
|
||||||
static NSString* ContainingDiskImageDevice(NSString* bundlePath);
|
static NSString* ContainingDiskImageDevice(NSString* bundlePath);
|
||||||
static void Relaunch(NSString* destinationPath);
|
static void Relaunch(NSString* destinationPath);
|
||||||
|
|
|
@ -14,6 +14,26 @@
|
||||||
#import <sys/param.h>
|
#import <sys/param.h>
|
||||||
|
|
||||||
#import "shell/browser/browser.h"
|
#import "shell/browser/browser.h"
|
||||||
|
#include "shell/common/native_mate_converters/once_callback.h"
|
||||||
|
|
||||||
|
namespace mate {
|
||||||
|
|
||||||
|
template <>
|
||||||
|
struct Converter<electron::BundlerMoverConflictType> {
|
||||||
|
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
|
||||||
|
electron::BundlerMoverConflictType value) {
|
||||||
|
switch (value) {
|
||||||
|
case electron::BundlerMoverConflictType::EXISTS:
|
||||||
|
return mate::StringToV8(isolate, "exists");
|
||||||
|
case electron::BundlerMoverConflictType::EXISTS_AND_RUNNING:
|
||||||
|
return mate::StringToV8(isolate, "existsAndRunning");
|
||||||
|
default:
|
||||||
|
return mate::StringToV8(isolate, "");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace mate
|
||||||
|
|
||||||
namespace electron {
|
namespace electron {
|
||||||
|
|
||||||
|
@ -21,6 +41,28 @@ namespace ui {
|
||||||
|
|
||||||
namespace cocoa {
|
namespace cocoa {
|
||||||
|
|
||||||
|
bool AtomBundleMover::ShouldContinueMove(BundlerMoverConflictType type,
|
||||||
|
mate::Arguments* args) {
|
||||||
|
mate::Dictionary options;
|
||||||
|
bool hasOptions = args->GetNext(&options);
|
||||||
|
base::OnceCallback<v8::Local<v8::Value>(BundlerMoverConflictType)>
|
||||||
|
conflict_cb;
|
||||||
|
|
||||||
|
if (hasOptions && options.Get("conflictHandler", &conflict_cb)) {
|
||||||
|
v8::Local<v8::Value> value = std::move(conflict_cb).Run(type);
|
||||||
|
if (value->IsBoolean()) {
|
||||||
|
if (!value.As<v8::Boolean>()->Value())
|
||||||
|
return false;
|
||||||
|
} else if (!value->IsUndefined()) {
|
||||||
|
// we only want to throw an error if a user has returned a non-boolean
|
||||||
|
// value; this allows for client-side error handling should something in
|
||||||
|
// the handler throw
|
||||||
|
args->ThrowError("Invalid conflict handler return type.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
bool AtomBundleMover::Move(mate::Arguments* args) {
|
bool AtomBundleMover::Move(mate::Arguments* args) {
|
||||||
// Path of the current bundle
|
// Path of the current bundle
|
||||||
NSString* bundlePath = [[NSBundle mainBundle] bundlePath];
|
NSString* bundlePath = [[NSBundle mainBundle] bundlePath];
|
||||||
|
@ -74,7 +116,12 @@ bool AtomBundleMover::Move(mate::Arguments* args) {
|
||||||
if ([fileManager fileExistsAtPath:destinationPath]) {
|
if ([fileManager fileExistsAtPath:destinationPath]) {
|
||||||
// But first, make sure that it's not running
|
// But first, make sure that it's not running
|
||||||
if (IsApplicationAtPathRunning(destinationPath)) {
|
if (IsApplicationAtPathRunning(destinationPath)) {
|
||||||
// Give the running app focus and terminate myself
|
// Check for callback handler and get user choice for open/quit
|
||||||
|
if (!ShouldContinueMove(BundlerMoverConflictType::EXISTS_AND_RUNNING,
|
||||||
|
args))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Unless explicitly denied, give running app focus and terminate self
|
||||||
[[NSTask
|
[[NSTask
|
||||||
launchedTaskWithLaunchPath:@"/usr/bin/open"
|
launchedTaskWithLaunchPath:@"/usr/bin/open"
|
||||||
arguments:[NSArray
|
arguments:[NSArray
|
||||||
|
@ -83,6 +130,11 @@ bool AtomBundleMover::Move(mate::Arguments* args) {
|
||||||
electron::Browser::Get()->Quit();
|
electron::Browser::Get()->Quit();
|
||||||
return true;
|
return true;
|
||||||
} else {
|
} else {
|
||||||
|
// Check callback handler and get user choice for app trashing
|
||||||
|
if (!ShouldContinueMove(BundlerMoverConflictType::EXISTS, args))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Unless explicitly denied, attempt to trash old app
|
||||||
if (!Trash([applicationsDirectory
|
if (!Trash([applicationsDirectory
|
||||||
stringByAppendingPathComponent:bundleName])) {
|
stringByAppendingPathComponent:bundleName])) {
|
||||||
args->ThrowError("Failed to delete existing application");
|
args->ThrowError("Failed to delete existing application");
|
||||||
|
@ -313,11 +365,7 @@ bool AtomBundleMover::CopyBundle(NSString* srcPath, NSString* dstPath) {
|
||||||
NSFileManager* fileManager = [NSFileManager defaultManager];
|
NSFileManager* fileManager = [NSFileManager defaultManager];
|
||||||
NSError* error = nil;
|
NSError* error = nil;
|
||||||
|
|
||||||
if ([fileManager copyItemAtPath:srcPath toPath:dstPath error:&error]) {
|
return [fileManager copyItemAtPath:srcPath toPath:dstPath error:&error];
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
NSString* AtomBundleMover::ShellQuotedString(NSString* string) {
|
NSString* AtomBundleMover::ShellQuotedString(NSString* string) {
|
||||||
|
|
Loading…
Reference in a new issue