9#include <QtCore/private/qwasmsuspendresumecontrol_p.h>
10#include <QtGui/qwindow.h>
16 QWasmAccessibility::enable();
19#if QT_CONFIG(accessibility)
21#include <QtGui/private/qaccessiblebridgeutils_p.h>
23Q_LOGGING_CATEGORY(lcQpaAccessibility,
"qt.qpa.accessibility")
26EM_JS(emscripten::EM_VAL, getActiveElement_js, (emscripten::EM_VAL undefHandle), {
27 var activeEl = document.activeElement;
31 }
else if (activeEl.shadowRoot) {
32 activeEl = activeEl.shadowRoot.activeElement;
34 return Emval.toHandle(activeEl);
50QWasmAccessibility::QWasmAccessibility()
54 if (qEnvironmentVariableIntValue(
"QT_WASM_ENABLE_ACCESSIBILITY") == 1)
55 enableAccessibility();
58 QWasmSuspendResumeControl *suspendResume = QWasmSuspendResumeControl::get();
60 m_eventHandlerIndex = suspendResume->registerEventHandler([
this](
const emscripten::val event){
61 this->handleEventFromHtmlElement(event);
65QWasmAccessibility::~QWasmAccessibility()
68 QWasmSuspendResumeControl *suspendResume = QWasmSuspendResumeControl::get();
69 suspendResume->removeEventHandler(m_eventHandlerIndex);
74QWasmAccessibility *QWasmAccessibility::s_instance =
nullptr;
76QWasmAccessibility* QWasmAccessibility::get()
81void QWasmAccessibility::addAccessibilityEnableButton(QWindow *window)
83 get()->addAccessibilityEnableButtonImpl(window);
86void QWasmAccessibility::onShowWindow(QWindow *window)
88 get()->onShowWindowImpl(window);
91void QWasmAccessibility::onRemoveWindow(QWindow *window)
93 get()->onRemoveWindowImpl(window);
96bool QWasmAccessibility::isEnabled()
98 return get()->m_accessibilityEnabled;
100void QWasmAccessibility::enable()
103 get()->enableAccessibility();
106void QWasmAccessibility::addAccessibilityEnableButtonImpl(QWindow *window)
108 if (m_accessibilityEnabled)
111 emscripten::val container = getElementContainer(window);
112 emscripten::val document = getDocument(container);
113 emscripten::val button = document.call<emscripten::val>(
"createElement", std::string(
"button"));
114 setProperty(button,
"innerText",
"Enable Screen Reader");
115 button[
"classList"].call<
void>(
"add", emscripten::val(
"hidden-visually-read-by-screen-reader"));
116 container.call<
void>(
"appendChild", button);
118 auto enableContext = std::make_tuple(button, std::make_unique<qstdweb::EventCallback>
119 (button, std::string(
"click"), [
this](emscripten::val) { enableAccessibility(); }));
120 m_enableButtons.insert(std::make_pair(window, std::move(enableContext)));
123void QWasmAccessibility::onShowWindowImpl(QWindow *window)
125 if (!m_accessibilityEnabled)
127 populateAccessibilityTree(window->accessibleRoot());
130void QWasmAccessibility::onRemoveWindowImpl(QWindow *window)
133 const auto it = m_enableButtons.find(window);
134 if (it != m_enableButtons.end()) {
136 auto [element, callback] = it->second;
138 element[
"parentElement"].call<
void>(
"removeChild", element);
139 m_enableButtons.erase(it);
143 auto a11yContainer = getA11yContainer(window);
144 auto describedByContainer =
145 getDescribedByContainer(window);
146 auto elementContainer = getElementContainer(window);
147 auto document = getDocument(a11yContainer);
150 if (!describedByContainer.isUndefined()) {
151 a11yContainer.call<
void>(
"removeChild", describedByContainer);
152 describedByContainer = document.call<emscripten::val>(
"createElement", std::string(
"div"));
154 a11yContainer.call<
void>(
"appendChild", elementContainer);
155 a11yContainer.call<
void>(
"appendChild", describedByContainer);
160void QWasmAccessibility::enableAccessibility()
165 Q_ASSERT(!m_accessibilityEnabled);
166 m_accessibilityEnabled =
true;
168 for (
const auto& [key, value] : m_enableButtons) {
169 const auto &[element, callback] = value;
171 if (
auto wasmWindow = QWasmWindow::fromWindow(key))
172 wasmWindow->onAccessibilityEnable();
173 onShowWindowImpl(key);
174 element[
"parentElement"].call<
void>(
"removeChild", element);
176 m_enableButtons.clear();
179bool QWasmAccessibility::isWindowNode(QAccessibleInterface *iface)
181 return (iface && !getWindow(iface->parent()) && getWindow(iface));
184emscripten::val QWasmAccessibility::getA11yContainer(QWindow *window)
186 const auto wasmWindow = QWasmWindow::fromWindow(window);
188 return emscripten::val::undefined();
190 auto a11yContainer = wasmWindow->a11yContainer();
191 if (a11yContainer[
"childElementCount"].as<
unsigned>() == 2)
192 return a11yContainer;
194 Q_ASSERT(a11yContainer[
"childElementCount"].as<
unsigned>() == 0);
196 const auto document = getDocument(a11yContainer);
197 if (document.isUndefined())
198 return emscripten::val::undefined();
200 auto elementContainer = document.call<emscripten::val>(
"createElement", std::string(
"div"));
201 elementContainer[
"classList"].call<
void>(
"add", emscripten::val(
"qt-window-a11y-elements-container"));
202 auto describedByContainer = document.call<emscripten::val>(
"createElement", std::string(
"div"));
203 describedByContainer[
"classList"].call<
void>(
"add", emscripten::val(
"qt-window-a11y-describedby-container"));
205 a11yContainer.call<
void>(
"appendChild", elementContainer);
206 a11yContainer.call<
void>(
"appendChild", describedByContainer);
208 return a11yContainer;
211emscripten::val QWasmAccessibility::getA11yContainer(QAccessibleInterface *iface)
213 return getA11yContainer(getWindow(iface));
216emscripten::val QWasmAccessibility::getDescribedByContainer(QWindow *window)
218 auto a11yContainer = getA11yContainer(window);
219 if (a11yContainer.isUndefined())
220 return emscripten ::val::undefined();
222 Q_ASSERT(a11yContainer[
"childElementCount"].as<
unsigned>() == 2);
223 Q_ASSERT(!a11yContainer[
"children"][1].isUndefined());
225 return a11yContainer[
"children"][1];
228emscripten::val QWasmAccessibility::getDescribedByContainer(QAccessibleInterface *iface)
230 return getDescribedByContainer(getWindow(iface));
233emscripten::val QWasmAccessibility::getElementContainer(QWindow *window)
235 auto a11yContainer = getA11yContainer(window);
236 if (a11yContainer.isUndefined())
237 return emscripten ::val::undefined();
239 Q_ASSERT(a11yContainer[
"childElementCount"].as<
unsigned>() == 2);
240 Q_ASSERT(!a11yContainer[
"children"][0].isUndefined());
241 return a11yContainer[
"children"][0];
244emscripten::val QWasmAccessibility::getElementContainer(QAccessibleInterface *iface)
251 if (!getWindow(iface))
252 return emscripten::val::undefined();
254 if (isWindowNode(iface))
255 return emscripten::val::undefined();
257 if (isWindowNode(iface->parent()))
258 return getElementContainer(getWindow(iface->parent()));
261 return getHtmlElement(iface->parent());
264QWindow *QWasmAccessibility::getWindow(QAccessibleInterface *iface)
269 QWindow *window = iface->window();
271 if (!window && iface->parent())
272 window = iface->parent()->window();
276emscripten::val QWasmAccessibility::getDocument(
const emscripten::val &container)
278 if (container.isUndefined())
279 return emscripten::val::global(
"document");
280 return container[
"ownerDocument"];
283emscripten::val QWasmAccessibility::getDocument(QAccessibleInterface *iface)
285 return getDocument(getA11yContainer(iface));
288void QWasmAccessibility::setAttribute(emscripten::val element,
const std::string &attr,
289 const std::string &val)
292 element.call<
void>(
"setAttribute", attr, val);
294 element.call<
void>(
"removeAttribute", attr);
297void QWasmAccessibility::setAttribute(emscripten::val element,
const std::string &attr,
300 setAttribute(element, attr, std::string(val));
303void QWasmAccessibility::setAttribute(emscripten::val element,
const std::string &attr,
bool val)
306 element.call<
void>(
"setAttribute", attr, val);
308 element.call<
void>(
"removeAttribute", attr);
311void QWasmAccessibility::setProperty(emscripten::val element,
const std::string &property,
312 const std::string &val)
314 element.set(property, val);
317void QWasmAccessibility::setProperty(emscripten::val element,
const std::string &property,
const char *val)
319 setProperty(element, property, std::string(val));
322void QWasmAccessibility::setProperty(emscripten::val element,
const std::string &property,
bool val)
324 element.set(property, val);
327void QWasmAccessibility::setNamedAttribute(QAccessibleInterface *iface,
const std::string &attribute, QAccessible::Text text)
329 const emscripten::val element = getHtmlElement(iface);
330 setAttribute(element, attribute, iface->text(text).toStdString());
332void QWasmAccessibility::setNamedProperty(QAccessibleInterface *iface,
const std::string &property, QAccessible::Text text)
334 const emscripten::val element = getHtmlElement(iface);
335 setProperty(element, property, iface->text(text).toStdString());
338void QWasmAccessibility::addEventListener(QAccessibleInterface *iface, emscripten::val element,
const char *eventType)
340 element.set(
"data-qta11yinterface",
reinterpret_cast<size_t>(iface));
341 element.call<
void>(
"addEventListener", emscripten::val(eventType),
342 QWasmSuspendResumeControl::get()->jsEventHandlerAt(m_eventHandlerIndex),
346void QWasmAccessibility::sendEvent(QAccessibleInterface *iface, QAccessible::Event eventType)
348 if (iface->object()) {
349 QAccessibleEvent event(iface->object(), eventType);
350 handleUpdateByInterfaceRole(&event);
352 QAccessibleEvent event(iface, eventType);
353 handleUpdateByInterfaceRole(&event);
357emscripten::val QWasmAccessibility::createHtmlElement(QAccessibleInterface *iface)
362 emscripten::val container = getElementContainer(iface);
366 emscripten::val document = getDocument(container);
371 emscripten::val element = [
this, iface, document] {
373 emscripten::val element = emscripten::val::undefined();
375 switch (iface->role()) {
377 case QAccessible::Button: {
378 element = document.call<emscripten::val>(
"createElement", std::string(
"button"));
379 addEventListener(iface, element,
"click");
382 case QAccessible::Grouping:
383 case QAccessible::CheckBox: {
389 emscripten::val checkbox = emscripten::val::undefined();
390 if (iface->role() == QAccessible::CheckBox) {
391 checkbox = document.call<emscripten::val>(
"createElement", std::string(
"input"));
392 setAttribute(checkbox,
"type",
"checkbox");
393 setAttribute(checkbox,
"checked", iface->state().checked);
394 setProperty(checkbox,
"indeterminate", iface->state().checkStateMixed);
395 addEventListener(iface, checkbox,
"change");
398 if (iface->childCount() > 0 || iface->role() == QAccessible::Grouping) {
399 auto label = document.call<emscripten::val>(
"createElement", std::string(
"span"));
401 const std::string id = QString::asprintf(
"lbid%p", iface).toStdString();
402 setAttribute(label,
"id", id);
404 element = document.call<emscripten::val>(
"createElement", std::string(
"div"));
405 element.call<
void>(
"appendChild", label);
407 setAttribute(element,
"role",
"group");
408 setAttribute(element,
"aria-labelledby", id);
410 if (!checkbox.isUndefined()) {
411 element.call<
void>(
"appendChild", checkbox);
412 addEventListener(iface, checkbox,
"focus");
419 case QAccessible::Switch: {
420 element = document.call<emscripten::val>(
"createElement", std::string(
"button"));
421 setAttribute(element,
"type",
"button");
422 setAttribute(element,
"role",
"switch");
423 if (iface->state().checked)
424 setAttribute(element,
"aria-checked",
"true");
426 setAttribute(element,
"aria-checked",
"false");
427 addEventListener(iface, element,
"change");
430 case QAccessible::RadioButton: {
431 element = document.call<emscripten::val>(
"createElement", std::string(
"input"));
432 setAttribute(element,
"type",
"radio");
433 setAttribute(element,
"checked", iface->state().checked);
434 setProperty(element,
"name",
"buttonGroup");
435 addEventListener(iface, element,
"change");
438 case QAccessible::Dial:
439 case QAccessible::SpinBox:
440 case QAccessible::Slider: {
441 const auto minValue = iface->valueInterface()->minimumValue().toString().toStdString();
442 const auto maxValue = iface->valueInterface()->maximumValue().toString().toStdString();
443 const auto stepValue =
444 iface->valueInterface()->minimumStepSize().toString().toStdString();
445 const auto value = iface->valueInterface()->currentValue().toString().toStdString();
446 element = document.call<emscripten::val>(
"createElement", std::string(
"input"));
447 setAttribute(element,
"type",
"number");
448 setAttribute(element,
"min", minValue);
449 setAttribute(element,
"max", maxValue);
450 setAttribute(element,
"step", stepValue);
451 setProperty(element,
"value", value);
452 addEventListener(iface, element,
"change");
455 case QAccessible::PageTabList:{
456 element = document.call<emscripten::val>(
"createElement", std::string(
"div"));
457 setAttribute(element,
"role",
"tablist");
458 setHtmlElementOrientation(element, iface);
460 m_elements[iface] = element;
462 for (
int i = 0; i < iface->childCount(); ++i)
463 createHtmlElement(iface->child(i));
467 case QAccessible::PageTab:{
468 const QString text = iface->text(QAccessible::Name);
469 element = document.call<emscripten::val>(
"createElement", std::string(
"button"));
470 setAttribute(element,
"role",
"tab");
471 setAttribute(element,
"title", text.toStdString());
472 addEventListener(iface, element,
"click");
475 case QAccessible::ScrollBar: {
478 const auto minValue = iface->valueInterface()->minimumValue().toString().toStdString();
479 const auto maxValue = iface->valueInterface()->maximumValue().toString().toStdString();
480 const auto value = iface->valueInterface()->currentValue().toString().toStdString();
481 element = document.call<emscripten::val>(
"createElement", std::string(
"div"));
482 setAttribute(element,
"tabindex",
"0");
483 setAttribute(element,
"role",
"scrollbar");
484 setAttribute(element,
"aria-valuemin", minValue);
485 setAttribute(element,
"aria-valuemax", maxValue);
486 setAttribute(element,
"aria-valuenow", value);
487 setHtmlElementOrientation(element, iface);
488 addEventListener(iface, element,
"change");
491 case QAccessible::StaticText: {
492 element = document.call<emscripten::val>(
"createElement", std::string(
"div"));
494 case QAccessible::Dialog: {
495 element = document.call<emscripten::val>(
"createElement", std::string(
"dialog"));
497 case QAccessible::ToolBar:{
498 const QString text = iface->text(QAccessible::Name);
499 element = document.call<emscripten::val>(
"createElement", std::string(
"div"));
500 setAttribute(element,
"role",
"toolbar");
501 setAttribute(element,
"title", text.toStdString());
502 setHtmlElementOrientation(element, iface);
503 addEventListener(iface, element,
"click");
505 case QAccessible::MenuItem:
506 case QAccessible::ButtonDropDown:
507 case QAccessible::ButtonMenu: {
508 const QString text = iface->text(QAccessible::Name);
509 element = document.call<emscripten::val>(
"createElement", std::string(
"button"));
510 setAttribute(element,
"role",
"menuitem");
511 setAttribute(element,
"title", text.toStdString());
512 addEventListener(iface, element,
"click");
514 case QAccessible::MenuBar:
515 case QAccessible::PopupMenu: {
516 const QString text = iface->text(QAccessible::Name);
517 element = document.call<emscripten::val>(
"createElement", std::string(
"div"));
518 setAttribute(element,
"role",
"menubar");
519 setAttribute(element,
"title", text.toStdString());
520 setHtmlElementOrientation(element, iface);
521 m_elements[iface] = element;
523 for (
int i = 0; i < iface->childCount(); ++i) {
524 emscripten::val childElement = createHtmlElement(iface->child(i));
525 setAttribute(childElement,
"aria-owns", text.toStdString());
528 case QAccessible::EditableText: {
529 element = document.call<emscripten::val>(
"createElement", std::string(
"input"));
530 setAttribute(element,
"type",
"text");
531 setAttribute(element,
"contenteditable",
"true");
532 setAttribute(element,
"readonly", iface->state().readOnly);
533 setAttribute(element,
"autocorrect",
"off");
534 setProperty(element,
"inputMode",
"text");
536 case QAccessible::List: {
537 element = document.call<emscripten::val>(
"createElement", std::string(
"div"));
538 setAttribute(element,
"tabindex",
"0");
539 setAttribute(element,
"role",
"list");
541 case QAccessible::ListItem: {
542 element = document.call<emscripten::val>(
"createElement", std::string(
"div"));
543 setAttribute(element,
"tabindex",
"0");
544 setAttribute(element,
"role",
"listitem");
547 qCDebug(lcQpaAccessibility) <<
"TODO: createHtmlElement() handle" << iface->role();
548 element = document.call<emscripten::val>(
"createElement", std::string(
"div"));
551 addEventListener(iface, element,
"focus");
556 m_elements[iface] = element;
558 setHtmlElementGeometry(iface);
559 setHtmlElementDisabled(iface);
560 setHtmlElementVisibility(iface, !iface->state().invisible);
561 handleIdentifierUpdate(iface);
562 handleDescriptionChanged(iface);
563 sendEvent(iface, QAccessible::NameChanged);
567 for (
int i = 0; i < iface->childCount(); ++i) {
568 if (!getHtmlElement(iface->child(i)).isUndefined())
569 linkToParent(iface->child(i));
575void QWasmAccessibility::destroyHtmlElement(QAccessibleInterface *iface)
578 qCDebug(lcQpaAccessibility) <<
"TODO destroyHtmlElement";
581emscripten::val QWasmAccessibility::getHtmlElement(QAccessibleInterface *iface)
583 auto it = m_elements.find(iface);
584 if (it != m_elements.end())
587 return emscripten::val::undefined();
590void QWasmAccessibility::repairLinks(QAccessibleInterface *iface)
595 for (
int i = 0; i < iface->childCount(); ++i) {
596 const auto elementI = getHtmlElement(iface->child(i));
597 const auto containerI = getElementContainer(iface->child(i));
599 if (!elementI.isUndefined() &&
600 !containerI.isUndefined() &&
601 !elementI[
"parentElement"].isUndefined() &&
602 !elementI[
"parentElement"].isNull() &&
603 elementI[
"parentElement"] != containerI) {
609 for (
int i = 0; i < iface->childCount(); ++i) {
610 const auto elementI = getHtmlElement(iface->child(i));
611 const auto containerI = getElementContainer(iface->child(i));
612 if (!elementI.isUndefined() && !containerI.isUndefined())
613 containerI.call<
void>(
"appendChild", elementI);
618void QWasmAccessibility::linkToParent(QAccessibleInterface *iface)
620 emscripten::val element = getHtmlElement(iface);
621 emscripten::val container = getElementContainer(iface);
623 if (container.isUndefined() || element.isUndefined())
627 const auto activeElementBefore = emscripten::val::take_ownership(
628 getActiveElement_js(emscripten::val::undefined().as_handle()));
631 repairLinks(iface->parent());
633 emscripten::val next = emscripten::val::undefined();
634 const int thisIndex = iface->parent()->indexOfChild(iface);
635 if (thisIndex >= 0) {
636 Q_ASSERT(thisIndex < iface->parent()->childCount());
637 for (
int i = thisIndex + 1; i < iface->parent()->childCount(); ++i) {
638 const auto elementI = getHtmlElement(iface->parent()->child(i));
639 if (!elementI.isUndefined() &&
640 elementI[
"parentElement"] == container) {
645 if (next.isUndefined())
646 container.call<
void>(
"appendChild", element);
648 container.call<
void>(
"insertBefore", element, next);
650 const auto activeElementAfter = emscripten::val::take_ownership(
651 getActiveElement_js(emscripten::val::undefined().as_handle()));
652 if (activeElementBefore != activeElementAfter) {
653 if (!activeElementBefore.isUndefined() && !activeElementBefore.isNull())
654 activeElementBefore.call<
void>(
"focus");
658void QWasmAccessibility::setHtmlElementVisibility(QAccessibleInterface *iface,
bool visible)
660 emscripten::val element = getHtmlElement(iface);
661 setAttribute(element,
"inert", !visible);
664void QWasmAccessibility::setHtmlElementGeometry(QAccessibleInterface *iface)
666 const emscripten::val element = getHtmlElement(iface);
668 QRect windowGeometry = iface->rect();
669 if (iface->parent()) {
673 const QRect parentRect = iface->parent()->rect();
674 const QRect thisRect = iface->rect();
675 const QRect result(thisRect.topLeft() - parentRect.topLeft(), thisRect.size());
676 windowGeometry = result;
680 Q_ASSERT(!getWindow(iface));
682 setHtmlElementGeometry(element, windowGeometry);
685void QWasmAccessibility::setHtmlElementGeometry(emscripten::val element, QRect geometry)
689 emscripten::val style = element[
"style"];
690 style.set(
"position", std::string(
"absolute"));
691 style.set(
"z-index", std::string(
"-1"));
693 style.set(
"left", std::to_string(geometry.x()) +
"px");
694 style.set(
"top", std::to_string(geometry.y()) +
"px");
695 style.set(
"width", std::to_string(geometry.width()) +
"px");
696 style.set(
"height", std::to_string(geometry.height()) +
"px");
699void QWasmAccessibility::setHtmlElementFocus(QAccessibleInterface *iface)
701 const auto element = getHtmlElement(iface);
702 element.call<
void>(
"focus");
705void QWasmAccessibility::setHtmlElementDisabled(QAccessibleInterface *iface)
707 auto element = getHtmlElement(iface);
708 setAttribute(element,
"aria-disabled", iface->state().disabled);
711void QWasmAccessibility::setHtmlElementOrientation(emscripten::val element, QAccessibleInterface *iface)
714 if (QAccessibleAttributesInterface *attributesIface = iface->attributesInterface()) {
715 const QVariant orientationVariant =
716 attributesIface->attributeValue(QAccessible::Attribute::Orientation);
717 if (orientationVariant.isValid()) {
718 Q_ASSERT(orientationVariant.canConvert<Qt::Orientation>());
719 const Qt::Orientation orientation = orientationVariant.value<Qt::Orientation>();
720 const std::string value = orientation == Qt::Horizontal ?
"horizontal" :
"vertical";
721 setAttribute(element,
"aria-orientation", value);
726void QWasmAccessibility::handleListItemUpdate(QAccessibleEvent *event)
728 auto iface = event->accessibleInterface();
730 switch (event->type()) {
731 case QAccessible::NameChanged: {
733 setNamedProperty(iface,
"innerText", QAccessible::Name);
736 qCDebug(lcQpaAccessibility) <<
"TODO: implement handleListItemUpdate for event" << event->type();
741void QWasmAccessibility::handleDialUpdate(QAccessibleEvent *event)
743 switch (event->type()) {
744 case QAccessible::ObjectCreated:
745 case QAccessible::StateChanged: {
747 case QAccessible::Focus:
748 case QAccessible::NameChanged: {
749 setNamedAttribute(event->accessibleInterface(),
"aria-label", QAccessible::Name);
751 case QAccessible::ValueChanged: {
752 QAccessibleInterface *accessible = event->accessibleInterface();
753 const emscripten::val element = getHtmlElement(accessible);
754 std::string valueString = accessible->valueInterface()->currentValue().toString().toStdString();
755 setProperty(element,
"value", valueString);
758 qDebug() <<
"TODO: implement handleSpinBoxUpdate for event" << event->type();
763void QWasmAccessibility::handleStaticTextUpdate(QAccessibleEvent *event)
765 switch (event->type()) {
766 case QAccessible::NameChanged: {
768 setNamedProperty(event->accessibleInterface(),
"innerText", QAccessible::Name);
771 qCDebug(lcQpaAccessibility) <<
"TODO: implement handleStaticTextUpdate for event" << event->type();
776void QWasmAccessibility::handleLineEditUpdate(QAccessibleEvent *event)
778 switch (event->type()) {
779 case QAccessible::StateChanged: {
780 auto iface = event->accessibleInterface();
781 auto element = getHtmlElement(iface);
782 setAttribute(element,
"readonly", iface->state().readOnly);
783 if (iface->state().passwordEdit)
784 setProperty(element,
"type",
"password");
786 setProperty(element,
"type",
"text");
788 case QAccessible::ValueChanged:
789 case QAccessible::NameChanged: {
790 setNamedProperty(event->accessibleInterface(),
"value", QAccessible::Value);
792 case QAccessible::ObjectShow:
793 case QAccessible::Focus: {
794 auto iface = event->accessibleInterface();
795 auto element = getHtmlElement(iface);
796 if (!element.isUndefined()) {
797 setAttribute(element,
"readonly", iface->state().readOnly);
798 if (iface->state().passwordEdit)
799 setProperty(element,
"type",
"password");
801 setProperty(element,
"type",
"text");
803 setNamedProperty(event->accessibleInterface(),
"value", QAccessible::Value);
805 case QAccessible::TextRemoved:
806 case QAccessible::TextInserted: {
807 setNamedProperty(event->accessibleInterface(),
"value", QAccessible::Value);
809 case QAccessible::TextCaretMoved: {
811 setNamedProperty(event->accessibleInterface(),
"value", QAccessible::Value);
814 qCDebug(lcQpaAccessibility) <<
"TODO: implement handleLineEditUpdate for event" << event->type();
819void QWasmAccessibility::handleEventFromHtmlElement(
const emscripten::val event)
821 if (event[
"target"].isNull() || event[
"target"].isUndefined())
824 if (event[
"target"][
"data-qta11yinterface"].isNull() || event[
"target"][
"data-qta11yinterface"].isUndefined())
827 auto iface =
reinterpret_cast<QAccessibleInterface *>(event[
"target"][
"data-qta11yinterface"].as<size_t>());
828 if (m_elements.find(iface) == m_elements.end())
831 const auto eventType = QString::fromStdString(event[
"type"].as<std::string>());
832 const auto& actionNames = QAccessibleBridgeUtils::effectiveActionNames(iface);
833 bool handled =
false;
835 if (eventType ==
"blur") {
837 }
else if (eventType ==
"keydown") {
839 }
else if (eventType ==
"keyup") {
841 }
else if (eventType ==
"focus") {
842 if (actionNames.contains(QAccessibleActionInterface::setFocusAction()))
843 iface->actionInterface()->doAction(QAccessibleActionInterface::setFocusAction());
844 }
else if (eventType ==
"click") {
846 if (actionNames.contains(QAccessibleActionInterface::pressAction()))
847 iface->actionInterface()->doAction(QAccessibleActionInterface::pressAction());
848 else if (actionNames.contains(QAccessibleActionInterface::showMenuAction()))
849 iface->actionInterface()->doAction(QAccessibleActionInterface::showMenuAction());
850 }
else if (eventType ==
"change") {
854 if (actionNames.contains(QAccessibleActionInterface::pressAction()))
855 iface->actionInterface()->doAction(QAccessibleActionInterface::pressAction());
856 else if (actionNames.contains(QAccessibleActionInterface::toggleAction()))
857 iface->actionInterface()->doAction(QAccessibleActionInterface::toggleAction());
859 qWarning() <<
" Unknown event" << eventType;
863 event.call<
void>(
"preventDefault");
864 event.call<
void>(
"stopPropagation");
868void QWasmAccessibility::handleButtonUpdate(QAccessibleEvent *event)
870 switch (event->type()) {
871 case QAccessible::Focus:
872 case QAccessible::NameChanged: {
873 setNamedAttribute(event->accessibleInterface(),
"aria-label", QAccessible::Name);
876 qCDebug(lcQpaAccessibility) <<
"TODO: implement handleCheckBoxUpdate for event" << event->type();
881void QWasmAccessibility::handleCheckBoxUpdate(QAccessibleEvent *event)
883 switch (event->type()) {
884 case QAccessible::Focus:
885 case QAccessible::NameChanged: {
886 setNamedAttribute(event->accessibleInterface(),
"aria-label", QAccessible::Name);
888 case QAccessible::StateChanged: {
889 QAccessibleInterface *accessible = event->accessibleInterface();
890 const emscripten::val element = getHtmlElement(accessible);
891 setAttribute(element,
"checked", accessible->state().checked);
892 setProperty(element,
"indeterminate", accessible->state().checkStateMixed);
895 qCDebug(lcQpaAccessibility) <<
"TODO: implement handleCheckBoxUpdate for event" << event->type();
900void QWasmAccessibility::handleGroupBoxUpdate(QAccessibleEvent *event)
902 QAccessibleInterface *iface = event->accessibleInterface();
904 emscripten::val parent = getHtmlElement(iface);
905 emscripten::val label = parent[
"children"][0];
906 emscripten::val checkbox = emscripten::val::undefined();
907 if (iface->role() == QAccessible::CheckBox)
908 checkbox = parent[
"children"][1];
910 switch (event->type()) {
911 case QAccessible::Focus:
912 case QAccessible::NameChanged: {
913 setProperty(label,
"innerText", iface->text(QAccessible::Name).toStdString());
914 if (!checkbox.isUndefined())
915 setAttribute(checkbox,
"aria-label", iface->text(QAccessible::Name).toStdString());
917 case QAccessible::StateChanged: {
918 if (!checkbox.isUndefined()) {
919 setAttribute(checkbox,
"checked", iface->state().checked);
920 setProperty(checkbox,
"indeterminate", iface->state().checkStateMixed);
924 qCDebug(lcQpaAccessibility) <<
"TODO: implement handleCheckBoxUpdate for event" << event->type();
929void QWasmAccessibility::handleSwitchUpdate(QAccessibleEvent *event)
931 switch (event->type()) {
932 case QAccessible::Focus:
933 case QAccessible::NameChanged: {
935 setNamedAttribute(event->accessibleInterface(),
"aria-label", QAccessible::Name);
937 case QAccessible::StateChanged: {
938 QAccessibleInterface *accessible = event->accessibleInterface();
939 const emscripten::val element = getHtmlElement(accessible);
940 if (accessible->state().checked)
941 setAttribute(element,
"aria-checked",
"true");
943 setAttribute(element,
"aria-checked",
"false");
946 qCDebug(lcQpaAccessibility) <<
"TODO: implement handleSwitchUpdate for event" << event->type();
951void QWasmAccessibility::handleToolUpdate(QAccessibleEvent *event)
953 QAccessibleInterface *iface = event->accessibleInterface();
954 QString text = iface->text(QAccessible::Name);
955 QString desc = iface->text(QAccessible::Description);
956 switch (event->type()) {
957 case QAccessible::NameChanged:
958 case QAccessible::StateChanged:{
959 const emscripten::val element = getHtmlElement(iface);
960 setAttribute(element,
"title", text.toStdString());
963 qCDebug(lcQpaAccessibility) <<
"TODO: implement handleToolUpdate for event" << event->type();
967void QWasmAccessibility::handleMenuUpdate(QAccessibleEvent *event)
969 QAccessibleInterface *iface = event->accessibleInterface();
970 QString text = iface->text(QAccessible::Name);
971 QString desc = iface->text(QAccessible::Description);
972 switch (event->type()) {
973 case QAccessible::Focus:
974 case QAccessible::NameChanged:
975 case QAccessible::MenuStart :
976 case QAccessible::StateChanged:{
977 const emscripten::val element = getHtmlElement(iface);
978 setAttribute(element,
"title", text.toStdString());
980 case QAccessible::PopupMenuStart: {
981 if (iface->childCount() > 0) {
982 const auto childElement = getHtmlElement(iface->child(0));
983 childElement.call<
void>(
"focus");
987 qCDebug(lcQpaAccessibility) <<
"TODO: implement handleMenuUpdate for event" << event->type();
991void QWasmAccessibility::handleDialogUpdate(QAccessibleEvent *event) {
993 switch (event->type()) {
994 case QAccessible::NameChanged:
995 case QAccessible::Focus:
996 case QAccessible::DialogStart:
997 case QAccessible::StateChanged: {
998 setNamedAttribute(event->accessibleInterface(),
"aria-label", QAccessible::Name);
1001 qCDebug(lcQpaAccessibility) <<
"TODO: implement handleLineEditUpdate for event" << event->type();
1006void QWasmAccessibility::populateAccessibilityTree(QAccessibleInterface *iface)
1013 const QWindow *window1 = getWindow(iface);
1014 const QWindow *window0 = (iface->parent()) ? getWindow(iface->parent()) :
nullptr;
1016 if (window1 && window0 == window1) {
1018 bool exists = !getHtmlElement(iface).isUndefined();
1020 exists = !createHtmlElement(iface).isUndefined();
1023 linkToParent(iface);
1024 setHtmlElementVisibility(iface, !iface->state().invisible);
1025 setHtmlElementGeometry(iface);
1026 setHtmlElementDisabled(iface);
1027 handleIdentifierUpdate(iface);
1028 handleDescriptionChanged(iface);
1029 sendEvent(iface, QAccessible::NameChanged);
1032 for (
int i = 0; i < iface->childCount(); ++i)
1033 populateAccessibilityTree(iface->child(i));
1036void QWasmAccessibility::handleRadioButtonUpdate(QAccessibleEvent *event)
1038 switch (event->type()) {
1039 case QAccessible::Focus:
1040 case QAccessible::NameChanged: {
1041 setNamedAttribute(event->accessibleInterface(),
"aria-label", QAccessible::Name);
1043 case QAccessible::StateChanged: {
1044 QAccessibleInterface *accessible = event->accessibleInterface();
1045 const emscripten::val element = getHtmlElement(accessible);
1046 setAttribute(element,
"checked", accessible->state().checked);
1049 qDebug() <<
"TODO: implement handleRadioButtonUpdate for event" << event->type();
1054void QWasmAccessibility::handleSpinBoxUpdate(QAccessibleEvent *event)
1056 switch (event->type()) {
1057 case QAccessible::ObjectCreated:
1058 case QAccessible::StateChanged: {
1060 case QAccessible::Focus:
1061 case QAccessible::NameChanged: {
1062 setNamedAttribute(event->accessibleInterface(),
"aria-label", QAccessible::Name);
1064 case QAccessible::ValueChanged: {
1065 QAccessibleInterface *accessible = event->accessibleInterface();
1066 const emscripten::val element = getHtmlElement(accessible);
1067 std::string valueString = accessible->valueInterface()->currentValue().toString().toStdString();
1068 setProperty(element,
"value", valueString);
1071 qDebug() <<
"TODO: implement handleSpinBoxUpdate for event" << event->type();
1076void QWasmAccessibility::handleSliderUpdate(QAccessibleEvent *event)
1078 switch (event->type()) {
1079 case QAccessible::ObjectCreated:
1080 case QAccessible::StateChanged: {
1082 case QAccessible::Focus:
1083 case QAccessible::NameChanged: {
1084 setNamedAttribute(event->accessibleInterface(),
"aria-label", QAccessible::Name);
1086 case QAccessible::ValueChanged: {
1087 QAccessibleInterface *accessible = event->accessibleInterface();
1088 const emscripten::val element = getHtmlElement(accessible);
1089 std::string valueString = accessible->valueInterface()->currentValue().toString().toStdString();
1090 setProperty(element,
"value", valueString);
1093 qDebug() <<
"TODO: implement handleSliderUpdate for event" << event->type();
1098void QWasmAccessibility::handleScrollBarUpdate(QAccessibleEvent *event)
1100 switch (event->type()) {
1101 case QAccessible::Focus:
1102 case QAccessible::NameChanged: {
1103 setNamedAttribute(event->accessibleInterface(),
"aria-label", QAccessible::Name);
1105 case QAccessible::ValueChanged: {
1106 QAccessibleInterface *accessible = event->accessibleInterface();
1107 const emscripten::val element = getHtmlElement(accessible);
1108 std::string valueString = accessible->valueInterface()->currentValue().toString().toStdString();
1109 setAttribute(element,
"aria-valuenow", valueString);
1112 qDebug() <<
"TODO: implement handleSliderUpdate for event" << event->type();
1118void QWasmAccessibility::handlePageTabUpdate(QAccessibleEvent *event)
1120 switch (event->type()) {
1121 case QAccessible::NameChanged: {
1122 setNamedAttribute(event->accessibleInterface(),
"aria-label", QAccessible::Name);
1124 case QAccessible::Focus: {
1125 setNamedAttribute(event->accessibleInterface(),
"aria-label", QAccessible::Name);
1128 qDebug() <<
"TODO: implement handlePageTabUpdate for event" << event->type();
1133void QWasmAccessibility::handlePageTabListUpdate(QAccessibleEvent *event)
1135 switch (event->type()) {
1136 case QAccessible::NameChanged: {
1137 setNamedAttribute(event->accessibleInterface(),
"aria-label", QAccessible::Name);
1139 case QAccessible::Focus: {
1140 setNamedAttribute(event->accessibleInterface(),
"aria-label", QAccessible::Name);
1143 qDebug() <<
"TODO: implement handlePageTabUpdate for event" << event->type();
1148void QWasmAccessibility::handleIdentifierUpdate(QAccessibleInterface *iface)
1150 const emscripten::val element = getHtmlElement(iface);
1151 QString id = iface->text(QAccessible::Identifier).replace(
" ",
"_");
1152 if (id.isEmpty() && iface->role() == QAccessible::PageTabList) {
1153 std::ostringstream oss;
1154 oss <<
"tabList_0x" << (
void *)iface;
1155 id = QString::fromUtf8(oss.str());
1158 setAttribute(element,
"id", id.toStdString());
1159 if (!id.isEmpty()) {
1160 if (iface->role() == QAccessible::PageTabList) {
1161 for (
int i = 0; i < iface->childCount(); ++i) {
1162 const auto child = getHtmlElement(iface->child(i));
1163 setAttribute(child,
"aria-owns", id.toStdString());
1169void QWasmAccessibility::handleDescriptionChanged(QAccessibleInterface *iface)
1171 const auto desc = iface->text(QAccessible::Description).toStdString();
1172 auto element = getHtmlElement(iface);
1173 auto container = getDescribedByContainer(iface);
1174 if (!container.isUndefined()) {
1175 std::ostringstream oss;
1176 oss <<
"dbid_" << (
void *)iface;
1177 auto id = oss.str();
1179 auto describedBy = container.call<emscripten::val>(
"querySelector",
"#" + std::string(id));
1181 if (!describedBy.isUndefined() && !describedBy.isNull()) {
1182 container.call<
void>(
"removeChild", describedBy);
1184 setAttribute(element,
"aria-describedby",
"");
1186 if (describedBy.isUndefined() || describedBy.isNull()) {
1187 auto document = getDocument(container);
1188 describedBy = document.call<emscripten::val>(
"createElement", std::string(
"p"));
1190 container.call<
void>(
"appendChild", describedBy);
1192 setAttribute(describedBy,
"id", id);
1193 setAttribute(describedBy,
"inert",
true);
1194 setAttribute(element,
"aria-describedby", id);
1195 setProperty(describedBy,
"innerText", desc);
1200void QWasmAccessibility::handleAnnouncement(QAccessibleInterface *iface,
1201 QAccessibleAnnouncementEvent *announcementEvent)
1203 emscripten::val element = getHtmlElement(iface);
1204 if (element[
"ariaNotify"].isUndefined())
1207 const std::string message = announcementEvent->message().toStdString();
1208 const std::string prio(announcementEvent->politeness()
1209 == QAccessible::AnnouncementPoliteness::Assertive
1212 emscripten::val options = emscripten::val::object();
1213 options.set(
"priority", prio);
1214 element.call<
void>(
"ariaNotify", message, options);
1217void QWasmAccessibility::createObject(QAccessibleInterface *iface)
1219 if (getHtmlElement(iface).isUndefined())
1220 createHtmlElement(iface);
1223void QWasmAccessibility::removeObject(QAccessibleInterface *iface)
1230 const auto it = m_elements.find(iface);
1231 if (it != m_elements.end()) {
1232 auto element = it.value();
1233 auto container = getDescribedByContainer(iface);
1234 if (!container.isUndefined()) {
1235 std::ostringstream oss;
1236 oss <<
"dbid_" << (
void *)iface;
1237 auto id = oss.str();
1238 auto describedBy = container.call<emscripten::val>(
"querySelector",
"#" + std::string(id));
1239 if (!describedBy.isUndefined() && !describedBy.isNull() &&
1240 !describedBy[
"parentElement"].isUndefined() && !describedBy[
"parentElement"].isNull())
1241 describedBy[
"parentElement"].call<
void>(
"removeChild", describedBy);
1243 if (!element[
"parentElement"].isUndefined() && !element[
"parentElement"].isNull())
1244 element[
"parentElement"].call<
void>(
"removeChild", element);
1245 m_elements.erase(it);
1249void QWasmAccessibility::unlinkParentForChildren(QAccessibleInterface *iface)
1251 auto element = getHtmlElement(iface);
1252 if (!element.isUndefined()) {
1253 auto oldContainer = element[
"parentElement"];
1254 auto newContainer = getElementContainer(iface);
1255 if (!oldContainer.isUndefined() &&
1256 !oldContainer.isNull() &&
1257 oldContainer != newContainer) {
1258 oldContainer.call<
void>(
"removeChild", element);
1261 for (
int i = 0; i < iface->childCount(); ++i)
1262 unlinkParentForChildren(iface->child(i));
1265void QWasmAccessibility::relinkParentForChildren(QAccessibleInterface *iface)
1267 auto element = getHtmlElement(iface);
1268 if (!element.isUndefined()) {
1269 if (element[
"parentElement"].isUndefined() ||
1270 element[
"parentElement"].isNull()) {
1271 linkToParent(iface);
1274 for (
int i = 0; i < iface->childCount(); ++i)
1275 relinkParentForChildren(iface->child(i));
1278void QWasmAccessibility::notifyAccessibilityUpdate(QAccessibleEvent *event)
1280 if (handleUpdateByEventType(event))
1281 handleUpdateByInterfaceRole(event);
1284bool QWasmAccessibility::handleUpdateByEventType(QAccessibleEvent *event)
1286 if (!m_accessibilityEnabled)
1289 QAccessibleInterface *iface = event->accessibleInterface();
1291 qWarning() <<
"handleUpdateByEventType with null a11y interface" << event->type() << event->object();
1296 switch (event->type()) {
1297 case QAccessible::ObjectCreated:
1302 case QAccessible::ObjectDestroyed:
1305 removeObject(iface);
1308 case QAccessible::ObjectShow:
1309 createObject(iface);
1312 case QAccessible::ParentChanged:
1313 unlinkParentForChildren(iface);
1314 relinkParentForChildren(iface);
1321 if (getHtmlElement(iface).isUndefined())
1326 switch (event->type()) {
1327 case QAccessible::Announcement:
1328 handleAnnouncement(iface,
static_cast<QAccessibleAnnouncementEvent *>(event));
1331 case QAccessible::StateChanged: {
1332 QAccessibleStateChangeEvent *stateChangeEvent = (QAccessibleStateChangeEvent *)event;
1333 if (stateChangeEvent->changedStates().disabled)
1334 setHtmlElementDisabled(iface);
1337 case QAccessible::DescriptionChanged:
1338 handleDescriptionChanged(iface);
1341 case QAccessible::Focus:
1344 setHtmlElementGeometry(iface);
1345 setHtmlElementFocus(iface);
1348 case QAccessible::IdentifierChanged:
1349 handleIdentifierUpdate(iface);
1352 case QAccessible::ObjectShow:
1353 linkToParent(iface);
1354 setHtmlElementVisibility(iface,
true);
1357 setHtmlElementGeometry(iface);
1358 sendEvent(iface, QAccessible::NameChanged);
1361 case QAccessible::ObjectHide:
1363 setHtmlElementVisibility(iface,
false);
1366 case QAccessible::LocationChanged:
1367 setHtmlElementGeometry(iface);
1378void QWasmAccessibility::handleUpdateByInterfaceRole(QAccessibleEvent *event)
1380 if (!m_accessibilityEnabled)
1383 QAccessibleInterface *iface = event->accessibleInterface();
1385 qWarning() <<
"handleUpdateByInterfaceRole with null a11y interface" << event->type() << event->object();
1391 switch (iface->role()) {
1392 case QAccessible::List:
1394 case QAccessible::ListItem:
1395 handleListItemUpdate(event);
1397 case QAccessible::StaticText:
1398 handleStaticTextUpdate(event);
1400 case QAccessible::Button:
1401 handleButtonUpdate(event);
1403 case QAccessible::CheckBox:
1404 if (iface->childCount() > 0)
1405 handleGroupBoxUpdate(event);
1407 handleCheckBoxUpdate(event);
1409 case QAccessible::Switch:
1410 handleSwitchUpdate(event);
1412 case QAccessible::EditableText:
1413 handleLineEditUpdate(event);
1415 case QAccessible::Dialog:
1416 handleDialogUpdate(event);
1418 case QAccessible::MenuItem:
1419 case QAccessible::MenuBar:
1420 case QAccessible::PopupMenu:
1421 handleMenuUpdate(event);
1423 case QAccessible::ToolBar:
1424 case QAccessible::ButtonMenu:
1425 handleToolUpdate(event);
1426 case QAccessible::RadioButton:
1427 handleRadioButtonUpdate(event);
1429 case QAccessible::Dial:
1430 handleDialUpdate(event);
1432 case QAccessible::SpinBox:
1433 handleSpinBoxUpdate(event);
1435 case QAccessible::Slider:
1436 handleSliderUpdate(event);
1438 case QAccessible::PageTab:
1439 handlePageTabUpdate(event);
1441 case QAccessible::PageTabList:
1442 handlePageTabListUpdate(event);
1444 case QAccessible::ScrollBar:
1445 handleScrollBarUpdate(event);
1447 case QAccessible::Grouping:
1448 handleGroupBoxUpdate(event);
1451 qCDebug(lcQpaAccessibility) <<
"TODO: implement notifyAccessibilityUpdate for role" << iface->role();
1455void QWasmAccessibility::setRootObject(QObject *root)
1457 m_rootObject = root;
1460void QWasmAccessibility::initialize()
1465void QWasmAccessibility::cleanup()
void QWasmAccessibilityEnable()