【问题标题】:How to make option menu appear on bottom of the screen?如何使选项菜单出现在屏幕底部?
【发布时间】:2017-06-21 12:46:19
【问题描述】:

这是我想要实现的:当人们点击工具栏右上角的菜单时,屏幕底部会出现一个选项菜单。见下图:

我不确定我应该为底部的项目调用什么方法。有人可以给我一些关于如何实现它的提示吗?

我用下面的代码成功实现了右上角菜单栏中的图标。但我不知道如何在屏幕底部显示选项,宽度match_parent,高度wrap_content

点击右上角

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.edit_profile_image_menu, menu);
        return true;
    }

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

        switch (id) {
            case R.id.more:
                //How to show the 2 option in the bottom of screen here


                return true;
        }

        return super.onOptionsItemSelected(item);
    }

更新

执行Nikesh的代码后,弹窗如下:

【问题讨论】:

  • 您可以使用可移动按钮并从中弹出一个菜单。
  • 一个从底部开始的弹出菜单??你能解释一下吗??
  • 您需要的是BottomSheet。阅读本教程:code.tutsplus.com/articles/…
  • 你也可以通过BottomSheetDialogFragment得到你想要的样子

标签: java android android-layout android-xml android-menu


【解决方案1】:

更新答案

您现在可以使用Material Design BottomAppBar

  • 底部应用栏在移动屏幕底部显示导航和关键操作

示例代码

在您的 build.gradle

中添加以下依赖项
implementation 'com.google.android.material:material:1.0.0'

现在在res/menu目录下创建3个菜单文件

bottom_menu.xml

<?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"
    xmlns:tools="http://schemas.android.com/tools">

    <item
        android:id="@+id/app_bar_search"
        android:icon="@drawable/ic_search"
        android:title="Search"
        app:showAsAction="ifRoom" />
    <item
        android:id="@+id/app_bar_fav"
        android:icon="@drawable/ic_favorite"
        android:title="Favorite"
        app:showAsAction="ifRoom"/>
    <item
        android:icon="@drawable/ic_favorite"
        android:title="Favorite"
        app:showAsAction=""/>
</menu>

nav_drawer_menu.xml

<?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/navItemOne"
        android:title="Nav Item 1"
        app:showAsAction="never"/>

    <item
        android:id="@+id/navItemTwo"
        android:title="Nav Item 2"
        app:showAsAction="never"/>

    <item
        android:id="@+id/navItemThree"
        android:title="Nav Item 3"
        app:showAsAction="never"/>
</menu>

toolbar_menu.xml

<?xml version="1.0" encoding="utf-8"?>
<menu xmlns:tools="http://schemas.android.com/tools"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:android="http://schemas.android.com/apk/res/android">

    <item
        android:id="@+id/app_bar_fav"
        android:icon="@drawable/ic_favorite"
        android:title="Favorite"
        app:showAsAction="ifRoom"/>
    <item
        android:id="@+id/app_bar_search"
        android:icon="@drawable/ic_search"
        android:title="Search"
        app:showAsAction="ifRoom"/>
    <item
        android:id="@+id/app_bar_settings"
        android:title="Settings"
        app:showAsAction="never"/>

    <item
        android:title="Menu Item 1"
        app:showAsAction="never"/>

    <item
        android:title="Menu Item 2"
        app:showAsAction="never"/>
    <item
        android:title="Menu Item 3"
        app:showAsAction="never"/>

</menu>

现在创建一个类名 BottomNavigationDrawerFragment 以从底部打开导航抽屉

import android.content.Context
import android.os.Bundle
import android.view.Gravity
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.widget.Toast
import com.google.android.material.bottomsheet.BottomSheetDialogFragment
import kotlinx.android.synthetic.main.bottom_nav_layout.*

class BottomNavigationDrawerFragment : BottomSheetDialogFragment() {

