i_slint_compiler/passes/
lower_component_container.rs1use crate::diagnostics::BuildDiagnostics;
25use crate::langtype::ElementType;
26use crate::typeloader::TypeLoader;
27use crate::{expression_tree, object_tree::*};
28use std::cell::RefCell;
29use std::rc::Rc;
30
31pub fn lower_component_container(
32 doc: &Document,
33 type_loader: &mut TypeLoader,
34 diag: &mut BuildDiagnostics,
35) {
36 let empty_type = type_loader.global_type_registry.borrow().empty_type();
37
38 doc.visit_all_used_components(|component| {
39 recurse_elem_including_sub_components_no_borrow(component, &(), &mut |elem, _| {
40 if matches!(&elem.borrow().builtin_type(), Some(b) if b.name == "ComponentContainer") {
41 diagnose_component_container(elem, diag);
42 process_component_container(elem, &empty_type);
43 }
44 })
45 });
46}
47
48fn diagnose_component_container(element: &ElementRc, diag: &mut BuildDiagnostics) {
49 let elem = element.borrow();
50 if !elem.children.is_empty() {
51 diag.push_error("ComponentContainers may not have children".into(), &*element.borrow());
52 }
53 if let Some(cip) =
54 elem.enclosing_component.upgrade().unwrap().child_insertion_point.borrow().clone()
55 && Rc::ptr_eq(&cip.parent, element)
56 {
57 diag.push_error(
58 "The @children placeholder cannot appear in a ComponentContainer".into(),
59 &*element.borrow(),
60 );
61 }
62}
63
64fn process_component_container(element: &ElementRc, empty_type: &ElementType) {
65 let suffix = element.borrow().id.clone();
66
67 let component = Rc::new_cyclic(|component_weak| {
68 let root_element = Rc::new(RefCell::new(Element {
69 id: smol_str::format_smolstr!("component_container_internal_{}", suffix),
70 base_type: empty_type.clone(),
71 enclosing_component: component_weak.clone(),
72 ..Default::default()
73 }));
74
75 Component {
76 id: smol_str::format_smolstr!("ComponentContainerInternal_{}", suffix),
77 root_element,
78 ..Default::default()
79 }
80 });
81
82 let mut elem = element.borrow_mut();
83
84 let embedded_element = Element::make_rc(Element {
85 base_type: ElementType::Component(component.clone()),
86 id: smol_str::format_smolstr!("component_container_placeholder_{}", suffix),
87 debug: elem.debug.clone(),
88 enclosing_component: elem.enclosing_component.clone(),
89 default_fill_parent: (true, true),
90 inline_depth: elem.inline_depth,
91 repeated: Some(RepeatedElementInfo {
92 model: expression_tree::Expression::BoolLiteral(false),
93 model_data_id: Default::default(),
94 index_id: Default::default(),
95 is_conditional_element: true,
96 is_listview: None,
97 }),
98 is_component_placeholder: true,
99 ..Default::default()
100 });
101 elem.children.push(embedded_element);
102}