/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at https://mozilla.org/MPL/2.0/. */ use std::cell::Cell; use std::convert::TryInto; use dom_struct::dom_struct; use js::jsval::JSVal; use lazy_static::lazy_static; use crate::dom::bindings::codegen::Bindings::NavigatorBinding::NavigatorMethods; use crate::dom::bindings::codegen::Bindings::WindowBinding::Window_Binding::WindowMethods; use crate::dom::bindings::reflector::{reflect_dom_object, DomObject, Reflector}; use crate::dom::bindings::root::{DomRoot, MutNullableDom}; use crate::dom::bindings::str::DOMString; use crate::dom::bindings::utils::to_frozen_array; use crate::dom::bluetooth::Bluetooth; use crate::dom::gamepad::Gamepad; use crate::dom::gamepadlist::GamepadList; use crate::dom::gpu::GPU; use crate::dom::mediadevices::MediaDevices; use crate::dom::mediasession::MediaSession; use crate::dom::mimetypearray::MimeTypeArray; use crate::dom::navigatorinfo; use crate::dom::permissions::Permissions; use crate::dom::pluginarray::PluginArray; use crate::dom::serviceworkercontainer::ServiceWorkerContainer; use crate::dom::window::Window; use crate::dom::xrsystem::XRSystem; use crate::script_runtime::JSContext; pub(super) fn hardware_concurrency() -> u64 { lazy_static! { static ref CPUS: u64 = num_cpus::get().try_into().unwrap_or(1); } *CPUS } #[dom_struct] pub struct Navigator { reflector_: Reflector, bluetooth: MutNullableDom, plugins: MutNullableDom, mime_types: MutNullableDom, service_worker: MutNullableDom, xr: MutNullableDom, mediadevices: MutNullableDom, /// gamepads: MutNullableDom, permissions: MutNullableDom, mediasession: MutNullableDom, gpu: MutNullableDom, /// has_gamepad_gesture: Cell, } impl Navigator { fn new_inherited() -> Navigator { Navigator { reflector_: Reflector::new(), bluetooth: Default::default(), plugins: Default::default(), mime_types: Default::default(), service_worker: Default::default(), xr: Default::default(), mediadevices: Default::default(), gamepads: Default::default(), permissions: Default::default(), mediasession: Default::default(), gpu: Default::default(), has_gamepad_gesture: Cell::new(false), } } pub fn new(window: &Window) -> DomRoot { reflect_dom_object(Box::new(Navigator::new_inherited()), window) } pub fn xr(&self) -> Option> { self.xr.get() } pub fn gamepads(&self) -> DomRoot { self.gamepads .or_init(|| GamepadList::new(&self.global(), &[])) } pub fn has_gamepad_gesture(&self) -> bool { self.has_gamepad_gesture.get() } pub fn set_has_gamepad_gesture(&self, has_gamepad_gesture: bool) { self.has_gamepad_gesture.set(has_gamepad_gesture); } } impl NavigatorMethods for Navigator { // https://html.spec.whatwg.org/multipage/#dom-navigator-product fn Product(&self) -> DOMString { navigatorinfo::Product() } // https://html.spec.whatwg.org/multipage/#dom-navigator-productsub fn ProductSub(&self) -> DOMString { navigatorinfo::ProductSub() } // https://html.spec.whatwg.org/multipage/#dom-navigator-vendor fn Vendor(&self) -> DOMString { navigatorinfo::Vendor() } // https://html.spec.whatwg.org/multipage/#dom-navigator-vendorsub fn VendorSub(&self) -> DOMString { navigatorinfo::VendorSub() } // https://html.spec.whatwg.org/multipage/#dom-navigator-taintenabled fn TaintEnabled(&self) -> bool { navigatorinfo::TaintEnabled() } // https://html.spec.whatwg.org/multipage/#dom-navigator-appname fn AppName(&self) -> DOMString { navigatorinfo::AppName() } // https://html.spec.whatwg.org/multipage/#dom-navigator-appcodename fn AppCodeName(&self) -> DOMString { navigatorinfo::AppCodeName() } // https://html.spec.whatwg.org/multipage/#dom-navigator-platform fn Platform(&self) -> DOMString { navigatorinfo::Platform() } // https://html.spec.whatwg.org/multipage/#dom-navigator-useragent fn UserAgent(&self) -> DOMString { navigatorinfo::UserAgent(self.global().get_user_agent()) } // https://html.spec.whatwg.org/multipage/#dom-navigator-appversion fn AppVersion(&self) -> DOMString { navigatorinfo::AppVersion() } // https://webbluetoothcg.github.io/web-bluetooth/#dom-navigator-bluetooth fn Bluetooth(&self) -> DomRoot { self.bluetooth.or_init(|| Bluetooth::new(&self.global())) } // https://html.spec.whatwg.org/multipage/#navigatorlanguage fn Language(&self) -> DOMString { navigatorinfo::Language() } // https://html.spec.whatwg.org/multipage/#dom-navigator-languages #[allow(unsafe_code)] fn Languages(&self, cx: JSContext) -> JSVal { to_frozen_array(&[self.Language()], cx) } // https://html.spec.whatwg.org/multipage/#dom-navigator-plugins fn Plugins(&self) -> DomRoot { self.plugins.or_init(|| PluginArray::new(&self.global())) } // https://html.spec.whatwg.org/multipage/#dom-navigator-mimetypes fn MimeTypes(&self) -> DomRoot { self.mime_types .or_init(|| MimeTypeArray::new(&self.global())) } // https://html.spec.whatwg.org/multipage/#dom-navigator-javaenabled fn JavaEnabled(&self) -> bool { false } // https://w3c.github.io/ServiceWorker/#navigator-service-worker-attribute fn ServiceWorker(&self) -> DomRoot { self.service_worker .or_init(|| ServiceWorkerContainer::new(&self.global())) } // https://html.spec.whatwg.org/multipage/#dom-navigator-cookieenabled fn CookieEnabled(&self) -> bool { true } /// fn GetGamepads(&self) -> Vec>> { let global = self.global(); let window = global.as_window(); let doc = window.Document(); // TODO: Handle permissions policy once implemented if !doc.is_fully_active() || !self.has_gamepad_gesture.get() { return Vec::new(); } let root = self.gamepads.or_init(|| GamepadList::new(&global, &[])); root.list() } // https://w3c.github.io/permissions/#navigator-and-workernavigator-extension fn Permissions(&self) -> DomRoot { self.permissions .or_init(|| Permissions::new(&self.global())) } /// fn Xr(&self) -> DomRoot { self.xr.or_init(|| XRSystem::new(self.global().as_window())) } /// fn MediaDevices(&self) -> DomRoot { self.mediadevices .or_init(|| MediaDevices::new(&self.global())) } /// fn MediaSession(&self) -> DomRoot { self.mediasession.or_init(|| { // There is a single MediaSession instance per Pipeline // and only one active MediaSession globally. // // MediaSession creation can happen in two cases: // // - If content gets `navigator.mediaSession` // - If a media instance (HTMLMediaElement so far) starts playing media. let global = self.global(); let window = global.as_window(); MediaSession::new(window) }) } // https://gpuweb.github.io/gpuweb/#dom-navigator-gpu fn Gpu(&self) -> DomRoot { self.gpu.or_init(|| GPU::new(&self.global())) } /// fn HardwareConcurrency(&self) -> u64 { hardware_concurrency() } }