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 を使っています。コメントデータの処理方法の詳細はこちらをご覧ください