Android Toolbar,你想知道的都在這裡了
Android 3.0之後,Google引入了ActionBar,想統一安卓應用的導航欄樣式。但由於ActionBar難以定製,很大程度上限制了開發人員,比如標題文字大小、間距等不易實現個性化,很多開發者放棄了ActionBar的使用,而是使用普通的ViewGroup來封裝自己的App Bar,或者使用JakeWharton大神的ActionBarSherlock庫。
後來,自2014年Google I/O 上Material Design橫空出世後,市場上的應用又逐步趨向了樣式的風格統一,support library中很快就出來了Toolbar控制元件,一個定製化的ViewGroup,來完善ActionBar的使用,App Bar又迎來了春天。
基本使用
第一步,在Theme中隱藏現有的ActionBar,有以下兩種方式:
<style name="AppTheme.Base" parent="Theme.AppCompat">
<item name="windowActionBar">false</item>
<item name="android:windowNoTitle">true</item>
......
</style>
或者
<style name="AppTheme.Base" parent="heme.AppCompat.Light.NoActionBar" >
......
</style>
第二步,在佈局中新增v7包中的Toolbar控制元件(注意在builde.gradle
檔案中新增support.v7包的依賴:compile ‘com.android.support:appcompat-v7:23.3.0’),如:
<android.support.v7.widget.Toolbar
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@+id/tb_toolbar"
android:layout_width ="match_parent"
android:layout_height="?attr/actionBarSize"
app:title="@string/app_name"
app:titleTextColor="@android:color/white"
android:background="@color/colorPrimary">
</android.support.v7.widget.Toolbar>
第三步,在Activity程式碼中使用Toolbar物件替換ActionBar:
Toolbar mToolbarTb = (Toolbar) findViewById(R.id.tb_toolbar);
setSupportActionBar(mToolbarTb);
效果如下:
oolbar-sample-01.png諸如logo、title、subTitle、navigationIcon等,都可以通過app:xxx
屬性和Java程式碼來控制。titleTextAppearance、subtitleTextAppearance也可以用來控制標題顏色和大小,如:
<style name="Theme.ToolBar.Base.Title" parent="@style/TextAppearance.Widget.AppCompat.Toolbar.Title">
<item name="android:textSize">18sp</item>
<item name="android:textColor">@android:color/white</item>
</style>
Options Menu
與ActionBar的用法一致,在menu資源目錄下新建一個search.xml
檔案,新增menu內容,如:
<?xml version="1.0" encoding="utf-8"?>
<menu xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<item
android:id="@+id/action_search"
android:title="search"
android:icon="@android:drawable/ic_menu_search"
app:showAsAction="collapseActionView"/>
</menu>
在Java程式碼中新增為Toolbar新增對應的Menu Item,並設定點選事件,如:
@Override
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate(R.menu.search, menu);
return super.onCreateOptionsMenu(menu);
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId()) {
case R.id.action_search:
//TODO search
break;
}
return super.onOptionsItemSelected(item);
}
效果如下:
toolbar-sample-02如圖,溢位按鈕的顏色是黑色,那麼怎麼修改成與主題色搭配的白色呢?可以通過修改Toolbar的theme和popupTheme屬性來改變,在Toolbar中新增如下屬性:
<android.support.v7.widget.Toolbar
......
app:popupTheme="@style/ThemeOverlay.AppCompat.Light"
app:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar">
</android.support.v7.widget.Toolbar>
還有另一種方式,就是使用theme中的actionMenuTextColor
屬性設定Menu Item的字型顏色,如:
<style name="OverFlowMenuTheme" parent="Theme.AppCompat.NoActionBar">
<item name="android:actionMenuTextColor">@android:color/white</item>
<item name="overlapAnchor">false</item>
</style>
Toolbar中的相關theme設定如下:
<android.support.v7.widget.Toolbar
......
app:popupTheme="@style/OverFlowMenuTheme"
app:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar">
</android.support.v7.widget.Toolbar>
一般也是採用這種做法,將Toolbar相關屬性集中寫到一個style中,比如:
<style name="OverFlowMenuTheme" parent="Theme.AppCompat.NoActionBar">
<!-- 設定Menu選單的背景色 -->
<item name="android:itemBackground">@android:color/white</item>
<!-- 設定Menu選單的字型顏色 -->
<item name="android:textColorPrimary">@android:color/black</item>
<!-- 設定Menu視窗不覆蓋Toolbar檢視 -->
<item name="overlapAnchor">false</item>
</style>
這樣設定下的效果如下圖所示:
toolbar-sample-04.pngUp Enable
在二級介面等Activity中,通過如下設定可以在Toolbar左邊顯示一個返回按鈕:
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
然後通過android.R.id.home
監聽返回按鈕的點選事件,比如可以返回上級Activity中:
@Override
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId()) {
case android.R.id.home:
finish();
break;
}
return super.onOptionsItemSelected(item);
}
當然,也可以通過為Toolbar設定導航圖示的點選事件來達到這個監聽效果:
mToolbarTb.setNavigationOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
finish();
}
});
顯示效果如下:
toolbar-sample-05.png至於導航返回按鈕的圖示,可以通過mToolbarTb.setNavigationIcon
方法或者app:navigationIcon
屬性來修改。
標題居中
前面我們說過,Toolbar就是一個定製化的ViewGroup,所以可以在Toolbar裡面放置一個TextView控制元件作為居中的標題來使用,再將Toolbar的Title隱藏起來即可實現Toolbar標題居中的效果,如:
<android.support.v7.widget.Toolbar
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@+id/tb_toolbar"
android:layout_width="match_parent"
android:layout_height="?attr/actionBarSize"
app:popupTheme="@style/OverFlowMenuTheme"
app:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
android:background="@color/colorPrimary">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:text="SecondActivity"
style="@style/TextAppearance.AppCompat.Widget.ActionBar.Title"/>
</android.support.v7.widget.Toolbar>
這裡給TextView設定了style屬性,與ActionBar.Title
保持一致,然後還需要去除Toolbar自有的Title,在佈局中使用app:title=""
是不會起作用的,會顯示ActionBar的標題,所以在程式碼中隱藏ActionBar的標題即可:
getSupportActionBar().setDisplayShowTitleEnabled(false);
效果如下:
toolbar-sample-06.pngFragment中使用
有時候需要在Fragment中使用Toolbar,比如Activity中不同的Tab顯示不同的Fragment,同時每個Tab的Toolbar標題、Menu均不相同,這時在Activity中使用同一個Toolbar就相當不方便了。我們可以在每個Fragment的佈局中新增各自的Toolbar,然後在Fragment中單獨控制。
與Activity中使用Toolbar有所不同。替換ActionBar時,需要給setSupportActionBar
方法新增作用物件:
((AppCompatActivity)getActivity()).setSupportActionBar((Toolbar) mContentView.findViewById(R.id.tb_toolbar));
新增Options Menu時,需要額外呼叫setHasOptionsMenu(true);
方法,確保onCreateOptionsMenu()
方法得以呼叫,並且onCreateOptionsMenu()
方法多了一個MenuInflater
引數:
@Override
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
super.onCreateOptionsMenu(menu, inflater);
inflater.inflate(R.menu.search, menu);
}
相關知識
關於Toolar的更多介紹可以從官網中學習借鑑,連結為:Adding the App Bar | Android Developers。從其使用上可以看出,Toolbar是ActionBar的一種替代解決方案,所以很多用法均可參考ActionBar,大家可以從郭神的譯文中掌握ActionBar的相關用法: