Files
servo/ports/servoshell/running_app_state.rs
Martin Robinson 824f551f03 Rename IOCompositor to Paint (#41176)
For a long time, the "Compositor" hasn't done any compositing. This is
handled by WebRender. In addition the "Compositor" does many other
tasks. This change renames `IOCompositor` to `Paint`.

`Paint` is Servo's paint subsystem and contains multiple `Painter`s.
This change does not rename the crate; that will be done in a
followup change.

Testing: This just renames types and updates comments, so no new tests
are necessary.

---------

Signed-off-by: Martin Robinson <mrobinson@igalia.com>
2025-12-10 15:09:49 +00:00

805 lines
29 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/. */
//! Shared state and methods for desktop and EGL implementations.
use std::cell::{Cell, Ref, RefCell};
use std::collections::HashMap;
use std::collections::hash_map::Entry;
use std::rc::Rc;
use crossbeam_channel::{Receiver, Sender, unbounded};
use euclid::Rect;
use image::{DynamicImage, ImageFormat, RgbaImage};
use log::{error, info, warn};
use servo::{
AllowOrDenyRequest, AuthenticationRequest, CSSPixel, DeviceIntPoint, DeviceIntSize,
EmbedderControl, EmbedderControlId, EventLoopWaker, GamepadHapticEffectType, GenericSender,
InputEvent, InputEventId, InputEventResult, IpcSender, JSValue, LoadStatus, MediaSessionEvent,
PermissionRequest, PrefValue, ScreenshotCaptureError, Servo, ServoDelegate, ServoError,
TraversalId, WebDriverCommandMsg, WebDriverJSResult, WebDriverLoadStatus,
WebDriverScriptCommand, WebDriverSenders, WebView, WebViewBuilder, WebViewDelegate, WebViewId,
pref,
};
use url::Url;
use crate::GamepadSupport;
use crate::prefs::{EXPERIMENTAL_PREFS, ServoShellPreferences};
use crate::webdriver::WebDriverEmbedderControls;
use crate::window::{PlatformWindow, ServoShellWindow, ServoShellWindowId};
#[derive(Default)]
pub struct WebViewCollection {
/// List of top-level browsing contexts.
/// Modified by EmbedderMsg::WebViewOpened and EmbedderMsg::WebViewClosed,
/// and we exit if it ever becomes empty.
webviews: HashMap<WebViewId, WebView>,
/// The order in which the webviews were created.
pub(crate) creation_order: Vec<WebViewId>,
/// The [`WebView`] that is currently active. This is the [`WebView`] that is shown and has
/// input focus.
active_webview_id: Option<WebViewId>,
}
impl WebViewCollection {
pub fn add(&mut self, webview: WebView) {
let id = webview.id();
self.creation_order.push(id);
self.webviews.insert(id, webview);
}
/// Removes a webview from the collection by [`WebViewId`]. If the removed [`WebView`] was the active
/// [`WebView`] then the next newest [`WebView`] will be activated.
pub fn remove(&mut self, id: WebViewId) -> Option<WebView> {
self.creation_order.retain(|&webview_id| webview_id != id);
let removed_webview = self.webviews.remove(&id);
if self.active_webview_id == Some(id) {
self.active_webview_id = None;
if let Some(newest) = self.creation_order.last() {
self.activate_webview(*newest);
}
}
removed_webview
}
pub fn get(&self, id: WebViewId) -> Option<&WebView> {
self.webviews.get(&id)
}
pub fn contains(&self, id: WebViewId) -> bool {
self.webviews.contains_key(&id)
}
pub fn active(&self) -> Option<&WebView> {
self.active_webview_id.and_then(|id| self.webviews.get(&id))
}
pub fn active_id(&self) -> Option<WebViewId> {
self.active_webview_id
}
/// Gets a reference to the most recently created webview, if any.
pub fn newest(&self) -> Option<&WebView> {
self.creation_order
.last()
.and_then(|id| self.webviews.get(id))
}
pub fn all_in_creation_order(&self) -> impl Iterator<Item = (WebViewId, &WebView)> {
self.creation_order
.iter()
.filter_map(move |id| self.webviews.get(id).map(|webview| (*id, webview)))
}
/// Returns an iterator over all webview references (in arbitrary order).
pub fn values(&self) -> impl Iterator<Item = &WebView> {
self.webviews.values()
}
/// Returns true if the collection contains no webviews.
pub fn is_empty(&self) -> bool {
self.webviews.is_empty()
}
pub(crate) fn activate_webview(&mut self, id_to_activate: WebViewId) {
assert!(self.creation_order.contains(&id_to_activate));
self.active_webview_id = Some(id_to_activate);
for (webview_id, webview) in self.all_in_creation_order() {
if id_to_activate == webview_id {
webview.show();
webview.focus();
} else {
webview.hide();
webview.blur();
}
}
}
pub(crate) fn activate_webview_by_index(&mut self, index: usize) {
self.activate_webview(
*self
.creation_order
.get(index)
.expect("Tried to activate an unknown WebView"),
);
}
}
/// A command received via the user interacting with the user interface.
#[cfg_attr(any(target_os = "android", target_env = "ohos"), expect(dead_code))]
pub(crate) enum UserInterfaceCommand {
Go(String),
Back,
Forward,
Reload,
ReloadAll,
NewWebView,
CloseWebView(WebViewId),
NewWindow,
}
pub(crate) struct RunningAppState {
/// Gamepad support, which may be `None` if it failed to initialize.
gamepad_support: RefCell<Option<GamepadSupport>>,
/// The [`WebDriverSenders`] used to reply to pending WebDriver requests.
pub(crate) webdriver_senders: RefCell<WebDriverSenders>,
/// When running in WebDriver mode, [`WebDriverEmbedderControls`] is a virtual container
/// for all embedder controls. This overrides the normal behavior where these controls
/// are shown in the GUI or not processed at all in headless mode.
pub(crate) webdriver_embedder_controls: WebDriverEmbedderControls,
/// A [`HashMap`] of pending WebDriver events. It is the WebDriver embedder's responsibility
/// to inform the WebDriver server when the event has been fully handled. This map is used
/// to report back to WebDriver when that happens.
pub(crate) pending_webdriver_events: RefCell<HashMap<InputEventId, Sender<()>>>,
/// A [`Receiver`] for receiving commands from a running WebDriver server, if WebDriver
/// was enabled.
pub(crate) webdriver_receiver: Option<Receiver<WebDriverCommandMsg>>,
/// servoshell specific preferences created during startup of the application.
pub(crate) servoshell_preferences: ServoShellPreferences,
/// A handle to the Servo instance.
pub(crate) servo: Servo,
/// Whether or not the application has achieved stable image output. This is used
/// for the `exit_after_stable_image` option.
pub(crate) achieved_stable_image: Rc<Cell<bool>>,
/// Whether or not program exit has been triggered. This means that all windows
/// will be destroyed and shutdown will start at the end of the current event loop.
exit_scheduled: Cell<bool>,
/// Whether the user has enabled experimental preferences.
experimental_preferences_enabled: Cell<bool>,
/// The set of [`ServoShellWindow`]s that currently exist for this instance of servoshell.
// This is the last field of the struct to ensure that windows are dropped *after* all
// other references to the relevant rendering contexts have been destroyed.
// See https://github.com/servo/servo/issues/36711.
windows: RefCell<HashMap<ServoShellWindowId, Rc<ServoShellWindow>>>,
}
impl RunningAppState {
pub(crate) fn new(
servo: Servo,
servoshell_preferences: ServoShellPreferences,
event_loop_waker: Box<dyn EventLoopWaker>,
) -> Self {
servo.set_delegate(Rc::new(ServoShellServoDelegate));
let gamepad_support = if pref!(dom_gamepad_enabled) {
GamepadSupport::maybe_new()
} else {
None
};
let webdriver_receiver = servoshell_preferences.webdriver_port.get().map(|port| {
let (embedder_sender, embedder_receiver) = unbounded();
webdriver_server::start_server(port, embedder_sender, event_loop_waker);
embedder_receiver
});
let experimental_preferences_enabled =
Cell::new(servoshell_preferences.experimental_preferences_enabled);
Self {
windows: Default::default(),
gamepad_support: RefCell::new(gamepad_support),
webdriver_senders: RefCell::default(),
webdriver_embedder_controls: Default::default(),
pending_webdriver_events: Default::default(),
webdriver_receiver,
servoshell_preferences,
servo,
achieved_stable_image: Default::default(),
exit_scheduled: Default::default(),
experimental_preferences_enabled,
}
}
pub(crate) fn open_window(
self: &Rc<Self>,
platform_window: Rc<dyn PlatformWindow>,
initial_url: Url,
) {
let window = Rc::new(ServoShellWindow::new(platform_window));
window.create_and_activate_toplevel_webview(self.clone(), initial_url);
self.windows.borrow_mut().insert(window.id(), window);
}
pub(crate) fn windows<'a>(
&'a self,
) -> Ref<'a, HashMap<ServoShellWindowId, Rc<ServoShellWindow>>> {
self.windows.borrow()
}
/// Get any [`ServoShellWindow`] from this state's collection of windows. This is used for
/// WebDriver, which currently doesn't have great support for per-window operation.
pub(crate) fn any_window(&self) -> Rc<ServoShellWindow> {
self.windows
.borrow()
.values()
.next()
.expect("Should always have at least one window open when running WebDriver")
.clone()
}
pub(crate) fn focused_window(&self) -> Option<Rc<ServoShellWindow>> {
self.windows
.borrow()
.values()
.find(|window| window.focused())
.cloned()
}
#[cfg_attr(any(target_os = "android", target_env = "ohos"), expect(dead_code))]
pub(crate) fn window(&self, id: ServoShellWindowId) -> Option<Rc<ServoShellWindow>> {
self.windows.borrow().get(&id).cloned()
}
pub(crate) fn webview_by_id(&self, webview_id: WebViewId) -> Option<WebView> {
self.maybe_window_for_webview_id(webview_id)?
.webview_by_id(webview_id)
}
pub(crate) fn webdriver_receiver(&self) -> Option<&Receiver<WebDriverCommandMsg>> {
self.webdriver_receiver.as_ref()
}
pub(crate) fn servo(&self) -> &Servo {
&self.servo
}
pub(crate) fn schedule_exit(&self) {
// When explicitly required to shutdown, unset webdriver port
// which allows normal shutdown.
// Note that when not explicitly required to shutdown, we still keep Servo alive
// when all tabs are closed when `webdriver_port` enabled, which is necessary
// to run wpt test using servodriver.
self.servoshell_preferences.webdriver_port.set(None);
self.exit_scheduled.set(true);
}
#[cfg_attr(any(target_os = "android", target_env = "ohos"), expect(dead_code))]
pub(crate) fn experimental_preferences_enabled(&self) -> bool {
self.experimental_preferences_enabled.get()
}
#[cfg_attr(any(target_os = "android", target_env = "ohos"), expect(dead_code))]
pub(crate) fn set_experimental_preferences_enabled(&self, new_value: bool) {
let old_value = self.experimental_preferences_enabled.replace(new_value);
if old_value == new_value {
return;
}
for pref in EXPERIMENTAL_PREFS {
self.servo.set_preference(pref, PrefValue::Bool(new_value));
}
}
/// Spins the internal application event loop.
///
/// - Notifies Servo about incoming gamepad events
/// - Spin the Servo event loop, which will update Servo's embedding layer and trigger
/// delegate methods.
///
/// Returns true if the event loop should continue spinning and false if it should exit.
pub(crate) fn spin_event_loop(self: &Rc<Self>) -> bool {
self.handle_webdriver_messages();
if pref!(dom_gamepad_enabled) {
self.handle_gamepad_events();
}
self.servo.spin_event_loop();
for window in self.windows.borrow().values() {
window.update_and_request_repaint_if_necessary(self);
}
if self.servoshell_preferences.exit_after_stable_image && self.achieved_stable_image.get() {
self.schedule_exit();
}
// When a ServoShellWindow has no more WebViews, close it. When no more windows are open, exit
// the application. Do not do this when running WebDriver, which expects to keep running with
// no WebView open.
if self.servoshell_preferences.webdriver_port.get().is_none() {
self.windows
.borrow_mut()
.retain(|_, window| !self.exit_scheduled.get() && !window.should_close());
if self.windows.borrow().is_empty() {
self.schedule_exit()
}
}
!self.exit_scheduled.get()
}
#[cfg_attr(any(target_os = "android", target_env = "ohos"), expect(dead_code))]
pub(crate) fn foreach_window_and_interface_commands(
self: &Rc<Self>,
callback: impl Fn(&ServoShellWindow, Vec<UserInterfaceCommand>),
) {
// We clone here to avoid a double borrow. User interface commands can update the list of windows.
let windows: Vec<_> = self.windows.borrow().values().cloned().collect();
for window in windows {
callback(
&window,
window.platform_window().take_user_interface_commands(),
)
}
}
pub(crate) fn maybe_window_for_webview_id(
&self,
webview_id: WebViewId,
) -> Option<Rc<ServoShellWindow>> {
for window in self.windows.borrow().values() {
if window.contains_webview(webview_id) {
return Some(window.clone());
}
}
None
}
pub(crate) fn window_for_webview_id(&self, webview_id: WebViewId) -> Rc<ServoShellWindow> {
self.maybe_window_for_webview_id(webview_id)
.expect("Looking for unexpected WebView: {webview_id:?}")
}
pub(crate) fn platform_window_for_webview_id(
&self,
webview_id: WebViewId,
) -> Rc<dyn PlatformWindow> {
self.window_for_webview_id(webview_id).platform_window()
}
/// If we are exiting after achieving a stable image or we want to save the display of the
/// [`WebView`] to an image file, request a screenshot of the [`WebView`].
fn maybe_request_screenshot(&self, webview: WebView) {
let output_path = self.servoshell_preferences.output_image_path.clone();
if !self.servoshell_preferences.exit_after_stable_image && output_path.is_none() {
return;
}
// Never request more than a single screenshot for now.
let achieved_stable_image = self.achieved_stable_image.clone();
if achieved_stable_image.get() {
return;
}
webview.take_screenshot(None, move |image| {
achieved_stable_image.set(true);
let Some(output_path) = output_path else {
return;
};
let image = match image {
Ok(image) => image,
Err(error) => {
error!("Could not take screenshot: {error:?}");
return;
},
};
let image_format = ImageFormat::from_path(&output_path).unwrap_or(ImageFormat::Png);
if let Err(error) =
DynamicImage::ImageRgba8(image).save_with_format(output_path, image_format)
{
error!("Failed to save screenshot: {error}.");
}
});
}
pub(crate) fn set_pending_traversal(
&self,
traversal_id: TraversalId,
sender: GenericSender<WebDriverLoadStatus>,
) {
self.webdriver_senders
.borrow_mut()
.pending_traversals
.insert(traversal_id, sender);
}
pub(crate) fn set_load_status_sender(
&self,
webview_id: WebViewId,
sender: GenericSender<WebDriverLoadStatus>,
) {
self.webdriver_senders
.borrow_mut()
.load_status_senders
.insert(webview_id, sender);
}
fn remove_load_status_sender(&self, webview_id: WebViewId) {
self.webdriver_senders
.borrow_mut()
.load_status_senders
.remove(&webview_id);
}
fn set_script_command_interrupt_sender(&self, sender: Option<IpcSender<WebDriverJSResult>>) {
self.webdriver_senders
.borrow_mut()
.script_evaluation_interrupt_sender = sender;
}
pub(crate) fn handle_webdriver_input_event(
&self,
webview_id: WebViewId,
input_event: InputEvent,
response_sender: Option<Sender<()>>,
) {
if let Some(webview) = self.webview_by_id(webview_id) {
let event_id = webview.notify_input_event(input_event);
if let Some(response_sender) = response_sender {
self.pending_webdriver_events
.borrow_mut()
.insert(event_id, response_sender);
}
} else {
error!("Could not find WebView ({webview_id:?}) for WebDriver event: {input_event:?}");
};
}
pub(crate) fn handle_webdriver_screenshot(
&self,
webview_id: WebViewId,
rect: Option<Rect<f32, CSSPixel>>,
result_sender: Sender<Result<RgbaImage, ScreenshotCaptureError>>,
) {
if let Some(webview) = self.webview_by_id(webview_id) {
let rect = rect.map(|rect| rect.to_box2d().into());
webview.take_screenshot(rect, move |result| {
if let Err(error) = result_sender.send(result) {
warn!("Failed to send response to TakeScreenshot: {error}");
}
});
} else if let Err(error) =
result_sender.send(Err(ScreenshotCaptureError::WebViewDoesNotExist))
{
error!("Failed to send response to TakeScreenshot: {error}");
}
}
pub(crate) fn handle_webdriver_script_command(&self, script_command: &WebDriverScriptCommand) {
match script_command {
WebDriverScriptCommand::ExecuteScript(_webview_id, response_sender) |
WebDriverScriptCommand::ExecuteAsyncScript(_webview_id, response_sender) => {
// Give embedder a chance to interrupt the script command.
// Webdriver only handles 1 script command at a time, so we can
// safely set a new interrupt sender and remove the previous one here.
self.set_script_command_interrupt_sender(Some(response_sender.clone()));
},
WebDriverScriptCommand::AddLoadStatusSender(webview_id, load_status_sender) => {
self.set_load_status_sender(*webview_id, load_status_sender.clone());
},
WebDriverScriptCommand::RemoveLoadStatusSender(webview_id) => {
self.remove_load_status_sender(*webview_id);
},
_ => {
self.set_script_command_interrupt_sender(None);
},
}
}
pub(crate) fn handle_webdriver_load_url(
&self,
webview_id: WebViewId,
url: Url,
load_status_sender: GenericSender<WebDriverLoadStatus>,
) {
let Some(webview) = self.webview_by_id(webview_id) else {
return;
};
self.platform_window_for_webview_id(webview_id)
.dismiss_embedder_controls_for_webview(webview_id);
info!("Loading URL in webview {}: {}", webview_id, url);
self.set_load_status_sender(webview_id, load_status_sender);
webview.load(url);
}
pub(crate) fn handle_gamepad_events(&self) {
let Some(active_webview) = self
.focused_window()
.and_then(|window| window.active_webview())
else {
return;
};
if let Some(gamepad_support) = self.gamepad_support.borrow_mut().as_mut() {
gamepad_support.handle_gamepad_events(active_webview);
}
}
/// Interrupt any ongoing WebDriver-based script evaluation.
///
/// From <https://w3c.github.io/webdriver/#dfn-execute-a-function-body>:
/// > The rules to execute a function body are as follows. The algorithm returns
/// > an ECMAScript completion record.
/// >
/// > If at any point during the algorithm a user prompt appears, immediately return
/// > Completion { Type: normal, Value: null, Target: empty }, but continue to run the
/// > other steps of this algorithm in parallel.
fn interrupt_webdriver_script_evaluation(&self) {
if let Some(sender) = &self
.webdriver_senders
.borrow()
.script_evaluation_interrupt_sender
{
sender.send(Ok(JSValue::Null)).unwrap_or_else(|err| {
info!(
"Notify dialog appear failed. Maybe the channel to webdriver is closed: {err}"
);
});
}
}
}
impl WebViewDelegate for RunningAppState {
fn screen_geometry(&self, webview: WebView) -> Option<servo::ScreenGeometry> {
Some(
self.platform_window_for_webview_id(webview.id())
.screen_geometry(),
)
}
fn notify_status_text_changed(&self, webview: WebView, _status: Option<String>) {
self.window_for_webview_id(webview.id()).set_needs_update();
}
fn notify_history_changed(&self, webview: WebView, _entries: Vec<Url>, _current: usize) {
self.window_for_webview_id(webview.id()).set_needs_update();
}
fn notify_page_title_changed(&self, webview: WebView, _: Option<String>) {
self.window_for_webview_id(webview.id()).set_needs_update();
}
fn notify_traversal_complete(&self, _webview: WebView, traversal_id: TraversalId) {
let mut webdriver_state = self.webdriver_senders.borrow_mut();
if let Entry::Occupied(entry) = webdriver_state.pending_traversals.entry(traversal_id) {
let sender = entry.remove();
let _ = sender.send(WebDriverLoadStatus::Complete);
}
}
fn request_move_to(&self, webview: WebView, new_position: DeviceIntPoint) {
self.platform_window_for_webview_id(webview.id())
.set_position(new_position);
}
fn request_resize_to(&self, webview: WebView, requested_outer_size: DeviceIntSize) {
self.platform_window_for_webview_id(webview.id())
.request_resize(&webview, requested_outer_size);
}
fn request_authentication(
&self,
webview: WebView,
authentication_request: AuthenticationRequest,
) {
self.platform_window_for_webview_id(webview.id())
.show_http_authentication_dialog(webview.id(), authentication_request);
}
fn request_open_auxiliary_webview(&self, parent_webview: WebView) -> Option<WebView> {
let window = self.window_for_webview_id(parent_webview.id());
let platform_window = window.platform_window();
let webview =
WebViewBuilder::new_auxiliary(&self.servo, platform_window.rendering_context())
.hidpi_scale_factor(platform_window.hidpi_scale_factor())
.delegate(parent_webview.delegate())
.build();
webview.notify_theme_change(platform_window.theme());
window.add_webview(webview.clone());
// When WebDriver is enabled, do not focus and raise the WebView to the top,
// as that is what the specification expects. Otherwise, we would like `window.open()`
// to create a new foreground tab
if self.servoshell_preferences.webdriver_port.get().is_none() {
window.activate_webview(webview.id());
} else {
webview.hide();
}
Some(webview)
}
fn notify_closed(&self, webview: WebView) {
self.window_for_webview_id(webview.id())
.close_webview(webview.id())
}
fn notify_input_event_handled(
&self,
webview: WebView,
id: InputEventId,
result: InputEventResult,
) {
self.platform_window_for_webview_id(webview.id())
.notify_input_event_handled(&webview, id, result);
if let Some(response_sender) = self.pending_webdriver_events.borrow_mut().remove(&id) {
let _ = response_sender.send(());
}
}
fn notify_cursor_changed(&self, webview: WebView, cursor: servo::Cursor) {
self.platform_window_for_webview_id(webview.id())
.set_cursor(cursor);
}
fn notify_load_status_changed(&self, webview: WebView, status: LoadStatus) {
self.window_for_webview_id(webview.id()).set_needs_update();
if status == LoadStatus::Complete {
if let Some(sender) = self
.webdriver_senders
.borrow_mut()
.load_status_senders
.remove(&webview.id())
{
let _ = sender.send(WebDriverLoadStatus::Complete);
}
self.maybe_request_screenshot(webview);
}
}
fn notify_fullscreen_state_changed(&self, webview: WebView, fullscreen_state: bool) {
self.platform_window_for_webview_id(webview.id())
.set_fullscreen(fullscreen_state);
}
fn show_bluetooth_device_dialog(
&self,
webview: WebView,
devices: Vec<String>,
response_sender: GenericSender<Option<String>>,
) {
self.platform_window_for_webview_id(webview.id())
.show_bluetooth_device_dialog(webview.id(), devices, response_sender);
}
fn request_permission(&self, webview: WebView, permission_request: PermissionRequest) {
self.platform_window_for_webview_id(webview.id())
.show_permission_dialog(webview.id(), permission_request);
}
fn notify_new_frame_ready(&self, webview: WebView) {
self.window_for_webview_id(webview.id()).set_needs_repaint();
}
fn play_gamepad_haptic_effect(
&self,
_webview: WebView,
index: usize,
effect_type: GamepadHapticEffectType,
effect_complete_sender: IpcSender<bool>,
) {
match self.gamepad_support.borrow_mut().as_mut() {
Some(gamepad_support) => {
gamepad_support.play_haptic_effect(index, effect_type, effect_complete_sender);
},
None => {
let _ = effect_complete_sender.send(false);
},
}
}
fn stop_gamepad_haptic_effect(
&self,
_webview: WebView,
index: usize,
haptic_stop_sender: IpcSender<bool>,
) {
let stopped = match self.gamepad_support.borrow_mut().as_mut() {
Some(gamepad_support) => gamepad_support.stop_haptic_effect(index),
None => false,
};
let _ = haptic_stop_sender.send(stopped);
}
fn show_embedder_control(&self, webview: WebView, embedder_control: EmbedderControl) {
if self.servoshell_preferences.webdriver_port.get().is_some() {
if matches!(&embedder_control, EmbedderControl::SimpleDialog(..)) {
self.interrupt_webdriver_script_evaluation();
// Dialogs block the page load, so need need to notify WebDriver
if let Some(sender) = self
.webdriver_senders
.borrow_mut()
.load_status_senders
.get(&webview.id())
{
let _ = sender.send(WebDriverLoadStatus::Blocked);
};
}
self.webdriver_embedder_controls
.show_embedder_control(webview.id(), embedder_control);
return;
}
self.window_for_webview_id(webview.id())
.show_embedder_control(webview, embedder_control);
}
fn hide_embedder_control(&self, webview: WebView, embedder_control_id: EmbedderControlId) {
if self.servoshell_preferences.webdriver_port.get().is_some() {
self.webdriver_embedder_controls
.hide_embedder_control(webview.id(), embedder_control_id);
return;
}
self.window_for_webview_id(webview.id())
.hide_embedder_control(webview, embedder_control_id);
}
fn notify_favicon_changed(&self, webview: WebView) {
self.window_for_webview_id(webview.id())
.notify_favicon_changed(webview);
}
fn notify_media_session_event(&self, webview: WebView, event: MediaSessionEvent) {
self.platform_window_for_webview_id(webview.id())
.notify_media_session_event(event);
}
fn notify_crashed(&self, webview: WebView, reason: String, backtrace: Option<String>) {
self.platform_window_for_webview_id(webview.id())
.notify_crashed(webview, reason, backtrace);
}
}
struct ServoShellServoDelegate;
impl ServoDelegate for ServoShellServoDelegate {
fn notify_devtools_server_started(&self, port: u16, _token: String) {
info!("Devtools Server running on port {port}");
}
fn request_devtools_connection(&self, request: AllowOrDenyRequest) {
request.allow();
}
fn notify_error(&self, error: ServoError) {
error!("Saw Servo error: {error:?}!");
}
}