Android KotlinでNavigationDrawerを実装する

やりたかったこと

あの、横からスッと出てくるメニュー。NavigationDrawerといいます。いろんなアプリでよく見かけます。

activity_main.xml

<android.support.v4.widget.DrawerLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/drawer_layout"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:fitsSystemWindows="true"
    tools:context="jp.sub.takelab.twitmorus.activity.MainActivity"
    tools:openDrawer="start">

    <!-- メインのコンテンツ -->
    <FrameLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent">

        <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
            xmlns:app="http://schemas.android.com/apk/res-auto"
            xmlns:tools="http://schemas.android.com/tools"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:orientation="vertical"
            android:paddingTop="@dimen/activity_vertical_margin"
            android:paddingBottom="@dimen/activity_vertical_margin">

            <android.support.v7.widget.Toolbar
                android:id="@+id/toolbar"
                android:layout_width="match_parent"
                android:layout_height="?attr/actionBarSize"
                android:background="?attr/colorPrimary"
                android:elevation="4dp"
                android:fitsSystemWindows="true"
                android:minHeight="?attr/actionBarSize" />

            <android.support.design.widget.TabLayout
                android:id="@+id/tabs"
                android:layout_width="match_parent"
                android:layout_height="wrap_content" />

            <android.support.v4.view.ViewPager
                android:id="@+id/container"
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                app:layout_behavior="@string/appbar_scrolling_view_behavior" />

            <android.support.design.widget.FloatingActionButton
                android:id="@+id/fab"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_gravity="end|bottom"
                android:layout_marginEnd="33dp"
                android:layout_marginBottom="33dp"
                android:src="@drawable/ic_action_compose" />

        </LinearLayout>
    </FrameLayout>

    <!-- 以下がNavigtionView(横からスッて出てくるメニュー) -->
    <LinearLayout
        android:layout_width="240dp"
        android:layout_height="match_parent"
        android:orientation="vertical"
        android:layout_gravity="start">

        <TextView
            android:id="@+id/mainTwitterAccount"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:padding="40dp"
            android:lines="2"
            android:background="@android:color/darker_gray"
            android:textColor="@android:color/white"
            android:textSize="20sp"
            android:textStyle="bold" />

        <android.support.design.widget.NavigationView
            android:id="@+id/navigation_view"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:choiceMode="singleChoice"
            android:divider="@null"
            android:scrollbars="vertical"
            app:layoutManager="LinearLayoutManager"
            app:menu="@menu/main" />
    </LinearLayout>

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


kotlinコード


         /** * NavigationDrawerの設定 **/ private fun setUpNavigationDrawer() { val toolbar = findViewById<android.support.v7.widget.Toolbar>(R.id.toolbar) val headerText = findViewById<TextView>(R.id.mainTwitterAccount) GlobalScope.launch { headerText.text = TwitterUtils.getTwitterInstance(applicationContext).screenName } drawerLayout = findViewById(R.id.drawer_layout) val drawerToggleButton: ActionBarDrawerToggle = object : ActionBarDrawerToggle( this, drawerLayout, toolbar, R.string.drawer_open, R.string.drawer_close ) { override fun onDrawerClosed(view: View) { super.onDrawerClosed(view) } override fun onDrawerOpened(drawerView: View) { super.onDrawerOpened(drawerView) } } drawerToggleButton.isDrawerIndicatorEnabled = true drawerLayout.addDrawerListener(drawerToggleButton) drawerToggleButton.syncState() findViewById<NavigationView>(R.id.navigation_view).setNavigationItemSelectedListener { when (it.itemId) { R.id.menu_tweet -> { // ここはmenuに記述したidたちを羅列する。 val intent = Intent(this, TweetActivity::class.java) startActivity(intent) true } else -> { // hogehoge false } } } }

実装のポイント

  • メインのコンテンツをFrameLayoutで囲むこと
  • ナビゲーションにしたい親要素にandroid:layout_gravity="start"を入れておくこと
  • setNavigationItemSelectedListenerでなにか処理したあとにtrue or falseをつけないといけません
  • trueを返すと次の動作に移ります。
                GlobalScope.launch {
            headerText.text = TwitterUtils.getTwitterInstance(applicationContext).screenName
        }

これを使うと非同期の通信処理を行えます。簡単に使えるので便利です。

参考
Kotlin Coroutineチュートリアル – qiita

メインコンテンツをFrameLayoutで囲まないと

java.lang.IllegalArgumentException: No drawer view found with gravity LEFT

が発生してクラッシュします。

以上、NavigationDrawer実装の備忘録に。


コメントを残す

このサイトはスパムを低減するために Akismet を使っています。コメントデータの処理方法の詳細はこちらをご覧ください