谷歌对讲API,以获取Android

时间:2017-05-08 18:45:27

标签: android android-orientation talkback android-accessibility accessibility-api

我的应用程序支持辅助功能,应用程序同时具有纵向和横向模式。

在我的屏幕中,我有一些视图,如按钮,textview,listview和自定义行。

我面临的问题是,当用户将任何项目聚焦在纵向模式并旋转屏幕时,应用程序不会将相同元素聚焦在横向模式中。有人可以建议如何将焦点设置为在纵向模式中选择到横向模式的项目吗?

我甚至对原生应用程序 - 无线网页和“ES文件探索”等现有应用程序进行了一些研究,在这些应用程序中,当用户将方向更改为横向模式时,也无法保持可访问性。系统正在选择景观中的一些随机元素到肖像,反之亦然。

以下是代码段

accessibility_sample.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >

<TextView
    android:id="@+id/name_text"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:contentDescription="Name"
    android:focusableInTouchMode="true"
    android:text="Name" />

<TextView
    android:id="@+id/email_text"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:contentDescription="Email"
    android:focusableInTouchMode="true"
    android:text="Email" />

<Button
    android:id="@+id/sample_button"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:contentDescription="Button description"
    android:text="ButtonText" />

<CheckBox
    android:id="@+id/checkbox"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:contentDescription="Checkbox description"
    android:text="Checkbox Text" />

<ListView
    android:id="@+id/sample_list"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:fadeScrollbars="false" >
</ListView>

</LinearLayout>

sample_list_row.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="horizontal" 
>

<CheckBox
    android:id="@+id/list_row_cb"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content" >
</CheckBox>

<TextView
    android:id="@+id/list_row_name"
    android:layout_width="match_parent"
    android:focusableInTouchMode="true"
    android:focusable="true"
    android:layout_height="wrap_content" />

  </LinearLayout>

AccessibilitySampleActivity.java

公共类AccessibilitySampleActivity扩展了Activity {

private String TAG = AccessibilitySampleActivity.class.getSimpleName();
ListView sampleList;
Button myButton;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.accessibility_sample);
    sampleList = (ListView) findViewById(R.id.sample_list);
    myButton = (Button) findViewById(R.id.sample_button);
    ArrayList<String> countriesList = new ArrayList<String>();
    countriesList.add("India");
    countriesList.add("America");
    countriesList.add("China");
    countriesList.add("Swis");
    countriesList.add("Paries");
    countriesList.add("Pak");
    countriesList.add("Aus");
    countriesList.add("Afg");
    countriesList.add("Nedharnalds");
    countriesList.add("Bangladhesh");
    countriesList.add("Srilanka");
    countriesList.add("France");
    countriesList.add("Japan");
    countriesList.add("SouthAfrica");
    countriesList.add("Iran");
    countriesList.add("Malaysia");
    countriesList.add("Nepal");

    sampleList.setAdapter(new CustomArrayAdapter(this, R.layout.sample_list_row, countriesList));

}

class CustomArrayAdapter extends ArrayAdapter<String> {

    Context context;
    int resource;
    ArrayList<String> countriesList;

    class ViewHolder {
        TextView name;
    }

    public CustomArrayAdapter(Context context, int resource, ArrayList<String> countriesList) {
        super(context, resource, countriesList);
        this.context = context;
        this.resource = resource;
        this.countriesList = countriesList;

    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ViewHolder holder = null;

        if (convertView == null) {
            LayoutInflater vi = (LayoutInflater) getSystemService(Context.LAYOUT_INFLATER_SERVICE);
            convertView = vi.inflate(resource, null);

            holder = new ViewHolder();
            holder.name = (TextView) convertView.findViewById(R.id.list_row_name);
            convertView.setTag(holder);
        } else {
            holder = (ViewHolder) convertView.getTag();
        }
        holder.name.setText(getItem(position));

        return convertView;
    }

    @Override
    public int getCount() {
        return super.getCount();
    }

}

@TargetApi(Build.VERSION_CODES.LOLLIPOP)
public static View findAccessibilityFocus(View view) {
    if (view == null)
        return view;

    if (view.isAccessibilityFocused())
        return view;

    if (view instanceof ViewGroup) {
        ViewGroup viewGroup = (ViewGroup) view;

        for (int i = 0; i < viewGroup.getChildCount(); i++) {
            View childView = viewGroup.getChildAt(i);

            View result = findAccessibilityFocus(childView);

            if (result != null)
                return result;
        }
    }

    return null;
}

@Override
public void onConfigurationChanged(Configuration newConfig) {
    super.onConfigurationChanged(newConfig);
    Log.d(TAG, "onConfigurationChanged");
    AccessibilityManager am = (AccessibilityManager) getSystemService(ACCESSIBILITY_SERVICE);
    boolean isAccessibilityEnabled = am.isEnabled();
    boolean isExploreByTouchEnabled = am.isTouchExplorationEnabled();
    Log.d(TAG, "isAccessibilityEnabled:" + isAccessibilityEnabled + " isExploreByTouchEnabled:"
            + isExploreByTouchEnabled);

    if (isAccessibilityEnabled) {
        View activityView = this.findViewById(android.R.id.content);
        Log.d(TAG, "activityView:" + activityView);
        View selectedView = findAccessibilityFocus(activityView);
        if (selectedView != null) {
            Log.d(TAG, "selectedView:" + selectedView);
            selectedView.sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_FOCUSED);
        }
    }
}

}