    override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
        return inflater.inflate(R.layout.bottom_nav_layout, container, false)
    }
    override fun onActivityCreated(savedInstanceState: Bundle?) {
        super.onActivityCreated(savedInstanceState)

        navigation_view.setNavigationItemSelectedListener { menuItem ->
            // Bottom Navigation Drawer menu item clicks
            when (menuItem!!.itemId) {
                R.id.navItemOne -> context!!.toast(" Nav item one is Clicked ")
                R.id.navItemTwo -> context!!.toast(" Nav item Two is Clicked ")
                R.id.navItemThree -> context!!.toast(" Nav item Three is Clicked ")
            }
            // Add code here to update the UI based on the item selected
            // For example, swap UI fragments here
            true
        }
    }

    // This is an extension method for easy Toast call
    fun Context.toast(message: CharSequence) {
        val toast = Toast.makeText(this, message, Toast.LENGTH_SHORT)
        toast.setGravity(Gravity.BOTTOM, 0, 600)
        toast.show()
    }
}

MainActivity代码

import android.os.Bundle
import android.view.Menu
import android.view.MenuItem
import android.widget.Toast
import androidx.appcompat.app.AppCompatActivity
import kotlinx.android.synthetic.main.activity_main.*

class MainActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        //  attach menu to your BottomAppBar
        bottomBar.replaceMenu(R.menu.bottom_menu)

        // handle click event of navigationIcon
        bottomBar.setNavigationOnClickListener {
            toast("Navigation Icon Clicked")
            val bottomNavDrawerFragment = BottomNavigationDrawerFragment()
            bottomNavDrawerFragment.show(supportFragmentManager, bottomNavDrawerFragment.tag)
        }

        // handle click event of FloatingActionButton
        fab.setOnClickListener {
            toast("Fab Icon Clicked")

        }

        //handle click event of menu of BottomAppBar
        bottomBar.setOnMenuItemClickListener { menuItem ->

            when (menuItem!!.itemId) {
                R.id.app_bar_search -> toast("Search menu of bottomBar is clicked!")
            }

            true
        }
    }

    // Overriding Actionbar menu
    override fun onCreateOptionsMenu(menu: Menu): Boolean {
        val inflater = menuInflater
        inflater.inflate(R.menu.toolbar_menu, menu)
        return true
    }


    //handle click event of menu of Actionbar
    override fun onOptionsItemSelected(item: MenuItem?): Boolean {
        when (item!!.itemId) {
            R.id.app_bar_fav -> toast("Fav menu item of toolbar is clicked!")
            R.id.app_bar_search -> toast("Search menu item of toolbar is clicked!")
            R.id.app_bar_settings -> toast("Settings item  of toolbar is clicked!")
            else -> toast("Menu item  of toolbar  is clicked!")
        }

        return true
    }

    // method to display toast
    private fun toast(message: String) {
        Toast.makeText(this, message, Toast.LENGTH_SHORT).show()
    }
}

layout.activity_main

<?xml version="1.0" encoding="utf-8"?>
<androidx.coordinatorlayout.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:id="@+id/main_content"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="#ffffff"
    android:fitsSystemWindows="true">

    <com.google.android.material.bottomappbar.BottomAppBar
        android:id="@+id/bottomBar"
        style="@style/Widget.MaterialComponents.BottomAppBar"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_gravity="bottom"
        app:fabCradleMargin="10dp"
        app:fabCradleVerticalOffset="4dp"
        android:backgroundTint="@color/colorPrimary"
        app:fabAlignmentMode="center"
        app:navigationIcon="@drawable/ic_drawer"/>

    <com.google.android.material.floatingactionbutton.FloatingActionButton
        android:id="@+id/fab"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:layout_anchor="@id/bottomBar"
        app:srcCompat="@drawable/ic_apps" />

</androidx.coordinatorlayout.widget.CoordinatorLayout>

bottom_nav_layout.xml

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content">


    <com.google.android.material.navigation.NavigationView
        android:id="@+id/navigation_view"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_gravity="bottom"
        android:layout_marginStart="8dp"
        android:layout_marginTop="8dp"
        android:layout_marginEnd="8dp"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        app:menu="@menu/nav_drawer_menu"/>

