slint/internal/compiler/passes/unique_id.rs
Olivier Goffart 28ae8f7bc4 Refactoring: split ElementType away from the types used as property type
These are two different concept, and it is confusing to keep them in the
same enum

We want to support component without any base element, and Void is
already used for global component, so do this refactoring before
2022-10-26 14:50:44 +02:00

84 lines
2.9 KiB
Rust

// Copyright © SixtyFPS GmbH <info@slint-ui.com>
// SPDX-License-Identifier: GPL-3.0-only OR LicenseRef-Slint-commercial
use crate::diagnostics::BuildDiagnostics;
use crate::langtype::ElementType;
use crate::object_tree::*;
use std::collections::HashMap;
use std::rc::Rc;
/// This pass make sure that the id of the elements are unique
///
/// It currently does so by adding a number to the existing id
pub fn assign_unique_id(doc: &Document) {
let mut count = 0;
assign_unique_id_in_component(&doc.root_component, &mut count);
for c in &doc.root_component.used_types.borrow().sub_components {
assign_unique_id_in_component(c, &mut count);
}
rename_globals(&doc.root_component, count);
}
fn assign_unique_id_in_component(component: &Rc<Component>, count: &mut u32) {
recurse_elem_including_sub_components(component, &(), &mut |elem, _| {
*count += 1;
let mut elem_mut = elem.borrow_mut();
let old_id = if !elem_mut.id.is_empty() {
elem_mut.id.clone()
} else {
elem_mut.base_type.to_string().to_ascii_lowercase()
};
elem_mut.id = format!("{}-{}", old_id, count);
});
}
/// Give globals unique name
fn rename_globals(component: &Rc<Component>, mut count: u32) {
for g in &component.used_types.borrow().globals {
count += 1;
let mut root = g.root_element.borrow_mut();
if matches!(&root.base_type, ElementType::Builtin(_)) {
// builtin global keeps its name
root.id = g.id.clone();
} else if let Some(s) = g.exported_global_names.borrow().first() {
root.id = s.to_string();
} else {
root.id = format!("{}-{}", g.id, count);
}
}
}
/// Checks that all ids in the Component are unique
pub fn check_unique_id(doc: &Document, diag: &mut BuildDiagnostics) {
for component in &doc.inner_components {
check_unique_id_in_component(component, diag);
}
}
fn check_unique_id_in_component(component: &Rc<Component>, diag: &mut BuildDiagnostics) {
struct SeenId {
element: ElementRc,
error_reported: bool,
}
let mut seen_ids: HashMap<String, SeenId> = HashMap::new();
recurse_elem(&component.root_element, &(), &mut |elem, _| {
let elem_bor = elem.borrow();
let id = &elem_bor.id;
if !id.is_empty() {
if let Some(other_loc) = seen_ids.get_mut(id) {
debug_assert!(!Rc::ptr_eq(&other_loc.element, elem));
let message = format!("duplicated element id '{}'", id);
if !other_loc.error_reported {
diag.push_error(message.clone(), &*other_loc.element.borrow());
other_loc.error_reported = true;
}
diag.push_error(message, &*elem_bor);
} else {
seen_ids
.insert(id.clone(), SeenId { element: elem.clone(), error_reported: false });
}
}
})
}