feat: new makeSingleInstance API (#12782)
* Refactor app.makeSingleInstance * new API `app.isPrimaryInstance()` * new API `app.isSingleInstance()` * new event `app.on('second-instance')` * deprecated old syntax `app.makeSingleInstance(cb)` * deprecated old syntax of `app.makeSingleInstance() --> bool` in favor of `app.isPrimaryInstance()` * Fix spec, we don't need process.nextTick hacks any more * Make deprecation TODO for the return value of makeSingleInstance * Refactor makeSingleInstance to requestSingleInstanceLock and add appropriate deprecation comments * I swear this isn't tricking the linter * Make const * Add deprecation warnings for release, and add to planned-breaking-changes BREAKING CHANGE
This commit is contained in:
parent
9c8952aef0
commit
5b5c161601
9 changed files with 176 additions and 52 deletions
|
@ -350,8 +350,8 @@ struct Converter<content::CertificateRequestResultType> {
|
||||||
namespace atom {
|
namespace atom {
|
||||||
|
|
||||||
ProcessMetric::ProcessMetric(int type,
|
ProcessMetric::ProcessMetric(int type,
|
||||||
base::ProcessId pid,
|
base::ProcessId pid,
|
||||||
std::unique_ptr<base::ProcessMetrics> metrics) {
|
std::unique_ptr<base::ProcessMetrics> metrics) {
|
||||||
this->type = type;
|
this->type = type;
|
||||||
this->pid = pid;
|
this->pid = pid;
|
||||||
this->metrics = std::move(metrics);
|
this->metrics = std::move(metrics);
|
||||||
|
@ -422,7 +422,9 @@ int GetPathConstant(const std::string& name) {
|
||||||
}
|
}
|
||||||
|
|
||||||
bool NotificationCallbackWrapper(
|
bool NotificationCallbackWrapper(
|
||||||
const ProcessSingleton::NotificationCallback& callback,
|
const base::Callback<
|
||||||
|
void(const base::CommandLine::StringVector& command_line,
|
||||||
|
const base::FilePath& current_directory)>& callback,
|
||||||
const base::CommandLine::StringVector& cmd,
|
const base::CommandLine::StringVector& cmd,
|
||||||
const base::FilePath& cwd) {
|
const base::FilePath& cwd) {
|
||||||
// Make sure the callback is called after app gets ready.
|
// Make sure the callback is called after app gets ready.
|
||||||
|
@ -864,30 +866,43 @@ std::string App::GetLocale() {
|
||||||
return g_browser_process->GetApplicationLocale();
|
return g_browser_process->GetApplicationLocale();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool App::MakeSingleInstance(
|
void App::OnSecondInstance(const base::CommandLine::StringVector& cmd,
|
||||||
const ProcessSingleton::NotificationCallback& callback) {
|
const base::FilePath& cwd) {
|
||||||
|
Emit("second-instance", cmd, cwd);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool App::HasSingleInstanceLock() const {
|
||||||
if (process_singleton_)
|
if (process_singleton_)
|
||||||
return false;
|
return true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool App::RequestSingleInstanceLock() {
|
||||||
|
if (HasSingleInstanceLock())
|
||||||
|
return true;
|
||||||
|
|
||||||
base::FilePath user_dir;
|
base::FilePath user_dir;
|
||||||
PathService::Get(brightray::DIR_USER_DATA, &user_dir);
|
PathService::Get(brightray::DIR_USER_DATA, &user_dir);
|
||||||
|
|
||||||
|
auto cb = base::Bind(&App::OnSecondInstance, base::Unretained(this));
|
||||||
|
|
||||||
process_singleton_.reset(new ProcessSingleton(
|
process_singleton_.reset(new ProcessSingleton(
|
||||||
user_dir, base::Bind(NotificationCallbackWrapper, callback)));
|
user_dir, base::Bind(NotificationCallbackWrapper, cb)));
|
||||||
|
|
||||||
switch (process_singleton_->NotifyOtherProcessOrCreate()) {
|
switch (process_singleton_->NotifyOtherProcessOrCreate()) {
|
||||||
case ProcessSingleton::NotifyResult::LOCK_ERROR:
|
case ProcessSingleton::NotifyResult::LOCK_ERROR:
|
||||||
case ProcessSingleton::NotifyResult::PROFILE_IN_USE:
|
case ProcessSingleton::NotifyResult::PROFILE_IN_USE:
|
||||||
case ProcessSingleton::NotifyResult::PROCESS_NOTIFIED: {
|
case ProcessSingleton::NotifyResult::PROCESS_NOTIFIED: {
|
||||||
process_singleton_.reset();
|
process_singleton_.reset();
|
||||||
return true;
|
return false;
|
||||||
}
|
}
|
||||||
case ProcessSingleton::NotifyResult::PROCESS_NONE:
|
case ProcessSingleton::NotifyResult::PROCESS_NONE:
|
||||||
default: // Shouldn't be needed, but VS warns if it is not there.
|
default: // Shouldn't be needed, but VS warns if it is not there.
|
||||||
return false;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void App::ReleaseSingleInstance() {
|
void App::ReleaseSingleInstanceLock() {
|
||||||
if (process_singleton_) {
|
if (process_singleton_) {
|
||||||
process_singleton_->Cleanup();
|
process_singleton_->Cleanup();
|
||||||
process_singleton_.reset();
|
process_singleton_.reset();
|
||||||
|
@ -1252,8 +1267,9 @@ void App::BuildPrototype(v8::Isolate* isolate,
|
||||||
#if defined(USE_NSS_CERTS)
|
#if defined(USE_NSS_CERTS)
|
||||||
.SetMethod("importCertificate", &App::ImportCertificate)
|
.SetMethod("importCertificate", &App::ImportCertificate)
|
||||||
#endif
|
#endif
|
||||||
.SetMethod("makeSingleInstance", &App::MakeSingleInstance)
|
.SetMethod("hasSingleInstanceLock", &App::HasSingleInstanceLock)
|
||||||
.SetMethod("releaseSingleInstance", &App::ReleaseSingleInstance)
|
.SetMethod("requestSingleInstanceLock", &App::RequestSingleInstanceLock)
|
||||||
|
.SetMethod("releaseSingleInstanceLock", &App::ReleaseSingleInstanceLock)
|
||||||
.SetMethod("relaunch", &App::Relaunch)
|
.SetMethod("relaunch", &App::Relaunch)
|
||||||
.SetMethod("isAccessibilitySupportEnabled",
|
.SetMethod("isAccessibilitySupportEnabled",
|
||||||
&App::IsAccessibilitySupportEnabled)
|
&App::IsAccessibilitySupportEnabled)
|
||||||
|
|
|
@ -178,9 +178,11 @@ class App : public AtomBrowserClient::Delegate,
|
||||||
|
|
||||||
void SetDesktopName(const std::string& desktop_name);
|
void SetDesktopName(const std::string& desktop_name);
|
||||||
std::string GetLocale();
|
std::string GetLocale();
|
||||||
bool MakeSingleInstance(
|
void OnSecondInstance(const base::CommandLine::StringVector& cmd,
|
||||||
const ProcessSingleton::NotificationCallback& callback);
|
const base::FilePath& cwd);
|
||||||
void ReleaseSingleInstance();
|
bool HasSingleInstanceLock() const;
|
||||||
|
bool RequestSingleInstanceLock();
|
||||||
|
void ReleaseSingleInstanceLock();
|
||||||
bool Relaunch(mate::Arguments* args);
|
bool Relaunch(mate::Arguments* args);
|
||||||
void DisableHardwareAcceleration(mate::Arguments* args);
|
void DisableHardwareAcceleration(mate::Arguments* args);
|
||||||
void DisableDomainBlockingFor3DAPIs(mate::Arguments* args);
|
void DisableDomainBlockingFor3DAPIs(mate::Arguments* args);
|
||||||
|
|
|
@ -373,6 +373,22 @@ app.on('session-created', (event, session) => {
|
||||||
})
|
})
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### Event: 'second-instance'
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
|
||||||
|
* `argv` String[] - An array of the second instance's command line arguments
|
||||||
|
* `workingDirectory` String - The second instance's working directory
|
||||||
|
|
||||||
|
This event will be emitted inside the primary instance of your application
|
||||||
|
when a second instance has been executed. `argv` is an Array of the second instance's
|
||||||
|
command line arguments, and `workingDirectory` is its current working directory. Usually
|
||||||
|
applications respond to this by making their primary window focused and
|
||||||
|
non-minimized.
|
||||||
|
|
||||||
|
This event is guaranteed to be emitted after the `ready` event of `app`
|
||||||
|
gets emitted.
|
||||||
|
|
||||||
## Methods
|
## Methods
|
||||||
|
|
||||||
The `app` object has the following methods:
|
The `app` object has the following methods:
|
||||||
|
@ -742,32 +758,24 @@ app.setJumpList([
|
||||||
])
|
])
|
||||||
```
|
```
|
||||||
|
|
||||||
### `app.makeSingleInstance(callback)`
|
### `app.requestSingleInstanceLock()`
|
||||||
|
|
||||||
* `callback` Function
|
Returns `Boolean`
|
||||||
* `argv` String[] - An array of the second instance's command line arguments
|
|
||||||
* `workingDirectory` String - The second instance's working directory
|
|
||||||
|
|
||||||
Returns `Boolean`.
|
|
||||||
|
|
||||||
This method makes your application a Single Instance Application - instead of
|
This method makes your application a Single Instance Application - instead of
|
||||||
allowing multiple instances of your app to run, this will ensure that only a
|
allowing multiple instances of your app to run, this will ensure that only a
|
||||||
single instance of your app is running, and other instances signal this
|
single instance of your app is running, and other instances signal this
|
||||||
instance and exit.
|
instance and exit.
|
||||||
|
|
||||||
`callback` will be called by the first instance with `callback(argv, workingDirectory)`
|
The return value of this method indicates whether or not this instance of your
|
||||||
when a second instance has been executed. `argv` is an Array of the second instance's
|
application successfully obtained the lock. If it failed to obtain the lock
|
||||||
command line arguments, and `workingDirectory` is its current working directory. Usually
|
you can assume that another instance of your application is already running with
|
||||||
applications respond to this by making their primary window focused and
|
the lock and exit immediately.
|
||||||
non-minimized.
|
|
||||||
|
|
||||||
The `callback` is guaranteed to be executed after the `ready` event of `app`
|
I.e. This method returns `true` if your process is the primary instance of your
|
||||||
gets emitted.
|
application and your app should continue loading. It returns `false` if your
|
||||||
|
process should immediately quit as it has sent its parameters to another
|
||||||
This method returns `false` if your process is the primary instance of the
|
instance that has already acquired the lock.
|
||||||
application and your app should continue loading. And returns `true` if your
|
|
||||||
process has sent its parameters to another instance, and you should immediately
|
|
||||||
quit.
|
|
||||||
|
|
||||||
On macOS the system enforces single instance automatically when users try to open
|
On macOS the system enforces single instance automatically when users try to open
|
||||||
a second instance of your app in Finder, and the `open-file` and `open-url`
|
a second instance of your app in Finder, and the `open-file` and `open-url`
|
||||||
|
@ -782,27 +790,38 @@ starts:
|
||||||
const {app} = require('electron')
|
const {app} = require('electron')
|
||||||
let myWindow = null
|
let myWindow = null
|
||||||
|
|
||||||
const isSecondInstance = app.makeSingleInstance((commandLine, workingDirectory) => {
|
const gotTheLock = app.requestSingleInstanceLock()
|
||||||
// Someone tried to run a second instance, we should focus our window.
|
|
||||||
if (myWindow) {
|
|
||||||
if (myWindow.isMinimized()) myWindow.restore()
|
|
||||||
myWindow.focus()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
if (isSecondInstance) {
|
if (!gotTheLock) {
|
||||||
app.quit()
|
app.quit()
|
||||||
}
|
} else {
|
||||||
|
app.on('second-instance', (commandLine, workingDirectory) => {
|
||||||
|
// Someone tried to run a second instance, we should focus our window.
|
||||||
|
if (myWindow) {
|
||||||
|
if (myWindow.isMinimized()) myWindow.restore()
|
||||||
|
myWindow.focus()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
// Create myWindow, load the rest of the app, etc...
|
// Create myWindow, load the rest of the app, etc...
|
||||||
app.on('ready', () => {
|
app.on('ready', () => {
|
||||||
})
|
})
|
||||||
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
### `app.releaseSingleInstance()`
|
### `app.hasSingleInstanceLock()`
|
||||||
|
|
||||||
Releases all locks that were created by `makeSingleInstance`. This will allow
|
Returns `Boolean`
|
||||||
multiple instances of the application to once again run side by side.
|
|
||||||
|
This method returns whether or not this instance of your app is currently
|
||||||
|
holding the single instance lock. You can request the lock with
|
||||||
|
`app.requestSingleInstanceLock()` and release with
|
||||||
|
`app.releaseSingleInstanceLock()`
|
||||||
|
|
||||||
|
### `app.releaseSingleInstanceLock()`
|
||||||
|
|
||||||
|
Releases all locks that were created by `requestSingleInstanceLock`. This will
|
||||||
|
allow multiple instances of the application to once again run side by side.
|
||||||
|
|
||||||
### `app.setUserActivity(type, userInfo[, webpageURL])` _macOS_
|
### `app.setUserActivity(type, userInfo[, webpageURL])` _macOS_
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
# Planned Breaking API Changes
|
# Planned Breaking API Changes (3.0)
|
||||||
|
|
||||||
The following list includes the APIs that will be removed in Electron 3.0.
|
The following list includes the APIs that will be removed in Electron 3.0.
|
||||||
|
|
||||||
|
@ -150,3 +150,33 @@ Replace with: https://atom.io/download/electron
|
||||||
The `FIXME` string is used in code comments to denote things that should be
|
The `FIXME` string is used in code comments to denote things that should be
|
||||||
fixed for the 3.0 release. See
|
fixed for the 3.0 release. See
|
||||||
https://github.com/electron/electron/search?q=fixme
|
https://github.com/electron/electron/search?q=fixme
|
||||||
|
|
||||||
|
# Planned Breaking API Changes (4.0)
|
||||||
|
|
||||||
|
The following list includes the APIs that will be removed in Electron 4.0.
|
||||||
|
|
||||||
|
There is no timetable for when this release will occur but deprecation
|
||||||
|
warnings will be added at least [one major version](electron-versioning.md#semver) beforehand.
|
||||||
|
|
||||||
|
## `app.makeSingleInstance`
|
||||||
|
|
||||||
|
```js
|
||||||
|
// Deprecated
|
||||||
|
app.makeSingleInstance(function (argv, cwd) {
|
||||||
|
|
||||||
|
})
|
||||||
|
// Replace with
|
||||||
|
app.requestSingleInstanceLock()
|
||||||
|
app.on('second-instance', function (argv, cwd) {
|
||||||
|
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
|
## `app.releaseSingleInstance`
|
||||||
|
|
||||||
|
```js
|
||||||
|
// Deprecated
|
||||||
|
app.releaseSingleInstance()
|
||||||
|
// Replace with
|
||||||
|
app.releaseSingleInstanceLock()
|
||||||
|
```
|
|
@ -109,6 +109,21 @@ for (let name of events) {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO(MarshallOfSound): Remove in 4.0
|
||||||
|
app.releaseSingleInstance = () => {
|
||||||
|
deprecate.warn('app.releaseSingleInstance(cb)', 'app.releaseSingleInstanceLock()')
|
||||||
|
app.releaseSingleInstanceLock()
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO(MarshallOfSound): Remove in 4.0
|
||||||
|
app.makeSingleInstance = (oldStyleFn) => {
|
||||||
|
deprecate.warn('app.makeSingleInstance(cb)', 'app.requestSingleInstanceLock() and app.on(\'second-instance\', cb)')
|
||||||
|
if (oldStyleFn && typeof oldStyleFn === 'function') {
|
||||||
|
app.on('second-instance', (event, ...args) => oldStyleFn(...args))
|
||||||
|
}
|
||||||
|
return !app.requestSingleInstanceLock()
|
||||||
|
}
|
||||||
|
|
||||||
// Wrappers for native classes.
|
// Wrappers for native classes.
|
||||||
const {DownloadItem} = process.atomBinding('download_item')
|
const {DownloadItem} = process.atomBinding('download_item')
|
||||||
Object.setPrototypeOf(DownloadItem.prototype, EventEmitter.prototype)
|
Object.setPrototypeOf(DownloadItem.prototype, EventEmitter.prototype)
|
||||||
|
|
|
@ -185,7 +185,29 @@ describe('app module', () => {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// TODO(MarshallOfSound) - Remove in 4.0.0
|
||||||
describe('app.makeSingleInstance', () => {
|
describe('app.makeSingleInstance', () => {
|
||||||
|
it('prevents the second launch of app', function (done) {
|
||||||
|
this.timeout(120000)
|
||||||
|
const appPath = path.join(__dirname, 'fixtures', 'api', 'singleton-old')
|
||||||
|
// First launch should exit with 0.
|
||||||
|
const first = ChildProcess.spawn(remote.process.execPath, [appPath])
|
||||||
|
first.once('exit', (code) => {
|
||||||
|
assert.equal(code, 0)
|
||||||
|
})
|
||||||
|
// Start second app when received output.
|
||||||
|
first.stdout.once('data', () => {
|
||||||
|
// Second launch should exit with 1.
|
||||||
|
const second = ChildProcess.spawn(remote.process.execPath, [appPath])
|
||||||
|
second.once('exit', (code) => {
|
||||||
|
assert.equal(code, 1)
|
||||||
|
done()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('app.requestSingleInstanceLock', () => {
|
||||||
it('prevents the second launch of app', function (done) {
|
it('prevents the second launch of app', function (done) {
|
||||||
this.timeout(120000)
|
this.timeout(120000)
|
||||||
const appPath = path.join(__dirname, 'fixtures', 'api', 'singleton')
|
const appPath = path.join(__dirname, 'fixtures', 'api', 'singleton')
|
||||||
|
|
13
spec/fixtures/api/singleton-old/main.js
vendored
Normal file
13
spec/fixtures/api/singleton-old/main.js
vendored
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
const {app} = require('electron')
|
||||||
|
|
||||||
|
app.once('ready', () => {
|
||||||
|
console.log('started') // ping parent
|
||||||
|
})
|
||||||
|
|
||||||
|
const shouldExit = app.makeSingleInstance(() => {
|
||||||
|
setImmediate(() => app.exit(0))
|
||||||
|
})
|
||||||
|
|
||||||
|
if (shouldExit) {
|
||||||
|
app.exit(1)
|
||||||
|
}
|
5
spec/fixtures/api/singleton-old/package.json
vendored
Normal file
5
spec/fixtures/api/singleton-old/package.json
vendored
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
{
|
||||||
|
"name": "electron-app-singleton",
|
||||||
|
"main": "main.js"
|
||||||
|
}
|
||||||
|
|
8
spec/fixtures/api/singleton/main.js
vendored
8
spec/fixtures/api/singleton/main.js
vendored
|
@ -4,10 +4,12 @@ app.once('ready', () => {
|
||||||
console.log('started') // ping parent
|
console.log('started') // ping parent
|
||||||
})
|
})
|
||||||
|
|
||||||
const shouldExit = app.makeSingleInstance(() => {
|
const gotTheLock = app.requestSingleInstanceLock()
|
||||||
process.nextTick(() => app.exit(0))
|
|
||||||
|
app.on('second-instance', () => {
|
||||||
|
setImmediate(() => app.exit(0))
|
||||||
})
|
})
|
||||||
|
|
||||||
if (shouldExit) {
|
if (!gotTheLock) {
|
||||||
app.exit(1)
|
app.exit(1)
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue