Android实例剖析笔记(二)
????? 上篇文章分析了NotesList這個Activity,并著重剖析了其中的intent機制,本文將繼續上篇未完的工作,以NotesList為實例介紹Android的菜單機制(尤其是動態菜單機制)。
簡介
android提供了三種菜單類型,分別為options menu,context menu,sub menu。
options menu就是通過按home鍵來顯示,context menu需要在view上按上2s后顯示。這兩種menu都有可以加入子菜單,子菜單不能種不能嵌套子菜單。options menu最多只能在屏幕最下面顯示6個菜單選項,稱為icon menu,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函數。
????????????????.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,這樣會覆蓋掉
????????????????.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;
????????????}
????? 實驗結果如下:
附記
感謝Evan JIANG對前一篇文章的錯誤之處進行指正,
“<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
只是指明會在Launcher中顯示圖標,同一個apk可以在桌面上加很多的圖標,分別啟動內部不同的多個界面。“,實驗后發現確實如此,學習了。
總結
以上是生活随笔為你收集整理的Android实例剖析笔记(二)的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Makefile 书写命令
- 下一篇: 冲动当机立断(2006-01-19 16