Android Action Bar 詳解篇
作為Android 3.0之後引入的新的物件,ActionBar可以說是一個方便快捷的導航神器。它可以作為活動的標題,突出活動的一些關鍵操作(如“搜尋”、“建立”、“共享”等)、作為選單的靈活使用,還可以實現類似TabWidget的標籤功能以及下拉導航的功能,系統能夠很好根據不同的螢幕配置來適應ActionBar的外觀,配合起Fragemtn可謂是十分強大。
那麼,對於今天的主角ActionBar怎麼去新增?在Android3.0預設主題HloleFraphic(全息)主題中,已經創造了ActionBar,所以只要targetSdkVersion的值不低於11,建立的Activity中預設都會帶有ActionBar。例如:
<manifest ... >
<uses-sdk android:minSdkVersion="4"
android:targetSdkVersion="11" />
...
</manifest>
當然了,如果你不想為一個特定的Activity設定Action Bar,設定Activity主題為Theme.Holo.NoActionBar。
<activity android:theme="@android:style/Theme.Holo.NoActionBar">
或者在執行時通過呼叫hide()隱藏Action
Bar。自然也有show()。
ActionBar
actionBar = getActionBar();
actionBar.hide();
下面我們從下拉導航、視窗操作、標籤導航三個方面逐一討論ActionBar
第一,下拉導航
下拉導航最典型的應用場景就是在Google+中的使用,效果如下圖:
圖1;Google+ 圖2:本文示例
實現此效果分如下幾個步驟:
1.初始化一個SpinnerAdapter
SpinnerAdapter mSpinnerAdapter = ArrayAdapter.createFromResource(this,
R.array.action_list,
android.R.layout.simple_spinner_dropdown_item);
2.生成一個OnNavigationListener來響應ActionBar的選單項點選操作
/**
* 在這裡配合Fragment,實現不同的頁面導航
*/
OnNavigationListener mOnNavigationListener = new OnNavigationListener() {
@Override
public boolean onNavigationItemSelected(int position, long itemId) {
Fragment newFragment = null;
switch (position) {
case 0:
newFragment = new Fragment1();
break;
case 1:
newFragment = new Fragment2();
break;
case 2:
newFragment = new Fragment3();
break;
default:
break;
}
getSupportFragmentManager().beginTransaction()
.replace(R.id.container, newFragment, strings[position])
.commit();
return true;
}
};
3,將生成好的適配去和監聽器塞給ActionBar
actionBar = getActionBar();
actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_LIST);//導航模式必須設為NAVIGATION_MODE_LIST
actionBar.setListNavigationCallbacks(mSpinnerAdapter,
mOnNavigationListener);
第二,操作視窗
先上效果圖
圖3 圖4 圖5
在上面的操作視窗裡,增加了一個用於搜尋的可選選單項以及分享和設定的兩個自定義ActionProVider。那麼如何在一個活動中,在已有的ActionBar上新增這些操作視窗。同建立可選選單一樣,定義options.xml的menu檔案如下:
<?xml version="1.0" encoding="utf-8"?>
仔細觀察可以發現每個Item裡都包含如下這兩個屬性:
<menu xmlns:android="http://schemas.android.com/apk/res/android" >
<item
android:id="@+id/menu_search"
android:actionViewClass="android.widget.SearchView"
android:icon="@drawable/ic_menu_search"
android:showAsAction="ifroom"
android:title="搜尋"/>
<item
android:id="@+id/menu_share"
android:actionProviderClass="android.widget.ShareActionProvider"
android:showAsAction="never"
android:title="分享"/>
<item
android:id="@+id/menu_setting"
android:actionProviderClass="com.example.tabdemo.MyActionProvider"
android:showAsAction="never"
android:title="設定">
<menu>
<item
android:id="@+id/menu_theme"
android:actionProviderClass="com.example.tabdemo.MyActionProvider"
android:showAsAction="always|withText"
android:title="更換主題"/>
<item
android:id="@+id/menu_system"
android:actionProviderClass="com.example.tabdemo.MyActionProvider"
android:showAsAction="always|withText"
android:title="系統設定"/>
</menu>
</item>
</menu>
android:actionProviderClass="com.example.tabdemo...."
android:showAsAction=""
對於actionProviderClass屬性用來指定一個構建視窗所使用的佈局資源,除了使用actionProviderClass指定外,還可以使用actionLayout或者actionViewClass都可以。SearchView和ShareActionProvider都是系統自帶的ActionProvider,MyActionProvider是我們要重寫的,後面將會看到如何去自定義一個ActionProvider。
showAsAction屬性共有五個值:ifRoom、never、always、withText、collapseActionView,可以混合使用。
ifRoom 會顯示在Item中,但是如果已經有4個或者4個以上的Item時會隱藏在溢位列表中。當然個
數並不僅僅侷限於4個,依據螢幕的寬窄而定never 永遠不會顯示。只會在溢位列表中顯示,而且只顯示標題,所以在定義item的時候,最好
把標題都帶上。always 無論是否溢位,總會顯示。 withText withText值示意Action bar要顯示文字標題。Action bar會盡可能的顯示這個
標題,但是,如果圖示有效並且受到Action bar空間的限制,文字標題有可
能顯示不全。collapseActionView 宣告瞭這個操作視窗應該被摺疊到一個按鈕中,當使用者選擇這個按鈕時,這個操作視窗展開。否則,
這個操作視窗在預設的情況下是可見的,並且即便在用於不適用的時候,也要佔據操作欄的有效空間。
一般要配合ifRoom一起使用才會有效果。
注: 當你的應用程式正在Android4.0(API 級別 14)或以上的版本上執行,那麼還有一種叫做“分隔操作欄”的額外模式對action bar有效。當你啟用分隔操作欄模式時,在螢幕的底部會顯示一個獨立的橫條,用於顯示Activity在窄屏裝置(如豎屏手機)上執行時的所有操作項。這裡我們不過過多描述,有興趣自己去研究。
就像載入menu一樣,在activity的onCreateOptionsMenu方法裡呼叫上述的xml檔案:
getMenuInflater().inflate(R.menu.options, menu);
//搜尋視窗,因為showAsAction="ifRoom",所以圖三中出現了搜尋按鈕
SearchView searchView = (SearchView) menu.findItem(R.id.menu_search)
.getActionView();
//分享視窗,因為showAsAction="never",所以只能在溢位選單中才看見到
ShareActionProvider mShareActionProvider = (ShareActionProvider) menu
.findItem(R.id.menu_share).getActionProvider();
Intent shareIntent = new Intent(Intent.ACTION_SEND);
shareIntent.setType("image/*");
mShareActionProvider.setShareIntent(shareIntent);
//設定視窗,MyActionProvider就是我們自定義的ActionProvider
MyActionProvider myactionprovider = (MyActionProvider) menu.findItem(
R.id.menu_setting).getActionProvider();
return super.onCreateOptionsMenu(menu);
顯然,當成功執行的時候,結果如圖三,當點選搜尋按鈕時,搜尋按鈕立刻變成了如圖四的樣子,變成可摺疊的操作視窗。
如何自定義操作視窗,定義一個類MyActionProvider繼承自ActionProvider,並實現它的兩口回撥函式即可。如下:
/**
* @ClassName: MyActionProvider
* @Description: 自定義一個視窗操作器,實現建構函式和onCreateActionView即可
* @author yuxianglong
* @date 2013-7-11 下午3:13:44
*
*/
public class MyActionProvider extends ActionProvider{
private Context context;
private LayoutInflater inflater;
private View view;
private ImageView button;
public MyActionProvider(Context context) {
super(context);
// TODO Auto-generated constructor stub
this.context = context;
inflater = LayoutInflater.from(context);
view = inflater.inflate(R.layout.myactionprovider, null);
}
@Override
public View onCreateActionView() {
// TODO Auto-generated method stub
button = (ImageView) view.findViewById(R.id.button);
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
Toast.makeText(context, "是我,沒錯", Toast.LENGTH_SHORT).show();
}
});
return view;
}
}
如此一來,只要在options.xml裡直接引用。執行成功效果如圖五,點選溢位選單,設定按鈕出來了,如果繼續點下去,回撥出它的子選單,因為我們在options.xml裡給自定義的ActionProvider分配了子選單。
當然了,最顯眼的就是處理Action Bar上的應用程式圖示,平時玩手機多的同學應該可以發現,好多應用的圖示都是可以點選的,而且大多數都是回到了上一個Activity,或者說是主Activity。那麼,如何觸發應用程式圖示呢,說白了應用程式圖示也是一個選單,並且其id是規定死的,所以只要我們在onOptionsItemSelected方法裡去捕捉它的點選事件,做出響應:
@Override
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId())
{
case android.R.id.home:
Intent intent = new Intent(this, HomeActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
startActivity(intent);
return true;
default:
return super.onOptionsItemSelected(item);
}
}
我們給Intent新增了FLAG_ACTIVITY_CLEAR_TOP標識,該標識的作用是在回到HomeActivity時,把在堆疊中處於HomeActivity上面的活動全部清除。如果這是候執行程式的話,如果系統版本小於4.0的話,是可以正常跑起來的,達到想要的效果,但如果系統大於或者等於4.0的話,那麼點選應用圖示是無效的。必須加上setHomeButtonEnabled=true,4.0一下 預設為true。如果還想要一個回退箭頭的話,再加上一句setDisplayHomeAsUpEnabled(true);效果如下:
這裡我在擴充套件一下:使用過Navigation Drawer的同學應該瞭解,這裡點選應用程式圖示通常會作為拉出導航抽屜。通常在那種情況下是把活動的onOptionsItemSelected,傳送給ActionBarDrawerToggle的onOptionsItemSelected。不多說了,感興趣的同學自己去研究,後面會把Navigation Bar寫出來。
第三,導航選項標籤
當你想要在一個Activity中提供導航選擇標籤時,使用操作欄的選項標籤是一個非常好的選擇(而不是使用TabWidget類),因為系統會調整操作欄選項標籤來適應不同尺寸的螢幕的需要,在螢幕足夠寬的時候,導航選項標籤會被放到主操作欄中;當螢幕太窄的時候,選項標籤會被放到一個分離的橫條中。如圖下:
要使用選項標籤在Fragmengt之間切換,選擇一個選項標籤時執行一個Fragment事務,佈局裡包含一個用於放置跟每個Fragment物件關聯的選項標籤的ViewGroup物件。該物件有一個資源ID,以便能夠在選項標籤的切換程式碼中能夠引用它。Activity的佈局檔案activity_main.xml定義如下:
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/container"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity"
tools:ignore="MergeRootFrame" />這裡的ViewGroup為FragmentLayout。Activity程式碼如下:
/*
* @ClassName: MainActivity
* @Description: 繼承自FragmentActivity,作為Fragment的holder-Activity使用,
* 實現TabListener介面,當切Tab的時候達到切換Fragment的效果
* @author yuxianglong
* @date 2013-7-11 下午7:40:35
*
*/
public class MainActivity extends FragmentActivity implements
ActionBar.TabListener {
private ActionBar actionBar;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
actionBar = getActionBar();
actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);// 導航模式必須設為NAVIGATION_MODE_Tabs
// For each of the sections in the app, add a tab to the action bar.
actionBar.addTab(actionBar.newTab().setText(R.string.title_section1)
.setTabListener(this));
actionBar.addTab(actionBar.newTab().setText(R.string.title_section2)
.setTabListener(this));
actionBar.addTab(actionBar.newTab().setText(R.string.title_section3)
.setTabListener(this));
}
@Override
public void onTabSelected(ActionBar.Tab tab,
FragmentTransaction fragmentTransaction) {
// When the given tab isselected, show the tabcontents in the
// //container view.
Fragment fragment3 = null;
Fragment fragment1 = null;
Fragment fragment2 = null;
switch (tab.getPosition()) {
case 0:
if (fragment1 == null) {
fragment1 = new Fragment1();
}
getSupportFragmentManager().beginTransaction()
.replace(R.id.container, fragment1).commit();
break;
case 1:
if (fragment2 == null) {
fragment2 = new Fragment2();
}
getSupportFragmentManager().beginTransaction()
.replace(R.id.container, fragment2).commit();
break;
case 2:
if (fragment3 == null) {
fragment3 = new Fragment3();
}
getSupportFragmentManager().beginTransaction()
.replace(R.id.container, fragment3).commit();
break;
default:
break;
}
}
@Override
public void onTabUnselected(ActionBar.Tab tab,
FragmentTransaction fragmentTransaction) {
}
@Override
public void onTabReselected(ActionBar.Tab tab,
FragmentTransaction fragmentTransaction) {
}
}
最後跑起來的效果如下:
至此ActionBar的一些常見使用場景,我們就熟悉了,後面繼續研究ActionBar的外觀樣式。
相關文章
- 自定義 Action Bar
- Action呼叫字首詳解
- Android 5.0——Material Design詳解(動畫篇)AndroidMaterial Design動畫
- Android ORM 框架:GreenDao 使用詳解(基礎篇)AndroidORM框架
- Android ORM 框架:GreenDao 使用詳解(進階篇)AndroidORM框架
- struts2 action中result引數詳解
- 詳解 Android 中的 IPC 機制:基礎篇Android
- Android Intent Action 大全AndroidIntent
- Android Translucent System Bar 的最佳實踐Android
- Android AsyncTask 詳解Android
- Android:動畫詳解Android動畫
- Android拖拽詳解Android
- Android:Service詳解Android
- Android Notification 詳解Android
- Android WebView 詳解AndroidWebView
- Android – Drawable 詳解Android
- Android RecyclerView詳解AndroidView
- Android Proguard 詳解Android
- android service詳解Android
- android廣播集合,intent,actionAndroidIntent
- Android自定義action與permission!!!Android
- Spark in action on Kubernetes - 儲存篇Spark
- 【Spark篇】---Spark中Action運算元Spark
- 證書詳解三篇
- Android開發:Translucent System Bar 的最佳實踐Android
- Android ADB使用之詳細篇Android
- Java反射詳解篇--一篇入魂Java反射
- Flutter完整開發實戰詳解(十四、混合開發打包 Android 篇)FlutterAndroid
- Android SecureRandom漏洞詳解Androidrandom
- Android Service詳解(二)Android
- Android Service詳解(一)Android
- Android 向量圖詳解Android
- Android元件詳解—TextViewAndroid元件TextView
- Android-Service詳解Android
- Android Gson使用詳解Android
- Android Paint 使用詳解AndroidAI
- Android HttpURLConnection詳解AndroidHTTP
- Android AsyncTask使用詳解Android