use std::default::Default;
use embedder_traits::ViewportDetails;
use layout_api::IFrameSizes;
use paint_api::PinchZoomInfos;
use rustc_hash::FxHashMap;
use script_bindings::script_runtime::CanGc;
use servo_base::id::BrowsingContextId;
use servo_constellation_traits::{IFrameSizeMsg, ScriptToConstellationMessage, WindowSizeType};
use crate::dom::bindings::inheritance::Castable;
use crate::dom::bindings::root::{Dom, DomRoot};
use crate::dom::html::htmliframeelement::HTMLIFrameElement;
use crate::dom::node::{Node, ShadowIncluding};
use crate::dom::types::{Document, Window};
use crate::script_thread::with_script_thread;
#[derive(JSTraceable, MallocSizeOf)]
#[cfg_attr(crown, crown::unrooted_must_root_lint::must_root)]
pub(crate) struct IFrame {
pub(crate) element: Dom<HTMLIFrameElement>,
#[no_trace]
pub(crate) size: Option<ViewportDetails>,
}
#[derive(Default, JSTraceable, MallocSizeOf)]
#[cfg_attr(crown, crown::unrooted_must_root_lint::must_root)]
pub(crate) struct IFrameCollection {
iframes: Vec<IFrame>,
invalid: bool,
}
impl IFrameCollection {
pub(crate) fn new() -> Self {
Self {
iframes: vec![],
invalid: true,
}
}
pub(crate) fn invalidate(&mut self) {
self.invalid = true;
}
pub(crate) fn validate(&mut self, document: &Document) {
if !self.invalid {
return;
}
let document_node = DomRoot::from_ref(document.upcast::<Node>());
let mut old_sizes: FxHashMap<_, _> = self
.iframes
.iter()
.filter_map(
|iframe| match (iframe.element.browsing_context_id(), iframe.size) {
(Some(browsing_context_id), Some(size)) => Some((browsing_context_id, size)),
_ => None,
},
)
.collect();
self.iframes = document_node
.traverse_preorder(ShadowIncluding::Yes)
.filter_map(DomRoot::downcast::<HTMLIFrameElement>)
.map(|element| {
let size = element
.browsing_context_id()
.and_then(|browsing_context_id| old_sizes.remove(&browsing_context_id));
IFrame {
element: element.as_traced(),
size,
}
})
.collect();
self.invalid = false;
}
pub(crate) fn get(&self, browsing_context_id: BrowsingContextId) -> Option<&IFrame> {
self.iframes
.iter()
.find(|iframe| iframe.element.browsing_context_id() == Some(browsing_context_id))
}
pub(crate) fn get_mut(
&mut self,
browsing_context_id: BrowsingContextId,
) -> Option<&mut IFrame> {
self.iframes
.iter_mut()
.find(|iframe| iframe.element.browsing_context_id() == Some(browsing_context_id))
}
pub(crate) fn set_viewport_details(
&mut self,
browsing_context_id: BrowsingContextId,
new_size: ViewportDetails,
) -> Option<ViewportDetails> {
self.get_mut(browsing_context_id)
.expect("Tried to set a size for an unknown <iframe>")
.size
.replace(new_size)
}
pub(crate) fn handle_new_iframe_sizes_after_layout(
&mut self,
window: &Window,
new_iframe_sizes: IFrameSizes,
) {
if new_iframe_sizes.is_empty() {
return;
}
let size_messages: Vec<_> = new_iframe_sizes
.into_iter()
.filter_map(|(browsing_context_id, iframe_size)| {
let viewport_details = iframe_size.viewport_details;
with_script_thread(|script_thread| {
script_thread.handle_resize_message(
iframe_size.pipeline_id,
viewport_details,
WindowSizeType::Resize,
);
script_thread.handle_update_pinch_zoom_infos(
iframe_size.pipeline_id,
PinchZoomInfos::new_from_viewport_size(viewport_details.size),
CanGc::note(),
)
});
let old_viewport_details =
self.set_viewport_details(browsing_context_id, viewport_details);
if old_viewport_details == Some(viewport_details) {
return None;
}
let size_type = match old_viewport_details {
Some(_) => WindowSizeType::Resize,
None => WindowSizeType::Initial,
};
Some(IFrameSizeMsg {
browsing_context_id,
size: viewport_details,
type_: size_type,
})
})
.collect();
if !size_messages.is_empty() {
window.send_to_constellation(ScriptToConstellationMessage::IFrameSizes(size_messages));
}
}
pub(crate) fn iter(&self) -> impl Iterator<Item = DomRoot<HTMLIFrameElement>> + use<'_> {
self.iframes.iter().map(|iframe| iframe.element.as_rooted())
}
}