动态填充项目时的导航抽屉组标题

时间:2019-03-14 21:25:59

标签: android navigation-drawer

我在我的应用程序中使用标准导航抽屉来显示两种菜单项:

  1. 高级常规项目,例如仪表板,详细信息视图等。

  2. 每个用户帐户的特定屏幕。

第二组项目需要动态填充,即我无法在XML中指定这些项目。我正在努力争取第二组的头衔。

这是我当前的menu.xml文件

<?xml version="1.0" encoding="utf-8"?>
<menu xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:title="General">
        <menu>
            <group
                android:id="@+id/main_group"
                android:orderInCategory="1">
                <item
                    android:id="@+id/nav_dashboard"
                    android:icon="@drawable/ic_dashboard"
                    android:title="@string/nav_dashboard"
                    android:checkable="true"
                    android:checked="true"/>
                <item
                    android:id="@+id/nav_all"
                    android:icon="@drawable/ic_list"
                    android:title="@string/nav_all"
                    android:checkable="true" />
            </group>

        </menu>
    </item>
    <item android:title="Accounts">
        <menu>
            <group
                android:id="@+id/accounts_group"
                android:orderInCategory="2">

            </group>
        </menu>
    </item>
</menu>

标题Accounts在我的导航抽屉中不可见。但是,如果我不是动态填充项目,而是在XML中放入一些随机项目,则它是可见的。

这是我填充菜单项的方式:

for (int index = 0; index < accountsList.size(); index++) {
    MainActivityPresenter.AccountInfo accountInfo = accountsList.get(index);
    menu.add(R.id.accounts_group, index + MENU_ITEM_ID, 1, accountInfo.name).setCheckable(true);
}

寻找提示。为什么我动态添加项目时没有显示项目标题Accounts?谢谢!

1 个答案:

答案 0 :(得分:1)

首先,您需要像下面那样更改菜单XML布局,这样将无法选择标题,而只能检查子项。

<menu xmlns:android="http://schemas.android.com/apk/res/android">
<item android:title="General"
    android:checkable="false"
    android:id="@+id/general_menu">
    <menu>
        <item
            android:id="@+id/nav_dashboard"
            android:icon="@drawable/ic_dashboard"
            android:title="@string/nav_dashboard"/>
        <item
            android:id="@+id/nav_all"
            android:icon="@drawable/ic_list"
            android:title="@string/nav_all"/>
    </menu>
</item>
<item android:title="Accounts"
    android:checkable="false"
    android:id="@+id/accounts_group">
    <menu>
        <!-- filled at runtime -->
    </menu>
</item>
</menu>

然后,在您的代码中,您需要访问正确的菜单item,对我们来说这就像一个“逻辑组”,您可以使用此代码来执行此操作:

//use the index of the group, in your case Account item is the 2nd in the
//XML this means its index is 1    
val menuItem: MenuItem = navigationView.menu.getItem(1)
//save the generated view Id, because if you need a reference after you need 
//to save newSubMenuId property somewhere before adding the item
val newSubMenuId = View.generateViewId()
//add the item to the submenu. The order value it's used to position the
//item inside the group. With 1 it will be the first item, 2 will be the second and so on
val addedItem = menuItem.subMenu.add(Menu.NONE, newSubMenuId, 1, R.string.new_account)
//add a nice icon to the inserted sub-item
addedItem.setIcon(R.drawable.ic_menu_account_icon)

现在,如果菜单XML中没有group,您将遇到问题,您将无法使用android:checkableBehavior="single"属性来阻止对NavigationView中的项目进行多选。

要解决最后一个问题,我们可以这样做:

val navigationView: NavigationView = findViewById(R.id.nav_view)
navigationView.setNavigationItemSelectedListener { selectedItem ->

    //loop trough all the main items (which are groups for us) to remove the checked state
    for (i in 0 until navigationView.menu.size()) {
        val menuItem = navigationView.menu.getItem(i)
        menuItem.isChecked = false
        if (menuItem.hasSubMenu()) {
            //if they have a subMenu, loop trough all the subitems
            //and remove the checked state from them too
            for (z in 0 until menuItem.subMenu.size()) {
                val subMenuItem = menuItem.subMenu.getItem(z)
                subMenuItem.isChecked = false
            }
        }
    }
    // set item as selected to persist highlight
    selectedItem.isChecked = true
    // close drawer when item is tapped
    drawerLayout.closeDrawers()

    // Add code here to update the UI based on the item selected
    // For example, swap UI fragments here

    true
}

所有代码都是用Kotlin编写的,但是我认为如果仍然使用它,将其转换为Java不会有太多麻烦。顺便说一句,如果您有问题,只问:)

  

PS:创建广告时无需指定.setCheckable(true)   新物品。默认情况下可以检查新项目