更改可聚焦视图组中元素的读取顺序



我的应用程序中有一些屏幕,其中TalkBack无法推断正确的阅读顺序。根据文档,我可以使用android:accessibilityTraversalAfter和朋友来更改阅读顺序。但它对我不起作用,因为应该一起阅读的可聚焦ViewGroup中的元素。

整个布局如下所示:

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:accessibilityTraversalBefore="@id/before"
android:focusable="true"
tools:context=".MainActivity">
<TextView
android:id="@+id/before"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:accessibilityTraversalAfter="@id/before"
android:text="Before"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent" />
<TextView
android:id="@+id/after"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:accessibilityTraversalBefore="@id/after"
android:text="After"
app:layout_constraintBottom_toTopOf="@+id/before"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
</androidx.constraintlayout.widget.ConstraintLayout>

它在屏幕中间呈现AfterBefore在底部。我希望 TalkBack 将整个屏幕视为单个连续元素,因此我将android:focusable设置为true。默认情况下,TalkBack 显示:">之后,之前"。但我希望它读作"之前,之后"。虽然我添加了android:accessibilityTraversalBeforeandroid:accessibilityTraversalAfter,但它仍然显示为"之后,之前"。这是节点树调试的输出:

TreeDebug: (-2147455381)429.FrameLayout:(0, 0 - 1080, 1920):A
TreeDebug:   (30189)429.TextView:(42, 101 - 397, 172):TEXT{My Application}:A:supportsTextLocation
TreeDebug:   (31150)429.ViewGroup:(0, 210 - 1080, 1794):Fa:focusable:accessibilityFocused
TreeDebug:     (33072)429.TextView:(499, 951 - 581, 1002):TEXT{After}:A:supportsTextLocation
TreeDebug:     (32111)429.TextView:(485, 1743 - 595, 1794):TEXT{Before}:A:supportsTextLocation

我做错了什么?

只是为了完整起见:minSdkVersion是 26,targetSdkVersion是 29。

我已经深入研究了这个问题并发现了以下内容:契约中的accessibilityTraversalBeforeaccessibilityTraversalAfter标志生效,但仅用于它们的用途 - 它们适用于辅助功能服务应用程序(例如对讲(。换句话说,如果从根布局中删除focusable属性,您将看到导航是正确的。

但是这些标志不会影响如何为根ViewGroup构建AccessibilityNode。从ViewGroup#onInitializeAccessibilityNodeInfoInternal()来源可以看出,实际的文本构造逻辑并没有考虑孩子如何使用上面提到的标志来构建他们的导航。

为了解决这个问题,我从布局xml中删除了过多的标志:

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/root"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:focusable="true"
tools:context=".MainActivity">
<TextView
android:id="@+id/before"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:importantForAccessibility="no"
android:text="Before"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<TextView
android:id="@+id/after"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:importantForAccessibility="no"
android:text="After"
app:layout_constraintBottom_toTopOf="@+id/before"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
</androidx.constraintlayout.widget.ConstraintLayout>

在主机活动/片段内部:

val root = findViewById<ViewGroup>(R.id.root)
ViewCompat.setAccessibilityDelegate(root, object : AccessibilityDelegateCompat() {
override fun onInitializeAccessibilityNodeInfo(
host: View?,
info: AccessibilityNodeInfoCompat?
) {
val stringBuilder = StringBuilder()
root.children.forEach { view ->
val label = if (view is TextView) view.text else ""
stringBuilder.append("$label, ")
}
info?.text = stringBuilder.toString()
super.onInitializeAccessibilityNodeInfo(host, info)
}
})

这将产生预期的结果:对讲将发音为"之前,之后"。

不幸的是,这是一个容易出错的代码,这意味着如果你以某种方式重构视图层次结构,子级的顺序被交换,那么这个节点文本构造逻辑将被破坏。尽管如此,我还是想不出更好的解决方案,也看不到可以指示父级考虑子排序标志(基于来源(。

最新更新