2021-09-23 11:00:11 +00:00
|
|
|
// Copyright (c) 2021 Microsoft, Inc.
|
|
|
|
// Use of this source code is governed by the MIT license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
#include "shell/browser/hid/electron_hid_delegate.h"
|
|
|
|
|
|
|
|
#include <string>
|
|
|
|
#include <utility>
|
|
|
|
|
2021-11-01 14:00:07 +00:00
|
|
|
#include "base/command_line.h"
|
2021-09-23 11:00:11 +00:00
|
|
|
#include "content/public/browser/web_contents.h"
|
2021-11-01 14:00:07 +00:00
|
|
|
#include "services/device/public/cpp/hid/hid_switches.h"
|
2021-09-23 11:00:11 +00:00
|
|
|
#include "shell/browser/hid/hid_chooser_context.h"
|
|
|
|
#include "shell/browser/hid/hid_chooser_context_factory.h"
|
|
|
|
#include "shell/browser/hid/hid_chooser_controller.h"
|
|
|
|
#include "shell/browser/web_contents_permission_helper.h"
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
electron::HidChooserContext* GetChooserContext(
|
|
|
|
content::RenderFrameHost* frame) {
|
|
|
|
auto* web_contents = content::WebContents::FromRenderFrameHost(frame);
|
|
|
|
auto* browser_context = web_contents->GetBrowserContext();
|
|
|
|
return electron::HidChooserContextFactory::GetForBrowserContext(
|
|
|
|
browser_context);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace electron {
|
|
|
|
|
|
|
|
ElectronHidDelegate::ElectronHidDelegate() = default;
|
|
|
|
|
|
|
|
ElectronHidDelegate::~ElectronHidDelegate() = default;
|
|
|
|
|
|
|
|
std::unique_ptr<content::HidChooser> ElectronHidDelegate::RunChooser(
|
|
|
|
content::RenderFrameHost* render_frame_host,
|
|
|
|
std::vector<blink::mojom::HidDeviceFilterPtr> filters,
|
2022-04-12 11:19:14 +00:00
|
|
|
std::vector<blink::mojom::HidDeviceFilterPtr> exclusion_filters,
|
2021-09-23 11:00:11 +00:00
|
|
|
content::HidChooser::Callback callback) {
|
|
|
|
electron::HidChooserContext* chooser_context =
|
|
|
|
GetChooserContext(render_frame_host);
|
|
|
|
if (!device_observation_.IsObserving())
|
|
|
|
device_observation_.Observe(chooser_context);
|
|
|
|
|
|
|
|
HidChooserController* controller = ControllerForFrame(render_frame_host);
|
|
|
|
if (controller) {
|
|
|
|
DeleteControllerForFrame(render_frame_host);
|
|
|
|
}
|
|
|
|
AddControllerForFrame(render_frame_host, std::move(filters),
|
2022-04-12 11:19:14 +00:00
|
|
|
std::move(exclusion_filters), std::move(callback));
|
2021-09-23 11:00:11 +00:00
|
|
|
|
|
|
|
// Return a nullptr because the return value isn't used for anything, eg
|
|
|
|
// there is no mechanism to cancel navigator.hid.requestDevice(). The return
|
|
|
|
// value is simply used in Chromium to cleanup the chooser UI once the serial
|
|
|
|
// service is destroyed.
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ElectronHidDelegate::CanRequestDevicePermission(
|
|
|
|
content::RenderFrameHost* render_frame_host) {
|
|
|
|
auto* web_contents =
|
|
|
|
content::WebContents::FromRenderFrameHost(render_frame_host);
|
|
|
|
auto* permission_helper =
|
|
|
|
WebContentsPermissionHelper::FromWebContents(web_contents);
|
|
|
|
return permission_helper->CheckHIDAccessPermission(
|
|
|
|
web_contents->GetMainFrame()->GetLastCommittedOrigin());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ElectronHidDelegate::HasDevicePermission(
|
|
|
|
content::RenderFrameHost* render_frame_host,
|
|
|
|
const device::mojom::HidDeviceInfo& device) {
|
|
|
|
auto* chooser_context = GetChooserContext(render_frame_host);
|
|
|
|
const auto& origin =
|
|
|
|
render_frame_host->GetMainFrame()->GetLastCommittedOrigin();
|
|
|
|
return chooser_context->HasDevicePermission(origin, device,
|
|
|
|
render_frame_host);
|
|
|
|
}
|
|
|
|
|
2022-02-10 02:58:52 +00:00
|
|
|
void ElectronHidDelegate::RevokeDevicePermission(
|
|
|
|
content::RenderFrameHost* render_frame_host,
|
|
|
|
const device::mojom::HidDeviceInfo& device) {
|
2022-05-23 19:13:18 +00:00
|
|
|
auto* chooser_context = GetChooserContext(render_frame_host);
|
|
|
|
const auto& origin =
|
|
|
|
render_frame_host->GetMainFrame()->GetLastCommittedOrigin();
|
|
|
|
return chooser_context->RevokeDevicePermission(origin, device,
|
|
|
|
render_frame_host);
|
2022-02-10 02:58:52 +00:00
|
|
|
}
|
|
|
|
|
2021-09-23 11:00:11 +00:00
|
|
|
device::mojom::HidManager* ElectronHidDelegate::GetHidManager(
|
|
|
|
content::RenderFrameHost* render_frame_host) {
|
|
|
|
auto* chooser_context = GetChooserContext(render_frame_host);
|
|
|
|
return chooser_context->GetHidManager();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ElectronHidDelegate::AddObserver(
|
|
|
|
content::RenderFrameHost* render_frame_host,
|
|
|
|
Observer* observer) {
|
|
|
|
observer_list_.AddObserver(observer);
|
|
|
|
auto* chooser_context = GetChooserContext(render_frame_host);
|
|
|
|
if (!device_observation_.IsObserving())
|
|
|
|
device_observation_.Observe(chooser_context);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ElectronHidDelegate::RemoveObserver(
|
|
|
|
content::RenderFrameHost* render_frame_host,
|
|
|
|
content::HidDelegate::Observer* observer) {
|
|
|
|
observer_list_.RemoveObserver(observer);
|
|
|
|
}
|
|
|
|
|
|
|
|
const device::mojom::HidDeviceInfo* ElectronHidDelegate::GetDeviceInfo(
|
|
|
|
content::RenderFrameHost* render_frame_host,
|
|
|
|
const std::string& guid) {
|
|
|
|
auto* chooser_context = GetChooserContext(render_frame_host);
|
|
|
|
return chooser_context->GetDeviceInfo(guid);
|
|
|
|
}
|
|
|
|
|
2022-02-10 02:58:52 +00:00
|
|
|
bool ElectronHidDelegate::IsFidoAllowedForOrigin(
|
|
|
|
content::RenderFrameHost* render_frame_host,
|
|
|
|
const url::Origin& origin) {
|
2021-11-01 14:00:07 +00:00
|
|
|
return base::CommandLine::ForCurrentProcess()->HasSwitch(
|
|
|
|
switches::kDisableHidBlocklist);
|
2021-09-23 11:00:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ElectronHidDelegate::OnDeviceAdded(
|
|
|
|
const device::mojom::HidDeviceInfo& device_info) {
|
|
|
|
for (auto& observer : observer_list_)
|
|
|
|
observer.OnDeviceAdded(device_info);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ElectronHidDelegate::OnDeviceRemoved(
|
|
|
|
const device::mojom::HidDeviceInfo& device_info) {
|
|
|
|
for (auto& observer : observer_list_)
|
|
|
|
observer.OnDeviceRemoved(device_info);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ElectronHidDelegate::OnDeviceChanged(
|
|
|
|
const device::mojom::HidDeviceInfo& device_info) {
|
|
|
|
for (auto& observer : observer_list_)
|
|
|
|
observer.OnDeviceChanged(device_info);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ElectronHidDelegate::OnHidManagerConnectionError() {
|
|
|
|
device_observation_.Reset();
|
|
|
|
|
|
|
|
for (auto& observer : observer_list_)
|
|
|
|
observer.OnHidManagerConnectionError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ElectronHidDelegate::OnHidChooserContextShutdown() {
|
|
|
|
device_observation_.Reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
HidChooserController* ElectronHidDelegate::ControllerForFrame(
|
|
|
|
content::RenderFrameHost* render_frame_host) {
|
|
|
|
auto mapping = controller_map_.find(render_frame_host);
|
|
|
|
return mapping == controller_map_.end() ? nullptr : mapping->second.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
HidChooserController* ElectronHidDelegate::AddControllerForFrame(
|
|
|
|
content::RenderFrameHost* render_frame_host,
|
|
|
|
std::vector<blink::mojom::HidDeviceFilterPtr> filters,
|
2022-04-12 11:19:14 +00:00
|
|
|
std::vector<blink::mojom::HidDeviceFilterPtr> exclusion_filters,
|
2021-09-23 11:00:11 +00:00
|
|
|
content::HidChooser::Callback callback) {
|
|
|
|
auto* web_contents =
|
|
|
|
content::WebContents::FromRenderFrameHost(render_frame_host);
|
|
|
|
auto controller = std::make_unique<HidChooserController>(
|
2022-04-12 11:19:14 +00:00
|
|
|
render_frame_host, std::move(filters), std::move(exclusion_filters),
|
|
|
|
std::move(callback), web_contents, weak_factory_.GetWeakPtr());
|
2021-09-23 11:00:11 +00:00
|
|
|
controller_map_.insert(
|
|
|
|
std::make_pair(render_frame_host, std::move(controller)));
|
|
|
|
return ControllerForFrame(render_frame_host);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ElectronHidDelegate::DeleteControllerForFrame(
|
|
|
|
content::RenderFrameHost* render_frame_host) {
|
|
|
|
controller_map_.erase(render_frame_host);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace electron
|