75 __android_log_print(ANDROID_LOG_WARN, m_qtTag,
76 "Could not run accessibility call in object context, no valid surface.");
80 QtAndroidPrivate::AndroidDeadlockProtector protector(
81 u"QtAndroidAccessibility::runInObjectContext()"_s);
82 if (!protector.acquire()) {
83 __android_log_print(ANDROID_LOG_WARN, m_qtTag,
84 "Could not run accessibility call in object context, accessing "
85 "main thread could lead to deadlock");
89 if (!QtAndroid::blockEventLoopsWhenSuspended()
90 || QGuiApplication::applicationState() != Qt::ApplicationSuspended) {
91 QMetaObject::invokeMethod(context, func, Qt::BlockingQueuedConnection, retVal);
93 __android_log_print(ANDROID_LOG_WARN, m_qtTag,
94 "Could not run accessibility call in object context, event loop suspended.");
310 QAccessibleInterface *iface = interfaceFromId(objectId);
311 if (!iface || !iface->isValid() || !iface->actionInterface())
314 const auto& actionNames = iface->actionInterface()->actionNames();
316 if (actionNames.contains(QAccessibleActionInterface::pressAction())) {
317 invokeActionOnInterfaceInMainThread(iface->actionInterface(),
318 QAccessibleActionInterface::pressAction());
319 }
else if (actionNames.contains(QAccessibleActionInterface::toggleAction())) {
320 invokeActionOnInterfaceInMainThread(iface->actionInterface(),
321 QAccessibleActionInterface::toggleAction());
419 QAccessibleValueInterface *valueIface = iface->valueInterface();
421 const QVariant valueVar = valueIface->currentValue();
422 const auto type = valueVar.typeId();
423 if (type == QMetaType::Double || type == QMetaType::Float) {
429 const double val = valueVar.toDouble();
431 bool stepIsValid =
false;
432 const double step = qAbs(valueIface->minimumStepSize().toDouble(&stepIsValid));
433 if (!stepIsValid || qFuzzyIsNull(step)) {
435 valueStr = qFuzzyIsNull(val) ? u"0"_s : QString::number(val,
'f');
437 const int precision = [](
double s) {
439 while (s < 1. && !qFuzzyCompare(s, 1.)) {
446 const int stop = count + 3;
447 const auto fractional = [](
double v) {
449 std::modf(v + 0.5, &whole);
450 return qAbs(v - whole);
453 while (count < stop && !qFuzzyIsNull(s)) {
455 s = fractional(s * 10);
459 valueStr = qFuzzyIsNull(val / step) ? u"0"_s
460 : QString::number(val,
'f', precision);
463 valueStr = valueVar.toString();
482 case QAccessible::Role::Button:
483 case QAccessible::Role::Link:
486 return QStringLiteral(
"android.widget.ToggleButton");
487 return QStringLiteral(
"android.widget.Button");
489 case QAccessible::Role::CheckBox:
492 return QStringLiteral(
"android.widget.CompoundButton");
493 case QAccessible::Role::Switch:
494 return QStringLiteral(
"android.widget.Switch");
495 case QAccessible::Role::Clock:
496 return QStringLiteral(
"android.widget.TextClock");
497 case QAccessible::Role::ComboBox:
498 return QStringLiteral(
"android.widget.Spinner");
499 case QAccessible::Role::Graphic:
501 return QStringLiteral(
"android.widget.ImageView");
502 case QAccessible::Role::Grouping:
503 return QStringLiteral(
"android.view.ViewGroup");
504 case QAccessible::Role::List:
507 return QStringLiteral(
"android.widget.AbsListView");
508 case QAccessible::Role::MenuItem:
509 return QStringLiteral(
"android.view.MenuItem");
510 case QAccessible::Role::PopupMenu:
511 return QStringLiteral(
"android.widget.PopupMenu");
512 case QAccessible::Role::Separator:
513 return QStringLiteral(
"android.widget.Space");
514 case QAccessible::Role::ToolBar:
515 return QStringLiteral(
"android.view.Toolbar");
516 case QAccessible::Role::Heading: [[fallthrough]];
517 case QAccessible::Role::StaticText:
519 return QStringLiteral(
"android.widget.TextView");
520 case QAccessible::Role::EditableText:
521 return QStringLiteral(
"android.widget.EditText");
522 case QAccessible::Role::RadioButton:
523 return QStringLiteral(
"android.widget.RadioButton");
524 case QAccessible::Role::ProgressBar:
525 return QStringLiteral(
"android.widget.ProgressBar");
526 case QAccessible::Role::SpinBox:
527 return QStringLiteral(
"android.widget.NumberPicker");
528 case QAccessible::Role::WebDocument:
529 return QStringLiteral(
"android.webkit.WebView");
530 case QAccessible::Role::Dialog:
531 return QStringLiteral(
"android.app.AlertDialog");
532 case QAccessible::Role::PageTab:
533 return QStringLiteral(
"android.app.ActionBar.Tab");
534 case QAccessible::Role::PageTabList:
535 return QStringLiteral(
"android.widget.TabWidget");
536 case QAccessible::Role::ScrollBar:
537 return QStringLiteral(
"android.widget.Scroller");
538 case QAccessible::Role::Slider:
539 return QStringLiteral(
"com.google.android.material.slider.Slider");
540 case QAccessible::Role::Table:
543 return QStringLiteral(
"android.widget.GridView");
544 case QAccessible::Role::Pane:
546 return QStringLiteral(
"android.view.ViewGroup");
547 case QAccessible::Role::AlertMessage:
548 case QAccessible::Role::Animation:
549 case QAccessible::Role::Application:
550 case QAccessible::Role::Assistant:
551 case QAccessible::Role::BlockQuote:
552 case QAccessible::Role::Border:
553 case QAccessible::Role::ButtonDropGrid:
554 case QAccessible::Role::ButtonDropDown:
555 case QAccessible::Role::ButtonMenu:
556 case QAccessible::Role::Canvas:
557 case QAccessible::Role::Caret:
558 case QAccessible::Role::Cell:
559 case QAccessible::Role::Chart:
560 case QAccessible::Role::Client:
561 case QAccessible::Role::ColorChooser:
562 case QAccessible::Role::Column:
563 case QAccessible::Role::ColumnHeader:
564 case QAccessible::Role::ComplementaryContent:
565 case QAccessible::Role::Cursor:
566 case QAccessible::Role::Desktop:
567 case QAccessible::Role::Dial:
568 case QAccessible::Role::Document:
569 case QAccessible::Role::Equation:
570 case QAccessible::Role::Footer:
571 case QAccessible::Role::Form:
572 case QAccessible::Role::Grip:
573 case QAccessible::Role::HelpBalloon:
574 case QAccessible::Role::HotkeyField:
575 case QAccessible::Role::Indicator:
576 case QAccessible::Role::LayeredPane:
577 case QAccessible::Role::ListItem:
578 case QAccessible::Role::MenuBar:
579 case QAccessible::Role::NoRole:
580 case QAccessible::Role::Note:
581 case QAccessible::Role::Notification:
582 case QAccessible::Role::Paragraph:
583 case QAccessible::Role::PropertyPage:
584 case QAccessible::Role::Row:
585 case QAccessible::Role::RowHeader:
586 case QAccessible::Role::Section:
587 case QAccessible::Role::Sound:
588 case QAccessible::Role::Splitter:
589 case QAccessible::Role::StatusBar:
590 case QAccessible::Role::Terminal:
591 case QAccessible::Role::TitleBar:
592 case QAccessible::Role::ToolTip:
593 case QAccessible::Role::Tree:
594 case QAccessible::Role::TreeItem:
595 case QAccessible::Role::UserRole:
596 case QAccessible::Role::Whitespace:
597 case QAccessible::Role::Window:
601 return QStringLiteral(
"android.view.TextView");
672 QAccessibleInterface *iface = interfaceFromId(objectId);
673 if (iface && iface->isValid()) {
675 info.state = iface->state();
676 info.role = iface->role();
677 info.actions = QAccessibleBridgeUtils::effectiveActionNames(iface);
678 info.description = descriptionForInterface(iface);
679 info.identifier = QAccessibleBridgeUtils::accessibleId(iface);
680 QAccessibleTextInterface *textIface = iface->textInterface();
681 if (textIface && (textIface->selectionCount() > 0)) {
685 QAccessibleValueInterface *valueInterface = iface->valueInterface();
686 if (valueInterface) {
688 info.minValue = valueInterface->minimumValue();
689 info.maxValue = valueInterface->maximumValue();
690 info.currentValue = valueInterface->currentValue();
691 info.valueStepSize = valueInterface->minimumStepSize();
700 if (m_accessibilityContext) {
701 runInObjectContext(m_accessibilityContext, [objectId]() {
702 return populateNode_helper(objectId);
706 __android_log_print(ANDROID_LOG_WARN, m_qtTag,
"Accessibility: populateNode for Invalid ID");
710 const QString role = classNameForRole(info.role, info.state);
711 jstring jrole = env->NewString((jchar*)role.constData(), (jsize)role.size());
712 env->CallVoidMethod(node, m_setClassNameMethodID, jrole);
714 const bool hasClickableAction =
715 (info.actions.contains(QAccessibleActionInterface::pressAction())
716 || info.actions.contains(QAccessibleActionInterface::toggleAction()))
717 && !(info.role == QAccessible::StaticText || info.role == QAccessible::Heading);
718 const bool hasIncreaseAction =
719 info.actions.contains(QAccessibleActionInterface::increaseAction());
720 const bool hasDecreaseAction =
721 info.actions.contains(QAccessibleActionInterface::decreaseAction());
724 env->CallVoidMethod(node, m_setTextSelectionMethodID, info
.selectionStart,
728 if (info
.hasValue && m_setRangeInfoMethodID) {
729 int valueType = info.currentValue.typeId();
732 case QMetaType::Float:
733 case QMetaType::Double:
741 float min = info.minValue.toFloat();
742 float max = info.maxValue.toFloat();
743 float current = info.currentValue.toFloat();
744 if (info.role == QAccessible::ProgressBar) {
746 current = 100 * (current - min) / (max - min);
751 QJniObject rangeInfo(
"android/view/accessibility/AccessibilityNodeInfo$RangeInfo",
752 "(IFFF)V", rangeType, min, max, current);
754 if (rangeInfo.isValid()) {
755 env->CallVoidMethod(node, m_setRangeInfoMethodID, rangeInfo.object());
759 env->CallVoidMethod(node, m_setCheckableMethodID, (
bool)info.state.checkable);
760 env->CallVoidMethod(node, m_setCheckedMethodID, (
bool)info.state.checked);
761 env->CallVoidMethod(node, m_setEditableMethodID, info.state.editable);
762 env->CallVoidMethod(node, m_setEnabledMethodID, !info.state.disabled);
763 env->CallVoidMethod(node, m_setFocusableMethodID, (
bool)info.state.focusable);
764 env->CallVoidMethod(node, m_setFocusedMethodID, (
bool)info.state.focused);
766 env->CallVoidMethod(node, m_setHeadingMethodID, info.role == QAccessible::Heading);
767 env->CallVoidMethod(node, m_setVisibleToUserMethodID, !info.state.invisible);
768 env->CallVoidMethod(node, m_setScrollableMethodID, hasIncreaseAction || hasDecreaseAction);
769 env->CallVoidMethod(node, m_setClickableMethodID, hasClickableAction || info.role == QAccessible::Link);
772 if (hasClickableAction)
773 env->CallVoidMethod(node, m_addActionMethodID, (
int)0x00000010);
776 if (hasIncreaseAction)
777 env->CallVoidMethod(node, m_addActionMethodID, (
int)0x00001000);
780 if (hasDecreaseAction)
781 env->CallVoidMethod(node, m_addActionMethodID, (
int)0x00002000);
784 jstring jdesc = env->NewString((jchar*)info.description.constData(),
785 (jsize)info.description.size());
787 env->CallVoidMethod(node, m_setContentDescriptionMethodID, jdesc);
789 QJniObject(node).callMethod<
void>(
"setViewIdResourceName", info.identifier);
825 if (!env.registerNativeMethods(
"org/qtproject/qt/android/QtNativeAccessibility",
827 __android_log_print(ANDROID_LOG_FATAL,
"Qt A11y",
"RegisterNatives failed");
831 jclass nodeInfoClass = env->FindClass(
"android/view/accessibility/AccessibilityNodeInfo");
837 GET_AND_CHECK_STATIC_METHOD(m_setContentDescriptionMethodID, nodeInfoClass,
"setContentDescription",
"(Ljava/lang/CharSequence;)V");
842 if (QtAndroidPrivate::androidSdkVersion() >= 28) {
849 m_setRangeInfoMethodID, nodeInfoClass,
"setRangeInfo",
850 "(Landroid/view/accessibility/AccessibilityNodeInfo$RangeInfo;)V");
852 jclass rangeInfoClass =
853 env->FindClass(
"android/view/accessibility/AccessibilityNodeInfo$RangeInfo");
857 if (QtAndroidPrivate::androidSdkVersion() >= 36) {
859 "RANGE_TYPE_INDETERMINATE");