mirror of
https://github.com/servo/servo
synced 2026-05-09 16:42:16 +02:00
This change moves the `UserContentManager` abstraction from the `ServoBuilder` to `WebViewBuilder` so that embedders can inject content for each `WebView` independently. It also adds basic support for runtime mutations to the `UserContentManager` API. Only adding new scripts is currently supported, but future changes will add support for both other mutations such as removal of scripts and addition & removal of stylesheets. Future changes could also optimize the way mutations are propagated to `ScriptThread`s by sending just the "delta" rather than the whole `UserContents` structure for each mutation. The `UserContentManager` now becomes just a convenient handle for the embedders to invoke the mutation API while the actual management of the manager's content is handled by the Constellation. The mutations are relayed to the constellation via messages. The change also separates the serialized version containg the user contents into a new `UserContent` structure so that the API cannot be misused. Testing: New unit tests have been added for the different scenarios involving UserContentManager. --------- Signed-off-by: Mukilan Thiyagarajan <mukilan@igalia.com>
442 lines
19 KiB
Rust
442 lines
19 KiB
Rust
/* 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 compositing_traits::CrossProcessPaintApi;
|
|
use compositing_traits::largest_contentful_paint_candidate::LargestContentfulPaintType;
|
|
use constellation_traits::{
|
|
KeyboardScroll, LoadData, NavigationHistoryBehavior, ScriptToConstellationSender,
|
|
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 ipc_channel::ipc::IpcReceiver;
|
|
use keyboard_types::Modifiers;
|
|
use malloc_size_of_derive::MallocSizeOf;
|
|
use media::WindowGLContext;
|
|
use net_traits::ResourceThreads;
|
|
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::units::{DevicePixel, LayoutVector2D};
|
|
use webrender_api::{ExternalScrollId, ImageKey};
|
|
|
|
/// 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<PipelineId>,
|
|
/// 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<BrowsingContextId>,
|
|
/// 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<UserContentManagerId>,
|
|
/// 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.
|
|
///
|
|
/// * <https://html.spec.whatwg.org/multipage/#active-document>
|
|
/// * <https://html.spec.whatwg.org/multipage/#fully-active>
|
|
#[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, Copy, 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 type of the largest contentful paint element.
|
|
lcp_type: LargestContentfulPaintType,
|
|
},
|
|
/// 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.
|
|
/// <https://html.spec.whatwg.org/multipage/#delaying-load-events-mode>
|
|
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<BrowsingContextId>,
|
|
/// The expected origin of the target.
|
|
target_origin: Option<ImmutableOrigin>,
|
|
/// The source origin of the message.
|
|
/// <https://html.spec.whatwg.org/multipage/#dom-messageevent-origin>
|
|
source_origin: ImmutableOrigin,
|
|
/// The data to be posted.
|
|
data: Box<StructuredSerializedData>,
|
|
},
|
|
/// 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<HistoryStateId>, ServoUrl),
|
|
/// Removes inaccesible history states.
|
|
RemoveHistoryStates(PipelineId, Vec<HistoryStateId>),
|
|
/// 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<WebViewId>),
|
|
/// 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<String>,
|
|
Option<String>,
|
|
Option<String>,
|
|
),
|
|
/// 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(IpcReceiver<WebGPUMsg>),
|
|
/// `Paint` scrolled and is updating the scroll states of the nodes in the given
|
|
/// pipeline via the Constellation.
|
|
SetScrollStates(PipelineId, FxHashMap<ExternalScrollId, LayoutVector2D>),
|
|
/// 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<ImageKey>),
|
|
/// 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 `<iframe>` to a parent pipeline.
|
|
ForwardKeyboardScroll(PipelineId, KeyboardScroll),
|
|
/// Request readiness for a screenshot from the given pipeline. The pipeline will
|
|
/// respond when it is ready to take the screenshot or will not be able to take it
|
|
/// in the future.
|
|
RequestScreenshotReadiness(WebViewId, PipelineId),
|
|
/// A response to a request to show an embedder user interface control.
|
|
EmbedderControlResponse(EmbedderControlId, EmbedderControlResponse),
|
|
/// Set the `UserContents` for the given `UserContentManagerId`. A `ScriptThread` can host many
|
|
/// `WebView`s which share the same `UserContentManager`. Only documents loaded after
|
|
/// the processing of this message will observe the new `UserContents` of the specified
|
|
/// `UserContentManagerId`.
|
|
SetUserContents(UserContentManagerId, UserContents),
|
|
/// Release all data for the given `UserContentManagerId` from the `ScriptThread`'s
|
|
/// `user_contents_for_manager_id` map.
|
|
DestroyUserContentManager(UserContentManagerId),
|
|
}
|
|
|
|
impl fmt::Debug for ScriptThreadMessage {
|
|
fn fmt(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
|
|
let variant_string: &'static str = self.into();
|
|
write!(formatter, "ConstellationControlMsg::{variant_string}")
|
|
}
|
|
}
|
|
|
|
/// Used to determine if a script has any pending asynchronous activity.
|
|
#[derive(Clone, Copy, Debug, Deserialize, PartialEq, Serialize)]
|
|
pub enum DocumentState {
|
|
/// The document has been loaded and is idle.
|
|
Idle,
|
|
/// The document is either loading or waiting on an event.
|
|
Pending,
|
|
}
|
|
|
|
/// Input events from the embedder that are sent via the `Constellation`` to the `ScriptThread`.
|
|
#[derive(Clone, Debug, Deserialize, Serialize)]
|
|
pub struct ConstellationInputEvent {
|
|
/// The hit test result of this input event, if any.
|
|
pub hit_test_result: Option<PaintHitTestResult>,
|
|
/// The pressed mouse button state of the constellation when this input
|
|
/// event was triggered.
|
|
pub pressed_mouse_buttons: u16,
|
|
/// The currently active keyboard modifiers.
|
|
pub active_keyboard_modifiers: Modifiers,
|
|
/// The [`InputEventAndId`] itself.
|
|
pub event: InputEventAndId,
|
|
}
|
|
|
|
/// All of the information necessary to create a new [`ScriptThread`] for a new [`EventLoop`].
|
|
///
|
|
/// NB: *DO NOT* add any Senders or Receivers here! pcwalton will have to rewrite your code if you
|
|
/// do! Use IPC senders and receivers instead.
|
|
#[derive(Deserialize, Serialize)]
|
|
pub struct InitialScriptState {
|
|
/// The id of the script event loop that this state will start. This is used to uniquely
|
|
/// identify an event loop.
|
|
pub id: ScriptEventLoopId,
|
|
/// The sender to use to install the `Pipeline` namespace into this process (if necessary).
|
|
pub namespace_request_sender: GenericSender<PipelineNamespaceRequest>,
|
|
/// A channel with which messages can be sent to us (the script thread).
|
|
pub constellation_to_script_sender: GenericSender<ScriptThreadMessage>,
|
|
/// A port on which messages sent by the constellation to script can be received.
|
|
pub constellation_to_script_receiver: GenericReceiver<ScriptThreadMessage>,
|
|
/// A channel on which messages can be sent to the constellation from script.
|
|
pub script_to_constellation_sender: ScriptToConstellationSender,
|
|
/// A channel which allows script to send messages directly to the Embedder
|
|
/// This will pump the embedder event loop.
|
|
pub script_to_embedder_sender: ScriptToEmbedderChan,
|
|
/// An IpcSender to the `SystemFontService` used to create a `SystemFontServiceProxy`.
|
|
pub system_font_service: SystemFontServiceProxySender,
|
|
/// A channel to the resource manager thread.
|
|
pub resource_threads: ResourceThreads,
|
|
/// A channel to the storage manager thread.
|
|
pub storage_threads: StorageThreads,
|
|
/// A channel to the bluetooth thread.
|
|
#[cfg(feature = "bluetooth")]
|
|
pub bluetooth_sender: GenericSender<BluetoothRequest>,
|
|
/// A channel to the time profiler thread.
|
|
pub time_profiler_sender: profile_traits::time::ProfilerChan,
|
|
/// A channel to the memory profiler thread.
|
|
pub memory_profiler_sender: mem::ProfilerChan,
|
|
/// A channel to the developer tools, if applicable.
|
|
pub devtools_server_sender: Option<GenericCallback<ScriptToDevtoolsControlMsg>>,
|
|
/// The ID of the pipeline namespace for this script thread.
|
|
pub pipeline_namespace_id: PipelineNamespaceId,
|
|
/// A channel to the WebGL thread used in this pipeline.
|
|
pub webgl_chan: Option<WebGLPipeline>,
|
|
/// The XR device registry
|
|
pub webxr_registry: Option<webxr_api::Registry>,
|
|
/// Access to `Paint` across a process boundary.
|
|
pub cross_process_paint_api: CrossProcessPaintApi,
|
|
/// Application window's GL Context for Media player
|
|
pub player_context: WindowGLContext,
|
|
/// A list of URLs that can access privileged internal APIs.
|
|
pub privileged_urls: Vec<ServoUrl>,
|
|
/// A copy of constellation's `UserContentManagerId` to `UserContents` map.
|
|
pub user_contents_for_manager_id: FxHashMap<UserContentManagerId, UserContents>,
|
|
}
|
|
|
|
/// Errors from executing a paint worklet
|
|
#[derive(Clone, Debug, Deserialize, Serialize)]
|
|
pub enum PaintWorkletError {
|
|
/// Execution timed out.
|
|
Timeout,
|
|
/// No such worklet.
|
|
WorkletNotFound,
|
|
}
|
|
|
|
impl From<RecvTimeoutError> for PaintWorkletError {
|
|
fn from(_: RecvTimeoutError) -> PaintWorkletError {
|
|
PaintWorkletError::Timeout
|
|
}
|
|
}
|
|
|
|
/// Execute paint code in the worklet thread pool.
|
|
pub trait Painter: SpeculativePainter {
|
|
/// <https://drafts.css-houdini.org/css-paint-api/#draw-a-paint-image>
|
|
fn draw_a_paint_image(
|
|
&self,
|
|
size: Size2D<f32, CSSPixel>,
|
|
zoom: Scale<f32, CSSPixel, DevicePixel>,
|
|
properties: Vec<(Atom, String)>,
|
|
arguments: Vec<String>,
|
|
) -> Result<DrawAPaintImageResult, PaintWorkletError>;
|
|
}
|
|
|
|
impl fmt::Debug for dyn Painter {
|
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
|
fmt.debug_tuple("Painter")
|
|
.field(&format_args!(".."))
|
|
.finish()
|
|
}
|
|
}
|
|
|
|
/// The result of executing paint code: the image together with any image URLs that need to be loaded.
|
|
///
|
|
/// TODO: this should return a WR display list. <https://github.com/servo/servo/issues/17497>
|
|
#[derive(Clone, Debug, Deserialize, MallocSizeOf, Serialize)]
|
|
pub struct DrawAPaintImageResult {
|
|
/// The image height
|
|
pub width: u32,
|
|
/// The image width
|
|
pub height: u32,
|
|
/// The image format
|
|
pub format: PixelFormat,
|
|
/// The image drawn, or None if an invalid paint image was drawn
|
|
pub image_key: Option<ImageKey>,
|
|
/// Drawing the image might have requested loading some image URLs.
|
|
pub missing_image_urls: Vec<ServoUrl>,
|
|
}
|