it-swarm.com.ru

Как показать значки в меню переполнения в ActionBar

Я знаю, что это невозможно с использованием нативного API. Есть ли обходной путь для реализации такого взгляда?

61
Mihir

Ранее опубликованный ответ в целом нормальный. Но это в основном удаляет поведение по умолчанию меню переполнения. Например, сколько значков может отображаться на экранах разных размеров, а затем они выпадают в меню переполнения, когда их невозможно отобразить. Делая выше, вы удаляете много важных функций.

Лучшим способом было бы сообщить меню переполнения для непосредственного отображения значков. Вы можете сделать это, добавив следующий код в свою активность.

@Override
public boolean onMenuOpened(int featureId, Menu menu)
{
    if(featureId == Window.FEATURE_ACTION_BAR && menu != null){
        if(menu.getClass().getSimpleName().equals("MenuBuilder")){
            try{
                Method m = menu.getClass().getDeclaredMethod(
                    "setOptionalIconsVisible", Boolean.TYPE);
                m.setAccessible(true);
                m.invoke(menu, true);
            }
            catch(NoSuchMethodException e){
                Log.e(TAG, "onMenuOpened", e);
            }
            catch(Exception e){
                throw new RuntimeException(e);
            }
        }
    }
    return super.onMenuOpened(featureId, menu);
}
84
Simon

В вашем меню XML, используйте следующий синтаксис для вложения меню, вы начнете получать меню с иконками

<item
    Android:id="@+id/empty"
    Android:icon="@drawable/ic_action_overflow"
    Android:orderInCategory="101"
    Android:showAsAction="always">
    <menu>
        <item
            Android:id="@+id/action_show_ir_list"
            Android:icon="@drawable/ic_menu_friendslist"
            Android:showAsAction="always|withText"
            Android:title="List"/>
    </menu>
</item>
53
iBabur

Попробовал это на основе предыдущих ответов, и он отлично работает, по крайней мере, с более поздними версиями библиотеки поддержки (25.1):

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    getMenuInflater().inflate(R.menu.menu_main, menu);

    if(menu instanceof MenuBuilder){
        MenuBuilder m = (MenuBuilder) menu;
        //noinspection RestrictedApi
        m.setOptionalIconsVisible(true);
    }

    return true;
}
35
lbarbosa

Вы можете использовать SpannableString

public boolean onCreateOptionsMenu(Menu menu) {
    getMenuInflater().inflate(R.menu.menu_tab, menu);

    MenuItem item = menu.findItem(R.id.action_login);
    SpannableStringBuilder builder = new SpannableStringBuilder("* Login");
    // replace "*" with icon
    builder.setSpan(new ImageSpan(this, R.drawable.login_icon), 0, 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
    item.setTitle(builder);
}
22
Desmond Lua

Ответ от Саймона был очень полезен для меня, поэтому я хочу поделиться тем, как я реализовал его в метод onCreateOptionsMenu-, как это было предложено:

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    // Inflate the menu items for use in the action bar
    MenuInflater inflater = getMenuInflater();
    inflater.inflate(R.menu.main_action_bar, menu);

    // To show icons in the actionbar's overflow menu:
    // http://stackoverflow.com/questions/18374183/how-to-show-icons-in-overflow-menu-in-actionbar
    //if(featureId == Window.FEATURE_ACTION_BAR && menu != null){
        if(menu.getClass().getSimpleName().equals("MenuBuilder")){
            try{
                Method m = menu.getClass().getDeclaredMethod(
                        "setOptionalIconsVisible", Boolean.TYPE);
                m.setAccessible(true);
                m.invoke(menu, true);
            }
            catch(NoSuchMethodException e){
                Log.e(TAG, "onMenuOpened", e);
            }
            catch(Exception e){
                throw new RuntimeException(e);
            }
        }
    //}

    return super.onCreateOptionsMenu(menu);
}
13
user2366975

Основываясь на ответе @Desmond Lua из выше , я создал статический метод для использования в раскрывающемся списке объекта, объявленного в XML, и обеспечения того, чтобы его окрашенный цвет не влиял на состояние константы Drawable.

    /**
 * Updates a menu item in the dropdown to show it's icon that was declared in XML.
 *
 * @param item
 *         the item to update
 * @param color
 *         the color to tint with
 */