对于像TextView,Button,Checkbox这样的普通视图,它可以在用户旋转屏幕时保持状态。

如果用户选择下面的列表视图是我面临的问题

  • 如果在纵向和旋转屏幕中选择了任何视图,则焦点不会保持横向。有时如果所选视图能够在不滚动的情况下显示(如果用户选择印度并且旋转印度将在可见状态下没有滚动)则保持状态
  • 当用户将画面从纵向旋转到横向时,应用滚动并选择任何视图(如日本)并将方向从横向更改为纵向模式,然后我们可以看到它始终选择第一行复选框即印度的复选框< /强>

1 个答案:

答案 0 :(得分:0)

WCAG目前没有指定在动态(当您的应用运行时)方向更改后焦点会发生什么。它确实讨论了方向变化,但没有讨论如何关注这些上下文变化,并且有充分的理由。专注管理不是这里的核心问题!事实上,如果有的话,WCAG会阻止您使用您正在尝试的解决方案,因为它可能被视为干扰了辅助技术的默认操作。考虑一下,在你的问题中,你指出大多数应用程序“不做”你正在尝试做的事情。如果你要完成这件事,世界上每一个其他应用都没有,谁真的无法进入? “预期行为”不是有益吗?

难以接近的部分是令人惊讶的方向改变而不是在完全重新绘制后重点发生的事情!在我看来,这样做的动机可能是对WCAG 2.0 3.2.#下的准则的错误解释。如果用户在关注焦点的位置时出乎意料地导致方向改变......这确实是一个问题!你刚刚解决了问题的错误部分。

焦点控制对于辅助功能非常重要,但方向更改与其他上下文更改不同。在方向更改中,屏幕可能完全不同。当然,屏幕上会有类似的元素,但不能保证焦点元素仍然存在。而不是试图跟踪焦点并将其移动到适当的(可能现在关闭屏幕元素)让AT做它的事情,并且更加选择性地发生方向变化。这里的最佳做法:

  • 切勿将用户锁定在特定方向。请记住,用户可能会将设备安装在他们面前,因此要求定位非常糟糕。 Mobile Guidelines 4.1
  • 然而,话虽这么说,一旦你确定用户(当使用触摸探索的辅助技术是其中之一)处于他们感觉舒适的方向时,锁定方向,并为他们提供另一种手动改变方向的机制。这样,用户可以自由地将手机放在耳朵上,而不必担心意外触发方向改变。 WCAG 2.0 - 3.2.5
  • 如果发生方向更改,请允许默认行为。即使您似乎无法访问,但由于用户不会意外触发方向更改,因此可能并不令人沮丧。如果您已正确解决上述问题,那将不会成为问题。
  • 最后,请记住,当您开发可访问性时,TalkBack只是一种辅助技术,盲人只是该技术用户的一部分。虽然您可能正在改善盲人TalkBack用户的用户体验,但您必须考虑使用Talkback支持其身体残疾的TalkBacks用户,或BrailleBack用户或SwitchAccess用户所做的事情。通常,支持多个AT意味着允许操作系统,AT和用户代理尽可能多地执行默认/支持的操作。虽然添加黑客解决方案可以改善盲人TalkBack用户的体验,但是在TalkBack v#。#上,它可能会让当前的SwitchAccess用户更加糟糕,甚至可以将其打破为流行的TalkBack v#。#+ 1用户。在WCAG 4.1下,WCAG实际上也涵盖了黑客解决方案不好的概念。

总之,从可访问性的角度来看,你要做的事实上有点误导,实际上会让事情变得更糟。话虽如此,在Android视图中跟踪可访问性焦点的概念很有意思,我将在下面为您保留代码。但是,在这种情况下,我建议不要采用这种方法。

原来的ANWER:

这很简单。

讲述对话以集中观点:

View theView = findViewById(R.id.theViewId);
theView.sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_FOCUSED);

就找到当前关注的节点而言,您希望在视图层次结构中搜索关注的视图。

@TargetApi(Build.VERSION_CODES.LOLLIPOP)
public static View findAccessibilityFocus(View view) {
    if (view == null) return view;

    if (view.isAccessibilityFocused()) return view;

    if (view instanceof ViewGroup) {
        ViewGroup viewGroup = (ViewGroup)view;

        for (int i = 0; i < viewGroup.getChildCount(); i++) {
            View childView = viewGroup.getChildAt(i);

            View result = findAccessibilityFocus(childView);

            if (result != null) return result;
        }
    }

    return null;
}

使用此功能时必须小心!当涉及竞争条件时,可访问性焦点可能很难找到。视图不必具有可访问性集中的视图。将可访问性委托附加到根视图并跟踪具有可访问性焦点的最后一个元素可能更容易,跟踪相应的可访问性事件。