[TOC]
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.
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, theobject:state-changed:checked
event tells Orca that thechecked
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.
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:
- The new ancestors of the object of interest, if any
- 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.)
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.
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.
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.
AT-SPI2/ATK v2.46 added an announcement
signal which can be used with Orca v45.2 and later.
Simple examples are provided below.
#!/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)
#!/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.
#!/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()
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.
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.
#!/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)
#!/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)
#!/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()
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()