DrawerLayout 和 ImageView 都实现了 onTouchListener

DrawerLayout and ImageView both implementing onTouchListener

我正在移动到 DrawerLayout,但是上述布局还包含一个 ImageView,在该布局之上实现了 onTouchListener 以实现 zoom/rotate 功能。

问题是当我尝试任何 ImageView 功能时,抽屉自己的触摸处理程序使我的应用程序崩溃。

我试过设置 drawerLayout.requestDisallowInterceptTouchEvent(false); 但这并没有真正帮助。

<android.support.v4.widget.DrawerLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/container"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent">

    <!-- map container -->
    <ImageView
        android:id="@+id/map_layout"
        android:background="@color/divider_color"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent" />

    <!-- navigation drawer -->
    <ListView
        android:id="@+id/navigation_drawer"
        android:choiceMode="singleChoice"
        android:divider="@color/divider_color"
        android:dividerHeight="1dp"
        android:background="@color/button_label"
        android:layout_gravity="start"
        android:layout_width="240dp"
        android:layout_height="fill_parent" />
</android.support.v4.widget.DrawerLayout>  

异常:

  java.lang.ArrayIndexOutOfBoundsException: length=1; index=1
            at android.support.v4.widget.ViewDragHelper.shouldInterceptTouchEvent(ViewDragHelper.java:1014)
            at android.support.v4.widget.DrawerLayout.onInterceptTouchEvent(DrawerLayout.java:1140)
            at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:1859)
            at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2216)
            at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:1959)
            at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2216)
            at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:1959)
            at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2216)
            at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:1959)
            at com.android.internal.policy.impl.PhoneWindow$DecorView.superDispatchTouchEvent(PhoneWindow.java:2072)
            at com.android.internal.policy.impl.PhoneWindow.superDispatchTouchEvent(PhoneWindow.java:1519)
            at android.app.Activity.dispatchTouchEvent(Activity.java:2467)
            at com.android.internal.policy.impl.PhoneWindow$DecorView.dispatchTouchEvent(PhoneWindow.java:2020)
            at android.view.View.dispatchPointerEvent(View.java:8017)
            at android.view.ViewRootImpl$ViewPostImeInputStage.processPointerEvent(ViewRootImpl.java:3977)
            at android.view.ViewRootImpl$ViewPostImeInputStage.onProcess(ViewRootImpl.java:3856)
            at android.view.ViewRootImpl$InputStage.deliver(ViewRootImpl.java:3416)
            at android.view.ViewRootImpl$InputStage.onDeliverToNext(ViewRootImpl.java:3466)
            at android.view.ViewRootImpl$InputStage.forward(ViewRootImpl.java:3435)
            at android.view.ViewRootImpl$AsyncInputStage.forward(ViewRootImpl.java:3542)
            at android.view.ViewRootImpl$InputStage.apply(ViewRootImpl.java:3443)
            at android.view.ViewRootImpl$AsyncInputStage.apply(ViewRootImpl.java:3599)
            at android.view.ViewRootImpl$InputStage.deliver(ViewRootImpl.java:3416)
            at android.view.ViewRootImpl$InputStage.onDeliverToNext(ViewRootImpl.java:3466)
            at android.view.ViewRootImpl$InputStage.forward(ViewRootImpl.java:3435)
            at android.view.ViewRootImpl$InputStage.apply(ViewRootImpl.java:3443)
            at android.view.ViewRootImpl$InputStage.deliver(ViewRootImpl.java:3416)
            at android.view.ViewRootImpl.deliverInputEvent(ViewRootImpl.java:5565)
            at android.view.ViewRootImpl.doProcessInputEvents(ViewRootImpl.java:5545)
            at android.view.ViewRootImpl.enqueueInputEvent(ViewRootImpl.java:5516)
            at android.view.ViewRootImpl$WindowInputEventReceiver.onInputEvent(ViewRootImpl.java:5645)
            at android.view.InputEventReceiver.dispatchInputEvent(InputEventReceiver.java:185)
            at android.view.InputEventReceiver.nativeConsumeBatchedInputEvents(Native Method)
            at android.view.InputEventReceiver.consumeBatchedInputEvents(InputEventReceiver.java:176)
            at android.view.ViewRootImpl.doConsumeBatchedInput(ViewRootImpl.java:5618)
            at android.view.ViewRootImpl$ConsumeBatchedInputRunnable.run(ViewRootImpl.java:5664)
            at android.view.Choreographer$CallbackRecord.run(Choreographer.java:761)
            at android.view.Choreographer.doCallbacks(Choreographer.java:574)
            at android.view.Choreographer.doFrame(Choreographer.java:542)
            at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:747)
            at android.os.Handler.handleCallback(Handler.java:743)
            at android.os.Handler.dispatchMessage(Handler.java:99)
            at android.os.Looper.loop(Looper.java:136)
            at android.app.ActivityThread.main(ActivityThread.java:5212)
            at java.lang.reflect.Method.invokeNative(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:515)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:786)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:602)
            at dalvik.system.NativeStart.main(Native Method)

