电竞比分网-中国电竞赛事及体育赛事平台

分享

Android Action Bar 詳解

 浮云沫沫33 2014-01-16
  作者原創(chuàng),轉(zhuǎn)載請標明出處:http://blog.csdn.net/yuxlong2010

          作為Android 3.0之后引入的新的對象,ActionBar可以說是一個方便快捷的導航神器。它可以作為活動的標題,突出活動的一些關鍵操作(如“搜索”、“創(chuàng)建”、“共享”等)、作為菜單的靈活使用,還可以實現(xiàn)類似TabWidget的標簽功能以及下拉導航的功能,系統(tǒng)能夠很好根據(jù)不同的屏幕配置來適應ActionBar的外觀,配合起Fragemtn可謂是十分強大。

          那么,對于今天的主角ActionBar怎么去添加?在Android3.0默認主題HloleFraphic(全息)主題中,已經(jīng)創(chuàng)造了ActionBar,所以只要targetSdkVersion的值不低于11,創(chuàng)建的Activity中默認都會帶有ActionBar。例如:

  1. <manifest ... >    
  2.     <uses-sdk android:minSdkVersion="4"  
  3.               android:targetSdkVersion="11" />    
  4.     ...    
  5. </manifest>  

          當然了,如果你不想為一個特定的Activity設置Action Bar,設置Activity主題為Theme.Holo.NoActionBar。

  1. <activity android:theme="@android:style/Theme.Holo.NoActionBar">  


          或者在運行時通過調(diào)用hide()隱藏Action Bar。自然也有show()。

  1. ActionBar actionBar = getActionBar();    
  2. actionBar.hide();  


          下面我們從下拉導航、視窗操作標簽導航三個方面逐一討論ActionBar

         第一,下拉導航

                    下拉導航最典型的應用場景就是在Google+中的使用,效果如下圖:

          

圖1;Google+                             圖2:本文示例

實現(xiàn)此效果分如下幾個步驟:

          1.初始化一個SpinnerAdapter 

  1. SpinnerAdapter mSpinnerAdapter = ArrayAdapter.createFromResource(this,  
  2.                 R.array.action_list,  
  3.                 android.R.layout.simple_spinner_dropdown_item);  

  2.生成一個OnNavigationListener來響應ActionBar的菜單項點擊操作

  1. /** 
  2.      * 在這里配合Fragment,實現(xiàn)不同的頁面導航 
  3.      */  
  4.     OnNavigationListener mOnNavigationListener = new OnNavigationListener() {  
  5.   
  6.         @Override  
  7.         public boolean onNavigationItemSelected(int position, long itemId) {  
  8.             Fragment newFragment = null;  
  9.             switch (position) {  
  10.             case 0:  
  11.                 newFragment = new Fragment1();  
  12.                 break;  
  13.             case 1:  
  14.                 newFragment = new Fragment2();  
  15.                 break;  
  16.             case 2:  
  17.                 newFragment = new Fragment3();  
  18.                 break;  
  19.             default:  
  20.                 break;  
  21.             }  
  22.             getSupportFragmentManager().beginTransaction()  
  23.                     .replace(R.id.container, newFragment, strings[position])  
  24.                     .commit();  
  25.             return true;  
  26.         }  
  27.     };  

   3,將生成好的適配去和監(jiān)聽器塞給ActionBar

  1. actionBar = getActionBar();  
  2. actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_LIST);//導航模式必須設為NAVIGATION_MODE_LIST   
  3. actionBar.setListNavigationCallbacks(mSpinnerAdapter,  
  4.         mOnNavigationListener);  


     第二,操作視窗

          先上效果圖

    

        圖3                                                                             圖4                                                                   圖5

          在上面的操作視窗里,增加了一個用于搜索的可選菜單項以及分享和設置的兩個自定義ActionProVider。那么如何在一個活動中,在已有的ActionBar上添加這些操作視窗。同創(chuàng)建可選菜單一樣,定義options.xml的menu文件如下:

  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <menu xmlns:android="http://schemas./apk/res/android" >  
  3.   
  4.     <item  
  5.         android:id="@+id/menu_search"  
  6.         android:actionViewClass="android.widget.SearchView"  
  7.         android:icon="@drawable/ic_menu_search"  
  8.         android:showAsAction="ifroom"  
  9.         android:title="搜索"/>  
  10.     <item  
  11.         android:id="@+id/menu_share"  
  12.         android:actionProviderClass="android.widget.ShareActionProvider"  
  13.         android:showAsAction="never"  
  14.         android:title="分享"/>  
  15.     <item  
  16.         android:id="@+id/menu_setting"  
  17.         android:actionProviderClass="com.example.tabdemo.MyActionProvider"  
  18.         android:showAsAction="never"  
  19.         android:title="設置">  
  20.         <menu>  
  21.             <item  
  22.                 android:id="@+id/menu_theme"  
  23.                 android:actionProviderClass="com.example.tabdemo.MyActionProvider"  
  24.                 android:showAsAction="always|withText"  
  25.                 android:title="更換主題"/>  
  26.             <item  
  27.                 android:id="@+id/menu_system"  
  28.                 android:actionProviderClass="com.example.tabdemo.MyActionProvider"  
  29.                 android:showAsAction="always|withText"  
  30.                 android:title="系統(tǒng)設置"/>  
  31.         </menu>  
  32.     </item>  
  33. </menu>  

          仔細觀察可以發(fā)現(xiàn)每個Item里都包含如下這兩個屬性:

  1. android:actionProviderClass="com.example.tabdemo...."  
  1. android:showAsAction=""  
          對于actionProviderClass屬性用來指定一個構(gòu)建視窗所使用的布局資源,除了使用actionProviderClass指定外,還可以使用actionLayout或者actionViewClass都可以。SearchView和ShareActionProvider都是系統(tǒng)自帶的ActionProvider,MyActionProvider是我們要重寫的,后面將會看到如何去自定義一個ActionProvider。

          showAsAction屬性共有五個值:ifRoom、never、always、withText、collapseActionView,可以混合使用。


    ifRoom 會顯示在Item中,但是如果已經(jīng)有4個或者4個以上的Item時會隱藏在溢出列表中。當然個
