| Index: trunk/src/content/public/android/java/src/org/chromium/content/browser/accessibility/BrowserAccessibilityManager.java
|
| ===================================================================
|
| --- trunk/src/content/public/android/java/src/org/chromium/content/browser/accessibility/BrowserAccessibilityManager.java (revision 207890)
|
| +++ trunk/src/content/public/android/java/src/org/chromium/content/browser/accessibility/BrowserAccessibilityManager.java (working copy)
|
| @@ -1,419 +0,0 @@
|
| -// Copyright (c) 2013 The Chromium Authors. All rights reserved.
|
| -// Use of this source code is governed by a BSD-style license that can be
|
| -// found in the LICENSE file.
|
| -
|
| -package org.chromium.content.browser.accessibility;
|
| -
|
| -import android.content.Context;
|
| -import android.graphics.Rect;
|
| -import android.os.Bundle;
|
| -import android.view.MotionEvent;
|
| -import android.view.View;
|
| -import android.view.accessibility.AccessibilityEvent;
|
| -import android.view.accessibility.AccessibilityManager;
|
| -import android.view.accessibility.AccessibilityNodeInfo;
|
| -import android.view.accessibility.AccessibilityNodeProvider;
|
| -import android.view.inputmethod.InputMethodManager;
|
| -
|
| -import org.chromium.base.CalledByNative;
|
| -import org.chromium.base.JNINamespace;
|
| -import org.chromium.content.browser.ContentViewCore;
|
| -import org.chromium.content.browser.RenderCoordinates;
|
| -
|
| -import java.util.ArrayList;
|
| -import java.util.List;
|
| -
|
| -/**
|
| - * Native accessibility for a {@link ContentViewCore}.
|
| - */
|
| -@JNINamespace("content")
|
| -public class BrowserAccessibilityManager extends AccessibilityNodeProvider {
|
| - private static final String TAG = BrowserAccessibilityManager.class.getSimpleName();
|
| -
|
| - private ContentViewCore mContentViewCore;
|
| - private AccessibilityManager mAccessibilityManager;
|
| - private RenderCoordinates mRenderCoordinates;
|
| - private int mNativeObj;
|
| - private int mAccessibilityFocusId;
|
| - private final int[] mTempLocation = new int[2];
|
| - private View mView;
|
| - private boolean mUserHasTouchExplored;
|
| - private boolean mFrameInfoInitialized;
|
| -
|
| - // If this is true, enables an experimental feature that focuses the web page after it
|
| - // finishes loading. Disabled for now because it can be confusing if the user was
|
| - // trying to do something when this happens.
|
| - private boolean mFocusPageOnLoad;
|
| -
|
| - /**
|
| - * Create a BrowserAccessibilityManager object, which is owned by the C++
|
| - * BrowserAccessibilityManagerAndroid instance, and connects to the content view.
|
| - * @param nativeBrowserAccessibilityManagerAndroid A pointer to the counterpart native
|
| - * C++ object that owns this object.
|
| - * @param contentViewCore The content view that this object provides accessibility for.
|
| - */
|
| - @CalledByNative
|
| - private static BrowserAccessibilityManager create(int nativeBrowserAccessibilityManagerAndroid,
|
| - ContentViewCore contentViewCore) {
|
| - return new BrowserAccessibilityManager(
|
| - nativeBrowserAccessibilityManagerAndroid, contentViewCore);
|
| - }
|
| -
|
| - private BrowserAccessibilityManager(int nativeBrowserAccessibilityManagerAndroid,
|
| - ContentViewCore contentViewCore) {
|
| - mNativeObj = nativeBrowserAccessibilityManagerAndroid;
|
| - mContentViewCore = contentViewCore;
|
| - mContentViewCore.setBrowserAccessibilityManager(this);
|
| - mAccessibilityFocusId = View.NO_ID;
|
| - mView = mContentViewCore.getContainerView();
|
| - mRenderCoordinates = mContentViewCore.getRenderCoordinates();
|
| - mAccessibilityManager =
|
| - (AccessibilityManager) mContentViewCore.getContext()
|
| - .getSystemService(Context.ACCESSIBILITY_SERVICE);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void onNativeObjectDestroyed() {
|
| - if (mContentViewCore.getBrowserAccessibilityManager() == this) {
|
| - mContentViewCore.setBrowserAccessibilityManager(null);
|
| - }
|
| - mNativeObj = 0;
|
| - mContentViewCore = null;
|
| - }
|
| -
|
| - @Override
|
| - public AccessibilityNodeInfo createAccessibilityNodeInfo(int virtualViewId) {
|
| - if (!mAccessibilityManager.isEnabled() || mNativeObj == 0 || !mFrameInfoInitialized) {
|
| - return null;
|
| - }
|
| -
|
| - int rootId = nativeGetRootId(mNativeObj);
|
| - if (virtualViewId == View.NO_ID) {
|
| - virtualViewId = rootId;
|
| - }
|
| - if (mAccessibilityFocusId == View.NO_ID) {
|
| - mAccessibilityFocusId = rootId;
|
| - }
|
| -
|
| - final AccessibilityNodeInfo info = AccessibilityNodeInfo.obtain(mView);
|
| - info.setPackageName(mContentViewCore.getContext().getPackageName());
|
| - info.setSource(mView, virtualViewId);
|
| -
|
| - if (nativePopulateAccessibilityNodeInfo(mNativeObj, info, virtualViewId)) {
|
| - return info;
|
| - } else {
|
| - return null;
|
| - }
|
| - }
|
| -
|
| - /** @inheritDoc */
|
| - public List<AccessibilityNodeInfo> findAccessibilityNodeInfosByText(String text,
|
| - int virtualViewId) {
|
| - return new ArrayList<AccessibilityNodeInfo>();
|
| - }
|
| -
|
| - /** @inheritDoc */
|
| - public boolean performAction(int virtualViewId, int action, Bundle arguments) {
|
| - if (!mAccessibilityManager.isEnabled() || mNativeObj == 0) {
|
| - return false;
|
| - }
|
| -
|
| - switch (action) {
|
| - case AccessibilityNodeInfo.ACTION_ACCESSIBILITY_FOCUS:
|
| - if (mAccessibilityFocusId == virtualViewId) {
|
| - return true;
|
| - }
|
| -
|
| - if (mAccessibilityFocusId != View.NO_ID) {
|
| - sendAccessibilityEvent(mAccessibilityFocusId,
|
| - AccessibilityEvent.TYPE_VIEW_HOVER_EXIT);
|
| - }
|
| - mAccessibilityFocusId = virtualViewId;
|
| - sendAccessibilityEvent(mAccessibilityFocusId,
|
| - AccessibilityEvent.TYPE_VIEW_HOVER_ENTER);
|
| - return true;
|
| - case AccessibilityNodeInfo.ACTION_CLEAR_ACCESSIBILITY_FOCUS:
|
| - if (mAccessibilityFocusId == virtualViewId) {
|
| - mAccessibilityFocusId = View.NO_ID;
|
| - }
|
| - return true;
|
| - case AccessibilityNodeInfo.ACTION_CLICK:
|
| - nativeClick(mNativeObj, virtualViewId);
|
| - break;
|
| - case AccessibilityNodeInfo.ACTION_FOCUS:
|
| - nativeFocus(mNativeObj, virtualViewId);
|
| - break;
|
| - case AccessibilityNodeInfo.ACTION_CLEAR_FOCUS:
|
| - nativeBlur(mNativeObj);
|
| - break;
|
| - default:
|
| - break;
|
| - }
|
| - return false;
|
| - }
|
| -
|
| - /**
|
| - * @see View#dispatchHoverEvent(MotionEvent)
|
| - */
|
| - public boolean dispatchHoverEvent(MotionEvent event) {
|
| - if (!mAccessibilityManager.isEnabled() || mNativeObj == 0) {
|
| - return false;
|
| - }
|
| -
|
| - mUserHasTouchExplored = true;
|
| - float x = event.getX();
|
| - float y = event.getY();
|
| -
|
| - // Offset by the location of the web content within the view.
|
| - // Note: the MotionEvent has view-relative coordinates already,
|
| - // so we don't have to offset by the view relative to the screen.
|
| - y -= mRenderCoordinates.getContentOffsetYPix();
|
| -
|
| - // Convert to CSS coordinates.
|
| - int cssX = (int) (mRenderCoordinates.fromPixToLocalCss(x) +
|
| - mRenderCoordinates.getScrollX());
|
| - int cssY = (int) (mRenderCoordinates.fromPixToLocalCss(y) +
|
| - mRenderCoordinates.getScrollY());
|
| - int id = nativeHitTest(mNativeObj, cssX, cssY);
|
| - if (mAccessibilityFocusId != id) {
|
| - mAccessibilityFocusId = id;
|
| - sendAccessibilityEvent(id, AccessibilityEvent.TYPE_VIEW_ACCESSIBILITY_FOCUSED);
|
| - }
|
| -
|
| - return true;
|
| - }
|
| -
|
| - /**
|
| - * Called by ContentViewCore to notify us when the frame info is initialized,
|
| - * the first time, since until that point, we can't use mRenderCoordinates to transform
|
| - * web coordinates to screen coordinates.
|
| - */
|
| - public void notifyFrameInfoInitialized() {
|
| - if (mFrameInfoInitialized) return;
|
| -
|
| - mFrameInfoInitialized = true;
|
| - // (Re-) focus focused element, since we weren't able to create an
|
| - // AccessibilityNodeInfo for this element before.
|
| - if (mAccessibilityFocusId != View.NO_ID) {
|
| - sendAccessibilityEvent(mAccessibilityFocusId,
|
| - AccessibilityEvent.TYPE_VIEW_ACCESSIBILITY_FOCUSED);
|
| - }
|
| - }
|
| -
|
| - private void sendAccessibilityEvent(int virtualViewId, int eventType) {
|
| - if (!mAccessibilityManager.isEnabled() || mNativeObj == 0) return;
|
| -
|
| - final AccessibilityEvent event = AccessibilityEvent.obtain(eventType);
|
| - event.setPackageName(mContentViewCore.getContext().getPackageName());
|
| - int rootId = nativeGetRootId(mNativeObj);
|
| - if (virtualViewId == rootId) {
|
| - virtualViewId = View.NO_ID;
|
| - }
|
| - event.setSource(mView, virtualViewId);
|
| - if (!nativePopulateAccessibilityEvent(mNativeObj, event, virtualViewId, eventType)) return;
|
| -
|
| - // This is currently needed if we want Android to draw the yellow box around
|
| - // the item that has accessibility focus. In practice, this doesn't seem to slow
|
| - // things down, because it's only called when the accessibility focus moves.
|
| - // TODO(dmazzoni): remove this if/when Android framework fixes bug.
|
| - mContentViewCore.getContainerView().postInvalidate();
|
| -
|
| - mContentViewCore.getContainerView().requestSendAccessibilityEvent(mView, event);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void handlePageLoaded(int id) {
|
| - if (mUserHasTouchExplored) return;
|
| -
|
| - if (mFocusPageOnLoad) {
|
| - // Focus the natively focused node (usually document),
|
| - // if this feature is enabled.
|
| - mAccessibilityFocusId = id;
|
| - sendAccessibilityEvent(id, AccessibilityEvent.TYPE_VIEW_FOCUSED);
|
| - }
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void handleFocusChanged(int id) {
|
| - if (mAccessibilityFocusId == id) return;
|
| -
|
| - mAccessibilityFocusId = id;
|
| - sendAccessibilityEvent(id, AccessibilityEvent.TYPE_VIEW_FOCUSED);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void handleCheckStateChanged(int id) {
|
| - sendAccessibilityEvent(id, AccessibilityEvent.TYPE_VIEW_CLICKED);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void handleTextSelectionChanged(int id) {
|
| - sendAccessibilityEvent(id, AccessibilityEvent.TYPE_WINDOW_CONTENT_CHANGED);
|
| - sendAccessibilityEvent(id, AccessibilityEvent.TYPE_VIEW_TEXT_SELECTION_CHANGED);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void handleEditableTextChanged(int id) {
|
| - sendAccessibilityEvent(id, AccessibilityEvent.TYPE_WINDOW_CONTENT_CHANGED);
|
| - sendAccessibilityEvent(id, AccessibilityEvent.TYPE_VIEW_TEXT_CHANGED);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void handleContentChanged(int id) {
|
| - sendAccessibilityEvent(id, AccessibilityEvent.TYPE_WINDOW_CONTENT_CHANGED);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void handleNavigate() {
|
| - mAccessibilityFocusId = View.NO_ID;
|
| - mUserHasTouchExplored = false;
|
| - mFrameInfoInitialized = false;
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void handleScrolledToAnchor(int id) {
|
| - if (mAccessibilityFocusId == id) {
|
| - return;
|
| - }
|
| -
|
| - mAccessibilityFocusId = id;
|
| - sendAccessibilityEvent(id, AccessibilityEvent.TYPE_VIEW_ACCESSIBILITY_FOCUSED);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void announceLiveRegionText(String text) {
|
| - mView.announceForAccessibility(text);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void setAccessibilityNodeInfoParent(AccessibilityNodeInfo node, int parentId) {
|
| - node.setParent(mView, parentId);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void addAccessibilityNodeInfoChild(AccessibilityNodeInfo node, int child_id) {
|
| - node.addChild(mView, child_id);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void setAccessibilityNodeInfoBooleanAttributes(AccessibilityNodeInfo node,
|
| - boolean checkable, boolean checked, boolean clickable,
|
| - boolean enabled, boolean focusable, boolean focused, boolean password,
|
| - boolean scrollable, boolean selected, boolean visibleToUser) {
|
| - node.setCheckable(checkable);
|
| - node.setChecked(checked);
|
| - node.setClickable(clickable);
|
| - node.setEnabled(enabled);
|
| - node.setFocusable(focusable);
|
| - node.setFocused(focused);
|
| - node.setPassword(password);
|
| - node.setScrollable(scrollable);
|
| - node.setSelected(selected);
|
| - node.setVisibleToUser(visibleToUser);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void setAccessibilityNodeInfoStringAttributes(AccessibilityNodeInfo node,
|
| - String className, String contentDescription) {
|
| - node.setClassName(className);
|
| - node.setContentDescription(contentDescription);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void setAccessibilityNodeInfoLocation(AccessibilityNodeInfo node,
|
| - int absoluteLeft, int absoluteTop, int parentRelativeLeft, int parentRelativeTop,
|
| - int width, int height, boolean isRootNode) {
|
| - // First set the bounds in parent.
|
| - Rect boundsInParent = new Rect(parentRelativeLeft, parentRelativeTop,
|
| - parentRelativeLeft + width, parentRelativeTop + height);
|
| - if (isRootNode) {
|
| - // Offset of the web content relative to the View.
|
| - boundsInParent.offset(0, (int) mRenderCoordinates.getContentOffsetYPix());
|
| - }
|
| - node.setBoundsInParent(boundsInParent);
|
| -
|
| - // Now set the absolute rect, which requires several transformations.
|
| - Rect rect = new Rect(absoluteLeft, absoluteTop, absoluteLeft + width, absoluteTop + height);
|
| -
|
| - // Offset by the scroll position.
|
| - rect.offset(-(int) mRenderCoordinates.getScrollX(),
|
| - -(int) mRenderCoordinates.getScrollY());
|
| -
|
| - // Convert CSS (web) pixels to Android View pixels
|
| - rect.left = (int) mRenderCoordinates.fromLocalCssToPix(rect.left);
|
| - rect.top = (int) mRenderCoordinates.fromLocalCssToPix(rect.top);
|
| - rect.bottom = (int) mRenderCoordinates.fromLocalCssToPix(rect.bottom);
|
| - rect.right = (int) mRenderCoordinates.fromLocalCssToPix(rect.right);
|
| -
|
| - // Offset by the location of the web content within the view.
|
| - rect.offset(0,
|
| - (int) mRenderCoordinates.getContentOffsetYPix());
|
| -
|
| - // Finally offset by the location of the view within the screen.
|
| - final int[] viewLocation = new int[2];
|
| - mView.getLocationOnScreen(viewLocation);
|
| - rect.offset(viewLocation[0], viewLocation[1]);
|
| -
|
| - node.setBoundsInScreen(rect);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void setAccessibilityEventBooleanAttributes(AccessibilityEvent event,
|
| - boolean checked, boolean enabled, boolean password, boolean scrollable) {
|
| - event.setChecked(checked);
|
| - event.setEnabled(enabled);
|
| - event.setPassword(password);
|
| - event.setScrollable(scrollable);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void setAccessibilityEventClassName(AccessibilityEvent event, String className) {
|
| - event.setClassName(className);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void setAccessibilityEventListAttributes(AccessibilityEvent event,
|
| - int currentItemIndex, int itemCount) {
|
| - event.setCurrentItemIndex(currentItemIndex);
|
| - event.setItemCount(itemCount);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void setAccessibilityEventScrollAttributes(AccessibilityEvent event,
|
| - int scrollX, int scrollY, int maxScrollX, int maxScrollY) {
|
| - event.setScrollX(scrollX);
|
| - event.setScrollY(scrollY);
|
| - event.setMaxScrollX(maxScrollX);
|
| - event.setMaxScrollY(maxScrollY);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void setAccessibilityEventTextChangedAttrs(AccessibilityEvent event,
|
| - int fromIndex, int addedCount, int removedCount, String beforeText, String text) {
|
| - event.setFromIndex(fromIndex);
|
| - event.setAddedCount(addedCount);
|
| - event.setRemovedCount(removedCount);
|
| - event.setBeforeText(beforeText);
|
| - event.getText().add(text);
|
| - }
|
| -
|
| - @CalledByNative
|
| - private void setAccessibilityEventSelectionAttrs(AccessibilityEvent event,
|
| - int fromIndex, int addedCount, int itemCount, String text) {
|
| - event.setFromIndex(fromIndex);
|
| - event.setAddedCount(addedCount);
|
| - event.setItemCount(itemCount);
|
| - event.getText().add(text);
|
| - }
|
| -
|
| - private native int nativeGetRootId(int nativeBrowserAccessibilityManagerAndroid);
|
| - private native int nativeHitTest(int nativeBrowserAccessibilityManagerAndroid, int x, int y);
|
| - private native boolean nativePopulateAccessibilityNodeInfo(
|
| - int nativeBrowserAccessibilityManagerAndroid, AccessibilityNodeInfo info, int id);
|
| - private native boolean nativePopulateAccessibilityEvent(
|
| - int nativeBrowserAccessibilityManagerAndroid, AccessibilityEvent event, int id,
|
| - int eventType);
|
| - private native void nativeClick(int nativeBrowserAccessibilityManagerAndroid, int id);
|
| - private native void nativeFocus(int nativeBrowserAccessibilityManagerAndroid, int id);
|
| - private native void nativeBlur(int nativeBrowserAccessibilityManagerAndroid);
|
| -}
|
|
|