Android側(cè)滑菜單之DrawerLayout用法詳解
onConfigurationChanged最早的時(shí)候?qū)崿F(xiàn)側(cè)滑菜單功能大多時(shí)候需要通過(guò)github上一個(gè)叫做SlidingMenu的開(kāi)源通過(guò)依賴包來(lái)實(shí)現(xiàn),后來(lái)谷歌在v4包中添加了DrawerLayout來(lái)實(shí)現(xiàn)這個(gè)功能,完全可以替代SlidingMenu,這里我們來(lái)學(xué)習(xí)DrawerLayout的用法
一)創(chuàng)建DrawerLayout
1)在布局文件里將布局設(shè)置為DrawerLaout,而且因?yàn)槭莢4包中的功能,所以必須寫(xiě)全包名,注意第一必須先寫(xiě)主視圖布局,然后再寫(xiě)抽屜里的視圖,這里我們放了ListView進(jìn)入抽屜
<pre name="code" class="java"><android.support.v4.widget.DrawerLayout xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/drawer_layout" android:layout_width="match_parent" android:layout_height="match_parent"> <!--The main content view 抽屜關(guān)閉時(shí)候的主視圖--> <FrameLayout android:id="@+id/content_frame" android:layout_width="match_parent" android:layout_height="match_parent" > </FrameLayout> <!-- The Navigation view 左側(cè)拉開(kāi)之后的導(dǎo)航視圖 --> <ListView android:id="@+id/left_drawer" android:layout_width="240dp" android:layout_height="match_parent" android:layout_gravity="start" //從左往右滑出菜單,如果為end就是從右往左滑出菜單 android:background="#ffffcc" android:choiceMode="singleChoice" android:divider="@android:color/transparent" android:dividerHeight="0dp" ></ListView> </android.support.v4.widget.DrawerLayout>
2)在activity里將DrawerLayout加載進(jìn)來(lái),并且給抽屜里的列表適配了數(shù)據(jù),這樣就將抽屜布局加載進(jìn)來(lái)了。
public class MainActivity extends Activity implements OnItemClickListener{ private DrawerLayout mDrawerLayout; private ListView mListView; private ArrayList<String> menuList;//數(shù)據(jù) private ArrayAdapter<String> adapter;//適配器 // private ActionBarDrawerToggle mDrawerToggle; // private String mTitle; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); // mTitle=(String) getTitle(); mDrawerLayout=(DrawerLayout) findViewById(R.id.drawer_layout); mListView=(ListView) findViewById(R.id.left_drawer); menuList=new ArrayList<String>(); for(int i=0;i<5;i++) menuList.add("劉家威"+i); adapter=new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1,menuList); mListView.setAdapter(adapter);
二)給抽屜里的列表添加點(diǎn)擊事件,動(dòng)態(tài)添加碎片進(jìn)入主視圖
1)給抽屜里的列表項(xiàng)添加點(diǎn)擊事件,每點(diǎn)擊列表項(xiàng),讓其在主內(nèi)容視圖里添加一個(gè)Fragment,讓這個(gè)Fragment里顯示點(diǎn)擊的列表項(xiàng)里的內(nèi)容,先寫(xiě)這個(gè)FragmentLayout里的布局,這里讓其顯示一段文本
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" > <TextView android:id="@+id/tv" android:layout_width="match_parent" android:layout_height="wrap_content" android:layout_gravity="center" android:textSize="25sp"/> </LinearLayout>
2)創(chuàng)建Fragment,在這里重寫(xiě)onCreateView(),把布局加載進(jìn)來(lái),然后獲取到點(diǎn)擊項(xiàng)里的內(nèi)容,這個(gè)是在MainActivity里通過(guò)讓碎片攜帶參數(shù)的方法來(lái)實(shí)現(xiàn)
package com.example.drawerlayout; import android.app.Fragment; import android.os.Bundle; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.TextView; public class ContentFragment extends Fragment { private TextView tv; @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { View view=inflater.inflate(R.layout.fragment_content,container,false); tv=(TextView) view.findViewById(R.id.tv); String text=getArguments().getString("text"); //獲取到傳入的值 tv.setText(text); return view; } }
3)給抽屜里的列表添加點(diǎn)擊事件,點(diǎn)擊后會(huì)進(jìn)入碎片界面,并且碎片攜帶了Bundle類(lèi)型的參數(shù),參數(shù)的值我們傳入了列表數(shù)據(jù)menuList.get(position),使用碎片的方法如下,獲取碎片的實(shí)例之后再通過(guò)getFragmentManager()獲取碎片管理器,給碎片管理器開(kāi)啟事務(wù),開(kāi)啟事務(wù)之后的碎片管理器用replace()方法將碎片的布局替換原來(lái)的主視圖布局,這樣一個(gè)碎片就添加進(jìn)來(lái)了
最后別忘了關(guān)閉抽屜
@Override public void onItemClick(AdapterView<?> parent, View view, int position, long id) { //在FrameLayout里動(dòng)態(tài)插入一個(gè)Fragment Fragment contentFragment=new ContentFragment(); //創(chuàng)建碎片實(shí)例 Bundle args=new Bundle(); //讓每個(gè)碎片都攜帶參數(shù),初始化為Bundle類(lèi)型 args.putString("text", menuList.get(position)); //參數(shù)put進(jìn)入String類(lèi)型的值,傳入當(dāng)前點(diǎn)擊的菜單項(xiàng)的值 contentFragment.setArguments(args); //讓碎片攜帶上參數(shù) FragmentManager fm=getFragmentManager(); fm.beginTransaction().replace(R.id.content_frame,contentFragment ).commit(); //開(kāi)啟事務(wù),向容器內(nèi)加入碎片,最后提交事務(wù) mDrawerLayout.closeDrawer(mListView); //進(jìn)入碎片界面時(shí)關(guān)閉掉抽屜的內(nèi)容 }
三)監(jiān)聽(tīng)抽屜的打開(kāi)關(guān)閉事件
使用ActionBarDrawerToggle,最早的時(shí)候在v4包里,現(xiàn)在谷歌把它搬到了v7包里,所以要引入v7依賴包
在引入v7包的時(shí)候,我遇到了一個(gè)錯(cuò)誤,support\v7\appcompat\res\values-21\themes-base.xml:error:Error retrieving parent for item :No resources found that matches the given name.
這是因?yàn)関7包版本過(guò)低,解決方法是在project:properties里target=android-8改為21以上就好
當(dāng)抽屜打開(kāi)的時(shí)候,標(biāo)題欄改為"請(qǐng)選擇",當(dāng)抽屜關(guān)閉的時(shí)候,標(biāo)題欄變?yōu)樵瓉?lái)的app標(biāo)題
mTitle=(String) getTitle()
mDrawerToggle=new ActionBarDrawerToggle(this, mDrawerLayout, R.string.drawer_open,R.string.drawer_close){ @Override public void onDrawerOpened(View drawerView) { super.onDrawerOpened(drawerView); getActionBar().setTitle("請(qǐng)選擇"); invalidateOptionsMenu();//重繪actionBar上的菜單項(xiàng),執(zhí)行這個(gè)方法的時(shí)候會(huì)回調(diào)onPreareOptionsMenu() } @Override public void onDrawerClosed(View drawerView) { super.onDrawerClosed(drawerView); getActionBar().setTitle(mTitle); invalidateOptionsMenu(); } }; mDrawerLayout.setDrawerListener(mDrawerToggle);
因?yàn)槲覀兏淖兞薃ctionBar的內(nèi)容,因此要調(diào)用invalidateOptionsMenu()會(huì)重繪actionBar上的菜單項(xiàng),執(zhí)行這個(gè)方法的時(shí)候會(huì)回調(diào)onPreareOptionsMenu(),所以我們可以在這里通過(guò)判斷抽屜處于打開(kāi)還是關(guān)閉狀態(tài),來(lái)進(jìn)行一些操作,比如隱藏actionbar的菜單項(xiàng)。
既然要隱藏,那么前提是我們得有菜單項(xiàng),這里先給actionbar添加一個(gè)搜索按鈕,關(guān)于給ActionBar添加按鈕的辦法,在關(guān)于ActionBar的博客里已經(jīng)講過(guò)了,所以這里直接操作
res/values/menu/main.xml
<menu xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" tools:context="com.example.drawerlayout.MainActivity" > <item android:id="@+id/websearch" android:icon="@drawable/action_search" android:showAsAction="ifRoom|withText" android:title="webSearch" /> </menu>
然后將菜單加載進(jìn)來(lái)
@Override public boolean onCreateOptionsMenu(Menu menu) { getMenuInflater().inflate(R.menu.main, menu); return super.onCreateOptionsMenu(menu); }
順路給這個(gè)搜索按鈕設(shè)置點(diǎn)擊事件
/*給actionBar的webSearch設(shè)置點(diǎn)擊事件*/ @Override public boolean onOptionsItemSelected(MenuItem item) { switch (item.getItemId()) { case R.id.websearch: Intent intent=new Intent(); intent.setAction("android.intent.action.VIEW"); intent.setData(Uri.parse("http://www.baidu.com")); startActivity(intent); break; default: break; } return super.onOptionsItemSelected(item); }
最后在onPrepareOptionsMenu()里通過(guò)DrawerLayout的isOpen()來(lái)判斷,如果打開(kāi)了就隱藏菜單項(xiàng)
@Override public boolean onPrepareOptionsMenu(Menu menu) { /*在這里通過(guò)判斷DrawerLayout是打開(kāi)還是關(guān)閉,來(lái)判斷是否顯示actionBar的webSearch圖標(biāo)*/ boolean isDrawerOpen=mDrawerLayout.isDrawerOpen(mListView); menu.findItem(R.id.websearch).setVisible(!isDrawerOpen); //如果DrawerLayout是關(guān)閉的就顯示actionBar里的webSearch return super.onPrepareOptionsMenu(menu); }
四)點(diǎn)擊圖標(biāo)開(kāi)閉抽屜
為了更加方便的使用抽屜,我們可以通過(guò)點(diǎn)擊圖標(biāo)來(lái)開(kāi)閉抽屜界面
1)在onCreate()里開(kāi)啟app icon的功能
//開(kāi)啟actionBar的APP icon的功能 getActionBar().setDisplayHomeAsUpEnabled(true); getActionBar().setHomeButtonEnabled(true);
2)因?yàn)閳D標(biāo)也是一個(gè)菜單項(xiàng),所以點(diǎn)擊它也是會(huì)調(diào)用onOptionsItemSelected(MenuItem item)
@Override public boolean onOptionsItemSelected(MenuItem item) { //將ActionBar上的圖標(biāo)與Drawer結(jié)合起來(lái) if(mDrawerToggle.onOptionsItemSelected(item)){ return true; } ....... }
這樣就實(shí)現(xiàn)了這個(gè)功能,但是圖標(biāo)還是原來(lái)的icon,現(xiàn)在把抽屜布局和ActionBarDrawerToggle同步
3)谷歌官方推薦在onPostCreate()里來(lái)同步,并且谷歌推薦寫(xiě)上onConfigurationChanged()
@Override protected void onPostCreate(Bundle savedInstanceState) { //將ActionBarDrawerToggle與DrawerLayout同步起來(lái) //將ActionBarDrawerToggle中的Drawer圖標(biāo)設(shè)置為ActionBar里的Home_Button的圖標(biāo) mDrawerToggle.syncState(); super.onPostCreate(savedInstanceState); } //這個(gè)方法用來(lái)監(jiān)測(cè)手機(jī)狀態(tài)的變化,比如橫屏豎屏的切換 @Override public void onConfigurationChanged(Configuration newConfig) { mDrawerToggle.onConfigurationChanged(newConfig); super.onConfigurationChanged(newConfig); }
綜合以上的所有內(nèi)容,效果圖如下
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請(qǐng)查看下面相關(guān)鏈接
相關(guān)文章
Android使用Sqlite存儲(chǔ)數(shù)據(jù)用法示例
這篇文章主要介紹了Android使用Sqlite存儲(chǔ)數(shù)據(jù)的方法,結(jié)合實(shí)例形式分析了Android操作SQLite數(shù)據(jù)庫(kù)的相關(guān)步驟與操作技巧,需要的朋友可以參考下2016-11-11Android布局控件DrawerLayout實(shí)現(xiàn)完美側(cè)滑效果
這篇文章主要為大家詳細(xì)介紹了Android布局控件DrawerLayout實(shí)現(xiàn)完美側(cè)滑效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-08-08Android開(kāi)發(fā)之自定義View(視圖)用法詳解
這篇文章主要介紹了Android開(kāi)發(fā)之自定義View(視圖)用法,結(jié)合實(shí)例形式分析了Android自定義視圖的實(shí)現(xiàn)方法與使用注意事項(xiàng),需要的朋友可以參考下2016-10-10Android廣播接收機(jī)制詳細(xì)介紹(附短信接收實(shí)現(xiàn))
這篇文章主要介紹了Android廣播接收機(jī)制詳細(xì)介紹(附短信接收實(shí)現(xiàn)),本文講解了BroadcastReceiver的注冊(cè)過(guò)程BroadcastReceiver的步驟,并給出一個(gè)短信廣播實(shí)現(xiàn)示例,需要的朋友可以參考下2015-06-06Android Intent實(shí)現(xiàn)頁(yè)面跳轉(zhuǎn)的方法示例
本篇文章主要介紹了Android Intent實(shí)現(xiàn)頁(yè)面跳轉(zhuǎn)的方法示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-03-03Android6.0動(dòng)態(tài)申請(qǐng)權(quán)限所遇到的問(wèn)題小結(jié)
這篇文章給大家介紹了Android6.0動(dòng)態(tài)申請(qǐng)權(quán)限所遇到的問(wèn)題,在沒(méi)給大家介紹這下問(wèn)題之前,先給大家說(shuō)下基本定義和基本使用方式,本文給大家介紹的非常詳細(xì),具有參考借鑒價(jià)值,對(duì)android 6.0 動(dòng)態(tài)權(quán)限遇到問(wèn)題感興趣的朋友一起看看吧2016-11-11Android App調(diào)用MediaRecorder實(shí)現(xiàn)錄音功能的實(shí)例
這篇文章主要介紹了Android App調(diào)用MediaRecorder實(shí)現(xiàn)錄音功能的實(shí)例,MediaRecorder非常強(qiáng)大,不僅能夠用來(lái)錄制音頻還可以錄制視頻,需要的朋友可以參考下2016-04-04