mirror of
https://github.com/slint-ui/slint.git
synced 2025-07-08 05:35:24 +00:00
143 lines
4.1 KiB
Rust
143 lines
4.1 KiB
Rust
// Copyright © SixtyFPS GmbH <info@slint.dev>
|
|
// SPDX-License-Identifier: GPL-3.0-only OR LicenseRef-Slint-Royalty-free-2.0 OR LicenseRef-Slint-Software-3.0
|
|
|
|
// cSpell: ignore descendents
|
|
|
|
use crate::{
|
|
item_tree::ItemTreeVTable,
|
|
items::{ItemRc, TextInput},
|
|
SharedString,
|
|
};
|
|
use alloc::{vec, vec::Vec};
|
|
use bitflags::bitflags;
|
|
use vtable::VRcMapped;
|
|
|
|
/// The property names of the accessible-properties
|
|
#[repr(u32)]
|
|
#[derive(PartialEq, Eq, Copy, Clone, strum::Display)]
|
|
#[strum(serialize_all = "kebab-case")]
|
|
pub enum AccessibleStringProperty {
|
|
Checkable,
|
|
Checked,
|
|
DelegateFocus,
|
|
Description,
|
|
Enabled,
|
|
Expandable,
|
|
Expanded,
|
|
ItemCount,
|
|
ItemIndex,
|
|
ItemSelectable,
|
|
ItemSelected,
|
|
Label,
|
|
PlaceholderText,
|
|
ReadOnly,
|
|
Value,
|
|
ValueMaximum,
|
|
ValueMinimum,
|
|
ValueStep,
|
|
}
|
|
|
|
/// The argument of an accessible action.
|
|
#[repr(u32)]
|
|
#[derive(PartialEq, Clone)]
|
|
pub enum AccessibilityAction {
|
|
Default,
|
|
Decrement,
|
|
Increment,
|
|
Expand,
|
|
/// This is currently unused
|
|
ReplaceSelectedText(SharedString),
|
|
SetValue(SharedString),
|
|
}
|
|
|
|
bitflags! {
|
|
/// Define a accessibility actions that supported by an item.
|
|
#[repr(transparent)]
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, Default)]
|
|
pub struct SupportedAccessibilityAction: u32 {
|
|
const Default = 1;
|
|
const Decrement = 1 << 1;
|
|
const Increment = 1 << 2;
|
|
const Expand = 1 << 3;
|
|
const ReplaceSelectedText = 1 << 4;
|
|
const SetValue = 1 << 5;
|
|
}
|
|
}
|
|
|
|
/// Find accessible descendents of `root_item`.
|
|
///
|
|
/// This will recurse through all children of `root_item`, but will not recurse
|
|
/// into nodes that are accessible.
|
|
pub fn accessible_descendents(root_item: &ItemRc) -> impl Iterator<Item = ItemRc> {
|
|
fn try_candidate_or_find_next_accessible_descendent(
|
|
candidate: ItemRc,
|
|
descendent_candidates: &mut Vec<ItemRc>,
|
|
) -> Option<ItemRc> {
|
|
if candidate.is_accessible() {
|
|
return Some(candidate);
|
|
}
|
|
|
|
candidate.first_child().and_then(|child| {
|
|
if let Some(next) = child.next_sibling() {
|
|
descendent_candidates.push(next);
|
|
}
|
|
try_candidate_or_find_next_accessible_descendent(child, descendent_candidates)
|
|
})
|
|
}
|
|
|
|
// Do not look on the root_item: That is either a component root or an
|
|
// accessible item already handled!
|
|
let mut descendent_candidates = Vec::new();
|
|
if let Some(child) = root_item.first_child() {
|
|
descendent_candidates.push(child);
|
|
}
|
|
|
|
core::iter::from_fn(move || loop {
|
|
let candidate = descendent_candidates.pop()?;
|
|
|
|
if let Some(next_candidate) = candidate.next_sibling() {
|
|
descendent_candidates.push(next_candidate);
|
|
}
|
|
|
|
if let Some(descendent) =
|
|
try_candidate_or_find_next_accessible_descendent(candidate, &mut descendent_candidates)
|
|
{
|
|
return Some(descendent);
|
|
}
|
|
})
|
|
}
|
|
|
|
/// Find the first built-in `TextInput` in the descendents of `item`.
|
|
pub fn find_text_input(item: &ItemRc) -> Option<VRcMapped<ItemTreeVTable, TextInput>> {
|
|
fn try_candidate_or_find_next_descendent(
|
|
candidate: ItemRc,
|
|
descendent_candidates: &mut Vec<ItemRc>,
|
|
) -> Option<VRcMapped<ItemTreeVTable, TextInput>> {
|
|
if let Some(input) = candidate.downcast::<TextInput>() {
|
|
return Some(input);
|
|
}
|
|
|
|
candidate.first_child().and_then(|child| {
|
|
if let Some(next) = child.next_sibling() {
|
|
descendent_candidates.push(next);
|
|
}
|
|
try_candidate_or_find_next_descendent(child, descendent_candidates)
|
|
})
|
|
}
|
|
|
|
let mut descendent_candidates = vec![item.clone()];
|
|
|
|
loop {
|
|
let candidate = descendent_candidates.pop()?;
|
|
|
|
if let Some(next_candidate) = candidate.next_sibling() {
|
|
descendent_candidates.push(next_candidate);
|
|
}
|
|
|
|
if let Some(input) =
|
|
try_candidate_or_find_next_descendent(candidate, &mut descendent_candidates)
|
|
{
|
|
return Some(input);
|
|
}
|
|
}
|
|
}
|