创建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未自动设置点击事件,所以不会触发,需要额外设置点击事件监听器。