Android TV 的辅助功能最佳实践

本指南提供了 Android TV 辅助功能的最佳实践,并针对原生和非原生应用提供了建议。

为什么辅助功能对我的电视应用很重要?

视力障碍在电视观众中并不少见。据世界卫生组织 (WHO) 估计,全球有 22 亿人患有视力障碍。根据2018 年全国健康访谈调查,在美国,3200 万 18 岁及以上美国人经历了严重的视力下降。在欧洲,根据欧洲盲人联合会 (EBU) 的数据,估计有3000 万盲人和弱视人士。

最重要的是,视力障碍用户与视力正常的用户一样享受媒体内容。康卡斯特委托进行的一项2017 年调查显示,96% 的盲人或低视力用户定期观看电视,其中 81% 的人每天观看超过一小时。然而,65% 的人还报告在查找电视节目时遇到问题。在2020 年英国的一项调查中,80% 的残疾人表示他们在点播视频流媒体服务中遇到了辅助功能问题。

虽然辅助技术可以帮助低视力用户,但在电视应用的内容发现过程中支持辅助功能也很重要。例如,特别注意提供导航指南和正确标记元素,并确保电视应用能够与 TalkBack 等辅助功能良好配合使用。这些步骤可以显著改善视力障碍用户的体验。

改善辅助功能的第一步是提高认识。本指南可以帮助您和您的团队发现电视应用中的辅助功能问题。

Android 辅助功能资源

要详细了解 Android 上的辅助功能,请参阅我们的辅助功能开发资源

文字缩放

Android TV 应用应通过支持不同的像素密度来尊重用户对文本缩放的偏好。

特别注意

  • 对 UI 组件中的尺寸使用wrap_content
  • 确保布局根据文本比例的变化重新排列组件的尺寸。
  • 确保组件在较大的文本比例下仍适合屏幕。
  • 不要对不灵活的组件使用 sp 文本大小单位。
  • 检查FONT_SCALE的值,以在自定义视图中进行调整

    // Checking font scale with Context
    val scale = resources.configuration.fontScale
    Log.d(TAG, "Text scale is: " + scale)
    

可以使用以下命令更改文本比例

adb shell settings put system font_scale 1.2f

在 Android 12 及更高版本中,用户可以通过设备设置更改文本缩放。

键盘布局

在 Android 13(API 级别 33)及更高版本中,您可以使用 getKeyCodeForKeyLocation()查找预期按键位置的键码。如果用户重新映射了一些按键位置,或者他们使用的是布局不典型的键盘,则可能需要这样做。

音频描述

在 Android 13(API 级别 33)及更高版本中,一个新的系统范围内的辅助功能偏好设置允许用户在所有应用中启用音频描述。Android TV 应用可以通过使用 isAudioDescriptionRequested() 查询用户偏好来检查用户的偏好。

Kotlin

private lateinit var accessibilityManager: AccessibilityManager

// In onCreate():
accessibilityManager = getSystemService(AccessibilityManager::class.java)

// Where your media player is initialized
if (am.isAudioDescriptionRequested) {
    // User has requested to enable audio descriptions
}

Java

private AccessibilityManager accessibilityManager;

// In onCreate():
accessibilityManager = getSystemService(AccessibilityManager.class);

// Where your media player is initialized
if(accessibilityManager.isAudioDescriptionRequested()) {
    // User has requested to enable audio descriptions
}

Android TV 应用可以通过向 AccessibilityManager 添加监听器来监控用户偏好何时发生变化。

Kotlin

private val listener =
    AccessibilityManager.AudioDescriptionRequestedChangeListener { enabled ->
        // Preference changed; reflect its state in your media player
    }

override fun onStart() {
    super.onStart()

    accessibilityManager.addAudioDescriptionRequestedChangeListener(mainExecutor, listener)
}

override fun onStop() {
    super.onStop()

    accessibilityManager.removeAudioDescriptionRequestedChangeListener(listener)
}

Java

private AccessibilityManager.AudioDescriptionRequestedChangeListener listener = enabled -> {
    // Preference changed; reflect its state in your media player
};

@Override
protected void onStart() {
    super.onStart();

    accessibilityManager.addAudioDescriptionRequestedChangeListener(getMainExecutor(), listener);
}

@Override
protected void onStop() {
    super.onStop();

    accessibilityManager.removeAudioDescriptionRequestedChangeListener(listener);
}