Appcompat v21抛出java.lang.UnsupportedOperationException异常

10

将我的项目更新为使用appcompat库的21.0.0版本后,我在使用gridview多选模态事件创建的上下文菜单时遇到了问题。相同的代码在appcompat v20中可以正常工作。

这是主活动的相关部分:

public class MainActivity extends android.support.v7.app.ActionBarActivity 
    implements AbsListView.MultiChoiceModeListener {

    ...
    mGridView.setChoiceMode(AbsListView.CHOICE_MODE_MULTIPLE_MODAL);
    mGridView.setMultiChoiceModeListener(this);

    @Override
    public boolean onCreateActionMode(final ActionMode mode, final Menu menu) {
        mode.setTitle("Started");
        mode.getMenuInflater().inflate(R.menu.context_menu, menu);
        return true;
    }
}

这是 context_menu.xml 文件:

<menu xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto">
    <item
        android:id="@+id/menu_item_share"
        android:title="Share..."
        app:showAsAction="ifRoom"
        android:icon="@android:drawable/ic_menu_share"
        app:actionProviderClass="android.support.v7.widget.ShareActionProvider" />
</menu>

这是我得到的堆栈跟踪:

java.lang.UnsupportedOperationException: This is not supported, use MenuItemCompat.setActionProvider()
        at android.support.v7.internal.view.menu.MenuItemImpl.setActionProvider(MenuItemImpl.java:628)
        at android.support.v7.internal.view.menu.MenuItemWrapperICS.setSupportActionProvider(MenuItemWrapperICS.java:315)
        at android.support.v4.view.MenuItemCompat.setActionProvider(MenuItemCompat.java:345)
        at android.support.v7.internal.view.SupportMenuInflater$MenuState.setItem(SupportMenuInflater.java:473)
        at android.support.v7.internal.view.SupportMenuInflater$MenuState.addSubMenuItem(SupportMenuInflater.java:485)
        at android.support.v7.internal.view.SupportMenuInflater.parseMenu(SupportMenuInflater.java:194)
        at android.support.v7.internal.view.SupportMenuInflater.inflate(SupportMenuInflater.java:118)
        at creativesdk.adobe.com.myapplication.MainActivity.onCreateActionMode(MainActivity.java:71)
        at android.widget.AbsListView$MultiChoiceModeWrapper.onCreateActionMode(AbsListView.java:6165)
        at android.support.v7.internal.view.SupportActionModeWrapper$CallbackWrapper.onCreateActionMode(SupportActionModeWrapper.java:151)
        at android.support.v7.app.ActionBarActivityDelegateBase$ActionModeCallbackWrapper.onCreateActionMode(ActionBarActivityDelegateBase.java:1367)
        at android.support.v7.internal.app.WindowDecorActionBar$ActionModeImpl.dispatchOnCreate(WindowDecorActionBar.java:1012)
        at android.support.v7.internal.app.WindowDecorActionBar.startActionMode(WindowDecorActionBar.java:510)
        at android.support.v7.app.ActionBarActivityDelegateBase.startSupportActionMode(ActionBarActivityDelegateBase.java:576)
        at android.support.v7.app.ActionBarActivityDelegateHC.startActionModeForChild(ActionBarActivityDelegateHC.java:62)
        at android.support.v7.internal.widget.NativeActionModeAwareLayout.startActionModeForChild(NativeActionModeAwareLayout.java:44)
        at android.view.ViewGroup.startActionModeForChild(ViewGroup.java:694)
        at android.view.View.startActionMode(View.java:4857)
        at android.widget.AbsListView.performLongPress(AbsListView.java:3102)
        at android.widget.AbsListView$CheckForLongPress.run(AbsListView.java:3061)
        at android.os.Handler.handleCallback(Handler.java:739)
        at android.os.Handler.dispatchMessage(Handler.java:95)
        at android.os.Looper.loop(Looper.java:135)
        at android.app.ActivityThread.main(ActivityThread.java:5221)
        at java.lang.reflect.Method.invoke(Native Method)
        at java.lang.reflect.Method.invoke(Method.java:372)
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:899)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:694)

我很好奇是否有人遇到了同样的问题,是否有解决方法。

4个回答

9
在使用AppCompat v21时,ICS设备在尝试从ActionMode中膨胀菜单项时确实存在问题。似乎菜单项被包裹了两次,并且调用了包装项方法而不是本机方法,导致了此异常。
谷歌需要在未来版本的AppCompat中修复这个问题。
无论如何,下面是我实现的一个hack以使其在当前版本中工作:
1)在android.support.v7.internal.view.menu包中创建一个实用程序类(使用此包是强制性的,以允许访问包保护方法而不使用反射)。
package android.support.v7.internal.view.menu;

import android.view.Menu;

/**
 * Hack to allow inflating ActionMode menus on Android 4.0.x with AppCompat v21
 */
public class MenuUnwrapper {

    public static Menu unwrap(Menu menu) {
        if (menu instanceof MenuWrapperICS) {
            return ((MenuWrapperICS) menu).getWrappedObject();
        }
        return menu;
    }
}

2) 您可以按照以下方式扩展您的菜单:

@Override
public boolean onCreateActionMode(ActionMode mode, Menu menu) {
    mode.getMenuInflater().inflate(R.menu.context_menu, MenuUnwrapper.unwrap(menu));
    return true;
}

编辑

该漏洞已在AppCompat v21.0.2中修复,因此不再需要此hack。

请更新您的工具。


尝试了这个解决方案,但对我没有用。问题在于mode.getMenuInflater()仍然返回一个android.support.v7.internal.view.SupportMenuInflater,它会填充MenuItemWrapperICS项。有什么想法吗? - André Restivo
我花了一些时间,但最终找出了问题所在。我还需要在onPrepareActionMode上取消菜单的包装。希望能对某人有所帮助。谢谢。 - André Restivo
是的,在每个需要填充菜单项的方法中,您需要解开菜单。问题在于在AppCompat代码的某个地方,菜单被包装了起来,而实际上不应该这样做,然后在每个回调中您会得到这个无效的菜单。请确认您在4.0.x(冰淇淋三明治)设备上遇到了这个问题,而不是在Jellybean或KitKat上?很高兴能帮到您! - BladeCoder
是的,看起来只支持ICS。再次感谢,我觉得我自己找不到这个。 - André Restivo

4

试试这个。对我有用。

MenuItem menuItem =  menu.findItem(R.id.search);
if (menuItem != null) {
    MenuItemCompat.setOnActionExpandListener(menuItem,this);
    MenuItemCompat.setActionView(menuItem, mSearchView);
}

3

试试这个

inflater.inflate(R.menu.menu, menu);

// search
MenuItem item = menu.findItem(R.id.settings_search);
MenuItemCompat.setOnActionExpandListener(
    item, new MenuItemCompat.OnActionExpandListener() {

    @Override
    public boolean onMenuItemActionExpand(MenuItem item) {
       return true;
    }

    @Override
    public boolean onMenuItemActionCollapse(MenuItem item) {
        return true;
    }
});

这似乎是正确的答案。异常提示您使用MenuItemCompat类的静态方法。 - Andrew

0
@Override
        public boolean onCreateActionMode(ActionMode mode, Menu menu) {
            Log.d(TAG, "onCreateActionMode");

            MenuItem item = menu.findItem(R.id.menu_item_share);
            mShareActionProvider = (ShareActionProvider) MenuItemCompat.getActionProvider(item);
            MenuItemCompat.setActionProvider(item, mShareActionProvider);

            return true;
        }

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接