Skip to main content

i_slint_compiler/passes/
check_public_api.rs

1// Copyright © SixtyFPS GmbH <info@slint.dev>
2// SPDX-License-Identifier: GPL-3.0-only OR LicenseRef-Slint-Royalty-free-2.0 OR LicenseRef-Slint-Software-3.0
3
4// cSpell: ignore Slintcomponents
5//! Pass that check that the public api is ok and mark the property as exposed
6
7use std::rc::Rc;
8
9use crate::diagnostics::{BuildDiagnostics, DiagnosticLevel};
10use crate::langtype::ElementType;
11use crate::object_tree::{Component, Document, ExportedName, PropertyVisibility};
12use crate::{CompilerConfiguration, ComponentSelection};
13use itertools::Either;
14
15pub fn check_public_api(
16    doc: &mut Document,
17    config: &CompilerConfiguration,
18    diag: &mut BuildDiagnostics,
19) {
20    let last = doc.last_exported_component();
21
22    if last.is_none() && !matches!(&config.components_to_generate, ComponentSelection::Named(_)) {
23        let last_imported = doc
24            .node
25            .as_ref()
26            .and_then(|n| {
27                let import_node = n.ImportSpecifier().last()?;
28                let import = crate::typeloader::ImportedName::extract_imported_names(&import_node.ImportIdentifierList()?).last()?;
29                let ElementType::Component(c) = doc.local_registry.lookup_element(&import.internal_name).ok()? else { return None };
30                diag.push_warning(format!("No component is exported. The last imported component '{}' will be used. This is deprecated", import.internal_name), &import_node);
31                let exported_name = ExportedName{ name: import.internal_name, name_ident: import_node.into() };
32                Some((exported_name, Either::Left(c)))
33            });
34        doc.exports.add_reexports(last_imported, diag);
35    }
36
37    match &config.components_to_generate {
38        ComponentSelection::ExportedWindows => doc.exports.retain(|export| {
39            // Warn about exported non-window (and remove them from the export unless it's the last for compatibility)
40            if let Either::Left(c) = &export.1
41                && !c.is_global() && !super::windows::inherits_window(c) {
42                    let is_last = last.as_ref().is_some_and(|last| !Rc::ptr_eq(last, c));
43
44                    if cfg!(feature = "experimental-library-module") && config.library_name.is_some() {
45                        // In library mode, we want to allow exporting for all Slintcomponents, as they might be used
46                        // as base components for the users of the library.
47                        return !is_last;
48                    }
49
50                    if is_last {
51                        diag.push_warning(format!("Exported component '{}' doesn't inherit Window. No code will be generated for it", export.0.name), &export.0.name_ident);
52                        return false;
53                    } else {
54                        diag.push_warning(format!("Exported component '{}' doesn't inherit Window. This is deprecated", export.0.name), &export.0.name_ident);
55                    }
56                }
57            true
58        }),
59        // Only keep the last component if there is one
60        ComponentSelection::LastExported => doc.exports.retain(|export| {
61            if let Either::Left(c) = &export.1 {
62                c.is_global() || last.as_ref().is_none_or(|last| Rc::ptr_eq(last, c))
63            } else {
64                true
65            }
66        }),
67        // Only keep the component with the given name
68        ComponentSelection::Named(name) => {
69            doc.exports.retain(|export| {
70                if let Either::Left(c) = &export.1 {
71                    c.is_global() || c.id == name
72                } else {
73                    true
74                }
75            });
76            if doc.last_exported_component().is_none() {
77                // We maybe requested to preview a non-exported component.
78                if let Ok(ElementType::Component(c)) = doc.local_registry.lookup_element(name)
79                    && let Some(name_ident) = c.node.as_ref().map(|n| n.DeclaredIdentifier().into()) {
80                        doc.exports.add_reexports(
81                            [(ExportedName{ name: name.into(), name_ident }, Either::Left(c))],
82                            diag,
83                        );
84                    }
85            }
86        },
87    }
88
89    for c in doc.exported_roots() {
90        check_public_api_component(&c, diag);
91    }
92    for (export_name, e) in &*doc.exports {
93        if let Some(c) = e.as_ref().left()
94            && c.is_global()
95        {
96            // This global will become part of the public API.
97            c.exported_global_names.borrow_mut().push(export_name.clone());
98            check_public_api_component(c, diag)
99        }
100    }
101}
102
103fn check_public_api_component(root_component: &Rc<Component>, diag: &mut BuildDiagnostics) {
104    let mut root_elem = root_component.root_element.borrow_mut();
105    let root_elem = &mut *root_elem;
106    let mut pa = root_elem.property_analysis.borrow_mut();
107    root_elem.property_declarations.iter_mut().for_each(|(n, d)| {
108        if d.property_type.ok_for_public_api() {
109            if d.visibility == PropertyVisibility::Private {
110                root_component.private_properties.borrow_mut().push((n.clone(), d.property_type.clone()));
111            } else {
112                d.expose_in_public_api = true;
113                if d.visibility != PropertyVisibility::Output {
114                    pa.entry(n.clone()).or_default().is_set = true;
115                }
116            }
117        } else {
118            diag.push_diagnostic(
119                 format!("Properties of type {} are not supported yet for public API. The property will not be exposed", d.property_type),
120                 &d.type_node(),
121                 DiagnosticLevel::Warning
122            );
123        }
124    });
125}