[英]View is not selectable after it is made VISIBLE with a delayed transition when Talkback is enabled in Android
当以某种方式使视图可见时,Talkback 不能再 select/单击/聚焦该视图,请参阅: https://github.com/kai-vala/DemoForAccessibilityBugs/blob/master/images/animation.gif
(没有足够的代表嵌入图像)
完整的示例应用程序在这里: https://github.com/kai-vala/DemoForAccessibilityBugs
意识到它是由过渡引起的,如果我们删除: TransitionManager.beginDelayedTransition(it)
,视图总是可选择的。
我正在寻找一种方法来使焦点/可点击即使在延迟过渡的情况下也能始终如一地工作。
以下情况应与模拟器上的 Android 11 一致:
重新启动应用程序
点击按钮执行过渡
尝试 select 'Edit text 2' 视图,注意它不能被点击(用 Talkback 滑动导航也会绕过它)
为了完整起见,我将在下面添加一个最小的代码示例,但我建议直接从 repo 中签出。
编辑:在我意识到这是由转换引起的之后,修改的示例代码更简单。
activity_main.xml:
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android">
<data>
<variable
name="vm"
type="com.valagroup.demoforaccessibilitybugs.MainViewModel" />
</data>
<androidx.constraintlayout.widget.ConstraintLayout 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:layout_marginStart="12.5dp"
android:layout_marginTop="12.5dp"
android:layout_marginEnd="12.5dp"
android:layout_marginBottom="12.5dp"
tools:context=".MainActivity">
<TextView
android:id="@+id/title_text"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:minWidth="44dp"
android:minHeight="44dp"
android:text="Remember to enable Talkback"
android:textAppearance="@style/TextAppearance.AppCompat.Headline"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<!-- See MainActivity.kt where the transition is started -->
<androidx.appcompat.widget.AppCompatButton
android:id="@+id/toggleButton"
style="@style/Widget.AppCompat.Button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="44dp"
android:text="Transition between text boxes"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@id/title_text" />
<androidx.appcompat.widget.AppCompatEditText
android:id="@+id/editTextByButton1"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="12.5dp"
android:text="Edit text other 1"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintTop_toBottomOf="@id/toggleButton"
app:visible="@{vm.buttonState == true}" />
<androidx.appcompat.widget.AppCompatEditText
android:id="@+id/editTextByButton2"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="12.5dp"
android:text="Edit text other 2"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintTop_toBottomOf="@id/editTextByButton1"
app:visible="@{vm.buttonState == false}" />
<TextView
android:id="@+id/bottom_text2"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="12.5dp"
android:text="Above text boxes are not enabled correctly \nWhen transitions are used with Talkback enabled"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintTop_toBottomOf="@id/editTextByButton2" />
</androidx.constraintlayout.widget.ConstraintLayout>
</layout>
主视图模型.kt
class MainViewModel : ViewModel() {
val buttonState = MutableLiveData<Boolean>().apply { value = true }
fun toggleButtonState() {
buttonState.value = !buttonState.value!!
}
init {
Log.d("MainViewModel", "init")
}
}
MainActivity.kt
class MainActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
val binding: ActivityMainBinding =
DataBindingUtil.setContentView(this, R.layout.activity_main)
binding.vm = ViewModelProvider(this).get(MainViewModel::class.java)
binding.lifecycleOwner = this
binding.toggleButton.setOnClickListener { v ->
if (v != null) {
Log.d("OnClick", "toggleButton")
val viewModel = binding.vm as MainViewModel
viewModel.toggleButtonState()
binding.toggleButton.findParent<ConstraintLayout>()?.let {
// TODO/FIXME: The issue where edit texts are not enabled is caused by the transition
TransitionManager.beginDelayedTransition(it)
}
}
}
}
}
@BindingAdapter("visible")
fun visible(view: View, visible: Boolean?) {
Log.d("BindingAdapter", "Changing visibility of view '${view.id}' to: $visible")
view.visibility = if (visible == true) View.VISIBLE else View.GONE
}
private inline fun <reified T : ViewGroup> View.findParent(): T? {
var view = this.parent
while (view != null) {
if (view is T)
return view
view = view.parent
}
return null
}
AOSP 开发人员回答的解决方法: https://issuetracker.google.com/185532478
我们修复了 S 中与可见性更改相关的错误,我认为这一定是导致此错误的原因,尽管有一些细节我无法完全解释。
问题是当视图变得不可见时我们没有发送可访问性事件,因为我们不允许不可见视图发送事件。 解决方法是让父级发送子树更改事件。
我认为,如果您检查 AccessibilityManager#isEnabled,如果是,则在父级上调用 notifySubtreeAccessibilityStateChanged,您将强制发送正确的事件并提醒服务旧视图已消失。
parent= view.parent
// null check, etc
parent.notifySubtreeAccessibilityStateChanged(parent, parent, AccessibilityEvent#CONTENT_CHANGE_TYPE_SUBTREE)
原则上,您应该只在视图变得不可见时才需要这样做,因为视图变得可见确实会发送一个事件。 但是按照这种推理,我不明白为什么使第二个编辑文本可见并不能解决问题。 无论哪种方式,我认为通知子树更改应该可以解决问题。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.