我找到的唯一简短有效的解决方案是扩展 DrawerLayout 并覆盖 onTouchEventonInterceptTouchEvent 方法。

所以我首先根据 DrawerLayout 创建自己的自定义 Layout 我称之为 NavigationDrawer:

NavigationDrawer.java

import android.content.Context;
import android.support.v4.widget.DrawerLayout;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.widget.ListView;

public class NavigationDrawer extends DrawerLayout {

    private ListView menu;

    public NavigationDrawer(Context context) {
        super(context);
    }

    public NavigationDrawer(Context context, AttributeSet attributeSet) {
        super(context, attributeSet);
    }

    public NavigationDrawer(Context context, AttributeSet attributeSet, int disp) {
        super(context, attributeSet, disp);
    }

    public void addMenu(ListView menu) {
        this.menu = menu;
    }

    /**
     * Override the method to correct the detection region for the navigation layout open/close case.
     * @param ev Motion event descriptor
     * @return
     */
    @Override
    public boolean onTouchEvent(MotionEvent ev) {
        super.onTouchEvent(ev);

        if (ev.getX() > 30 && ev.getAction() == MotionEvent.ACTION_DOWN) {
            if (isDrawerOpen(menu) || isDrawerVisible(menu)) {
                return true;
            } else {
                return false;
            }
        }

        return false;
    }

    /**
     * Override the method to catch ArrayIndexOutOfBounds exception thrown in conflict between
     * the two touch listeners.
     * @param ev Motion event descriptor
     * @return
     */
    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
        try {
            return super.onInterceptTouchEvent(ev);
        } catch (ArrayIndexOutOfBoundsException e ) {
            return false;
        }
    }
}

创建它的布局:

<?xml version="1.0" encoding="utf-8"?>
<...views.NavigationDrawer
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/content_layout"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent">

    <FrameLayout
        android:layout_width="fill_parent"
        android:layout_height="fill_parent">

       <!-- My own content in here most likely you will want swappable fragment. -->

    </FrameLayout>

    <!-- List of our menu options -->
    <ListView
        android:id="@+id/menu"
        android:choiceMode="singleChoice"
        android:divider="@color/divider_color"
        android:dividerHeight="1dp"
        android:background="@color/button_label"
        android:clickable="true"
        android:layout_gravity="start"
        android:layout_width="240dp"
        android:layout_height="fill_parent" />

</...views.NavigationDrawer>

还需要ListView项:

<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/navigation_item"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:gravity="center_vertical"
    android:padding="20dp"
    android:textSize="@dimen/main_title"
    android:background="#FFFFFF"
    android:textColor="@color/checkbox_label" />

然后创建一切:

    // Navigation menu
    private NavigationDrawer menuLayout;
    private ActionBarDrawerToggle menuToggle;
    private ListView menu;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.layout_navigator);

        // Setup the navigation drawer layout
        String[] menuOptions = getResources().getStringArray(R.array.menu_options);
        menu = (ListView) findViewById(R.id.menu);
        menuLayout = (NavigationDrawer) findViewById(R.id.content_layout);
        menuLayout.addMenu(menu);
        menuLayout.requestDisallowInterceptTouchEvent(false);

        // Populate the menu with options
        menu.setAdapter(new ArrayAdapter<String>(this, R.layout.layout_navigator_item, menuOptions));
        menu.setOnItemClickListener(new MenuItemSelected());

        menuToggle = new ActionBarDrawerToggle(
                this, menuLayout, null, R.string.open, R.string.closed) {
            @Override
            public void onDrawerOpened(View drawerView) {
                super.onDrawerOpened(drawerView);
            }

            @Override
            public void onDrawerClosed(View drawerView) {
                super.onDrawerClosed(drawerView);
            }
        };
}