导航抽屉项目点击监听器无法工作

13

很抱歉问这个愚蠢的问题,我是一名Android Studio的新手,正在学习中。 我已经尝试了很多次,但点击监听器无法工作,请帮忙。 我使用了Android Studio的默认抽屉布局。导航可以正常工作,但我想执行一个特殊操作,比如使用一个新意图来打开另一个应用程序。我正在尝试在ID nav_link上执行一个简单的Toast,但它不起作用。

package com.demo.navdraw;
import android.content.ClipData;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;

import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.appcompat.widget.Toolbar;
import androidx.drawerlayout.widget.DrawerLayout;
import androidx.navigation.NavController;
import androidx.navigation.Navigation;
import androidx.navigation.ui.AppBarConfiguration;
import androidx.navigation.ui.NavigationUI;
import android.widget.Toast;



import com.google.android.material.floatingactionbutton.FloatingActionButton;
import com.google.android.material.navigation.NavigationView;
import com.google.android.material.snackbar.Snackbar;

public class MainActivity extends AppCompatActivity implements NavigationView.OnNavigationItemSelectedListener {

private AppBarConfiguration mAppBarConfiguration;
private MenuItem item;

@Override
public boolean onCreateOptionsMenu(Menu menu) {

    getMenuInflater().inflate(R.menu.activity_main_drawer, menu);
    return true;
}

@Override
public boolean onOptionsItemSelected(MenuItem item) {
    int id = item.getItemId();


    if (id == R.id.nav_link) {
        Toast.makeText(this, "Setting", Toast.LENGTH_LONG).show();
        return true;
    }

    return super.onOptionsItemSelected(item);
}


@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    Toolbar toolbar=findViewById(R.id.toolbar);
    setSupportActionBar(toolbar);
    FloatingActionButton fab=findViewById(R.id.fab);



    fab.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View view) {
            Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG)
                    .setAction("Action", null).show();
        }
    });



    DrawerLayout drawer=findViewById(R.id.drawer_layout);
    NavigationView navigationView=findViewById(R.id.nav_view);
    navigationView.setNavigationItemSelectedListener(this);
    // Passing each menu ID as a set of Ids because each
    // menu should be considered as top level destinations.
    mAppBarConfiguration=new AppBarConfiguration.Builder(
            R.id.nav_home, R.id.nav_gallery, R.id.nav_slideshow,
            R.id.nav_tools, R.id.nav_share, R.id.nav_send, R.id.nav_profile, R.id.nav_link)
            .setDrawerLayout(drawer)
            .build();
    NavController navController=Navigation.findNavController(this, R.id.nav_host_fragment);
    NavigationUI.setupActionBarWithNavController(this, navController, mAppBarConfiguration);
    NavigationUI.setupWithNavController(navigationView, navController);


}

@Override
public boolean onSupportNavigateUp() {
    NavController navController=Navigation.findNavController(this, R.id.nav_host_fragment);
    return NavigationUI.navigateUp(navController, mAppBarConfiguration)
            || super.onSupportNavigateUp();
}


@Override
public boolean onNavigationItemSelected(@NonNull MenuItem Item) {
    int id=item.getItemId();

    if (id==R.id.nav_link){

            Toast.makeText(getApplicationContext(), "Link", Toast.LENGTH_LONG).show();
            return true;

        }
    return true;
    }


}

1
此答案可能有助于更清晰地理解并解决此问题:https://dev59.com/nbjoa4cB1Zd3GeqPCaqX#59761704 - Mohd Qasim
对于像我这样的人,请在此行代码后使用'navigationView.setNavigationItemSelectedListener(this);':'NavigationUI.setupWithNavController(navigationView,navController);' - KamDroid
4个回答

38

好的,你可以尝试这样做,但你需要手动完成一些操作,这就是你想要做的代价。以下是你需要做的:

  1. 删除 NavigationView.OnNavigationItemSelectedListener 的实现,它不是必需的。
  2. 在调用以下方法后:

NavigationUI.setupWithNavController(navigationView, navController);

插入此代码片段:

navigationView.setNavigationItemSelectedListener(new NavigationView.OnNavigationItemSelectedListener() {
            @Override
            public boolean onNavigationItemSelected(@NonNull MenuItem menuItem) {
                int id=menuItem.getItemId();
                //it's possible to do more actions on several items, if there is a large amount of items I prefer switch(){case} instead of if()
                if (id==R.id.nav_home){
                    Toast.makeText(getApplicationContext(), "Home", Toast.LENGTH_SHORT).show();
                }
                //This is for maintaining the behavior of the Navigation view
                NavigationUI.onNavDestinationSelected(menuItem,navController);
                //This is for closing the drawer after acting on it
                drawer.closeDrawer(GravityCompat.START);
                return true;
            }
        });

我已经为主页片段创建了自己的示例,但您也可以使用自己的“链接”片段。请注意变量名称可能会更改,但是思想是相同的。

我已经为首页片段创建了自己的示例,但是您可以使用自己的“链接”片段。请注意变量名称可能会更改,但思路相同。

太好了!我浪费了很多时间才明白在使用Navigation Drawer的Android Studio模板时进行了哪些更改。谢谢! - Mark Delphi

17

你可以采用两种方法来实现这个功能。
第一种方法
是在导航抽屉中仅为单个项实现监听器时使用 setOnMenuItemClickListener。这将为导航抽屉中的单个项添加一个监听器,而不影响其他导航项。

 val navigationView: NavigationView = findViewById(R.id.nav_view) as NavigationView

    navigationView.menu!!.findItem(R.id.nav_logout).setOnMenuItemClickListener { menuItem:MenuItem? ->
        //write your implementation here
        //to close the navigation drawer
        if (drawer_layout.isDrawerOpen(GravityCompat.START)) {
            drawer_layout.closeDrawer(GravityCompat.START)
        }
        Toast.makeText(applicationContext, "single item click listener implemented", Toast.LENGTH_SHORT).show()
        true
    }



第二种方法
是使用 setNavigationItemSelectedListener 来为导航抽屉中的每个项目编写监听器。

val navigationView: NavigationView = findViewById(R.id.nav_view) as NavigationView
 navigationView.setNavigationItemSelectedListener { menuItem ->
    when (menuItem.itemId) {
        R.id.nav_gallery -> {
            //write your implementation here
            if (drawer_layout.isDrawerOpen(GravityCompat.START)) {
                drawer_layout.closeDrawer(GravityCompat.START)
            }
            true
        }
        else -> {
            if (drawer_layout.isDrawerOpen(GravityCompat.START)) {
                drawer_layout.closeDrawer(GravityCompat.START)
            }
            false
        }
    }
}

5

这里有两种导航项单击的方式。

第一种方法: 如果你想在单击导航抽屉中的项目后显示片段,那么可以按照以下简单步骤操作。

  • 您必须创建一个导航图。有关更多信息,请单击此处

  • 您必须将菜单项和导航图中的片段的ID设为相同。

请参阅此图片: 菜单项ID

请参阅此图片: 导航图 - 片段ID

DrawerLayout drawer = binding.drawerLayout;
    NavigationView navigationView = binding.navView;
    // Passing each menu ID as a set of Ids because each
    // menu should be considered as top level destinations.
    mAppBarConfiguration = new AppBarConfiguration.Builder(
            R.id.home, R.id.gallery, R.id.nav_slideshow)
            .setOpenableLayout(drawer)
            .build();
    NavController navController = Navigation.findNavController(this, R.id.nav_host_fragment_content_main3);
    NavigationUI.setupActionBarWithNavController(this, navController, mAppBarConfiguration);
    NavigationUI.setupWithNavController(navigationView, navController);

当id相同时,您的导航视图将根据id处理导航项目的点击。

第二种方法:如果您想在点击导航抽屉项后显示自定义内容,则可以实现导航项点击监听器。

navigationView.setNavigationItemSelectedListener(item -> {
        if (item.getItemId()==R.id.home){
            // your code
            Toast.makeText(this, "Clicked..", Toast.LENGTH_SHORT).show();
            drawer.close();
            return true;
        }
        return false;
    });

0

对于 Kotlin,

   bottom_navigation.setOnItemSelectedListener {

        when(it.itemId) {
            R.id.page_1 -> {
                // Respond to navigation item 1 click
                Toast.makeText(this,"Home Clicked",Toast.LENGTH_SHORT).show()
                true
            }
            R.id.page_2 -> {
                // Respond to navigation item 2 click
                true
            }
            R.id.page_3 -> {
                // Respond to navigation item 2 click
                true
            }
            R.id.page_4 -> {
                // Respond to navigation item 2 click
                true
            }

            else -> false
        }
    }

OnNavigationItemSelectedListener已经过时了


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