Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to create a Menu instance programmatically? i.e. inflate a Menu outside onCreateOptionsMenu

I want to inflate a menu object outside onCreateOptionsMenu method (which means to create/show the menu when the user doesn't press the button), so I need to create a menu instance to pass it to the inflate method.

Here is an example of what I am trying to achieve:

Menu menu = // How to create an instance !? 
new MenuInflater(context).inflate(R.menu.my_menu, menu)

Menu is an interface, so I need to know which class is implementing it. I did browse Android code to get any hint on how a Menu object is created, but still could not find what I am looking for.

Edit 1

My goal is to fire an onOptionsItemSelected(MenuItem item) event from a custom view, which will be handled by the activity, so I need to have a MenuItem object with specific itemId and title to pass it with the event.

If I can successfully create a Menu object, it will be easy to get its children MenuItems.

Edit 2

I am not trying to display a menu at all, what I want is to populate a ListView with elements defined in a menu XML that have title, icon and itemId and whenever a ListViewItem is clicked I want to fire a onOptionsItemSelected(MenuItem item) event that is handled in my activity.

I know that I can parse the menu XML to extract items information, however I will not be able to fire onOptionsItemSelected(MenuItem item) without creating a standard MenuItem object to pass it as argument.

Any help will be appreciated. Thanks!

like image 226
iTech Avatar asked Jan 02 '13 08:01

iTech


3 Answers

Here's a trick to get an instance of Menu:

PopupMenu p  = new PopupMenu(getContext(), null);
Menu menu = p.getMenu();
like image 138
Musma Avatar answered Oct 23 '22 05:10

Musma


I found two solutions to programmatically create a Menu instance and inflate it:

  • Using ActionbarSherlock library or AppCompat v7 library Menu menu = new MenuBuilder(context); or you can write your own MenuBuilder class

  • Using standard android SDK:

// Creating an instance by reflection

Menu menu = newMenuInstance(context);


protected Menu newMenuInstance(Context context) {
    try {
        Class<?> menuBuilderClass = Class.forName("com.android.internal.view.menu.MenuBuilder");

        Constructor<?> constructor = menuBuilderClass.getDeclaredConstructor(Context.class);

        return (Menu) constructor.newInstance(context);

    } catch (Exception e) {e.printStackTrace();}

    return null;
}

Once you have a Menu instance you can easily inflate it from a menu XML resource anywhere in your program

new MenuInflater(context).inflate(menuId, menu);

I tested both methods and they are working perfectly, I would recommend using the second method with the standard Menu and MenuItem classes from android SDK even if your activity extends SherlockActivity because it will still receive onOptionsItemSelected(MenuItem item) regardless if you fire it with android.view.MenuItem or com.actionbarsherlock.view.MenuItem

like image 29
iTech Avatar answered Oct 23 '22 05:10

iTech


I'm not sure why this isn't an answer already, and I know this is an old question, but for future readers..

If you simply do this:

val menu = MenuBuilder(context)
MenuInflater(context).inflate(R.menu.menu_XXXX, menu)

It works!

androidx.appcompat.view.menu.MenuBuilder implements android.view.Menu. Upon inspection, that's all that PopupMenu does.

Note that the com.android.internal.view.menu.MenuBuilder mentioned by @iTech and used by PopupMenu is not public and should not be used.

Here are two helper functions and a usage example:

fun Context.inflateMenu(@MenuRes menuRes: Int): Lazy<MenuBuilder> = lazy {
    MenuBuilder(this)
        .also { MenuInflater(this).inflate(menuRes, it) }
}

fun Fragment.inflateMenu(@MenuRes menuRes: Int): Lazy<MenuBuilder> = lazy {
    MenuBuilder(context)
        .also { MenuInflater(context).inflate(menuRes, it) }
}

Usage:

Activity

class MyActivity : AppCompatActivity(R.layout.activity_my) {

    val menu by inflateMenu(R.menu.menu_my)

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)

        ...
    }
}

Fragment:

class MyFragment : Fragment(R.layout.fragment_my) {

    val menu by inflateMenu(R.menu.menu_my)

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)

        ...
    }
}
like image 6
Joshua King Avatar answered Oct 23 '22 05:10

Joshua King