am eb178890: Merge " Adding a code sample for implementing accessibility in a custom view." into ics-mr1
* commit 'eb1788903298fc12ac800fe8fcfc6b7adec7b258': Adding a code sample for implementing accessibility in a custom view.
This commit is contained in:
@@ -542,7 +542,7 @@
|
|||||||
</activity>
|
</activity>
|
||||||
|
|
||||||
<!-- ============================ -->
|
<!-- ============================ -->
|
||||||
<!-- Accessibility examples strings -->
|
<!-- Accessibility examples -->
|
||||||
<!-- ============================ -->
|
<!-- ============================ -->
|
||||||
|
|
||||||
<activity android:name=".accessibility.ClockBackActivity"
|
<activity android:name=".accessibility.ClockBackActivity"
|
||||||
@@ -580,6 +580,15 @@
|
|||||||
android:resource="@xml/taskbackconfig" />
|
android:resource="@xml/taskbackconfig" />
|
||||||
</service>
|
</service>
|
||||||
|
|
||||||
|
<activity android:name=".accessibility.CustomViewAccessibilityActivity"
|
||||||
|
android:label="@string/accessibility_custom_view"
|
||||||
|
android:enabled="@bool/atLeastIceCreamSandwich">
|
||||||
|
<intent-filter>
|
||||||
|
<action android:name="android.intent.action.MAIN" />
|
||||||
|
<category android:name="android.intent.category.SAMPLE_CODE" />
|
||||||
|
</intent-filter>
|
||||||
|
</activity>
|
||||||
|
|
||||||
<!-- Instrumentation Samples -->
|
<!-- Instrumentation Samples -->
|
||||||
|
|
||||||
<activity android:name=".app.LocalSample" android:label="@string/activity_local_sample">
|
<activity android:name=".app.LocalSample" android:label="@string/activity_local_sample">
|
||||||
|
|||||||
@@ -39,6 +39,10 @@ support</a></li>
|
|||||||
<li><a
|
<li><a
|
||||||
href="src/com/example/android/apis/accessibility/TaskBackService.html">Window
|
href="src/com/example/android/apis/accessibility/TaskBackService.html">Window
|
||||||
Querying Accessibility Service</a></li>
|
Querying Accessibility Service</a></li>
|
||||||
|
<li><a
|
||||||
|
href="src/com/example/android/apis/accessibility/CustomViewAccessibilityActivity.html">Custom View
|
||||||
|
Accessibility</a></li>
|
||||||
|
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|||||||
44
samples/ApiDemos/res/layout/custom_view_accessibility.xml
Normal file
44
samples/ApiDemos/res/layout/custom_view_accessibility.xml
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<!-- Copyright (C) 2011 The Android Open Source Project
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
-->
|
||||||
|
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:orientation="vertical">
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_marginBottom="50dip"
|
||||||
|
android:text="@string/accessibility_custom_view_instructions">
|
||||||
|
</TextView>
|
||||||
|
|
||||||
|
<view
|
||||||
|
class="com.example.android.apis.accessibility.CustomViewAccessibilityActivity$AccessibleCompoundButtonInheritance"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:padding="20dip"
|
||||||
|
android:clickable="true" >
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<view
|
||||||
|
class="com.example.android.apis.accessibility.CustomViewAccessibilityActivity$AccessibleCompoundButtonComposition"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:padding="20dip"
|
||||||
|
android:clickable="true" >
|
||||||
|
</view>
|
||||||
|
|
||||||
|
</LinearLayout>
|
||||||
@@ -1296,7 +1296,7 @@
|
|||||||
<string name="accessibility_service_label">ClockBack</string>
|
<string name="accessibility_service_label">ClockBack</string>
|
||||||
<string name="accessibility_service_instructions">
|
<string name="accessibility_service_instructions">
|
||||||
1. Enable TalkBack (Settings -> Accessibility -> TalkBack).
|
1. Enable TalkBack (Settings -> Accessibility -> TalkBack).
|
||||||
\n\n2. Enable Explore-byTouch (Settings -> Accessibility -> Explore by Touch).
|
\n\n2. Enable Explore-by-Touch (Settings -> Accessibility -> Explore by Touch).
|
||||||
\n\n3. Touch explore the Clock application and the home screen.
|
\n\n3. Touch explore the Clock application and the home screen.
|
||||||
\n\n4. Go to the Clock application and change the time of an alarm.
|
\n\n4. Go to the Clock application and change the time of an alarm.
|
||||||
\n\n5. Enable ClockBack (Settings -> Accessibility -> ClockBack).
|
\n\n5. Enable ClockBack (Settings -> Accessibility -> ClockBack).
|
||||||
@@ -1316,10 +1316,19 @@
|
|||||||
<string name="accessibility_query_window_description">Task App Accessibility Service</string>
|
<string name="accessibility_query_window_description">Task App Accessibility Service</string>
|
||||||
<string name="accessibility_query_window_instructions">
|
<string name="accessibility_query_window_instructions">
|
||||||
1. Enable QueryBack (Settings -> Accessibility -> QueryBack).
|
1. Enable QueryBack (Settings -> Accessibility -> QueryBack).
|
||||||
\n\n2. Enable Explore-byTouch (Settings -> Accessibility -> Explore by Touch).
|
\n\n2. Enable Explore-by-Touch (Settings -> Accessibility -> Explore by Touch).
|
||||||
\n\n3. Touch explore the list.
|
\n\n3. Touch explore the list.
|
||||||
</string>
|
</string>
|
||||||
|
|
||||||
|
<string name="accessibility_custom_view">Accessibility/Custom View</string>
|
||||||
|
<string name="accessibility_custom_view_instructions">
|
||||||
|
1. Enable TalkBack (Settings -> Accessibility -> TalkBack).
|
||||||
|
\n\n2. Enable Explore-by-Touch (Settings -> Accessibility -> Explore by Touch).
|
||||||
|
\n\n3. Touch explore/poke the buttons.
|
||||||
|
</string>
|
||||||
|
<string name="accessibility_custom_on">On</string>
|
||||||
|
<string name="accessibility_custom_off">Off</string>
|
||||||
|
|
||||||
<string name="task_name">Task</string>
|
<string name="task_name">Task</string>
|
||||||
<string name="task_complete_template">Task %1$s %2$s</string>
|
<string name="task_complete_template">Task %1$s %2$s</string>
|
||||||
<string name="task_complete">is complete</string>
|
<string name="task_complete">is complete</string>
|
||||||
|
|||||||
@@ -0,0 +1,280 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2011 The Android Open Source Project
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.example.android.apis.accessibility;
|
||||||
|
|
||||||
|
import android.app.Activity;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.graphics.Canvas;
|
||||||
|
import android.graphics.Paint;
|
||||||
|
import android.os.Build;
|
||||||
|
import android.os.Bundle;
|
||||||
|
import android.text.Layout;
|
||||||
|
import android.text.StaticLayout;
|
||||||
|
import android.text.TextPaint;
|
||||||
|
import android.text.TextUtils;
|
||||||
|
import android.util.AttributeSet;
|
||||||
|
import android.util.TypedValue;
|
||||||
|
import android.view.View;
|
||||||
|
import android.view.accessibility.AccessibilityEvent;
|
||||||
|
import android.view.accessibility.AccessibilityNodeInfo;
|
||||||
|
|
||||||
|
import com.example.android.apis.R;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Demonstrates how to implement accessibility support of custom views. Custom view
|
||||||
|
* is a tailored widget developed by extending the base classes in the android.view
|
||||||
|
* package. This sample shows how to implement the accessibility behavior via both
|
||||||
|
* inheritance (non backwards compatible) and composition (backwards compatible).
|
||||||
|
* <p>
|
||||||
|
* While the Android framework has a diverse portfolio of views tailored for various
|
||||||
|
* use cases, sometimes a developer needs a specific functionality not implemented
|
||||||
|
* by the standard views. A solution is to write a custom view that extends one the
|
||||||
|
* base view classes. While implementing the desired functionality a developer should
|
||||||
|
* also implement accessibility support for that new functionality such that
|
||||||
|
* disabled users can leverage it.
|
||||||
|
* </p>
|
||||||
|
*/
|
||||||
|
public class CustomViewAccessibilityActivity extends Activity {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onCreate(Bundle savedInstanceState) {
|
||||||
|
super.onCreate(savedInstanceState);
|
||||||
|
setContentView(R.layout.custom_view_accessibility);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Demonstrates how to enhance the accessibility support via inheritance.
|
||||||
|
* <p>
|
||||||
|
* <strong>Note:</strong> Using inheritance may break your application's
|
||||||
|
* backwards compatibility. In particular, overriding a method that takes as
|
||||||
|
* an argument or returns a class not present on an older platform
|
||||||
|
* version will prevent your application from running on that platform.
|
||||||
|
* For example, {@link AccessibilityNodeInfo} was introduced in
|
||||||
|
* {@link Build.VERSION_CODES#ICE_CREAM_SANDWICH API 14}, thus overriding
|
||||||
|
* {@link View#onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo)
|
||||||
|
* View.onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo)}
|
||||||
|
* will prevent you application from running on a platform older than
|
||||||
|
* {@link Build.VERSION_CODES#ICE_CREAM_SANDWICH API 14}.
|
||||||
|
* </p>
|
||||||
|
*/
|
||||||
|
public static class AccessibleCompoundButtonInheritance extends BaseToggleButton {
|
||||||
|
|
||||||
|
public AccessibleCompoundButtonInheritance(Context context, AttributeSet attrs) {
|
||||||
|
super(context, attrs);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onInitializeAccessibilityEvent(AccessibilityEvent event) {
|
||||||
|
super.onInitializeAccessibilityEvent(event);
|
||||||
|
// We called the super implementation to let super classes
|
||||||
|
// set appropriate event properties. Then we add the new property
|
||||||
|
// (checked) which is not supported by a super class.
|
||||||
|
event.setChecked(isChecked());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) {
|
||||||
|
super.onInitializeAccessibilityNodeInfo(info);
|
||||||
|
// We called the super implementation to let super classes set
|
||||||
|
// appropriate info properties. Then we add our properties
|
||||||
|
// (checkable and checked) which are not supported by a super class.
|
||||||
|
info.setCheckable(true);
|
||||||
|
info.setChecked(isChecked());
|
||||||
|
// Very often you will need to add only the text on the custom view.
|
||||||
|
CharSequence text = getText();
|
||||||
|
if (!TextUtils.isEmpty(text)) {
|
||||||
|
info.setText(text);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onPopulateAccessibilityEvent(AccessibilityEvent event) {
|
||||||
|
super.onPopulateAccessibilityEvent(event);
|
||||||
|
// We called the super implementation to populate its text to the
|
||||||
|
// event. Then we add our text not present in a super class.
|
||||||
|
// Very often you will need to add only the text on the custom view.
|
||||||
|
CharSequence text = getText();
|
||||||
|
if (!TextUtils.isEmpty(text)) {
|
||||||
|
event.getText().add(text);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Demonstrates how to enhance the accessibility support via composition.
|
||||||
|
* <p>
|
||||||
|
* <strong>Note:</strong> Using composition ensures that your application is
|
||||||
|
* backwards compatible. The android-support-v4 library has API that allow
|
||||||
|
* using the accessibility APIs in a backwards compatible manner.
|
||||||
|
* </p>
|
||||||
|
*/
|
||||||
|
public static class AccessibleCompoundButtonComposition extends BaseToggleButton {
|
||||||
|
|
||||||
|
public AccessibleCompoundButtonComposition(Context context, AttributeSet attrs) {
|
||||||
|
super(context, attrs);
|
||||||
|
tryInstallAccessibilityDelegate();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void tryInstallAccessibilityDelegate() {
|
||||||
|
// If the API version of the platform we are running is too old
|
||||||
|
// and does not support the AccessibilityDelegate APIs, do not
|
||||||
|
// call View.setAccessibilityDelegate(AccessibilityDelegate) or
|
||||||
|
// refer to AccessibilityDelegate, otherwise an exception will
|
||||||
|
// be thrown.
|
||||||
|
// NOTE: The android-support-v4 library contains APIs the enable
|
||||||
|
// using the accessibility APIs in a backwards compatible fashion.
|
||||||
|
if (Build.VERSION.SDK_INT < 14) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// AccessibilityDelegate allows clients to override its methods that
|
||||||
|
// correspond to the accessibility methods in View and register the
|
||||||
|
// delegate in the View essentially injecting the accessibility support.
|
||||||
|
setAccessibilityDelegate(new AccessibilityDelegate() {
|
||||||
|
@Override
|
||||||
|
public void onInitializeAccessibilityEvent(View host, AccessibilityEvent event) {
|
||||||
|
super.onInitializeAccessibilityEvent(host, event);
|
||||||
|
// We called the super implementation to let super classes
|
||||||
|
// set appropriate event properties. Then we add the new property
|
||||||
|
// (checked) which is not supported by a super class.
|
||||||
|
event.setChecked(isChecked());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onInitializeAccessibilityNodeInfo(View host,
|
||||||
|
AccessibilityNodeInfo info) {
|
||||||
|
super.onInitializeAccessibilityNodeInfo(host, info);
|
||||||
|
// We called the super implementation to let super classes set
|
||||||
|
// appropriate info properties. Then we add our properties
|
||||||
|
// (checkable and checked) which are not supported by a super class.
|
||||||
|
info.setCheckable(true);
|
||||||
|
info.setChecked(isChecked());
|
||||||
|
// Very often you will need to add only the text on the custom view.
|
||||||
|
CharSequence text = getText();
|
||||||
|
if (!TextUtils.isEmpty(text)) {
|
||||||
|
info.setText(text);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onPopulateAccessibilityEvent(View host, AccessibilityEvent event) {
|
||||||
|
super.onPopulateAccessibilityEvent(host, event);
|
||||||
|
// We called the super implementation to populate its text to the
|
||||||
|
// event. Then we add our text not present in a super class.
|
||||||
|
// Very often you will need to add only the text on the custom view.
|
||||||
|
CharSequence text = getText();
|
||||||
|
if (!TextUtils.isEmpty(text)) {
|
||||||
|
event.getText().add(text);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is a base toggle button class whose accessibility is not tailored
|
||||||
|
* to reflect the new functionality it implements.
|
||||||
|
* <p>
|
||||||
|
* <strong>Note:</strong> This is not a sample implementation of a toggle
|
||||||
|
* button, rather a simple class needed to demonstrate how to refine the
|
||||||
|
* accessibility support of a custom View.
|
||||||
|
* </p>
|
||||||
|
*/
|
||||||
|
private static class BaseToggleButton extends View {
|
||||||
|
private boolean mChecked;
|
||||||
|
|
||||||
|
private CharSequence mTextOn;
|
||||||
|
private CharSequence mTextOff;
|
||||||
|
|
||||||
|
private Layout mOnLayout;
|
||||||
|
private Layout mOffLayout;
|
||||||
|
|
||||||
|
private TextPaint mTextPaint;
|
||||||
|
|
||||||
|
public BaseToggleButton(Context context, AttributeSet attrs) {
|
||||||
|
this(context, attrs, android.R.attr.buttonStyle);
|
||||||
|
}
|
||||||
|
|
||||||
|
public BaseToggleButton(Context context, AttributeSet attrs, int defStyle) {
|
||||||
|
super(context, attrs, defStyle);
|
||||||
|
|
||||||
|
mTextPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG);
|
||||||
|
|
||||||
|
TypedValue typedValue = new TypedValue();
|
||||||
|
context.getTheme().resolveAttribute(android.R.attr.textSize, typedValue, true);
|
||||||
|
final int textSize = (int) typedValue.getDimension(
|
||||||
|
context.getResources().getDisplayMetrics());
|
||||||
|
mTextPaint.setTextSize(textSize);
|
||||||
|
|
||||||
|
context.getTheme().resolveAttribute(android.R.attr.textColorPrimary, typedValue, true);
|
||||||
|
final int textColor = context.getResources().getColor(typedValue.resourceId);
|
||||||
|
mTextPaint.setColor(textColor);
|
||||||
|
|
||||||
|
mTextOn = context.getString(R.string.accessibility_custom_on);
|
||||||
|
mTextOff = context.getString(R.string.accessibility_custom_off);
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isChecked() {
|
||||||
|
return mChecked;
|
||||||
|
}
|
||||||
|
|
||||||
|
public CharSequence getText() {
|
||||||
|
return mChecked ? mTextOn : mTextOff;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean performClick() {
|
||||||
|
final boolean handled = super.performClick();
|
||||||
|
if (!handled) {
|
||||||
|
mChecked ^= true;
|
||||||
|
invalidate();
|
||||||
|
}
|
||||||
|
return handled;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
|
||||||
|
if (mOnLayout == null) {
|
||||||
|
mOnLayout = makeLayout(mTextOn);
|
||||||
|
}
|
||||||
|
if (mOffLayout == null) {
|
||||||
|
mOffLayout = makeLayout(mTextOff);
|
||||||
|
}
|
||||||
|
final int minWidth = Math.max(mOnLayout.getWidth(), mOffLayout.getWidth())
|
||||||
|
+ getPaddingLeft() + getPaddingRight();
|
||||||
|
final int minHeight = Math.max(mOnLayout.getHeight(), mOffLayout.getHeight())
|
||||||
|
+ getPaddingLeft() + getPaddingRight();
|
||||||
|
setMeasuredDimension(resolveSizeAndState(minWidth, widthMeasureSpec, 0),
|
||||||
|
resolveSizeAndState(minHeight, heightMeasureSpec, 0));
|
||||||
|
}
|
||||||
|
|
||||||
|
private Layout makeLayout(CharSequence text) {
|
||||||
|
return new StaticLayout(text, mTextPaint,
|
||||||
|
(int) Math.ceil(Layout.getDesiredWidth(text, mTextPaint)),
|
||||||
|
Layout.Alignment.ALIGN_NORMAL, 1.f, 0, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onDraw(Canvas canvas) {
|
||||||
|
super.onDraw(canvas);
|
||||||
|
canvas.save();
|
||||||
|
canvas.translate(getPaddingLeft(), getPaddingRight());
|
||||||
|
Layout switchText = mChecked ? mOnLayout : mOffLayout;
|
||||||
|
switchText.draw(canvas);
|
||||||
|
canvas.restore();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -9,7 +9,7 @@
|
|||||||
how to provide dynamic, context-dependent feedback — feedback type changes depending on
|
how to provide dynamic, context-dependent feedback — feedback type changes depending on
|
||||||
the ringer mode.
|
the ringer mode.
|
||||||
</dd>
|
</dd>
|
||||||
<dt>
|
</dl>
|
||||||
|
|
||||||
<dl>
|
<dl>
|
||||||
<dt><a href="TaskBackService.html">Window Querying Accessibility Service</a></dt>
|
<dt><a href="TaskBackService.html">Window Querying Accessibility Service</a></dt>
|
||||||
@@ -21,3 +21,12 @@
|
|||||||
AccessibilityRecords.
|
AccessibilityRecords.
|
||||||
</dd>
|
</dd>
|
||||||
</dl>
|
</dl>
|
||||||
|
|
||||||
|
<dl>
|
||||||
|
<dt><a href="CustomViewAccessibilityActivity.html">Custom View Accessibility</a></dt>
|
||||||
|
<dd>Demonstrates how to implement accessibility support of custom views. Custom view
|
||||||
|
is a tailored widget developed by extending the base classes in the android.view
|
||||||
|
package. This sample shows how to implement the accessibility behavior via both
|
||||||
|
inheritance (non backwards compatible) and composition (backwards compatible).
|
||||||
|
</dd>
|
||||||
|
</dl>
|
||||||
|
|||||||
Reference in New Issue
Block a user