/* 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/. */ //! This module contains traits in script used generically in the rest of Servo. //! The traits are here instead of in script so that these modules won't have //! to depend on script. #![deny(missing_docs)] #![deny(unsafe_code)] use std::fmt; use base::cross_process_instant::CrossProcessInstant; use base::generic_channel::{GenericCallback, GenericReceiver, GenericSender}; use base::id::{ BrowsingContextId, HistoryStateId, PipelineId, PipelineNamespaceId, PipelineNamespaceRequest, ScriptEventLoopId, WebViewId, }; #[cfg(feature = "bluetooth")] use bluetooth_traits::BluetoothRequest; use canvas_traits::webgl::WebGLPipeline; use constellation_traits::{ KeyboardScroll, LoadData, NavigationHistoryBehavior, ScriptToConstellationSender, ScrollStateUpdate, StructuredSerializedData, WindowSizeType, }; use crossbeam_channel::RecvTimeoutError; use devtools_traits::ScriptToDevtoolsControlMsg; use embedder_traits::user_contents::{UserContentManagerId, UserContents}; use embedder_traits::{ EmbedderControlId, EmbedderControlResponse, FocusSequenceNumber, InputEventAndId, JavaScriptEvaluationId, MediaSessionActionType, PaintHitTestResult, ScriptToEmbedderChan, Theme, ViewportDetails, WebDriverScriptCommand, }; use euclid::{Scale, Size2D}; use fonts_traits::SystemFontServiceProxySender; use keyboard_types::Modifiers; use malloc_size_of_derive::MallocSizeOf; use media::WindowGLContext; use net_traits::ResourceThreads; use paint_api::{CrossProcessPaintApi, PinchZoomInfos}; use pixels::PixelFormat; use profile_traits::mem; use rustc_hash::FxHashMap; use serde::{Deserialize, Serialize}; use servo_config::prefs::PrefValue; use servo_url::{ImmutableOrigin, ServoUrl}; use storage_traits::StorageThreads; use storage_traits::webstorage_thread::WebStorageType; use strum::IntoStaticStr; use style_traits::{CSSPixel, SpeculativePainter}; use stylo_atoms::Atom; #[cfg(feature = "webgpu")] use webgpu_traits::WebGPUMsg; use webrender_api::ImageKey; use webrender_api::units::DevicePixel; /// The initial data required to create a new `Pipeline` attached to an existing `ScriptThread`. #[derive(Clone, Debug, Deserialize, Serialize)] pub struct NewPipelineInfo { /// The ID of the parent pipeline and frame type, if any. /// If `None`, this is a root pipeline. pub parent_info: Option, /// Id of the newly-created pipeline. pub new_pipeline_id: PipelineId, /// Id of the browsing context associated with this pipeline. pub browsing_context_id: BrowsingContextId, /// Id of the top-level browsing context associated with this pipeline. pub webview_id: WebViewId, /// Id of the opener, if any pub opener: Option, /// Network request data which will be initiated by the script thread. pub load_data: LoadData, /// Initial [`ViewportDetails`] for this layout. pub viewport_details: ViewportDetails, /// The ID of the `UserContentManager` associated with this new pipeline's `WebView`. pub user_content_manager_id: Option, /// The [`Theme`] of the new layout. pub theme: Theme, } /// When a pipeline is closed, should its browsing context be discarded too? #[derive(Clone, Copy, Debug, Deserialize, Eq, Hash, PartialEq, Serialize)] pub enum DiscardBrowsingContext { /// Discard the browsing context Yes, /// Don't discard the browsing context No, } /// Is a document fully active, active or inactive? /// A document is active if it is the current active document in its session history, /// it is fuly active if it is active and all of its ancestors are active, /// and it is inactive otherwise. /// /// * /// * #[derive(Clone, Copy, Debug, Deserialize, Eq, Hash, MallocSizeOf, PartialEq, Serialize)] pub enum DocumentActivity { /// An inactive document Inactive, /// An active but not fully active document Active, /// A fully active document FullyActive, } /// Type of recorded progressive web metric #[derive(Clone, Debug, Deserialize, Serialize)] pub enum ProgressiveWebMetricType { /// Time to first Paint FirstPaint, /// Time to first contentful paint FirstContentfulPaint, /// Time for the largest contentful paint LargestContentfulPaint { /// The pixel area of the largest contentful element. area: usize, /// The URL of the largest contentful element, if any. url: Option, }, /// Time to interactive TimeToInteractive, } impl ProgressiveWebMetricType { /// Returns the area if the metric type is LargestContentfulPaint pub fn area(&self) -> usize { match self { ProgressiveWebMetricType::LargestContentfulPaint { area, .. } => *area, _ => 0, } } } /// The reason why the pipeline id of an iframe is being updated. #[derive(Clone, Copy, Debug, Deserialize, Eq, Hash, MallocSizeOf, PartialEq, Serialize)] pub enum UpdatePipelineIdReason { /// The pipeline id is being updated due to a navigation. Navigation, /// The pipeline id is being updated due to a history traversal. Traversal, } /// Messages sent to the `ScriptThread` event loop from the `Constellation`, `Paint`, and (for /// now) `Layout`. #[derive(Deserialize, IntoStaticStr, Serialize)] pub enum ScriptThreadMessage { /// Span a new `Pipeline` in this `ScriptThread` and start fetching the contents /// according to the provided `LoadData`. This will ultimately create a `Window` /// and all associated data structures such as `Layout` in the `ScriptThread`. SpawnPipeline(NewPipelineInfo), /// Takes the associated window proxy out of "delaying-load-events-mode", /// used if a scheduled navigated was refused by the embedder. /// StopDelayingLoadEventsMode(PipelineId), /// Window resized. Sends a DOM event eventually, but first we combine events. Resize(PipelineId, ViewportDetails, WindowSizeType), /// Theme changed. ThemeChange(PipelineId, Theme), /// Notifies script that window has been resized but to not take immediate action. ResizeInactive(PipelineId, ViewportDetails), /// Window switched from fullscreen mode. ExitFullScreen(PipelineId), /// Notifies the script that the document associated with this pipeline should 'unload'. UnloadDocument(PipelineId), /// Notifies the script that a pipeline should be closed. ExitPipeline(WebViewId, PipelineId, DiscardBrowsingContext), /// Notifies the script that the whole thread should be closed. ExitScriptThread, /// Sends a DOM event. SendInputEvent(WebViewId, PipelineId, ConstellationInputEvent), /// Request that the given pipeline refresh the cursor by doing a hit test at the most /// recently hovered cursor position and resetting the cursor. This happens after a /// display list update is rendered. RefreshCursor(PipelineId), /// Requests that the script thread immediately send the constellation the title of a pipeline. GetTitle(PipelineId), /// Notifies script thread of a change to one of its document's activity SetDocumentActivity(PipelineId, DocumentActivity), /// Set whether to use less resources by running timers at a heavily limited rate. SetThrottled(WebViewId, PipelineId, bool), /// Notify the containing iframe (in PipelineId) that the nested browsing context (BrowsingContextId) is throttled. SetThrottledInContainingIframe(WebViewId, PipelineId, BrowsingContextId, bool), /// Notifies script thread that a url should be loaded in this iframe. /// PipelineId is for the parent, BrowsingContextId is for the nested browsing context NavigateIframe( PipelineId, BrowsingContextId, LoadData, NavigationHistoryBehavior, ), /// Post a message to a given window. PostMessage { /// The target of the message. target: PipelineId, /// The webview associated with the source pipeline. source_webview: WebViewId, /// The ancestry of browsing context associated with the source, /// starting with the source itself. source_with_ancestry: Vec, /// The expected origin of the target. target_origin: Option, /// The source origin of the message. /// source_origin: ImmutableOrigin, /// The data to be posted. data: Box, }, /// Updates the current pipeline ID of a given iframe. /// First PipelineId is for the parent, second is the new PipelineId for the frame. UpdatePipelineId( PipelineId, BrowsingContextId, WebViewId, PipelineId, UpdatePipelineIdReason, ), /// Updates the history state and url of a given pipeline. UpdateHistoryState(PipelineId, Option, ServoUrl), /// Removes inaccesible history states. RemoveHistoryStates(PipelineId, Vec), /// Set an iframe to be focused. Used when an element in an iframe gains focus. /// PipelineId is for the parent, BrowsingContextId is for the nested browsing context FocusIFrame(PipelineId, BrowsingContextId, FocusSequenceNumber), /// Focus the document. Used when the container gains focus. FocusDocument(PipelineId, FocusSequenceNumber), /// Notifies that the document's container (e.g., an iframe) is not included /// in the top-level browsing context's focus chain (not considering system /// focus) anymore. /// /// Obviously, this message is invalid for a top-level document. Unfocus(PipelineId, FocusSequenceNumber), /// Passes a webdriver command to the script thread for execution WebDriverScriptCommand(PipelineId, WebDriverScriptCommand), /// Notifies script thread that all animations are done TickAllAnimations(Vec), /// Notifies the script thread that a new Web font has been loaded, and thus the page should be /// reflowed. WebFontLoaded(PipelineId, bool /* success */), /// Cause a `load` event to be dispatched at the appropriate iframe element. DispatchIFrameLoadEvent { /// The frame that has been marked as loaded. target: BrowsingContextId, /// The pipeline that contains a frame loading the target pipeline. parent: PipelineId, /// The pipeline that has completed loading. child: PipelineId, }, /// Cause a `storage` event to be dispatched at the appropriate window. /// The strings are key, old value and new value. DispatchStorageEvent( PipelineId, WebStorageType, ServoUrl, Option, Option, Option, ), /// Report an error from a CSS parser for the given pipeline ReportCSSError(PipelineId, String, u32, u32, String), /// Reload the given page. Reload(PipelineId), /// Notifies the script thread about a new recorded paint metric. PaintMetric( PipelineId, ProgressiveWebMetricType, CrossProcessInstant, bool, /* first_reflow */ ), /// Notifies the media session about a user requested media session action. MediaSessionAction(PipelineId, MediaSessionActionType), /// Notifies script thread that WebGPU server has started #[cfg(feature = "webgpu")] SetWebGPUPort(GenericReceiver), /// `Paint` scrolled and is updating the scroll states of the nodes in the given /// pipeline via the Constellation. SetScrollStates(PipelineId, ScrollStateUpdate), /// Evaluate the given JavaScript and return a result via a corresponding message /// to the Constellation. EvaluateJavaScript(WebViewId, PipelineId, JavaScriptEvaluationId, String), /// A new batch of keys for the image cache for the specific pipeline. SendImageKeysBatch(PipelineId, Vec), /// Preferences were updated in the parent process. PreferencesUpdated(Vec<(String, PrefValue)>), /// Notify the `ScriptThread` that the Servo renderer is no longer waiting on /// asynchronous image uploads for the given `Pipeline`. These are mainly used /// by canvas to perform uploads while the display list is being built. NoLongerWaitingOnAsychronousImageUpdates(PipelineId), /// Forward a keyboard scroll operation from an `