數(shù)并不僅僅局限于4個,依據(jù)屏幕的寬窄而定
    never 永遠不會顯示。只會在溢出列表中顯示,而且只顯示標題,所以在定義item的時候,最好
把標題都帶上。
    always 無論是否溢出,總會顯示。
    withText withText值示意Action bar要顯示文本標題。Action bar會盡可能的顯示這個
標題,但是,如果圖標有效并且受到Action bar空間的限制,文本標題有可
能顯示不全。
   collapseActionView   聲明了這個操作視窗應該被折疊到一個按鈕中,當用戶選擇這個按鈕時,這個操作視窗展開。否則,
這個操作視窗在默認的情況下是可見的,并且即便在用于不適用的時候,也要占據(jù)操作欄的有效空間。
一般要配合ifRoom一起使用才會有效果。

          注: 當你的應用程序正在Android4.0(API 級別 14)或以上的版本上運行,那么還有一種叫做“分隔操作欄”的額外模式對action bar有效。當你啟用分隔操作欄模式時,在屏幕的底部會顯示一個獨立的橫條,用于顯示Activity在窄屏設備(如豎屏手機)上運行時的所有操作項。這里我們不過過多描述,有興趣自己去研究。

          就像加載menu一樣,在activity的onCreateOptionsMenu方法里調(diào)用上述的xml文件:

  1. getMenuInflater().inflate(R.menu.options, menu);  
  2.   
  3. //搜索視窗,因為showAsAction="ifRoom",所以圖三中出現(xiàn)了搜索按鈕   
  4. SearchView searchView = (SearchView) menu.findItem(R.id.menu_search)  
  5.         .getActionView();  
  6.   
  7. //分享視窗,因為showAsAction="never",所以只能在溢出菜單中才看見到   
  8. ShareActionProvider mShareActionProvider = (ShareActionProvider) menu  
  9.         .findItem(R.id.menu_share).getActionProvider();  
  10. Intent shareIntent = new Intent(Intent.ACTION_SEND);  
  11. shareIntent.setType("image/*");  
  12. mShareActionProvider.setShareIntent(shareIntent);  
  13.   
  14. //設置視窗,MyActionProvider就是我們自定義的ActionProvider   
  15. MyActionProvider myactionprovider = (MyActionProvider) menu.findItem(  
  16.         R.id.menu_setting).getActionProvider();  
  17. return super.onCreateOptionsMenu(menu);  

          顯然,當成功運行的時候,結(jié)果如圖三,當點擊搜索按鈕時,搜索按鈕立刻變成了如圖四的樣子,變成可折疊的操作視窗。

          如何自定義操作視窗,定義一個類MyActionProvider繼承自ActionProvider,并實現(xiàn)它的兩口回調(diào)函數(shù)即可。如下:

  1. /**  
  2.  * @ClassName: MyActionProvider  
  3.  * @Description: 自定義一個視窗操作器,實現(xiàn)構(gòu)造函數(shù)和onCreateActionView即可 
  4.  * @author yuxianglong  
  5.  * @date 2013-7-11 下午3:13:44  
  6.  *   
  7.  */  
  8. public class MyActionProvider extends ActionProvider{  
  9.   
  10.     private Context context;  
  11.     private LayoutInflater inflater;  
  12.     private View view;  
  13.     private ImageView button;  
  14.     public MyActionProvider(Context context) {  
  15.         super(context);  
  16.         // TODO Auto-generated constructor stub   
  17.         this.context = context;  
  18.         inflater = LayoutInflater.from(context);  
  19.         view = inflater.inflate(R.layout.myactionprovider, null);  
  20.     }  
  21.   
  22.       
  23.     @Override  
  24.     public View onCreateActionView() {  
  25.         // TODO Auto-generated method stub   
  26.         button = (ImageView) view.findViewById(R.id.button);  
  27.         button.setOnClickListener(new View.OnClickListener() {  
  28.               
  29.             @Override  
  30.             public void onClick(View v) {  
  31.                 // TODO Auto-generated method stub   
  32.                 Toast.makeText(context, "是我,沒錯", Toast.LENGTH_SHORT).show();  
  33.             }  
  34.         });  
  35.         return view;  
  36.     }  
  37.   
  38. }  

          如此一來,只要在options.xml里直接引用。運行成功效果如圖五,點擊溢出菜單,設置按鈕出來了,如果繼續(xù)點下去,回調(diào)出它的子菜單,因為我們在options.xml里給自定義的ActionProvider分配了子菜單。

                 當然了,最顯眼的就是處理Action Bar上的應用程序圖標,平時玩手機多的同學應該可以發(fā)現(xiàn),好多應用的圖標都是可以點擊的,而且大多數(shù)都是回到了上一個Activity,或者說是主Activity。那么,如何觸發(fā)應用程序圖標呢,說白了應用程序圖標也是一個菜單,并且其id是規(guī)定死的,所以只要我們在onOptionsItemSelected方法里去捕捉它的點擊事件,做出響應:

  1. @Override  
  2.     public boolean onOptionsItemSelected(MenuItem item) {  
  3.         switch (item.getItemId())   
  4.            {          
  5.               case android.R.id.home:              
  6.                  Intent intent = new Intent(this, HomeActivity.class);              
  7.                  intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);   
  8.                  startActivity(intent);              
  9.                  return true;          
  10.               default:              
  11.                  return super.onOptionsItemSelected(item);      
  12.            }  
  13.     }  

          我們給Intent添加了FLAG_ACTIVITY_CLEAR_TOP標識,該標識的作用是在回到HomeActivity時,把在堆棧中處于HomeActivity上面的活動全部清除。如果這是候運行程序的話,如果系統(tǒng)版本小于4.0的話,是可以正常跑起來的,達到想要的效果,但如果系統(tǒng)大于或者等于4.0的話,那么點擊應用圖標是無效的。必須加上setHomeButtonEnabled=true,4.0一下 默認為true。如果還想要一個回退箭頭的話,再加上一句setDisplayHomeAsUpEnabled(true);效果如下:

          這里我在擴展一下:使用過Navigation Drawer的同學應該了解,這里點擊應用程序圖標通常會作為拉出導航抽屜。通常在那種情況下是把活動的onOptionsItemSelected,傳送給ActionBarDrawerToggle的onOptionsItemSelected。不多說了,感興趣的同學自己去研究,后面會把Navigation Bar寫出來。


     第三,導航選項標簽

          當你想要在一個Activity中提供導航選擇標簽時,使用操作欄的選項標簽是一個非常好的選擇(而不是使用TabWidget類),因為系統(tǒng)會調(diào)整操作欄選項標簽來適應不同尺寸的屏幕的需要,在屏幕足夠?qū)挼臅r候,導航選項標簽會被放到主操作欄中;當屏幕太窄的時候,選項標簽會被放到一個分離的橫條中。如圖下:

   

          要使用選項標簽在Fragmengt之間切換,選擇一個選項標簽時執(zhí)行一個Fragment事務,布局里包含一個用于放置跟每個Fragment對象關聯(lián)的選項標簽的ViewGroup對象。該對象有一個資源ID,以便能夠在選項標簽的切換代碼中能夠引用它。Activity的布局文件activity_main.xml定義如下:

  1. <FrameLayout xmlns:android="http://schemas./apk/res/android"  
  2.     xmlns:tools="http://schemas./tools"  
  3.     android:id="@+id/container"  
  4.     android:layout_width="match_parent"  
  5.     android:layout_height="match_parent"  
  6.     tools:context=".MainActivity"  
  7.     tools:ignore="MergeRootFrame" />  
          這里的ViewGroup為FragmentLayout。Activity代碼如下:

  1. /* 
  2. * @ClassName: MainActivity  
  3. * @Description: 繼承自FragmentActivity,作為Fragment的holder-Activity使用, 
  4. *               實現(xiàn)TabListener接口,當切Tab的時候達到切換Fragment的效果 
  5. * @author yuxianglong  
  6. * @date 2013-7-11 下午7:40:35  
  7. * 
  8.  */  
  9. public class MainActivity extends FragmentActivity implements  
  10.         ActionBar.TabListener {  
  11.     private ActionBar actionBar;  
  12.   
  13.     @Override  
  14.     protected void onCreate(Bundle savedInstanceState) {  
  15.         super.onCreate(savedInstanceState);  
  16.         setContentView(R.layout.activity_main);  
  17.         actionBar = getActionBar();  
  18.         actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);// 導航模式必須設為NAVIGATION_MODE_Tabs   
  19.   
  20.         // For each of the sections in the app, add a tab to the action bar.   
  21.   
  22.         actionBar.addTab(actionBar.newTab().setText(R.string.title_section1)  
  23.                 .setTabListener(this));  
  24.         actionBar.addTab(actionBar.newTab().setText(R.string.title_section2)  
  25.                 .setTabListener(this));  
  26.         actionBar.addTab(actionBar.newTab().setText(R.string.title_section3)  
  27.                 .setTabListener(this));  
  28.   
  29.     }  
  30.   
  31.   
  32.     @Override  
  33.     public void onTabSelected(ActionBar.Tab tab,  
  34.             FragmentTransaction fragmentTransaction) {  
  35.         // When the given tab isselected, show the tabcontents in the   
  36.         // //container view.   
  37.         Fragment fragment3 = null;  
  38.         Fragment fragment1 = null;  
  39.         Fragment fragment2 = null;  
  40.         switch (tab.getPosition()) {  
  41.         case 0:  
  42.             if (fragment1 == null) {  
  43.                 fragment1 = new Fragment1();  
  44.             }  
  45.             getSupportFragmentManager().beginTransaction()  
  46.                     .replace(R.id.container, fragment1).commit();  
  47.             break;  
  48.         case 1:  
  49.             if (fragment2 == null) {  
  50.                 fragment2 = new Fragment2();  
  51.             }  
  52.             getSupportFragmentManager().beginTransaction()  
  53.                     .replace(R.id.container, fragment2).commit();  
  54.             break;  
  55.         case 2:  
  56.             if (fragment3 == null) {  
  57.                 fragment3 = new Fragment3();  
  58.             }  
  59.             getSupportFragmentManager().beginTransaction()  
  60.                     .replace(R.id.container, fragment3).commit();  
  61.             break;  
  62.   
  63.         default:  
  64.             break;  
  65.         }  
  66.   
  67.     }  
  68.   
  69.     @Override  
  70.     public void onTabUnselected(ActionBar.Tab tab,  
  71.             FragmentTransaction fragmentTransaction) {  
  72.     }  
  73.   
  74.     @Override  
  75.     public void onTabReselected(ActionBar.Tab tab,  
  76.             FragmentTransaction fragmentTransaction) {  
  77.     }  
  78.   
  79. }  
          最后跑起來的效果如下:

          至此ActionBar的一些常見使用場景,我們就熟悉了,后面繼續(xù)研究ActionBar的外觀樣式。



    本站是提供個人知識管理的網(wǎng)絡存儲空間,所有內(nèi)容均由用戶發(fā)布,不代表本站觀點。請注意甄別內(nèi)容中的聯(lián)系方式、誘導購買等信息,謹防詐騙。如發(fā)現(xiàn)有害或侵權內(nèi)容,請點擊一鍵舉報。
    轉(zhuǎn)藏 分享 獻花(0

    0條評論

    發(fā)表

    請遵守用戶 評論公約

    類似文章 更多