Toolbar动态改变menu

关于Toolbar的使用网上有很多介绍了,这里不赘述了。

1.Toolbar主题

一般使用系统的主题ThemeOverlay.AppCompat.Dark.ActionBar。也可以自定义主题:


<style name="ToolbarTheme" parent="@style/ThemeOverlay.AppCompat.Dark.ActionBar">
<item name="titleTextColor">@color/colorAccent</item>   <!--标题颜色-->
<item name="actionMenuTextColor">@color/white</item> <!-- 菜单颜色-->
</style>

这样我们可以控制标题的颜色和菜单的颜色,当然还有其他的标签…

2.动态改变menu

可能会有这样的需求,Toolbar有个确定的menu,选择条目时这个menu会显示当前选择数量,如:确定(3).

  • 1.不使用menu,在Toolbar中添加一个TextView控件

这种方式虽然也能实现,但是相当于只是把Toolbar当做一个ViewGroup使用了,我之前也是这样做的,哈哈哈

  • 2.在onPrepareOptionsMenu方法中控制

onPrepareOptionsMenu这个方法在创建菜单时(onCreateOptionMenu)会调用一次.

这里写图片描述

比如我们现在就实现 “确定(3)”这种和动态设置显示和隐藏:


@Override
public boolean onPrepareOptionsMenu(Menu menu) {
    Log.e(TAG, "onPrepareOptionsMenu");
    if(isSave){
        MenuItem save = menu.findItem(R.id.menu_save);
        save.setVisible(true);
        String title = count == 0 ? "保存" : String.format("保存(%d)", count);
        save.setTitle(title);
        menu.findItem(R.id.menu_share).setVisible(false);
    } else {
        menu.findItem(R.id.menu_save).setVisible(false);
        MenuItem share = menu.findItem(R.id.menu_share);
        share.setVisible(true);
        String title = count == 0 ? "分享" : String.format("分享(%d)", count);
        share.setTitle(title);
    }
    return super.onPrepareOptionsMenu(menu);
}

然后在数量改变的时候调用该方法?当然不是。系统提供给我们有两个方法可以间接调用该方法:
supportInvalidateOptionMenu()和invalidateOptionMenu()

如:

    public void increase(View view) {
        count++;
        supportInvalidateOptionsMenu();
    }

效果图:
onPrepareOptionsMenu

  • 3.自定义ActionProvider方法自定义菜单

这种方式也可以动态改变menu,而且布局没有限制。

menu文件:通过actionProviderClass指定Provider


<item
    android:actionLayout="@+id/menu_custom"
    android:title="自定义"
    android:id="@+id/menu_custom"
    app:actionProviderClass="com.beiing.toolbar_menu.MainActionProvider"
    app:showAsAction="always"
    />

菜单布局:background设置为actionBarItemBackground,这样点击时会有涟漪效果


<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:gravity="center"
    android:background="?actionBarItemBackground">

    <TextView
        android:id="@+id/tv_custom_menu"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="自定义"
        android:textColor="@color/white"/>

    <View
        android:layout_width="match_parent"
        android:layout_marginTop="2dp"
        android:layout_height="1dp"
        android:background="@color/white"/>

</LinearLayout>

Provider类:提供了settext方法动态改变menu文本

public class MainActionProvider extends ActionProvider {

    LinearLayout customMenu;

    TextView tvTitle;

    View.OnClickListener clickListener;

    /**
     * Creates a new instance.
     *
     * @param context Context for accessing resources.
     */
    public MainActionProvider(Context context) {
        super(context);
    }

    @Override
    public View onCreateActionView() {
        Log.e("toolbar-menu", "onCreateActionView");
        customMenu = (LinearLayout) LayoutInflater.from(getContext()).inflate(R.layout.layout_custom_menu, null);
        tvTitle = (TextView) customMenu.findViewById(R.id.tv_custom_menu);
        customMenu.setOnClickListener(clickListener);


 //return null 时才会调用onPrepareSubMenu方法
//        return null;

        return customMenu;
    }


    @Override
    public void onPrepareSubMenu(SubMenu subMenu) {

        subMenu.clear();
        subMenu.add("子菜单1").setIcon(R.mipmap.ic_launcher)
                .setOnMenuItemClickListener(new MenuItem.OnMenuItemClickListener() {
                    @Override
                    public boolean onMenuItemClick(MenuItem item) {
                        return true;
                    }
                });
        subMenu.add("子菜单2").setIcon(R.mipmap.ic_launcher)
                .setOnMenuItemClickListener(new MenuItem.OnMenuItemClickListener() {
                    @Override
                    public boolean onMenuItemClick(MenuItem item) {
                        return true;
                    }
                });

    }

    @Override
    public boolean hasSubMenu() {
        return true;
    }

    public void setonClickListener(View.OnClickListener listener){
        clickListener = listener;
    }

    public void setTvTitle(String title){
        tvTitle.setText(title);
    }

}

Activity中获取Provider:

 @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        Log.e(TAG, "onCreateOptionsMenu");
        getMenuInflater().inflate(R.menu.share_and_save, menu);

        MenuItem customItem = menu.findItem(R.id.menu_custom);
        MainActionProvider actionProvider = (MainActionProvider) MenuItemCompat.getActionProvider(customItem);
        actionProvider.setonClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Toast.makeText(MainActivity.this, "自定义菜单", Toast.LENGTH_SHORT).show();
            }
        });
        return super.onCreateOptionsMenu(menu);
}

效果图:
效果图

第一种方式对于菜单显示样式有限制,一般情况可满足。第二种方式使用自己的布局,基本可以满足各种需要了。

3.Extra

添加子菜单

  • 1.menu文件
<item android:id="@+id/action_more"
    android:icon="@drawable/abc_ic_menu_moreoverflow_mtrl_alpha"
    android:title="more"
    app:showAsAction="ifRoom">
    <menu>
        <item
            android:id="@+id/setting"
            android:title="设置" />
        <item
            android:id="@+id/about"
            android:title="关于" />
    </menu>
</item>
  • 2.Provider方式

重写onPrepareSubMenu和hasSubMenu方法,代码见上文,需要注意的是:

在onCreateActionView中返回自定义的菜单view时,onPrepareSubMenu是不会执行的,点击该菜单不会出现子菜单列表,因为只有返回null时onPrepareSubMenu才会调用。

当然,这种方式添加子菜单的方式太麻烦了,在xml文件里就可以搞定。

4.源码

https://github.com/LineChen/ToolBar-menu

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值