%PDF- %PDF-
| Direktori : /lib/python3/dist-packages/orca/ |
| Current File : //lib/python3/dist-packages/orca/ax_event_synthesizer.py |
# Orca
#
# Copyright 2005-2008 Sun Microsystems Inc.
# Copyright 2018-2023 Igalia, S.L.
#
# This library is free software; you can redistribute it and/or
# modify it under the terms of the GNU Lesser General Public
# License as published by the Free Software Foundation; either
# version 2.1 of the License, or (at your option) any later version.
#
# This library is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public
# License along with this library; if not, write to the
# Free Software Foundation, Inc., Franklin Street, Fifth Floor,
# Boston MA 02110-1301 USA.
"""Provides support for synthesizing accessible input events."""
__id__ = "$Id$"
__version__ = "$Revision$"
__date__ = "$Date$"
__copyright__ = "Copyright (c) 2005-2008 Sun Microsystems Inc." \
"Copyright (c) 2018-2023 Igalia, S.L."
__license__ = "LGPL"
import time
import gi
gi.require_version("Atspi", "2.0")
from gi.repository import Atspi
gi.require_version("Gtk", "3.0")
from gi.repository import Gtk
from . import debug
from . import focus_manager
from .ax_component import AXComponent
from .ax_object import AXObject
from .ax_text import AXText
from .ax_utilities import AXUtilities
class AXEventSynthesizer:
"""Provides support for synthesizing accessible input events."""
_banner = None
@staticmethod
def _window_coordinates_to_screen_coordinates(x, y):
# TODO - JD: This is a workaround to keep things working until we have something like
# https://gitlab.gnome.org/GNOME/at-spi2-core/-/issues/158
active_window = focus_manager.getManager().get_active_window()
if active_window is None:
msg = "AXEventSynthesizer: Could not get active window to adjust coordinates"
debug.printMessage(debug.LEVEL_INFO, msg, True)
return x, y
try:
point = Atspi.Component.get_position(active_window, Atspi.CoordType.SCREEN)
except Exception as error:
msg = f"AXEventSynthesizer: Exception in calling get_position: {error}"
debug.printMessage(debug.LEVEL_INFO, msg, True)
return x, y
msg = f"AXEventSynthesizer: Active window position: {point.x}, {point.y}"
debug.printMessage(debug.LEVEL_INFO, msg, True)
# Unfortunately, the position we get does not seem to include window decorations.
# So we have to do more work to adjust. This is why we cannot have nice things.
gdk_window = Gtk.Window().get_screen().get_active_window()
frame_extents = gdk_window.get_frame_extents()
title_bar_height = frame_extents.height - gdk_window.get_height()
msg = f"AXEventSynthesizer: Title bar height believed to be: {title_bar_height}px"
debug.printMessage(debug.LEVEL_INFO, msg, True)
new_x = x + point.x
new_y = y + point.y + title_bar_height
msg = f"AXEventSynthesizer: x: {x}->{new_x}, y: {y}->{new_y}"
debug.printMessage(debug.LEVEL_INFO, msg, True)
return new_x, new_y
@staticmethod
def _get_mouse_coordinates():
"""Returns the current mouse coordinates."""
root_window = Gtk.Window().get_screen().get_root_window()
window, x_coord, y_coord, modifiers = root_window.get_pointer()
tokens = ["AXEventSynthesizer: Mouse coordinates:", x_coord, ",", y_coord]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return x_coord, y_coord
@staticmethod
def _generate_mouse_event(x_coord, y_coord, event):
"""Synthesize a mouse event at a specific screen coordinate."""
old_x, old_y = AXEventSynthesizer._get_mouse_coordinates()
tokens = ["AXEventSynthesizer: Generating", event, "mouse event at", x_coord, ",", y_coord]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
screen_x, screen_y = AXEventSynthesizer._window_coordinates_to_screen_coordinates(
x_coord, y_coord)
try:
success = Atspi.generate_mouse_event(screen_x, screen_y, event)
except Exception as error:
tokens = ["AXEventSynthesizer: Exception in _generate_mouse_event:", error]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
success = False
else:
tokens = ["AXEventSynthesizer: Atspi.generate_mouse_event returned", success]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
# There seems to be a timeout / lack of reply from this blocking call.
# But often the mouse event is successful. Pause briefly before checking.
time.sleep(1)
new_x, new_y = AXEventSynthesizer._get_mouse_coordinates()
if old_x == new_x and old_y == new_y and (old_x, old_y) != (screen_x, screen_y):
msg = "AXEventSynthesizer: Mouse event possible failure. Pointer didn't move"
debug.println(debug.LEVEL_INFO, msg, True)
return False
return True
@staticmethod
def _mouse_event_on_character(obj, event):
"""Performs the specified mouse event on the current character in obj."""
extents = AXText.get_character_rect(obj)
if AXComponent.is_empty_rect(extents):
return False
rect = AXComponent.get_rect(obj)
intersection = AXComponent.get_rect_intersection(extents, rect)
if AXComponent.is_empty_rect(intersection):
tokens = ["AXEventSynthesizer:", obj, "'s caret", extents, "not in obj", rect]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return False
x_coord = max(extents.x, extents.y + (extents.width / 2) - 1)
y_coord = extents.y + extents.height / 2
return AXEventSynthesizer._generate_mouse_event(x_coord, y_coord, event)
@staticmethod
def _mouse_event_on_object(obj, event):
"""Performs the specified mouse event on obj."""
x_coord, y_coord = AXComponent.get_center_point(obj)
return AXEventSynthesizer._generate_mouse_event(x_coord, y_coord, event)
@staticmethod
def route_to_character(obj):
"""Routes the pointer to the current character in obj."""
tokens = ["AXEventSynthesizer: Attempting to route to character in", obj]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return AXEventSynthesizer._mouse_event_on_character(obj, "abs")
@staticmethod
def route_to_object(obj):
"""Moves the mouse pointer to the center of obj."""
tokens = ["AXEventSynthesizer: Attempting to route to", obj]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return AXEventSynthesizer._mouse_event_on_object(obj, "abs")
@staticmethod
def route_to_point(x_coord, y_coord):
"""Routes the pointer to the specified coordinates."""
return AXEventSynthesizer._generate_mouse_event(x_coord, y_coord, "abs")
@staticmethod
def click_character(obj, button=1):
"""Single click on the current character in obj using the specified button."""
return AXEventSynthesizer._mouse_event_on_character(obj, f"b{button}c")
@staticmethod
def click_object(obj, button=1):
"""Single click on obj using the specified button."""
return AXEventSynthesizer._mouse_event_on_object(obj, f"b{button}c")
@staticmethod
def click_point(x_coord, y_coord, button=1):
"""Single click on the given point using the specified button."""
return AXEventSynthesizer._generate_mouse_event(x_coord, y_coord, f"b{button}c")
@staticmethod
def double_click_character(obj, button=1):
"""Double click on the current character in obj using the specified button."""
return AXEventSynthesizer._mouse_event_on_character(obj, f"b{button}d")
@staticmethod
def double_click_object(obj, button=1):
"""Double click on obj using the specified button."""
return AXEventSynthesizer._mouse_event_on_object(obj, f"b{button}d")
@staticmethod
def double_click_point(x_coord, y_coord, button=1):
"""Double click on the given point using the specified button."""
return AXEventSynthesizer._generate_mouse_event(x_coord, y_coord, f"b{button}d")
@staticmethod
def press_at_character(obj, button=1):
"""Performs a press on the current character in obj using the specified button."""
return AXEventSynthesizer._mouse_event_on_character(obj, f"b{button}p")
@staticmethod
def press_at_object(obj, button=1):
"""Performs a press on obj using the specified button."""
return AXEventSynthesizer._mouse_event_on_object(obj, f"b{button}p")
@staticmethod
def press_at_point(x_coord, y_coord, button=1):
"""Performs a press on the given point using the specified button."""
return AXEventSynthesizer._generate_mouse_event(x_coord, y_coord, f"b{button}p")
@staticmethod
def release_at_character(obj, button=1):
"""Performs a release on the current character in obj using the specified button."""
return AXEventSynthesizer._mouse_event_on_character(obj, f"b{button}r")
@staticmethod
def release_at_object(obj, button=1):
"""Performs a release on obj using the specified button."""
return AXEventSynthesizer._mouse_event_on_object(obj, f"b{button}r")
@staticmethod
def release_at_point(x_coord, y_coord, button=1):
"""Performs a release on the given point using the specified button."""
return AXEventSynthesizer._generate_mouse_event(x_coord, y_coord, f"b{button}r")
@staticmethod
def _scroll_to_location(obj, location, start_offset=None, end_offset=None):
"""Attempts to scroll to the specified location."""
before = AXComponent.get_position(obj)
if not AXText.scroll_substring_to_location(obj, location, start_offset, end_offset):
AXComponent.scroll_object_to_location(obj, location)
after = AXComponent.get_position(obj)
tokens = ["AXEventSynthesizer: Before scroll:", before, "After scroll:", after]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
@staticmethod
def _scroll_to_point(obj, x_coord, y_coord, start_offset=None, end_offset=None):
"""Attempts to scroll obj to the specified point."""
before = AXComponent.get_position(obj)
if not AXText.scroll_substring_to_point(obj, x_coord, y_coord, start_offset, end_offset):
AXComponent.scroll_object_to_point(obj, x_coord, y_coord)
after = AXComponent.get_position(obj)
tokens = ["AXEventSynthesizer: Before scroll:", before, "After scroll:", after]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
@staticmethod
def scroll_into_view(obj, start_offset=None, end_offset=None):
"""Attempts to scroll obj into view."""
AXEventSynthesizer._scroll_to_location(
obj, Atspi.ScrollType.ANYWHERE, start_offset, end_offset)
@staticmethod
def _containing_document(obj):
"""Returns the document containing obj"""
document = AXObject.find_ancestor(obj, AXUtilities.is_document)
while document:
ancestor = AXObject.find_ancestor(document, AXUtilities.is_document)
if ancestor is None or ancestor == document:
break
document = ancestor
return document
@staticmethod
def _get_obscuring_banner(obj):
""""Returns the banner obscuring obj from view."""
document = AXEventSynthesizer._containing_document(obj)
if not document:
tokens = ["AXEventSynthesizer: No obscuring banner found for", obj, ". No document."]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return None
if not AXObject.supports_component(document):
tokens = ["AXEventSynthesizer: No obscuring banner found for", obj, ". No doc iface."]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return None
obj_rect = AXComponent.get_rect(obj)
doc_rect = AXComponent.get_rect(document)
left = AXComponent.get_descendant_at_point(document, doc_rect.x, obj_rect.y)
right = AXComponent.get_descendant_at_point(
document, doc_rect.x + doc_rect.width, obj_rect.y)
if not (left and right and left == right != document):
tokens = ["AXEventSynthesizer: No obscuring banner found for", obj]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return None
tokens = ["AXEventSynthesizer:", obj, "believed to be obscured by banner", left]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return left
@staticmethod
def _scroll_below_banner(obj, banner, start_offset, end_offset, margin=25):
"""Attempts to scroll obj below banner."""
obj_rect = AXComponent.get_rect(obj)
banner_rect = AXComponent.get_rect(banner)
tokens = ["AXEventSynthesizer: Extents of banner: ", banner_rect]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
AXEventSynthesizer._scroll_to_point(
obj, obj_rect.x, banner_rect.y + banner_rect.height + margin, start_offset, end_offset)
@staticmethod
def scroll_to_top_edge(obj, start_offset=None, end_offset=None):
"""Attempts to scroll obj to the top edge."""
if AXEventSynthesizer._banner and not AXObject.is_dead(AXEventSynthesizer._banner):
msg = (
f"AXEventSynthesizer: Suspected existing banner found: "
f"{AXEventSynthesizer._banner}"
)
debug.println(debug.LEVEL_INFO, msg, True)
AXEventSynthesizer._scroll_below_banner(
obj, AXEventSynthesizer._banner, start_offset, end_offset)
return
AXEventSynthesizer._scroll_to_location(
obj, Atspi.ScrollType.TOP_EDGE, start_offset, end_offset)
AXEventSynthesizer._banner = AXEventSynthesizer._get_obscuring_banner(obj)
if AXEventSynthesizer._banner:
msg = f"AXEventSynthesizer: Re-scrolling {obj} due to banner"
AXEventSynthesizer._scroll_below_banner(
obj, AXEventSynthesizer._banner, start_offset, end_offset)
debug.println(debug.LEVEL_INFO, msg, True)
@staticmethod
def scroll_to_top_left(obj, start_offset=None, end_offset=None):
"""Attempts to scroll obj to the top left."""
AXEventSynthesizer._scroll_to_location(
obj, Atspi.ScrollType.TOP_LEFT, start_offset, end_offset)
@staticmethod
def scroll_to_left_edge(obj, start_offset=None, end_offset=None):
"""Attempts to scroll obj to the left edge."""
AXEventSynthesizer._scroll_to_location(
obj, Atspi.ScrollType.LEFT_EDGE, start_offset, end_offset)
@staticmethod
def scroll_to_bottom_edge(obj, start_offset=None, end_offset=None):
"""Attempts to scroll obj to the bottom edge."""
AXEventSynthesizer._scroll_to_location(
obj, Atspi.ScrollType.BOTTOM_EDGE, start_offset, end_offset)
@staticmethod
def scroll_to_bottom_right(obj, start_offset=None, end_offset=None):
"""Attempts to scroll obj to the bottom right."""
AXEventSynthesizer._scroll_to_location(
obj, Atspi.ScrollType.BOTTOM_RIGHT, start_offset, end_offset)
@staticmethod
def scroll_to_right_edge(obj, start_offset=None, end_offset=None):
"""Attempts to scroll obj to the right edge."""
AXEventSynthesizer._scroll_to_location(
obj, Atspi.ScrollType.RIGHT_EDGE, start_offset, end_offset)
@staticmethod
def try_all_clickable_actions(obj):
"""Attempts to perform a click-like action if one is available."""
actions = ["click", "press", "jump", "open"]
for action in actions:
if AXObject.do_named_action(obj, action):
tokens = ["AXEventSynthesizer: '", action, "' on", obj, "performed successfully"]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return True
if debug.LEVEL_INFO < debug.debugLevel:
return False
tokens = ["AXEventSynthesizer: Actions on", obj, ":", AXObject.actions_as_string(obj)]
debug.printTokens(debug.LEVEL_INFO, tokens, True)
return False
_synthesizer = AXEventSynthesizer()
def getSynthesizer():
return _synthesizer