| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  | // Copyright (c) 2022 Microsoft, Inc.
 | 
					
						
							|  |  |  | // Use of this source code is governed by the MIT license that can be
 | 
					
						
							|  |  |  | // found in the LICENSE file.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "shell/browser/usb/usb_chooser_controller.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-02 21:21:59 -05:00
										 |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <cstddef>
 | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  | #include <utility>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-03 12:43:42 +01:00
										 |  |  | #include "base/functional/bind.h"
 | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  | #include "components/strings/grit/components_strings.h"
 | 
					
						
							|  |  |  | #include "content/public/browser/render_frame_host.h"
 | 
					
						
							|  |  |  | #include "content/public/browser/web_contents.h"
 | 
					
						
							|  |  |  | #include "gin/data_object_builder.h"
 | 
					
						
							|  |  |  | #include "services/device/public/cpp/usb/usb_utils.h"
 | 
					
						
							|  |  |  | #include "services/device/public/mojom/usb_enumeration_options.mojom.h"
 | 
					
						
							| 
									
										
										
										
											2024-07-25 04:25:45 -05:00
										 |  |  | #include "shell/browser/api/electron_api_session.h"
 | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  | #include "shell/browser/javascript_environment.h"
 | 
					
						
							| 
									
										
										
										
											2024-07-29 12:42:57 -05:00
										 |  |  | #include "shell/browser/usb/electron_usb_delegate.h"
 | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  | #include "shell/browser/usb/usb_chooser_context_factory.h"
 | 
					
						
							|  |  |  | #include "shell/common/gin_converters/callback_converter.h"
 | 
					
						
							|  |  |  | #include "shell/common/gin_converters/content_converter.h"
 | 
					
						
							|  |  |  | #include "shell/common/gin_converters/frame_converter.h"
 | 
					
						
							|  |  |  | #include "shell/common/gin_converters/usb_device_info_converter.h"
 | 
					
						
							| 
									
										
										
										
											2024-09-16 15:53:04 -05:00
										 |  |  | #include "shell/common/node_util.h"
 | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  | #include "ui/base/l10n/l10n_util.h"
 | 
					
						
							|  |  |  | #include "url/gurl.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | using content::RenderFrameHost; | 
					
						
							|  |  |  | using content::WebContents; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace electron { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | UsbChooserController::UsbChooserController( | 
					
						
							|  |  |  |     RenderFrameHost* render_frame_host, | 
					
						
							| 
									
										
										
										
											2023-07-01 16:22:55 -04:00
										 |  |  |     blink::mojom::WebUsbRequestDeviceOptionsPtr options, | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  |     blink::mojom::WebUsbService::GetPermissionCallback callback, | 
					
						
							|  |  |  |     content::WebContents* web_contents, | 
					
						
							|  |  |  |     base::WeakPtr<ElectronUsbDelegate> usb_delegate) | 
					
						
							|  |  |  |     : WebContentsObserver(web_contents), | 
					
						
							| 
									
										
										
										
											2023-07-01 16:22:55 -04:00
										 |  |  |       options_(std::move(options)), | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  |       callback_(std::move(callback)), | 
					
						
							|  |  |  |       origin_(render_frame_host->GetMainFrame()->GetLastCommittedOrigin()), | 
					
						
							|  |  |  |       usb_delegate_(usb_delegate), | 
					
						
							|  |  |  |       render_frame_host_id_(render_frame_host->GetGlobalId()) { | 
					
						
							|  |  |  |   chooser_context_ = UsbChooserContextFactory::GetForBrowserContext( | 
					
						
							|  |  |  |                          web_contents->GetBrowserContext()) | 
					
						
							|  |  |  |                          ->AsWeakPtr(); | 
					
						
							|  |  |  |   DCHECK(chooser_context_); | 
					
						
							|  |  |  |   chooser_context_->GetDevices(base::BindOnce( | 
					
						
							|  |  |  |       &UsbChooserController::GotUsbDeviceList, weak_factory_.GetWeakPtr())); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | UsbChooserController::~UsbChooserController() { | 
					
						
							| 
									
										
										
										
											2023-04-26 07:09:54 -07:00
										 |  |  |   RunCallback(/*device_info=*/nullptr); | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | api::Session* UsbChooserController::GetSession() { | 
					
						
							|  |  |  |   if (!web_contents()) { | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return api::Session::FromBrowserContext(web_contents()->GetBrowserContext()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UsbChooserController::OnDeviceAdded( | 
					
						
							|  |  |  |     const device::mojom::UsbDeviceInfo& device_info) { | 
					
						
							|  |  |  |   if (DisplayDevice(device_info)) { | 
					
						
							|  |  |  |     api::Session* session = GetSession(); | 
					
						
							|  |  |  |     if (session) { | 
					
						
							|  |  |  |       session->Emit("usb-device-added", device_info.Clone(), web_contents()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UsbChooserController::OnDeviceRemoved( | 
					
						
							|  |  |  |     const device::mojom::UsbDeviceInfo& device_info) { | 
					
						
							|  |  |  |   api::Session* session = GetSession(); | 
					
						
							|  |  |  |   if (session) { | 
					
						
							|  |  |  |     session->Emit("usb-device-removed", device_info.Clone(), web_contents()); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UsbChooserController::OnDeviceChosen(gin::Arguments* args) { | 
					
						
							|  |  |  |   std::string device_id; | 
					
						
							|  |  |  |   if (!args->GetNext(&device_id) || device_id.empty()) { | 
					
						
							| 
									
										
										
										
											2023-04-26 07:09:54 -07:00
										 |  |  |     RunCallback(/*device_info=*/nullptr); | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  |   } else { | 
					
						
							|  |  |  |     auto* device_info = chooser_context_->GetDeviceInfo(device_id); | 
					
						
							|  |  |  |     if (device_info) { | 
					
						
							|  |  |  |       RunCallback(device_info->Clone()); | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2024-09-16 15:53:04 -05:00
										 |  |  |       util::EmitWarning( | 
					
						
							|  |  |  |           base::StrCat({"The device id ", device_id, " was not found."}), | 
					
						
							|  |  |  |           "UnknownUsbDeviceId"); | 
					
						
							| 
									
										
										
										
											2023-04-26 07:09:54 -07:00
										 |  |  |       RunCallback(/*device_info=*/nullptr); | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UsbChooserController::OnBrowserContextShutdown() { | 
					
						
							|  |  |  |   observation_.Reset(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Get a list of devices that can be shown in the chooser bubble UI for
 | 
					
						
							| 
									
										
										
										
											2023-11-12 19:51:56 -08:00
										 |  |  | // user to grant permission.
 | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  | void UsbChooserController::GotUsbDeviceList( | 
					
						
							|  |  |  |     std::vector<::device::mojom::UsbDeviceInfoPtr> devices) { | 
					
						
							|  |  |  |   // Listen to UsbChooserContext for OnDeviceAdded/Removed events after the
 | 
					
						
							|  |  |  |   // enumeration.
 | 
					
						
							|  |  |  |   if (chooser_context_) | 
					
						
							|  |  |  |     observation_.Observe(chooser_context_.get()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bool prevent_default = false; | 
					
						
							|  |  |  |   api::Session* session = GetSession(); | 
					
						
							|  |  |  |   if (session) { | 
					
						
							|  |  |  |     auto* rfh = content::RenderFrameHost::FromID(render_frame_host_id_); | 
					
						
							|  |  |  |     v8::Isolate* isolate = JavascriptEnvironment::GetIsolate(); | 
					
						
							| 
									
										
										
										
											2024-09-16 15:53:04 -05:00
										 |  |  |     v8::HandleScope handle_scope{isolate}; | 
					
						
							| 
									
										
										
										
											2024-01-31 15:53:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // "select-usb-device" should respect |filters|.
 | 
					
						
							| 
									
										
										
										
											2024-04-19 10:55:59 -05:00
										 |  |  |     std::erase_if(devices, [this](const auto& device_info) { | 
					
						
							|  |  |  |       return !DisplayDevice(*device_info); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2024-01-31 15:53:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  |     v8::Local<v8::Object> details = gin::DataObjectBuilder(isolate) | 
					
						
							|  |  |  |                                         .Set("deviceList", devices) | 
					
						
							|  |  |  |                                         .Set("frame", rfh) | 
					
						
							|  |  |  |                                         .Build(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     prevent_default = | 
					
						
							|  |  |  |         session->Emit("select-usb-device", details, | 
					
						
							| 
									
										
										
										
											2023-05-18 23:01:44 +02:00
										 |  |  |                       base::BindRepeating(&UsbChooserController::OnDeviceChosen, | 
					
						
							|  |  |  |                                           weak_factory_.GetWeakPtr())); | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (!prevent_default) { | 
					
						
							| 
									
										
										
										
											2023-04-26 07:09:54 -07:00
										 |  |  |     RunCallback(/*device_info=*/nullptr); | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool UsbChooserController::DisplayDevice( | 
					
						
							|  |  |  |     const device::mojom::UsbDeviceInfo& device_info) const { | 
					
						
							| 
									
										
										
										
											2023-07-01 16:22:55 -04:00
										 |  |  |   if (!device::UsbDeviceFilterMatchesAny(options_->filters, device_info)) { | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2023-07-01 16:22:55 -04:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-02 21:21:59 -05:00
										 |  |  |   if (std::ranges::any_of( | 
					
						
							| 
									
										
										
										
											2023-07-01 16:22:55 -04:00
										 |  |  |           options_->exclusion_filters, [&device_info](const auto& filter) { | 
					
						
							|  |  |  |             return device::UsbDeviceFilterMatches(*filter, device_info); | 
					
						
							|  |  |  |           })) { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-11-22 16:50:32 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UsbChooserController::RenderFrameDeleted( | 
					
						
							|  |  |  |     content::RenderFrameHost* render_frame_host) { | 
					
						
							|  |  |  |   if (usb_delegate_) { | 
					
						
							|  |  |  |     usb_delegate_->DeleteControllerForFrame(render_frame_host); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UsbChooserController::RunCallback( | 
					
						
							|  |  |  |     device::mojom::UsbDeviceInfoPtr device_info) { | 
					
						
							|  |  |  |   if (callback_) { | 
					
						
							|  |  |  |     if (!chooser_context_ || !device_info) { | 
					
						
							|  |  |  |       std::move(callback_).Run(nullptr); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       chooser_context_->GrantDevicePermission(origin_, *device_info); | 
					
						
							|  |  |  |       std::move(callback_).Run(std::move(device_info)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace electron
 |