feat(powerMonitor): expose interface to query system idle state (#11807)

* feat(BrowserWindow): expose interface to query system idle state

* test(BrowserWindow): update test cases for querySystemIdle interface

* docs(BrowserWindow): add querySystemIdle interface documentation

* refactor(powerMonitor): move querySystemIdle into powerMonitor

* test(powerMonitor): split test cases for all platform
This commit is contained in:
OJ Kwon 2018-03-13 22:42:08 -07:00 committed by Charles Kerr
parent 90dc897f71
commit e7181eb89c
5 changed files with 143 additions and 24 deletions

View file

@ -5,12 +5,33 @@
#include "atom/browser/api/atom_api_power_monitor.h" #include "atom/browser/api/atom_api_power_monitor.h"
#include "atom/browser/browser.h" #include "atom/browser/browser.h"
#include "atom/common/native_mate_converters/callback.h"
#include "base/power_monitor/power_monitor.h" #include "base/power_monitor/power_monitor.h"
#include "base/power_monitor/power_monitor_device_source.h" #include "base/power_monitor/power_monitor_device_source.h"
#include "native_mate/dictionary.h" #include "native_mate/dictionary.h"
#include "atom/common/node_includes.h" #include "atom/common/node_includes.h"
namespace mate {
template<>
struct Converter<ui::IdleState> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
const ui::IdleState& in) {
switch (in) {
case ui::IDLE_STATE_ACTIVE:
return mate::StringToV8(isolate, "active");
case ui::IDLE_STATE_IDLE:
return mate::StringToV8(isolate, "idle");
case ui::IDLE_STATE_LOCKED:
return mate::StringToV8(isolate, "locked");
case ui::IDLE_STATE_UNKNOWN:
default:
return mate::StringToV8(isolate, "unknown");
}
}
};
} // namespace mate
namespace atom { namespace atom {
namespace api { namespace api {
@ -60,6 +81,22 @@ void PowerMonitor::OnResume() {
Emit("resume"); Emit("resume");
} }
void PowerMonitor::QuerySystemIdleState(v8::Isolate* isolate,
int idle_threshold,
const ui::IdleCallback& callback) {
if (idle_threshold > 0) {
ui::CalculateIdleState(idle_threshold, callback);
} else {
isolate->ThrowException(v8::Exception::TypeError(
mate::StringToV8(isolate,
"Invalid idle threshold, must be greater than 0")));
}
}
void PowerMonitor::QuerySystemIdleTime(const ui::IdleTimeCallback& callback) {
ui::CalculateIdleTime(callback);
}
// static // static
v8::Local<v8::Value> PowerMonitor::Create(v8::Isolate* isolate) { v8::Local<v8::Value> PowerMonitor::Create(v8::Isolate* isolate) {
if (!Browser::Get()->is_ready()) { if (!Browser::Get()->is_ready()) {
@ -76,11 +113,15 @@ v8::Local<v8::Value> PowerMonitor::Create(v8::Isolate* isolate) {
void PowerMonitor::BuildPrototype( void PowerMonitor::BuildPrototype(
v8::Isolate* isolate, v8::Local<v8::FunctionTemplate> prototype) { v8::Isolate* isolate, v8::Local<v8::FunctionTemplate> prototype) {
prototype->SetClassName(mate::StringToV8(isolate, "PowerMonitor")); prototype->SetClassName(mate::StringToV8(isolate, "PowerMonitor"));
#if defined(OS_LINUX)
mate::ObjectTemplateBuilder(isolate, prototype->PrototypeTemplate()) mate::ObjectTemplateBuilder(isolate, prototype->PrototypeTemplate())
.SetMethod("blockShutdown", &PowerMonitor::BlockShutdown) .MakeDestroyable()
.SetMethod("unblockShutdown", &PowerMonitor::UnblockShutdown); #if defined(OS_LINUX)
.SetMethod("blockShutdown", &PowerMonitor::BlockShutdown)
.SetMethod("unblockShutdown", &PowerMonitor::UnblockShutdown)
#endif #endif
.SetMethod("querySystemIdleState", &PowerMonitor::QuerySystemIdleState)
.SetMethod("querySystemIdleTime", &PowerMonitor::QuerySystemIdleTime);
} }
} // namespace api } // namespace api

View file

@ -9,6 +9,7 @@
#include "atom/browser/lib/power_observer.h" #include "atom/browser/lib/power_observer.h"
#include "base/compiler_specific.h" #include "base/compiler_specific.h"
#include "native_mate/handle.h" #include "native_mate/handle.h"
#include "ui/base/idle/idle.h"
namespace atom { namespace atom {
@ -41,6 +42,11 @@ class PowerMonitor : public mate::TrackableObject<PowerMonitor>,
void OnResume() override; void OnResume() override;
private: private:
void QuerySystemIdleState(v8::Isolate* isolate,
int idle_threshold,
const ui::IdleCallback& callback);
void QuerySystemIdleTime(const ui::IdleTimeCallback& callback);
DISALLOW_COPY_AND_ASSIGN(PowerMonitor); DISALLOW_COPY_AND_ASSIGN(PowerMonitor);
}; };

View file

@ -24,6 +24,7 @@
#include "content/public/browser/child_process_security_policy.h" #include "content/public/browser/child_process_security_policy.h"
#include "device/geolocation/geolocation_delegate.h" #include "device/geolocation/geolocation_delegate.h"
#include "device/geolocation/geolocation_provider.h" #include "device/geolocation/geolocation_provider.h"
#include "ui/base/idle/idle.h"
#include "ui/base/l10n/l10n_util.h" #include "ui/base/l10n/l10n_util.h"
#include "v8/include/v8-debug.h" #include "v8/include/v8-debug.h"
@ -159,6 +160,11 @@ int AtomBrowserMainParts::PreCreateThreads() {
fake_browser_process_->SetApplicationLocale( fake_browser_process_->SetApplicationLocale(
brightray::BrowserClient::Get()->GetApplicationLocale()); brightray::BrowserClient::Get()->GetApplicationLocale());
} }
#if defined(OS_MACOSX)
ui::InitIdleMonitor();
#endif
return result; return result;
} }

View file

@ -46,3 +46,25 @@ Emitted when the system is about to reboot or shut down. If the event handler
invokes `e.preventDefault()`, Electron will attempt to delay system shutdown in invokes `e.preventDefault()`, Electron will attempt to delay system shutdown in
order for the app to exit cleanly. If `e.preventDefault()` is called, the app order for the app to exit cleanly. If `e.preventDefault()` is called, the app
should exit as soon as possible by calling something like `app.quit()`. should exit as soon as possible by calling something like `app.quit()`.
## Methods
The `powerMonitor` module has the following methods:
#### `powerMonitor.querySystemIdleState(idleThreshold, callback)`
* `idleThreshold` Integer
* `callback` Function
* `idleState` String - Can be `active`, `idle`, `locked` or `unknown`
Calculate the system idle state. `idleThreshold` is the amount of time (in seconds)
before considered idle. `callback` will be called synchronously on some systems
and with an `idleState` argument that describes the system's state. `locked` is
available on supported systems only.
#### `powerMonitor.querySystemIdleTime(callback)`
* `callback` Function
* `idleTime` Integer - Idle time in seconds
Calculate system idle time in seconds.

View file

@ -10,26 +10,28 @@ const assert = require('assert')
const dbus = require('dbus-native') const dbus = require('dbus-native')
const Promise = require('bluebird') const Promise = require('bluebird')
const skip = process.platform !== 'linux' || !process.env.DBUS_SYSTEM_BUS_ADDRESS; const skip = process.platform !== 'linux' || !process.env.DBUS_SYSTEM_BUS_ADDRESS
(skip ? describe.skip : describe)('powerMonitor', () => { describe('powerMonitor', () => {
let logindMock, powerMonitor, getCalls, emitSignal, reset let logindMock, dbusMockPowerMonitor, getCalls, emitSignal, reset
before(async () => { if (!skip) {
const systemBus = dbus.systemBus() before(async () => {
const loginService = systemBus.getService('org.freedesktop.login1') const systemBus = dbus.systemBus()
const getInterface = Promise.promisify(loginService.getInterface, {context: loginService}) const loginService = systemBus.getService('org.freedesktop.login1')
logindMock = await getInterface('/org/freedesktop/login1', 'org.freedesktop.DBus.Mock') const getInterface = Promise.promisify(loginService.getInterface, {context: loginService})
getCalls = Promise.promisify(logindMock.GetCalls, {context: logindMock}) logindMock = await getInterface('/org/freedesktop/login1', 'org.freedesktop.DBus.Mock')
emitSignal = Promise.promisify(logindMock.EmitSignal, {context: logindMock}) getCalls = Promise.promisify(logindMock.GetCalls, {context: logindMock})
reset = Promise.promisify(logindMock.Reset, {context: logindMock}) emitSignal = Promise.promisify(logindMock.EmitSignal, {context: logindMock})
}) reset = Promise.promisify(logindMock.Reset, {context: logindMock})
})
after(async () => { after(async () => {
await reset() await reset()
}) })
}
describe('when powerMonitor module is loaded', () => { (skip ? describe.skip : describe)('when powerMonitor module is loaded with dbus mock', () => {
function onceMethodCalled (done) { function onceMethodCalled (done) {
function cb () { function cb () {
logindMock.removeListener('MethodCalled', cb) logindMock.removeListener('MethodCalled', cb)
@ -41,7 +43,7 @@ const skip = process.platform !== 'linux' || !process.env.DBUS_SYSTEM_BUS_ADDRES
before((done) => { before((done) => {
logindMock.on('MethodCalled', onceMethodCalled(done)) logindMock.on('MethodCalled', onceMethodCalled(done))
// lazy load powerMonitor after we listen to MethodCalled mock signal // lazy load powerMonitor after we listen to MethodCalled mock signal
powerMonitor = require('electron').remote.powerMonitor dbusMockPowerMonitor = require('electron').remote.powerMonitor
}) })
it('should call Inhibit to delay suspend', async () => { it('should call Inhibit to delay suspend', async () => {
@ -59,14 +61,14 @@ const skip = process.platform !== 'linux' || !process.env.DBUS_SYSTEM_BUS_ADDRES
describe('when PrepareForSleep(true) signal is sent by logind', () => { describe('when PrepareForSleep(true) signal is sent by logind', () => {
it('should emit "suspend" event', (done) => { it('should emit "suspend" event', (done) => {
powerMonitor.once('suspend', () => done()) dbusMockPowerMonitor.once('suspend', () => done())
emitSignal('org.freedesktop.login1.Manager', 'PrepareForSleep', emitSignal('org.freedesktop.login1.Manager', 'PrepareForSleep',
'b', [['b', true]]) 'b', [['b', true]])
}) })
describe('when PrepareForSleep(false) signal is sent by logind', () => { describe('when PrepareForSleep(false) signal is sent by logind', () => {
it('should emit "resume" event', (done) => { it('should emit "resume" event', (done) => {
powerMonitor.once('resume', () => done()) dbusMockPowerMonitor.once('resume', () => done())
emitSignal('org.freedesktop.login1.Manager', 'PrepareForSleep', emitSignal('org.freedesktop.login1.Manager', 'PrepareForSleep',
'b', [['b', false]]) 'b', [['b', false]])
}) })
@ -90,7 +92,7 @@ const skip = process.platform !== 'linux' || !process.env.DBUS_SYSTEM_BUS_ADDRES
before(async () => { before(async () => {
const calls = await getCalls() const calls = await getCalls()
assert.equal(calls.length, 2) assert.equal(calls.length, 2)
powerMonitor.once('shutdown', () => { }) dbusMockPowerMonitor.once('shutdown', () => { })
}) })
it('should call Inhibit to delay shutdown', async () => { it('should call Inhibit to delay shutdown', async () => {
@ -108,11 +110,53 @@ const skip = process.platform !== 'linux' || !process.env.DBUS_SYSTEM_BUS_ADDRES
describe('when PrepareForShutdown(true) signal is sent by logind', () => { describe('when PrepareForShutdown(true) signal is sent by logind', () => {
it('should emit "shutdown" event', (done) => { it('should emit "shutdown" event', (done) => {
powerMonitor.once('shutdown', () => { done() }) dbusMockPowerMonitor.once('shutdown', () => { done() })
emitSignal('org.freedesktop.login1.Manager', 'PrepareForShutdown', emitSignal('org.freedesktop.login1.Manager', 'PrepareForShutdown',
'b', [['b', true]]) 'b', [['b', true]])
}) })
}) })
}) })
}) })
describe('when powerMonitor module is loaded', () => {
let powerMonitor
before(() => {
powerMonitor = require('electron').remote.powerMonitor
})
describe('powerMonitor.querySystemIdleState', () => {
it('notify current system idle state', (done) => {
powerMonitor.querySystemIdleState(1, (idleState) => {
assert.ok(idleState)
done()
})
})
it('does not accept non positive integer threshold', () => {
assert.throws(() => {
powerMonitor.querySystemIdleState(-1, (idleState) => {
})
})
assert.throws(() => {
powerMonitor.querySystemIdleState(NaN, (idleState) => {
})
})
assert.throws(() => {
powerMonitor.querySystemIdleState('a', (idleState) => {
})
})
})
})
describe('powerMonitor.querySystemIdleTime', () => {
it('notify current system idle time', (done) => {
powerMonitor.querySystemIdleTime((idleTime) => {
assert.ok(idleTime >= 0)
done()
})
})
})
})
}) })