pub mod accessibility_manager;
pub use accessibility_manager::*;
use crate::{
dom::{DioxusDOM, DioxusNode},
types::AccessibilityId,
};
use freya_native_core::{node::NodeType, real_dom::NodeImmutable, tags::TagName, NodeId};
use freya_node_state::AccessibilityNodeState;
use torin::torin::Torin;
#[derive(PartialEq)]
pub enum AccessibilityFocusDirection {
Forward,
Backward,
}
trait NodeAccessibility {
fn get_inner_texts(&self) -> Option<String>;
fn get_accessibility_children(&self) -> Vec<AccessibilityId>;
}
impl NodeAccessibility for DioxusNode<'_> {
fn get_inner_texts(&self) -> Option<String> {
let children = self.children();
let first_child = children.first()?;
let node_type = first_child.node_type();
if let NodeType::Text(text) = &*node_type {
Some(text.to_owned())
} else {
None
}
}
fn get_accessibility_children(&self) -> Vec<AccessibilityId> {
self.children()
.iter()
.filter_map(|child| {
if child.node_type().is_visible_element() {
let node_accessibility = &*child.get::<AccessibilityNodeState>().unwrap();
node_accessibility.accessibility_id
} else {
None
}
})
.collect::<Vec<AccessibilityId>>()
}
}
pub fn process_accessibility(
layout: &Torin<NodeId>,
rdom: &DioxusDOM,
accessibility_manager: &mut AccessibilityManager,
) {
rdom.traverse_depth_first_advanced(|node| {
if !node.node_type().is_element() {
return false;
}
let layout_node = layout.get(node.id()).unwrap();
let node_accessibility = &*node.get::<AccessibilityNodeState>().unwrap();
if let Some(accessibility_id) = node_accessibility.accessibility_id {
accessibility_manager.add_node(
&node,
layout_node,
accessibility_id,
node_accessibility,
);
}
if let Some(tag) = node.node_type().tag() {
if *tag == TagName::Paragraph || *tag == TagName::Label {
return false;
}
}
true
});
}