| 
									
										
										
										
											2021-09-23 07:00:11 -04: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/hid_chooser_controller.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-02 21:21:59 -05:00
										 |  |  | #include <algorithm>
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | #include <utility>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "base/command_line.h"
 | 
					
						
							|  |  |  | #include "base/containers/contains.h"
 | 
					
						
							| 
									
										
										
										
											2023-02-03 12:43:42 +01:00
										 |  |  | #include "base/functional/bind.h"
 | 
					
						
							| 
									
										
										
										
											2024-07-29 12:42:57 -05:00
										 |  |  | #include "content/public/browser/web_contents.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | #include "gin/data_object_builder.h"
 | 
					
						
							|  |  |  | #include "services/device/public/cpp/hid/hid_blocklist.h"
 | 
					
						
							|  |  |  | #include "services/device/public/cpp/hid/hid_switches.h"
 | 
					
						
							|  |  |  | #include "shell/browser/api/electron_api_session.h"
 | 
					
						
							| 
									
										
										
										
											2024-07-29 12:42:57 -05:00
										 |  |  | #include "shell/browser/hid/electron_hid_delegate.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | #include "shell/browser/hid/hid_chooser_context.h"
 | 
					
						
							|  |  |  | #include "shell/browser/hid/hid_chooser_context_factory.h"
 | 
					
						
							|  |  |  | #include "shell/browser/javascript_environment.h"
 | 
					
						
							|  |  |  | #include "shell/common/gin_converters/callback_converter.h"
 | 
					
						
							|  |  |  | #include "shell/common/gin_converters/content_converter.h"
 | 
					
						
							| 
									
										
										
										
											2022-05-23 15:13:18 -04:00
										 |  |  | #include "shell/common/gin_converters/hid_device_info_converter.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | #include "shell/common/gin_converters/value_converter.h"
 | 
					
						
							| 
									
										
										
										
											2024-09-16 15:53:04 -05:00
										 |  |  | #include "shell/common/node_util.h"
 | 
					
						
							| 
									
										
										
										
											2024-11-04 12:58:16 -06:00
										 |  |  | #include "third_party/abseil-cpp/absl/strings/str_format.h"
 | 
					
						
							| 
									
										
										
										
											2024-09-17 14:58:56 -05:00
										 |  |  | #include "third_party/blink/public/mojom/devtools/console_message.mojom.h"
 | 
					
						
							|  |  |  | #include "third_party/blink/public/mojom/hid/hid.mojom.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | #include "ui/base/l10n/l10n_util.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-12 13:19:14 +02:00
										 |  |  | bool FilterMatch(const blink::mojom::HidDeviceFilterPtr& filter, | 
					
						
							|  |  |  |                  const device::mojom::HidDeviceInfo& device) { | 
					
						
							|  |  |  |   if (filter->device_ids) { | 
					
						
							|  |  |  |     if (filter->device_ids->is_vendor()) { | 
					
						
							|  |  |  |       if (filter->device_ids->get_vendor() != device.vendor_id) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } else if (filter->device_ids->is_vendor_and_product()) { | 
					
						
							|  |  |  |       const auto& vendor_and_product = | 
					
						
							|  |  |  |           filter->device_ids->get_vendor_and_product(); | 
					
						
							|  |  |  |       if (vendor_and_product->vendor != device.vendor_id) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |       if (vendor_and_product->product != device.product_id) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (filter->usage) { | 
					
						
							|  |  |  |     if (filter->usage->is_page()) { | 
					
						
							|  |  |  |       const uint16_t usage_page = filter->usage->get_page(); | 
					
						
							| 
									
										
										
										
											2024-08-26 09:58:32 -05:00
										 |  |  |       auto find_it = std::ranges::find_if( | 
					
						
							|  |  |  |           device.collections, | 
					
						
							|  |  |  |           [=](const device::mojom::HidCollectionInfoPtr& c) { | 
					
						
							|  |  |  |             return usage_page == c->usage->usage_page; | 
					
						
							|  |  |  |           }); | 
					
						
							| 
									
										
										
										
											2022-04-12 13:19:14 +02:00
										 |  |  |       if (find_it == device.collections.end()) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } else if (filter->usage->is_usage_and_page()) { | 
					
						
							|  |  |  |       const auto& usage_and_page = filter->usage->get_usage_and_page(); | 
					
						
							|  |  |  |       auto find_it = std::find_if( | 
					
						
							|  |  |  |           device.collections.begin(), device.collections.end(), | 
					
						
							|  |  |  |           [&usage_and_page](const device::mojom::HidCollectionInfoPtr& c) { | 
					
						
							|  |  |  |             return usage_and_page->usage_page == c->usage->usage_page && | 
					
						
							|  |  |  |                    usage_and_page->usage == c->usage->usage; | 
					
						
							|  |  |  |           }); | 
					
						
							|  |  |  |       if (find_it == device.collections.end()) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | }  // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace electron { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | HidChooserController::HidChooserController( | 
					
						
							|  |  |  |     content::RenderFrameHost* render_frame_host, | 
					
						
							|  |  |  |     std::vector<blink::mojom::HidDeviceFilterPtr> filters, | 
					
						
							| 
									
										
										
										
											2022-04-12 13:19:14 +02:00
										 |  |  |     std::vector<blink::mojom::HidDeviceFilterPtr> exclusion_filters, | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |     content::HidChooser::Callback callback, | 
					
						
							|  |  |  |     content::WebContents* web_contents, | 
					
						
							|  |  |  |     base::WeakPtr<ElectronHidDelegate> hid_delegate) | 
					
						
							|  |  |  |     : WebContentsObserver(web_contents), | 
					
						
							|  |  |  |       filters_(std::move(filters)), | 
					
						
							| 
									
										
										
										
											2022-04-12 13:19:14 +02:00
										 |  |  |       exclusion_filters_(std::move(exclusion_filters)), | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |       callback_(std::move(callback)), | 
					
						
							| 
									
										
										
										
											2023-10-19 01:19:51 +02:00
										 |  |  |       initiator_document_(render_frame_host->GetWeakDocumentPtr()), | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |       origin_(content::WebContents::FromRenderFrameHost(render_frame_host) | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |                   ->GetPrimaryMainFrame() | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |                   ->GetLastCommittedOrigin()), | 
					
						
							|  |  |  |       hid_delegate_(hid_delegate), | 
					
						
							|  |  |  |       render_frame_host_id_(render_frame_host->GetGlobalId()) { | 
					
						
							| 
									
										
										
										
											2023-10-19 01:19:51 +02:00
										 |  |  |   // The use above of GetMainFrame is safe as content::HidService instances are
 | 
					
						
							|  |  |  |   // not created for fenced frames.
 | 
					
						
							|  |  |  |   DCHECK(!render_frame_host->IsNestedWithinFencedFrame()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   chooser_context_ = HidChooserContextFactory::GetForBrowserContext( | 
					
						
							|  |  |  |                          web_contents->GetBrowserContext()) | 
					
						
							|  |  |  |                          ->AsWeakPtr(); | 
					
						
							|  |  |  |   DCHECK(chooser_context_); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   chooser_context_->GetHidManager()->GetDevices(base::BindOnce( | 
					
						
							|  |  |  |       &HidChooserController::OnGotDevices, weak_factory_.GetWeakPtr())); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | HidChooserController::~HidChooserController() { | 
					
						
							|  |  |  |   if (callback_) | 
					
						
							|  |  |  |     std::move(callback_).Run(std::vector<device::mojom::HidDeviceInfoPtr>()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-23 15:13:18 -04:00
										 |  |  | // static
 | 
					
						
							| 
									
										
										
										
											2023-07-19 09:54:30 -05:00
										 |  |  | const std::string& HidChooserController::PhysicalDeviceIdFromDeviceInfo( | 
					
						
							| 
									
										
										
										
											2022-05-23 15:13:18 -04:00
										 |  |  |     const device::mojom::HidDeviceInfo& device) { | 
					
						
							|  |  |  |   // A single physical device may expose multiple HID interfaces, each
 | 
					
						
							|  |  |  |   // represented by a HidDeviceInfo object. When a device exposes multiple
 | 
					
						
							|  |  |  |   // HID interfaces, the HidDeviceInfo objects will share a common
 | 
					
						
							|  |  |  |   // |physical_device_id|. Group these devices so that a single chooser item
 | 
					
						
							|  |  |  |   // is shown for each physical device. If a device's physical device ID is
 | 
					
						
							|  |  |  |   // empty, use its GUID instead.
 | 
					
						
							|  |  |  |   return device.physical_device_id.empty() ? device.guid | 
					
						
							|  |  |  |                                            : device.physical_device_id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | api::Session* HidChooserController::GetSession() { | 
					
						
							|  |  |  |   if (!web_contents()) { | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return api::Session::FromBrowserContext(web_contents()->GetBrowserContext()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserController::OnDeviceAdded( | 
					
						
							|  |  |  |     const device::mojom::HidDeviceInfo& device) { | 
					
						
							|  |  |  |   if (!DisplayDevice(device)) | 
					
						
							|  |  |  |     return; | 
					
						
							| 
									
										
										
										
											2023-10-19 01:19:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   if (AddDeviceInfo(device)) { | 
					
						
							|  |  |  |     api::Session* session = GetSession(); | 
					
						
							|  |  |  |     if (session) { | 
					
						
							|  |  |  |       auto* rfh = content::RenderFrameHost::FromID(render_frame_host_id_); | 
					
						
							|  |  |  |       v8::Isolate* isolate = JavascriptEnvironment::GetIsolate(); | 
					
						
							|  |  |  |       v8::HandleScope scope(isolate); | 
					
						
							|  |  |  |       v8::Local<v8::Object> details = gin::DataObjectBuilder(isolate) | 
					
						
							|  |  |  |                                           .Set("device", device.Clone()) | 
					
						
							|  |  |  |                                           .Set("frame", rfh) | 
					
						
							|  |  |  |                                           .Build(); | 
					
						
							|  |  |  |       session->Emit("hid-device-added", details); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserController::OnDeviceRemoved( | 
					
						
							|  |  |  |     const device::mojom::HidDeviceInfo& device) { | 
					
						
							| 
									
										
										
										
											2023-07-19 09:54:30 -05:00
										 |  |  |   if (!base::Contains(items_, PhysicalDeviceIdFromDeviceInfo(device))) | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |     return; | 
					
						
							| 
									
										
										
										
											2023-07-19 09:54:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-19 01:19:51 +02:00
										 |  |  |   api::Session* session = GetSession(); | 
					
						
							|  |  |  |   if (session) { | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |     auto* rfh = content::RenderFrameHost::FromID(render_frame_host_id_); | 
					
						
							|  |  |  |     v8::Isolate* isolate = JavascriptEnvironment::GetIsolate(); | 
					
						
							|  |  |  |     v8::HandleScope scope(isolate); | 
					
						
							|  |  |  |     v8::Local<v8::Object> details = gin::DataObjectBuilder(isolate) | 
					
						
							|  |  |  |                                         .Set("device", device.Clone()) | 
					
						
							|  |  |  |                                         .Set("frame", rfh) | 
					
						
							|  |  |  |                                         .Build(); | 
					
						
							|  |  |  |     session->Emit("hid-device-removed", details); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   RemoveDeviceInfo(device); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserController::OnDeviceChanged( | 
					
						
							|  |  |  |     const device::mojom::HidDeviceInfo& device) { | 
					
						
							|  |  |  |   bool has_chooser_item = | 
					
						
							|  |  |  |       base::Contains(items_, PhysicalDeviceIdFromDeviceInfo(device)); | 
					
						
							|  |  |  |   if (!DisplayDevice(device)) { | 
					
						
							|  |  |  |     if (has_chooser_item) | 
					
						
							|  |  |  |       OnDeviceRemoved(device); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!has_chooser_item) { | 
					
						
							|  |  |  |     OnDeviceAdded(device); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Update the item to replace the old device info with |device|.
 | 
					
						
							|  |  |  |   UpdateDeviceInfo(device); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserController::OnDeviceChosen(gin::Arguments* args) { | 
					
						
							|  |  |  |   std::string device_id; | 
					
						
							|  |  |  |   if (!args->GetNext(&device_id) || device_id.empty()) { | 
					
						
							|  |  |  |     RunCallback({}); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     auto find_it = device_map_.find(device_id); | 
					
						
							|  |  |  |     if (find_it != device_map_.end()) { | 
					
						
							|  |  |  |       auto& device_infos = find_it->second; | 
					
						
							|  |  |  |       std::vector<device::mojom::HidDeviceInfoPtr> devices; | 
					
						
							|  |  |  |       devices.reserve(device_infos.size()); | 
					
						
							|  |  |  |       for (auto& device : device_infos) { | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |         chooser_context_->GrantDevicePermission(origin_, *device); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |         devices.push_back(device->Clone()); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       RunCallback(std::move(devices)); | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2024-09-16 15:53:04 -05:00
										 |  |  |       util::EmitWarning( | 
					
						
							|  |  |  |           base::StrCat({"The device id ", device_id, " was not found."}), | 
					
						
							|  |  |  |           "UnknownHIDDeviceId"); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |       RunCallback({}); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserController::OnHidManagerConnectionError() { | 
					
						
							|  |  |  |   observation_.Reset(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserController::OnHidChooserContextShutdown() { | 
					
						
							|  |  |  |   observation_.Reset(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserController::OnGotDevices( | 
					
						
							|  |  |  |     std::vector<device::mojom::HidDeviceInfoPtr> devices) { | 
					
						
							|  |  |  |   std::vector<device::mojom::HidDeviceInfoPtr> devicesToDisplay; | 
					
						
							|  |  |  |   devicesToDisplay.reserve(devices.size()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (auto& device : devices) { | 
					
						
							|  |  |  |     if (DisplayDevice(*device)) { | 
					
						
							| 
									
										
										
										
											2023-10-19 01:19:51 +02:00
										 |  |  |       if (AddDeviceInfo(*device)) | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |         devicesToDisplay.push_back(device->Clone()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Listen to HidChooserContext for OnDeviceAdded/Removed events after the
 | 
					
						
							|  |  |  |   // enumeration.
 | 
					
						
							|  |  |  |   if (chooser_context_) | 
					
						
							|  |  |  |     observation_.Observe(chooser_context_.get()); | 
					
						
							| 
									
										
										
										
											2023-10-19 01:19:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   bool prevent_default = false; | 
					
						
							|  |  |  |   api::Session* session = GetSession(); | 
					
						
							|  |  |  |   if (session) { | 
					
						
							|  |  |  |     auto* rfh = content::RenderFrameHost::FromID(render_frame_host_id_); | 
					
						
							|  |  |  |     v8::Isolate* isolate = JavascriptEnvironment::GetIsolate(); | 
					
						
							|  |  |  |     v8::HandleScope scope(isolate); | 
					
						
							|  |  |  |     v8::Local<v8::Object> details = gin::DataObjectBuilder(isolate) | 
					
						
							|  |  |  |                                         .Set("deviceList", devicesToDisplay) | 
					
						
							|  |  |  |                                         .Set("frame", rfh) | 
					
						
							|  |  |  |                                         .Build(); | 
					
						
							|  |  |  |     prevent_default = | 
					
						
							|  |  |  |         session->Emit("select-hid-device", details, | 
					
						
							| 
									
										
										
										
											2023-05-18 23:01:44 +02:00
										 |  |  |                       base::BindRepeating(&HidChooserController::OnDeviceChosen, | 
					
						
							|  |  |  |                                           weak_factory_.GetWeakPtr())); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (!prevent_default) { | 
					
						
							|  |  |  |     RunCallback({}); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool HidChooserController::DisplayDevice( | 
					
						
							|  |  |  |     const device::mojom::HidDeviceInfo& device) const { | 
					
						
							| 
									
										
										
										
											2023-10-19 01:19:51 +02:00
										 |  |  |   // Check if `device` has a top-level collection with a FIDO usage. FIDO
 | 
					
						
							|  |  |  |   // devices may be displayed if the origin is privileged or the blocklist is
 | 
					
						
							|  |  |  |   // disabled.
 | 
					
						
							|  |  |  |   const bool has_fido_collection = | 
					
						
							|  |  |  |       base::Contains(device.collections, device::mojom::kPageFido, | 
					
						
							|  |  |  |                      [](const auto& c) { return c->usage->usage_page; }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (has_fido_collection) { | 
					
						
							|  |  |  |     if (base::CommandLine::ForCurrentProcess()->HasSwitch( | 
					
						
							|  |  |  |             switches::kDisableHidBlocklist) || | 
					
						
							|  |  |  |         (chooser_context_ && | 
					
						
							|  |  |  |          chooser_context_->IsFidoAllowedForOrigin(origin_))) { | 
					
						
							|  |  |  |       return FilterMatchesAny(device) && !IsExcluded(device); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     AddMessageToConsole( | 
					
						
							|  |  |  |         blink::mojom::ConsoleMessageLevel::kInfo, | 
					
						
							| 
									
										
										
										
											2024-11-04 12:58:16 -06:00
										 |  |  |         absl::StrFormat( | 
					
						
							| 
									
										
										
										
											2023-10-19 01:19:51 +02:00
										 |  |  |             "Chooser dialog is not displaying a FIDO HID device: vendorId=%d, " | 
					
						
							|  |  |  |             "productId=%d, name='%s', serial='%s'", | 
					
						
							| 
									
										
										
										
											2025-06-26 08:18:06 -05:00
										 |  |  |             device.vendor_id, device.product_id, device.product_name, | 
					
						
							|  |  |  |             device.serial_number)); | 
					
						
							| 
									
										
										
										
											2023-10-19 01:19:51 +02:00
										 |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (device.is_excluded_by_blocklist) { | 
					
						
							|  |  |  |     AddMessageToConsole( | 
					
						
							|  |  |  |         blink::mojom::ConsoleMessageLevel::kInfo, | 
					
						
							| 
									
										
										
										
											2024-11-04 12:58:16 -06:00
										 |  |  |         absl::StrFormat("Chooser dialog is not displaying a device excluded by " | 
					
						
							|  |  |  |                         "the HID blocklist: vendorId=%d, " | 
					
						
							|  |  |  |                         "productId=%d, name='%s', serial='%s'", | 
					
						
							|  |  |  |                         device.vendor_id, device.product_id, | 
					
						
							| 
									
										
										
										
											2025-06-26 08:18:06 -05:00
										 |  |  |                         device.product_name, device.serial_number)); | 
					
						
							| 
									
										
										
										
											2023-10-19 01:19:51 +02:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-12 13:19:14 +02:00
										 |  |  |   return FilterMatchesAny(device) && !IsExcluded(device); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool HidChooserController::FilterMatchesAny( | 
					
						
							|  |  |  |     const device::mojom::HidDeviceInfo& device) const { | 
					
						
							|  |  |  |   if (filters_.empty()) | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (const auto& filter : filters_) { | 
					
						
							| 
									
										
										
										
											2022-04-12 13:19:14 +02:00
										 |  |  |     if (FilterMatch(filter, device)) | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-12 13:19:14 +02:00
										 |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-12 13:19:14 +02:00
										 |  |  | bool HidChooserController::IsExcluded( | 
					
						
							|  |  |  |     const device::mojom::HidDeviceInfo& device) const { | 
					
						
							|  |  |  |   for (const auto& exclusion_filter : exclusion_filters_) { | 
					
						
							|  |  |  |     if (FilterMatch(exclusion_filter, device)) | 
					
						
							|  |  |  |       return true; | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-19 01:19:51 +02:00
										 |  |  | void HidChooserController::AddMessageToConsole( | 
					
						
							|  |  |  |     blink::mojom::ConsoleMessageLevel level, | 
					
						
							|  |  |  |     const std::string& message) const { | 
					
						
							|  |  |  |   if (content::RenderFrameHost* rfh = | 
					
						
							|  |  |  |           initiator_document_.AsRenderFrameHostIfValid()) { | 
					
						
							|  |  |  |     rfh->AddMessageToConsole(level, message); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | bool HidChooserController::AddDeviceInfo( | 
					
						
							|  |  |  |     const device::mojom::HidDeviceInfo& device) { | 
					
						
							| 
									
										
										
										
											2023-07-19 09:54:30 -05:00
										 |  |  |   const auto& id = PhysicalDeviceIdFromDeviceInfo(device); | 
					
						
							|  |  |  |   auto [iter, is_new_physical_device] = device_map_.try_emplace(id); | 
					
						
							|  |  |  |   iter->second.emplace_back(device.Clone()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // append new devices to the chooser list
 | 
					
						
							|  |  |  |   if (is_new_physical_device) | 
					
						
							|  |  |  |     items_.emplace_back(id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return is_new_physical_device; | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool HidChooserController::RemoveDeviceInfo( | 
					
						
							|  |  |  |     const device::mojom::HidDeviceInfo& device) { | 
					
						
							| 
									
										
										
										
											2023-07-19 09:54:30 -05:00
										 |  |  |   const auto& id = PhysicalDeviceIdFromDeviceInfo(device); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   auto find_it = device_map_.find(id); | 
					
						
							|  |  |  |   DCHECK(find_it != device_map_.end()); | 
					
						
							|  |  |  |   auto& device_infos = find_it->second; | 
					
						
							| 
									
										
										
										
											2024-03-12 10:15:41 +01:00
										 |  |  |   std::erase_if(device_infos, | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |                 [&device](const device::mojom::HidDeviceInfoPtr& d) { | 
					
						
							|  |  |  |                   return d->guid == device.guid; | 
					
						
							|  |  |  |                 }); | 
					
						
							|  |  |  |   if (!device_infos.empty()) | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   // A device was disconnected. Remove it from the chooser list.
 | 
					
						
							|  |  |  |   device_map_.erase(find_it); | 
					
						
							| 
									
										
										
										
											2024-03-12 10:15:41 +01:00
										 |  |  |   std::erase(items_, id); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserController::UpdateDeviceInfo( | 
					
						
							|  |  |  |     const device::mojom::HidDeviceInfo& device) { | 
					
						
							| 
									
										
										
										
											2023-07-19 09:54:30 -05:00
										 |  |  |   const auto& id = PhysicalDeviceIdFromDeviceInfo(device); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   auto physical_device_it = device_map_.find(id); | 
					
						
							|  |  |  |   DCHECK(physical_device_it != device_map_.end()); | 
					
						
							|  |  |  |   auto& device_infos = physical_device_it->second; | 
					
						
							| 
									
										
										
										
											2024-08-02 21:21:59 -05:00
										 |  |  |   auto device_it = std::ranges::find(device_infos, device.guid, | 
					
						
							|  |  |  |                                      &device::mojom::HidDeviceInfo::guid); | 
					
						
							| 
									
										
										
										
											2021-09-23 07:00:11 -04:00
										 |  |  |   DCHECK(device_it != device_infos.end()); | 
					
						
							|  |  |  |   *device_it = device.Clone(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserController::RunCallback( | 
					
						
							|  |  |  |     std::vector<device::mojom::HidDeviceInfoPtr> devices) { | 
					
						
							|  |  |  |   if (callback_) { | 
					
						
							|  |  |  |     std::move(callback_).Run(std::move(devices)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HidChooserController::RenderFrameDeleted( | 
					
						
							|  |  |  |     content::RenderFrameHost* render_frame_host) { | 
					
						
							|  |  |  |   if (hid_delegate_) { | 
					
						
							|  |  |  |     hid_delegate_->DeleteControllerForFrame(render_frame_host); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace electron
 |