#![allow(non_upper_case_globals)]
use accesskit::{
Action, ActionData, ActionRequest, Live, NodeId, NodeIdContent, Orientation, Point, Role,
Toggled,
};
use accesskit_consumer::{FilterResult, Node, TreeState};
use paste::paste;
use std::sync::{atomic::Ordering, Arc, Weak};
use windows::{
core::*,
Win32::{Foundation::*, System::Com::*, UI::Accessibility::*},
};
use crate::{
context::Context,
filters::{filter, filter_with_root_exception},
text::PlatformRange as PlatformTextRange,
util::*,
};
const RUNTIME_ID_SIZE: usize = 3;
fn runtime_id_from_node_id(id: NodeId) -> [i32; RUNTIME_ID_SIZE] {
static_assertions::assert_eq_size!(NodeIdContent, u64);
let id = id.0;
[
UiaAppendRuntimeId as _,
((id >> 32) & 0xFFFFFFFF) as _,
(id & 0xFFFFFFFF) as _,
]
}
pub(crate) struct NodeWrapper<'a>(pub(crate) &'a Node<'a>);
impl<'a> NodeWrapper<'a> {
fn control_type(&self) -> UIA_CONTROLTYPE_ID {
let role = self.0.role();
match role {
Role::Unknown => UIA_CustomControlTypeId,
Role::TextRun => UIA_CustomControlTypeId,
Role::Cell => UIA_DataItemControlTypeId,
Role::Label => UIA_TextControlTypeId,
Role::Image => UIA_ImageControlTypeId,
Role::Link => UIA_HyperlinkControlTypeId,
Role::Row => UIA_DataItemControlTypeId,
Role::ListItem => UIA_ListItemControlTypeId,
Role::ListMarker => UIA_GroupControlTypeId,
Role::TreeItem => UIA_TreeItemControlTypeId,
Role::ListBoxOption => UIA_ListItemControlTypeId,
Role::MenuItem => UIA_MenuItemControlTypeId,
Role::MenuListOption => UIA_ListItemControlTypeId,
Role::Paragraph => UIA_GroupControlTypeId,
Role::GenericContainer => UIA_GroupControlTypeId,
Role::CheckBox => UIA_CheckBoxControlTypeId,
Role::RadioButton => UIA_RadioButtonControlTypeId,
Role::TextInput
| Role::MultilineTextInput
| Role::SearchInput
| Role::DateInput
| Role::DateTimeInput
| Role::WeekInput
| Role::MonthInput
| Role::TimeInput
| Role::EmailInput
| Role::NumberInput
| Role::PasswordInput
| Role::PhoneNumberInput
| Role::UrlInput => UIA_EditControlTypeId,
Role::Button | Role::DefaultButton => UIA_ButtonControlTypeId,
Role::Pane => UIA_PaneControlTypeId,
Role::RowHeader => UIA_DataItemControlTypeId,
Role::ColumnHeader => UIA_DataItemControlTypeId,
Role::RowGroup => UIA_GroupControlTypeId,
Role::List => UIA_ListControlTypeId,
Role::Table => UIA_TableControlTypeId,
Role::LayoutTableCell => UIA_DataItemControlTypeId,
Role::LayoutTableRow => UIA_DataItemControlTypeId,
Role::LayoutTable => UIA_TableControlTypeId,
Role::Switch => UIA_ButtonControlTypeId,
Role::Menu => UIA_MenuControlTypeId,
Role::Abbr => UIA_TextControlTypeId,
Role::Alert => UIA_TextControlTypeId,
Role::AlertDialog => {
UIA_TextControlTypeId
}
Role::Application => UIA_PaneControlTypeId,
Role::Article => UIA_GroupControlTypeId,
Role::Audio => UIA_GroupControlTypeId,
Role::Banner => UIA_GroupControlTypeId,
Role::Blockquote => UIA_GroupControlTypeId,
Role::Canvas => UIA_ImageControlTypeId,
Role::Caption => UIA_TextControlTypeId,
Role::Caret => UIA_GroupControlTypeId,
Role::Code => UIA_TextControlTypeId,
Role::ColorWell => UIA_ButtonControlTypeId,
Role::ComboBox | Role::EditableComboBox => UIA_ComboBoxControlTypeId,
Role::Complementary => UIA_GroupControlTypeId,
Role::Comment => UIA_GroupControlTypeId,
Role::ContentDeletion => UIA_GroupControlTypeId,
Role::ContentInsertion => UIA_GroupControlTypeId,
Role::ContentInfo => UIA_GroupControlTypeId,
Role::Definition => UIA_GroupControlTypeId,
Role::DescriptionList => UIA_ListControlTypeId,
Role::DescriptionListDetail => UIA_TextControlTypeId,
Role::DescriptionListTerm => UIA_ListItemControlTypeId,
Role::Details => UIA_GroupControlTypeId,
Role::Dialog => UIA_PaneControlTypeId,
Role::Directory => UIA_ListControlTypeId,
Role::DisclosureTriangle => UIA_ButtonControlTypeId,
Role::Document | Role::Terminal => UIA_DocumentControlTypeId,
Role::EmbeddedObject => UIA_PaneControlTypeId,
Role::Emphasis => UIA_TextControlTypeId,
Role::Feed => UIA_GroupControlTypeId,
Role::FigureCaption => UIA_TextControlTypeId,
Role::Figure => UIA_GroupControlTypeId,
Role::Footer => UIA_GroupControlTypeId,
Role::FooterAsNonLandmark => UIA_GroupControlTypeId,
Role::Form => UIA_GroupControlTypeId,
Role::Grid => UIA_DataGridControlTypeId,
Role::Group => UIA_GroupControlTypeId,
Role::Header => UIA_GroupControlTypeId,
Role::HeaderAsNonLandmark => UIA_GroupControlTypeId,
Role::Heading => UIA_TextControlTypeId,
Role::Iframe => UIA_DocumentControlTypeId,
Role::IframePresentational => UIA_GroupControlTypeId,
Role::ImeCandidate => UIA_PaneControlTypeId,
Role::Keyboard => UIA_PaneControlTypeId,
Role::Legend => UIA_TextControlTypeId,
Role::LineBreak => UIA_TextControlTypeId,
Role::ListBox => UIA_ListControlTypeId,
Role::Log => UIA_GroupControlTypeId,
Role::Main => UIA_GroupControlTypeId,
Role::Mark => UIA_TextControlTypeId,
Role::Marquee => UIA_TextControlTypeId,
Role::Math => UIA_GroupControlTypeId,
Role::MenuBar => UIA_MenuBarControlTypeId,
Role::MenuItemCheckBox => UIA_CheckBoxControlTypeId,
Role::MenuItemRadio => UIA_RadioButtonControlTypeId,
Role::MenuListPopup => UIA_ListControlTypeId,
Role::Meter => UIA_ProgressBarControlTypeId,
Role::Navigation => UIA_GroupControlTypeId,
Role::Note => UIA_GroupControlTypeId,
Role::PluginObject => UIA_GroupControlTypeId,
Role::Portal => UIA_ButtonControlTypeId,
Role::Pre => UIA_GroupControlTypeId,
Role::ProgressIndicator => UIA_ProgressBarControlTypeId,
Role::RadioGroup => UIA_GroupControlTypeId,
Role::Region => UIA_GroupControlTypeId,
Role::RootWebArea => UIA_DocumentControlTypeId,
Role::Ruby => UIA_GroupControlTypeId,
Role::RubyAnnotation => {
UIA_TextControlTypeId
}
Role::ScrollBar => UIA_ScrollBarControlTypeId,
Role::ScrollView => UIA_PaneControlTypeId,
Role::Search => UIA_GroupControlTypeId,
Role::Section => UIA_GroupControlTypeId,
Role::Slider => UIA_SliderControlTypeId,
Role::SpinButton => UIA_SpinnerControlTypeId,
Role::Splitter => UIA_SeparatorControlTypeId,
Role::Status => UIA_StatusBarControlTypeId,
Role::Strong => UIA_TextControlTypeId,
Role::Suggestion => UIA_GroupControlTypeId,
Role::SvgRoot => UIA_ImageControlTypeId,
Role::Tab => UIA_TabItemControlTypeId,
Role::TabList => UIA_TabControlTypeId,
Role::TabPanel => UIA_PaneControlTypeId,
Role::Term => UIA_ListItemControlTypeId,
Role::Time => UIA_TextControlTypeId,
Role::Timer => UIA_PaneControlTypeId,
Role::TitleBar => UIA_PaneControlTypeId,
Role::Toolbar => UIA_ToolBarControlTypeId,
Role::Tooltip => UIA_ToolTipControlTypeId,
Role::Tree => UIA_TreeControlTypeId,
Role::TreeGrid => UIA_DataGridControlTypeId,
Role::Video => UIA_GroupControlTypeId,
Role::WebView => UIA_DocumentControlTypeId,
Role::Window => {
UIA_WindowControlTypeId
}
Role::PdfActionableHighlight => UIA_CustomControlTypeId,
Role::PdfRoot => UIA_DocumentControlTypeId,
Role::GraphicsDocument => UIA_DocumentControlTypeId,
Role::GraphicsObject => UIA_PaneControlTypeId,
Role::GraphicsSymbol => UIA_ImageControlTypeId,
Role::DocAbstract => UIA_GroupControlTypeId,
Role::DocAcknowledgements => UIA_GroupControlTypeId,
Role::DocAfterword => UIA_GroupControlTypeId,
Role::DocAppendix => UIA_GroupControlTypeId,
Role::DocBackLink => UIA_HyperlinkControlTypeId,
Role::DocBiblioEntry => UIA_ListItemControlTypeId,
Role::DocBibliography => UIA_GroupControlTypeId,
Role::DocBiblioRef => UIA_HyperlinkControlTypeId,
Role::DocChapter => UIA_GroupControlTypeId,
Role::DocColophon => UIA_GroupControlTypeId,
Role::DocConclusion => UIA_GroupControlTypeId,
Role::DocCover => UIA_ImageControlTypeId,
Role::DocCredit => UIA_GroupControlTypeId,
Role::DocCredits => UIA_GroupControlTypeId,
Role::DocDedication => UIA_GroupControlTypeId,
Role::DocEndnote => UIA_ListItemControlTypeId,
Role::DocEndnotes => UIA_GroupControlTypeId,
Role::DocEpigraph => UIA_GroupControlTypeId,
Role::DocEpilogue => UIA_GroupControlTypeId,
Role::DocErrata => UIA_GroupControlTypeId,
Role::DocExample => UIA_GroupControlTypeId,
Role::DocFootnote => UIA_ListItemControlTypeId,
Role::DocForeword => UIA_GroupControlTypeId,
Role::DocGlossary => UIA_GroupControlTypeId,
Role::DocGlossRef => UIA_HyperlinkControlTypeId,
Role::DocIndex => UIA_GroupControlTypeId,
Role::DocIntroduction => UIA_GroupControlTypeId,
Role::DocNoteRef => UIA_HyperlinkControlTypeId,
Role::DocNotice => UIA_GroupControlTypeId,
Role::DocPageBreak => UIA_SeparatorControlTypeId,
Role::DocPageFooter => UIA_GroupControlTypeId,
Role::DocPageHeader => UIA_GroupControlTypeId,
Role::DocPageList => UIA_GroupControlTypeId,
Role::DocPart => UIA_GroupControlTypeId,
Role::DocPreface => UIA_GroupControlTypeId,
Role::DocPrologue => UIA_GroupControlTypeId,
Role::DocPullquote => UIA_GroupControlTypeId,
Role::DocQna => UIA_GroupControlTypeId,
Role::DocSubtitle => UIA_GroupControlTypeId,
Role::DocTip => UIA_GroupControlTypeId,
Role::DocToc => UIA_GroupControlTypeId,
Role::ListGrid => UIA_DataGridControlTypeId,
}
}
fn localized_control_type(&self) -> Option<String> {
self.0.role_description()
}
pub(crate) fn name(&self) -> Option<String> {
if self.0.label_comes_from_value() {
self.0.value()
} else {
self.0.label()
}
}
fn description(&self) -> Option<String> {
self.0.description()
}
fn placeholder(&self) -> Option<String> {
self.0.placeholder()
}
fn is_content_element(&self) -> bool {
filter(self.0) == FilterResult::Include
}
fn is_enabled(&self) -> bool {
!self.0.is_disabled()
}
fn is_focusable(&self) -> bool {
self.0.is_focusable()
}
fn is_focused(&self) -> bool {
self.0.is_focused()
}
fn live_setting(&self) -> LiveSetting {
let live = self.0.live();
match live {
Live::Off => Off,
Live::Polite => Polite,
Live::Assertive => Assertive,
}
}
fn automation_id(&self) -> Option<&str> {
self.0.author_id()
}
fn class_name(&self) -> Option<&str> {
self.0.class_name()
}
fn orientation(&self) -> OrientationType {
match self.0.orientation() {
Some(Orientation::Horizontal) => OrientationType_Horizontal,
Some(Orientation::Vertical) => OrientationType_Vertical,
None => OrientationType_None,
}
}
fn is_toggle_pattern_supported(&self) -> bool {
self.0.toggled().is_some() && !self.is_selection_item_pattern_supported()
}
fn toggle_state(&self) -> ToggleState {
match self.0.toggled().unwrap() {
Toggled::False => ToggleState_Off,
Toggled::True => ToggleState_On,
Toggled::Mixed => ToggleState_Indeterminate,
}
}
fn is_invoke_pattern_supported(&self) -> bool {
self.0.is_invocable()
}
fn is_value_pattern_supported(&self) -> bool {
self.0.has_value() && !self.0.label_comes_from_value()
}
fn is_range_value_pattern_supported(&self) -> bool {
self.0.numeric_value().is_some()
}
fn value(&self) -> String {
self.0.value().unwrap()
}
fn is_read_only(&self) -> bool {
self.0.is_read_only()
}
fn numeric_value(&self) -> f64 {
self.0.numeric_value().unwrap()
}
fn min_numeric_value(&self) -> f64 {
self.0.min_numeric_value().unwrap_or(0.0)
}
fn max_numeric_value(&self) -> f64 {
self.0.max_numeric_value().unwrap_or(0.0)
}
fn numeric_value_step(&self) -> f64 {
self.0.numeric_value_step().unwrap_or(0.0)
}
fn numeric_value_jump(&self) -> f64 {
self.0
.numeric_value_jump()
.unwrap_or_else(|| self.numeric_value_step())
}
fn is_selection_item_pattern_supported(&self) -> bool {
match self.0.role() {
Role::RadioButton | Role::MenuItemRadio => {
matches!(self.0.toggled(), Some(Toggled::True | Toggled::False))
}
Role::ListBoxOption
| Role::ListItem
| Role::MenuListOption
| Role::Tab
| Role::TreeItem => self.0.is_selected().is_some(),
_ => false,
}
}
fn is_selected(&self) -> bool {
match self.0.role() {
Role::RadioButton | Role::MenuItemRadio => self.0.toggled() == Some(Toggled::True),
_ => self.0.is_selected().unwrap_or(false),
}
}
fn is_text_pattern_supported(&self) -> bool {
self.0.supports_text_ranges()
}
pub(crate) fn enqueue_property_changes(
&self,
queue: &mut Vec<QueuedEvent>,
element: &IRawElementProviderSimple,
old: &NodeWrapper,
) {
self.enqueue_simple_property_changes(queue, element, old);
self.enqueue_pattern_property_changes(queue, element, old);
self.enqueue_property_implied_events(queue, element, old);
}
fn enqueue_property_implied_events(
&self,
queue: &mut Vec<QueuedEvent>,
element: &IRawElementProviderSimple,
old: &NodeWrapper,
) {
if self.is_selection_item_pattern_supported()
&& self.is_selected()
&& !(old.is_selection_item_pattern_supported() && old.is_selected())
{
queue.push(QueuedEvent::Simple {
element: element.clone(),
event_id: UIA_SelectionItem_ElementSelectedEventId,
});
}
if self.is_text_pattern_supported()
&& old.is_text_pattern_supported()
&& self.0.raw_text_selection() != old.0.raw_text_selection()
{
queue.push(QueuedEvent::Simple {
element: element.clone(),
event_id: UIA_Text_TextSelectionChangedEventId,
});
}
}
fn enqueue_property_change(
&self,
queue: &mut Vec<QueuedEvent>,
element: &IRawElementProviderSimple,
property_id: UIA_PROPERTY_ID,
old_value: Variant,
new_value: Variant,
) {
let old_value: VARIANT = old_value.into();
let new_value: VARIANT = new_value.into();
queue.push(QueuedEvent::PropertyChanged {
element: element.clone(),
property_id,
old_value,
new_value,
});
}
}
#[implement(
IRawElementProviderSimple,
IRawElementProviderFragment,
IRawElementProviderFragmentRoot,
IToggleProvider,
IInvokeProvider,
IValueProvider,
IRangeValueProvider,
ISelectionItemProvider,
ITextProvider
)]
pub(crate) struct PlatformNode {
pub(crate) context: Weak<Context>,
pub(crate) node_id: Option<NodeId>,
}
impl PlatformNode {
pub(crate) fn new(context: &Arc<Context>, node_id: NodeId) -> Self {
Self {
context: Arc::downgrade(context),
node_id: Some(node_id),
}
}
pub(crate) fn unspecified_root(context: &Arc<Context>) -> Self {
Self {
context: Arc::downgrade(context),
node_id: None,
}
}
fn upgrade_context(&self) -> Result<Arc<Context>> {
upgrade(&self.context)
}
fn with_tree_state_and_context<F, T>(&self, f: F) -> Result<T>
where
F: FnOnce(&TreeState, &Context) -> Result<T>,
{
let context = self.upgrade_context()?;
let tree = context.read_tree();
f(tree.state(), &context)
}
fn with_tree_state<F, T>(&self, f: F) -> Result<T>
where
F: FnOnce(&TreeState) -> Result<T>,
{
self.with_tree_state_and_context(|state, _| f(state))
}
fn node<'a>(&self, state: &'a TreeState) -> Result<Node<'a>> {
if let Some(id) = self.node_id {
if let Some(node) = state.node_by_id(id) {
Ok(node)
} else {
Err(element_not_available())
}
} else {
Ok(state.root())
}
}
fn resolve_with_context<F, T>(&self, f: F) -> Result<T>
where
for<'a> F: FnOnce(Node<'a>, &Context) -> Result<T>,
{
self.with_tree_state_and_context(|state, context| {
let node = self.node(state)?;
f(node, context)
})
}
fn resolve_with_tree_state_and_context<F, T>(&self, f: F) -> Result<T>
where
for<'a> F: FnOnce(Node<'a>, &TreeState, &Context) -> Result<T>,
{
self.with_tree_state_and_context(|state, context| {
let node = self.node(state)?;
f(node, state, context)
})
}
fn resolve<F, T>(&self, f: F) -> Result<T>
where
for<'a> F: FnOnce(Node<'a>) -> Result<T>,
{
self.resolve_with_context(|node, _| f(node))
}
fn resolve_with_context_for_text_pattern<F, T>(&self, f: F) -> Result<T>
where
for<'a> F: FnOnce(Node<'a>, &Context) -> Result<T>,
{
self.with_tree_state_and_context(|state, context| {
let node = self.node(state)?;
if node.supports_text_ranges() {
f(node, context)
} else {
Err(element_not_available())
}
})
}
fn resolve_for_text_pattern<F, T>(&self, f: F) -> Result<T>
where
for<'a> F: FnOnce(Node<'a>) -> Result<T>,
{
self.resolve_with_context_for_text_pattern(|node, _| f(node))
}
fn do_action<F>(&self, f: F) -> Result<()>
where
F: FnOnce() -> (Action, Option<ActionData>),
{
let context = self.upgrade_context()?;
if context.is_placeholder.load(Ordering::SeqCst) {
return Ok(());
}
let tree = context.read_tree();
let node_id = if let Some(id) = self.node_id {
if !tree.state().has_node(id) {
return Err(element_not_available());
}
id
} else {
tree.state().root_id()
};
drop(tree);
let (action, data) = f();
let request = ActionRequest {
target: node_id,
action,
data,
};
context.do_action(request);
Ok(())
}
fn click(&self) -> Result<()> {
self.do_action(|| (Action::Click, None))
}
fn relative(&self, node_id: NodeId) -> Self {
Self {
context: self.context.clone(),
node_id: Some(node_id),
}
}
fn is_root(&self, state: &TreeState) -> bool {
self.node_id.map_or(false, |id| id == state.root_id())
}
}
#[allow(non_snake_case)]
impl IRawElementProviderSimple_Impl for PlatformNode_Impl {
fn ProviderOptions(&self) -> Result<ProviderOptions> {
Ok(ProviderOptions_ServerSideProvider)
}
fn GetPatternProvider(&self, pattern_id: UIA_PATTERN_ID) -> Result<IUnknown> {
self.pattern_provider(pattern_id)
}
fn GetPropertyValue(&self, property_id: UIA_PROPERTY_ID) -> Result<VARIANT> {
self.resolve_with_tree_state_and_context(|node, state, context| {
let wrapper = NodeWrapper(&node);
let mut result = wrapper.get_property_value(property_id);
if result.is_empty() {
if node.is_root() {
match property_id {
UIA_NamePropertyId => {
result = window_title(context.hwnd).into();
}
UIA_NativeWindowHandlePropertyId => {
result = (context.hwnd.0 .0 as i32).into();
}
_ => (),
}
}
match property_id {
UIA_FrameworkIdPropertyId => result = state.toolkit_name().into(),
UIA_ProviderDescriptionPropertyId => {
result = app_and_toolkit_description(state).into()
}
_ => (),
}
}
Ok(result.into())
})
}
fn HostRawElementProvider(&self) -> Result<IRawElementProviderSimple> {
self.with_tree_state_and_context(|state, context| {
if self.is_root(state) {
unsafe { UiaHostProviderFromHwnd(context.hwnd.0) }
} else {
Err(Error::empty())
}
})
}
}
#[allow(non_snake_case)]
impl IRawElementProviderFragment_Impl for PlatformNode_Impl {
fn Navigate(&self, direction: NavigateDirection) -> Result<IRawElementProviderFragment> {
self.resolve(|node| {
let result = match direction {
NavigateDirection_Parent => node.filtered_parent(&filter_with_root_exception),
NavigateDirection_NextSibling => node.following_filtered_siblings(&filter).next(),
NavigateDirection_PreviousSibling => {
node.preceding_filtered_siblings(&filter).next()
}
NavigateDirection_FirstChild => node.filtered_children(&filter).next(),
NavigateDirection_LastChild => node.filtered_children(&filter).next_back(),
_ => None,
};
match result {
Some(result) => Ok(self.relative(result.id()).into()),
None => Err(Error::empty()),
}
})
}
fn GetRuntimeId(&self) -> Result<*mut SAFEARRAY> {
let node_id = if let Some(id) = self.node_id {
id
} else {
return Err(not_implemented());
};
let runtime_id = runtime_id_from_node_id(node_id);
Ok(safe_array_from_i32_slice(&runtime_id))
}
fn BoundingRectangle(&self) -> Result<UiaRect> {
self.resolve_with_context(|node, context| {
let rect = node.bounding_box().map_or(UiaRect::default(), |rect| {
let client_top_left = context.client_top_left();
UiaRect {
left: rect.x0 + client_top_left.x,
top: rect.y0 + client_top_left.y,
width: rect.width(),
height: rect.height(),
}
});
Ok(rect)
})
}
fn GetEmbeddedFragmentRoots(&self) -> Result<*mut SAFEARRAY> {
Ok(std::ptr::null_mut())
}
fn SetFocus(&self) -> Result<()> {
self.do_action(|| (Action::Focus, None))
}
fn FragmentRoot(&self) -> Result<IRawElementProviderFragmentRoot> {
self.with_tree_state(|state| {
if self.is_root(state) {
unsafe { self.cast() }
} else {
let root_id = state.root_id();
Ok(self.relative(root_id).into())
}
})
}
}
#[allow(non_snake_case)]
impl IRawElementProviderFragmentRoot_Impl for PlatformNode_Impl {
fn ElementProviderFromPoint(&self, x: f64, y: f64) -> Result<IRawElementProviderFragment> {
self.resolve_with_context(|node, context| {
let client_top_left = context.client_top_left();
let point = Point::new(x - client_top_left.x, y - client_top_left.y);
let point = node.transform().inverse() * point;
node.node_at_point(point, &filter).map_or_else(
|| Err(Error::empty()),
|node| Ok(self.relative(node.id()).into()),
)
})
}
fn GetFocus(&self) -> Result<IRawElementProviderFragment> {
self.with_tree_state(|state| {
if let Some(id) = state.focus_id() {
let self_id = if let Some(id) = self.node_id {
id
} else {
state.root_id()
};
if id != self_id {
return Ok(self.relative(id).into());
}
}
Err(Error::empty())
})
}
}
macro_rules! properties {
($(($base_id:ident, $m:ident)),+) => {
impl NodeWrapper<'_> {
fn get_property_value(&self, property_id: UIA_PROPERTY_ID) -> Variant {
match property_id {
$(paste! { [< UIA_ $base_id PropertyId>] } => {
self.$m().into()
})*
_ => Variant::empty()
}
}
fn enqueue_simple_property_changes(
&self,
queue: &mut Vec<QueuedEvent>,
element: &IRawElementProviderSimple,
old: &NodeWrapper,
) {
$({
let old_value = old.$m();
let new_value = self.$m();
if old_value != new_value {
self.enqueue_property_change(
queue,
element,
paste! { [<UIA_ $base_id PropertyId>] },
old_value.into(),
new_value.into(),
);
}
})*
}
}
};
}
macro_rules! patterns {
($(($base_pattern_id:ident, $is_supported:ident, (
$(($base_property_id:ident, $getter:ident, $com_type:ident)),*
), (
$($extra_trait_method:item),*
))),+) => {
impl PlatformNode {
fn pattern_provider(&self, pattern_id: UIA_PATTERN_ID) -> Result<IUnknown> {
self.resolve(|node| {
let wrapper = NodeWrapper(&node);
match pattern_id {
$(paste! { [< UIA_ $base_pattern_id PatternId>] } => {
if wrapper.$is_supported() {
let intermediate: paste! { [< I $base_pattern_id Provider>] } =
unsafe { self.cast() }?;
return intermediate.cast();
}
})*
_ => (),
}
Err(Error::empty())
})
}
}
impl NodeWrapper<'_> {
fn enqueue_pattern_property_changes(
&self,
queue: &mut Vec<QueuedEvent>,
element: &IRawElementProviderSimple,
old: &NodeWrapper,
) {
$(if self.$is_supported() && old.$is_supported() {
$({
let old_value = old.$getter();
let new_value = self.$getter();
if old_value != new_value {
self.enqueue_property_change(
queue,
element,
paste! { [<UIA_ $base_pattern_id $base_property_id PropertyId>] },
old_value.into(),
new_value.into(),
);
}
})*
})*
}
}
paste! {
$(#[allow(non_snake_case)]
impl [< I $base_pattern_id Provider_Impl>] for PlatformNode_Impl {
$(fn $base_property_id(&self) -> Result<$com_type> {
self.resolve(|node| {
let wrapper = NodeWrapper(&node);
Ok(wrapper.$getter().into())
})
})*
$($extra_trait_method)*
})*
}
};
}
properties! {
(ControlType, control_type),
(LocalizedControlType, localized_control_type),
(Name, name),
(FullDescription, description),
(HelpText, placeholder),
(IsContentElement, is_content_element),
(IsControlElement, is_content_element),
(IsEnabled, is_enabled),
(IsKeyboardFocusable, is_focusable),
(HasKeyboardFocus, is_focused),
(LiveSetting, live_setting),
(AutomationId, automation_id),
(ClassName, class_name),
(Orientation, orientation)
}
patterns! {
(Toggle, is_toggle_pattern_supported, (
(ToggleState, toggle_state, ToggleState)
), (
fn Toggle(&self) -> Result<()> {
self.click()
}
)),
(Invoke, is_invoke_pattern_supported, (), (
fn Invoke(&self) -> Result<()> {
self.click()
}
)),
(Value, is_value_pattern_supported, (
(Value, value, BSTR),
(IsReadOnly, is_read_only, BOOL)
), (
fn SetValue(&self, value: &PCWSTR) -> Result<()> {
self.do_action(|| {
let value = unsafe { value.to_string() }.unwrap();
(Action::SetValue, Some(ActionData::Value(value.into())))
})
}
)),
(RangeValue, is_range_value_pattern_supported, (
(Value, numeric_value, f64),
(IsReadOnly, is_read_only, BOOL),
(Minimum, min_numeric_value, f64),
(Maximum, max_numeric_value, f64),
(SmallChange, numeric_value_step, f64),
(LargeChange, numeric_value_jump, f64)
), (
fn SetValue(&self, value: f64) -> Result<()> {
self.do_action(|| {
(Action::SetValue, Some(ActionData::NumericValue(value)))
})
}
)),
(SelectionItem, is_selection_item_pattern_supported, (
(IsSelected, is_selected, BOOL)
), (
fn Select(&self) -> Result<()> {
self.click()
},
fn AddToSelection(&self) -> Result<()> {
Err(not_implemented())
},
fn RemoveFromSelection(&self) -> Result<()> {
Err(not_implemented())
},
fn SelectionContainer(&self) -> Result<IRawElementProviderSimple> {
Err(E_FAIL.into())
}
)),
(Text, is_text_pattern_supported, (), (
fn GetSelection(&self) -> Result<*mut SAFEARRAY> {
self.resolve_for_text_pattern(|node| {
if let Some(range) = node.text_selection() {
let platform_range: ITextRangeProvider = PlatformTextRange::new(&self.context, range).into();
let iunknown: IUnknown = platform_range.cast()?;
Ok(safe_array_from_com_slice(&[iunknown]))
} else {
Ok(std::ptr::null_mut())
}
})
},
fn GetVisibleRanges(&self) -> Result<*mut SAFEARRAY> {
Ok(std::ptr::null_mut())
},
fn RangeFromChild(&self, _child: Option<&IRawElementProviderSimple>) -> Result<ITextRangeProvider> {
Err(not_implemented())
},
fn RangeFromPoint(&self, point: &UiaPoint) -> Result<ITextRangeProvider> {
self.resolve_with_context_for_text_pattern(|node, context| {
let client_top_left = context.client_top_left();
let point = Point::new(point.x - client_top_left.x, point.y - client_top_left.y);
let point = node.transform().inverse() * point;
let pos = node.text_position_at_point(point);
let range = pos.to_degenerate_range();
Ok(PlatformTextRange::new(&self.context, range).into())
})
},
fn DocumentRange(&self) -> Result<ITextRangeProvider> {
self.resolve_for_text_pattern(|node| {
let range = node.document_range();
Ok(PlatformTextRange::new(&self.context, range).into())
})
},
fn SupportedTextSelection(&self) -> Result<SupportedTextSelection> {
self.resolve_for_text_pattern(|node| {
if node.has_text_selection() {
Ok(SupportedTextSelection_Single)
} else {
Ok(SupportedTextSelection_None)
}
})
}
))
}
#[test]
fn platform_node_impl_send_sync() {
fn assert_send_sync<T: Send + Sync>() {}
assert_send_sync::<PlatformNode>();
}