android - Android Talkback 在 TextInputLayout 上的错误公告
问题描述
我的布局中有以下代码
<com.google.android.material.textfield.TextInputLayout
android:id="@+id/tilPassword"
style="@style/Widget.MaterialComponents.TextInputLayout.OutlinedBox"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:hint="Password"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@id/tilUserName"
app:passwordToggleEnabled="true">
<com.google.android.material.textfield.TextInputEditText
android:id="@+id/tiePassword"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:imeOptions="actionDone"
android:inputType="textPassword"
android:selectAllOnFocus="true"
android:singleLine="true" />
</com.google.android.material.textfield.TextInputLayout>
导航到 TextInputLayout 时 TalkBack 宣布:“密码密码编辑框”
期望的公告:“密码编辑框”
如果我删除任何一个,android:hint="Password"
或者 android:inputType="textPassword"
它按预期工作。
关于设置提示的注意事项
提示应设置在 TextInputLayout,而不是 TextInputEditText 或 EditText。如果在 XML 中的子 EditText 上指定了提示,TextInputLayout 可能仍然可以正常工作;TextInputLayout 将使用 EditText 的提示作为其浮动标签。但是,将来修改提示的调用不会更新 TextInputLayout 的提示。为避免意外行为,请在 TextInputLayout 上而不是在 EditText 上调用 setHint() 和 getHint()。
解决方案
通过检查源代码:
文本输入布局类
可以发现:
TextInputLayout 辅助功能信息通过以下公共类提供:
public static class AccessibilityDelegate extends AccessibilityDelegateCompat {
private final TextInputLayout layout;
public AccessibilityDelegate(TextInputLayout layout) {
this.layout = layout;
}
@Override
public void onInitializeAccessibilityNodeInfo(View host, AccessibilityNodeInfoCompat info) {
super.onInitializeAccessibilityNodeInfo(host, info);
EditText editText = layout.getEditText();
CharSequence text = (editText != null) ? editText.getText() : null;
CharSequence hintText = layout.getHint();
CharSequence errorText = layout.getError();
CharSequence counterDesc = layout.getCounterOverflowDescription();
boolean showingText = !TextUtils.isEmpty(text);
boolean hasHint = !TextUtils.isEmpty(hintText);
boolean showingError = !TextUtils.isEmpty(errorText);
boolean contentInvalid = showingError || !TextUtils.isEmpty(counterDesc);
if (showingText) {
info.setText(text);
} else if (hasHint) {
info.setText(hintText);
}
if (hasHint) {
info.setHintText(hintText);
info.setShowingHintText(!showingText && hasHint);
}
if (contentInvalid) {
info.setError(showingError ? errorText : counterDesc);
info.setContentInvalid(true);
}
}
}
并通过调用以下公共方法将其应用于 TextInputLayout:
public void setTextInputAccessibilityDelegate(TextInputLayout.AccessibilityDelegate delegate) {
if (editText != null) {
ViewCompat.setAccessibilityDelegate(editText, delegate);
}
}
因此,可以扩展:
TextInputLayout.AccessibilityDelegate 类并覆盖 onInitializeAccessibilityNodeInfo() 以仅宣布需要的内容。例如,在您的情况下,您可以执行以下操作:
private class CustomTextInputLayoutAccessibilityDelegate extends TextInputLayout.AccessibilityDelegate{
private final TextInputLayout layout;
public CustomTextInputLayoutAccessibilityDelegate(TextInputLayout layout) {
super(layout);
this.layout = layout;
}
@Override
public void onInitializeAccessibilityNodeInfo(View host, AccessibilityNodeInfoCompat info) {
super.onInitializeAccessibilityNodeInfo(host, info);
EditText editText = layout.getEditText();
CharSequence text = (editText != null) ? editText.getText() : null;
CharSequence hintText = layout.getHint();
CharSequence errorText = layout.getError();
//CharSequence counterDesc = layout.getCounterOverflowDescription();
boolean showingText = !TextUtils.isEmpty(text);
boolean hasHint = !TextUtils.isEmpty(hintText);
//boolean showingError = !TextUtils.isEmpty(errorText);
//boolean contentInvalid = showingError || !TextUtils.isEmpty(counterDesc);
if (showingText) {
info.setText(text);
} else if (hasHint) {
info.setText("");
}
if (hasHint) {
info.setHintText("");
info.setShowingHintText(!showingText && hasHint);
}
//if (contentInvalid) {
// info.setError(showingError ? errorText : counterDesc);
// info.setContentInvalid(true);
//}
}
}
然后调用:
tilPassword.setTextInputAccessibilityDelegate(new CustomTextInputLayoutAccessibilityDelegate(tilPassword));
推荐阅读
- javascript - React / TypeScript - 类型“{ index:number}”缺少类型“IList”的以下属性:length、pop、push、concat 和另外 26 个
- vue.js - 如何从 vuetify 数据表槽中消除和理解这个错误警告?
- flutter - 是否可以简化我对飞镖的逻辑?
- javascript - 如何使用过滤器方法删除特定数字?
- pine-script - “输入'行尾没有行继续'的语法错误。”
- python - SqlAlchemy RuntimeError:在迭代期间双端队列突变
- flutter - 如何检测用户何时在flutter inappwebview中导航到新屏幕
- c# - 如何使用 C# 分割多层 TIFF 图像?
- unity3d - 在 Unity 编辑器中查看或设置场景相机
- sql-server - SQL Server - 通过存储过程中的一些相等字段获取