Pro Android学习笔记(三三):Menu(4):Alternative菜单
什么是Alternative menu(替代菜單)
舉個(gè)例子,Activity顯示一個(gè)文本文件。如果用戶想對(duì)文本文件進(jìn)行編輯,Activity不提供編輯能力,但可由其他activity或者其他應(yīng)用提供。我們將相關(guān)信息存儲(chǔ)在一個(gè)intent中,例如該文本的Uri。這個(gè)intent可以匹配系統(tǒng)的多個(gè)應(yīng)用,替代菜單將這些應(yīng)用一一列出,菜單項(xiàng)的title就是該可被調(diào)用的activity的名字,圖標(biāo)也為該可被調(diào)用的activity的圖表。
小例子說明
我們通過一個(gè)小例子進(jìn)行學(xué)習(xí),簡單地打開一個(gè)URL:wei://flowingflying/helloworld。在之前Intent的學(xué)習(xí)中,我們通過schema的配置,匹配該URL,也就是我們已經(jīng)有其他應(yīng)用的Activity(Intent Basic Test)可以打開該URL。我們同時(shí)在App中新增一個(gè)activity也能打開該URL。這樣,將在alternative菜單中加入兩個(gè)菜單項(xiàng),點(diǎn)擊它們,將打開相應(yīng)的activity,并通過intent傳遞相關(guān)的數(shù)據(jù)信息。
新增的acitivity名字為Invoke Action(好像應(yīng)該是invoked才對(duì),不好意思)。在AndroidManifest.xml中加入intent-fliter的描述即可,具體見:Pro Android學(xué)習(xí)筆記(十一):了解Intent(中)?。
<activity android:name=".InvokeAction" android:label="@string/invokeAction" android:icon="@drawable/leaf" >
??? <intent-filter >?
??????? <action android:name="android.intent.action.VIEW"?/>?
????????<data android:scheme="wei" />?
??????? <category android:name="android.intent.category.DEFAULT" />?
??????? <category android:name="android.intent.category.ALTERNATIVE" /><!-- 將在最后討論 -->
??? </intent-filter>?
</activity>?
Alternative menu代碼
我們看看如何將替代菜單加入到OptionMenu中。Alternative menu還可以加載subMenu,Context Menu中。
@Override?
public boolean onCreateOptionsMenu(Menu menu) {?
????// 對(duì)比:加入一個(gè)普通菜單項(xiàng)??
????menu.add("普通菜單項(xiàng)");??
????//【步驟1】設(shè)置intent,本例簡單實(shí)用一個(gè)已知的Uri。
??? Intent menuIntent = new Intent(null,Uri.parse("wei://flowingflying/helloWorld"));?
????//【步驟2】加入Alternative菜單。在之前的Item ID類別中已經(jīng)講過,Android對(duì)ID進(jìn)行了劃分,有alternative的ID范圍。
??? int menuGroup = Menu.CATEGORY_ALTERNATIVE;?
??? int startingItemId =?Menu.CATEGORY_ALTERNATIVE;?
??? int orderId = Menu.CATEGORY_ALTERNATIVE;??
??? menu.addIntentOptions(??//返回增加的菜單項(xiàng)數(shù)目,本例為2
??????????? menuGroup,??/* int groupId */?
??????????? startingItemId,??/* int itemId:由于自動(dòng)跳轉(zhuǎn)到,此參數(shù)可以設(shè)置為Menu.NONE。 */
??????????? orderId,???/*int order*/?
??????????? this.getComponentName(),?/* ComponentName caller:當(dāng)前的activity名字,這是android系統(tǒng)處理alternatice menu是調(diào)用的queryIntentActivityOptions()函數(shù)所需要的參數(shù)。getComponentName()返回package名字和class名字,系統(tǒng)以此獲知源activity是誰。 */
??????????? null,??/* Intent[] specifics:匹配可能有多個(gè)intent,此用于過濾,但具體用途不詳 */
??????????? menuIntent,??/* Intent intent:關(guān)鍵的intent */
??????????? 0,??/* flages:關(guān)于items如何加入。0表示 no flag*/
??????????? null);??/* MenuItem[] outSpecificItems ,與specifice相關(guān)*/
????return super.onCreateOptionsMenu(menu);?
}?
關(guān)于Category和規(guī)范代碼寫法
我們注意到,在被喚起的actvity中有下面的描述:
[html]?view plaincopy在試驗(yàn)中,發(fā)現(xiàn)此項(xiàng)可有可無,并不真正影響結(jié)果。而在reference中卻明確表示要為CATEGORY_ALTERNATIVE或者CATEGORY_SELECTED_ALTERNATIVE。為何?
我們以Alternative menu的方式調(diào)用其他activity,正規(guī)的做法是,被喚起的activity應(yīng)允許被alternative菜單喚起。因此被喚起的activity在intent-fliter中需給出類別。同時(shí)alternative菜單的intent也應(yīng)當(dāng)標(biāo)明自己類型。因此規(guī)范的代碼是:
Intent menuIntent = new Intent(null,this.getIntent().getData());?
menuIntent.addCategory(Intent.CATEGORY_ALTERNATIVE);?
在小例子中,由于其他應(yīng)用的Activity(Intent Basic Test)在Manifest XML中并沒有給出相應(yīng)的類別,不被匹配。運(yùn)行結(jié)果如圖:
關(guān)于flags
menu.addIntentOptions()的倒數(shù)第二個(gè)參數(shù)是flags,表示菜單項(xiàng)添加的方式。0,即缺省,表示如果groupId相同,則替代菜單將取代原有的菜單項(xiàng)設(shè)置。如果我們想保留原有的同一Group的菜單項(xiàng),可以將flags設(shè)置為Menu.FLAG_APPEND_TO_GROUP。注意,如果groupId為Menu.NONE是不進(jìn)行替換的,這個(gè)表示不設(shè)置GroupId,并非GroupId為0。
多個(gè)匹配的itemId等參數(shù)
讓我們看看系統(tǒng)是如何實(shí)現(xiàn)Alternative菜單的。從reference中看到,Menu是一個(gè)interface,具體是通過MenuBuilder實(shí)現(xiàn)(源代碼見android-17(version)/com/android/internal/view/menu/MenuBuilder.java。相關(guān)代碼如下:
public int addIntentOptions(int group, int id, int categoryOrder, ComponentName caller,
??????? Intent[] specifics, Intent intent, int flags, MenuItem[] outSpecificItems) {
??? PackageManager pm = mContext.getPackageManager();?
??? final List<ResolveInfo> lri =??//查詢匹配的Activity信息?
??????????? pm.queryIntentActivityOptions(caller, specifics, intent, 0);?
????final int N = lri != null ? lri.size() : 0;
????//下面說明如果flag表示FLAG_APPEND_TO_GROUP,會(huì)刪除整個(gè)group,取而代之
??? if ((flags & FLAG_APPEND_TO_GROUP) == 0) {?
??????? removeGroup(group);?
??? }?
??? for (int i=0; i<N; i++) {?
??????? final ResolveInfo ri = lri.get(i);?
??????? Intent rintent = new Intent(?
??????????? ri.specificIndex < 0 ? intent : specifics[ri.specificIndex]);?
??????? rintent.setComponent(new ComponentName(?
??????????????? ri.activityInfo.applicationInfo.packageName,?
??????????????? ri.activityInfo.name));?
????????final MenuItem item = add(group, id, categoryOrder, ri.loadLabel(pm))
??????????????? .setIcon(ri.loadIcon(pm))?
??????????????? .setIntent(rintent);?
??????? if (outSpecificItems != null && ri.specificIndex >= 0) {
??????????? outSpecificItems[ri.specificIndex] = item;?
??????? }?
??? }?
??? return N;?
}?
從源代碼,可能看出如果有多個(gè)匹配,這些菜單項(xiàng)具有相同的group,相同的id,和相同categoryOrder。雖然我們?cè)谛±又惺褂昧藄tartingItemId,但是實(shí)際上itemId是相同的。在小例子中,我們?cè)黾恿藀ublic boolean onOptionsItemSelected(MenuItem item),并在里面檢查item的參數(shù)值,證實(shí)確實(shí)相同。
這段代碼還說明了菜單項(xiàng)的名字和圖片為何,以及為何能喚起Activity。采用了setIntent(),是在Pro Android學(xué)習(xí)筆記(三十):Menu(1):了解Menu學(xué)習(xí)過的的一種觸發(fā)機(jī)制。
本博文涉及的例子代碼,可以在Pro Android學(xué)習(xí):Menu中下載。
轉(zhuǎn)載于:https://www.cnblogs.com/mzsoft/p/4467780.html
總結(jié)
以上是生活随笔為你收集整理的Pro Android学习笔记(三三):Menu(4):Alternative菜单的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 设计模式之禅读书笔记
- 下一篇: Android应用安全开发之浅谈网页打开