private static void updateMenuWithIcon(@NonNull final MenuItem item, final int color) {
    SpannableStringBuilder builder = new SpannableStringBuilder()
            .append("*") // the * will be replaced with the icon via ImageSpan
            .append("    ") // This extra space acts as padding. Adjust as you wish
            .append(item.getTitle());

    // Retrieve the icon that was declared in XML and assigned during inflation
    if (item.getIcon() != null && item.getIcon().getConstantState() != null) {
        Drawable drawable = item.getIcon().getConstantState().newDrawable();

        // Mutate this drawable so the tint only applies here
        drawable.mutate().setTint(color);

        // Needs bounds, or else it won't show up (doesn't know how big to be)
        drawable.setBounds(0, 0, drawable.getIntrinsicWidth(), drawable.getIntrinsicHeight());
        ImageSpan imageSpan = new ImageSpan(drawable);
        builder.setSpan(imageSpan, 0, 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
        item.setTitle(builder);
    }
}

И использование этого будет выглядеть примерно так при использовании в деятельности. Это может быть еще более элегантно в зависимости от ваших индивидуальных потребностей.

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    getMenuInflater().inflate(R.menu.menu_activity_provider_connect, menu);
    int color = ContextCompat.getColor(this, R.color.accent_dark_grey);
    updateMenuWithIcon(menu.findItem(R.id.email), color);
    updateMenuWithIcon(menu.findItem(R.id.sms), color);
    updateMenuWithIcon(menu.findItem(R.id.call), color);
    return true;
}
7
Kevin Grant

Ответ @Simon действительно работает хорошо ... но вы используете AppCompat Activity ... вам нужно будет использовать этот код вместо этого ... Потому что onMenuOpened () больше не вызывается в appcompat-v7: 22.x

@Override
    protected boolean onPrepareOptionsPanel(View view, Menu menu) {
        if(menu != null){
            if(menu.getClass().getSimpleName().equals("MenuBuilder")){
                try{
                    Method m = menu.getClass().getDeclaredMethod(
                            "setOptionalIconsVisible", Boolean.TYPE);
                    m.setAccessible(true);
                    m.invoke(menu, true);
                }
                catch(NoSuchMethodException e){
                    Log.e(Constants.DEBUG_LOG, "onMenuOpened", e);
                }
                catch(Exception e){
                    throw new RuntimeException(e);
                }
            }
        }
        return super.onPrepareOptionsPanel(view, menu);
    }
3
Vishal Kumar

Текущий лучший, но не принятый решение , вероятно, работает на старых платформах. В любом случае в новом AppCompat21 + требуемый метод не существует, а метод getDeclaredMethod возвращает исключительную ситуацию NoSuchMethodException.

Поэтому обходной путь для меня (проверенный и работающий на устройствах 4.x, 5.x) основан на параметре фона прямого изменения. Так что просто поместите этот код в свой класс Activity.

@Override
public boolean onMenuOpened(int featureId, Menu menu) {
    // enable visible icons in action bar
    if (featureId == Window.FEATURE_ACTION_BAR && menu != null) {
        if (menu.getClass().getSimpleName().equals("MenuBuilder")) {
            try {
                Field field = menu.getClass().
                        getDeclaredField("mOptionalIconsVisible");
                field.setAccessible(true);
                field.setBoolean(menu, true);
            } catch (IllegalAccessException | NoSuchFieldException e) {
                Logger.w(TAG, "onMenuOpened(" + featureId + ", " + menu + ")", e);
            }
        }
    }
    return super.onMenuOpened(featureId, menu);
}
3
Menion Asamm

Самый простой способ, который я нашел, это:

