簡介
android提供了三種菜單類型,分別為options menu,context menu,sub menu。
options menu就是通過按home鍵來顯示,context menu需要在view上按上2s後顯示。這兩種menu都有可以加入子功能表,子功能表不能種不能嵌套子功能表。options menu最多隻能在螢幕最下面顯示6個菜單選項,稱為iconmenu,icon menu不能有checkable選項。多於6的功能表項目會以more icon menu來調出,稱為expanded menu。options menu通過activity的onCreateOptionsMenu來產生,這個函數只會在menu第一次產生時調用。任何想改變options menu的想法只能在onPrepareOptionsMenu來實現,這個函數會在menu顯示前調用。onOptionsItemSelected 用來處理選中的功能表項目。
context menu是跟某個具體的view綁定在一起,在activity種用registerForContextMenu來為某個view註冊context menu。context menu在顯示前都會調用onCreateContextMenu來產生menu。onContextItemSelected用來處理選中的功能表項目。
android還提供了對功能表項目進行分組的功能,可以把相似功能的功能表項目分成同一個組,這樣就可以通過調用setGroupCheckable,setGroupEnabled,setGroupVisible來設定菜單屬性,而無須單獨設定。
Options Menu
Notepad中使用了options menu和context menu兩種菜單。首先來看產生options menu的onCreateOptionsMenu函數。
menu.add(0, MENU_ITEM_INSERT, 0, R.string.menu_insert)
.setShortcut('3', 'a')
.setIcon(android.R.drawable.ic_menu_add);
這是一個標準的插入一個功能表項目的方法,功能表項目的id為MENU_ITEM_INSERT。有意思的是下面這幾句代碼:
Intent intent = new Intent(null, getIntent().getData());
intent.addCategory(Intent.CATEGORY_ALTERNATIVE);
menu.addIntentOptions(Menu.CATEGORY_ALTERNATIVE, 0, 0,
new ComponentName(this, NotesList.class), null, intent, 0, null);
這到底有何用處呢?其實這是一種動態菜單技術(也有點像外掛程式機制),若某一個activity,其類型是”android.intent.category.ALTERNATIVE”,資料是”vnd.android.cursor.dir/vnd.google.note”的話,系統就會為這個activity增加一個功能表項目。在androidmanfest.xml中查看後發現,沒有一個activity符合條件,所以這段代碼並沒有動態添加出任何一個功能表項目。
為了驗證上述分析,我們可以來做一個實驗,在androidmanfest.xml中進行修改,看是否會動態產生出功能表項目。
實驗一
首先我們來建立一個新的activity作為目標activity,名為HelloAndroid,沒有什麼功能,就是顯示一個介面。
public class HelloAndroid extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
this.setContentView(R.layout.main);
}
}
它所對應的布局介面XML檔案如下:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
>
<TextView
android:layout_width="fill_parent"
android:layout_height="wrap_content" android:id="@+id/TextView01"/>
<Button android:id="@+id/Button01" android:layout_height="wrap_content" android:layout_width="fill_parent" android:text="@string/txtInfo"></Button>
</LinearLayout>
然後修改androidmanfest.xml,加入下面這段配置,讓HelloAndroid滿足上述兩個條件:
<activity android:name="HelloAndroid" android:label="@string/txtInfo">
<intent-filter>
<action android:name="com.android.notepad.action.HELLO_TEST" />
<category android:name="android.intent.category.ALTERNATIVE"/>
<data android:mimeType="vnd.android.cursor.dir/vnd.google.note" />
</intent-filter>
</activity>
好了,運行下試試,哎,還是沒有動態功能表項目加入呀!怎麼回事呢?查看代碼後發現,原來是onPrepareOptionsMenu搞的鬼!這個函數在onCreateOptionsMenu之後運行,下面這段代碼中,由於Menu.CATEGORY_ALTERNATIVE是指向同一個組,所以把onCreateOptionsMenu中設定的功能表項目給覆蓋掉了,而由於onPrepareOptionsMenu沒有給Menu.CATEGORY_ALTERNATIVE附新值,故Menu.CATEGORY_ALTERNATIVE還是為空白。
Intent intent = new Intent(null, uri);
intent.addCategory(Intent.CATEGORY_ALTERNATIVE);
menu.addIntentOptions(Menu.CATEGORY_ALTERNATIVE, 0, 0, null, specifics, intent, 0,items);
好的,那我們暫時把上面這幾句給注釋掉,當然,也可以不注釋這幾句,在onCreateOptionsMenu中改groupid號,即將Menu.CATEGORY_ALTERNATIVE改為Menu.first,其他的也行,但注意不要改為menu.none,這樣會覆蓋掉。
menu.add(0, MENU_ITEM_INSERT, 0, R.string.menu_insert)
.setShortcut('3', 'a')
.setIcon(android.R.drawable.ic_menu_add);
添加的菜單。因為menu.none也為0。運行後就可以看到動態菜單出來了!
上面這個options menu是在NotesList介面上沒有日誌列表選中的情況下產生的,若先選中一個日誌,然後再點”menu”,則產生的options menu是下面這樣的:
哎,又動態增加了兩個功能表項目”Edit note”和”Edit title”,這又是如何動態加入的呢?這就是onPrepareOptionsMenu的功勞了。
Uri uri = ContentUris.withAppendedId(getIntent().getData(), getSelectedItemId());
首先擷取選中的日誌(若沒有選擇,則uri為空白)
Intent[] specifics = new Intent[1];
specifics[0] = new Intent(Intent.ACTION_EDIT, uri);
MenuItem[] items = new MenuItem[1];
然後為選中的日誌建立一個intent,操作類型為Intent.ACTION_EDIT,資料為選中日誌的URI.於是會為選中的日誌建立一個”Edit note”功能表項目。
Intent intent = new Intent(null, uri);
intent.addCategory(Intent.CATEGORY_ALTERNATIVE);
menu.addIntentOptions(Menu.CATEGORY_ALTERNATIVE, 0, 0, null, specifics, intent, 0,
items);
這幾句和上面onCreateOptionsMenu函數中類似,用於動態增加功能表項目,若某一個activity,其類型是”android.intent.category.ALTERNATIVE”,資料是”vnd.android.cursor.item/vnd.google.note”的話,系統就會為這個activity增加一個功能表項目。在androidmanfest.xml中查看後發現,TitleEditor這個activity符合條件,於是系統就為TitleEditor這個activity動態添加一個功能表項目”Edit title”。
else {
menu.removeGroup(Menu.CATEGORY_ALTERNATIVE);
}
若日誌列表為空白,則從菜單中刪除群組號為Menu.CATEGORY_ALTERNATIVE的功能表項目,只剩下”Add note”功能表項目。
處理“選中功能表項目”事件
功能表項目選中事件的處理非常簡單,通過onOptionsItemSelected來完成,這裡只是簡單地調用 startActivity(new Intent(Intent.ACTION_INSERT, getIntent().getData()));這個intent的操作類型為Intent.ACTION_INSERT,資料為日誌列表的URI,即”content:// com.google.provider.NotePad/notes”
@Override
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId()) {
case MENU_ITEM_INSERT:
// Launch activity to insert a new item
startActivity(new Intent(Intent.ACTION_INSERT, getIntent().getData()));
return true;
}
return super.onOptionsItemSelected(item);
}
Context Menu
下面介紹另一種菜單---操作功能表,這通過重載onCreateContextMenu函數實現。首先確認已經選中了日誌列表中的一個日誌,若沒選擇,則直接返回。Cursor指向選中的日誌項。
Cursor cursor = (Cursor) getListAdapter().getItem(info.position);
if (cursor == null) {
// For some reason the requested item isn't available, do nothing
return;
}
然後,設定操作功能表的標題為日誌標題
// Setup the menu header
menu.setHeaderTitle(cursor.getString(COLUMN_INDEX_TITLE));
最後為操作功能表增加一個功能表項目
// Add a menu item to delete the note
menu.add(0, MENU_ITEM_DELETE, 0, R.string.menu_delete);
對於操作功能表項目選中的事件處理,是通過重載onContextItemSelected實現的。
switch (item.getItemId()) {
case MENU_ITEM_DELETE: {
// Delete the note that the context menu is for
Uri noteUri = ContentUris.withAppendedId(getIntent().getData(), info.id);
getContentResolver().delete(noteUri, null, null);
return true;
}
}
return false;
}
對於日誌的刪除,首先調用ContentUris.withAppendedId(getIntent().getData(), info.id);來拼接出待刪除日誌的URI.然後getContentResolver().delete(noteUri, null, null);調用下層的Content Provider去刪除此日誌。
實驗二
來做個簡單實驗,在上述代碼基礎上增加一個操作功能表項目。首先在onCreateContextMenu函數中增加一個操作功能表項目:
menu.add(0,MENU_ITEM_INSERT,0,R.string.menu_insert);
然後為其在onContextItemSelected函數中增加一個處理過程:
case MENU_ITEM_INSERT:
{
new AlertDialog.Builder(this).setIcon(R.drawable.app_notes)
.setTitle(R.string.app_name).setMessage(R.string.error_message).setPositiveButton(R.string.button_ok, new OnClickListener(){
public void onClick(DialogInterface dialog, int which) {
// TODO Auto-generated method stub
}
}).show();
return true;
}
實驗結果如下: