繁体   English   中英

在 Android 中启用对讲时,视图在变为可见并延迟转换后不可选择

[英]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 一致:

  1. 重新启动应用程序

  2. 点击按钮执行过渡

  3. 尝试 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.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM