Skip to content

Latest commit

 

History

History
519 lines (396 loc) · 19.5 KB

README-APPLICATION-DEVELOPERS.md

File metadata and controls

519 lines (396 loc) · 19.5 KB

Tips for Application Developers

[TOC]

Introduction

Please note: This document is a work in progress and will be expanded over time to include details about what Orca expects from applications. In the meantime, it contains tips based on frequently-asked questions. We hope you find them useful.

Accessible Events

Orca has many commands and modes which make it possible for users to interact with your application. However, the most basic need for applications is presentation of changes in location, such as focus changes and navigation within text. In order for Orca to present such changes, it must be notified via one or more accessible events. Unless you have created a custom widget, you should expect that accessible events will be fired by your application's toolkit.

Examples of accessible events include:

  • window:activate tells Orca that the user is in a different window. When the active window changes, Orca will announce the new window. Orca also keeps track of the active window in order to filter many accessibility events which get fired by apps other than the one currently being used.
  • object:state-changed:focused tells Orca that the user has moved to a different widget. Orca presents newly-focused widgets. Orca also keeps track of the focused widget in order to filter out some accessibility events. As an example, the object:state-changed:checked event tells Orca that the checked state of a checkbox-like widget has changed. That change is likely not of interest to the user if that widget is not focused. Therefore Orca will present that change only if the widget emitting it is focused.
  • object:text-caret-moved tells Orca that the user has moved to a different location in a text widget or document. Assuming the event is fired from the active object (focused text field or current document), Orca will present the new character, word, or line in response, using some heuristics to determine what unit of text should be presented.

If you are interested in seeing what accessible events your application is firing, you can use Accerciser's event monitor.

What Orca Presents In Response To Location Changes

When an accessible event informs Orca that the object of interest, or the user's location therein, has changed, Orca will generally present two things:

  1. The new ancestors of the object of interest, if any
  2. The object of interest, or the new location, and any role-specific relevant details

For instance if the user tabs into a group of checkboxes labelled "Permissions" and lands on the unchecked "Allow access to microphone" checkbox, Orca will say:

  • "Permissions panel"
  • "Allow access to microphone checkbox. Not checked."

If the user then tabs to another widget in the "Permissions" group, Orca will only present that widget and will not repeat "Permissions."