</androidx.constraintlayout.widget.ConstraintLayout>

JAVA 爱好者的示例代码

BottomNavigationDrawerFragment

import android.content.Context;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.MenuItem;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Toast;
import com.google.android.material.bottomsheet.BottomSheetDialogFragment;
import com.google.android.material.navigation.NavigationView;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import neel.com.bottomappbar.R;

public class BottomNavigationDrawerFragment extends BottomSheetDialogFragment {

    private Context mContext;

    @Override
    public void onAttach(Context context) {
        super.onAttach(context);
        mContext=context;
    }

    @Nullable
    @Override
    public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {

        View rootView= inflater.inflate(R.layout.bottom_nav_layout, container, false);

        NavigationView navigationView=rootView.findViewById(R.id.navigation_view);

        navigationView.setNavigationItemSelectedListener(new NavigationView.OnNavigationItemSelectedListener() {
            @Override
            public boolean onNavigationItemSelected(@NonNull MenuItem menuItem) {

                switch (menuItem.getItemId()){
                    case R.id.navItemOne:
                        Toast.makeText(mContext, "Nav item One is Clicked ", Toast.LENGTH_SHORT).show();
                        return true;
                    case R.id.navItemTwo:
                        Toast.makeText(mContext, "Nav item Two is Clicked ", Toast.LENGTH_SHORT).show();
                        return true;
                    case R.id.navItemThree:
                        Toast.makeText(mContext, "Nav item Three is Clicked ", Toast.LENGTH_SHORT).show();
                        return true;
                }
                return false;
            }
        });

        return rootView;
    }

    @Override
    public void onActivityCreated(@Nullable Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);
    }
}

主活动

import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.Toast;
import com.google.android.material.bottomappbar.BottomAppBar;
import com.google.android.material.floatingactionbutton.FloatingActionButton;
import androidx.annotation.Nullable;
import androidx.appcompat.app.AppCompatActivity;
import androidx.appcompat.widget.Toolbar;
import neel.com.bottomappbar.R;

public class MainActivity extends AppCompatActivity {

    BottomAppBar bottomAppBar;
    FloatingActionButton floatingActionButton;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        bottomAppBar = findViewById(R.id.bottomBar);

        //  attach menu to your BottomAppBar
        bottomAppBar.replaceMenu(R.menu.bottom_menu);

        // handle click event of navigationIcon of bottomAppBar
        bottomAppBar.setNavigationOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                BottomNavigationDrawerFragment bottomNavigationDrawerFragment = new BottomNavigationDrawerFragment();
                bottomNavigationDrawerFragment.show(getSupportFragmentManager(), bottomNavigationDrawerFragment.getTag());
            }
        });

        //handle click event of menu of BottomAppBar
        bottomAppBar.setOnMenuItemClickListener(new Toolbar.OnMenuItemClickListener() {
            @Override
            public boolean onMenuItemClick(MenuItem item) {
                switch (item.getItemId()) {
                    case R.id.app_bar_search:
                        Toast.makeText(MainActivity.this, "Search menu of bottomBar is clicked!", Toast.LENGTH_SHORT).show();
                        return true;
                    case R.id.app_bar_fav:
                        Toast.makeText(MainActivity.this, "Favorite menu of bottomBar is clicked!", Toast.LENGTH_SHORT).show();
                        return true;

                }
                return false;
            }
        });

        floatingActionButton.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Toast.makeText(MainActivity.this, "Fab Is clicked ", Toast.LENGTH_SHORT).show();
            }
        });
    }

    // Overriding Actionbar menu
    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        getMenuInflater().inflate(R.menu.toolbar_menu, menu);
        return true;
    }

    //handle click event of menu of Actionbar
    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        switch (item.getItemId()) {
            case R.id.app_bar_fav:
                Toast.makeText(MainActivity.this, "Fav menu item of toolbar is clicked!", Toast.LENGTH_SHORT).show();
                return true;
            case R.id.app_bar_search:
                Toast.makeText(MainActivity.this, "Search menu item of toolbar is clicked!", Toast.LENGTH_SHORT).show();
                return true;
            case R.id.app_bar_settings:
                Toast.makeText(MainActivity.this, "app_bar_settings", Toast.LENGTH_SHORT).show();
                return true;

        }
        return false;
    }
}

