1use atomic_refcell::AtomicRefCell;
10use devtools_traits::DevtoolScriptControlMsg::{self, GetCssDatabase, SimulateColorScheme};
11use devtools_traits::{DevtoolsPageInfo, NavigationState};
12use embedder_traits::Theme;
13use malloc_size_of_derive::MallocSizeOf;
14use rustc_hash::FxHashMap;
15use serde::Serialize;
16use serde_json::{Map, Value};
17use servo_base::generic_channel::{self, GenericSender, SendError};
18use servo_base::id::PipelineId;
19
20use crate::actor::{Actor, ActorEncode, ActorError, ActorRegistry};
21use crate::actors::inspector::InspectorActor;
22use crate::actors::inspector::accessibility::AccessibilityActor;
23use crate::actors::inspector::css_properties::CssPropertiesActor;
24use crate::actors::reflow::ReflowActor;
25use crate::actors::stylesheets::StyleSheetsActor;
26use crate::actors::tab::TabDescriptorActor;
27use crate::actors::thread::ThreadActor;
28use crate::actors::watcher::{SessionContext, SessionContextType, WatcherActor};
29use crate::id::{DevtoolsBrowserId, DevtoolsBrowsingContextId, DevtoolsOuterWindowId, IdMap};
30use crate::protocol::{ClientRequest, DevtoolsConnection, JsonPacketStream};
31use crate::resource::ResourceAvailable;
32use crate::{EmptyReplyMsg, StreamId};
33
34#[derive(Serialize)]
35struct ListWorkersReply {
36 from: String,
37 workers: Vec<()>,
38}
39
40#[derive(Serialize)]
41struct FrameUpdateReply {
42 from: String,
43 #[serde(rename = "type")]
44 type_: String,
45 frames: Vec<FrameUpdateMsg>,
46}
47
48#[derive(Serialize)]
49#[serde(rename_all = "camelCase")]
50struct FrameUpdateMsg {
51 id: u32,
52 is_top_level: bool,
53 url: String,
54 title: String,
55}
56
57#[derive(Serialize)]
58struct TabNavigated {
59 from: String,
60 #[serde(rename = "type")]
61 type_: String,
62 url: String,
63 title: Option<String>,
64 #[serde(rename = "nativeConsoleAPI")]
65 native_console_api: bool,
66 state: String,
67 is_frame_switching: bool,
68}
69
70#[derive(Serialize)]
71#[serde(rename_all = "camelCase")]
72struct BrowsingContextTraits {
73 frames: bool,
74 is_browsing_context: bool,
75 log_in_page: bool,
76 navigation: bool,
77 supports_top_level_target_flag: bool,
78 watchpoints: bool,
79}
80
81#[derive(Serialize)]
82#[serde(rename_all = "lowercase")]
83enum TargetType {
84 Frame,
85 }
87
88#[derive(Serialize)]
89#[serde(rename_all = "camelCase")]
90pub(crate) struct BrowsingContextActorMsg {
91 actor: String,
92 title: String,
93 url: String,
94 #[serde(rename = "browserId")]
96 browser_id: u32,
97 #[serde(rename = "outerWindowID")]
98 outer_window_id: u32,
99 #[serde(rename = "browsingContextID")]
100 browsing_context_id: u32,
101 is_top_level_target: bool,
102 traits: BrowsingContextTraits,
103 accessibility_actor: String,
105 console_actor: String,
106 css_properties_actor: String,
107 inspector_actor: String,
108 reflow_actor: String,
109 style_sheets_actor: String,
110 thread_actor: String,
111 target_type: TargetType,
112 }
127
128#[derive(MallocSizeOf)]
132pub(crate) struct BrowsingContextActor {
133 name: String,
134 pub title: AtomicRefCell<String>,
135 pub url: AtomicRefCell<String>,
136 pub browser_id: DevtoolsBrowserId,
138 active_pipeline_id: AtomicRefCell<PipelineId>,
140 active_outer_window_id: AtomicRefCell<DevtoolsOuterWindowId>,
141 pub browsing_context_id: DevtoolsBrowsingContextId,
142 accessibility_name: String,
143 pub console_name: String,
144 css_properties_name: String,
145 pub(crate) inspector_name: String,
146 reflow_name: String,
147 style_sheets_name: String,
148 pub thread_name: String,
149 _tab: String,
150 script_chans: AtomicRefCell<FxHashMap<PipelineId, GenericSender<DevtoolScriptControlMsg>>>,
157 pub watcher_name: String,
158}
159
160impl ResourceAvailable for BrowsingContextActor {
161 fn actor_name(&self) -> String {
162 self.name.clone()
163 }
164}
165
166impl Actor for BrowsingContextActor {
167 fn name(&self) -> String {
168 self.name.clone()
169 }
170
171 fn handle_message(
172 &self,
173 request: ClientRequest,
174 _registry: &ActorRegistry,
175 msg_type: &str,
176 _msg: &Map<String, Value>,
177 _id: StreamId,
178 ) -> Result<(), ActorError> {
179 match msg_type {
180 "listFrames" => {
181 let msg = EmptyReplyMsg { from: self.name() };
183 request.reply_final(&msg)?
184 },
185 "listWorkers" => {
186 request.reply_final(&ListWorkersReply {
187 from: self.name(),
188 workers: vec![],
190 })?
191 },
192 _ => return Err(ActorError::UnrecognizedPacketType),
193 };
194 Ok(())
195 }
196}
197
198impl BrowsingContextActor {
199 #[expect(clippy::too_many_arguments)]
200 pub(crate) fn register(
201 registry: &ActorRegistry,
202 console_name: String,
203 browser_id: DevtoolsBrowserId,
204 browsing_context_id: DevtoolsBrowsingContextId,
205 page_info: DevtoolsPageInfo,
206 pipeline_id: PipelineId,
207 outer_window_id: DevtoolsOuterWindowId,
208 script_sender: GenericSender<DevtoolScriptControlMsg>,
209 ) -> String {
210 let name = registry.new_name::<BrowsingContextActor>();
211 let DevtoolsPageInfo {
212 title,
213 url,
214 is_top_level_global,
215 ..
216 } = page_info;
217
218 let accessibility_name = AccessibilityActor::register(registry);
219
220 let properties = (|| {
221 let (properties_sender, properties_receiver) = generic_channel::channel()?;
222 script_sender.send(GetCssDatabase(properties_sender)).ok()?;
223 properties_receiver.recv().ok()
224 })()
225 .unwrap_or_default();
226 let css_properties_name = CssPropertiesActor::register(registry, properties);
227
228 let inspector_name = InspectorActor::register(registry, name.clone());
229
230 let reflow_name = ReflowActor::register(registry);
231
232 let style_sheets_name = StyleSheetsActor::register(registry);
233
234 let tab_descriptor_name =
235 TabDescriptorActor::register(registry, name.clone(), is_top_level_global);
236
237 let thread_name =
238 ThreadActor::register(registry, script_sender.clone(), Some(name.clone()));
239
240 let watcher_name = WatcherActor::register(
241 registry,
242 name.clone(),
243 SessionContext::new(SessionContextType::BrowserElement),
244 );
245
246 let mut script_chans = FxHashMap::default();
247 script_chans.insert(pipeline_id, script_sender);
248
249 let actor = BrowsingContextActor {
250 name: name.clone(),
251 script_chans: AtomicRefCell::new(script_chans),
252 title: AtomicRefCell::new(title),
253 url: AtomicRefCell::new(url.into_string()),
254 active_pipeline_id: AtomicRefCell::new(pipeline_id),
255 active_outer_window_id: AtomicRefCell::new(outer_window_id),
256 browser_id,
257 browsing_context_id,
258 accessibility_name,
259 console_name,
260 css_properties_name,
261 inspector_name,
262 reflow_name,
263 style_sheets_name,
264 _tab: tab_descriptor_name,
265 thread_name,
266 watcher_name,
267 };
268
269 registry.register::<Self>(actor);
270
271 name
272 }
273
274 pub(crate) fn handle_new_global(
275 &self,
276 pipeline: PipelineId,
277 script_sender: GenericSender<DevtoolScriptControlMsg>,
278 ) {
279 self.script_chans
280 .borrow_mut()
281 .insert(pipeline, script_sender);
282 }
283
284 pub(crate) fn handle_navigate<'a>(
285 &self,
286 state: NavigationState,
287 id_map: &mut IdMap,
288 connections: impl Iterator<Item = &'a mut DevtoolsConnection>,
289 ) {
290 let (pipeline_id, title, url, state) = match state {
291 NavigationState::Start(url) => (None, None, url, "start"),
292 NavigationState::Stop(pipeline, info) => {
293 (Some(pipeline), Some(info.title), info.url, "stop")
294 },
295 };
296 if let Some(pipeline_id) = pipeline_id {
297 let outer_window_id = id_map.outer_window_id(pipeline_id);
298 *self.active_outer_window_id.borrow_mut() = outer_window_id;
299 *self.active_pipeline_id.borrow_mut() = pipeline_id;
300 }
301 url.as_str().clone_into(&mut self.url.borrow_mut());
302 if let Some(ref t) = title {
303 self.title.borrow_mut().clone_from(t);
304 }
305
306 let msg = TabNavigated {
307 from: self.name(),
308 type_: "tabNavigated".to_owned(),
309 url: url.as_str().to_owned(),
310 title,
311 native_console_api: true,
312 state: state.to_owned(),
313 is_frame_switching: false,
314 };
315
316 for stream in connections {
317 let _ = stream.write_json_packet(&msg);
318 }
319 }
320
321 pub(crate) fn title_changed(&self, pipeline_id: PipelineId, title: String) {
322 if pipeline_id != self.pipeline_id() {
323 return;
324 }
325 *self.title.borrow_mut() = title;
326 }
327
328 pub(crate) fn frame_update(&self, request: &mut ClientRequest) {
329 let _ = request.write_json_packet(&FrameUpdateReply {
330 from: self.name(),
331 type_: "frameUpdate".into(),
332 frames: vec![FrameUpdateMsg {
333 id: self.browsing_context_id.value(),
334 is_top_level: true,
335 title: self.title.borrow().clone(),
336 url: self.url.borrow().clone(),
337 }],
338 });
339 }
340
341 pub fn simulate_color_scheme(&self, theme: Theme) -> Result<(), ()> {
342 self.script_chan()
343 .send(SimulateColorScheme(self.pipeline_id(), theme))
344 .map_err(|_| ())
345 }
346
347 pub(crate) fn pipeline_id(&self) -> PipelineId {
348 *self.active_pipeline_id.borrow()
349 }
350
351 pub(crate) fn outer_window_id(&self) -> DevtoolsOuterWindowId {
352 *self.active_outer_window_id.borrow()
353 }
354
355 pub(crate) fn script_chan(&self) -> GenericSender<DevtoolScriptControlMsg> {
357 self.script_chans
358 .borrow()
359 .get(&self.pipeline_id())
360 .unwrap()
361 .clone()
362 }
363
364 pub(crate) fn instruct_script_to_send_live_updates(&self, should_send_updates: bool) {
365 let result = self
366 .script_chan()
367 .send(DevtoolScriptControlMsg::WantsLiveNotifications(
368 self.pipeline_id(),
369 should_send_updates,
370 ));
371
372 debug_assert!(matches!(result, Ok(_) | Err(SendError::Disconnected)));
375 }
376}
377
378impl ActorEncode<BrowsingContextActorMsg> for BrowsingContextActor {
379 fn encode(&self, _: &ActorRegistry) -> BrowsingContextActorMsg {
380 BrowsingContextActorMsg {
381 actor: self.name(),
382 traits: BrowsingContextTraits {
383 is_browsing_context: true,
384 frames: true,
385 log_in_page: false,
386 navigation: true,
387 supports_top_level_target_flag: true,
388 watchpoints: true,
389 },
390 title: self.title.borrow().clone(),
391 url: self.url.borrow().clone(),
392 browser_id: self.browser_id.value(),
393 browsing_context_id: self.browsing_context_id.value(),
394 outer_window_id: self.outer_window_id().value(),
395 is_top_level_target: true,
396 accessibility_actor: self.accessibility_name.clone(),
397 console_actor: self.console_name.clone(),
398 css_properties_actor: self.css_properties_name.clone(),
399 inspector_actor: self.inspector_name.clone(),
400 reflow_actor: self.reflow_name.clone(),
401 style_sheets_actor: self.style_sheets_name.clone(),
402 thread_actor: self.thread_name.clone(),
403 target_type: TargetType::Frame,
404 }
405 }
406}