Appcompat v21 throws java.lang.UnsupportedOperationException

После обновления моего проекта для использования библиотеки appcompat до версии 21.0.0 у меня возникла проблема с контекстным меню, созданным с помощью модального события gridview multichoice. Тот же код хорошо работает с 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)

Мне любопытно узнать, нашли ли кто-нибудь ту же проблему и если есть способ ее преодолеть.

Ответ 1

На устройствах ICS действительно возникает проблема при попытке раздувания элемента меню из ActionMode с помощью AppCompat v21. Кажется, что пункты меню обернуты 2 раза, а метод обернутого элемента вызывается вместо собственного, вызывая это исключение.

Google должен исправить это в будущей версии AppCompat.

В любом случае, здесь был взломан я, чтобы он работал с текущей версией:

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, и этот хак больше не нужен.

Обновите свои инструменты.

Ответ 2

Попробуйте это. Это сработало для меня.

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;
    }
});

Ответ 4

@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;
        }