mirror of
				https://github.com/slint-ui/slint.git
				synced 2025-11-03 21:24:17 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			752 lines
		
	
	
	
		
			28 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			752 lines
		
	
	
	
		
			28 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
 | 
						|
 | 
						|
use std::collections::HashMap;
 | 
						|
use std::pin::Pin;
 | 
						|
use std::ptr::NonNull;
 | 
						|
use std::rc::Weak;
 | 
						|
 | 
						|
use accesskit::{Action, ActionRequest, Node, NodeId, Role, Toggled, Tree, TreeUpdate};
 | 
						|
use i_slint_core::accessibility::{
 | 
						|
    AccessibilityAction, AccessibleStringProperty, SupportedAccessibilityAction,
 | 
						|
};
 | 
						|
use i_slint_core::api::Window;
 | 
						|
use i_slint_core::item_tree::{ItemTreeRc, ItemTreeRef, ItemTreeWeak};
 | 
						|
use i_slint_core::items::{ItemRc, WindowItem};
 | 
						|
use i_slint_core::lengths::{LogicalPoint, ScaleFactor};
 | 
						|
use i_slint_core::window::{PopupWindowLocation, WindowInner};
 | 
						|
use i_slint_core::SharedString;
 | 
						|
use i_slint_core::{properties::PropertyTracker, window::WindowAdapter};
 | 
						|
 | 
						|
use super::WinitWindowAdapter;
 | 
						|
use crate::SlintUserEvent;
 | 
						|
use winit::event_loop::EventLoopProxy;
 | 
						|
 | 
						|
/// The AccessKit adapter tries to keep the given window adapter's item tree in sync with accesskit's node tree.
 | 
						|
///
 | 
						|
/// The entire item tree is mapped to accesskit's node tree. Any changes to an individual accessible item results
 | 
						|
/// in an access kit tree update with just changed nodes. Any changes in the tree structure result in a complete
 | 
						|
/// tree rebuild. This could be implemented more efficiently, but that isn't essential; AccessKit will avoid firing
 | 
						|
/// gratuitous events for full-tree updates as long as the node IDs are stable.
 | 
						|
///
 | 
						|
/// For unix it's necessary to inform accesskit about any changes to the position or size of the window, hence
 | 
						|
/// the `on_event` function that needs calling.
 | 
						|
///
 | 
						|
/// Similarly, when the window adapter is informed about a focus change, handle_focus_change must be called.
 | 
						|
/// Finally, when a component is destroyed, `unregister_item_tree` must be called, which rebuilds the entire
 | 
						|
/// tree at the moment.
 | 
						|
///
 | 
						|
/// If we wanted to move this to corelib, `on_event` gets replaced with listening to the events sent from the
 | 
						|
/// platform adapter to the slint::Window. `handle_focus_change` is already internal to WindowInner, as well
 | 
						|
/// as `component_destroyed`. The `WindowInner` would own this `AccessKit`.
 | 
						|
pub struct AccessKitAdapter {
 | 
						|
    inner: accesskit_winit::Adapter,
 | 
						|
    window_adapter_weak: Weak<WinitWindowAdapter>,
 | 
						|
    nodes: NodeCollection,
 | 
						|
    global_property_tracker: Pin<Box<PropertyTracker<AccessibilitiesPropertyTracker>>>,
 | 
						|
    pending_update: bool,
 | 
						|
    initial_tree_sent: bool,
 | 
						|
}
 | 
						|
 | 
						|
