app:actionLayout定义的menu不能触发onOptionsItemSelected

创建option menu时,在menu的定义xml文件中使用android:title定义的menu item当点击时可以触发activity或者fragmentonOptionsItemSelected方法,而使用app:actionLayoutapp:actionViewClass定义的menu item无法触发onOptionsItemSelected

查看加载menu文件的inflate()方法源码,这个方法位于MenuInflater.java文件中。

public void inflate(@MenuRes int menuRes, Menu menu) {
        XmlResourceParser parser = null;
        try {
            parser = mContext.getResources().getLayout(menuRes);
            AttributeSet attrs = Xml.asAttributeSet(parser);
            parseMenu(parser, attrs, menu);
        } catch (XmlPullParserException e) {
            throw new InflateException("Error inflating menu XML", e);
        } catch (IOException e) {
            throw new InflateException("Error inflating menu XML", e);
        } finally {
            if (parser != null) parser.close();
        }
    }
复制代码

最终会调用到下面的setItem()方法

private void setItem(MenuItem item) {
    ...省略部分代码
    boolean actionViewSpecified = false;
    if (itemActionViewClassName != null) {
        View actionView = (View) newInstance(itemActionViewClassName,
                        ACTION_VIEW_CONSTRUCTOR_SIGNATURE, mActionViewConstructorArguments);
        item.setActionView(actionView);
        actionViewSpecified = true;
     }
     if (itemActionViewLayout > 0) {
         if (!actionViewSpecified) {
             item.setActionView(itemActionViewLayout);
             actionViewSpecified = true;
         } else {
             Log.w(LOG_TAG, "Ignoring attribute 'itemActionViewLayout'."
                            + " Action view already specified.");
         }
     }
     if (itemActionProvider != null) {
         item.setActionProvider(itemActionProvider);
     }
     ...省略部分代码
}

复制代码

在xml文件的item标签中如果设置了了app:actionLayoutapp:actionViewClass,会直接调用item.setActionView(),如果只是设置android:title并不会调用这个方法。代码后面会执行到ActionMenuPresenter.java类,在getItemView()方法中

public View getItemView(final MenuItemImpl item, View convertView, ViewGroup parent) {
        View actionView = item.getActionView();
        if (actionView == null || item.hasCollapsibleActionView()) {
            actionView = super.getItemView(item, convertView, parent);
        }
        actionView.setVisibility(item.isActionViewExpanded() ? View.GONE : View.VISIBLE);

        final ActionMenuView menuParent = (ActionMenuView) parent;
        final ViewGroup.LayoutParams lp = actionView.getLayoutParams();
        if (!menuParent.checkLayoutParams(lp)) {
            actionView.setLayoutParams(menuParent.generateLayoutParams(lp));
        }
        return actionView;
    }
复制代码

如果item.getActionView()拿到的actionViewnull就执行父类的getActionView(),上面代码处设置了actionView则不会执行到if语句里。来到父类BaseMenuPresenter.java中的getItemView的方法:

public View getItemView(MenuItemImpl item, View convertView, ViewGroup parent) {
        MenuView.ItemView itemView;
        if (convertView instanceof MenuView.ItemView) {
            itemView = (MenuView.ItemView) convertView;
        } else {
            itemView = createItemView(parent);
        }
        bindItemView(item, itemView);
        return (View) itemView;
    }
复制代码

然后执行到bindItemView()方法,这是个抽象方法,实现在ActionMenuPresenter.java

public void bindItemView(MenuItemImpl item, MenuView.ItemView itemView) {
        itemView.initialize(item, 0);

        final ActionMenuView menuView = (ActionMenuView) mMenuView;
        final ActionMenuItemView actionItemView = (ActionMenuItemView) itemView;
        actionItemView.setItemInvoker(menuView);

        if (mPopupCallback == null) {
            mPopupCallback = new ActionMenuPopupCallback();
        }
        actionItemView.setPopupCallback(mPopupCallback);
    }
复制代码

这里调用了ActionMenuItemView类中的setItemInvoker(menuView),此方法设置了一个mItemInvoker,

@Override
    public void onClick(View v) {
        if (mItemInvoker != null) {
            mItemInvoker.invokeItem(mItemData);
        }
    }

    public void setItemInvoker(MenuBuilder.ItemInvoker invoker) {
        mItemInvoker = invoker;
    }
复制代码

到这终于发现mItemInvoker其实是拦截了ActionMenuItemView的点击事件,当点击ActionMenuItemView时,会调用到MenuBuilder类中的performItemAction()方法:

public boolean performItemAction(MenuItem item, MenuPresenter preferredPresenter, int flags) {
    ...
    boolean invoked = itemImpl.invoke();
    ...
}
复制代码

itemImpl.invoke()最终会调用到Activity中的onOptionsItemSelected()

总结一下,未使用app:actionLayoutapp:actionViewClass标签定义的menu最终生成一个ActionMenuItemView,同时对其设置点击事件从而触发了Activity中的onOptionsItemSelected(),而使用了app:actionLayoutapp:actionViewClass定义的menu未自动设置点击事件,所以不会触发,需要额外设置点击事件监听器。

© 版权声明
THE END
喜欢就支持一下吧
点赞0 分享