`
XiangdongLee
  • 浏览: 91929 次
  • 性别: Icon_minigender_1
  • 来自: 长沙
社区版块
存档分类
最新评论

【攻克Android (17)】Side nav 抽屉导航

 
阅读更多
本文围绕以下三个部分展开:

一、DrawerLayout 抽屉布局
二、案例
        附   代码补充
三、案例二:抽屉布局结合SwipeTab





一、DrawerLayout 抽屉布局



        1. DrawerLayout的使用场景

        (1)超过三个顶级视图(否则可用 Tab 导航)

        (2)需要从底层视图切换导航位置

        (3)分支视图层次较深

        2. 实现抽屉导航

        相关类:

        (1)抽屉布局:android.support.v4.widget.DrawerLayout

        (2)抽屉把手:android.support.v4.app.ActionBarDrawerToggle

        需要在工程中先导入 support-v4 。


二、案例







        1. 抽屉布局需要用到v4支持包,因此先导入v4支持包:在 build.gradle 中加入以下导包语句:

compile 'com.android.support:support-v4:21.0.3'


        2. 在 strings.xml 中定义所需字符串:

<resources>
  <string name="app_name">DrawerNav</string>

  <string name="action_settings">Settings</string>

  <string name="drawer_open">open</string>
  <string name="drawer_close">close</string>
  <string name="search">搜索</string>
  <string name="title">目录</string>

</resources>


        3. 复制 icon 到 res 下面:



        4. 在 menu_main.xml 中定义 search 菜单 (搜索菜单):

<item
      android:id="@+id/action_search"
      android:title="@string/search"
      android:orderInCategory="101"
      android:showAsAction="always"
      android:icon="@drawable/ic_search_white_24dp"/>


        在 menu_main.xml 中,原本就有设置菜单(settings),加入搜索菜单后,界面如下:



        menu_main.xml 代码如下:

<menu xmlns:android="http://schemas.android.com/apk/res/android"
      xmlns:tools="http://schemas.android.com/tools"
      tools:context=".MainActivity">
  <item
      android:id="@+id/action_settings"
      android:title="@string/action_settings"
      android:orderInCategory="100"
      android:showAsAction="never"/>

  <item
      android:id="@+id/action_search"
      android:title="@string/search"
      android:orderInCategory="101"
      android:showAsAction="always"
      android:icon="@drawable/ic_search_white_24dp"/>
</menu>


        5. 在 activity_main.xml 中定义内容视图(帧布局)和左侧、右侧两个抽屉:

<android.support.v4.widget.DrawerLayout
    android:id="@+id/drawer_layout"
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_height="match_parent"
    android:layout_width="match_parent">

  <!--内容视图-->
  <FrameLayout
      android:id="@+id/content_layout"
      android:layout_width="match_parent"
      android:layout_height="match_parent"/>

  <!--
  左侧抽屉 :
    因为左侧抽屉,写了一个单独的布局文件 drawer_left,因此要包含进来。
    单独写布局文件,内容可以更丰富一些。
  -->
  <!-- layout_gravity:定义抽屉的位置(start:左侧;end:右侧。) -->
  <include
      layout="@layout/drawer_left"
      android:layout_width="240dp"
      android:layout_height="match_parent"
      android:layout_gravity="start"/>

  <!--
  右侧抽屉 :
    因为右侧抽屉没有单独写布局文件,因此只需写一个控件 ListView 即可。
  -->
  <ListView
      android:id="@+id/right_drawer"
      android:layout_width="240dp"
      android:layout_height="match_parent"
      android:layout_gravity="end"
      android:background="@android:color/white"/>

</android.support.v4.widget.DrawerLayout>


        其中,左侧抽屉写了一个单独的布局文件 drawer_left.xml,其界面如下:



<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:orientation="vertical"
              android:layout_width="match_parent"
              android:layout_height="match_parent"
              android:background="@android:color/white">

  <LinearLayout
      android:orientation="vertical"
      android:layout_width="match_parent"
      android:layout_height="150dp"
      android:background="@android:color/darker_gray">

    <ImageView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center"
        android:src="@mipmap/ic_launcher"/>

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@string/app_name"
        android:layout_gravity="center"
        android:paddingTop="30dp"/>

  </LinearLayout>

  <ListView
      android:id="@+id/listView"
      android:layout_width="match_parent"
      android:layout_height="match_parent"
      android:choiceMode="singleChoice"
      android:divider="@android:color/transparent"
      android:dividerHeight="0dp"/>
</LinearLayout>

    <!--
    上面的 LinearLayout 不会被调用,因此不需要写 id。
    下面的 ListView 会在里面添加内容,因此需要写 id。
    -->


        6. MainActivity 。 声明变量。

// 声明变量
    private String[] data = {"选项一", "选项二", "选项三", "选项四", "选项五"};
    private DrawerLayout drawerLayout;
    private ActionBarDrawerToggle toggle;
    private ListView listView;


        7. MainActivity 。 初始化变量(同时设置抽屉的阴影、设置显示抽屉把手)。

// 初始化变量
        drawerLayout = (DrawerLayout) findViewById(R.id.drawer_layout);
        // 设置抽屉的阴影
        drawerLayout.setDrawerShadow(R.drawable.drawer_shadow, Gravity.START);
        drawerLayout.setDrawerShadow(R.drawable.drawer_shadow, Gravity.END);
        // 显示 Home 按钮(此处显示抽屉把手)
        getActionBar().setDisplayHomeAsUpEnabled(true);
        getActionBar().setHomeButtonEnabled(true);


        8. MainActivity 。 创建抽屉把手(可以部署到手机上看效果)。

   // 创建抽屉把手(看效果)
        // 参数:上下文,抽屉布局对象,把手菜单图标,打开抽屉的文本,关闭抽屉的文本
        toggle = new ActionBarDrawerToggle(this,
                drawerLayout,
                R.drawable.ic_menu_white_24dp,
                R.string.drawer_open,
                R.string.drawer_close) {
            // 抽屉打开
            @Override
            public void onDrawerOpened(View drawerView) {
                super.onDrawerOpened(drawerView);
                getActionBar().setTitle(getString(R.string.title));
            }

            // 抽屉关闭
            @Override
            public void onDrawerClosed(View drawerView) {
                super.onDrawerClosed(drawerView);

            }
        };

        // 同步状态(更新 Home 位置显示的图标)
        toggle.syncState();
        // 设置监听器
        drawerLayout.setDrawerListener(toggle);


        9. MainActivity 。 写菜单事件(搜索图标)。完成后可以在手机上部署看效果。

// 菜单 (看效果:要重绘菜单)
    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.menu_main, menu);
        // 8.1 控制搜索图标的显示/隐藏
        // 左侧抽屉打开,隐藏操作栏按钮
        // 左侧抽屉关闭,显示操作栏按钮
        boolean isOpen = drawerLayout.isDrawerOpen(Gravity.START);
        menu.findItem(R.id.action_search).setVisible(!isOpen);
        return true;
    }


    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // 8.2 点击抽屉把手,打开/关闭抽屉
        if (toggle.onOptionsItemSelected(item)) {
            return true;
        }
        return super.onOptionsItemSelected(item);
    }


        接着在上一步的“创建抽屉把手”中的打开抽屉(onDrawerOpened)和关闭抽屉(onDrawerClosed)的类中,重新绘制菜单(不重新绘制的时候,菜单没有改变)

invalidateOptionsMenu();


        10. MainActivity 。 初始化左侧抽屉。

        (1)在 onCreate()方法中,调用自己创建的初始化左侧抽屉的方法:initLeftDrawer() 。

// 初始化左侧抽屉
        initLeftDrawer();


        (2)在initLeftDrawer()方法中,将 左侧抽屉的listView(列表框)、适配器、数据 进行绑定 。

   // 左侧抽屉的listView(列表框)、适配器、数据 进行绑定
        // 之所以能找到 listView,是因为 activity_main.xml 把它 include 了。
        listView = (ListView) findViewById(R.id.listView);
        ArrayAdapter adapter = new ArrayAdapter(this,
                android.R.layout.simple_list_item_activated_1, data);
        // 设置单选模式
        listView.setChoiceMode(ListView.CHOICE_MODE_SINGLE);
        listView.setAdapter(adapter);


        补充:
        simple_list_item_1 :有一个控件。
        simple_list_item_2 :有两个控件。
        simple_list_item_activated_1 : 有一个控件。当选项选中时,背景色改变(即为激活状态)
        simple_list_item_activated_2 : 有两个控件。当选项选中时,背景色改变(即为激活状态)

        (3)在initLeftDrawer()方法中,写 点击左侧抽屉ListView(列表框)的事件。

listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
                // 点击ListView中的项的时候,就设置标题
                getActionBar().setTitle(data[position]);
                // 点击完项后,关闭左侧抽屉
                drawerLayout.closeDrawer(Gravity.START);




        11. MainActivity 。 创建两个碎片:NewsFragment & fragment_news.xml,TechFragment & fragment_tech.xml。

        12.当选中左侧抽屉中ListView中的不同选项的时候,使用不同的 碎片 替换 内容视图。

        (1)在“声明变量”部分,定义 碎片管理器

// 定义 碎片管理器
    private FragmentManager fm;


        (2)在“初始化变量”部分,获得 碎片管理器

// 获得 碎片管理器
        fm = this.getFragmentManager();


        (3)在“初始化左侧抽屉”的点击ListView事件部分,当选中“选项一”的时候,用 News 碎片 替换 内容视图;当选中“选项二”的时候,用 Tech 碎片 替换 内容视图。

FragmentTransaction ft = fm.beginTransaction();
                switch (position) {
                    case 0:
                        ft.replace(R.id.content_layout, new NewsFragment());
                        break;
                    case 1:
                        ft.replace(R.id.content_layout, new TechFragment());
                        break;
                    default:
                        break;
                }
                // 提交事务
                ft.commit();



        附   代码补充



        MainActivity

package com.xiangdong.drawernav;

import android.app.Activity;
import android.app.FragmentManager;
import android.app.FragmentTransaction;
import android.os.Bundle;
import android.support.v4.app.ActionBarDrawerToggle;
import android.support.v4.widget.DrawerLayout;
import android.view.Gravity;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.AdapterView;
import android.widget.ArrayAdapter;
import android.widget.ListView;


public class MainActivity extends Activity {
    // 5. 声明变量
    private String[] data = {"选项一", "选项二", "选项三", "选项四", "选项五"};
    private DrawerLayout drawerLayout;
    private ActionBarDrawerToggle toggle;
    private ListView listView;

    // 11.1 定义 碎片管理器
    private FragmentManager fm;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        // 6. 初始化变量
        drawerLayout = (DrawerLayout) findViewById(R.id.drawer_layout);
        // 设置抽屉的阴影
        drawerLayout.setDrawerShadow(R.drawable.drawer_shadow, Gravity.START);
        drawerLayout.setDrawerShadow(R.drawable.drawer_shadow, Gravity.END);
        // 显示 Home 按钮(此处显示抽屉把手)
        getActionBar().setDisplayHomeAsUpEnabled(true);
        getActionBar().setHomeButtonEnabled(true);

        // 11.2 获得 碎片管理器
        fm = this.getFragmentManager();

        // 7. 创建抽屉把手(看效果)
        // 参数:上下文,抽屉布局对象,把手菜单图标,打开抽屉的文本,关闭抽屉的文本
        toggle = new ActionBarDrawerToggle(this,
                drawerLayout,
                R.drawable.ic_menu_white_24dp,
                R.string.drawer_open,
                R.string.drawer_close) {
            // 抽屉打开
            @Override
            public void onDrawerOpened(View drawerView) {
                super.onDrawerOpened(drawerView);
                // 8.3重新绘制菜单 (不重新绘制的时候,菜单没有改变)
                invalidateOptionsMenu();
                getActionBar().setTitle(getString(R.string.title));
            }

            // 抽屉关闭
            @Override
            public void onDrawerClosed(View drawerView) {
                super.onDrawerClosed(drawerView);
                // 8.3 重新绘制菜单
                invalidateOptionsMenu();
            }
        };

        // 同步状态(更新 Home 位置显示的图标)
        toggle.syncState();
        // 设置监听器
        drawerLayout.setDrawerListener(toggle);

        // 9. 初始化左侧抽屉
        initLeftDrawer();
    }

    /**
     * 9. 初始化左侧抽屉
     */
    private void initLeftDrawer() {
        // 9.1 左侧抽屉的listView(列表框)、适配器、数据 进行绑定
        // 之所以能找到 listView,是因为 activity_main.xml 把它 include 了。
        listView = (ListView) findViewById(R.id.listView);
        /*
        simple_list_item_1 :有一个控件。
        simple_list_item_2 :有两个控件。
        simple_list_item_activated_1 : 有一个控件。当选项选中时,背景色改变(即为激活状态)
        simple_list_item_activated_2 : 有两个控件。当选项选中时,背景色改变(即为激活状态)
         */
        ArrayAdapter adapter = new ArrayAdapter(this,
                android.R.layout.simple_list_item_activated_1, data);
        // 设置单选模式
        listView.setChoiceMode(ListView.CHOICE_MODE_SINGLE);
        listView.setAdapter(adapter);

        // 9.2 点击左侧抽屉ListView(列表框)的事件
        listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
                // 点击ListView中的项的时候,就设置标题
                getActionBar().setTitle(data[position]);
                // 点击完项后,关闭左侧抽屉
                drawerLayout.closeDrawer(Gravity.START);

                // 10. 创建碎片(XxxFragment & fragment_Xxx.xml)
                // 11. 使用 碎片 替换 内容视图
                // 11.3
                FragmentTransaction ft = fm.beginTransaction();
                switch (position) {
                    case 0:
                        ft.replace(R.id.content_layout, new NewsFragment());
                        break;
                    case 1:
                        ft.replace(R.id.content_layout, new TechFragment());
                        break;
                    default:
                        break;
                }
                // 提交事务
                ft.commit();

            }
        });

    }


    // 菜单 (看效果:要重绘菜单)
    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.menu_main, menu);
        // 8.1 控制搜索图标的显示/隐藏
        // 左侧抽屉打开,隐藏操作栏按钮
        // 左侧抽屉关闭,显示操作栏按钮
        boolean isOpen = drawerLayout.isDrawerOpen(Gravity.START);
        menu.findItem(R.id.action_search).setVisible(!isOpen);
        return true;
    }


    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // 8.2 点击抽屉把手,打开/关闭抽屉
        if (toggle.onOptionsItemSelected(item)) {
            return true;
        }
        return super.onOptionsItemSelected(item);
    }
}



三、案例二:抽屉布局结合SwipeTab





        本案例与上一个案例,唯一不同的是,当点击左侧抽屉中的“选项一”后,加载了一个碎片:MainFragment ,该碎片使用了 SwipeTab。



        1. strings.xml

<resources>
  <string name="app_name">DrawerNav</string>

  <string name="drawer_open">open</string>
  <string name="drawer_close">close</string>
  <string name="search">搜索</string>
  <string name="title">目录</string>

  <string name="news">新闻</string>
  <string name="tech">科技</string>
  <string name="game">游戏</string>

  <string name="action_settings">Settings</string>

</resources>


        2. menu_main.xml

<menu xmlns:android="http://schemas.android.com/apk/res/android"
      xmlns:tools="http://schemas.android.com/tools"
      tools:context=".MainActivity">
  <item
      android:id="@+id/action_settings"
      android:orderInCategory="100"
      android:showAsAction="never"
      android:title="@string/action_settings"/>

  <item
      android:id="@+id/action_search"
      android:icon="@drawable/ic_search_white_24dp"
      android:orderInCategory="101"
      android:showAsAction="always"
      android:title="@string/search"/>
</menu>


        3. activity_main.xml

<android.support.v4.widget.DrawerLayout
    android:id="@+id/drawer_layout"
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

  <!-- 内容视图 -->
  <FrameLayout
      android:id="@+id/content_layout"
      android:layout_width="match_parent"
      android:layout_height="match_parent"/>

  <!-- 左侧抽屉 -->
  <!--layout_gravity: 定义抽屉的位置(start 左侧,end 右侧)-->
  <include
      layout="@layout/drawer_left"
      android:layout_width="240dp"
      android:layout_height="match_parent"
      android:layout_gravity="start"/>

  <!--右侧抽屉-->
  <ListView
      android:id="@+id/right_drawer"
      android:layout_width="240dp"
      android:layout_height="match_parent"
      android:layout_gravity="end"
      android:background="@android:color/white"/>
</android.support.v4.widget.DrawerLayout>


        4. 三个碎片的布局:

        fragment_news.xml:

<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
             xmlns:tools="http://schemas.android.com/tools"
             android:layout_width="match_parent"
             android:layout_height="match_parent"
             tools:context="com.android.drawernav.NewsFragment">

  <TextView
      android:layout_width="match_parent"
      android:layout_height="match_parent"
      android:text="@string/news"/>

</FrameLayout>


        fragment_tech.xml:

<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
             xmlns:tools="http://schemas.android.com/tools"
             android:layout_width="match_parent"
             android:layout_height="match_parent"
             tools:context="com.android.drawernav.TechFragment">

  <TextView
      android:layout_width="match_parent"
      android:layout_height="match_parent"
      android:text="@string/tech"/>

</FrameLayout>


        fragment_game.xml:

<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
             xmlns:tools="http://schemas.android.com/tools"
             android:layout_width="match_parent"
             android:layout_height="match_parent"
             tools:context="com.android.drawernav.GameFragment">

  <TextView
      android:layout_width="match_parent"
      android:layout_height="match_parent"
      android:text="@string/game"/>

</FrameLayout>


        5. SwipeTab的布局文件和事件:

        (1)fragment_main.xml

<android.support.v4.view.ViewPager
    android:id="@+id/pager"
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"/>

<!--
  因为要调用,因此要写 id。
-->


        (2)MainFragment

package com.android.drawernav;


import android.app.ActionBar;
import android.app.FragmentTransaction;
import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentManager;
import android.support.v4.app.FragmentPagerAdapter;
import android.support.v4.view.ViewPager;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;


/**
 * A simple {@link Fragment} subclass.
 */
public class MainFragment extends Fragment {
    private ActionBar actionBar;
    private String[] tabs = {"新闻", "科技"};
    private ViewPager viewPager;
    private TabFragmentAdapter adapter;

    public MainFragment() {

    }


    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.fragment_main, container, false);
        actionBar = getActivity().getActionBar();

        // 8.创建 ViewPager 适配器
        adapter = new TabFragmentAdapter(getActivity().getSupportFragmentManager());

        // 获得分页控件 (fragment_main.xml)
        viewPager = (ViewPager) view.findViewById(R.id.pager);
        viewPager.setAdapter(adapter);

        // 9. 设置 Pager 改变的监听器
        viewPager.setOnPageChangeListener(new ViewPager.OnPageChangeListener() {
            @Override
            public void onPageScrolled(int i, float v, int i1) {

            }

            @Override
            public void onPageSelected(int i) {
                // Pager 改变时后,选中对应的 Tab
                actionBar.setSelectedNavigationItem(i);
            }

            @Override
            public void onPageScrollStateChanged(int i) {

            }
        });

        // 11.设置应用栏导航模式,文本/监听事件
        actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
        if (actionBar.getTabCount() == 0) {
            for (int i = 0; i < tabs.length; i++) {
                actionBar.addTab(actionBar.newTab().setText(tabs[i]).
                        setTabListener(tabListener));
            }
        }else{
           // 获得 已初始化的 actionBar
        }
        return view;
    }

    private class TabFragmentAdapter extends FragmentPagerAdapter {

        public TabFragmentAdapter(FragmentManager fm) {
            super(fm);
        }

        @Override
        public Fragment getItem(int i) {
            switch (i) {
                case 0:
                    // 需要兼容低版本:android.support.v4.app.Fragment
                    return new NewsFragment();
                case 1:
                    return new TechFragment();
                default:
                    return null;
            }
        }

        @Override
        public int getCount() {
            return tabs.length;
        }
    }

    private ActionBar.TabListener tabListener = new ActionBar.TabListener() {

        @Override
        public void onTabSelected(ActionBar.Tab tab, FragmentTransaction ft) {
            // 点击 Tab,ViewPager 改变选中项
            viewPager.setCurrentItem(tab.getPosition());
        }

        @Override
        public void onTabUnselected(ActionBar.Tab tab, FragmentTransaction ft) {

        }

        @Override
        public void onTabReselected(ActionBar.Tab tab, FragmentTransaction ft) {

        }
    };
}


        6. MainActivity:

package com.android.drawernav;

import android.support.v4.app.FragmentActivity;
import android.support.v4.app.FragmentManager;
import android.support.v4.app.FragmentTransaction;
import android.os.Bundle;

import android.support.v4.app.ActionBarDrawerToggle;
import android.support.v4.widget.DrawerLayout;
import android.view.Gravity;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.AdapterView;
import android.widget.ArrayAdapter;
import android.widget.ListView;


public class MainActivity extends FragmentActivity {
    // 5.声明变量
    private String[] data = {"选项一", "选项二", "选项三", "选项四", "选项五"};
    private DrawerLayout drawerLayout;
    private ActionBarDrawerToggle toggle;
    private ListView listView;

    private FragmentManager fm;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        // 6.初始化变量
        drawerLayout = (DrawerLayout) findViewById(R.id.drawer_layout);
        // 设置抽屉的阴影
        drawerLayout.setDrawerShadow(R.drawable.drawer_shadow, Gravity.START);
        drawerLayout.setDrawerShadow(R.drawable.drawer_shadow, Gravity.END);
        // 显示 Home 按钮(此处显示抽屉把手)
        getActionBar().setDisplayHomeAsUpEnabled(true);
        getActionBar().setHomeButtonEnabled(true);

        fm = this.getSupportFragmentManager();

        // 7.创建抽屉把手(看效果)
        // 参数:上下文,抽屉布局对象,把手菜单图标,打开抽屉的文本,关闭抽屉的文本
        toggle = new ActionBarDrawerToggle(this, drawerLayout,
                R.drawable.ic_menu_white_24dp,
                R.string.drawer_open, R.string.drawer_close
        ) {

            // 抽屉打开
            @Override
            public void onDrawerOpened(View drawerView) {
                super.onDrawerOpened(drawerView);
                // 重建选项菜单
                invalidateOptionsMenu();
                getActionBar().setTitle(getString(R.string.title));
            }

            // 抽屉关闭
            @Override
            public void onDrawerClosed(View drawerView) {
                super.onDrawerClosed(drawerView);
                // 重建选项菜单
                invalidateOptionsMenu();
            }
        };
        // 同步状态(跟新 Home 位置显示的图标)
        toggle.syncState();
        // 设置监听器
        drawerLayout.setDrawerListener(toggle);

        // 9. 初始化左侧抽屉
        initLeftDrawer();
    }

    /**
     * 9.1 初始化左侧抽屉
     */
    private void initLeftDrawer() {
        listView = (ListView) findViewById(R.id.listView);
        // simple_list_item_1 一个控件
        // simple_list_item_2 两个控件
        // simple_list_item_activated_1 当选项选中时,改背景色(激活状态)
        ArrayAdapter adapter = new ArrayAdapter(this,
                android.R.layout.simple_list_item_activated_1, data);
        // 设置单选模式
        listView.setChoiceMode(ListView.CHOICE_MODE_SINGLE);
        listView.setAdapter(adapter);

        // 9.2 点击左侧抽屉列表项的事件处理
        listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
                // 设置标题
                getActionBar().setTitle(data[position]);
                // 关闭抽屉
                drawerLayout.closeDrawer(Gravity.START);

                // 10. 创建 XxxFragment & fragment_xxx.xml
                // 11. 使用 Fragment 替换 内容视图
                FragmentTransaction ft = fm.beginTransaction();
                switch (position) {
                    case 0:
                        ft.replace(R.id.content_layout, new MainFragment());
                        break;
                    case 1:
                        ft.replace(R.id.content_layout, new GameFragment());
                        break;
                    default:
                        // 3 / 4 / 5 (省略)
                        break;
                }
                ft.commit();
            }
        });
    }

    // 8. 菜单 [看效果 - 重建选项菜单 invalidateOptionsMenu()]
    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        getMenuInflater().inflate(R.menu.menu_main, menu);
        // 8.1 搜索图标 显示/隐藏
        // 左侧抽屉打开,隐藏操作栏按钮;
        // 左侧抽屉关闭,显示操作栏按钮
        boolean isOpen = drawerLayout.isDrawerOpen(Gravity.START);
        menu.findItem(R.id.action_search).setVisible(!isOpen);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // 8.2 点击抽屉把手,打开/关闭抽屉
        if (toggle.onOptionsItemSelected(item)) {
            return true;
        }

        return super.onOptionsItemSelected(item);
    }
}
  • 大小: 108.1 KB
  • 大小: 7.8 KB
  • 大小: 10.1 KB
  • 大小: 8.6 KB
  • 大小: 15.3 KB
  • 大小: 26.1 KB
  • 大小: 35.1 KB
  • 大小: 14.3 KB
  • 大小: 8.1 KB
  • 大小: 8.8 KB
  • 大小: 15.7 KB
分享到:
评论

相关推荐

    攻克Data动态获取网页评论,保存数据库

    【攻克Data动态获取网页评论,保存数据库】是一个关于利用特定工具——攻克Data,来抓取网页上的评论数据并存储到数据库的过程。这个过程涉及到网络爬虫技术、JSON解析以及数据库管理等多个IT领域的知识点。 1. **...

    大学生攻克Linux系统教程

    【大学生攻克Linux系统教程】 本教程专为对Linux操作系统感兴趣的初学者设计,旨在提供一个从零开始学习Linux的全面指南。教程内容涵盖了Linux系统的安装、基本操作、文本编辑器VI的使用、调试工具GDB的基础知识,...

    【备战2014】高考化学 精讲巧解分类攻克17

    【备战2014】高考化学 精讲巧解分类攻克17

    24学时攻克c++

    根据提供的信息,“24学时攻克C++”这本书旨在通过一系列高效的学习计划帮助读者在较短的时间内掌握C++编程语言。尽管仅上传了23页的内容,我们仍然可以从书名、描述以及部分可见的内容中推断出一些关键知识点。 ##...

    24小时攻克c++代码

    《24小时攻克C++代码》是一门针对C++编程语言深入浅出的学习教程,旨在帮助初学者在短时间内掌握C++编程基础,并逐步提升至进阶水平。C++是一种广泛应用的面向对象编程语言,它的强大功能和高效性能使得它在系统软件...

    大学生攻克Linux系统教程.rar

    大学生攻克Linux系统教程(又名天下没有难学的Linux)。一位在校大学生整理的教程,献给每一位要征服Linux的学习者-五星级的Linux教程。 本书汇集众多Linux学习者的心血与贡献,有很多作者在默默的为你呼喊,感谢...

    8天攻克8000英语词汇

    8天攻克8000词汇,word版,方便学习,背单词的好东东

    Android程序技术:开拓创新.pptx

    Android 程序技术 本节课程内容:开拓创新 开拓创新 开拓创新 Blaze new trails in a pioneering spirit 开拓创新 Blaze new trails in a pioneering spirit 创新区别于发明 发明是从无到有,而创新是除旧创新。从...

    新东方8天攻克8000单词魔鬼训练营

    标题中的“新东方8天攻克8000单词魔鬼训练营”揭示了一个高效学习英语单词的课程项目。新东方作为知名的教育机构,以其在英语教学领域的专业性而著称,这个训练营可能是一个针对英语学习者设计的高强度、短期记忆...

    Android自动来电录音

    在Android平台上实现自动来电录音是一项技术挑战,但并非无法攻克。Android系统提供了丰富的API和工具,使得开发者可以创建各种功能丰富的应用,包括来电录音。在本文中,我们将深入探讨如何利用Android的...

    【备战2014】高考数学 精讲巧解分类攻克17

    【备战2014】高考数学精讲巧解分类攻克17主要涵盖了高中数学中的基本概念、公式和解题技巧,特别关注了选择题的解答。以下是对部分内容的详细解析: 1. 第一个题目考察了三角函数的诱导公式。已知 sin θ = -1/2,...

    24学时攻克c++_笔记

    从给定的文件信息来看,这是一份关于学习C++编程的笔记,涵盖了从环境搭建到高级主题的多个方面。下面将对这份笔记中的关键知识点进行详细的解释和扩展。 ### 1.... 笔记中提到了几种推荐使用的编译器,包括BCB5、VS...

    如何深入学习Android Framework.pdf

    ### 如何深入学习Android Framework #### 一、学习Android Framework的重要性 在当今移动互联网时代,Android作为全球最大的移动操作系统之一,其重要性不言而喻。对于开发者而言,深入理解Android Framework不仅...

    24学时攻克C++源码及习题答案

    在学习编程语言C++的过程中,24学时的课程安排是一个相当常见的学习进度,这个压缩包文件名为"24学时攻克C++源码及习题答案",显然是为初学者设计的一套完整教程。C++是面向对象编程的重要语言,它的强大功能和广泛...

    【备战2014】高考历史 精讲巧解分类攻克17

    【知识点详解】 1. 牛顿的科学贡献:在历史选择题中,提及了英国1970年发行的英镑纸币,上面描绘了科学家牛顿的形象,提及其发明的反射式望远镜以及钞票上的天体运行轨道图案。这涉及到牛顿的科学成就,包括他在...

    40天攻克大学英语四级

    本文将重点解析“40天攻克大学英语四级”中的核心知识点,尤其是关于作文和翻译部分的倒装句技巧。 倒装句是英语写作中的一种高级表达方式,它能够使句子更具表现力,提升文章的层次感。倒装句分为完全倒装和部分倒...

    大学生攻克Linux系统教程(又名天下没有难学的Linux)

    本教程“大学生攻克Linux系统教程”旨在帮助初学者轻松理解并熟练运用Linux,让看似复杂的Linux变得易学易用。 首先,让我们从基础开始。Linux系统基于Unix,其命令行界面(CLI)是核心操作方式。学习Linux,首先要...

    8天攻克8000单词

    8天攻克8000单词,魔鬼训练营,速度攻克新东方英语

    android-ndkr10环境搭建.pdf

    在本文中,我们将详细介绍如何搭建Android NDK r10开发环境,并通过实例来理解这一过程。Android NDK是Android开发工具的一部分,它允许开发者使用C和C++原生代码来编写部分应用程序,以提高性能或利用特定硬件功能...

Global site tag (gtag.js) - Google Analytics