public boolean onCreateOptionsMenu(Menu menu){
     MenuInflater inflater = getMenuInflater();
     inflater.inflate(R.menu.toolbar_menu,menu);
     if(menu instanceof MenuBuilder) {  //To display icon on overflow menu

          MenuBuilder m = (MenuBuilder) menu; 
          m.setOptionalIconsVisible(true);

     }
   return true;
}        `
2
Paras Singh

По моему мнению, это возможно только путем создания пользовательской панели инструментов. Потому что ActionBar по умолчанию не дает вам эту функцию. Но вы можете поместить значки, взяв подменю в качестве дочернего элемента. И если у вас есть лучшее решение, чем я .. просто сообщите мне.

<?xml version="1.0" encoding="utf-8"?>
<menu xmlns:Android="http://schemas.Android.com/apk/res/Android"
      xmlns:app="http://schemas.Android.com/apk/res-auto">
<item
    Android:id="@+id/action_settings"
    Android:icon="@drawable/ic_menu_camera"
    Android:showAsAction="never"
    Android:title="@string/action_settings" />

<item
    Android:id="@+id/action_1"
    Android:icon="@drawable/ic_menu_gallery"
    Android:showAsAction="never"
    Android:title="Hello" />

<item
    Android:id="@+id/action_search"
    Android:icon="@Android:drawable/ic_search_category_default"
    Android:showAsAction="never"
    Android:title="action_search">

    <menu>
        <item
            Android:id="@+id/version1"
            Android:icon="@Android:drawable/ic_dialog_alert"
            Android:showAsAction="never"
            Android:title="Cup cake" />

        <item
            Android:id="@+id/version2"
            Android:icon="@drawable/ic_menu_camera"
            Android:showAsAction="never"
            Android:title="Donut" />


        <item
            Android:id="@+id/version3"
            Android:icon="@drawable/ic_menu_send"
            Android:showAsAction="never"
            Android:title="Eclair" />

        <item
            Android:id="@+id/version4"
            Android:icon="@drawable/ic_menu_gallery"
            Android:showAsAction="never"
            Android:title="Froyo" />
    </menu>
</item>
</menu> 
2
MashukKhan

Мой простой мод для отличного решения Саймона для использования с ActionMode:

 @Override
    public boolean onPrepareActionMode(ActionMode mode, Menu menu) {
        if(menu != null){
            if(menu.getClass().getSimpleName().equals("MenuBuilder")){
                try{
                    Method m = menu.getClass().getDeclaredMethod(
                            "setOptionalIconsVisible", Boolean.TYPE);
                    m.setAccessible(true);
                    m.invoke(menu, true);
                }
                catch(NoSuchMethodException e){
                    Log.e(TAG, "onPrepareActionMode", e);
                }
                catch(Exception e){
                    throw new RuntimeException(e);
                }
            }
        }
        return true;
    }
2
Kaamel

Это слишком поздно, но кто-то может помочь моей попытке, я получил помощь от ответа @Desmond Lua, который помогает тем, кто использует menu.xml

Мой ответ для динамического создания меню вот мой код:

  int ACTION_MENU_ID =1;
  SpannableStringBuilder builder;
   @Override
  public boolean onCreateOptionsMenu(Menu menu) {

  //this space for icon 
    builder = new SpannableStringBuilder("  " + getString(R.string.your_menu_title));
    builder.setSpan(new ImageSpan(this,  R.drawable.ic_your_menu_icon), 0, 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
  //dynamic menu added 
    menu.add(Menu.NONE,ACTION_MENU_ID, Menu.NONE,
             getString(R.string.your_menu_title))
            .setShowAsAction(MenuItemCompat.SHOW_AS_ACTION_WITH_TEXT | MenuItemCompat.SHOW_AS_ACTION_COLLAPSE_ACTION_VIEW);
  //set icon in overflow menu      
        menu.findItem(ACTION_MENU_ID).setTitle(builder);
}
1
MilapTank

Добавьте это в стиле:

<menu xmlns:Android="http://schemas.Android.com/apk/res/Android">
    <item
        Android:id="@+id/action_settings"
        app:showAsAction="always"
        Android:icon="@drawable/ic_more_vert_white"
        Android:orderInCategory="100"
        Android:title="">
        <menu>

            <item
                Android:id="@+id/Login"
                Android:icon="@drawable/ic_menu_user_icon"
                Android:showAsAction="collapseActionView|withText"
                Android:title="@string/str_Login" />

            <item
                Android:id="@+id/str_WishList"
                Android:icon="@drawable/ic_menu_wish_list_icon"
                Android:showAsAction="collapseActionView"
                Android:title="@string/str_WishList" />

            <item
                Android:id="@+id/TrackOrder"
                Android:icon="@drawable/ic_menu_my_order_icon"
                Android:showAsAction="collapseActionView"
                Android:title="@string/str_TrackOrder" />

            <item
                Android:id="@+id/Ratetheapp"
                Android:icon="@drawable/ic_menu_rate_the_apps"
                Android:showAsAction="collapseActionView"
                Android:title="@string/str_Ratetheapp" />

            <item
                Android:id="@+id/Sharetheapp"
                Android:icon="@drawable/ic_menu_shar_the_apps"
                Android:showAsAction="collapseActionView"
                Android:title="@string/str_Sharetheapp" />

            <item
                Android:id="@+id/Contactus"
                Android:icon="@drawable/ic_menu_contact"
                Android:showAsAction="collapseActionView"
                Android:title="@string/str_Contactus" />

            <item
                Android:id="@+id/Policies"
                Android:icon="@drawable/ic_menu_policy_icon"
                Android:showAsAction="collapseActionView"
                Android:title="@string/str_Policies" />
        </menu>
    </item>
</menu>
1
Dashrath Rathod
 public void showContextMenuIconVisible(Menu menu){
    if (menu.getClass().getSimpleName().equals("MenuBuilder")) {
        try {
            Field field = menu.getClass().getDeclaredField("mOptionalIconsVisible");
            field.setAccessible(true);
            field.setBoolean(menu, true);
        } catch (Exception ignored) {
            ignored.printStackTrace();
        }
    }
}
0
chry