更多信息请查看以下文章

输出

https://www.youtube.com/watch?v=k145bGLrleo

旧答案

尝试将此PopupMenu 附加到您的底部视图单击事件中,例如按钮单击事件

第 1 步 像这样在布局的底部创建一个视图

<View
android:layout_gravity="center"
android:id="@+id/myView"
android:layout_width="match_parent"
android:layout_height="wrap_content"/>

第 2 步 像这样创建一个 new_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/temp"
    android:title="@string/str_menu_account_logout"
    android:icon="@drawable/next"
    app:showAsAction="ifRoom"></item>

</menu>

现在添加此代码以在您的 java 文件中创建选项菜单

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    // Inflate the menu; this adds items to the action bar if it is present.
    getMenuInflater().inflate(R.menu.new_menu, menu);
    return true;
}

@Override
public boolean onOptionsItemSelected(MenuItem item) {

    if (item.getItemId() == R.id.temp) {
        PopupMenu popupMenu = new PopupMenu(this, findViewById(R.id.myView),Gravity.CENTER);
        popupMenu.inflate(R.menu.home_menu);
        popupMenu.show();
        return true;
    }

    return false;

}

【讨论】:

  • menu.xml 文件是放“从相册中选择”和“保存到相册”的地方??
  • 您必须执行“从相册中选择”或“保存到相册”的点击事件才能显示弹出菜单
  • 不,我希望人们点击右上角的图标,然后在底部显示“从相册中选择”和“保存到手机”的菜单
  • 在菜单中放置“从相册中选择”和“保存到相册”的地方
  • 我试过了,但是当我点击右上角的图标时,菜单没有显示在底部
【解决方案2】:

我建议使用Material design App Bar(Bottom),稍加修改即可获得效果。

试试这个https://material.io/design/components/app-bars-bottom.html

希望这会有所帮助。

