创建option menu时,在menu的定义xml文件中使用android:title定义的menu item当点击时可以触发activity或者fragment的onOptionsItemSelected方法,而使用app:actionLayout或app: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:actionLayout或app: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()拿到的actionView是null就执行父类的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:actionLayout或app:actionViewClass标签定义的menu最终生成一个ActionMenuItemView,同时对其设置点击事件从而触发了Activity中的onOptionsItemSelected(),而使用了app:actionLayout或app:actionViewClass定义的menu未自动设置点击事件,所以不会触发,需要额外设置点击事件监听器。



















![[02/27][官改] Simplicity@MIX2 ROM更新-一一网](https://www.proyy.com/wp-content/uploads/2020/02/3168457341.jpg)


![[桜井宁宁]COS和泉纱雾超可爱写真福利集-一一网](https://www.proyy.com/skycj/data/images/2020-12-13/4d3cf227a85d7e79f5d6b4efb6bde3e8.jpg)

![[桜井宁宁] 爆乳奶牛少女cos写真-一一网](https://www.proyy.com/skycj/data/images/2020-12-13/d40483e126fcf567894e89c65eaca655.jpg)
