/* 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::borrow::ToOwned; use std::cell::{Cell, RefCell, RefMut}; use std::cmp; use std::collections::hash_map::Entry; use std::collections::{HashMap, HashSet}; use std::default::Default; use std::io::{Write, stderr, stdout}; use std::rc::Rc; use std::sync::{Arc, Mutex}; use std::time::{Duration, Instant}; use app_units::Au; use backtrace::Backtrace; use base::cross_process_instant::CrossProcessInstant; use base::id::{BrowsingContextId, PipelineId, WebViewId}; use base64::Engine; #[cfg(feature = "bluetooth")] use bluetooth_traits::BluetoothRequest; use canvas_traits::webgl::WebGLChan; use compositing_traits::CrossProcessCompositorApi; use constellation_traits::{ DocumentState, LoadData, LoadOrigin, NavigationHistoryBehavior, ScriptToConstellationChan, ScriptToConstellationMessage, StructuredSerializedData, WindowSizeType, }; use crossbeam_channel::{Sender, unbounded}; use cssparser::SourceLocation; use devtools_traits::{ScriptToDevtoolsControlMsg, TimelineMarker, TimelineMarkerType}; use dom_struct::dom_struct; use embedder_traits::user_content_manager::{UserContentManager, UserScript}; use embedder_traits::{ AlertResponse, ConfirmResponse, EmbedderMsg, GamepadEvent, GamepadSupportedHapticEffects, GamepadUpdateType, PromptResponse, SimpleDialog, Theme, ViewportDetails, WebDriverJSError, WebDriverJSResult, }; use euclid::default::{Point2D as UntypedPoint2D, Rect as UntypedRect, Size2D as UntypedSize2D}; use euclid::{Point2D, Scale, Size2D, Vector2D}; use fonts::FontContext; use ipc_channel::ipc::{self, IpcSender}; use js::conversions::ToJSValConvertible; use js::glue::DumpJSStack; use js::jsapi::{ GCReason, Heap, JS_GC, JSAutoRealm, JSContext as RawJSContext, JSObject, JSPROP_ENUMERATE, }; use js::jsval::{NullValue, UndefinedValue}; use js::rust::wrappers::JS_DefineProperty; use js::rust::{ CustomAutoRooter, CustomAutoRooterGuard, HandleObject, HandleValue, MutableHandleObject, MutableHandleValue, }; use layout_api::{ FragmentType, Layout, PendingImageState, QueryMsg, ReflowGoal, ReflowRequest, TrustedNodeAddress, combine_id_with_fragment_type, }; use malloc_size_of::MallocSizeOf; use media::WindowGLContext; use net_traits::ResourceThreads; use net_traits::image_cache::{ ImageCache, ImageCacheResponseMessage, ImageLoadListener, ImageResponse, PendingImageId, PendingImageResponse, RasterizationCompleteResponse, }; use net_traits::storage_thread::StorageType; use num_traits::ToPrimitive; use profile_traits::ipc as ProfiledIpc; use profile_traits::mem::ProfilerChan as MemProfilerChan; use profile_traits::time::ProfilerChan as TimeProfilerChan; use script_bindings::codegen::GenericBindings::NavigatorBinding::NavigatorMethods; use script_bindings::codegen::GenericBindings::PerformanceBinding::PerformanceMethods; use script_bindings::interfaces::WindowHelpers; use script_bindings::root::Root; use script_traits::ScriptThreadMessage; use selectors::attr::CaseSensitivity; use servo_arc::Arc as ServoArc; use servo_config::{opts, pref}; use servo_geometry::{DeviceIndependentIntRect, f32_rect_to_au_rect}; use servo_url::{ImmutableOrigin, MutableOrigin, ServoUrl}; use style::error_reporting::{ContextualParseError, ParseErrorReporter}; use style::properties::PropertyId; use style::properties::style_structs::Font; use style::selector_parser::PseudoElement; use style::str::HTML_SPACE_CHARACTERS; use style::stylesheets::UrlExtraData; use style_traits::CSSPixel; use stylo_atoms::Atom; use url::Position; use webrender_api::ExternalScrollId; use webrender_api::units::{DeviceIntSize, DevicePixel, LayoutPixel}; use super::bindings::codegen::Bindings::MessagePortBinding::StructuredSerializeOptions; use super::bindings::trace::HashMapTracedValues; use crate::dom::bindings::cell::{DomRefCell, Ref}; use crate::dom::bindings::codegen::Bindings::DocumentBinding::{ DocumentMethods, DocumentReadyState, NamedPropertyValue, }; use crate::dom::bindings::codegen::Bindings::HTMLIFrameElementBinding::HTMLIFrameElementMethods; use crate::dom::bindings::codegen::Bindings::HistoryBinding::History_Binding::HistoryMethods; use crate::dom::bindings::codegen::Bindings::ImageBitmapBinding::{ ImageBitmapOptions, ImageBitmapSource, }; use crate::dom::bindings::codegen::Bindings::MediaQueryListBinding::MediaQueryList_Binding::MediaQueryListMethods; use crate::dom::bindings::codegen::Bindings::RequestBinding::RequestInit; use crate::dom::bindings::codegen::Bindings::VoidFunctionBinding::VoidFunction; use crate::dom::bindings::codegen::Bindings::WindowBinding::{ self, FrameRequestCallback, ScrollBehavior, ScrollToOptions, WindowMethods, WindowPostMessageOptions, }; use crate::dom::bindings::codegen::UnionTypes::{RequestOrUSVString, StringOrFunction}; use crate::dom::bindings::error::{Error, ErrorResult, Fallible}; use crate::dom::bindings::inheritance::{Castable, ElementTypeId, HTMLElementTypeId, NodeTypeId}; use crate::dom::bindings::num::Finite; use crate::dom::bindings::refcounted::Trusted; use crate::dom::bindings::reflector::{DomGlobal, DomObject}; use crate::dom::bindings::root::{Dom, DomRoot, MutNullableDom}; use crate::dom::bindings::str::{DOMString, USVString}; use crate::dom::bindings::structuredclone; use crate::dom::bindings::trace::{CustomTraceable, JSTraceable, RootedTraceableBox}; use crate::dom::bindings::utils::GlobalStaticData; use crate::dom::bindings::weakref::DOMTracker; #[cfg(feature = "bluetooth")] use crate::dom::bluetooth::BluetoothExtraPermissionData; use crate::dom::crypto::Crypto; use crate::dom::cssstyledeclaration::{CSSModificationAccess, CSSStyleDeclaration, CSSStyleOwner}; use crate::dom::customelementregistry::CustomElementRegistry; use crate::dom::document::{AnimationFrameCallback, Document, ReflowTriggerCondition}; use crate::dom::element::Element; use crate::dom::event::{Event, EventBubbles, EventCancelable, EventStatus}; use crate::dom::eventtarget::EventTarget; use crate::dom::gamepad::{Gamepad, contains_user_gesture}; use crate::dom::gamepadevent::GamepadEventType; use crate::dom::globalscope::GlobalScope; use crate::dom::hashchangeevent::HashChangeEvent; use crate::dom::history::History; use crate::dom::htmlcollection::{CollectionFilter, HTMLCollection}; use crate::dom::htmliframeelement::HTMLIFrameElement; use crate::dom::idbfactory::IDBFactory; use crate::dom::location::Location; use crate::dom::medialist::MediaList; use crate::dom::mediaquerylist::{MediaQueryList, MediaQueryListMatchState}; use crate::dom::mediaquerylistevent::MediaQueryListEvent; use crate::dom::messageevent::MessageEvent; use crate::dom::navigator::Navigator; use crate::dom::node::{Node, NodeDamage, NodeTraits, from_untrusted_node_address}; use crate::dom::performance::Performance; use crate::dom::promise::Promise; use crate::dom::screen::Screen; use crate::dom::selection::Selection; use crate::dom::shadowroot::ShadowRoot; use crate::dom::storage::Storage; #[cfg(feature = "bluetooth")] use crate::dom::testrunner::TestRunner; use crate::dom::trustedtypepolicyfactory::TrustedTypePolicyFactory; use crate::dom::types::{ImageBitmap, UIEvent}; use crate::dom::webglrenderingcontext::WebGLCommandSender; #[cfg(feature = "webgpu")] use crate::dom::webgpu::identityhub::IdentityHub; use crate::dom::windowproxy::{WindowProxy, WindowProxyHandler}; use crate::dom::worklet::Worklet; use crate::dom::workletglobalscope::WorkletGlobalScopeType; use crate::layout_image::fetch_image_for_layout; use crate::messaging::{MainThreadScriptMsg, ScriptEventLoopReceiver, ScriptEventLoopSender}; use crate::microtask::MicrotaskQueue; use crate::realms::{InRealm, enter_realm}; use crate::script_runtime::{CanGc, JSContext, Runtime}; use crate::script_thread::ScriptThread; use crate::timers::{IsInterval, TimerCallback}; use crate::unminify::unminified_path; use crate::webdriver_handlers::{find_node_by_unique_id_in_document, jsval_to_webdriver}; use crate::{fetch, window_named_properties}; /// A callback to call when a response comes back from the `ImageCache`. /// /// This is wrapped in a struct so that we can implement `MallocSizeOf` /// for this type. #[derive(MallocSizeOf)] pub struct PendingImageCallback( #[ignore_malloc_size_of = "dyn Fn is currently impossible to measure"] Box<dyn Fn(PendingImageResponse) + 'static>, ); /// Current state of the window object #[derive(Clone, Copy, Debug, JSTraceable, MallocSizeOf, PartialEq)] enum WindowState { Alive, Zombie, // Pipeline is closed, but the window hasn't been GCed yet. } /// How long we should wait before performing the initial reflow after `<body>` is parsed, /// assuming that `<body>` take this long to parse. const INITIAL_REFLOW_DELAY: Duration = Duration::from_millis(200); /// During loading and parsing, layouts are suppressed to avoid flashing incomplete page /// contents. /// /// Exceptions: /// - Parsing the body takes so long, that layouts are no longer suppressed in order /// to show the user that the page is loading. /// - Script triggers a layout query or scroll event in which case, we want to layout /// but not display the contents. /// /// For more information see: <https://github.com/servo/servo/pull/6028>. #[derive(Clone, Copy, MallocSizeOf)] enum LayoutBlocker { /// The first load event hasn't been fired and we have not started to parse the `<body>` yet. WaitingForParse, /// The body is being parsed the `<body>` starting at the `Instant` specified. Parsing(Instant), /// The body finished parsing and the `load` event has been fired or parsing took so /// long, that we are going to do layout anyway. Note that subsequent changes to the body /// can trigger parsing again, but the `Window` stays in this state. FiredLoadEventOrParsingTimerExpired, } impl LayoutBlocker { fn layout_blocked(&self) -> bool { !matches!(self, Self::FiredLoadEventOrParsingTimerExpired) } } type PendingImageRasterizationKey = (PendingImageId, DeviceIntSize); #[dom_struct] pub(crate) struct Window { globalscope: GlobalScope, /// The webview that contains this [`Window`]. /// /// This may not be the top-level [`Window`], in the case of frames. #[no_trace] webview_id: WebViewId, script_chan: Sender<MainThreadScriptMsg>, #[no_trace] #[ignore_malloc_size_of = "TODO: Add MallocSizeOf support to layout"] layout: RefCell<Box<dyn Layout>>, /// A [`FontContext`] which is used to store and match against fonts for this `Window` and to /// trigger the download of web fonts. #[no_trace] #[conditional_malloc_size_of] font_context: Arc<FontContext>, navigator: MutNullableDom<Navigator>, #[ignore_malloc_size_of = "Arc"] #[no_trace] image_cache: Arc<dyn ImageCache>, #[no_trace] image_cache_sender: IpcSender<ImageCacheResponseMessage>, window_proxy: MutNullableDom<WindowProxy>, document: MutNullableDom<Document>, location: MutNullableDom<Location>, history: MutNullableDom<History>, indexeddb: MutNullableDom<IDBFactory>, custom_element_registry: MutNullableDom<CustomElementRegistry>, performance: MutNullableDom<Performance>, #[no_trace] navigation_start: Cell<CrossProcessInstant>, screen: MutNullableDom<Screen>, session_storage: MutNullableDom<Storage>, local_storage: MutNullableDom<Storage>, status: DomRefCell<DOMString>, trusted_types: MutNullableDom<TrustedTypePolicyFactory>, /// For sending timeline markers. Will be ignored if /// no devtools server #[no_trace] devtools_markers: DomRefCell<HashSet<TimelineMarkerType>>, #[no_trace] devtools_marker_sender: DomRefCell<Option<IpcSender<Option<TimelineMarker>>>>, /// Most recent unhandled resize event, if any. #[no_trace] unhandled_resize_event: DomRefCell<Option<(ViewportDetails, WindowSizeType)>>, /// Platform theme. #[no_trace] theme: Cell<Theme>, /// Parent id associated with this page, if any. #[no_trace] parent_info: Option<PipelineId>, /// Global static data related to the DOM. dom_static: GlobalStaticData, /// The JavaScript runtime. #[ignore_malloc_size_of = "Rc<T> is hard"] js_runtime: DomRefCell<Option<Rc<Runtime>>>, /// The [`ViewportDetails`] of this [`Window`]'s frame. #[no_trace] viewport_details: Cell<ViewportDetails>, /// A handle for communicating messages to the bluetooth thread. #[no_trace] #[cfg(feature = "bluetooth")] bluetooth_thread: IpcSender<BluetoothRequest>, #[cfg(feature = "bluetooth")] bluetooth_extra_permission_data: BluetoothExtraPermissionData, /// See the documentation for [`LayoutBlocker`]. Essentially, this flag prevents /// layouts from happening before the first load event, apart from a few exceptional /// cases. #[no_trace] layout_blocker: Cell<LayoutBlocker>, /// A channel for communicating results of async scripts back to the webdriver server #[no_trace] webdriver_script_chan: DomRefCell<Option<IpcSender<WebDriverJSResult>>>, /// The current state of the window object current_state: Cell<WindowState>, /// The current size of the viewport. This might change if the `WebView` or containing `<iframe>` /// for this `Window` object change. #[no_trace] current_viewport_size: Cell<UntypedSize2D<Au>>, error_reporter: CSSErrorReporter, /// All the MediaQueryLists we need to update media_query_lists: DOMTracker<MediaQueryList>, #[cfg(feature = "bluetooth")] test_runner: MutNullableDom<TestRunner>, /// A handle for communicating messages to the WebGL thread, if available. #[ignore_malloc_size_of = "channels are hard"] #[no_trace] webgl_chan: Option<WebGLChan>, #[ignore_malloc_size_of = "defined in webxr"] #[no_trace] #[cfg(feature = "webxr")] webxr_registry: Option<webxr_api::Registry>, /// When an element triggers an image load or starts watching an image load from the /// `ImageCache` it adds an entry to this list. When those loads are triggered from /// layout, they also add an etry to [`Self::pending_layout_images`]. #[no_trace] pending_image_callbacks: DomRefCell<HashMap<PendingImageId, Vec<PendingImageCallback>>>, /// All of the elements that have an outstanding image request that was /// initiated by layout during a reflow. They are stored in the [`ScriptThread`] /// to ensure that the element can be marked dirty when the image data becomes /// available at some point in the future. pending_layout_images: DomRefCell<HashMapTracedValues<PendingImageId, Vec<Dom<Node>>>>, /// Vector images for which layout has intiated rasterization at a specific size /// and whose results are not yet available. They are stored in the [`ScriptThread`] /// so that the element can be marked dirty once the rasterization is completed. pending_images_for_rasterization: DomRefCell<HashMapTracedValues<PendingImageRasterizationKey, Vec<Dom<Node>>>>, /// Directory to store unminified css for this window if unminify-css /// opt is enabled. unminified_css_dir: DomRefCell<Option<String>>, /// Directory with stored unminified scripts local_script_source: Option<String>, /// Worklets test_worklet: MutNullableDom<Worklet>, /// <https://drafts.css-houdini.org/css-paint-api-1/#paint-worklet> paint_worklet: MutNullableDom<Worklet>, /// Flag to identify whether mutation observers are present(true)/absent(false) exists_mut_observer: Cell<bool>, /// Cross-process access to the compositor. #[ignore_malloc_size_of = "Wraps an IpcSender"] #[no_trace] compositor_api: CrossProcessCompositorApi, /// Indicate whether a SetDocumentStatus message has been sent after a reflow is complete. /// It is used to avoid sending idle message more than once, which is unneccessary. has_sent_idle_message: Cell<bool>, /// Emits notifications when there is a relayout. relayout_event: bool, /// Unminify Css. unminify_css: bool, /// User content manager #[no_trace] user_content_manager: UserContentManager, /// Window's GL context from application #[ignore_malloc_size_of = "defined in script_thread"] #[no_trace] player_context: WindowGLContext, throttled: Cell<bool>, /// A shared marker for the validity of any cached layout values. A value of true /// indicates that any such values remain valid; any new layout that invalidates /// those values will cause the marker to be set to false. #[ignore_malloc_size_of = "Rc is hard"] layout_marker: DomRefCell<Rc<Cell<bool>>>, /// <https://dom.spec.whatwg.org/#window-current-event> current_event: DomRefCell<Option<Dom<Event>>>, } impl Window { pub(crate) fn webview_id(&self) -> WebViewId { self.webview_id } pub(crate) fn as_global_scope(&self) -> &GlobalScope { self.upcast::<GlobalScope>() } pub(crate) fn layout(&self) -> Ref<Box<dyn Layout>> { self.layout.borrow() } pub(crate) fn layout_mut(&self) -> RefMut<Box<dyn Layout>> { self.layout.borrow_mut() } pub(crate) fn get_exists_mut_observer(&self) -> bool { self.exists_mut_observer.get() } pub(crate) fn set_exists_mut_observer(&self) { self.exists_mut_observer.set(true); } #[allow(unsafe_code)] pub(crate) fn clear_js_runtime_for_script_deallocation(&self) { self.as_global_scope() .remove_web_messaging_and_dedicated_workers_infra(); unsafe { *self.js_runtime.borrow_for_script_deallocation() = None; self.window_proxy.set(None); self.current_state.set(WindowState::Zombie); self.as_global_scope() .task_manager() .cancel_all_tasks_and_ignore_future_tasks(); } } /// A convenience method for /// <https://html.spec.whatwg.org/multipage/#a-browsing-context-is-discarded> pub(crate) fn discard_browsing_context(&self) { let proxy = match self.window_proxy.get() { Some(proxy) => proxy, None => panic!("Discarding a BC from a window that has none"), }; proxy.discard_browsing_context(); // Step 4 of https://html.spec.whatwg.org/multipage/#discard-a-document // Other steps performed when the `PipelineExit` message // is handled by the ScriptThread. self.as_global_scope() .task_manager() .cancel_all_tasks_and_ignore_future_tasks(); } /// Get a sender to the time profiler thread. pub(crate) fn time_profiler_chan(&self) -> &TimeProfilerChan { self.globalscope.time_profiler_chan() } pub(crate) fn origin(&self) -> &MutableOrigin { self.globalscope.origin() } #[allow(unsafe_code)] pub(crate) fn get_cx(&self) -> JSContext { unsafe { JSContext::from_ptr(self.js_runtime.borrow().as_ref().unwrap().cx()) } } pub(crate) fn get_js_runtime(&self) -> Ref<Option<Rc<Runtime>>> { self.js_runtime.borrow() } pub(crate) fn main_thread_script_chan(&self) -> &Sender<MainThreadScriptMsg> { &self.script_chan } pub(crate) fn parent_info(&self) -> Option<PipelineId> { self.parent_info } pub(crate) fn new_script_pair(&self) -> (ScriptEventLoopSender, ScriptEventLoopReceiver) { let (sender, receiver) = unbounded(); ( ScriptEventLoopSender::MainThread(sender), ScriptEventLoopReceiver::MainThread(receiver), ) } pub(crate) fn event_loop_sender(&self) -> ScriptEventLoopSender { ScriptEventLoopSender::MainThread(self.script_chan.clone()) } pub(crate) fn image_cache(&self) -> Arc<dyn ImageCache> { self.image_cache.clone() } /// This can panic if it is called after the browsing context has been discarded pub(crate) fn window_proxy(&self) -> DomRoot<WindowProxy> { self.window_proxy.get().unwrap() } /// Returns the window proxy if it has not been discarded. /// <https://html.spec.whatwg.org/multipage/#a-browsing-context-is-discarded> pub(crate) fn undiscarded_window_proxy(&self) -> Option<DomRoot<WindowProxy>> { self.window_proxy.get().and_then(|window_proxy| { if window_proxy.is_browsing_context_discarded() { None } else { Some(window_proxy) } }) } /// Returns the window proxy of the webview, which is the top-level ancestor browsing context. /// <https://html.spec.whatwg.org/multipage/#top-level-browsing-context> pub(crate) fn webview_window_proxy(&self) -> Option<DomRoot<WindowProxy>> { self.undiscarded_window_proxy() .and_then(|window_proxy| ScriptThread::find_window_proxy(window_proxy.webview_id().0)) } #[cfg(feature = "bluetooth")] pub(crate) fn bluetooth_thread(&self) -> IpcSender<BluetoothRequest> { self.bluetooth_thread.clone() } #[cfg(feature = "bluetooth")] pub(crate) fn bluetooth_extra_permission_data(&self) -> &BluetoothExtraPermissionData { &self.bluetooth_extra_permission_data } pub(crate) fn css_error_reporter(&self) -> Option<&dyn ParseErrorReporter> { Some(&self.error_reporter) } pub(crate) fn webgl_chan(&self) -> Option<WebGLCommandSender> { self.webgl_chan .as_ref() .map(|chan| WebGLCommandSender::new(chan.clone())) } #[cfg(feature = "webxr")] pub(crate) fn webxr_registry(&self) -> Option<webxr_api::Registry> { self.webxr_registry.clone() } fn new_paint_worklet(&self, can_gc: CanGc) -> DomRoot<Worklet> { debug!("Creating new paint worklet."); Worklet::new(self, WorkletGlobalScopeType::Paint, can_gc) } pub(crate) fn register_image_cache_listener( &self, id: PendingImageId, callback: impl Fn(PendingImageResponse) + 'static, ) -> IpcSender<ImageCacheResponseMessage> { self.pending_image_callbacks .borrow_mut() .entry(id) .or_default() .push(PendingImageCallback(Box::new(callback))); self.image_cache_sender.clone() } fn pending_layout_image_notification(&self, response: PendingImageResponse) { let mut images = self.pending_layout_images.borrow_mut(); let nodes = images.entry(response.id); let nodes = match nodes { Entry::Occupied(nodes) => nodes, Entry::Vacant(_) => return, }; for node in nodes.get() { node.dirty(NodeDamage::Other); } match response.response { ImageResponse::MetadataLoaded(_) => {}, ImageResponse::Loaded(_, _) | ImageResponse::PlaceholderLoaded(_, _) | ImageResponse::None => { nodes.remove(); }, } } pub(crate) fn handle_image_rasterization_complete_notification( &self, response: RasterizationCompleteResponse, ) { let mut images = self.pending_images_for_rasterization.borrow_mut(); let nodes = images.entry((response.image_id, response.requested_size)); let nodes = match nodes { Entry::Occupied(nodes) => nodes, Entry::Vacant(_) => return, }; for node in nodes.get() { node.dirty(NodeDamage::Other); } nodes.remove(); } pub(crate) fn pending_image_notification(&self, response: PendingImageResponse) { // We take the images here, in order to prevent maintaining a mutable borrow when // image callbacks are called. These, in turn, can trigger garbage collection. // Normally this shouldn't trigger more pending image notifications, but just in // case we do not want to cause a double borrow here. let mut images = std::mem::take(&mut *self.pending_image_callbacks.borrow_mut()); let Entry::Occupied(callbacks) = images.entry(response.id) else { let _ = std::mem::replace(&mut *self.pending_image_callbacks.borrow_mut(), images); return; }; for callback in callbacks.get() { callback.0(response.clone()); } match response.response { ImageResponse::MetadataLoaded(_) => {}, ImageResponse::Loaded(_, _) | ImageResponse::PlaceholderLoaded(_, _) | ImageResponse::None => { callbacks.remove(); }, } let _ = std::mem::replace(&mut *self.pending_image_callbacks.borrow_mut(), images); } pub(crate) fn compositor_api(&self) -> &CrossProcessCompositorApi { &self.compositor_api } pub(crate) fn userscripts(&self) -> &[UserScript] { self.user_content_manager.scripts() } pub(crate) fn get_player_context(&self) -> WindowGLContext { self.player_context.clone() } // see note at https://dom.spec.whatwg.org/#concept-event-dispatch step 2 pub(crate) fn dispatch_event_with_target_override( &self, event: &Event, can_gc: CanGc, ) -> EventStatus { event.dispatch(self.upcast(), true, can_gc) } pub(crate) fn font_context(&self) -> &Arc<FontContext> { &self.font_context } pub(crate) fn handle_gamepad_event(&self, gamepad_event: GamepadEvent) { match gamepad_event { GamepadEvent::Connected(index, name, bounds, supported_haptic_effects) => { self.handle_gamepad_connect( index.0, name, bounds.axis_bounds, bounds.button_bounds, supported_haptic_effects, ); }, GamepadEvent::Disconnected(index) => { self.handle_gamepad_disconnect(index.0); }, GamepadEvent::Updated(index, update_type) => { self.receive_new_gamepad_button_or_axis(index.0, update_type); }, }; } /// <https://www.w3.org/TR/gamepad/#dfn-gamepadconnected> fn handle_gamepad_connect( &self, // As the spec actually defines how to set the gamepad index, the GilRs index // is currently unused, though in practice it will almost always be the same. // More infra is currently needed to track gamepads across windows. _index: usize, name: String, axis_bounds: (f64, f64), button_bounds: (f64, f64), supported_haptic_effects: GamepadSupportedHapticEffects, ) { // TODO: 2. If document is not null and is not allowed to use the "gamepad" permission, // then abort these steps. let this = Trusted::new(self); self.upcast::<GlobalScope>() .task_manager() .gamepad_task_source() .queue(task!(gamepad_connected: move || { let window = this.root(); let navigator = window.Navigator(); let selected_index = navigator.select_gamepad_index(); let gamepad = Gamepad::new( &window, selected_index, name, "standard".into(), axis_bounds, button_bounds, supported_haptic_effects, false, CanGc::note(), ); navigator.set_gamepad(selected_index as usize, &gamepad, CanGc::note()); })); } /// <https://www.w3.org/TR/gamepad/#dfn-gamepaddisconnected> fn handle_gamepad_disconnect(&self, index: usize) { let this = Trusted::new(self); self.upcast::<GlobalScope>() .task_manager() .gamepad_task_source() .queue(task!(gamepad_disconnected: move || { let window = this.root(); let navigator = window.Navigator(); if let Some(gamepad) = navigator.get_gamepad(index) { if window.Document().is_fully_active() { gamepad.update_connected(false, gamepad.exposed(), CanGc::note()); navigator.remove_gamepad(index); } } })); } /// <https://www.w3.org/TR/gamepad/#receiving-inputs> fn receive_new_gamepad_button_or_axis(&self, index: usize, update_type: GamepadUpdateType) { let this = Trusted::new(self); // <https://w3c.github.io/gamepad/#dfn-update-gamepad-state> self.upcast::<GlobalScope>().task_manager().gamepad_task_source().queue( task!(update_gamepad_state: move || { let window = this.root(); let navigator = window.Navigator(); if let Some(gamepad) = navigator.get_gamepad(index) { let current_time = window.Performance().Now(); gamepad.update_timestamp(*current_time); match update_type { GamepadUpdateType::Axis(index, value) => { gamepad.map_and_normalize_axes(index, value); }, GamepadUpdateType::Button(index, value) => { gamepad.map_and_normalize_buttons(index, value); } }; if !navigator.has_gamepad_gesture() && contains_user_gesture(update_type) { navigator.set_has_gamepad_gesture(true); navigator.GetGamepads() .iter() .filter_map(|g| g.as_ref()) .for_each(|gamepad| { gamepad.set_exposed(true); gamepad.update_timestamp(*current_time); let new_gamepad = Trusted::new(&**gamepad); if window.Document().is_fully_active() { window.upcast::<GlobalScope>().task_manager().gamepad_task_source().queue( task!(update_gamepad_connect: move || { let gamepad = new_gamepad.root(); gamepad.notify_event(GamepadEventType::Connected, CanGc::note()); }) ); } }); } } }) ); } } // https://html.spec.whatwg.org/multipage/#atob pub(crate) fn base64_btoa(input: DOMString) -> Fallible<DOMString> { // "The btoa() method must throw an InvalidCharacterError exception if // the method's first argument contains any character whose code point // is greater than U+00FF." if input.chars().any(|c: char| c > '\u{FF}') { Err(Error::InvalidCharacter) } else { // "Otherwise, the user agent must convert that argument to a // sequence of octets whose nth octet is the eight-bit // representation of the code point of the nth character of // the argument," let octets = input.chars().map(|c: char| c as u8).collect::<Vec<u8>>(); // "and then must apply the base64 algorithm to that sequence of // octets, and return the result. [RFC4648]" let config = base64::engine::general_purpose::GeneralPurposeConfig::new().with_encode_padding(true); let engine = base64::engine::GeneralPurpose::new(&base64::alphabet::STANDARD, config); Ok(DOMString::from(engine.encode(octets))) } } // https://html.spec.whatwg.org/multipage/#atob pub(crate) fn base64_atob(input: DOMString) -> Fallible<DOMString> { // "Remove all space characters from input." fn is_html_space(c: char) -> bool { HTML_SPACE_CHARACTERS.iter().any(|&m| m == c) } let without_spaces = input .chars() .filter(|&c| !is_html_space(c)) .collect::<String>(); let mut input = &*without_spaces; // "If the length of input divides by 4 leaving no remainder, then: // if input ends with one or two U+003D EQUALS SIGN (=) characters, // remove them from input." if input.len() % 4 == 0 { if input.ends_with("==") { input = &input[..input.len() - 2] } else if input.ends_with('=') { input = &input[..input.len() - 1] } } // "If the length of input divides by 4 leaving a remainder of 1, // throw an InvalidCharacterError exception and abort these steps." if input.len() % 4 == 1 { return Err(Error::InvalidCharacter); } // "If input contains a character that is not in the following list of // characters and character ranges, throw an InvalidCharacterError // exception and abort these steps: // // U+002B PLUS SIGN (+) // U+002F SOLIDUS (/) // Alphanumeric ASCII characters" if input .chars() .any(|c| c != '+' && c != '/' && !c.is_alphanumeric()) { return Err(Error::InvalidCharacter); } let config = base64::engine::general_purpose::GeneralPurposeConfig::new() .with_decode_padding_mode(base64::engine::DecodePaddingMode::RequireNone) .with_decode_allow_trailing_bits(true); let engine = base64::engine::GeneralPurpose::new(&base64::alphabet::STANDARD, config); let data = engine.decode(input).map_err(|_| Error::InvalidCharacter)?; Ok(data.iter().map(|&b| b as char).collect::<String>().into()) } impl WindowMethods<crate::DomTypeHolder> for Window { // https://html.spec.whatwg.org/multipage/#dom-alert fn Alert_(&self) { self.Alert(DOMString::new()); } // https://html.spec.whatwg.org/multipage/#dom-alert fn Alert(&self, s: DOMString) { // Print to the console. // Ensure that stderr doesn't trample through the alert() we use to // communicate test results (see executorservo.py in wptrunner). { let stderr = stderr(); let mut stderr = stderr.lock(); let stdout = stdout(); let mut stdout = stdout.lock(); writeln!(&mut stdout, "\nALERT: {}", s).unwrap(); stdout.flush().unwrap(); stderr.flush().unwrap(); } let (sender, receiver) = ProfiledIpc::channel(self.global().time_profiler_chan().clone()).unwrap(); let dialog = SimpleDialog::Alert { message: s.to_string(), response_sender: sender, }; let msg = EmbedderMsg::ShowSimpleDialog(self.webview_id(), dialog); self.send_to_embedder(msg); let AlertResponse::Ok = receiver.recv().unwrap(); } // https://html.spec.whatwg.org/multipage/#dom-confirm fn Confirm(&self, s: DOMString) -> bool { let (sender, receiver) = ProfiledIpc::channel(self.global().time_profiler_chan().clone()).unwrap(); let dialog = SimpleDialog::Confirm { message: s.to_string(), response_sender: sender, }; let msg = EmbedderMsg::ShowSimpleDialog(self.webview_id(), dialog); self.send_to_embedder(msg); receiver.recv().unwrap() == ConfirmResponse::Ok } // https://html.spec.whatwg.org/multipage/#dom-prompt fn Prompt(&self, message: DOMString, default: DOMString) -> Option<DOMString> { let (sender, receiver) = ProfiledIpc::channel(self.global().time_profiler_chan().clone()).unwrap(); let dialog = SimpleDialog::Prompt { message: message.to_string(), default: default.to_string(), response_sender: sender, }; let msg = EmbedderMsg::ShowSimpleDialog(self.webview_id(), dialog); self.send_to_embedder(msg); match receiver.recv().unwrap() { PromptResponse::Ok(input) => Some(input.into()), PromptResponse::Cancel => None, } } // https://html.spec.whatwg.org/multipage/#dom-window-stop fn Stop(&self, can_gc: CanGc) { // TODO: Cancel ongoing navigation. let doc = self.Document(); doc.abort(can_gc); } /// <https://html.spec.whatwg.org/multipage/#dom-window-focus> fn Focus(&self) { // > 1. Let `current` be this `Window` object's browsing context. // > // > 2. If `current` is null, then return. let current = match self.undiscarded_window_proxy() { Some(proxy) => proxy, None => return, }; // > 3. Run the focusing steps with `current`. current.focus(); // > 4. If current is a top-level browsing context, user agents are // > encouraged to trigger some sort of notification to indicate to // > the user that the page is attempting to gain focus. // // TODO: Step 4 } // https://html.spec.whatwg.org/multipage/#dom-window-blur fn Blur(&self) { // > User agents are encouraged to ignore calls to this `blur()` method // > entirely. } // https://html.spec.whatwg.org/multipage/#dom-open fn Open( &self, url: USVString, target: DOMString, features: DOMString, can_gc: CanGc, ) -> Fallible<Option<DomRoot<WindowProxy>>> { self.window_proxy().open(url, target, features, can_gc) } // https://html.spec.whatwg.org/multipage/#dom-opener fn GetOpener( &self, cx: JSContext, in_realm_proof: InRealm, mut retval: MutableHandleValue, ) -> Fallible<()> { // Step 1, Let current be this Window object's browsing context. let current = match self.window_proxy.get() { Some(proxy) => proxy, // Step 2, If current is null, then return null. None => { retval.set(NullValue()); return Ok(()); }, }; // Still step 2, since the window's BC is the associated doc's BC, // see https://html.spec.whatwg.org/multipage/#window-bc // and a doc's BC is null if it has been discarded. // see https://html.spec.whatwg.org/multipage/#concept-document-bc if current.is_browsing_context_discarded() { retval.set(NullValue()); return Ok(()); } // Step 3 to 5. current.opener(*cx, in_realm_proof, retval); Ok(()) } #[allow(unsafe_code)] // https://html.spec.whatwg.org/multipage/#dom-opener fn SetOpener(&self, cx: JSContext, value: HandleValue) -> ErrorResult { // Step 1. if value.is_null() { if let Some(proxy) = self.window_proxy.get() { proxy.disown(); } return Ok(()); } // Step 2. let obj = self.reflector().get_jsobject(); unsafe { let result = JS_DefineProperty(*cx, obj, c"opener".as_ptr(), value, JSPROP_ENUMERATE as u32); if result { Ok(()) } else { Err(Error::JSFailed) } } } // https://html.spec.whatwg.org/multipage/#dom-window-closed fn Closed(&self) -> bool { self.window_proxy .get() .map(|ref proxy| proxy.is_browsing_context_discarded() || proxy.is_closing()) .unwrap_or(true) } // https://html.spec.whatwg.org/multipage/#dom-window-close fn Close(&self) { // Step 1, Let current be this Window object's browsing context. // Step 2, If current is null or its is closing is true, then return. let window_proxy = match self.window_proxy.get() { Some(proxy) => proxy, None => return, }; if window_proxy.is_closing() { return; } // Note: check the length of the "session history", as opposed to the joint session history? // see https://github.com/whatwg/html/issues/3734 if let Ok(history_length) = self.History().GetLength() { let is_auxiliary = window_proxy.is_auxiliary(); // https://html.spec.whatwg.org/multipage/#script-closable let is_script_closable = (self.is_top_level() && history_length == 1) || is_auxiliary || pref!(dom_allow_scripts_to_close_windows); // TODO: rest of Step 3: // Is the incumbent settings object's responsible browsing context familiar with current? // Is the incumbent settings object's responsible browsing context allowed to navigate current? if is_script_closable { // Step 3.1, set current's is closing to true. window_proxy.close(); // Step 3.2, queue a task on the DOM manipulation task source to close current. let this = Trusted::new(self); let task = task!(window_close_browsing_context: move || { let window = this.root(); let document = window.Document(); // https://html.spec.whatwg.org/multipage/#closing-browsing-contexts // Step 1, check if traversable is closing, was already done above. // Steps 2 and 3, prompt to unload for all inclusive descendant navigables. // TODO: We should be prompting for all inclusive descendant navigables, // but we pass false here, which suggests we are not doing that. Why? if document.prompt_to_unload(false, CanGc::note()) { // Step 4, unload. document.unload(false, CanGc::note()); // https://html.spec.whatwg.org/multipage/#a-browsing-context-is-discarded // which calls into https://html.spec.whatwg.org/multipage/#discard-a-document. window.discard_browsing_context(); window.send_to_constellation(ScriptToConstellationMessage::DiscardTopLevelBrowsingContext); } }); self.as_global_scope() .task_manager() .dom_manipulation_task_source() .queue(task); } } } // https://html.spec.whatwg.org/multipage/#dom-document-2 fn Document(&self) -> DomRoot<Document> { self.document .get() .expect("Document accessed before initialization.") } // https://html.spec.whatwg.org/multipage/#dom-history fn History(&self) -> DomRoot<History> { self.history.or_init(|| History::new(self, CanGc::note())) } // https://w3c.github.io/IndexedDB/#factory-interface fn IndexedDB(&self) -> DomRoot<IDBFactory> { self.indexeddb.or_init(|| { let global_scope = self.upcast::<GlobalScope>(); IDBFactory::new(global_scope, CanGc::note()) }) } // https://html.spec.whatwg.org/multipage/#dom-window-customelements fn CustomElements(&self) -> DomRoot<CustomElementRegistry> { self.custom_element_registry .or_init(|| CustomElementRegistry::new(self, CanGc::note())) } // https://html.spec.whatwg.org/multipage/#dom-location fn Location(&self) -> DomRoot<Location> { self.location.or_init(|| Location::new(self, CanGc::note())) } // https://html.spec.whatwg.org/multipage/#dom-sessionstorage fn SessionStorage(&self) -> DomRoot<Storage> { self.session_storage .or_init(|| Storage::new(self, StorageType::Session, CanGc::note())) } // https://html.spec.whatwg.org/multipage/#dom-localstorage fn LocalStorage(&self) -> DomRoot<Storage> { self.local_storage .or_init(|| Storage::new(self, StorageType::Local, CanGc::note())) } // https://dvcs.w3.org/hg/webcrypto-api/raw-file/tip/spec/Overview.html#dfn-GlobalCrypto fn Crypto(&self) -> DomRoot<Crypto> { self.as_global_scope().crypto(CanGc::note()) } // https://html.spec.whatwg.org/multipage/#dom-frameelement fn GetFrameElement(&self) -> Option<DomRoot<Element>> { // Steps 1-3. let window_proxy = self.window_proxy.get()?; // Step 4-5. let container = window_proxy.frame_element()?; // Step 6. let container_doc = container.owner_document(); let current_doc = GlobalScope::current() .expect("No current global object") .as_window() .Document(); if !current_doc .origin() .same_origin_domain(container_doc.origin()) { return None; } // Step 7. Some(DomRoot::from_ref(container)) } // https://html.spec.whatwg.org/multipage/#dom-navigator fn Navigator(&self) -> DomRoot<Navigator> { self.navigator .or_init(|| Navigator::new(self, CanGc::note())) } // https://html.spec.whatwg.org/multipage/#dom-windowtimers-settimeout fn SetTimeout( &self, _cx: JSContext, callback: StringOrFunction, timeout: i32, args: Vec<HandleValue>, ) -> i32 { let callback = match callback { StringOrFunction::String(i) => TimerCallback::StringTimerCallback(i), StringOrFunction::Function(i) => TimerCallback::FunctionTimerCallback(i), }; self.as_global_scope().set_timeout_or_interval( callback, args, Duration::from_millis(timeout.max(0) as u64), IsInterval::NonInterval, ) } // https://html.spec.whatwg.org/multipage/#dom-windowtimers-cleartimeout fn ClearTimeout(&self, handle: i32) { self.as_global_scope().clear_timeout_or_interval(handle); } // https://html.spec.whatwg.org/multipage/#dom-windowtimers-setinterval fn SetInterval( &self, _cx: JSContext, callback: StringOrFunction, timeout: i32, args: Vec<HandleValue>, ) -> i32 { let callback = match callback { StringOrFunction::String(i) => TimerCallback::StringTimerCallback(i), StringOrFunction::Function(i) => TimerCallback::FunctionTimerCallback(i), }; self.as_global_scope().set_timeout_or_interval( callback, args, Duration::from_millis(timeout.max(0) as u64), IsInterval::Interval, ) } // https://html.spec.whatwg.org/multipage/#dom-windowtimers-clearinterval fn ClearInterval(&self, handle: i32) { self.ClearTimeout(handle); } // https://html.spec.whatwg.org/multipage/#dom-queuemicrotask fn QueueMicrotask(&self, callback: Rc<VoidFunction>) { self.as_global_scope().queue_function_as_microtask(callback); } /// <https://html.spec.whatwg.org/multipage/#dom-createimagebitmap> fn CreateImageBitmap( &self, image: ImageBitmapSource, options: &ImageBitmapOptions, can_gc: CanGc, ) -> Rc<Promise> { let p = ImageBitmap::create_image_bitmap( self.as_global_scope(), image, 0, 0, None, None, options, can_gc, ); p } /// <https://html.spec.whatwg.org/multipage/#dom-createimagebitmap> fn CreateImageBitmap_( &self, image: ImageBitmapSource, sx: i32, sy: i32, sw: i32, sh: i32, options: &ImageBitmapOptions, can_gc: CanGc, ) -> Rc<Promise> { let p = ImageBitmap::create_image_bitmap( self.as_global_scope(), image, sx, sy, Some(sw), Some(sh), options, can_gc, ); p } // https://html.spec.whatwg.org/multipage/#dom-window fn Window(&self) -> DomRoot<WindowProxy> { self.window_proxy() } // https://html.spec.whatwg.org/multipage/#dom-self fn Self_(&self) -> DomRoot<WindowProxy> { self.window_proxy() } // https://html.spec.whatwg.org/multipage/#dom-frames fn Frames(&self) -> DomRoot<WindowProxy> { self.window_proxy() } // https://html.spec.whatwg.org/multipage/#accessing-other-browsing-contexts fn Length(&self) -> u32 { self.Document().iframes().iter().count() as u32 } // https://html.spec.whatwg.org/multipage/#dom-parent fn GetParent(&self) -> Option<DomRoot<WindowProxy>> { // Steps 1-3. let window_proxy = self.undiscarded_window_proxy()?; // Step 4. if let Some(parent) = window_proxy.parent() { return Some(DomRoot::from_ref(parent)); } // Step 5. Some(window_proxy) } // https://html.spec.whatwg.org/multipage/#dom-top fn GetTop(&self) -> Option<DomRoot<WindowProxy>> { // Steps 1-3. let window_proxy = self.undiscarded_window_proxy()?; // Steps 4-5. Some(DomRoot::from_ref(window_proxy.top())) } // https://dvcs.w3.org/hg/webperf/raw-file/tip/specs/ // NavigationTiming/Overview.html#sec-window.performance-attribute fn Performance(&self) -> DomRoot<Performance> { self.performance.or_init(|| { Performance::new( self.as_global_scope(), self.navigation_start.get(), CanGc::note(), ) }) } // https://html.spec.whatwg.org/multipage/#globaleventhandlers global_event_handlers!(); // https://html.spec.whatwg.org/multipage/#windoweventhandlers window_event_handlers!(); // https://developer.mozilla.org/en-US/docs/Web/API/Window/screen fn Screen(&self) -> DomRoot<Screen> { self.screen.or_init(|| Screen::new(self, CanGc::note())) } // https://html.spec.whatwg.org/multipage/#dom-windowbase64-btoa fn Btoa(&self, btoa: DOMString) -> Fallible<DOMString> { base64_btoa(btoa) } // https://html.spec.whatwg.org/multipage/#dom-windowbase64-atob fn Atob(&self, atob: DOMString) -> Fallible<DOMString> { base64_atob(atob) } /// <https://html.spec.whatwg.org/multipage/#dom-window-requestanimationframe> fn RequestAnimationFrame(&self, callback: Rc<FrameRequestCallback>) -> u32 { self.Document() .request_animation_frame(AnimationFrameCallback::FrameRequestCallback { callback }) } /// <https://html.spec.whatwg.org/multipage/#dom-window-cancelanimationframe> fn CancelAnimationFrame(&self, ident: u32) { let doc = self.Document(); doc.cancel_animation_frame(ident); } // https://html.spec.whatwg.org/multipage/#dom-window-postmessage fn PostMessage( &self, cx: JSContext, message: HandleValue, target_origin: USVString, transfer: CustomAutoRooterGuard<Vec<*mut JSObject>>, ) -> ErrorResult { let incumbent = GlobalScope::incumbent().expect("no incumbent global?"); let source = incumbent.as_window(); let source_origin = source.Document().origin().immutable().clone(); self.post_message_impl(&target_origin, source_origin, source, cx, message, transfer) } /// <https://html.spec.whatwg.org/multipage/#dom-messageport-postmessage> fn PostMessage_( &self, cx: JSContext, message: HandleValue, options: RootedTraceableBox<WindowPostMessageOptions>, ) -> ErrorResult { let mut rooted = CustomAutoRooter::new( options .parent .transfer .iter() .map(|js: &RootedTraceableBox<Heap<*mut JSObject>>| js.get()) .collect(), ); let transfer = CustomAutoRooterGuard::new(*cx, &mut rooted); let incumbent = GlobalScope::incumbent().expect("no incumbent global?"); let source = incumbent.as_window(); let source_origin = source.Document().origin().immutable().clone(); self.post_message_impl( &options.targetOrigin, source_origin, source, cx, message, transfer, ) } // https://html.spec.whatwg.org/multipage/#dom-window-captureevents fn CaptureEvents(&self) { // This method intentionally does nothing } // https://html.spec.whatwg.org/multipage/#dom-window-releaseevents fn ReleaseEvents(&self) { // This method intentionally does nothing } // check-tidy: no specs after this line fn Debug(&self, message: DOMString) { debug!("{}", message); } #[allow(unsafe_code)] fn Gc(&self) { unsafe { JS_GC(*self.get_cx(), GCReason::API); } } #[allow(unsafe_code)] fn Js_backtrace(&self) { unsafe { println!("Current JS stack:"); dump_js_stack(*self.get_cx()); let rust_stack = Backtrace::new(); println!("Current Rust stack:\n{:?}", rust_stack); } } fn WebdriverCallback(&self, cx: JSContext, val: HandleValue, realm: InRealm, can_gc: CanGc) { let rv = jsval_to_webdriver(cx, &self.globalscope, val, realm, can_gc); let opt_chan = self.webdriver_script_chan.borrow_mut().take(); if let Some(chan) = opt_chan { let _ = chan.send(rv); } } fn WebdriverException(&self, cx: JSContext, val: HandleValue, realm: InRealm, can_gc: CanGc) { let rv = jsval_to_webdriver(cx, &self.globalscope, val, realm, can_gc); let opt_chan = self.webdriver_script_chan.borrow_mut().take(); if let Some(chan) = opt_chan { if let Ok(rv) = rv { let _ = chan.send(Err(WebDriverJSError::JSException(rv))); } else { let _ = chan.send(rv); } } } fn WebdriverTimeout(&self) { let opt_chan = self.webdriver_script_chan.borrow_mut().take(); if let Some(chan) = opt_chan { let _ = chan.send(Err(WebDriverJSError::Timeout)); } } fn WebdriverElement(&self, id: DOMString) -> Option<DomRoot<Element>> { find_node_by_unique_id_in_document(&self.Document(), id.into()).and_then(Root::downcast) } fn WebdriverFrame(&self, id: DOMString) -> Option<DomRoot<Element>> { find_node_by_unique_id_in_document(&self.Document(), id.into()) .and_then(Root::downcast::<HTMLIFrameElement>) .map(Root::upcast::<Element>) } fn WebdriverWindow(&self, _id: DOMString) -> Option<DomRoot<Window>> { warn!("Window references are not supported in webdriver yet"); None } fn WebdriverShadowRoot(&self, id: DOMString) -> Option<DomRoot<ShadowRoot>> { find_node_by_unique_id_in_document(&self.Document(), id.into()).and_then(Root::downcast) } // https://drafts.csswg.org/cssom/#dom-window-getcomputedstyle fn GetComputedStyle( &self, element: &Element, pseudo: Option<DOMString>, ) -> DomRoot<CSSStyleDeclaration> { // Step 2: Let obj be elt. // We don't store CSSStyleOwner directly because it stores a `Dom` which must be // rooted. This avoids the rooting the value temporarily. let mut is_null = false; // Step 3: If pseudoElt is provided, is not the empty string, and starts with a colon, then: // Step 3.1: Parse pseudoElt as a <pseudo-element-selector>, and let type be the result. let pseudo = pseudo.map(|mut s| { s.make_ascii_lowercase(); s }); let pseudo = match pseudo { Some(ref pseudo) if pseudo == ":before" || pseudo == "::before" => { Some(PseudoElement::Before) }, Some(ref pseudo) if pseudo == ":after" || pseudo == "::after" => { Some(PseudoElement::After) }, Some(ref pseudo) if pseudo == "::selection" => Some(PseudoElement::Selection), Some(ref pseudo) if pseudo == "::marker" => Some(PseudoElement::Marker), Some(ref pseudo) if pseudo.starts_with(':') => { // Step 3.2: If type is failure, or is a ::slotted() or ::part() // pseudo-element, let obj be null. is_null = true; None }, _ => None, }; // Step 4. Let decls be an empty list of CSS declarations. // Step 5: If obj is not null, and elt is connected, part of the flat tree, and // its shadow-including root has a browsing context which either doesn’t have a // browsing context container, or whose browsing context container is being // rendered, set decls to a list of all longhand properties that are supported CSS // properties, in lexicographical order, with the value being the resolved value // computed for obj using the style rules associated with doc. Additionally, // append to decls all the custom properties whose computed value for obj is not // the guaranteed-invalid value. // // Note: The specification says to generate the list of declarations beforehand, yet // also says the list should be alive. This is why we do not do step 4 and 5 here. // See: https://github.com/w3c/csswg-drafts/issues/6144 // // Step 6: Return a live CSSStyleProperties object with the following properties: CSSStyleDeclaration::new( self, if is_null { CSSStyleOwner::Null } else { CSSStyleOwner::Element(Dom::from_ref(element)) }, pseudo, CSSModificationAccess::Readonly, CanGc::note(), ) } // https://drafts.csswg.org/cssom-view/#dom-window-innerheight //TODO Include Scrollbar fn InnerHeight(&self) -> i32 { self.viewport_details .get() .size .height .to_i32() .unwrap_or(0) } // https://drafts.csswg.org/cssom-view/#dom-window-innerwidth //TODO Include Scrollbar fn InnerWidth(&self) -> i32 { self.viewport_details.get().size.width.to_i32().unwrap_or(0) } /// <https://drafts.csswg.org/cssom-view/#dom-window-scrollx> fn ScrollX(&self) -> i32 { self.scroll_offset_query_with_external_scroll_id( self.pipeline_id().root_scroll_id(), CanGc::note(), ) .x as i32 } // https://drafts.csswg.org/cssom-view/#dom-window-pagexoffset fn PageXOffset(&self) -> i32 { self.ScrollX() } /// <https://drafts.csswg.org/cssom-view/#dom-window-scrolly> fn ScrollY(&self) -> i32 { self.scroll_offset_query_with_external_scroll_id( self.pipeline_id().root_scroll_id(), CanGc::note(), ) .y as i32 } // https://drafts.csswg.org/cssom-view/#dom-window-pageyoffset fn PageYOffset(&self) -> i32 { self.ScrollY() } // https://drafts.csswg.org/cssom-view/#dom-window-scroll fn Scroll(&self, options: &ScrollToOptions, can_gc: CanGc) { // Step 1 let left = options.left.unwrap_or(0.0f64); let top = options.top.unwrap_or(0.0f64); self.scroll(left, top, options.parent.behavior, can_gc); } // https://drafts.csswg.org/cssom-view/#dom-window-scroll fn Scroll_(&self, x: f64, y: f64, can_gc: CanGc) { self.scroll(x, y, ScrollBehavior::Auto, can_gc); } // https://drafts.csswg.org/cssom-view/#dom-window-scrollto fn ScrollTo(&self, options: &ScrollToOptions) { self.Scroll(options, CanGc::note()); } // https://drafts.csswg.org/cssom-view/#dom-window-scrollto fn ScrollTo_(&self, x: f64, y: f64) { self.scroll(x, y, ScrollBehavior::Auto, CanGc::note()); } // https://drafts.csswg.org/cssom-view/#dom-window-scrollby fn ScrollBy(&self, options: &ScrollToOptions, can_gc: CanGc) { // Step 1 let x = options.left.unwrap_or(0.0f64); let y = options.top.unwrap_or(0.0f64); self.ScrollBy_(x, y, can_gc); self.scroll(x, y, options.parent.behavior, can_gc); } // https://drafts.csswg.org/cssom-view/#dom-window-scrollby fn ScrollBy_(&self, x: f64, y: f64, can_gc: CanGc) { // Step 3 let left = x + self.ScrollX() as f64; // Step 4 let top = y + self.ScrollY() as f64; // Step 5 self.scroll(left, top, ScrollBehavior::Auto, can_gc); } // https://drafts.csswg.org/cssom-view/#dom-window-resizeto fn ResizeTo(&self, width: i32, height: i32) { // Step 1 let window_proxy = match self.window_proxy.get() { Some(proxy) => proxy, None => return, }; // If target is not an auxiliary browsing context that was created by a script // (as opposed to by an action of the user), then return. if !window_proxy.is_auxiliary() { return; } let dpr = self.device_pixel_ratio(); let size = Size2D::new(width, height).to_f32() * dpr; self.send_to_embedder(EmbedderMsg::ResizeTo(self.webview_id(), size.to_i32())); } // https://drafts.csswg.org/cssom-view/#dom-window-resizeby fn ResizeBy(&self, x: i32, y: i32) { let (size, _) = self.client_window(); // Step 1 self.ResizeTo( x + size.width.to_i32().unwrap_or(1), y + size.height.to_i32().unwrap_or(1), ) } // https://drafts.csswg.org/cssom-view/#dom-window-moveto fn MoveTo(&self, x: i32, y: i32) { // Step 1 //TODO determine if this operation is allowed let dpr = self.device_pixel_ratio(); let point = Point2D::new(x, y).to_f32() * dpr; let msg = EmbedderMsg::MoveTo(self.webview_id(), point.to_i32()); self.send_to_embedder(msg); } // https://drafts.csswg.org/cssom-view/#dom-window-moveby fn MoveBy(&self, x: i32, y: i32) { let (_, origin) = self.client_window(); // Step 1 self.MoveTo(x + origin.x, y + origin.y) } // https://drafts.csswg.org/cssom-view/#dom-window-screenx fn ScreenX(&self) -> i32 { let (_, origin) = self.client_window(); origin.x } // https://drafts.csswg.org/cssom-view/#dom-window-screeny fn ScreenY(&self) -> i32 { let (_, origin) = self.client_window(); origin.y } // https://drafts.csswg.org/cssom-view/#dom-window-outerheight fn OuterHeight(&self) -> i32 { let (size, _) = self.client_window(); size.height.to_i32().unwrap_or(1) } // https://drafts.csswg.org/cssom-view/#dom-window-outerwidth fn OuterWidth(&self) -> i32 { let (size, _) = self.client_window(); size.width.to_i32().unwrap_or(1) } // https://drafts.csswg.org/cssom-view/#dom-window-devicepixelratio fn DevicePixelRatio(&self) -> Finite<f64> { Finite::wrap(self.device_pixel_ratio().get() as f64) } // https://html.spec.whatwg.org/multipage/#dom-window-status fn Status(&self) -> DOMString { self.status.borrow().clone() } // https://html.spec.whatwg.org/multipage/#dom-window-status fn SetStatus(&self, status: DOMString) { *self.status.borrow_mut() = status } // https://drafts.csswg.org/cssom-view/#dom-window-matchmedia fn MatchMedia(&self, query: DOMString) -> DomRoot<MediaQueryList> { let media_query_list = MediaList::parse_media_list(&query, self); let document = self.Document(); let mql = MediaQueryList::new(&document, media_query_list, CanGc::note()); self.media_query_lists.track(&*mql); mql } // https://fetch.spec.whatwg.org/#fetch-method fn Fetch( &self, input: RequestOrUSVString, init: RootedTraceableBox<RequestInit>, comp: InRealm, can_gc: CanGc, ) -> Rc<Promise> { fetch::Fetch(self.upcast(), input, init, comp, can_gc) } #[cfg(feature = "bluetooth")] fn TestRunner(&self) -> DomRoot<TestRunner> { self.test_runner .or_init(|| TestRunner::new(self.upcast(), CanGc::note())) } fn RunningAnimationCount(&self) -> u32 { self.document .get() .map_or(0, |d| d.animations().running_animation_count() as u32) } // https://html.spec.whatwg.org/multipage/#dom-name fn SetName(&self, name: DOMString) { if let Some(proxy) = self.undiscarded_window_proxy() { proxy.set_name(name); } } // https://html.spec.whatwg.org/multipage/#dom-name fn Name(&self) -> DOMString { match self.undiscarded_window_proxy() { Some(proxy) => proxy.get_name(), None => "".into(), } } // https://html.spec.whatwg.org/multipage/#dom-origin fn Origin(&self) -> USVString { USVString(self.origin().immutable().ascii_serialization()) } // https://w3c.github.io/selection-api/#dom-window-getselection fn GetSelection(&self) -> Option<DomRoot<Selection>> { self.document .get() .and_then(|d| d.GetSelection(CanGc::note())) } // https://dom.spec.whatwg.org/#dom-window-event #[allow(unsafe_code)] fn Event(&self, cx: JSContext, rval: MutableHandleValue) { if let Some(ref event) = *self.current_event.borrow() { unsafe { event.reflector().get_jsobject().to_jsval(*cx, rval); } } } fn IsSecureContext(&self) -> bool { self.as_global_scope().is_secure_context() } /// <https://html.spec.whatwg.org/multipage/#dom-window-nameditem> fn NamedGetter(&self, name: DOMString) -> Option<NamedPropertyValue> { if name.is_empty() { return None; } let document = self.Document(); // https://html.spec.whatwg.org/multipage/#document-tree-child-browsing-context-name-property-set let iframes: Vec<_> = document .iframes() .iter() .filter(|iframe| { if let Some(window) = iframe.GetContentWindow() { return window.get_name() == name; } false }) .collect(); let iframe_iter = iframes.iter().map(|iframe| iframe.upcast::<Element>()); let name = Atom::from(&*name); // Step 1. let elements_with_name = document.get_elements_with_name(&name); let name_iter = elements_with_name .iter() .map(|element| &**element) .filter(|elem| is_named_element_with_name_attribute(elem)); let elements_with_id = document.get_elements_with_id(&name); let id_iter = elements_with_id .iter() .map(|element| &**element) .filter(|elem| is_named_element_with_id_attribute(elem)); // Step 2. for elem in iframe_iter.clone() { if let Some(nested_window_proxy) = elem .downcast::<HTMLIFrameElement>() .and_then(|iframe| iframe.GetContentWindow()) { return Some(NamedPropertyValue::WindowProxy(nested_window_proxy)); } } let mut elements = iframe_iter.chain(name_iter).chain(id_iter); let first = elements.next()?; if elements.next().is_none() { // Step 3. return Some(NamedPropertyValue::Element(DomRoot::from_ref(first))); } // Step 4. #[derive(JSTraceable, MallocSizeOf)] struct WindowNamedGetter { #[no_trace] name: Atom, } impl CollectionFilter for WindowNamedGetter { fn filter(&self, elem: &Element, _root: &Node) -> bool { let type_ = match elem.upcast::<Node>().type_id() { NodeTypeId::Element(ElementTypeId::HTMLElement(type_)) => type_, _ => return false, }; if elem.get_id().as_ref() == Some(&self.name) { return true; } match type_ { HTMLElementTypeId::HTMLEmbedElement | HTMLElementTypeId::HTMLFormElement | HTMLElementTypeId::HTMLImageElement | HTMLElementTypeId::HTMLObjectElement => { elem.get_name().as_ref() == Some(&self.name) }, _ => false, } } } let collection = HTMLCollection::create( self, document.upcast(), Box::new(WindowNamedGetter { name }), CanGc::note(), ); Some(NamedPropertyValue::HTMLCollection(collection)) } // https://html.spec.whatwg.org/multipage/#dom-tree-accessors:supported-property-names fn SupportedPropertyNames(&self) -> Vec<DOMString> { let mut names_with_first_named_element_map: HashMap<&Atom, &Element> = HashMap::new(); let document = self.Document(); let name_map = document.name_map(); for (name, elements) in &name_map.0 { if name.is_empty() { continue; } let mut name_iter = elements .iter() .filter(|elem| is_named_element_with_name_attribute(elem)); if let Some(first) = name_iter.next() { names_with_first_named_element_map.insert(name, first); } } let id_map = document.id_map(); for (id, elements) in &id_map.0 { if id.is_empty() { continue; } let mut id_iter = elements .iter() .filter(|elem| is_named_element_with_id_attribute(elem)); if let Some(first) = id_iter.next() { match names_with_first_named_element_map.entry(id) { Entry::Vacant(entry) => drop(entry.insert(first)), Entry::Occupied(mut entry) => { if first.upcast::<Node>().is_before(entry.get().upcast()) { *entry.get_mut() = first; } }, } } } let mut names_with_first_named_element_vec: Vec<(&Atom, &Element)> = names_with_first_named_element_map .iter() .map(|(k, v)| (*k, *v)) .collect(); names_with_first_named_element_vec.sort_unstable_by(|a, b| { if a.1 == b.1 { // This can happen if an img has an id different from its name, // spec does not say which string to put first. a.0.cmp(b.0) } else if a.1.upcast::<Node>().is_before(b.1.upcast::<Node>()) { cmp::Ordering::Less } else { cmp::Ordering::Greater } }); names_with_first_named_element_vec .iter() .map(|(k, _v)| DOMString::from(&***k)) .collect() } /// <https://html.spec.whatwg.org/multipage/#dom-structuredclone> fn StructuredClone( &self, cx: JSContext, value: HandleValue, options: RootedTraceableBox<StructuredSerializeOptions>, retval: MutableHandleValue, ) -> Fallible<()> { self.as_global_scope() .structured_clone(cx, value, options, retval) } fn TrustedTypes(&self, can_gc: CanGc) -> DomRoot<TrustedTypePolicyFactory> { self.trusted_types .or_init(|| TrustedTypePolicyFactory::new(self.as_global_scope(), can_gc)) } } impl Window { // https://heycam.github.io/webidl/#named-properties-object // https://html.spec.whatwg.org/multipage/#named-access-on-the-window-object #[allow(unsafe_code)] pub(crate) fn create_named_properties_object( cx: JSContext, proto: HandleObject, object: MutableHandleObject, ) { window_named_properties::create(cx, proto, object) } pub(crate) fn current_event(&self) -> Option<DomRoot<Event>> { self.current_event .borrow() .as_ref() .map(|e| DomRoot::from_ref(&**e)) } pub(crate) fn set_current_event(&self, event: Option<&Event>) -> Option<DomRoot<Event>> { let current = self.current_event(); *self.current_event.borrow_mut() = event.map(Dom::from_ref); current } /// <https://html.spec.whatwg.org/multipage/#window-post-message-steps> fn post_message_impl( &self, target_origin: &USVString, source_origin: ImmutableOrigin, source: &Window, cx: JSContext, message: HandleValue, transfer: CustomAutoRooterGuard<Vec<*mut JSObject>>, ) -> ErrorResult { // Step 1-2, 6-8. let data = structuredclone::write(cx, message, Some(transfer))?; // Step 3-5. let target_origin = match target_origin.0[..].as_ref() { "*" => None, "/" => Some(source_origin.clone()), url => match ServoUrl::parse(url) { Ok(url) => Some(url.origin().clone()), Err(_) => return Err(Error::Syntax), }, }; // Step 9. self.post_message(target_origin, source_origin, &source.window_proxy(), data); Ok(()) } // https://drafts.css-houdini.org/css-paint-api-1/#paint-worklet pub(crate) fn paint_worklet(&self) -> DomRoot<Worklet> { self.paint_worklet .or_init(|| self.new_paint_worklet(CanGc::note())) } pub(crate) fn has_document(&self) -> bool { self.document.get().is_some() } pub(crate) fn clear_js_runtime(&self) { self.as_global_scope() .remove_web_messaging_and_dedicated_workers_infra(); // Clean up any active promises // https://github.com/servo/servo/issues/15318 if let Some(custom_elements) = self.custom_element_registry.get() { custom_elements.teardown(); } // The above code may not catch all DOM objects (e.g. DOM // objects removed from the tree that haven't been collected // yet). There should not be any such DOM nodes with layout // data, but if there are, then when they are dropped, they // will attempt to send a message to layout. // This causes memory safety issues, because the DOM node uses // the layout channel from its window, and the window has // already been GC'd. For nodes which do not have a live // pointer, we can avoid this by GCing now: self.Gc(); // but there may still be nodes being kept alive by user // script. // TODO: ensure that this doesn't happen! self.current_state.set(WindowState::Zombie); *self.js_runtime.borrow_mut() = None; // If this is the currently active pipeline, // nullify the window_proxy. if let Some(proxy) = self.window_proxy.get() { let pipeline_id = self.pipeline_id(); if let Some(currently_active) = proxy.currently_active() { if currently_active == pipeline_id { self.window_proxy.set(None); } } } if let Some(performance) = self.performance.get() { performance.clear_and_disable_performance_entry_buffer(); } self.as_global_scope() .task_manager() .cancel_all_tasks_and_ignore_future_tasks(); } /// <https://drafts.csswg.org/cssom-view/#dom-window-scroll> pub(crate) fn scroll(&self, x_: f64, y_: f64, behavior: ScrollBehavior, can_gc: CanGc) { // Step 3 let xfinite = if x_.is_finite() { x_ } else { 0.0f64 }; let yfinite = if y_.is_finite() { y_ } else { 0.0f64 }; // TODO Step 4 - determine if a window has a viewport // Step 5 & 6 // TODO: Remove scrollbar dimensions. let viewport = self.viewport_details.get().size; // Step 7 & 8 // TODO: Consider `block-end` and `inline-end` overflow direction. let scrolling_area = self.scrolling_area_query(None, can_gc); let x = xfinite .min(scrolling_area.width() as f64 - viewport.width as f64) .max(0.0f64); let y = yfinite .min(scrolling_area.height() as f64 - viewport.height as f64) .max(0.0f64); // Step 10 //TODO handling ongoing smooth scrolling if x == self.ScrollX() as f64 && y == self.ScrollY() as f64 { return; } //TODO Step 11 // Step 12: Perform a scroll of the viewport to position, document’s root element // as the associated element, if there is one, or null otherwise, and the scroll // behavior being the value of the behavior dictionary member of options. self.perform_a_scroll( x.to_f32().unwrap_or(0.0f32), y.to_f32().unwrap_or(0.0f32), self.pipeline_id().root_scroll_id(), behavior, None, can_gc, ); } /// <https://drafts.csswg.org/cssom-view/#perform-a-scroll> pub(crate) fn perform_a_scroll( &self, x: f32, y: f32, scroll_id: ExternalScrollId, _behavior: ScrollBehavior, _element: Option<&Element>, can_gc: CanGc, ) { // TODO Step 1 // TODO(mrobinson, #18709): Add smooth scrolling support to WebRender so that we can // properly process ScrollBehavior here. self.reflow( ReflowGoal::UpdateScrollNode(scroll_id, Vector2D::new(x, y)), can_gc, ); } pub(crate) fn device_pixel_ratio(&self) -> Scale<f32, CSSPixel, DevicePixel> { self.viewport_details.get().hidpi_scale_factor } fn client_window(&self) -> (Size2D<u32, CSSPixel>, Point2D<i32, CSSPixel>) { let timer_profile_chan = self.global().time_profiler_chan().clone(); let (sender, receiver) = ProfiledIpc::channel::<DeviceIndependentIntRect>(timer_profile_chan).unwrap(); let _ = self.compositor_api.sender().send( compositing_traits::CompositorMsg::GetClientWindowRect(self.webview_id(), sender), ); let rect = receiver.recv().unwrap_or_default(); ( Size2D::new(rect.size().width as u32, rect.size().height as u32), Point2D::new(rect.min.x, rect.min.y), ) } /// Prepares to tick animations and then does a reflow which also advances the /// layout animation clock. #[allow(unsafe_code)] pub(crate) fn advance_animation_clock(&self, delta_ms: i32) { self.Document() .advance_animation_timeline_for_testing(delta_ms as f64 / 1000.); ScriptThread::handle_tick_all_animations_for_testing(self.pipeline_id()); } /// Reflows the page unconditionally if possible and not suppressed. This method will wait for /// the layout to complete. If there is no window size yet, the page is presumed invisible and /// no reflow is performed. If reflow is suppressed, no reflow will be performed for ForDisplay /// goals. /// /// Returns true if layout actually happened and it sent a new display list to the renderer. /// /// NOTE: This method should almost never be called directly! Layout and rendering updates should /// happen as part of the HTML event loop via *update the rendering*. #[allow(unsafe_code)] fn force_reflow( &self, reflow_goal: ReflowGoal, condition: Option<ReflowTriggerCondition>, ) -> bool { self.Document().ensure_safe_to_run_script_or_layout(); // If layouts are blocked, we block all layouts that are for display only. Other // layouts (for queries and scrolling) are not blocked, as they do not display // anything and script excpects the layout to be up-to-date after they run. let layout_blocked = self.layout_blocker.get().layout_blocked(); let pipeline_id = self.pipeline_id(); if reflow_goal == ReflowGoal::UpdateTheRendering && layout_blocked { debug!("Suppressing pre-load-event reflow pipeline {pipeline_id}"); return false; } if condition != Some(ReflowTriggerCondition::PaintPostponed) { debug!( "Invalidating layout cache due to reflow condition {:?}", condition ); // Invalidate any existing cached layout values. self.layout_marker.borrow().set(false); // Create a new layout caching token. *self.layout_marker.borrow_mut() = Rc::new(Cell::new(true)); } else { debug!("Not invalidating cached layout values for paint-only reflow."); } debug!("script: performing reflow for goal {reflow_goal:?}"); let marker = if self.need_emit_timeline_marker(TimelineMarkerType::Reflow) { Some(TimelineMarker::start("Reflow".to_owned())) } else { None }; // On debug mode, print the reflow event information. if self.relayout_event { debug_reflow_events(pipeline_id, &reflow_goal); } let document = self.Document(); let stylesheets_changed = document.flush_stylesheets_for_reflow(); // If this reflow is for display, ensure webgl canvases are composited with // up-to-date contents. let for_display = reflow_goal.needs_display(); if for_display { document.flush_dirty_webgl_canvases(); document.flush_dirty_2d_canvases(); } let pending_restyles = document.drain_pending_restyles(); let dirty_root = document .take_dirty_root() .filter(|_| !stylesheets_changed) .or_else(|| document.GetDocumentElement()) .map(|root| root.upcast::<Node>().to_trusted_node_address()); let highlighted_dom_node = document.highlighted_dom_node().map(|node| node.to_opaque()); // Send new document and relevant styles to layout. let reflow = ReflowRequest { document: document.upcast::<Node>().to_trusted_node_address(), dirty_root, stylesheets_changed, viewport_details: self.viewport_details.get(), origin: self.origin().immutable().clone(), reflow_goal, dom_count: document.dom_count(), pending_restyles, animation_timeline_value: document.current_animation_timeline_value(), animations: document.animations().sets.clone(), node_to_image_animation_map: document .image_animation_manager_mut() .take_image_animate_set(), theme: self.theme.get(), highlighted_dom_node, }; let Some(results) = self.layout.borrow_mut().reflow(reflow) else { return false; }; debug!("script: layout complete"); if let Some(marker) = marker { self.emit_timeline_marker(marker.end()); } // Either this reflow caused new contents to be displayed or on the next // full layout attempt a reflow should be forced in order to update the // visual contents of the page. A case where full display might be delayed // is when reflowing just for the purpose of doing a layout query. document.set_needs_paint(!for_display); for image in results.pending_images { let id = image.id; let node = unsafe { from_untrusted_node_address(image.node) }; if let PendingImageState::Unrequested(ref url) = image.state { fetch_image_for_layout(url.clone(), &node, id, self.image_cache.clone()); } let mut images = self.pending_layout_images.borrow_mut(); if !images.contains_key(&id) { let trusted_node = Trusted::new(&*node); let sender = self.register_image_cache_listener(id, move |response| { trusted_node .root() .owner_window() .pending_layout_image_notification(response); }); self.image_cache.add_listener(ImageLoadListener::new( sender, self.pipeline_id(), id, )); } let nodes = images.entry(id).or_default(); if !nodes.iter().any(|n| std::ptr::eq(&**n, &*node)) { nodes.push(Dom::from_ref(&*node)); } } for image in results.pending_rasterization_images { let node = unsafe { from_untrusted_node_address(image.node) }; let mut images = self.pending_images_for_rasterization.borrow_mut(); if !images.contains_key(&(image.id, image.size)) { self.image_cache.add_rasterization_complete_listener( pipeline_id, image.id, image.size, self.image_cache_sender.clone(), ); } let nodes = images.entry((image.id, image.size)).or_default(); if !nodes.iter().any(|n| std::ptr::eq(&**n, &*node)) { nodes.push(Dom::from_ref(&*node)); } } let size_messages = self .Document() .iframes_mut() .handle_new_iframe_sizes_after_layout(results.iframe_sizes); if !size_messages.is_empty() { self.send_to_constellation(ScriptToConstellationMessage::IFrameSizes(size_messages)); } document .image_animation_manager_mut() .restore_image_animate_set(results.node_to_image_animation_map); document.update_animations_post_reflow(); self.update_constellation_epoch(); results.built_display_list } /// Reflows the page if it's possible to do so and the page is dirty. Returns true if layout /// actually happened and produced a new display list, false otherwise. /// /// NOTE: This method should almost never be called directly! Layout and rendering updates /// should happen as part of the HTML event loop via *update the rendering*. Currerntly, the /// only exceptions are script queries and scroll requests. pub(crate) fn reflow(&self, reflow_goal: ReflowGoal, can_gc: CanGc) -> bool { // Never reflow inactive Documents. if !self.Document().is_fully_active() { return false; } // Count the pending web fonts before layout, in case a font loads during the layout. let waiting_for_web_fonts_to_load = self.font_context.web_fonts_still_loading() != 0; self.Document().ensure_safe_to_run_script_or_layout(); let mut issued_reflow = false; let condition = self.Document().needs_reflow(); let updating_the_rendering = reflow_goal == ReflowGoal::UpdateTheRendering; let for_display = reflow_goal.needs_display(); if !updating_the_rendering || condition.is_some() { debug!("Reflowing document ({:?})", self.pipeline_id()); issued_reflow = self.force_reflow(reflow_goal, condition); // We shouldn't need a reflow immediately after a completed reflow, unless the reflow didn't // display anything and it wasn't for display. Queries can cause this to happen. if issued_reflow { let condition = self.Document().needs_reflow(); let display_is_pending = condition == Some(ReflowTriggerCondition::PaintPostponed); assert!( condition.is_none() || (display_is_pending && !for_display), "Needed reflow after reflow: {:?}", condition ); } } else { debug!( "Document ({:?}) doesn't need reflow - skipping it (goal {reflow_goal:?})", self.pipeline_id() ); } let document = self.Document(); let font_face_set = document.Fonts(can_gc); let is_ready_state_complete = document.ReadyState() == DocumentReadyState::Complete; // From https://drafts.csswg.org/css-font-loading/#font-face-set-ready: // > A FontFaceSet is pending on the environment if any of the following are true: // > - the document is still loading // > - the document has pending stylesheet requests // > - the document has pending layout operations which might cause the user agent to request // > a font, or which depend on recently-loaded fonts // // Thus, we are queueing promise resolution here. This reflow should have been triggered by // a "rendering opportunity" in `ScriptThread::handle_web_font_loaded, which should also // make sure a microtask checkpoint happens, triggering the promise callback. if !waiting_for_web_fonts_to_load && is_ready_state_complete { font_face_set.fulfill_ready_promise_if_needed(can_gc); } // If writing a screenshot, check if the script has reached a state // where it's safe to write the image. This means that: // 1) The reflow is for display (otherwise it could be a query) // 2) The html element doesn't contain the 'reftest-wait' class // 3) The load event has fired. // When all these conditions are met, notify the constellation // that this pipeline is ready to write the image (from the script thread // perspective at least). if opts::get().wait_for_stable_image && updating_the_rendering { // Checks if the html element has reftest-wait attribute present. // See http://testthewebforward.org/docs/reftests.html // and https://web-platform-tests.org/writing-tests/crashtest.html let html_element = document.GetDocumentElement(); let reftest_wait = html_element.is_some_and(|elem| { elem.has_class(&atom!("reftest-wait"), CaseSensitivity::CaseSensitive) || elem.has_class(&Atom::from("test-wait"), CaseSensitivity::CaseSensitive) }); let has_sent_idle_message = self.has_sent_idle_message.get(); let no_pending_images = self.pending_layout_images.borrow().is_empty() && self.pending_images_for_rasterization.borrow().is_empty(); if !has_sent_idle_message && is_ready_state_complete && !reftest_wait && no_pending_images && !waiting_for_web_fonts_to_load { debug!( "{:?}: Sending DocumentState::Idle to Constellation", self.pipeline_id() ); let event = ScriptToConstellationMessage::SetDocumentState(DocumentState::Idle); self.send_to_constellation(event); self.has_sent_idle_message.set(true); } } issued_reflow } /// If parsing has taken a long time and reflows are still waiting for the `load` event, /// start allowing them. See <https://github.com/servo/servo/pull/6028>. pub(crate) fn reflow_if_reflow_timer_expired(&self, can_gc: CanGc) { // Only trigger a long parsing time reflow if we are in the first parse of `<body>` // and it started more than `INITIAL_REFLOW_DELAY` ago. if !matches!( self.layout_blocker.get(), LayoutBlocker::Parsing(instant) if instant + INITIAL_REFLOW_DELAY < Instant::now() ) { return; } self.allow_layout_if_necessary(can_gc); } /// Block layout for this `Window` until parsing is done. If parsing takes a long time, /// we want to layout anyway, so schedule a moment in the future for when layouts are /// allowed even though parsing isn't finished and we havne't sent a load event. pub(crate) fn prevent_layout_until_load_event(&self) { // If we have already started parsing or have already fired a load event, then // don't delay the first layout any longer. if !matches!(self.layout_blocker.get(), LayoutBlocker::WaitingForParse) { return; } self.layout_blocker .set(LayoutBlocker::Parsing(Instant::now())); } /// Inform the [`Window`] that layout is allowed either because `load` has happened /// or because parsing the `<body>` took so long that we cannot wait any longer. pub(crate) fn allow_layout_if_necessary(&self, can_gc: CanGc) { if matches!( self.layout_blocker.get(), LayoutBlocker::FiredLoadEventOrParsingTimerExpired ) { return; } self.layout_blocker .set(LayoutBlocker::FiredLoadEventOrParsingTimerExpired); self.Document().set_needs_paint(true); // We do this immediately instead of scheduling a future task, because this can // happen if parsing is taking a very long time, which means that the // `ScriptThread` is busy doing the parsing and not doing layouts. // // TOOD(mrobinson): It's expected that this is necessary when in the process of // parsing, as we need to interrupt it to update contents, but why is this // necessary when parsing finishes? Not doing the synchronous update in that case // causes iframe tests to become flaky. It seems there's an issue with the timing of // iframe size updates. // // See <https://github.com/servo/servo/issues/14719> self.reflow(ReflowGoal::UpdateTheRendering, can_gc); } pub(crate) fn layout_blocked(&self) -> bool { self.layout_blocker.get().layout_blocked() } /// If writing a screenshot, synchronously update the layout epoch that it set /// in the constellation. pub(crate) fn update_constellation_epoch(&self) { if !opts::get().wait_for_stable_image { return; } let epoch = self.layout.borrow().current_epoch(); debug!( "{:?}: Updating constellation epoch: {epoch:?}", self.pipeline_id() ); let (sender, receiver) = ipc::channel().expect("Failed to create IPC channel!"); let event = ScriptToConstellationMessage::SetLayoutEpoch(epoch, sender); self.send_to_constellation(event); let _ = receiver.recv(); } pub(crate) fn layout_reflow(&self, query_msg: QueryMsg, can_gc: CanGc) -> bool { self.reflow(ReflowGoal::LayoutQuery(query_msg), can_gc) } pub(crate) fn resolved_font_style_query( &self, node: &Node, value: String, can_gc: CanGc, ) -> Option<ServoArc<Font>> { self.layout_reflow(QueryMsg::ResolvedFontStyleQuery, can_gc); let document = self.Document(); let animations = document.animations().sets.clone(); self.layout.borrow().query_resolved_font_style( node.to_trusted_node_address(), &value, animations, document.current_animation_timeline_value(), ) } // Query content box without considering any reflow pub(crate) fn content_box_query_unchecked(&self, node: &Node) -> Option<UntypedRect<Au>> { self.layout .borrow() .query_content_box(node.to_trusted_node_address()) } pub(crate) fn content_box_query(&self, node: &Node, can_gc: CanGc) -> Option<UntypedRect<Au>> { self.layout_reflow(QueryMsg::ContentBox, can_gc); self.content_box_query_unchecked(node) } pub(crate) fn content_boxes_query(&self, node: &Node, can_gc: CanGc) -> Vec<UntypedRect<Au>> { self.layout_reflow(QueryMsg::ContentBoxes, can_gc); self.layout .borrow() .query_content_boxes(node.to_trusted_node_address()) } pub(crate) fn client_rect_query(&self, node: &Node, can_gc: CanGc) -> UntypedRect<i32> { self.layout_reflow(QueryMsg::ClientRectQuery, can_gc); self.layout .borrow() .query_client_rect(node.to_trusted_node_address()) } /// Find the scroll area of the given node, if it is not None. If the node /// is None, find the scroll area of the viewport. pub(crate) fn scrolling_area_query( &self, node: Option<&Node>, can_gc: CanGc, ) -> UntypedRect<i32> { self.layout_reflow(QueryMsg::ScrollingAreaOrOffsetQuery, can_gc); self.layout .borrow() .query_scrolling_area(node.map(Node::to_trusted_node_address)) } pub(crate) fn scroll_offset_query( &self, node: &Node, can_gc: CanGc, ) -> Vector2D<f32, LayoutPixel> { let external_scroll_id = ExternalScrollId( combine_id_with_fragment_type(node.to_opaque().id(), FragmentType::FragmentBody), self.pipeline_id().into(), ); self.scroll_offset_query_with_external_scroll_id(external_scroll_id, can_gc) } fn scroll_offset_query_with_external_scroll_id( &self, external_scroll_id: ExternalScrollId, can_gc: CanGc, ) -> Vector2D<f32, LayoutPixel> { self.layout_reflow(QueryMsg::ScrollingAreaOrOffsetQuery, can_gc); self.layout .borrow() .scroll_offset(external_scroll_id) .unwrap_or_default() } // https://drafts.csswg.org/cssom-view/#element-scrolling-members pub(crate) fn scroll_node( &self, node: &Node, x_: f64, y_: f64, behavior: ScrollBehavior, can_gc: CanGc, ) { let scroll_id = ExternalScrollId( combine_id_with_fragment_type(node.to_opaque().id(), FragmentType::FragmentBody), self.pipeline_id().into(), ); // Step 12 self.perform_a_scroll( x_.to_f32().unwrap_or(0.0f32), y_.to_f32().unwrap_or(0.0f32), scroll_id, behavior, None, can_gc, ); } pub(crate) fn resolved_style_query( &self, element: TrustedNodeAddress, pseudo: Option<PseudoElement>, property: PropertyId, can_gc: CanGc, ) -> DOMString { self.layout_reflow(QueryMsg::ResolvedStyleQuery, can_gc); let document = self.Document(); let animations = document.animations().sets.clone(); DOMString::from(self.layout.borrow().query_resolved_style( element, pseudo, property, animations, document.current_animation_timeline_value(), )) } /// If the given |browsing_context_id| refers to an `<iframe>` that is an element /// in this [`Window`] and that `<iframe>` has been laid out, return its size. /// Otherwise, return `None`. pub(crate) fn get_iframe_viewport_details_if_known( &self, browsing_context_id: BrowsingContextId, can_gc: CanGc, ) -> Option<ViewportDetails> { // Reflow might fail, but do a best effort to return the right size. self.layout_reflow(QueryMsg::InnerWindowDimensionsQuery, can_gc); self.Document() .iframes() .get(browsing_context_id) .and_then(|iframe| iframe.size) } #[allow(unsafe_code)] pub(crate) fn offset_parent_query( &self, node: &Node, can_gc: CanGc, ) -> (Option<DomRoot<Element>>, UntypedRect<Au>) { self.layout_reflow(QueryMsg::OffsetParentQuery, can_gc); let response = self .layout .borrow() .query_offset_parent(node.to_trusted_node_address()); let element = response.node_address.and_then(|parent_node_address| { let node = unsafe { from_untrusted_node_address(parent_node_address) }; DomRoot::downcast(node) }); (element, response.rect) } pub(crate) fn text_index_query( &self, node: &Node, point_in_node: UntypedPoint2D<f32>, can_gc: CanGc, ) -> Option<usize> { self.layout_reflow(QueryMsg::TextIndexQuery, can_gc); self.layout .borrow() .query_text_indext(node.to_opaque(), point_in_node) } #[allow(unsafe_code)] pub(crate) fn init_window_proxy(&self, window_proxy: &WindowProxy) { assert!(self.window_proxy.get().is_none()); self.window_proxy.set(Some(window_proxy)); } #[allow(unsafe_code)] pub(crate) fn init_document(&self, document: &Document) { assert!(self.document.get().is_none()); assert!(document.window() == self); self.document.set(Some(document)); if self.unminify_css { *self.unminified_css_dir.borrow_mut() = Some(unminified_path("unminified-css")); } } /// Commence a new URL load which will either replace this window or scroll to a fragment. /// /// <https://html.spec.whatwg.org/multipage/#navigating-across-documents> pub(crate) fn load_url( &self, history_handling: NavigationHistoryBehavior, force_reload: bool, load_data: LoadData, can_gc: CanGc, ) { let doc = self.Document(); // Step 3. Let initiatorOriginSnapshot be sourceDocument's origin. let initiator_origin_snapshot = &load_data.load_origin; // TODO: Important re security. See https://github.com/servo/servo/issues/23373 // Step 5. check that the source browsing-context is "allowed to navigate" this window. if !force_reload && load_data.url.as_url()[..Position::AfterQuery] == doc.url().as_url()[..Position::AfterQuery] { // Step 6 // TODO: Fragment handling appears to have moved to step 13 if let Some(fragment) = load_data.url.fragment() { self.send_to_constellation(ScriptToConstellationMessage::NavigatedToFragment( load_data.url.clone(), history_handling, )); doc.check_and_scroll_fragment(fragment, can_gc); let this = Trusted::new(self); let old_url = doc.url().into_string(); let new_url = load_data.url.clone().into_string(); let task = task!(hashchange_event: move || { let this = this.root(); let event = HashChangeEvent::new( &this, atom!("hashchange"), false, false, old_url, new_url, CanGc::note()); event.upcast::<Event>().fire(this.upcast::<EventTarget>(), CanGc::note()); }); self.as_global_scope() .task_manager() .dom_manipulation_task_source() .queue(task); doc.set_url(load_data.url.clone()); return; } } // Step 4 and 5 let pipeline_id = self.pipeline_id(); let window_proxy = self.window_proxy(); if let Some(active) = window_proxy.currently_active() { if pipeline_id == active && doc.is_prompting_or_unloading() { return; } } // Step 8 if doc.prompt_to_unload(false, can_gc) { let window_proxy = self.window_proxy(); if window_proxy.parent().is_some() { // Step 10 // If browsingContext is a nested browsing context, // then put it in the delaying load events mode. window_proxy.start_delaying_load_events_mode(); } // Step 11. If historyHandling is "auto", then: let resolved_history_handling = if history_handling == NavigationHistoryBehavior::Auto { // Step 11.1. If url equals navigable's active document's URL, and // initiatorOriginSnapshot is same origin with targetNavigable's active document's // origin, then set historyHandling to "replace". // Note: `targetNavigable` is not actually defined in the spec, "active document" is // assumed to be the correct reference based on WPT results if let LoadOrigin::Script(initiator_origin) = initiator_origin_snapshot { if load_data.url == doc.url() && initiator_origin.same_origin(doc.origin()) { NavigationHistoryBehavior::Replace } else { NavigationHistoryBehavior::Push } } else { // Step 11.2. Otherwise, set historyHandling to "push". NavigationHistoryBehavior::Push } // Step 12. If the navigation must be a replace given url and navigable's active // document, then set historyHandling to "replace". } else if load_data.url.scheme() == "javascript" || doc.is_initial_about_blank() { NavigationHistoryBehavior::Replace } else { NavigationHistoryBehavior::Push }; // Step 13 ScriptThread::navigate( window_proxy.browsing_context_id(), pipeline_id, load_data, resolved_history_handling, ); }; } pub(crate) fn set_viewport_details(&self, size: ViewportDetails) { self.viewport_details.set(size); } pub(crate) fn viewport_details(&self) -> ViewportDetails { self.viewport_details.get() } /// Get the theme of this [`Window`]. pub(crate) fn theme(&self) -> Theme { self.theme.get() } /// Handle a theme change request, triggering a reflow is any actual change occured. pub(crate) fn handle_theme_change(&self, new_theme: Theme) { if self.theme.get() == new_theme { return; } self.theme.set(new_theme); self.Document().set_needs_paint(true); } pub(crate) fn get_url(&self) -> ServoUrl { self.Document().url() } pub(crate) fn windowproxy_handler(&self) -> &'static WindowProxyHandler { self.dom_static.windowproxy_handler } pub(crate) fn add_resize_event(&self, event: ViewportDetails, event_type: WindowSizeType) { // Whenever we receive a new resize event we forget about all the ones that came before // it, to avoid unnecessary relayouts *self.unhandled_resize_event.borrow_mut() = Some((event, event_type)) } pub(crate) fn take_unhandled_resize_event(&self) -> Option<(ViewportDetails, WindowSizeType)> { self.unhandled_resize_event.borrow_mut().take() } pub(crate) fn set_viewport_size(&self, new_viewport_size: UntypedSize2D<f32>) { let new_viewport_size = Size2D::new( Au::from_f32_px(new_viewport_size.width), Au::from_f32_px(new_viewport_size.height), ); if new_viewport_size == self.current_viewport_size.get() { return; } self.current_viewport_size.set(new_viewport_size); // The document needs to be repainted, because the initial containing block // is now a different size. self.Document().set_needs_paint(true); } pub(crate) fn suspend(&self, can_gc: CanGc) { // Suspend timer events. self.as_global_scope().suspend(); // Set the window proxy to be a cross-origin window. if self.window_proxy().currently_active() == Some(self.global().pipeline_id()) { self.window_proxy().unset_currently_active(can_gc); } // A hint to the JS runtime that now would be a good time to // GC any unreachable objects generated by user script, // or unattached DOM nodes. Attached DOM nodes can't be GCd yet, // as the document might be reactivated later. self.Gc(); } pub(crate) fn resume(&self, can_gc: CanGc) { // Resume timer events. self.as_global_scope().resume(); // Set the window proxy to be this object. self.window_proxy().set_currently_active(self, can_gc); // Push the document title to the compositor since we are // activating this document due to a navigation. self.Document().title_changed(); } pub(crate) fn need_emit_timeline_marker(&self, timeline_type: TimelineMarkerType) -> bool { let markers = self.devtools_markers.borrow(); markers.contains(&timeline_type) } pub(crate) fn emit_timeline_marker(&self, marker: TimelineMarker) { let sender = self.devtools_marker_sender.borrow(); let sender = sender.as_ref().expect("There is no marker sender"); sender.send(Some(marker)).unwrap(); } pub(crate) fn set_devtools_timeline_markers( &self, markers: Vec<TimelineMarkerType>, reply: IpcSender<Option<TimelineMarker>>, ) { *self.devtools_marker_sender.borrow_mut() = Some(reply); self.devtools_markers.borrow_mut().extend(markers); } pub(crate) fn drop_devtools_timeline_markers(&self, markers: Vec<TimelineMarkerType>) { let mut devtools_markers = self.devtools_markers.borrow_mut(); for marker in markers { devtools_markers.remove(&marker); } if devtools_markers.is_empty() { *self.devtools_marker_sender.borrow_mut() = None; } } pub(crate) fn set_webdriver_script_chan(&self, chan: Option<IpcSender<WebDriverJSResult>>) { *self.webdriver_script_chan.borrow_mut() = chan; } pub(crate) fn is_alive(&self) -> bool { self.current_state.get() == WindowState::Alive } // https://html.spec.whatwg.org/multipage/#top-level-browsing-context pub(crate) fn is_top_level(&self) -> bool { self.parent_info.is_none() } /// An implementation of: /// <https://drafts.csswg.org/cssom-view/#document-run-the-resize-steps> /// /// Returns true if there were any pending resize events. pub(crate) fn run_the_resize_steps(&self, can_gc: CanGc) -> bool { let Some((new_size, size_type)) = self.take_unhandled_resize_event() else { return false; }; if self.viewport_details() == new_size { return false; } let _realm = enter_realm(self); debug!( "Resizing Window for pipeline {:?} from {:?} to {new_size:?}", self.pipeline_id(), self.viewport_details(), ); self.set_viewport_details(new_size); // http://dev.w3.org/csswg/cssom-view/#resizing-viewports if size_type == WindowSizeType::Resize { let uievent = UIEvent::new( self, DOMString::from("resize"), EventBubbles::DoesNotBubble, EventCancelable::NotCancelable, Some(self), 0i32, can_gc, ); uievent.upcast::<Event>().fire(self.upcast(), can_gc); } // The document needs to be repainted, because the initial containing block // is now a different size. self.Document().set_needs_paint(true); true } /// Evaluate media query lists and report changes /// <https://drafts.csswg.org/cssom-view/#evaluate-media-queries-and-report-changes> pub(crate) fn evaluate_media_queries_and_report_changes(&self, can_gc: CanGc) { let _realm = enter_realm(self); rooted_vec!(let mut mql_list); self.media_query_lists.for_each(|mql| { if let MediaQueryListMatchState::Changed = mql.evaluate_changes() { // Recording list of changed Media Queries mql_list.push(Dom::from_ref(&*mql)); } }); // Sending change events for all changed Media Queries for mql in mql_list.iter() { let event = MediaQueryListEvent::new( &mql.global(), atom!("change"), false, false, mql.Media(), mql.Matches(), can_gc, ); event .upcast::<Event>() .fire(mql.upcast::<EventTarget>(), can_gc); } } /// Set whether to use less resources by running timers at a heavily limited rate. pub(crate) fn set_throttled(&self, throttled: bool) { self.throttled.set(throttled); if throttled { self.as_global_scope().slow_down_timers(); } else { self.as_global_scope().speed_up_timers(); } } pub(crate) fn throttled(&self) -> bool { self.throttled.get() } pub(crate) fn unminified_css_dir(&self) -> Option<String> { self.unminified_css_dir.borrow().clone() } pub(crate) fn local_script_source(&self) -> &Option<String> { &self.local_script_source } pub(crate) fn set_navigation_start(&self) { self.navigation_start.set(CrossProcessInstant::now()); } pub(crate) fn send_to_embedder(&self, msg: EmbedderMsg) { self.send_to_constellation(ScriptToConstellationMessage::ForwardToEmbedder(msg)); } pub(crate) fn send_to_constellation(&self, msg: ScriptToConstellationMessage) { self.as_global_scope() .script_to_constellation_chan() .send(msg) .unwrap(); } #[cfg(feature = "webxr")] pub(crate) fn in_immersive_xr_session(&self) -> bool { self.navigator .get() .as_ref() .and_then(|nav| nav.xr()) .is_some_and(|xr| xr.pending_or_active_session()) } #[cfg(not(feature = "webxr"))] pub(crate) fn in_immersive_xr_session(&self) -> bool { false } } impl Window { #[allow(unsafe_code)] #[allow(clippy::too_many_arguments)] pub(crate) fn new( webview_id: WebViewId, runtime: Rc<Runtime>, script_chan: Sender<MainThreadScriptMsg>, layout: Box<dyn Layout>, font_context: Arc<FontContext>, image_cache_sender: IpcSender<ImageCacheResponseMessage>, image_cache: Arc<dyn ImageCache>, resource_threads: ResourceThreads, #[cfg(feature = "bluetooth")] bluetooth_thread: IpcSender<BluetoothRequest>, mem_profiler_chan: MemProfilerChan, time_profiler_chan: TimeProfilerChan, devtools_chan: Option<IpcSender<ScriptToDevtoolsControlMsg>>, constellation_chan: ScriptToConstellationChan, control_chan: IpcSender<ScriptThreadMessage>, pipeline_id: PipelineId, parent_info: Option<PipelineId>, viewport_details: ViewportDetails, origin: MutableOrigin, creation_url: ServoUrl, top_level_creation_url: ServoUrl, navigation_start: CrossProcessInstant, webgl_chan: Option<WebGLChan>, #[cfg(feature = "webxr")] webxr_registry: Option<webxr_api::Registry>, microtask_queue: Rc<MicrotaskQueue>, compositor_api: CrossProcessCompositorApi, relayout_event: bool, unminify_js: bool, unminify_css: bool, local_script_source: Option<String>, user_content_manager: UserContentManager, player_context: WindowGLContext, #[cfg(feature = "webgpu")] gpu_id_hub: Arc<IdentityHub>, inherited_secure_context: Option<bool>, theme: Theme, ) -> DomRoot<Self> { let error_reporter = CSSErrorReporter { pipelineid: pipeline_id, script_chan: Arc::new(Mutex::new(control_chan)), }; let initial_viewport = f32_rect_to_au_rect(UntypedRect::new( Point2D::zero(), viewport_details.size.to_untyped(), )); let win = Box::new(Self { webview_id, globalscope: GlobalScope::new_inherited( pipeline_id, devtools_chan, mem_profiler_chan, time_profiler_chan, constellation_chan, resource_threads, origin, creation_url, Some(top_level_creation_url), microtask_queue, #[cfg(feature = "webgpu")] gpu_id_hub, inherited_secure_context, unminify_js, ), script_chan, layout: RefCell::new(layout), font_context, image_cache_sender, image_cache, navigator: Default::default(), location: Default::default(), history: Default::default(), indexeddb: Default::default(), custom_element_registry: Default::default(), window_proxy: Default::default(), document: Default::default(), performance: Default::default(), navigation_start: Cell::new(navigation_start), screen: Default::default(), session_storage: Default::default(), local_storage: Default::default(), status: DomRefCell::new(DOMString::new()), parent_info, dom_static: GlobalStaticData::new(), js_runtime: DomRefCell::new(Some(runtime.clone())), #[cfg(feature = "bluetooth")] bluetooth_thread, #[cfg(feature = "bluetooth")] bluetooth_extra_permission_data: BluetoothExtraPermissionData::new(), unhandled_resize_event: Default::default(), viewport_details: Cell::new(viewport_details), current_viewport_size: Cell::new(initial_viewport.to_untyped().size), layout_blocker: Cell::new(LayoutBlocker::WaitingForParse), current_state: Cell::new(WindowState::Alive), devtools_marker_sender: Default::default(), devtools_markers: Default::default(), webdriver_script_chan: Default::default(), error_reporter, media_query_lists: DOMTracker::new(), #[cfg(feature = "bluetooth")] test_runner: Default::default(), webgl_chan, #[cfg(feature = "webxr")] webxr_registry, pending_image_callbacks: Default::default(), pending_layout_images: Default::default(), pending_images_for_rasterization: Default::default(), unminified_css_dir: Default::default(), local_script_source, test_worklet: Default::default(), paint_worklet: Default::default(), exists_mut_observer: Cell::new(false), compositor_api, has_sent_idle_message: Cell::new(false), relayout_event, unminify_css, user_content_manager, player_context, throttled: Cell::new(false), layout_marker: DomRefCell::new(Rc::new(Cell::new(true))), current_event: DomRefCell::new(None), theme: Cell::new(theme), trusted_types: Default::default(), }); unsafe { WindowBinding::Wrap::<crate::DomTypeHolder>(JSContext::from_ptr(runtime.cx()), win) } } pub(crate) fn pipeline_id(&self) -> PipelineId { self.as_global_scope().pipeline_id() } /// Create a new cached instance of the given value. pub(crate) fn cache_layout_value<T>(&self, value: T) -> LayoutValue<T> where T: Copy + MallocSizeOf, { LayoutValue::new(self.layout_marker.borrow().clone(), value) } } /// An instance of a value associated with a particular snapshot of layout. This stored /// value can only be read as long as the associated layout marker that is considered /// valid. It will automatically become unavailable when the next layout operation is /// performed. #[derive(MallocSizeOf)] pub(crate) struct LayoutValue<T: MallocSizeOf> { #[ignore_malloc_size_of = "Rc is hard"] is_valid: Rc<Cell<bool>>, value: T, } #[allow(unsafe_code)] unsafe impl<T: JSTraceable + MallocSizeOf> JSTraceable for LayoutValue<T> { unsafe fn trace(&self, trc: *mut js::jsapi::JSTracer) { self.value.trace(trc) } } impl<T: Copy + MallocSizeOf> LayoutValue<T> { fn new(marker: Rc<Cell<bool>>, value: T) -> Self { LayoutValue { is_valid: marker, value, } } /// Retrieve the stored value if it is still valid. pub(crate) fn get(&self) -> Result<T, ()> { if self.is_valid.get() { return Ok(self.value); } Err(()) } } fn should_move_clip_rect(clip_rect: UntypedRect<Au>, new_viewport: UntypedRect<f32>) -> bool { let clip_rect = UntypedRect::new( Point2D::new( clip_rect.origin.x.to_f32_px(), clip_rect.origin.y.to_f32_px(), ), Size2D::new( clip_rect.size.width.to_f32_px(), clip_rect.size.height.to_f32_px(), ), ); // We only need to move the clip rect if the viewport is getting near the edge of // our preexisting clip rect. We use half of the size of the viewport as a heuristic // for "close." static VIEWPORT_SCROLL_MARGIN_SIZE: f32 = 0.5; let viewport_scroll_margin = new_viewport.size * VIEWPORT_SCROLL_MARGIN_SIZE; (clip_rect.origin.x - new_viewport.origin.x).abs() <= viewport_scroll_margin.width || (clip_rect.max_x() - new_viewport.max_x()).abs() <= viewport_scroll_margin.width || (clip_rect.origin.y - new_viewport.origin.y).abs() <= viewport_scroll_margin.height || (clip_rect.max_y() - new_viewport.max_y()).abs() <= viewport_scroll_margin.height } fn debug_reflow_events(id: PipelineId, reflow_goal: &ReflowGoal) { let goal_string = match *reflow_goal { ReflowGoal::UpdateTheRendering => "\tFull", ReflowGoal::UpdateScrollNode(..) => "\tUpdateScrollNode", ReflowGoal::LayoutQuery(ref query_msg) => match *query_msg { QueryMsg::ContentBox => "\tContentBoxQuery", QueryMsg::ContentBoxes => "\tContentBoxesQuery", QueryMsg::NodesFromPointQuery => "\tNodesFromPointQuery", QueryMsg::ClientRectQuery => "\tClientRectQuery", QueryMsg::ScrollingAreaOrOffsetQuery => "\tNodeScrollGeometryQuery", QueryMsg::ResolvedStyleQuery => "\tResolvedStyleQuery", QueryMsg::ResolvedFontStyleQuery => "\nResolvedFontStyleQuery", QueryMsg::OffsetParentQuery => "\tOffsetParentQuery", QueryMsg::StyleQuery => "\tStyleQuery", QueryMsg::TextIndexQuery => "\tTextIndexQuery", QueryMsg::ElementInnerOuterTextQuery => "\tElementInnerOuterTextQuery", QueryMsg::InnerWindowDimensionsQuery => "\tInnerWindowDimensionsQuery", }, }; println!("**** pipeline={id}\t{goal_string}"); } impl Window { // https://html.spec.whatwg.org/multipage/#dom-window-postmessage step 7. pub(crate) fn post_message( &self, target_origin: Option<ImmutableOrigin>, source_origin: ImmutableOrigin, source: &WindowProxy, data: StructuredSerializedData, ) { let this = Trusted::new(self); let source = Trusted::new(source); let task = task!(post_serialised_message: move || { let this = this.root(); let source = source.root(); let document = this.Document(); // Step 7.1. if let Some(ref target_origin) = target_origin { if !target_origin.same_origin(document.origin()) { return; } } // Steps 7.2.-7.5. let cx = this.get_cx(); let obj = this.reflector().get_jsobject(); let _ac = JSAutoRealm::new(*cx, obj.get()); rooted!(in(*cx) let mut message_clone = UndefinedValue()); if let Ok(ports) = structuredclone::read(this.upcast(), data, message_clone.handle_mut()) { // Step 7.6, 7.7 MessageEvent::dispatch_jsval( this.upcast(), this.upcast(), message_clone.handle(), Some(&source_origin.ascii_serialization()), Some(&*source), ports, CanGc::note() ); } else { // Step 4, fire messageerror. MessageEvent::dispatch_error( this.upcast(), this.upcast(), CanGc::note() ); } }); // TODO(#12718): Use the "posted message task source". self.as_global_scope() .task_manager() .dom_manipulation_task_source() .queue(task); } } #[derive(Clone, MallocSizeOf)] pub(crate) struct CSSErrorReporter { pub(crate) pipelineid: PipelineId, // Arc+Mutex combo is necessary to make this struct Sync, // which is necessary to fulfill the bounds required by the // uses of the ParseErrorReporter trait. #[ignore_malloc_size_of = "Arc is defined in libstd"] pub(crate) script_chan: Arc<Mutex<IpcSender<ScriptThreadMessage>>>, } unsafe_no_jsmanaged_fields!(CSSErrorReporter); impl ParseErrorReporter for CSSErrorReporter { fn report_error( &self, url: &UrlExtraData, location: SourceLocation, error: ContextualParseError, ) { if log_enabled!(log::Level::Info) { info!( "Url:\t{}\n{}:{} {}", url.0.as_str(), location.line, location.column, error ) } //TODO: report a real filename let _ = self .script_chan .lock() .unwrap() .send(ScriptThreadMessage::ReportCSSError( self.pipelineid, url.0.to_string(), location.line, location.column, error.to_string(), )); } } fn is_named_element_with_name_attribute(elem: &Element) -> bool { let type_ = match elem.upcast::<Node>().type_id() { NodeTypeId::Element(ElementTypeId::HTMLElement(type_)) => type_, _ => return false, }; matches!( type_, HTMLElementTypeId::HTMLEmbedElement | HTMLElementTypeId::HTMLFormElement | HTMLElementTypeId::HTMLImageElement | HTMLElementTypeId::HTMLObjectElement ) } fn is_named_element_with_id_attribute(elem: &Element) -> bool { elem.is_html_element() } #[allow(unsafe_code)] #[unsafe(no_mangle)] /// Helper for interactive debugging sessions in lldb/gdb. unsafe extern "C" fn dump_js_stack(cx: *mut RawJSContext) { unsafe { DumpJSStack(cx, true, false, false); } } impl WindowHelpers for Window { fn create_named_properties_object( cx: JSContext, proto: HandleObject, object: MutableHandleObject, ) { Self::create_named_properties_object(cx, proto, object) } }