Accessibility for message header

Thanks @pvagner
pull/170/head
M66B 5 years ago
parent dc5df7b6a2
commit 15bf4907ae

@ -78,6 +78,7 @@ import android.view.ViewAnimationUtils;
import android.view.ViewConfiguration; import android.view.ViewConfiguration;
import android.view.ViewGroup; import android.view.ViewGroup;
import android.view.ViewStub; import android.view.ViewStub;
import android.view.accessibility.AccessibilityManager;
import android.view.accessibility.AccessibilityNodeInfo; import android.view.accessibility.AccessibilityNodeInfo;
import android.view.animation.AccelerateDecelerateInterpolator; import android.view.animation.AccelerateDecelerateInterpolator;
import android.webkit.WebView; import android.webkit.WebView;
@ -174,6 +175,7 @@ public class AdapterMessage extends RecyclerView.Adapter<AdapterMessage.ViewHold
private Context context; private Context context;
private LifecycleOwner owner; private LifecycleOwner owner;
private LayoutInflater inflater; private LayoutInflater inflater;
private boolean accessibility;
private boolean suitable; private boolean suitable;
private boolean unmetered; private boolean unmetered;
@ -422,7 +424,15 @@ public class AdapterMessage extends RecyclerView.Adapter<AdapterMessage.ViewHold
tvSubject.setEllipsize(TextUtils.TruncateAt.MIDDLE); tvSubject.setEllipsize(TextUtils.TruncateAt.MIDDLE);
} }
if (view != null) // Accessibility
int vt = getItemViewType();
if (vt != R.layout.item_message_compact && vt != R.layout.item_message_normal)
return;
if (!BuildConfig.DEBUG && !accessibility)
return;
view.setAccessibilityDelegate(new View.AccessibilityDelegate() { view.setAccessibilityDelegate(new View.AccessibilityDelegate() {
@Override @Override
public void onInitializeAccessibilityNodeInfo(View host, AccessibilityNodeInfo info) { public void onInitializeAccessibilityNodeInfo(View host, AccessibilityNodeInfo info) {
@ -432,42 +442,103 @@ public class AdapterMessage extends RecyclerView.Adapter<AdapterMessage.ViewHold
if (message == null) if (message == null)
return; return;
if (ibExpander != null && ibExpander.getVisibility() == View.VISIBLE) {
boolean expanded = properties.getValue("expanded", message.id); boolean expanded = properties.getValue("expanded", message.id);
List<String> result = new ArrayList<>();
vwColor.setImportantForAccessibility(View.IMPORTANT_FOR_ACCESSIBILITY_NO);
if (selectionTracker != null && selectionTracker.isSelected(message.id))
result.add(context.getString(R.string.title_accessibility_selected));
if (message.count == 1)
result.add(context.getString(message.unseen > 0 ? R.string.title_accessibility_unseen : R.string.title_accessibility_seen));
else if (message.unseen == 0)
result.add(context.getResources().getQuantityString(
R.plurals.title_accessibility_all_of_seen, message.count, message.count));
else
result.add(context.getResources().getQuantityString(
R.plurals.title_accessibility_count_of_seen, message.unseen, message.unseen, message.count));
if (ibExpander.getVisibility() == View.VISIBLE) {
result.add(context.getString(expanded ? R.string.title_accessibility_expanded : R.string.title_accessibility_collapsed));
info.addAction(new AccessibilityNodeInfo.AccessibilityAction(R.id.ibExpander, info.addAction(new AccessibilityNodeInfo.AccessibilityAction(R.id.ibExpander,
context.getString(expanded ? R.string.title_accessibility_collapse : R.string.title_accessibility_expand))); context.getString(expanded ? R.string.title_accessibility_collapse : R.string.title_accessibility_expand)));
} }
ibExpander.setImportantForAccessibility(View.IMPORTANT_FOR_ACCESSIBILITY_NO);
if (ibAvatar != null && ibAvatar.getVisibility() == View.VISIBLE && ibAvatar.isEnabled()) if (ibAvatar.getVisibility() == View.VISIBLE && ibAvatar.isEnabled())
info.addAction(new AccessibilityNodeInfo.AccessibilityAction(R.id.ibAvatar, info.addAction(new AccessibilityNodeInfo.AccessibilityAction(R.id.ibAvatar,
context.getString(R.string.title_accessibility_view_contact))); context.getString(R.string.title_accessibility_view_contact)));
ibAvatar.setImportantForAccessibility(View.IMPORTANT_FOR_ACCESSIBILITY_NO);
if (message.drafts > 0)
result.add(context.getString(R.string.title_legend_draft));
if (message.ui_answered)
result.add(context.getString(R.string.title_accessibility_answered));
if (ibFlagged != null && ibFlagged.getVisibility() == View.VISIBLE && ibFlagged.isEnabled()) { if (ibFlagged.getVisibility() == View.VISIBLE && ibFlagged.isEnabled()) {
int flagged = (message.count - message.unflagged); int flagged = (message.count - message.unflagged);
if (flagged > 0)
result.add(context.getString(R.string.title_accessibility_flagged));
info.addAction(new AccessibilityNodeInfo.AccessibilityAction(R.id.ibFlagged, info.addAction(new AccessibilityNodeInfo.AccessibilityAction(R.id.ibFlagged,
context.getString(flagged > 0 ? R.string.title_unflag : R.string.title_flag))); context.getString(flagged > 0 ? R.string.title_unflag : R.string.title_flag)));
} }
} ibFlagged.setImportantForAccessibility(View.IMPORTANT_FOR_ACCESSIBILITY_NO);
@Override if (EntityMessage.PRIORITIY_HIGH.equals(message.priority))
public boolean performAccessibilityAction(View host, int action, Bundle args) { result.add(context.getString(R.string.title_legend_priority));
TupleMessageEx message = getMessage(); if (EntityMessage.PRIORITIY_LOW.equals(message.priority))
if (message == null) result.add(context.getString(R.string.title_legend_priority_low));
return false; if (message.attachments > 0)
result.add(context.getString(R.string.title_legend_attachment));
switch (action) { // For a11y purpose report addresses first in case of incoming message
case R.id.ibExpander: boolean outgoing = isOutgoing(message);
onToggleMessage(message); if (!outgoing || message.count > 1)
return true; result.add(tvFrom.getText().toString());
case R.id.ibAvatar: else
onViewContact(message); result.add(message.subject); // Don't want to ellipsize for a11y
return true; result.add(tvTime.getText().toString());
case R.id.ibFlagged: if (outgoing && message.count == 1)
onToggleFlag(message); result.add(tvFrom.getText().toString());
return true; else
default: result.add(message.subject);
return super.performAccessibilityAction(host, action, args);
if (message.encrypted > 0)
result.add(context.getString(R.string.title_legend_encrypted));
if (message.signed > 0)
result.add(context.getString(R.string.title_legend_signed));
if (ibAuth.getVisibility() == View.VISIBLE) {
result.add(context.getString(R.string.title_legend_auth));
info.addAction(new AccessibilityNodeInfo.AccessibilityAction(R.id.ibAuth,
context.getString(R.string.title_accessibility_show_authentication_result)));
} }
ibAuth.setImportantForAccessibility(View.IMPORTANT_FOR_ACCESSIBILITY_NO);
if (ibSnoozed.getVisibility() == View.VISIBLE) {
result.add(context.getString(R.string.title_legend_snoozed));
info.addAction(new AccessibilityNodeInfo.AccessibilityAction(R.id.ibSnoozed,
context.getString(R.string.title_accessibility_show_snooze_time)));
}
ibSnoozed.setImportantForAccessibility(View.IMPORTANT_FOR_ACCESSIBILITY_NO);
if (tvFolder.getVisibility() == View.VISIBLE)
result.add(tvFolder.getText().toString());
if (tvSize.getVisibility() == View.VISIBLE)
result.add(tvSize.getText().toString());
if (tvError.getVisibility() == View.VISIBLE)
result.add(tvError.getText().toString());
if (tvPreview.getVisibility() == View.VISIBLE)
result.add(tvPreview.getText().toString());
if (ibHelp.getVisibility() == View.VISIBLE)
info.addAction(new AccessibilityNodeInfo.AccessibilityAction(R.id.ibHelp,
context.getString(R.string.title_accessibility_view_help)));
ibHelp.setImportantForAccessibility(View.IMPORTANT_FOR_ACCESSIBILITY_NO);
info.setContentDescription(TextUtils.join(", ", result));
} }
}); });
} }
@ -905,18 +976,6 @@ public class AdapterMessage extends RecyclerView.Adapter<AdapterMessage.ViewHold
// Expand warning // Expand warning
bindExpandWarning(message, expanded); bindExpandWarning(message, expanded);
// Accessibility
header.setContentDescription(context.getString(
R.string.title_accessibility_header,
position + 1,
tvFrom.getText(),
tvTime.getText(),
tvSubject.getText(),
context.getString(message.unseen > 0 ? R.string.title_accessibility_unseen : R.string.title_accessibility_seen),
message.count - message.unflagged > 0 ? context.getString(R.string.title_accessibility_flagged) : "",
message.count
));
// Message text preview // Message text preview
int textColor = (contrast ? textColorPrimary : textColorSecondary); int textColor = (contrast ? textColorPrimary : textColorSecondary);
if (tvPreview.getTag() == null || (int) tvPreview.getTag() != textColor) { if (tvPreview.getTag() == null || (int) tvPreview.getTag() != textColor) {
@ -3744,6 +3803,10 @@ public class AdapterMessage extends RecyclerView.Adapter<AdapterMessage.ViewHold
this.context = parentFragment.getContext(); this.context = parentFragment.getContext();
this.owner = parentFragment.getViewLifecycleOwner(); this.owner = parentFragment.getViewLifecycleOwner();
this.inflater = LayoutInflater.from(context); this.inflater = LayoutInflater.from(context);
AccessibilityManager am = (AccessibilityManager) context.getSystemService(Context.ACCESSIBILITY_SERVICE);
this.accessibility = (am != null && am.isEnabled());
this.TF = Helper.getTimeInstance(context, SimpleDateFormat.SHORT); this.TF = Helper.getTimeInstance(context, SimpleDateFormat.SHORT);
this.DTF = Helper.getDateTimeInstance(context, SimpleDateFormat.LONG, SimpleDateFormat.LONG); this.DTF = Helper.getDateTimeInstance(context, SimpleDateFormat.LONG, SimpleDateFormat.LONG);

@ -1020,16 +1020,29 @@
<string name="title_widget_unseen">Unread messages only</string> <string name="title_widget_unseen">Unread messages only</string>
<string name="title_widget_flagged">Starred messages only</string> <string name="title_widget_flagged">Starred messages only</string>
<string name="title_accessibility_header" translatable="false">Conversation %1$d: %2$s at %3$s %4$s %5$s %6$s %7$d messages</string>
<string name="title_accessibility_collapsed">Collapsed</string> <string name="title_accessibility_collapsed">Collapsed</string>
<string name="title_accessibility_expanded">Expanded</string> <string name="title_accessibility_expanded">Expanded</string>
<string name="title_accessibility_seen">Read</string> <string name="title_accessibility_seen">Read</string>
<string name="title_accessibility_unseen">Unread</string> <string name="title_accessibility_unseen">Unread</string>
<plurals name="title_accessibility_count_of_seen">
<item quantity="one"></item>
<item quantity="other">%1$d of %2$d read</item>
</plurals>
<plurals name="title_accessibility_all_of_seen">
<item quantity="one"></item>
<item quantity="other">All of %1$d read</item>
</plurals>
<string name="title_accessibility_flagged">Starred</string> <string name="title_accessibility_flagged">Starred</string>
<string name="title_accessibility_unflagged">Unstarred</string> <string name="title_accessibility_unflagged">Unstarred</string>
<string name="title_accessibility_collapse">Collapse</string> <string name="title_accessibility_collapse">Collapse</string>
<string name="title_accessibility_expand">Expand</string> <string name="title_accessibility_expand">Expand</string>
<string name="title_accessibility_view_contact">View contact</string> <string name="title_accessibility_view_contact">View contact</string>
<string name="title_accessibility_show_authentication_result">Show authentication result</string>
<string name="title_accessibility_show_snooze_time">Show snooze time</string>
<string name="title_accessibility_view_help">View help</string>
<string name="title_accessibility_selected">Selected</string>
<string name="title_accessibility_answered">Replied</string>
<string-array name="pollIntervalNames"> <string-array name="pollIntervalNames">
<item>Always</item> <item>Always</item>

Loading…
Cancel
Save