Cross Platform C#
Standardized Navigation Principles for Android App Development
Navigation in mobile devices is an important consideration. If a user isn't able to navigate an app, he might quickly abandon it. Learn the navigation principles of Android to keep this from happening to you.
Mobile applications have a number of screens (Activities and Fragments) associated with them. Navigating through these is an important mechanism. There are a number of ways to provide standardized navigation in Android. This article will look at three mechanisms to provide navigation:
- Menus: Menus provide a common UI component for an application. The Menu APIs are used to present actions, as well as other options within an application.
- Navigation Drawer: The navigation drawer is a panel shown from the side of the display; it provides the user with the application's main navigation.
- Action Bar: The action bar provides a familiar navigation mechanism to the user across the top of the screen via an app icon, action items and action overflow.
Menus
Android has several types of menus, including:
- Options Menu: The primary set of items for an activity. This is where a user can perform actions such as Compose E-mail, User Settings, or in the case of the sample app presented in this article, which pet you want to load.
- Context Menu: Allows for a menu to be created that's associated with a view.
To create an options menu, two code methods must be created: OnCreateOptionsMenu and OnOptionsItemSelected. The OnCreateOptionsMenu method creates a menu. In the code sample in Listing 1, the menu is loaded from a resource .xml file. There are additional methods to create menus programmatically via the Menu.Add method. The OnCreateOptionsMenu is called several ways:
- In Android 2.3 and earlier, OnCreateOptionsMenu is called when a user touches the physical menu key on a device.
- In Android 3.0 and later, OnCreateOptionsMenu is called when a user touches the soft menu key in the action bar.
The OnOptionsItemSelected method is called when a user selects a menu item. In Listing 1, a basic switch statement is used to determine the menu item selected.
Listing 1: The OnCreateOptionsMenu Method
public override bool OnCreateOptionsMenu (IMenu menu)
{
MenuInflater inflater = this.MenuInflater;
inflater.Inflate(Resource.Menu.examplemenu, menu);
return true;
}
public override bool OnOptionsItemSelected (IMenuItem item)
{
switch (item.ItemId) {
case Resource.Id.catexample:
Android.Util.Log.Info (this.Application.PackageName, "Cat Example Selected");
ic.SetImageResource (Resource.Drawable.wells);
break;
case Resource.Id.dogexample:
Android.Util.Log.Info (this.Application.PackageName, "Dog Example Selected");
ic.SetImageResource (Resource.Drawable.dog);
break;
default:
break;
}
return base.OnOptionsItemSelected (item);
}
In Listing 2, the XML is loaded from the Resource/Menu directory.
Listing 2: Content for Creating a Menu with XML
<?xml version="1.0" encoding="utf-8"?>
<menu xmlns:android="http://schemas.android.com/apk/res/android">
<item android:id="@+id/dogexample"
android:title="@string/thedog" />
<item android:id="@+id/catexample"
android:icon="@drawable/wellssmall"
android:title="@string/thecat" />
</menu>
The key tags and properties in Listing 2 are:
- item. An <item> tag defines an item in a menu.
- android:id. An android:id property defines the id of a menu item. This property will be used inside the software code to determine the menu item selected.
- android:icon. An android:icon property defines the image/icon used when the menu item's displayed. In the code sample in Listing 2, the image is pulled from the drawable resources.
- android:title. An android:title defines the text displayed when the menu is activated. In the example code shown in Listing 2, the text is pulled from the values resource; specifically, from the string.xml file.
Figure 1 shows the menu displayed in an application running as an Android 2.3 application. Notice that the menu is displayed at the bottom, and the icon is shown.
[Click on image for larger view.]
Figure 1. The menu for an Android 2.3 application. Notice that it's displayed at the bottom, and the icon is shown.
Figure 2 shows the same code running as an Android 4.0 application. Notice that the menu is displayed from the top of the image in the android action bar.
[Click on image for larger view.]
Figure 2. The same code from Figure 1, running as an Android 4.0 application. Notice the menu is displayed from the top of the image in the Android action bar.
Context Menu
A contextual menu allows for actions performed against a view. The process:
- The view must be registered for a context menu. This is done via a call to RegisterForContextMenu(view).
- Override the OnCreateContextMenu method in an activity or fragment.
- Override the OnContextItemSelected method.
Listing 3 shows the setup and processing of a context menu in an imageview. The key details:
- The call to RegisterForContextMenu and passing the view to the method. This registers the example imageview for context menu operations.
- The OnCreateContextMenu is particular for a given activity or fragment. Because of this, it's important to check and make sure the view passed in is appropriate for a given context menu request.
- The OnContextItemSelected is the same as the OnOptionsItemSelected shown previously. A switch command is used to determine the selected menu item.
Listing 3: Setup and Processing a Context Menu
protected override void OnCreate (Bundle bundle)
{
base.OnCreate (bundle);
SetContentView (Resource.Layout.Main);
ic = FindViewById<ImageView> (Resource.Id.iv);
ic.SetImageResource (Resource.Drawable.wells);
RegisterForContextMenu (ic);
}
public override void OnCreateContextMenu (IContextMenu menu, View v, IContextMenuContextMenuInfo menuInfo)
{
MenuInflater inflater = this.MenuInflater;
if (v.Id == ic.Id) {
inflater.Inflate (Resource.Menu.catmenu, menu);
}
base.OnCreateContextMenu (menu, v, menuInfo);
}
public override bool OnContextItemSelected (IMenuItem item)
{
switch (item.ItemId) {
case Resource.Id.meow:
Android.Util.Log.Info (this.Application.PackageName, "Meow!");
break;
case Resource.Id.eat:
Android.Util.Log.Info (this.Application.PackageName, "Eat");
break;
case Resource.Id.shed:
Android.Util.Log.Info (this.Application.PackageName, "Shedding on everything");
break;
default:
break;
}
return base.OnContextItemSelected (item);
}
Figure 3 shows the output of having the Listing 3 code as a context menu.
[Click on image for larger view.]
Figure 3. The output from Listing 3, showing the context menu.
Navigation Drawer
The navigation drawer is a panel that transitions from the left edge of the screen to display some navigation options. With the navigation drawer, the user activates it by swiping from the left edge of the screen or touching the application icon in the action bar. As the navigation drawer expands, it overlays the screen content except for the navigation bar.
When should an application make use of the navigation drawer? The navigation drawer is a great tool when there will be three or more top-level views.
Note: The navigation drawer is available via the Android Support Library v4. This article assumes Android 4 or later.
Figure 4 shows the navigation drawer once the user has activated the drawer.
[Click on image for larger view.]
Figure 4. The navigation drawer, once the user has activated it.
Figure 5 shows the result of selecting one of the items in the navigation drawer. In this case, the user selected Mars.
[Click on image for larger view.]
Figure 5. The result of selecting an item in the navigation drawer. In this case, the user selected Mars.
Let's start by looking at the layout used to store the display. There are three widgets in the layout in Listing 4:
- DrawerLayout: A container layout that holds both the framelayout that contains the fragment, as well as a listview.
- FrameLayout: Contains the content that will be displayed. The framelayout will be changed programmatically to contain the fragment that contains the imageview of the planet.
- ListView: Contains the items that will be navigated to.
Listing 4: The XML Used to Create the DrawerLayout
<?xml version="1.0" encoding="utf-8"?>
<android.support.v4.widget.DrawerLayout xmlns:android=
"http://schemas.android.com/apk/res/android"
android:id="@+id/drawer_layout"
android:layout_width="match_parent"
android:layout_height="match_parent">
<FrameLayout
android:id="@+id/content_frame"
android:layout_width="match_parent"
android:layout_height="match_parent" />
<ListView
android:id="@+id/left_drawer"
android:layout_width="240dp"
android:layout_height="match_parent"
android:layout_gravity="start"
android:choiceMode="singleChoice"
android:divider="@android:color/transparent"
android:dividerHeight="0dp"
android:background="#111" />
</android.support.v4.widget.DrawerLayout>
Listing 5 shows the activity that displays the content, as well as processes the selection. The key items:
- The listview within the drawer is loaded with content. The content comes from the string.xml file in the resources.
- Clicking on the item in the listview is set up and handled by the SelectItem method. The SelectItem method is going to handle the screen changes by handling the fragment changes. After the fragment is instantiated, a value is passed to the fragment.
Listing 5: Setting Up the MenuDrawer Code
[Activity (Label = "Navigation Drawer", MainLauncher = true)]
public class MainActivity : Activity
{
private String[] mPlanetTitles;
private DrawerLayout mDrawerLayout;
private ListView mDrawerList;
private myActionBarDrawerToggle mDrawerToggle;
private string mTitle, mDrawerTitle;
protected override void OnCreate (Bundle bundle)
{
base.OnCreate (bundle);
SetContentView(Resource.Layout.activity_main);
mTitle = mDrawerTitle = Title;
mPlanetTitles = Resources.GetStringArray(Resource.Array.planets_array);
mDrawerLayout = FindViewById<DrawerLayout>(Resource.Id.drawer_layout);
mDrawerList = FindViewById<ListView>(Resource.Id.left_drawer);
// Set the adapter for the list view
mDrawerList.Adapter = (new ArrayAdapter<String>(this,
Resource.Layout.drawer_list_item, mPlanetTitles));
mDrawerList.ItemClick += (sender, args) => SelectItem(args.Position);
mDrawerToggle = new myActionBarDrawerToggle (
this, /* host Activity */
mDrawerLayout, /* DrawerLayout object */
Resource.Drawable.ic_drawer, /* nav drawer icon to replace 'Up' caret */
Resource.String.drawer_open, /* "open drawer" description */
Resource.String.drawer_close /* "close drawer" description */
);
ActionBar.SetDisplayHomeAsUpEnabled(true);
ActionBar.SetHomeButtonEnabled(true);
mDrawerToggle.DrawerClosed += delegate
{
ActionBar.Title = mTitle;
};
mDrawerToggle.DrawerOpened += delegate
{
ActionBar.Title = mDrawerTitle;
};
mDrawerLayout.SetDrawerListener(mDrawerToggle);
if (null == bundle)
SelectItem(0);
}
private void SelectItem(int position)
{
var fragment = new PlanetFragment();
var arguments = new Bundle();
arguments.PutInt(PlanetFragment.ArgPlanetNumber, position);
fragment.Arguments = arguments;
FragmentManager.BeginTransaction()
.Replace(Resource.Id.content_frame, fragment)
.Commit();
mDrawerList.SetItemChecked(position, true);
ActionBar.Title = mTitle = mPlanetTitles[position];
mDrawerLayout.CloseDrawer(mDrawerList);
}
protected override void OnPostCreate(Bundle savedInstanceState)
{
base.OnPostCreate(savedInstanceState);
mDrawerToggle.SyncState();
}
public override void OnConfigurationChanged(Configuration newConfig)
{
base.OnConfigurationChanged(newConfig);
mDrawerToggle.OnConfigurationChanged(newConfig);
}
public override bool OnCreateOptionsMenu(IMenu menu)
{
MenuInflater.Inflate(Resource.Menu.main, menu);
return base.OnCreateOptionsMenu(menu);
}
public override bool OnPrepareOptionsMenu(IMenu menu)
{
var drawerOpen = mDrawerLayout.IsDrawerOpen(Resource.Id.left_drawer);
menu.FindItem(Resource.Id.action_websearch).SetVisible(!drawerOpen);
return base.OnPrepareOptionsMenu(menu);
}
public override bool OnOptionsItemSelected(IMenuItem item)
{
if (mDrawerToggle.OnOptionsItemSelected(item))
return true;
switch (item.ItemId)
{
case Resource.Id.action_websearch:
{
var intent = new Intent(Intent.ActionWebSearch);
intent.PutExtra(SearchManager.Query, ActionBar.Title);
if ((intent.ResolveActivity(PackageManager)) != null)
StartActivity(intent);
else
Toast.MakeText(this, Resource.String.app_not_available,
ToastLength.Long).Show();
return true;
}
default:
return base.OnOptionsItemSelected(item);
}
}
}