1use std::{
37 borrow::Cow,
38 cell::RefCell,
39 collections::HashMap,
40 fs::File,
41 io::Write,
42 path::PathBuf,
43 rc::Rc,
44 time::{
45 Duration,
46 Instant,
47 },
48};
49
50use freya_clipboard::copypasta::{
51 ClipboardContext,
52 ClipboardProvider,
53};
54use freya_components::{
55 cache::AssetCacher,
56 integration::integration,
57};
58use freya_core::{
59 integration::*,
60 prelude::*,
61};
62use freya_engine::prelude::{
63 EncodedImageFormat,
64 FontCollection,
65 FontMgr,
66 SkData,
67 TypefaceFontProvider,
68 raster_n32_premul,
69};
70use ragnarok::{
71 CursorPoint,
72 EventsExecutorRunner,
73 EventsMeasurerRunner,
74 NodesState,
75};
76use torin::prelude::{
77 LayoutNode,
78 Size2D,
79};
80
81pub mod prelude {
82 pub use freya_core::{
83 events::platform::*,
84 prelude::*,
85 };
86
87 pub use crate::{
88 DocRunner,
89 TestingRunner,
90 launch_doc,
91 launch_test,
92 };
93}
94
95type DocRunnerHook = Box<dyn FnOnce(&mut TestingRunner)>;
96
97pub struct DocRunner {
98 app: AppComponent,
99 size: Size2D,
100 scale_factor: f64,
101 hook: Option<DocRunnerHook>,
102 image_path: PathBuf,
103}
104
105impl DocRunner {
106 pub fn render(self) {
107 let (mut test, _) = TestingRunner::new(self.app, self.size, |_| {}, self.scale_factor);
108 if let Some(hook) = self.hook {
109 (hook)(&mut test);
110 }
111 test.render_to_file(self.image_path);
112 }
113
114 pub fn with_hook(mut self, hook: impl FnOnce(&mut TestingRunner) + 'static) -> Self {
115 self.hook = Some(Box::new(hook));
116 self
117 }
118
119 pub fn with_image_path(mut self, image_path: PathBuf) -> Self {
120 self.image_path = image_path;
121 self
122 }
123
124 pub fn with_scale_factor(mut self, scale_factor: f64) -> Self {
125 self.scale_factor = scale_factor;
126 self
127 }
128
129 pub fn with_size(mut self, size: Size2D) -> Self {
130 self.size = size;
131 self
132 }
133}
134
135pub fn launch_doc(app: impl Into<AppComponent>, path: impl Into<PathBuf>) -> DocRunner {
136 DocRunner {
137 app: app.into(),
138 size: Size2D::new(250., 250.),
139 scale_factor: 1.0,
140 hook: None,
141 image_path: path.into(),
142 }
143}
144
145pub fn launch_test(app: impl Into<AppComponent>) -> TestingRunner {
146 TestingRunner::new(app, Size2D::new(500., 500.), |_| {}, 1.0).0
147}
148
149pub struct TestingRunner {
150 nodes_state: NodesState<NodeId>,
151 runner: Runner,
152 tree: Rc<RefCell<Tree>>,
153 size: Size2D,
154
155 accessibility: AccessibilityTree,
156
157 events_receiver: futures_channel::mpsc::UnboundedReceiver<EventsChunk>,
158 events_sender: futures_channel::mpsc::UnboundedSender<EventsChunk>,
159
160 font_manager: FontMgr,
161 font_collection: FontCollection,
162
163 platform: Platform,
164
165 animation_clock: AnimationClock,
166 ticker_sender: RenderingTickerSender,
167
168 default_fonts: Vec<Cow<'static, str>>,
169 scale_factor: f64,
170}
171
172impl TestingRunner {
173 pub fn new<T>(
174 app: impl Into<AppComponent>,
175 size: Size2D,
176 hook: impl FnOnce(&mut Runner) -> T,
177 scale_factor: f64,
178 ) -> (Self, T) {
179 let (events_sender, events_receiver) = futures_channel::mpsc::unbounded();
180 let app = app.into();
181 let mut runner = Runner::new(move || integration(app.clone()).into_element());
182
183 runner.provide_root_context(ScreenReader::new);
184
185 let (mut ticker_sender, ticker) = RenderingTicker::new();
186 ticker_sender.set_overflow(true);
187 runner.provide_root_context(|| ticker);
188
189 let animation_clock = runner.provide_root_context(AnimationClock::new);
190
191 runner.provide_root_context(AssetCacher::create);
192
193 let tree = Tree::default();
194 let tree = Rc::new(RefCell::new(tree));
195
196 let platform = runner.provide_root_context({
197 let tree = tree.clone();
198 || Platform {
199 focused_accessibility_id: State::create(ACCESSIBILITY_ROOT_ID),
200 focused_accessibility_node: State::create(accesskit::Node::new(
201 accesskit::Role::Window,
202 )),
203 root_size: State::create(size),
204 navigation_mode: State::create(NavigationMode::NotKeyboard),
205 preferred_theme: State::create(PreferredTheme::Light),
206 is_app_focused: State::create(true),
207 accent_color: State::create(AccentColor::default()),
208 sender: Rc::new(move |user_event| {
209 match user_event {
210 UserEvent::RequestRedraw => {
211 }
213 UserEvent::FocusAccessibilityNode(strategy) => {
214 tree.borrow_mut().accessibility_diff.request_focus(strategy);
215 }
216 UserEvent::SetCursorIcon(_) => {
217 }
219 UserEvent::Erased(_) => {
220 }
222 }
223 }),
224 }
225 });
226
227 runner.provide_root_context(|| {
228 let clipboard: Option<Box<dyn ClipboardProvider>> = ClipboardContext::new()
229 .ok()
230 .map(|c| Box::new(c) as Box<dyn ClipboardProvider>);
231
232 State::create(clipboard)
233 });
234
235 runner.provide_root_context(|| tree.borrow().accessibility_generator.clone());
236
237 let hook_result = hook(&mut runner);
238
239 let mut font_collection = FontCollection::new();
240 let def_mgr = FontMgr::default();
241 let provider = TypefaceFontProvider::new();
242 let font_manager: FontMgr = provider.into();
243 font_collection.set_default_font_manager(def_mgr, None);
244 font_collection.set_dynamic_font_manager(font_manager.clone());
245 font_collection.paragraph_cache_mut().turn_on(false);
246
247 runner.provide_root_context(|| font_collection.clone());
248
249 let nodes_state = NodesState::default();
250 let accessibility = AccessibilityTree::default();
251
252 let mut runner = Self {
253 runner,
254 tree,
255 size,
256
257 accessibility,
258 platform,
259
260 nodes_state,
261 events_receiver,
262 events_sender,
263
264 font_manager,
265 font_collection,
266
267 animation_clock,
268 ticker_sender,
269
270 default_fonts: default_fonts(),
271 scale_factor,
272 };
273
274 runner.sync_and_update();
275
276 (runner, hook_result)
277 }
278
279 pub fn set_fonts(&mut self, fonts: HashMap<&str, &[u8]>) {
280 let mut provider = TypefaceFontProvider::new();
281 for (font_name, font_data) in fonts {
282 let ft_type = self
283 .font_collection
284 .fallback_manager()
285 .unwrap()
286 .new_from_data(font_data, None)
287 .unwrap_or_else(|| panic!("Failed to load font {font_name}."));
288 provider.register_typeface(ft_type, Some(font_name));
289 }
290 let font_manager: FontMgr = provider.into();
291 self.font_manager = font_manager.clone();
292 self.font_collection.set_dynamic_font_manager(font_manager);
293 }
294
295 pub fn set_default_fonts(&mut self, fonts: &[Cow<'static, str>]) {
296 self.default_fonts.clear();
297 self.default_fonts.extend_from_slice(fonts);
298 self.tree.borrow_mut().layout.reset();
299 self.tree.borrow_mut().text_cache.reset();
300 self.tree.borrow_mut().measure_layout(
301 self.size,
302 &mut self.font_collection,
303 &self.font_manager,
304 &self.events_sender,
305 self.scale_factor,
306 &self.default_fonts,
307 );
308 self.tree.borrow_mut().accessibility_diff.clear();
309 self.accessibility.focused_id = ACCESSIBILITY_ROOT_ID;
310 self.accessibility.init(&mut self.tree.borrow_mut());
311 self.sync_and_update();
312 }
313
314 pub async fn handle_events(&mut self) {
315 self.runner.handle_events().await
316 }
317
318 pub fn handle_events_immediately(&mut self) {
319 self.runner.handle_events_immediately()
320 }
321
322 pub fn sync_and_update(&mut self) {
323 while let Ok(events_chunk) = self.events_receiver.try_recv() {
324 match events_chunk {
325 EventsChunk::Processed(processed_events) => {
326 let events_executor_adapter = EventsExecutorAdapter {
327 runner: &mut self.runner,
328 };
329 events_executor_adapter.run(&mut self.nodes_state, processed_events);
330 }
331 EventsChunk::Batch(events) => {
332 for event in events {
333 self.runner.handle_event(
334 event.node_id,
335 event.name,
336 event.data,
337 event.bubbles,
338 );
339 }
340 }
341 }
342 }
343
344 let mutations = self.runner.sync_and_update();
345 self.runner.run_in(|| {
346 self.tree.borrow_mut().apply_mutations(mutations);
347 });
348 self.tree.borrow_mut().measure_layout(
349 self.size,
350 &mut self.font_collection,
351 &self.font_manager,
352 &self.events_sender,
353 self.scale_factor,
354 &self.default_fonts,
355 );
356
357 let accessibility_update = self
358 .accessibility
359 .process_updates(&mut self.tree.borrow_mut(), &self.events_sender);
360
361 self.platform
362 .focused_accessibility_id
363 .set_if_modified(accessibility_update.focus);
364 let node_id = self.accessibility.focused_node_id().unwrap();
365 let tree = self.tree.borrow();
366 let layout_node = tree.layout.get(&node_id).unwrap();
367 self.platform
368 .focused_accessibility_node
369 .set_if_modified(AccessibilityTree::create_node(node_id, layout_node, &tree));
370 }
371
372 pub fn poll(&mut self, step: Duration, duration: Duration) {
375 let started = Instant::now();
376 while started.elapsed() < duration {
377 self.handle_events_immediately();
378 self.sync_and_update();
379 std::thread::sleep(step);
380 self.ticker_sender.broadcast_blocking(()).unwrap();
381 }
382 }
383
384 pub fn poll_n(&mut self, step: Duration, times: u32) {
387 for _ in 0..times {
388 self.handle_events_immediately();
389 self.sync_and_update();
390 std::thread::sleep(step);
391 self.ticker_sender.broadcast_blocking(()).unwrap();
392 }
393 }
394
395 pub fn send_event(&mut self, platform_event: PlatformEvent) {
396 let mut events_measurer_adapter = EventsMeasurerAdapter {
397 tree: &mut self.tree.borrow_mut(),
398 scale_factor: self.scale_factor,
399 };
400 let processed_events = events_measurer_adapter.run(
401 &mut vec![platform_event],
402 &mut self.nodes_state,
403 self.accessibility.focused_node_id(),
404 );
405 self.events_sender
406 .unbounded_send(EventsChunk::Processed(processed_events))
407 .unwrap();
408 }
409
410 pub fn move_cursor(&mut self, cursor: impl Into<CursorPoint>) {
411 self.send_event(PlatformEvent::Mouse {
412 name: MouseEventName::MouseMove,
413 cursor: cursor.into(),
414 button: Some(MouseButton::Left),
415 })
416 }
417
418 pub fn write_text(&mut self, text: impl ToString) {
419 let text = text.to_string();
420 self.send_event(PlatformEvent::Keyboard {
421 name: KeyboardEventName::KeyDown,
422 key: Key::Character(text),
423 code: Code::Unidentified,
424 modifiers: Modifiers::default(),
425 });
426 self.sync_and_update();
427 }
428
429 pub fn press_key(&mut self, key: Key) {
430 self.send_event(PlatformEvent::Keyboard {
431 name: KeyboardEventName::KeyDown,
432 key,
433 code: Code::Unidentified,
434 modifiers: Modifiers::default(),
435 });
436 self.sync_and_update();
437 }
438
439 pub fn press_cursor(&mut self, cursor: impl Into<CursorPoint>) {
440 let cursor = cursor.into();
441 self.send_event(PlatformEvent::Mouse {
442 name: MouseEventName::MouseDown,
443 cursor,
444 button: Some(MouseButton::Left),
445 });
446 self.sync_and_update();
447 }
448
449 pub fn release_cursor(&mut self, cursor: impl Into<CursorPoint>) {
450 let cursor = cursor.into();
451 self.send_event(PlatformEvent::Mouse {
452 name: MouseEventName::MouseUp,
453 cursor,
454 button: Some(MouseButton::Left),
455 });
456 self.sync_and_update();
457 }
458
459 pub fn click_cursor(&mut self, cursor: impl Into<CursorPoint>) {
460 let cursor = cursor.into();
461 self.send_event(PlatformEvent::Mouse {
462 name: MouseEventName::MouseDown,
463 cursor,
464 button: Some(MouseButton::Left),
465 });
466 self.sync_and_update();
467 self.send_event(PlatformEvent::Mouse {
468 name: MouseEventName::MouseUp,
469 cursor,
470 button: Some(MouseButton::Left),
471 });
472 self.sync_and_update();
473 }
474
475 pub fn scroll(&mut self, cursor: impl Into<CursorPoint>, scroll: impl Into<CursorPoint>) {
476 let cursor = cursor.into();
477 let scroll = scroll.into();
478 self.send_event(PlatformEvent::Wheel {
479 name: WheelEventName::Wheel,
480 scroll,
481 cursor,
482 source: WheelSource::Device,
483 });
484 self.sync_and_update();
485 }
486
487 pub fn animation_clock(&mut self) -> &mut AnimationClock {
488 &mut self.animation_clock
489 }
490
491 pub fn render(&mut self) -> SkData {
492 let mut surface = raster_n32_premul((self.size.width as i32, self.size.height as i32))
493 .expect("Failed to create the surface.");
494
495 let render_pipeline = RenderPipeline {
496 font_collection: &mut self.font_collection,
497 font_manager: &self.font_manager,
498 tree: &self.tree.borrow(),
499 canvas: surface.canvas(),
500 scale_factor: self.scale_factor,
501 background: Color::WHITE,
502 };
503 render_pipeline.render();
504
505 let image = surface.image_snapshot();
506 let mut context = surface.direct_context();
507 image
508 .encode(context.as_mut(), EncodedImageFormat::PNG, None)
509 .expect("Failed to encode the snapshot.")
510 }
511
512 pub fn render_to_file(&mut self, path: impl Into<PathBuf>) {
513 let path = path.into();
514
515 let image = self.render();
516
517 let mut snapshot_file = File::create(path).expect("Failed to create the snapshot file.");
518
519 snapshot_file
520 .write_all(&image)
521 .expect("Failed to save the snapshot file.");
522 }
523
524 pub fn find<T>(
525 &self,
526 matcher: impl Fn(TestingNode, &dyn ElementExt) -> Option<T>,
527 ) -> Option<T> {
528 let mut matched = None;
529 {
530 let tree = self.tree.borrow();
531 tree.traverse_depth(|id| {
532 if matched.is_some() {
533 return;
534 }
535 let element = tree.elements.get(&id).unwrap();
536 let node = TestingNode {
537 tree: self.tree.clone(),
538 id,
539 };
540 matched = matcher(node, element.as_ref());
541 });
542 }
543
544 matched
545 }
546
547 pub fn find_many<T>(
548 &self,
549 matcher: impl Fn(TestingNode, &dyn ElementExt) -> Option<T>,
550 ) -> Vec<T> {
551 let mut matched = Vec::new();
552 {
553 let tree = self.tree.borrow();
554 tree.traverse_depth(|id| {
555 let element = tree.elements.get(&id).unwrap();
556 let node = TestingNode {
557 tree: self.tree.clone(),
558 id,
559 };
560 if let Some(result) = matcher(node, element.as_ref()) {
561 matched.push(result);
562 }
563 });
564 }
565
566 matched
567 }
568}
569
570pub struct TestingNode {
571 tree: Rc<RefCell<Tree>>,
572 id: NodeId,
573}
574
575impl TestingNode {
576 pub fn layout(&self) -> LayoutNode {
577 self.tree.borrow().layout.get(&self.id).cloned().unwrap()
578 }
579
580 pub fn children(&self) -> Vec<Self> {
581 let children = self
582 .tree
583 .borrow()
584 .children
585 .get(&self.id)
586 .cloned()
587 .unwrap_or_default();
588
589 children
590 .into_iter()
591 .map(|child_id| Self {
592 id: child_id,
593 tree: self.tree.clone(),
594 })
595 .collect()
596 }
597
598 pub fn is_visible(&self) -> bool {
599 let layout = self.layout();
600 let effect_state = self
601 .tree
602 .borrow()
603 .effect_state
604 .get(&self.id)
605 .cloned()
606 .unwrap();
607
608 effect_state.is_visible(&self.tree.borrow().layout, &layout.area)
609 }
610
611 pub fn element(&self) -> Rc<dyn ElementExt> {
612 self.tree
613 .borrow()
614 .elements
615 .get(&self.id)
616 .cloned()
617 .expect("Element does not exist.")
618 }
619}