[Jetpack Compose] Firebase Authenticationでログイン画面を作る

Firebaseの準備

Firebase構成管理ファイルをプロジェクトにダウンロードして、Authenticationを有効化、メール / パスワード認証を許可する。

実装

build.gradle (アプリ)

plugins {
	alias(libs.plugins.android.application) apply false
	alias(libs.plugins.jetbrains.kotlin.android) apply false
	// これを追加、バージョンは適したものを
	id("com.google.gms.google-services") version "4.4.1" apply false
}

build.gradle (モジュール)

plugins {
	alias(libs.plugins.android.application)
	alias(libs.plugins.jetbrains.kotlin.android)
	// これを追加
	id("com.google.gms.google-services")
}

dependencies {

	implementation(libs.androidx.core.ktx)
	implementation(libs.androidx.lifecycle.runtime.ktx)
	implementation(libs.androidx.activity.compose)
	implementation(platform(libs.androidx.compose.bom))
	implementation(libs.androidx.ui)
	implementation(libs.androidx.ui.graphics)
	implementation(libs.androidx.ui.tooling.preview)
	implementation(libs.androidx.material3)
	implementation(platform(libs.firebase.bom)) // 追加
	implementation(libs.firebase.auth.ktx) // 追加
	testImplementation(libs.junit)
	androidTestImplementation(libs.androidx.junit)
	androidTestImplementation(libs.androidx.espresso.core)
	androidTestImplementation(platform(libs.androidx.compose.bom))
	androidTestImplementation(libs.androidx.ui.test.junit4)
	debugImplementation(libs.androidx.ui.tooling)
	debugImplementation(libs.androidx.ui.test.manifest)
	implementation(libs.androidx.navigation.compose)
	implementation(libs.coil.compose)
}

libs.versions.toml

[versions]
agp = "8.5.2"
coilCompose = "1.4.0"
kotlin = "1.9.0"
coreKtx = "1.15.0"
junit = "4.13.2"
junitVersion = "1.2.1"
espressoCore = "3.6.1"
lifecycleRuntimeKtx = "2.8.7"
activityCompose = "1.10.1"
composeBom = "2024.04.01" # 追加
navigationCompose = "2.8.9" # 追加
firebaseAuthKtx = "22.3.0" # 追加
firebase-bom = "32.8.1" # 追加

[libraries]
androidx-core-ktx = { group = "androidx.core", name = "core-ktx", version.ref = "coreKtx" }
androidx-navigation-compose = { module = "androidx.navigation:navigation-compose", version.ref = "navigationCompose" }
coil-compose = { module = "io.coil-kt:coil-compose", version.ref = "coilCompose" }
junit = { group = "junit", name = "junit", version.ref = "junit" }
# 追加
firebase-auth-ktx = { group = "com.google.firebase", name = "firebase-auth-ktx", version.ref = "firebaseAuthKtx" }
# 追加
firebase-bom = { module = "com.google.firebase:firebase-bom", version.ref = "firebase-bom" }
androidx-junit = { group = "androidx.test.ext", name = "junit", version.ref = "junitVersion" }
androidx-espresso-core = { group = "androidx.test.espresso", name = "espresso-core", version.ref = "espressoCore" }
androidx-lifecycle-runtime-ktx = { group = "androidx.lifecycle", name = "lifecycle-runtime-ktx", version.ref = "lifecycleRuntimeKtx" }
androidx-activity-compose = { group = "androidx.activity", name = "activity-compose", version.ref = "activityCompose" }
androidx-compose-bom = { group = "androidx.compose", name = "compose-bom", version.ref = "composeBom" }
androidx-ui = { group = "androidx.compose.ui", name = "ui" }
androidx-ui-graphics = { group = "androidx.compose.ui", name = "ui-graphics" }
androidx-ui-tooling = { group = "androidx.compose.ui", name = "ui-tooling" }
androidx-ui-tooling-preview = { group = "androidx.compose.ui", name = "ui-tooling-preview" }
androidx-ui-test-manifest = { group = "androidx.compose.ui", name = "ui-test-manifest" }
androidx-ui-test-junit4 = { group = "androidx.compose.ui", name = "ui-test-junit4" }
androidx-material3 = { group = "androidx.compose.material3", name = "material3" }

[plugins]
android-application = { id = "com.android.application", version.ref = "agp" }
jetbrains-kotlin-android = { id = "org.jetbrains.kotlin.android", version.ref = "kotlin" }

AndroidManifest.xml

インターネット接続を行うので追加

<uses-permission android:name="android.permission.INTERNET" />

MainActivity.kt

サンプルなのでNavigation管理なども一箇所に全部まとめてます、リファクタリングはお好きなように。

object AuthNavRoutes {
	const val AUTH_SCREEN = "auth_screen"
	const val HOME_SCREEN = "home_screen"
}

class MainActivity : ComponentActivity() {
	override fun onCreate(savedInstanceState: Bundle?) {
		super.onCreate(savedInstanceState)
		enableEdgeToEdge()
		setContent {
			MyFirebaseAuthenticationTheme {
				Scaffold(modifier = Modifier.fillMaxSize()) { innerPadding ->
					FirebaseAuthApp(
						modifier = Modifier.padding(innerPadding)
					)
				}
			}
		}
	}
}