What Orca presents for any given object depends on the role of that object. However, for UI components, Orca will always present the accessible name and, by default, the accessible description. (The latter can be disabled by the user either globally or on a per-app basis, though it is always available on demand via Orca's "Where Am I?" command.)

Speaking Your Application's Non-Focusable Static Text

Technique: Use An Accessible Description

If your application has static, on-screen text of an explanatory nature and you do not want to make that text focusable, it is still possible to have Orca present it automatically using the accessible description property or the described-by relationship.

Examples:

  • If the focused object is a search input, and the text to be presented is "n matches found", set "n matches found" as the accessible description of the search input and update that description each time the count changes. In response, the toolkit should fire an object:property-change:accessible-description event which Orca will handle by presenting the new description.

  • If there is an on-screen message associated with a group of widgets, set the accessible description of that group to the on-screen message. As stated in the previous section, Orca will present the name and description of new ancestors prior to presenting the focused object.

Note: Orca v47.alpha or later is required to have Orca present the accessible description, and any changes to that description, on ancestors of the object of interest. In Orca v46 and earlier, Orca only presents the name of ancestors and description changes on the object of interest.

Can I Use Details/Details-For And Error-Message/Errors-For Relations?

The details/details-for and error-message/error-for relation types were created for ARIA, and there was no indication that they might be of interest to developers of native applications. As a result, support in Orca for these new relation types was implemented only for web apps. There are plans to support these relation types globally, hopefully during the v47 release cycle.

Why Is Orca Speaking My Labels As Static Text?

Using the accessible description property to get screen readers to automatically announce text in a newly-shown dialog originated as a web-browser practice, e.g. for alerts. Historically, application developers simply used toolkit labels, e.g. GtkLabel, to add static text. And those developers, and Orca users, expected Orca to read that text automatically.

In order to distinguish static text labels from widget labels, Orca checks the accessible relations of the label. If it finds any relation, Orca assumes the label is NOT static text. Otherwise Orca applies some additional heuristics to filter out false positives. But in the end, Orca may conclude incorrectly that the unrelated label is indeed static text which should be read automatically.

While less than ideal, keeping this functionality in place is important, because there are many applications that do not use the new description approach and likely will not be updated to do so. Breaking the user experience in all of those apps would be bad. However, it's easy to ensure Orca doesn't mistakenly treat your labels as static text to be read automatically:

  • Orca v47.alpha and later prefers the description as the source of static text. If your app uses that technique, Orca will not search for unrelated labels to present.
  • Any label that is not static text should have the accessible label-for relationship pointing to the widget it labels. That will prevent all versions of Orca from concluding it is static text that should be automatically presented.

Speaking Your Application's Custom Message/Announcement

AT-SPI2/ATK v2.46 added an announcement signal which can be used with Orca v45.2 and later. Simple examples are provided below.

Announcement Example: GTK 3

#!/usr/bin/python

import gi
gi.require_version("Gtk", "3.0")
from gi.repository import Gtk

def on_button_clicked(button):
    button.get_accessible().emit("announcement", "Hello world. I am an announcement.")

def on_activate(application):
    window = Gtk.ApplicationWindow(application=application)
    button = Gtk.Button(label="Make an announcement")
    button.connect("clicked", on_button_clicked)
    window.add(button)
    window.show_all()

app = Gtk.Application()
app.connect("activate", on_activate)
app.run(None)

If you are running Orca v45.2 or later, launch the sample application above and press the "Make an announcement" button. You should hear Orca say "Hello world. I am an announcement."

Beginning with ATK v2.50, the announcement signal was deprecated in favor of a new notification signal to provide native applications similar functionality to ARIA's live regions which allow web applications to specify that a notification is urgent/"assertive."

Here is an example of using the notification signal:

#!/usr/bin/python

import gi
gi.require_version("Atk", "1.0")
gi.require_version("Gtk", "3.0")

from gi.repository import Atk, Gtk

def on_button_clicked(button):
    button.get_accessible().emit("notification", "Hello world. I am a notification.", Atk.Live.POLITE)

def on_activate(application):
    window = Gtk.ApplicationWindow(application=application)
    button = Gtk.Button(label="Make a notification")
    button.connect("clicked", on_button_clicked)
    window.add(button)
    window.show_all()

app = Gtk.Application()
app.connect("activate", on_activate)
app.run(None)

Announcement Example: GTK 4 (Minimum Version: 4.14)

#!/usr/bin/python

import gi
gi.require_version("Gtk", "4.0")

from gi.repository import Gtk

def on_button_clicked(button):
    button.announce("Hello world. I am a notification.", Gtk.AccessibleAnnouncementPriority.MEDIUM)

def on_activate(application):
    window = Gtk.ApplicationWindow(application=application)
    button = Gtk.Button(label="Make a notification")
    button.connect("clicked", on_button_clicked)
    window.set_child(button)
    window.present()

app = Gtk.Application()
app.connect("activate", on_activate)
app.run(None)

Note that in older GTK 4 releases there is no way how to do this, as you can't emit raw AT-SPI2 events, or do similar platform-specific things.

Announcement Example: Qt 6 (Minimum Version: 6.8)

#!/usr/bin/python

import sys
from PySide6.QtCore import Slot
from PySide6.QtGui import QAccessible, QAccessibleAnnouncementEvent
from PySide6.QtWidgets import QApplication, QMainWindow, QPushButton

@Slot()
def on_button_clicked(checked):
    announcement_event = QAccessibleAnnouncementEvent(button, "Hello world. I am a notification.")
    # prio could be set like this (Polite is the default anyway)
    announcement_event.setPoliteness(QAccessible.AnnouncementPoliteness.Polite)
    QAccessible.updateAccessibility(announcement_event)

app = QApplication(sys.argv)
main_window = QMainWindow()
button = QPushButton("Make a notification", main_window)
button.resize(200, 50)
button.clicked.connect(on_button_clicked)

main_window.show()
app.exec()

Announcement Example: Headless Application using Dasbus for DBus Communication

In an application without a GUI, you can pretend to be enough of an accessible object to fire this event as well. To hear the result, Orca must be running first, though.

#!/usr/bin/python3
from dasbus.connection import SessionMessageBus, AddressedMessageBus
from dasbus.loop import EventLoop
from dasbus.server.interface import dbus_class, dbus_interface, dbus_signal
from dasbus.typing import Int32, UInt32, Variant, Structure, Dict, List, Tuple, ObjPath, get_variant
import threading
import time

ANNOUNCER_PATH = "/com/example/Announcer"

@dbus_interface("org.a11y.atspi.Application")
class Application:
    """A minimal accessible application to fulfill AT-SPI2's expectations that events come from a
    valid accessible application."""

    @property
    def ToolkitName(self) -> str:
        return "Announcer"

@dbus_interface("org.a11y.atspi.Event.Object")
class ObjectEvents:
    """Just a holder for the one needed signal."""

    @dbus_signal
    def Announcement(self, subtype: str, detail1: int, detail2: int, value: Variant, props: Structure):
        pass

@dbus_interface("org.a11y.atspi.Accessible")
class Accessible:
    """A minimal accessible object to fulfill AT-SPI2's expectations that events come from a valid
    accessible object."""

    def GetState(self) -> list[UInt32]:
        return [1<<25, 0] # ATSPI_STATE_SHOWING

    @property
    def Name(self) -> str:
        return "Announcer"

    @property
    def Parent(self) -> Tuple[str, ObjPath]:
        return "", ObjPath("/org/a11y/atspi/null")

    def GetRole(self) -> UInt32:
        return 75 # ATSPI_ROLE_APPLICATION

    def GetAttributes(self) -> Dict[str, str]:
        return {}

@dbus_class
class Announcer(Accessible, Application, ObjectEvents):
    pass

CacheEntry = Tuple[Tuple[str, ObjPath], Tuple[str, ObjPath], Tuple[str, ObjPath], Int32, Int32, List[str], str, UInt32, str, List[UInt32]]

@dbus_interface("org.a11y.atspi.Cache")
class Cache:
    """A minimal accessible objects cache to fulfill AT-SPI2's expectations that events come from an
    accessible application which has a valid accessible objects cache."""

    def GetItems(self) -> List[CacheEntry]:
        return []

session_bus = SessionMessageBus()
a11y_bus_info_provider = session_bus.get_proxy("org.a11y.Bus", "/org/a11y/bus")
address = a11y_bus_info_provider.GetAddress()
a11y_bus = AddressedMessageBus(address)
announcer = Announcer()
a11y_bus.publish_object(ANNOUNCER_PATH, announcer)
a11y_bus.publish_object("/org/a11y/atspi/cache", Cache())
loop = EventLoop()
threading.Thread(target=loop.run, daemon=True).start()
print("About to announce Hello, world!")
announcer.Announcement("", 1, 0, get_variant(str, "Hello, world!"), [])
# Give the announcement time to be processed
time.sleep(0.5)
print("Done announcing Hello, world!")

Please note: Because "assertive" messages can be disruptive if presented at the wrong time, Orca currently treats an "assertive" notification from non-web applications the same as a regular/"polite" notification. Adding support for "assertive" notifications from non-web applications is planned and depends on Orca's live-region support being made global so that users have full control over when and how notifications are presented to them.

Providing Context-Sensitive Help Messages

AT-SPI2/ATK v2.52 added support for setting and retrieving "help text" on accessible objects. Help text makes it possible to provide context-sensitive information that might not be immediately obvious to the user. For instance in a slide presentation editor, when the user tabs to a placeholder on a slide, an appropriate message might be "You are on a placeholder. Use the arrow keys to reposition it on the slide. Press Return to edit its contents." (As the user moves the placeholder on the slide, the Announcement feature described above could be used to inform the user of the new location.)

Please note: Help text should not be used to announce mnemonics. Mnemonics are expected to be exposed to Orca via the accessible Action interface via the toolkit. Orca has a setting, disabled by default, to present mnemonics to the user.

Help text is supported in Orca v46 and later. Prior to Orca v47.alpha, this feature was disabled by default. Starting with Orca v47.alpha, help text is presented by default when focus changes, but that presentation can be disabled by the user either globally or on per-app basis. However, even when disabled for focus changes, users can always obtain the help text on demand by using Orca's "Where Am I?" command.

Help Message Example: GTK 3

#!/usr/bin/python

import gi
gi.require_version("Gtk", "3.0")

from gi.repository import Gtk

def on_activate(application):
    window = Gtk.ApplicationWindow(application=application)
    box = Gtk.HBox()
    window.add(box)
    label = Gtk.Label(label="Type something here:")
    box.add(label)
    entry = Gtk.Entry()
    box.add(entry)

    # Setting the mnemonic widget will cause the accessible labelled-by relation to be
    # set. Orca uses that to say "Type something here:" when the entry gains focus.
    label.set_mnemonic_widget(entry)

    # This text is presented by Orca as a "tutorial message."
    entry.get_accessible().set_help_text("Enter 10 characters.")
    window.show_all()

app = Gtk.Application()
app.connect("activate", on_activate)
app.run(None)

Help Message Example: GTK 4 (Minimum Version: 4.16)

#!/usr/bin/python

import gi
gi.require_version("Gtk", "4.0")

from gi.repository import Gtk

def on_activate(application):
    window = Gtk.ApplicationWindow(application=application)
    box = Gtk.Box.new(Gtk.Orientation.HORIZONTAL, 6)
    window.set_child(box)
    label = Gtk.Label(label="Type something here:")
    box.append(label)
    entry = Gtk.Entry()
    box.append(entry)

    # Setting the mnemonic widget will cause the accessible labelled-by relation to be
    # set. Orca uses that to say "Type something here:" when the entry gains focus.
    label.set_mnemonic_widget(entry)

    # This text is presented by Orca as a "tutorial message."
    entry.update_property([Gtk.AccessibleProperty.HELP_TEXT], ["Enter 10 characters."])
    window.present()

app = Gtk.Application()
app.connect("activate", on_activate)
app.run(None)

Help Message Example: Qt 6 (Minimum Version: 6.8)

#!/usr/bin/python
import sys
from PySide6.QtWidgets import QMainWindow, QLabel, QLineEdit, QHBoxLayout, QApplication, QWidget

app = QApplication(sys.argv)
window = QMainWindow()
central_widget = QWidget()
box = QHBoxLayout(central_widget)
window.setCentralWidget(central_widget)
label = QLabel("Type something here:")
box.addWidget(label)
entry = QLineEdit()
box.addWidget(entry)

# Setting the label's buddy will cause the accessible label-for relation to be
# set. Orca uses that to say "Type something here:" when the entry gains focus.
label.setBuddy(entry)

# This text is presented by Orca as a "tutorial message."
entry.setWhatsThis("Enter 10 characters.")
window.show()

app.exec()

Stand-Alone Tools For Debugging

Dump The Focused Object And Its Ancestors

This tool listens for object:state-changed:focused events. When an object emits this event, the tool will print the event along with the accessibility tree from the application down to the object which just claimed focus. Note that the information dumped is limited to the basics: role, name/label, description, and help text.

#!/usr/bin/python

import gi
gi.require_version("Atspi", "2.0")
from gi.repository import Atspi

def as_string(obj):
    try:
        help_text = Atspi.Accessible.get_help_text(obj)
    except Exception as error:
        help_text = f"({error})"
    return (f"{Atspi.Accessible.get_role(obj).value_name} "
            f"name:'{get_name(obj)}' "
            f"description:'{get_description(obj)}' "
            f"help text: '{help_text}'")

def get_name(obj):
    name = Atspi.Accessible.get_name(obj)
    if name:
        return name

    relations = Atspi.Accessible.get_relation_set(obj)
    for relation in relations:
        if relation.get_relation_type() != Atspi.RelationType.LABELLED_BY:
            continue
        targets = [relation.get_target(i) for i in range(relation.get_n_targets())]
        return " ".join([target.name for target in targets])

    return ""

def get_description(obj):
    description = Atspi.Accessible.get_description(obj)
    if description:
        return description

    relations = Atspi.Accessible.get_relation_set(obj)
    for relation in relations:
        if relation.get_relation_type() != Atspi.RelationType.DESCRIBED_BY:
            continue
        targets = [relation.get_target(i) for i in range(relation.get_n_targets())]
        return " ".join([target.name for target in targets])

    return ""

def on_event(e):
    if not e.detail1:
        return

    print(f"\n{as_string(e.source)} is now focused")
    ancestors = []
    parent = e.source
    while parent:
      grandparent = Atspi.Accessible.get_parent(parent)
      ancestors.append(as_string(parent))
      if Atspi.Accessible.get_role(parent) == Atspi.Role.APPLICATION:
        break
      parent = grandparent

    ancestors.reverse()
    indent = 0
    for ancestor in ancestors:
        print(f"{' ' * indent}--> {ancestor}")
        indent += 2

    if Atspi.Accessible.get_role(e.source) == Atspi.Role.TERMINAL:
      print("Exiting.")
      Atspi.event_quit()

listener = Atspi.EventListener.new(on_event)
listener.register("object:state-changed:focused")
print("Return focus to your terminal to exit")
Atspi.event_main()