impl AccessKitAdapter {
 | 
						|
    pub fn new(
 | 
						|
        window_adapter_weak: Weak<WinitWindowAdapter>,
 | 
						|
        winit_window: &winit::window::Window,
 | 
						|
        proxy: EventLoopProxy<SlintUserEvent>,
 | 
						|
    ) -> Self {
 | 
						|
        Self {
 | 
						|
            inner: accesskit_winit::Adapter::with_event_loop_proxy(winit_window, proxy),
 | 
						|
            window_adapter_weak: window_adapter_weak.clone(),
 | 
						|
            nodes: NodeCollection {
 | 
						|
                next_component_id: 1,
 | 
						|
                root_node_id: NodeId(0),
 | 
						|
                components_by_id: Default::default(),
 | 
						|
                component_ids: Default::default(),
 | 
						|
                all_nodes: Default::default(),
 | 
						|
                focused_node_tracker: Box::pin(PropertyTracker::new_with_dirty_handler(
 | 
						|
                    DelegateFocusPropertyTracker {
 | 
						|
                        window_adapter_weak: window_adapter_weak.clone(),
 | 
						|
                    },
 | 
						|
                )),
 | 
						|
            },
 | 
						|
            global_property_tracker: Box::pin(PropertyTracker::new_with_dirty_handler(
 | 
						|
                AccessibilitiesPropertyTracker { window_adapter_weak: window_adapter_weak.clone() },
 | 
						|
            )),
 | 
						|
            pending_update: false,
 | 
						|
            initial_tree_sent: false,
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    pub fn process_event(
 | 
						|
        &mut self,
 | 
						|
        window: &winit::window::Window,
 | 
						|
        event: &winit::event::WindowEvent,
 | 
						|
    ) {
 | 
						|
        if matches!(event, winit::event::WindowEvent::Focused(_)) {
 | 
						|
            self.global_property_tracker.set_dirty();
 | 
						|
            self.invoke_later(|self_cell, _| self_cell.borrow_mut().rebuild_tree_of_dirty_nodes());
 | 
						|
        }
 | 
						|
        self.inner.process_event(window, event);
 | 
						|
    }
 | 
						|
 | 
						|
    pub fn process_accesskit_event(
 | 
						|
        &mut self,
 | 
						|
        window_event: accesskit_winit::WindowEvent,
 | 
						|
    ) -> Option<DeferredAccessKitAction> {
 | 
						|
        match window_event {
 | 
						|
            accesskit_winit::WindowEvent::InitialTreeRequested => {
 | 
						|
                self.inner.update_if_active(|| {
 | 
						|
                    self.nodes.build_new_tree(
 | 
						|
                        &self.window_adapter_weak,
 | 
						|
                        self.global_property_tracker.as_ref(),
 | 
						|
                    )
 | 
						|
                });
 | 
						|
                self.initial_tree_sent = true;
 | 
						|
                None
 | 
						|
            }
 | 
						|
            accesskit_winit::WindowEvent::ActionRequested(r) => self.handle_request(r),
 | 
						|
            accesskit_winit::WindowEvent::AccessibilityDeactivated => {
 | 
						|
                self.initial_tree_sent = false;
 | 
						|
                None
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    pub fn handle_focus_item_change(&mut self) {
 | 
						|
        // Ignore focus changes until an initial tree was sent, to avoid sending `tree: None`.
 | 
						|
        if !self.initial_tree_sent {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
        // Don't send a tree update now with an empty tree/node list when we know that the structure
 | 
						|
        // of the tree has changed. It might be that the focus node is not known yet to AccessKit.
 | 
						|
        // The pending update will take care of setting the focus node.
 | 
						|
        if self.pending_update {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
        self.inner.update_if_active(|| TreeUpdate {
 | 
						|
            nodes: vec![],
 | 
						|
            tree: None,
 | 
						|
            focus: self.nodes.focus_node(&self.window_adapter_weak),
 | 
						|
        })
 | 
						|
    }
 | 
						|
 | 
						|
    fn handle_request(&self, request: ActionRequest) -> Option<DeferredAccessKitAction> {
 | 
						|
        let a = match request.action {
 | 
						|
            Action::Click => AccessibilityAction::Default,
 | 
						|
            Action::Focus => {
 | 
						|
                return self
 | 
						|
                    .nodes
 | 
						|
                    .item_rc_for_node_id(request.target)
 | 
						|
                    .map(DeferredAccessKitAction::SetFocus);
 | 
						|
            }
 | 
						|
            Action::Decrement => AccessibilityAction::Decrement,
 | 
						|
            Action::Increment => AccessibilityAction::Increment,
 | 
						|
            Action::ReplaceSelectedText => {
 | 
						|
                let Some(accesskit::ActionData::Value(v)) = request.data else { return None };
 | 
						|
                AccessibilityAction::ReplaceSelectedText(SharedString::from(&*v))
 | 
						|
            }
 | 
						|
            Action::SetValue => match request.data.unwrap() {
 | 
						|
                accesskit::ActionData::Value(v) => {
 | 
						|
                    AccessibilityAction::SetValue(SharedString::from(&*v))
 | 
						|
                }
 | 
						|
                accesskit::ActionData::NumericValue(v) => {
 | 
						|
                    AccessibilityAction::SetValue(i_slint_core::format!("{v}"))
 | 
						|
                }
 | 
						|
                _ => return None,
 | 
						|
            },
 | 
						|
            Action::Expand => AccessibilityAction::Expand,
 | 
						|
            _ => return None,
 | 
						|
        };
 | 
						|
        self.nodes
 | 
						|
            .item_rc_for_node_id(request.target)
 | 
						|
            .map(|item| DeferredAccessKitAction::InvokeAccessibleAction(item, a))
 | 
						|
    }
 | 
						|
 | 
						|
    pub fn reload_tree(&mut self) {
 | 
						|
        if self.pending_update {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
        self.pending_update = true;
 | 
						|
 | 
						|
        self.invoke_later(|self_cell, win| {
 | 
						|
            let mut self_ = self_cell.borrow_mut();
 | 
						|
            let self_ = &mut *self_;
 | 
						|
            self_.pending_update = false;
 | 
						|
            self_.inner.update_if_active(|| {
 | 
						|
                self_.nodes.build_new_tree(&win, self_.global_property_tracker.as_ref())
 | 
						|
            })
 | 
						|
        });
 | 
						|
    }
 | 
						|
 | 
						|
    pub fn unregister_item_tree(&mut self, component: ItemTreeRef) {
 | 
						|
        let component_ptr = ItemTreeRef::as_ptr(component);
 | 
						|
        if let Some(component_id) = self.nodes.component_ids.remove(&component_ptr) {
 | 
						|
            self.nodes.components_by_id.remove(&component_id);
 | 
						|
        }
 | 
						|
        self.reload_tree();
 | 
						|
    }
 | 
						|
 | 
						|
    fn rebuild_tree_of_dirty_nodes(&mut self) {
 | 
						|
        if !self.global_property_tracker.is_dirty() && !self.nodes.focused_node_tracker.is_dirty() {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        // It's possible that we may have been triggered by a timer, but in the meantime
 | 
						|
        // the node tree has been emptied due to a tree structure change.
 | 
						|
        if self.nodes.all_nodes.is_empty() {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        let Some(window_adapter) = self.window_adapter_weak.upgrade() else { return };
 | 
						|
        let window = window_adapter.window();
 | 
						|
 | 
						|
        self.inner.update_if_active(|| {
 | 
						|
            self.global_property_tracker.as_ref().evaluate_as_dependency_root(|| {
 | 
						|
                let nodes = self.nodes.all_nodes.iter().filter_map(|cached_node| {
 | 
						|
                    cached_node.tracker.as_ref().evaluate_if_dirty(|| {
 | 
						|
                        let scale_factor = ScaleFactor::new(window.scale_factor());
 | 
						|
                        let item = self.nodes.item_rc_for_node_id(cached_node.id)?;
 | 
						|
 | 
						|
                        let mut node = self.nodes.build_node_without_children(
 | 
						|
                            &item,
 | 
						|
                            scale_factor,
 | 
						|
                            Default::default(),
 | 
						|
                        );
 | 
						|
 | 
						|
                        node.set_children(cached_node.children.clone());
 | 
						|
 | 
						|
                        Some((cached_node.id, node))
 | 
						|
                    })?
 | 
						|
                });
 | 
						|
 | 
						|
                TreeUpdate {
 | 
						|
                    nodes: nodes.collect(),
 | 
						|
                    tree: None,
 | 
						|
                    focus: self.nodes.focus_node(&self.window_adapter_weak),
 | 
						|
                }
 | 
						|
            })
 | 
						|
        })
 | 
						|
    }
 | 
						|
 | 
						|
    fn invoke_later(
 | 
						|
        &self,
 | 
						|
        callback: impl FnOnce(&std::cell::RefCell<Self>, Weak<WinitWindowAdapter>) + 'static,
 | 
						|
    ) {
 | 
						|
        let win = self.window_adapter_weak.clone();
 | 
						|
        i_slint_core::timers::Timer::single_shot(Default::default(), move || {
 | 
						|
            WinitWindowAdapter::with_access_kit_adapter_from_weak_window_adapter(
 | 
						|
                win.clone(),
 | 
						|
                move |self_cell| callback(self_cell, win),
 | 
						|
            );
 | 
						|
        });
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
fn accessible_parent_for_item_rc(mut item: ItemRc) -> ItemRc {
 | 
						|
    while !item.is_accessible() {
 | 
						|
        if let Some(parent) = item.parent_item() {
 | 
						|
            item = parent;
 | 
						|
        } else {
 | 
						|
            break;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    item
 | 
						|
}
 | 
						|
 | 
						|
struct NodeCollection {
 | 
						|
    next_component_id: u32,
 | 
						|
    components_by_id: HashMap<u32, ItemTreeWeak>,
 | 
						|
    component_ids: HashMap<NonNull<u8>, u32>,
 | 
						|
    all_nodes: Vec<CachedNode>,
 | 
						|
    root_node_id: NodeId,
 | 
						|
    focused_node_tracker: Pin<Box<PropertyTracker<DelegateFocusPropertyTracker>>>,
 | 
						|
}
 | 
						|
 | 
						|
impl NodeCollection {
 | 
						|
    fn focus_node(&mut self, window_adapter_weak: &Weak<WinitWindowAdapter>) -> NodeId {
 | 
						|
        window_adapter_weak
 | 
						|
            .upgrade()
 | 
						|
            .filter(|window_adapter| {
 | 
						|
                window_adapter.winit_window().is_some_and(|winit_window| winit_window.has_focus())
 | 
						|
            })
 | 
						|
            .and_then(|window_adapter| {
 | 
						|
                let window_inner = WindowInner::from_pub(window_adapter.window());
 | 
						|
                window_inner
 | 
						|
                    .focus_item
 | 
						|
                    .borrow()
 | 
						|
                    .upgrade()
 | 
						|
                    .map(|focus_item| {
 | 
						|
                        let parent = accessible_parent_for_item_rc(focus_item);
 | 
						|
                        self.focused_node_tracker
 | 
						|
                            .as_ref()
 | 
						|
                            .evaluate(|| {
 | 
						|
                                parent.accessible_string_property(
 | 
						|
                                    AccessibleStringProperty::DelegateFocus,
 | 
						|
                                )
 | 
						|
                            })
 | 
						|
                            .and_then(|s| s.parse::<usize>().ok())
 | 
						|
                            .and_then(|i| {
 | 
						|
                                i_slint_core::accessibility::accessible_descendents(&parent).nth(i)
 | 
						|
                            })
 | 
						|
                            .unwrap_or(parent)
 | 
						|
                    })
 | 
						|
                    .or_else(|| {
 | 
						|
                        window_inner
 | 
						|
                            .try_component()
 | 
						|
                            .map(|component_rc| ItemRc::new(component_rc, 0))
 | 
						|
                    })
 | 
						|
                    .map(|focus_item| self.find_node_id_by_item_rc(focus_item))
 | 
						|
            })
 | 
						|
            .unwrap_or(self.root_node_id)
 | 
						|
    }
 | 
						|
 | 
						|
    fn item_rc_for_node_id(&self, id: NodeId) -> Option<ItemRc> {
 | 
						|
        let component_id: u32 = (id.0 >> u32::BITS) as _;
 | 
						|
        let index: u32 = (id.0 & u32::MAX as u64) as _;
 | 
						|
        let component = self.components_by_id.get(&component_id)?.upgrade()?;
 | 
						|
        Some(ItemRc::new(component, index))
 | 
						|
    }
 | 
						|
 | 
						|
    fn find_node_id_by_item_rc(&mut self, mut item: ItemRc) -> NodeId {
 | 
						|
        item = accessible_parent_for_item_rc(item);
 | 
						|
 | 
						|
        self.encode_item_node_id(&item)
 | 
						|
    }
 | 
						|
 | 
						|
    fn encode_item_node_id(&mut self, item: &ItemRc) -> NodeId {
 | 
						|
        let component = item.item_tree();
 | 
						|
        let component_ptr = ItemTreeRef::as_ptr(ItemTreeRc::borrow(component));
 | 
						|
        let component_id = match self.component_ids.get(&component_ptr) {
 | 
						|
            Some(&component_id) => component_id,
 | 
						|
            None => {
 | 
						|
                let component_id = self.next_component_id;
 | 
						|
                self.next_component_id += 1;
 | 
						|
                self.component_ids.insert(component_ptr, component_id);
 | 
						|
                self.components_by_id.insert(component_id, ItemTreeRc::downgrade(component));
 | 
						|
                component_id
 | 
						|
            }
 | 
						|
        };
 | 
						|
 | 
						|
        let index = item.index();
 | 
						|
        NodeId((component_id as u64) << u32::BITS | (index as u64 & u32::MAX as u64))
 | 
						|
    }
 | 
						|
 | 
						|
    fn build_node_for_item_recursively(
 | 
						|
        &mut self,
 | 
						|
        item: ItemRc,
 | 
						|
        nodes: &mut Vec<(NodeId, Node)>,
 | 
						|
        popups: &[AccessiblePopup],
 | 
						|
        scale_factor: ScaleFactor,
 | 
						|
        window_position: LogicalPoint,
 | 
						|
    ) -> NodeId {
 | 
						|
        let tracker = Box::pin(PropertyTracker::default());
 | 
						|
 | 
						|
        let mut node = tracker
 | 
						|
            .as_ref()
 | 
						|
            .evaluate(|| self.build_node_without_children(&item, scale_factor, window_position));
 | 
						|
 | 
						|
        let id = self.encode_item_node_id(&item);
 | 
						|
 | 
						|
        let popup_children = popups
 | 
						|
            .iter()
 | 
						|
            .filter_map(|popup| {
 | 
						|
                if popup.parent_node != id {
 | 
						|
                    return None;
 | 
						|
                }
 | 
						|
 | 
						|
                let popup_item = ItemRc::new(popup.component.clone(), 0);
 | 
						|
                Some(self.build_node_for_item_recursively(
 | 
						|
                    popup_item,
 | 
						|
                    nodes,
 | 
						|
                    popups,
 | 
						|
                    scale_factor,
 | 
						|
                    popup.location,
 | 
						|
                ))
 | 
						|
            })
 | 
						|
            .collect::<Vec<_>>();
 | 
						|
 | 
						|
        let children = i_slint_core::accessibility::accessible_descendents(&item)
 | 
						|
            .map(|child| {
 | 
						|
                self.build_node_for_item_recursively(
 | 
						|
                    child,
 | 
						|
                    nodes,
 | 
						|
                    popups,
 | 
						|
                    scale_factor,
 | 
						|
                    window_position,
 | 
						|
                )
 | 
						|
            })
 | 
						|
            .chain(popup_children)
 | 
						|
            .collect::<Vec<NodeId>>();
 | 
						|
 | 
						|
        node.set_children(children.clone());
 | 
						|
 | 
						|
        self.all_nodes.push(CachedNode { id, children, tracker });
 | 
						|
 | 
						|
        nodes.push((id, node));
 | 
						|
 | 
						|
        id
 | 
						|
    }
 | 
						|
 | 
						|
    fn build_new_tree(
 | 
						|
        &mut self,
 | 
						|
        window_adapter_weak: &Weak<WinitWindowAdapter>,
 | 
						|
        property_tracker: Pin<&PropertyTracker<AccessibilitiesPropertyTracker>>,
 | 
						|
    ) -> TreeUpdate {
 | 
						|
        let Some(window_adapter) = window_adapter_weak.upgrade() else {
 | 
						|
            return TreeUpdate {
 | 
						|
                nodes: Default::default(),
 | 
						|
                tree: Default::default(),
 | 
						|
                focus: self.root_node_id,
 | 
						|
            };
 | 
						|
        };
 | 
						|
        let window = window_adapter.window();
 | 
						|
        let window_inner = i_slint_core::window::WindowInner::from_pub(window);
 | 
						|
 | 
						|
        let root_item = ItemRc::new(window_inner.component(), 0);
 | 
						|
 | 
						|
        let popups = window_inner
 | 
						|
            .active_popups()
 | 
						|
            .iter()
 | 
						|
            .filter_map(|popup| {
 | 
						|
                let PopupWindowLocation::ChildWindow(location) = popup.location else {
 | 
						|
                    return None;
 | 
						|
                };
 | 
						|
 | 
						|
                let parent_item = accessible_parent_for_item_rc(popup.parent_item.upgrade()?);
 | 
						|
                let parent_node = self.encode_item_node_id(if parent_item.is_accessible() {
 | 
						|
                    &parent_item
 | 
						|
                } else {
 | 
						|
                    &root_item
 | 
						|
                });
 | 
						|
 | 
						|
                Some(AccessiblePopup { location, parent_node, component: popup.component.clone() })
 | 
						|
            })
 | 
						|
            .collect::<Vec<_>>();
 | 
						|
 | 
						|
        self.all_nodes.clear();
 | 
						|
        let mut nodes = Vec::new();
 | 
						|
 | 
						|
        let root_id = property_tracker.evaluate_as_dependency_root(|| {
 | 
						|
            self.build_node_for_item_recursively(
 | 
						|
                root_item,
 | 
						|
                &mut nodes,
 | 
						|
                &popups,
 | 
						|
                ScaleFactor::new(window.scale_factor()),
 | 
						|
                Default::default(),
 | 
						|
            )
 | 
						|
        });
 | 
						|
        self.root_node_id = root_id;
 | 
						|
 | 
						|
        TreeUpdate {
 | 
						|
            nodes,
 | 
						|
            tree: Some(Tree::new(root_id)),
 | 
						|
            focus: self.focus_node(window_adapter_weak),
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    fn build_node_without_children(
 | 
						|
        &self,
 | 
						|
        item: &ItemRc,
 | 
						|
        scale_factor: ScaleFactor,
 | 
						|
        window_position: LogicalPoint,
 | 
						|
    ) -> Node {
 | 
						|
        let is_checkable = item
 | 
						|
            .accessible_string_property(AccessibleStringProperty::Checkable)
 | 
						|
            .is_some_and(|x| x == "true");
 | 
						|
 | 
						|
        let (role, label) = if let Some(window_item) = item.downcast::<WindowItem>() {
 | 
						|
            (Role::Window, Some(window_item.as_pin_ref().title().to_string()))
 | 
						|
        } else {
 | 
						|
            (
 | 
						|
                match item.accessible_role() {
 | 
						|
                    i_slint_core::items::AccessibleRole::None => Role::Unknown,
 | 
						|
                    i_slint_core::items::AccessibleRole::Button => Role::Button,
 | 
						|
                    i_slint_core::items::AccessibleRole::Checkbox => Role::CheckBox,
 | 
						|
                    i_slint_core::items::AccessibleRole::Combobox => Role::ComboBox,
 | 
						|
                    i_slint_core::items::AccessibleRole::Groupbox => Role::Group,
 | 
						|
                    i_slint_core::items::AccessibleRole::List => Role::ListBox,
 | 
						|
                    i_slint_core::items::AccessibleRole::Slider => Role::Slider,
 | 
						|
                    i_slint_core::items::AccessibleRole::Spinbox => Role::SpinButton,
 | 
						|
                    i_slint_core::items::AccessibleRole::Tab => Role::Tab,
 | 
						|
                    i_slint_core::items::AccessibleRole::TabList => Role::TabList,
 | 
						|
                    i_slint_core::items::AccessibleRole::TabPanel => Role::TabPanel,
 | 
						|
                    i_slint_core::items::AccessibleRole::Text => Role::Label,
 | 
						|
                    i_slint_core::items::AccessibleRole::Table => Role::Table,
 | 
						|
                    i_slint_core::items::AccessibleRole::Tree => Role::Tree,
 | 
						|
                    i_slint_core::items::AccessibleRole::TextInput => {
 | 
						|
                        if let Some(text_input) = i_slint_core::accessibility::find_text_input(item)
 | 
						|
                        {
 | 
						|
                            if !text_input.single_line.get_internal() {
 | 
						|
                                Role::MultilineTextInput
 | 
						|
                            } else {
 | 
						|
                                match text_input.input_type.get_internal() {
 | 
						|
                                    i_slint_core::items::InputType::Decimal
 | 
						|
                                    | i_slint_core::items::InputType::Number => Role::NumberInput,
 | 
						|
                                    i_slint_core::items::InputType::Password => Role::PasswordInput,
 | 
						|
                                    _ => Role::TextInput,
 | 
						|
                                }
 | 
						|
                            }
 | 
						|
                        } else {
 | 
						|
                            Role::TextInput
 | 
						|
                        }
 | 
						|
                    }
 | 
						|
                    i_slint_core::items::AccessibleRole::ProgressIndicator => {
 | 
						|
                        Role::ProgressIndicator
 | 
						|
                    }
 | 
						|
                    i_slint_core::items::AccessibleRole::Switch => Role::Switch,
 | 
						|
                    i_slint_core::items::AccessibleRole::ListItem => Role::ListBoxOption,
 | 
						|
                    i_slint_core::items::AccessibleRole::Image => Role::Image,
 | 
						|
                    _ => Role::Unknown,
 | 
						|
                },
 | 
						|
                item.accessible_string_property(
 | 
						|
                    i_slint_core::accessibility::AccessibleStringProperty::Label,
 | 
						|
                )
 | 
						|
                .map(|x| x.to_string()),
 | 
						|
            )
 | 
						|
        };
 | 
						|
 | 
						|
        let mut node = Node::new(role);
 | 
						|
 | 
						|
        if let Some(label) = label {
 | 
						|
            if role == Role::Label {
 | 
						|
                node.set_value(label);
 | 
						|
            } else {
 | 
						|
                node.set_label(label);
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        if item
 | 
						|
            .accessible_string_property(AccessibleStringProperty::Enabled)
 | 
						|
            .is_some_and(|x| x != "true")
 | 
						|
        {
 | 
						|
            node.set_disabled();
 | 
						|
        }
 | 
						|
 | 
						|
        if !item.is_visible() {
 | 
						|
            node.set_hidden();
 | 
						|
        }
 | 
						|
 | 
						|
        if item.borrow().as_ref().clips_children() {
 | 
						|
            node.set_clips_children();
 | 
						|
        }
 | 
						|
 | 
						|
        let geometry = item.geometry();
 | 
						|
        let absolute_origin = item.map_to_window(geometry.origin) + window_position.to_vector();
 | 
						|
        let physical_origin = (absolute_origin * scale_factor).cast::<f64>();
 | 
						|
        let physical_size = (geometry.size * scale_factor).cast::<f64>();
 | 
						|
        node.set_bounds(accesskit::Rect {
 | 
						|
            x0: physical_origin.x,
 | 
						|
            y0: physical_origin.y,
 | 
						|
            x1: physical_origin.x + physical_size.width,
 | 
						|
            y1: physical_origin.y + physical_size.height,
 | 
						|
        });
 | 
						|
 | 
						|
        let is_checked = is_checkable
 | 
						|
            && item
 | 
						|
                .accessible_string_property(AccessibleStringProperty::Checked)
 | 
						|
                .is_some_and(|x| x == "true");
 | 
						|
        if is_checkable {
 | 
						|
            node.set_toggled(if is_checked { Toggled::True } else { Toggled::False });
 | 
						|
        }
 | 
						|
 | 
						|
        if let Some(description) =
 | 
						|
            item.accessible_string_property(AccessibleStringProperty::Description)
 | 
						|
        {
 | 
						|
            node.set_description(description.to_string());
 | 
						|
        }
 | 
						|
 | 
						|
        if item
 | 
						|
            .accessible_string_property(AccessibleStringProperty::Expandable)
 | 
						|
            .is_some_and(|x| x == "true")
 | 
						|
        {
 | 
						|
            node.set_expanded(
 | 
						|
                item.accessible_string_property(AccessibleStringProperty::Expanded)
 | 
						|
                    .is_some_and(|x| x == "true"),
 | 
						|
            );
 | 
						|
        }
 | 
						|
 | 
						|
        if matches!(
 | 
						|
            role,
 | 
						|
            Role::Button
 | 
						|
                | Role::CheckBox
 | 
						|
                | Role::ComboBox
 | 
						|
                | Role::ListBoxOption
 | 
						|
                | Role::MultilineTextInput
 | 
						|
                | Role::NumberInput
 | 
						|
                | Role::PasswordInput
 | 
						|
                | Role::Slider
 | 
						|
                | Role::SpinButton
 | 
						|
                | Role::Tab
 | 
						|
                | Role::TextInput
 | 
						|
        ) {
 | 
						|
            node.add_action(Action::Focus);
 | 
						|
        }
 | 
						|
 | 
						|
        if let Some(min) = item
 | 
						|
            .accessible_string_property(AccessibleStringProperty::ValueMinimum)
 | 
						|
            .and_then(|min| min.parse().ok())
 | 
						|
        {
 | 
						|
            node.set_min_numeric_value(min);
 | 
						|
        }
 | 
						|
        if let Some(max) = item
 | 
						|
            .accessible_string_property(AccessibleStringProperty::ValueMaximum)
 | 
						|
            .and_then(|max| max.parse().ok())
 | 
						|
        {
 | 
						|
            node.set_max_numeric_value(max);
 | 
						|
        }
 | 
						|
        if let Some(step) = item
 | 
						|
            .accessible_string_property(AccessibleStringProperty::ValueStep)
 | 
						|
            .and_then(|step| step.parse().ok())
 | 
						|
        {
 | 
						|
            node.set_numeric_value_step(step);
 | 
						|
        }
 | 
						|
 | 
						|
        if let Some(value) = item.accessible_string_property(AccessibleStringProperty::Value) {
 | 
						|
            if let Ok(value) = value.parse() {
 | 
						|
                node.set_numeric_value(value);
 | 
						|
            } else {
 | 
						|
                node.set_value(value.to_string());
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        if let Some(placeholder) = item
 | 
						|
            .accessible_string_property(AccessibleStringProperty::PlaceholderText)
 | 
						|
            .filter(|x| !x.is_empty())
 | 
						|
        {
 | 
						|
            node.set_placeholder(placeholder.to_string());
 | 
						|
        }
 | 
						|
 | 
						|
        if item
 | 
						|
            .accessible_string_property(AccessibleStringProperty::ReadOnly)
 | 
						|
            .is_some_and(|x| x == "true")
 | 
						|
        {
 | 
						|
            node.set_read_only();
 | 
						|
        }
 | 
						|
 | 
						|
        if item
 | 
						|
            .accessible_string_property(AccessibleStringProperty::ItemSelectable)
 | 
						|
            .is_some_and(|x| x == "true")
 | 
						|
        {
 | 
						|
            node.set_selected(
 | 
						|
                item.accessible_string_property(AccessibleStringProperty::ItemSelected)
 | 
						|
                    .is_some_and(|x| x == "true"),
 | 
						|
            );
 | 
						|
        }
 | 
						|
 | 
						|
        if let Some(position_in_set) = item
 | 
						|
            .accessible_string_property(AccessibleStringProperty::ItemIndex)
 | 
						|
            .and_then(|s| s.parse::<usize>().ok())
 | 
						|
        {
 | 
						|
            node.set_position_in_set(position_in_set);
 | 
						|
            let mut item = item.clone();
 | 
						|
            while let Some(parent) = item.parent_item() {
 | 
						|
                if !parent.is_accessible() {
 | 
						|
                    item = parent;
 | 
						|
                    continue;
 | 
						|
                }
 | 
						|
                if let Some(size_of_set) = parent
 | 
						|
                    .accessible_string_property(AccessibleStringProperty::ItemCount)
 | 
						|
                    .and_then(|s| s.parse::<usize>().ok())
 | 
						|
                {
 | 
						|
                    node.set_size_of_set(size_of_set);
 | 
						|
                }
 | 
						|
                break;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        let supported = item.supported_accessibility_actions();
 | 
						|
        if supported.contains(SupportedAccessibilityAction::Default) {
 | 
						|
            node.add_action(accesskit::Action::Click);
 | 
						|
        }
 | 
						|
        if supported.contains(SupportedAccessibilityAction::Decrement) {
 | 
						|
            node.add_action(accesskit::Action::Decrement);
 | 
						|
        }
 | 
						|
        if supported.contains(SupportedAccessibilityAction::Increment) {
 | 
						|
            node.add_action(accesskit::Action::Increment);
 | 
						|
        }
 | 
						|
        if supported.contains(SupportedAccessibilityAction::SetValue) {
 | 
						|
            node.add_action(accesskit::Action::SetValue);
 | 
						|
        }
 | 
						|
        if supported.contains(SupportedAccessibilityAction::ReplaceSelectedText) {
 | 
						|
            node.add_action(accesskit::Action::ReplaceSelectedText);
 | 
						|
        }
 | 
						|
        if supported.contains(SupportedAccessibilityAction::Expand) {
 | 
						|
            node.add_action(accesskit::Action::Expand);
 | 
						|
        }
 | 
						|
 | 
						|
        node
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
struct AccessibilitiesPropertyTracker {
 | 
						|
    window_adapter_weak: Weak<WinitWindowAdapter>,
 | 
						|
}
 | 
						|
 | 
						|
impl i_slint_core::properties::PropertyDirtyHandler for AccessibilitiesPropertyTracker {
 | 
						|
    fn notify(self: Pin<&Self>) {
 | 
						|
        let win = self.window_adapter_weak.clone();
 | 
						|
        i_slint_core::timers::Timer::single_shot(Default::default(), move || {
 | 
						|
            WinitWindowAdapter::with_access_kit_adapter_from_weak_window_adapter(
 | 
						|
                win,
 | 
						|
                |self_cell| {
 | 
						|
                    self_cell.borrow_mut().rebuild_tree_of_dirty_nodes();
 | 
						|
                },
 | 
						|
            );
 | 
						|
        })
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
struct DelegateFocusPropertyTracker {
 | 
						|
    window_adapter_weak: Weak<WinitWindowAdapter>,
 | 
						|
}
 | 
						|
 | 
						|
impl i_slint_core::properties::PropertyDirtyHandler for DelegateFocusPropertyTracker {
 | 
						|
    fn notify(self: Pin<&Self>) {
 | 
						|
        let win = self.window_adapter_weak.clone();
 | 
						|
        i_slint_core::timers::Timer::single_shot(Default::default(), move || {
 | 
						|
            WinitWindowAdapter::with_access_kit_adapter_from_weak_window_adapter(
 | 
						|
                win,
 | 
						|
                |self_cell| {
 | 
						|
                    self_cell.borrow_mut().handle_focus_item_change();
 | 
						|
                },
 | 
						|
            );
 | 
						|
        })
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
struct CachedNode {
 | 
						|
    id: NodeId,
 | 
						|
    children: Vec<NodeId>,
 | 
						|
    tracker: Pin<Box<PropertyTracker>>,
 | 
						|
}
 | 
						|
 | 
						|
impl From<accesskit_winit::Event> for SlintUserEvent {
 | 
						|
    fn from(value: accesskit_winit::Event) -> Self {
 | 
						|
        SlintUserEvent(crate::event_loop::CustomEvent::Accesskit(value))
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
pub enum DeferredAccessKitAction {
 | 
						|
    SetFocus(ItemRc),
 | 
						|
    InvokeAccessibleAction(ItemRc, AccessibilityAction),
 | 
						|
}
 | 
						|
 | 
						|
impl DeferredAccessKitAction {
 | 
						|
    pub fn invoke(&self, window: &Window) {
 | 
						|
        match self {
 | 
						|
            DeferredAccessKitAction::SetFocus(item) => {
 | 
						|
                WindowInner::from_pub(window).set_focus_item(item, true);
 | 
						|
            }
 | 
						|
            DeferredAccessKitAction::InvokeAccessibleAction(item, accessibility_action) => {
 | 
						|
                item.accessible_action(accessibility_action);
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
struct AccessiblePopup {
 | 
						|
    location: LogicalPoint,
 | 
						|
    parent_node: NodeId,
 | 
						|
    component: ItemTreeRc,
 | 
						|
}
 |