@OptIn(ExperimentalMaterial3Api::class)
@Composable
fun FirebaseAuthApp(modifier: Modifier = Modifier) {
	val navController = rememberNavController()
	val auth = remember { FirebaseAuth.getInstance() }
	val context = LocalContext.current
	var isLoggedIn by remember { mutableStateOf(auth.currentUser != null) }

    // ログイン状態監視
	DisposableEffect(auth) {
		val authStateListener = FirebaseAuth.AuthStateListener { firebaseAuth ->
			val user = firebaseAuth.currentUser
			isLoggedIn = (user != null)
			if (user != null) {
				navController.navigate(AuthNavRoutes.HOME_SCREEN) {
					popUpTo(AuthNavRoutes.AUTH_SCREEN) { inclusive = true }
				}
			}
		}
		auth.addAuthStateListener(authStateListener)
		onDispose {
			auth.removeAuthStateListener(authStateListener)
		}
	}

	Scaffold(
		topBar = {
			CenterAlignedTopAppBar(title = { Text("Firebase Auth Demo") })
		}
	) { paddingValues ->
		AuthNavGraph(navController, isLoggedIn, paddingValues, auth, context)
	}
}

@Composable
private fun AuthNavGraph(
	navController: NavHostController,
	isLoggedIn: Boolean,
	paddingValues: PaddingValues,
	auth: FirebaseAuth,
	context: Context
) {
	NavHost(
		navController = navController,
		startDestination = if (isLoggedIn) AuthNavRoutes.HOME_SCREEN else AuthNavRoutes.AUTH_SCREEN,
		modifier = Modifier.padding(paddingValues)
	) {
		composable(AuthNavRoutes.AUTH_SCREEN) {
			AuthScreen(
				auth = auth,
				onAuthSuccess = {
					// 認証成功時の処理はAuthStateListenerでハンドリングされるため、ここでは何もしない
				}
			)
		}
		composable(AuthNavRoutes.HOME_SCREEN) {
			HomeScreen(
				auth = auth,
				onLogout = {
					auth.signOut()
					Toast.makeText(context, "ログアウトしました", Toast.LENGTH_SHORT).show()
				}
			)
		}
	}
}

@Composable
fun AuthScreen(
	auth: FirebaseAuth,
	onAuthSuccess: () -> Unit
) {
	var email by remember { mutableStateOf("") }
	var password by remember { mutableStateOf("") }
	var isLoading by remember { mutableStateOf(false) }
	val context = LocalContext.current
	val scope = rememberCoroutineScope()

	Column(
		modifier = Modifier
			.fillMaxSize()
			.padding(24.dp),
		horizontalAlignment = Alignment.CenterHorizontally,
		verticalArrangement = Arrangement.Center
	) {
		Text(text = "Firebase認証", fontSize = 32.sp)
		Spacer(modifier = Modifier.height(32.dp))
		TextField(
			value = email,
			onValueChange = { email = it },
			label = { Text("メールアドレス") },
			modifier = Modifier.fillMaxWidth()
		)
		Spacer(modifier = Modifier.height(16.dp))
		TextField(
			value = password,
			onValueChange = { password = it },
			label = { Text("パスワード") },
			visualTransformation = PasswordVisualTransformation(),
			modifier = Modifier.fillMaxWidth()
		)
		Spacer(modifier = Modifier.height(32.dp))

		if (isLoading) {
			CircularProgressIndicator(modifier = Modifier.size(48.dp))
		} else {
			Button(
				onClick = {
					isLoading = true
					scope.launch {
						try {
							auth.createUserWithEmailAndPassword(email, password).await()
							Toast.makeText(context, "サインアップ成功",  Toast.LENGTH_SHORT).show()
							onAuthSuccess()
						} catch (e: Exception) {
							Toast.makeText(context, "サインアップ失敗: ${e.message}", Toast.LENGTH_LONG).show()
							e.printStackTrace()
						} finally {
							isLoading = false
						}
					}
				},
				enabled = email.isNotBlank() && password.isNotBlank(),
				modifier = Modifier.fillMaxWidth()
			) {
				Text("サインアップ", fontSize = 20.sp)
			}
			Spacer(modifier = Modifier.height(16.dp))
			Button(
				onClick = {
					isLoading = true
					scope.launch {
						try {
							auth.signInWithEmailAndPassword(email, password).await()
							Toast.makeText(context, "ログイン成功!", Toast.LENGTH_SHORT).show()
							onAuthSuccess()
						} catch (e: Exception) {
							Toast.makeText(context, "ログイン失敗: ${e.message}", Toast.LENGTH_LONG).show()
							e.printStackTrace() // エラー詳細をログに出力
						} finally {
							isLoading = false
						}
					}
				},
				enabled = email.isNotBlank() && password.isNotBlank(),
				modifier = Modifier.fillMaxWidth()
			) {
				Text("ログイン", fontSize = 20.sp)
			}
		}
	}
}

@Composable
fun HomeScreen(auth: FirebaseAuth, onLogout: () -> Unit) {
	val currentUser = auth.currentUser // 現在のログインユーザー情報を取得

	Column(
		modifier = Modifier
			.fillMaxSize()
			.padding(24.dp),
		horizontalAlignment = Alignment.CenterHorizontally,
		verticalArrangement = Arrangement.Center
	) {
		Text(text = "ログイン成功!", fontSize = 32.sp)
		Spacer(modifier = Modifier.height(16.dp))
		Text(
			text = "ようこそ、${currentUser?.email ?: "ゲスト"}さん!",
			fontSize = 24.sp
		)
		Spacer(modifier = Modifier.height(32.dp))
		Button(
			onClick = onLogout,
			modifier = Modifier.fillMaxWidth()
		) {
			Text("ログアウト", fontSize = 20.sp)
		}
	}
}