// Copyright (c) 2015 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include "shell/browser/api/atom_api_power_save_blocker.h" #include #include "base/bind_helpers.h" #include "base/task/post_task.h" #include "base/threading/thread_task_runner_handle.h" #include "content/public/browser/system_connector.h" #include "gin/dictionary.h" #include "gin/function_template.h" #include "services/device/public/mojom/constants.mojom.h" #include "services/device/public/mojom/wake_lock_provider.mojom.h" #include "services/service_manager/public/cpp/connector.h" #include "shell/common/node_includes.h" namespace gin { template <> struct Converter { static bool FromV8(v8::Isolate* isolate, v8::Local val, device::mojom::WakeLockType* out) { std::string type; if (!ConvertFromV8(isolate, val, &type)) return false; if (type == "prevent-app-suspension") *out = device::mojom::WakeLockType::kPreventAppSuspension; else if (type == "prevent-display-sleep") *out = device::mojom::WakeLockType::kPreventDisplaySleep; else return false; return true; } }; } // namespace gin namespace electron { namespace api { gin::WrapperInfo PowerSaveBlocker::kWrapperInfo = {gin::kEmbedderNativeGin}; PowerSaveBlocker::PowerSaveBlocker(v8::Isolate* isolate) : current_lock_type_(device::mojom::WakeLockType::kPreventAppSuspension), is_wake_lock_active_(false) {} PowerSaveBlocker::~PowerSaveBlocker() = default; void PowerSaveBlocker::UpdatePowerSaveBlocker() { if (wake_lock_types_.empty()) { if (is_wake_lock_active_) { GetWakeLock()->CancelWakeLock(); is_wake_lock_active_ = false; } return; } // |WakeLockType::kPreventAppSuspension| keeps system active, but allows // screen to be turned off. // |WakeLockType::kPreventDisplaySleep| keeps system and screen active, has a // higher precedence level than |WakeLockType::kPreventAppSuspension|. // // Only the highest-precedence blocker type takes effect. device::mojom::WakeLockType new_lock_type = device::mojom::WakeLockType::kPreventAppSuspension; for (const auto& element : wake_lock_types_) { if (element.second == device::mojom::WakeLockType::kPreventDisplaySleep) { new_lock_type = device::mojom::WakeLockType::kPreventDisplaySleep; break; } } if (current_lock_type_ != new_lock_type) { GetWakeLock()->ChangeType(new_lock_type, base::DoNothing()); current_lock_type_ = new_lock_type; } if (!is_wake_lock_active_) { GetWakeLock()->RequestWakeLock(); is_wake_lock_active_ = true; } } device::mojom::WakeLock* PowerSaveBlocker::GetWakeLock() { if (!wake_lock_) { mojo::Remote wake_lock_provider; DCHECK(content::GetSystemConnector()); content::GetSystemConnector()->Connect( device::mojom::kServiceName, wake_lock_provider.BindNewPipeAndPassReceiver()); wake_lock_provider->GetWakeLockWithoutContext( device::mojom::WakeLockType::kPreventAppSuspension, device::mojom::WakeLockReason::kOther, ELECTRON_PRODUCT_NAME, wake_lock_.BindNewPipeAndPassReceiver()); } return wake_lock_.get(); } int PowerSaveBlocker::Start(device::mojom::WakeLockType type) { static int count = 0; wake_lock_types_[count] = type; UpdatePowerSaveBlocker(); return count++; } bool PowerSaveBlocker::Stop(int id) { bool success = wake_lock_types_.erase(id) > 0; UpdatePowerSaveBlocker(); return success; } bool PowerSaveBlocker::IsStarted(int id) { return wake_lock_types_.find(id) != wake_lock_types_.end(); } // static gin::Handle PowerSaveBlocker::Create(v8::Isolate* isolate) { return gin::CreateHandle(isolate, new PowerSaveBlocker(isolate)); } gin::ObjectTemplateBuilder PowerSaveBlocker::GetObjectTemplateBuilder( v8::Isolate* isolate) { return gin::Wrappable::GetObjectTemplateBuilder(isolate) .SetMethod("start", &PowerSaveBlocker::Start) .SetMethod("stop", &PowerSaveBlocker::Stop) .SetMethod("isStarted", &PowerSaveBlocker::IsStarted); } } // namespace api } // namespace electron namespace { void Initialize(v8::Local exports, v8::Local unused, v8::Local context, void* priv) { v8::Isolate* isolate = context->GetIsolate(); gin::Dictionary dict(isolate, exports); dict.Set("powerSaveBlocker", electron::api::PowerSaveBlocker::Create(isolate)); } } // namespace NODE_LINKED_MODULE_CONTEXT_AWARE(atom_browser_power_save_blocker, Initialize)