slint/internal/compiler/passes/lower_accessibility.rs

78 lines
3.2 KiB
Rust

// Copyright © SixtyFPS GmbH <info@slint-ui.com>
// SPDX-License-Identifier: GPL-3.0-only OR LicenseRef-Slint-commercial
//! Pass that lowers synthetic `accessible-*` properties
use crate::diagnostics::BuildDiagnostics;
use crate::expression_tree::{Expression, NamedReference};
use crate::langtype::EnumerationValue;
use crate::object_tree::{Component, ElementRc};
use std::rc::Rc;
pub fn lower_accessibility_properties(component: &Rc<Component>, diag: &mut BuildDiagnostics) {
crate::object_tree::recurse_elem_including_sub_components_no_borrow(
component,
&(),
&mut |elem, _| {
if elem.borrow().repeated.is_some() {
return;
};
apply_builtin(elem);
let accessible_role_set = match elem.borrow().bindings.get("accessible-role") {
Some(role) => {
if let Expression::EnumerationValue(val) = &role.borrow().expression {
debug_assert_eq!(val.enumeration.name, "AccessibleRole");
debug_assert_eq!(val.enumeration.values[0], "none");
if val.value == 0 {
return;
}
} else {
diag.push_error(
"The `accessible-role` property must be a constant expression".into(),
&*role.borrow(),
);
}
true
}
// maybe it was set on the parent
None => elem.borrow().is_binding_set("accessible-role", false),
};
for prop_name in crate::typeregister::RESERVED_ACCESSIBILITY_PROPERTIES
.iter()
.map(|x| x.0)
.chain(std::iter::once("accessible-role"))
{
if accessible_role_set {
if elem.borrow().is_binding_set(prop_name, false) {
let nr = NamedReference::new(elem, prop_name);
elem.borrow_mut().accessibility_props.0.insert(prop_name.into(), nr);
}
} else if let Some(b) = elem.borrow().bindings.get(prop_name) {
diag.push_error(
format!("The `{prop_name}` property can only be set in combination to `accessible-role`"),
&*b.borrow(),
);
}
}
},
)
}
fn apply_builtin(e: &ElementRc) {
let bty = if let Some(bty) = e.borrow().builtin_type() { bty } else { return };
if bty.name == "Text" {
e.borrow_mut().set_binding_if_not_set("accessible-role".into(), || {
let enum_ty = crate::typeregister::BUILTIN_ENUMS.with(|e| e.AccessibleRole.clone());
Expression::EnumerationValue(EnumerationValue {
value: enum_ty.values.iter().position(|v| v == "text").unwrap(),
enumeration: enum_ty,
})
});
let text_prop = NamedReference::new(e, "text");
e.borrow_mut().set_binding_if_not_set("accessible-label".into(), || {
Expression::PropertyReference(text_prop)
});
}
}