【讨论】:

    【解决方案3】:

    简单。 使用自己的内容创建一个对话框,该对话框将构成底部菜单。 当用户在上面点击时,对话框应该会显示,但它的位置会在下面,即固定在底部。您可以找到许多关于如何在锚定的碎石底部定位对话框的代码!所以不添加任何内容。希望对您有所帮助。!

    【讨论】:

      【解决方案4】:

      @Nilesh 的回答非常详尽且有点复杂。

      简单的解决方案

      1. 根据您的要求创建 FrameLayout 并在框架布局内排列按钮。设置 id -- 我用 id = "menu"

      2. 将可见性设置为gone android:visibility = "gone"

      3. 点击菜单按钮

        findViewById(R.id.menu).setVisibility(View.VISIBLE);

      4. 现在要使菜单在您在菜单外单击时消失,请在菜单后面添加一个空视图或布局。设置 ontouch 监听器。触摸背景视图时设置可见性 GONE。

      【讨论】:

      • 我认为这很简单,充分证明,不需要我添加完整的代码。但是,如果您需要工作代码,请在 cmets 中告诉我。
      • 这种方法的另一个好处是它可以完全定制。
      【解决方案5】:

      没有必要重新发明轮子;使用BottomNavigationView,由以下提供:

      dependencies {
          api "com.google.android.material:material:1.0.0"
      }
      

      可以像这样添加到布局中(默认设置gone):

       <android.support.design.widget.BottomNavigationView
           android:id="@+id/navigation"
           android:layout_width="match_parent"
           android:layout_height="56dp"
           android:layout_gravity="start"
           android:visiblity="gone"
           app:menu="@menu/bottom" />
      

      menu/bottom.xml:

      <menu
          xmlns:android="http://schemas.android.com/apk/res/android">
      
          <item android:id="@+id/action_search"
              android:title="@string/menu_search"
              android:icon="@drawable/ic_search" />
      
          <item android:id="@+id/action_settings"
              android:title="@string/menu_settings"
              android:icon="@drawable/ic_add" />
      
      </menu>
      

      然后挂钩到onMenuOpened() 事件:

      @Override
      public boolean onMenuOpened(int featureId, Menu menu) {
          this.mBottomNavigationView.setVisibility(View.VISIBLE);
          return super.onMenuOpened(featureId, menu);
      }
      

      同时实现方法onOptionsItemSelected():

      @Override
      public boolean onOptionsItemSelected(MenuItem item) {
          switch (item.getItemId()) {
      
              case R.id.action_search:
                  ...
                  this.hideBottomNavigation();
                  return true;
      
              case R.id.action_settings:
                  ...
                  this.hideBottomNavigation();
                  return true;
      
              default:
                  this.hideBottomNavigation();
                  return super.onOptionsItemSelected(item);
          }
      }
      
      private void hideBottomNavigation() {
          this.mBottomNavigationView.setVisibility(View.GONE);
      }
      

      BottomSheetDialogFragment 确实是另一个可用的选项,虽然它不支持对menu 进行膨胀,但对fragment 进行膨胀......尽管如此,它只是相同的方法。

      【讨论】:

        【解决方案6】:

        您现在可以使用 Material Design BottomAppBar

        首先将其添加到所有正在实施的 build.gradle 中

        implementation 'com.google.android.material:material:1.0.0'
        

        创建菜单 anyname.xml

        第二次添加协调器布局并在其中底部AppBaradd menu and navigation as given here

        第三次通过初始化添加代码

        private var mBottomAppBar: BottomAppBar ?= null
        

        然后将它们添加到方法中

        //bottom bar
                mBottomAppBar = findViewById(R.id.bar)
                mBottomAppBar?.setOnMenuItemClickListener{
                    when(it.itemId){
                        R.id.itemConfig->showConfigBottomSheetDialogFragment()
        
                    }
                    true
                }
                mBottomAppBar?.setNavigationOnClickListener{
                        startContentHighlightActivity()
                }
        
            }
        

        导航将直接起作用,菜单将根据您要添加的项目起作用。

        【讨论】:

          【解决方案7】:
             //This is click event of button 
          @Override
          public void onClick(View view) {
          
              switch (view.getId()){
                  case R.id.buttonSort:
                      PopupMenu popupMenu = new PopupMenu(this, findViewById(R.id.myView),Gravity.CENTER);
                      popupMenu.inflate(R.menu.bottom_menu);
                      popupMenu.show();
                      break;
          }
          

          这是我的 home_menu.xml

              <?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/bottom_menu_price"
              android:title="Price low to high"
              android:icon="@drawable/ic_price_low"
          
              />
          <item
              android:id="@+id/bottom_menu_price_high"
              android:title="Price high to low"
              android:icon="@drawable/ic_price_high"
          
              />
          <item
              android:id="@+id/bottom_menu_date"
              android:title="New Added"
              android:icon="@drawable/ic_date_sort"
          
              />
          <item
              android:id="@+id/bottom_menu_rate"
              android:title="Best Rated"
              android:icon="@drawable/ic_rate_sort"
          
              />
          

          【讨论】:

          • 虽然这段代码可能会回答这个问题,但如果你能包含一些关于它在做什么以及它为什么会解决问题的解释会更好。阅读How do I write a good answer? 以了解什么是有用的答案是个好主意。
          猜你喜欢
          • 1970-01-01
          • 1970-01-01
          • 2013-10-17
          • 2017-10-22
          • 1970-01-01
          • 2021-07-12
          • 2013-07-14
          • 2019-01-25
          • 2016-06-01
          相关资源
          最近更新 更多