commit a8e53e344c215572430f0ed20a96c9076750a3ff Author: Ultradesu Date: Fri Jun 5 12:04:49 2026 +0300 init diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..aa724b7 --- /dev/null +++ b/.gitignore @@ -0,0 +1,15 @@ +*.iml +.gradle +/local.properties +/.idea/caches +/.idea/libraries +/.idea/modules.xml +/.idea/workspace.xml +/.idea/navEditor.xml +/.idea/assetWizardSettings.xml +.DS_Store +/build +/captures +.externalNativeBuild +.cxx +local.properties diff --git a/.idea/.gitignore b/.idea/.gitignore new file mode 100644 index 0000000..26d3352 --- /dev/null +++ b/.idea/.gitignore @@ -0,0 +1,3 @@ +# Default ignored files +/shelf/ +/workspace.xml diff --git a/.idea/.name b/.idea/.name new file mode 100644 index 0000000..7a9188e --- /dev/null +++ b/.idea/.name @@ -0,0 +1 @@ +furumi-android \ No newline at end of file diff --git a/.idea/AndroidProjectSystem.xml b/.idea/AndroidProjectSystem.xml new file mode 100644 index 0000000..4a53bee --- /dev/null +++ b/.idea/AndroidProjectSystem.xml @@ -0,0 +1,6 @@ + + + + + \ No newline at end of file diff --git a/.idea/codeStyles/Project.xml b/.idea/codeStyles/Project.xml new file mode 100644 index 0000000..7643783 --- /dev/null +++ b/.idea/codeStyles/Project.xml @@ -0,0 +1,123 @@ + + + + + + + + + + \ No newline at end of file diff --git a/.idea/codeStyles/codeStyleConfig.xml b/.idea/codeStyles/codeStyleConfig.xml new file mode 100644 index 0000000..79ee123 --- /dev/null +++ b/.idea/codeStyles/codeStyleConfig.xml @@ -0,0 +1,5 @@ + + + + \ No newline at end of file diff --git a/.idea/compiler.xml b/.idea/compiler.xml new file mode 100644 index 0000000..b86273d --- /dev/null +++ b/.idea/compiler.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/.idea/deploymentTargetSelector.xml b/.idea/deploymentTargetSelector.xml new file mode 100644 index 0000000..be32e44 --- /dev/null +++ b/.idea/deploymentTargetSelector.xml @@ -0,0 +1,18 @@ + + + + + + + + + \ No newline at end of file diff --git a/.idea/deviceManager.xml b/.idea/deviceManager.xml new file mode 100644 index 0000000..91f9558 --- /dev/null +++ b/.idea/deviceManager.xml @@ -0,0 +1,13 @@ + + + + + + \ No newline at end of file diff --git a/.idea/gradle.xml b/.idea/gradle.xml new file mode 100644 index 0000000..02c4aa5 --- /dev/null +++ b/.idea/gradle.xml @@ -0,0 +1,18 @@ + + + + + + + \ No newline at end of file diff --git a/.idea/inspectionProfiles/Project_Default.xml b/.idea/inspectionProfiles/Project_Default.xml new file mode 100644 index 0000000..7061a0d --- /dev/null +++ b/.idea/inspectionProfiles/Project_Default.xml @@ -0,0 +1,61 @@ + + + + \ No newline at end of file diff --git a/.idea/misc.xml b/.idea/misc.xml new file mode 100644 index 0000000..74dd639 --- /dev/null +++ b/.idea/misc.xml @@ -0,0 +1,10 @@ + + + + + + + + + \ No newline at end of file diff --git a/.idea/runConfigurations.xml b/.idea/runConfigurations.xml new file mode 100644 index 0000000..16660f1 --- /dev/null +++ b/.idea/runConfigurations.xml @@ -0,0 +1,17 @@ + + + + + + \ No newline at end of file diff --git a/.idea/vcs.xml b/.idea/vcs.xml new file mode 100644 index 0000000..94a25f7 --- /dev/null +++ b/.idea/vcs.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/app/.gitignore b/app/.gitignore new file mode 100644 index 0000000..42afabf --- /dev/null +++ b/app/.gitignore @@ -0,0 +1 @@ +/build \ No newline at end of file diff --git a/app/build.gradle.kts b/app/build.gradle.kts new file mode 100644 index 0000000..990516a --- /dev/null +++ b/app/build.gradle.kts @@ -0,0 +1,77 @@ +plugins { + alias(libs.plugins.android.application) + alias(libs.plugins.kotlin.compose) + id("com.google.devtools.ksp") + id("com.google.dagger.hilt.android") +} + +android { + namespace = "com.example.furumi_android" + compileSdk = 35 + + defaultConfig { + applicationId = "com.example.furumi_android" + minSdk = 24 + targetSdk = 35 + versionCode = 1 + versionName = "1.0" + + testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" + } + + buildTypes { + release { + optimization { + enable = false + } + } + } + compileOptions { + sourceCompatibility = JavaVersion.VERSION_11 + targetCompatibility = JavaVersion.VERSION_11 + } + buildFeatures { + compose = true + } +} + +dependencies { + implementation(platform(libs.androidx.compose.bom)) + implementation(libs.androidx.activity.compose) + implementation(libs.androidx.compose.material3) + implementation(libs.androidx.compose.ui) + implementation(libs.androidx.compose.ui.graphics) + implementation(libs.androidx.compose.ui.tooling.preview) + implementation(libs.androidx.core.ktx) + implementation(libs.androidx.lifecycle.runtime.ktx) + implementation(libs.androidx.lifecycle.viewmodel.compose) + implementation(libs.androidx.navigation.compose) + + // Networking + implementation(libs.retrofit) + implementation(libs.retrofit.moshi) + implementation(libs.okhttp) + implementation(libs.okhttp.logging) + implementation(libs.moshi) + implementation(libs.androidx.media3.exoplayer) + implementation(libs.androidx.media3.datasource.okhttp) + implementation(libs.androidx.media3.session) + + // Security & Storage + implementation(libs.androidx.security.crypto) + implementation(libs.androidx.datastore.preferences) + implementation(libs.androidx.browser) + + // DI + implementation(libs.hilt.android) + ksp(libs.hilt.compiler) + implementation(libs.hilt.navigation.compose) + + testImplementation(libs.junit) + androidTestImplementation(platform(libs.androidx.compose.bom)) + androidTestImplementation(libs.androidx.compose.ui.test.junit4) + androidTestImplementation(libs.androidx.espresso.core) + androidTestImplementation(libs.androidx.junit) + debugImplementation(libs.androidx.compose.ui.test.manifest) + debugImplementation(libs.androidx.compose.ui.tooling) +} diff --git a/app/src/androidTest/java/com/example/furumi_android/ExampleInstrumentedTest.kt b/app/src/androidTest/java/com/example/furumi_android/ExampleInstrumentedTest.kt new file mode 100644 index 0000000..0cb5b0b --- /dev/null +++ b/app/src/androidTest/java/com/example/furumi_android/ExampleInstrumentedTest.kt @@ -0,0 +1,24 @@ +package com.example.furumi_android + +import androidx.test.platform.app.InstrumentationRegistry +import androidx.test.ext.junit.runners.AndroidJUnit4 + +import org.junit.Test +import org.junit.runner.RunWith + +import org.junit.Assert.* + +/** + * Instrumented test, which will execute on an Android device. + * + * See [testing documentation](http://d.android.com/tools/testing). + */ +@RunWith(AndroidJUnit4::class) +class ExampleInstrumentedTest { + @Test + fun useAppContext() { + // Context of the app under test. + val appContext = InstrumentationRegistry.getInstrumentation().targetContext + assertEquals("com.example.furumi_android", appContext.packageName) + } +} \ No newline at end of file diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml new file mode 100644 index 0000000..a27ead7 --- /dev/null +++ b/app/src/main/AndroidManifest.xml @@ -0,0 +1,61 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/app/src/main/java/com/example/furumi_android/FurumiApplication.kt b/app/src/main/java/com/example/furumi_android/FurumiApplication.kt new file mode 100644 index 0000000..da8fe96 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/FurumiApplication.kt @@ -0,0 +1,7 @@ +package com.example.furumi_android + +import android.app.Application +import dagger.hilt.android.HiltAndroidApp + +@HiltAndroidApp +class FurumiApplication : Application() diff --git a/app/src/main/java/com/example/furumi_android/MainActivity.kt b/app/src/main/java/com/example/furumi_android/MainActivity.kt new file mode 100644 index 0000000..18487e0 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/MainActivity.kt @@ -0,0 +1,113 @@ +package com.example.furumi_android + +import android.Manifest +import android.content.Intent +import android.content.pm.PackageManager +import android.net.Uri +import android.os.Build +import android.os.Bundle +import androidx.activity.result.contract.ActivityResultContracts +import androidx.activity.ComponentActivity +import androidx.activity.compose.setContent +import androidx.activity.enableEdgeToEdge +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.padding +import androidx.compose.material3.Scaffold +import androidx.compose.runtime.Composable +import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.setValue +import androidx.compose.ui.Modifier +import androidx.hilt.navigation.compose.hiltViewModel +import androidx.navigation.compose.NavHost +import androidx.navigation.compose.composable +import androidx.navigation.compose.rememberNavController +import com.example.furumi_android.ui.login.LoginScreen +import com.example.furumi_android.ui.login.LoginViewModel +import com.example.furumi_android.ui.player.PlayerScreen +import com.example.furumi_android.ui.theme.FurumiandroidTheme +import dagger.hilt.android.AndroidEntryPoint + +@AndroidEntryPoint +class MainActivity : ComponentActivity() { + + private var deepLinkUri by mutableStateOf(null) + private val notificationPermissionLauncher = registerForActivityResult( + ActivityResultContracts.RequestPermission() + ) { } + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + enableEdgeToEdge() + requestNotificationPermissionIfNeeded() + + if (deepLinkUri == null) { + deepLinkUri = intent?.data + } + + setContent { + FurumiandroidTheme { + Scaffold(modifier = Modifier.fillMaxSize()) { innerPadding -> + Box(modifier = Modifier.padding(innerPadding)) { + FurumiApp( + deepLinkUri = deepLinkUri, + onDeepLinkHandled = { deepLinkUri = null } + ) + } + } + } + } + } + + override fun onNewIntent(intent: Intent) { + super.onNewIntent(intent) + setIntent(intent) + deepLinkUri = intent.data + } + + private fun requestNotificationPermissionIfNeeded() { + if (Build.VERSION.SDK_INT < Build.VERSION_CODES.TIRAMISU) return + + if (checkSelfPermission(Manifest.permission.POST_NOTIFICATIONS) != PackageManager.PERMISSION_GRANTED) { + notificationPermissionLauncher.launch(Manifest.permission.POST_NOTIFICATIONS) + } + } +} + +@Composable +fun FurumiApp(deepLinkUri: Uri?, onDeepLinkHandled: () -> Unit) { + val navController = rememberNavController() + + NavHost(navController = navController, startDestination = "login") { + composable("login") { + val loginViewModel: LoginViewModel = hiltViewModel() + + // Handle deep link when it arrives + androidx.compose.runtime.LaunchedEffect(deepLinkUri) { + deepLinkUri?.let { uri -> + loginViewModel.handleDeepLink(uri) + onDeepLinkHandled() + } + } + + LoginScreen( + viewModel = loginViewModel, + onLoginSuccess = { + navController.navigate("player") { + popUpTo("login") { inclusive = true } + } + } + ) + } + composable("player") { + PlayerScreen( + onLoggedOut = { + navController.navigate("login") { + popUpTo("player") { inclusive = true } + } + } + ) + } + } +} diff --git a/app/src/main/java/com/example/furumi_android/data/local/AuthSessionStorage.kt b/app/src/main/java/com/example/furumi_android/data/local/AuthSessionStorage.kt new file mode 100644 index 0000000..f2c802e --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/data/local/AuthSessionStorage.kt @@ -0,0 +1,117 @@ +package com.example.furumi_android.data.local + +import android.content.Context +import androidx.security.crypto.EncryptedSharedPreferences +import androidx.security.crypto.MasterKey +import com.example.furumi_android.domain.model.AuthSession +import com.example.furumi_android.domain.model.AuthTokens +import com.example.furumi_android.domain.model.User +import com.squareup.moshi.Moshi + +class AuthSessionStorage(context: Context, moshi: Moshi) { + private val masterKey = MasterKey.Builder(context) + .setKeyScheme(MasterKey.KeyScheme.AES256_GCM) + .build() + + private val prefs = EncryptedSharedPreferences.create( + context, + PREFS_NAME, + masterKey, + EncryptedSharedPreferences.PrefKeyEncryptionScheme.AES256_SIV, + EncryptedSharedPreferences.PrefValueEncryptionScheme.AES256_GCM + ) + + private val userAdapter = moshi.adapter(User::class.java) + + fun saveSession(session: AuthSession) { + prefs.edit() + .putString(KEY_BASE_URL, session.serverBaseUrl) + .putString(KEY_ACCESS_TOKEN, session.tokens.accessToken) + .putString(KEY_REFRESH_TOKEN, session.tokens.refreshToken) + .putString(KEY_TOKEN_TYPE, session.tokens.tokenType) + .putInt(KEY_EXPIRES_IN_SECONDS, session.tokens.expiresInSeconds) + .putLong(KEY_EXPIRES_AT_EPOCH_SECONDS, session.tokens.expiresAtEpochSeconds) + .putString(KEY_USER_DATA, userAdapter.toJson(session.user)) + .remove(KEY_PENDING_SSO_BASE_URL) + .apply() + } + + fun saveTokens(tokens: AuthTokens) { + prefs.edit() + .putString(KEY_ACCESS_TOKEN, tokens.accessToken) + .putString(KEY_REFRESH_TOKEN, tokens.refreshToken) + .putString(KEY_TOKEN_TYPE, tokens.tokenType) + .putInt(KEY_EXPIRES_IN_SECONDS, tokens.expiresInSeconds) + .putLong(KEY_EXPIRES_AT_EPOCH_SECONDS, tokens.expiresAtEpochSeconds) + .apply() + } + + fun savePendingSsoBaseUrl(baseUrl: String) { + prefs.edit() + .putString(KEY_PENDING_SSO_BASE_URL, baseUrl) + .apply() + } + + fun getPendingSsoBaseUrl(): String? { + return prefs.getString(KEY_PENDING_SSO_BASE_URL, null) + } + + fun getBaseUrl(): String? { + return prefs.getString(KEY_BASE_URL, null) + } + + fun getSession(): AuthSession? { + val baseUrl = getBaseUrl() ?: return null + val user = getUser() ?: return null + val tokens = getTokens() ?: return null + return AuthSession(baseUrl, user, tokens) + } + + fun getTokens(): AuthTokens? { + val accessToken = prefs.getString(KEY_ACCESS_TOKEN, null) ?: return null + val refreshToken = prefs.getString(KEY_REFRESH_TOKEN, null) ?: return null + val tokenType = prefs.getString(KEY_TOKEN_TYPE, AuthTokens.DEFAULT_TOKEN_TYPE) + ?: AuthTokens.DEFAULT_TOKEN_TYPE + + return AuthTokens( + accessToken = accessToken, + refreshToken = refreshToken, + tokenType = tokenType, + expiresInSeconds = prefs.getInt(KEY_EXPIRES_IN_SECONDS, 0), + expiresAtEpochSeconds = prefs.getLong(KEY_EXPIRES_AT_EPOCH_SECONDS, 0L) + ) + } + + private fun getUser(): User? { + val json = prefs.getString(KEY_USER_DATA, null) ?: return null + return runCatching { userAdapter.fromJson(json) }.getOrNull() + } + + fun clear() { + prefs.edit().clear().apply() + } + + fun clearSession() { + prefs.edit() + .remove(KEY_ACCESS_TOKEN) + .remove(KEY_REFRESH_TOKEN) + .remove(KEY_TOKEN_TYPE) + .remove(KEY_EXPIRES_IN_SECONDS) + .remove(KEY_EXPIRES_AT_EPOCH_SECONDS) + .remove(KEY_PENDING_SSO_BASE_URL) + .remove(KEY_USER_DATA) + .apply() + } + + companion object { + private const val PREFS_NAME = "auth_session" + private const val KEY_ACCESS_TOKEN = "access_token" + private const val KEY_REFRESH_TOKEN = "refresh_token" + private const val KEY_TOKEN_TYPE = "token_type" + private const val KEY_EXPIRES_IN_SECONDS = "expires_in_seconds" + private const val KEY_EXPIRES_AT_EPOCH_SECONDS = "expires_at_epoch_seconds" + private const val KEY_BASE_URL = "base_url" + private const val KEY_PENDING_SSO_BASE_URL = "pending_sso_base_url" + private const val KEY_USER_DATA = "user_data" + } +} diff --git a/app/src/main/java/com/example/furumi_android/data/remote/AccessTokenInterceptor.kt b/app/src/main/java/com/example/furumi_android/data/remote/AccessTokenInterceptor.kt new file mode 100644 index 0000000..6d6182f --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/data/remote/AccessTokenInterceptor.kt @@ -0,0 +1,33 @@ +package com.example.furumi_android.data.remote + +import com.example.furumi_android.data.local.AuthSessionStorage +import okhttp3.Interceptor +import okhttp3.Response +import javax.inject.Inject + +class AccessTokenInterceptor @Inject constructor( + private val sessionStorage: AuthSessionStorage, + private val authEndpoints: AuthEndpoints +) : Interceptor { + + override fun intercept(chain: Interceptor.Chain): Response { + val request = chain.request() + + if (request.header(HEADER_AUTHORIZATION) != null || + authEndpoints.isLoginOrRefreshPath(request.url.encodedPath) + ) { + return chain.proceed(request) + } + + val tokens = sessionStorage.getTokens() ?: return chain.proceed(request) + val authorizedRequest = request.newBuilder() + .header(HEADER_AUTHORIZATION, tokens.authorizationHeader) + .build() + + return chain.proceed(authorizedRequest) + } + + companion object { + const val HEADER_AUTHORIZATION = "Authorization" + } +} diff --git a/app/src/main/java/com/example/furumi_android/data/remote/AuthApiErrorParser.kt b/app/src/main/java/com/example/furumi_android/data/remote/AuthApiErrorParser.kt new file mode 100644 index 0000000..2efe35a --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/data/remote/AuthApiErrorParser.kt @@ -0,0 +1,31 @@ +package com.example.furumi_android.data.remote + +import com.example.furumi_android.data.remote.model.ErrorResponse +import com.squareup.moshi.Moshi +import retrofit2.Response +import javax.inject.Inject + +class AuthApiErrorParser @Inject constructor( + moshi: Moshi +) { + private val errorAdapter = moshi.adapter(ErrorResponse::class.java) + + fun messageFrom(response: Response<*>): String { + val code = response.code() + val rawError = response.errorBody()?.string() + + if (rawError.isNullOrBlank()) { + return "Server error $code" + } + + val apiError = runCatching { errorAdapter.fromJson(rawError)?.error } + .getOrNull() + ?.takeIf { it.isNotBlank() } + + return apiError ?: "Server error $code: ${rawError.take(MAX_ERROR_PREVIEW_LENGTH)}" + } + + companion object { + private const val MAX_ERROR_PREVIEW_LENGTH = 160 + } +} diff --git a/app/src/main/java/com/example/furumi_android/data/remote/AuthAuthenticator.kt b/app/src/main/java/com/example/furumi_android/data/remote/AuthAuthenticator.kt new file mode 100644 index 0000000..96b54a9 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/data/remote/AuthAuthenticator.kt @@ -0,0 +1,94 @@ +package com.example.furumi_android.data.remote + +import com.example.furumi_android.data.local.AuthSessionStorage +import com.example.furumi_android.data.remote.AccessTokenInterceptor.Companion.HEADER_AUTHORIZATION +import com.example.furumi_android.data.remote.api.AuthApi +import com.example.furumi_android.data.remote.model.RefreshRequest +import com.example.furumi_android.data.remote.model.toDomain +import dagger.Lazy +import kotlinx.coroutines.runBlocking +import okhttp3.Authenticator +import okhttp3.Request +import okhttp3.Response +import okhttp3.Route +import javax.inject.Inject + +class AuthAuthenticator @Inject constructor( + private val sessionStorage: AuthSessionStorage, + private val authEndpoints: AuthEndpoints, + private val authApiLazy: Lazy +) : Authenticator { + + private val refreshLock = Any() + + override fun authenticate(route: Route?, response: Response): Request? { + if (responseCount(response) >= MAX_AUTH_ATTEMPTS || + authEndpoints.isLoginOrRefreshPath(response.request.url.encodedPath) + ) { + return null + } + + val requestAuthorization = response.request.header(HEADER_AUTHORIZATION) ?: return null + + return synchronized(refreshLock) { + val currentTokens = sessionStorage.getTokens() ?: return@synchronized null + if (requestAuthorization != currentTokens.authorizationHeader) { + return@synchronized response.withAuthorization(currentTokens.authorizationHeader) + } + + val baseUrl = sessionStorage.getBaseUrl() ?: return@synchronized null + val refreshedTokens = runBlocking { + refreshTokens(baseUrl, currentTokens.refreshToken) + } + + if (refreshedTokens == null) { + sessionStorage.clearSession() + null + } else { + sessionStorage.saveTokens(refreshedTokens) + response.withAuthorization(refreshedTokens.authorizationHeader) + } + } + } + + private suspend fun refreshTokens( + baseUrl: String, + refreshToken: String + ): com.example.furumi_android.domain.model.AuthTokens? { + val refreshResponse = runCatching { + authApiLazy.get().refresh( + url = authEndpoints.refresh(baseUrl), + request = RefreshRequest(refreshToken) + ) + }.getOrNull() + + if (refreshResponse?.isSuccessful != true) { + return null + } + + val body = refreshResponse.body() ?: return null + return body.toDomain(nowEpochSeconds()) + } + + private fun Response.withAuthorization(authorizationHeader: String): Request { + return request.newBuilder() + .header(HEADER_AUTHORIZATION, authorizationHeader) + .build() + } + + private fun responseCount(response: Response): Int { + var count = 1 + var priorResponse = response.priorResponse + while (priorResponse != null) { + count++ + priorResponse = priorResponse.priorResponse + } + return count + } + + private fun nowEpochSeconds(): Long = System.currentTimeMillis() / 1000L + + companion object { + private const val MAX_AUTH_ATTEMPTS = 2 + } +} diff --git a/app/src/main/java/com/example/furumi_android/data/remote/AuthEndpoints.kt b/app/src/main/java/com/example/furumi_android/data/remote/AuthEndpoints.kt new file mode 100644 index 0000000..010054c --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/data/remote/AuthEndpoints.kt @@ -0,0 +1,67 @@ +package com.example.furumi_android.data.remote + +import android.net.Uri +import com.example.furumi_android.domain.model.AuthException +import javax.inject.Inject + +class AuthEndpoints @Inject constructor() { + fun passwordLogin(baseUrl: String): String = "$baseUrl$PASSWORD_LOGIN_PATH" + + fun ssoExchange(baseUrl: String): String = "$baseUrl$SSO_EXCHANGE_PATH" + + fun refresh(baseUrl: String): String = "$baseUrl$REFRESH_PATH" + + fun logout(baseUrl: String): String = "$baseUrl$LOGOUT_PATH" + + fun ssoStart(baseUrl: String): String { + return Uri.parse("$baseUrl$SSO_START_PATH") + .buildUpon() + .appendQueryParameter("redirect_uri", SSO_REDIRECT_URI) + .build() + .toString() + } + + fun parseSsoCallback(callbackUrl: String): Result { + val uri = runCatching { Uri.parse(callbackUrl) }.getOrElse { + return Result.failure(AuthException("Invalid SSO callback")) + } + + if (uri.scheme != SSO_REDIRECT_SCHEME || uri.host != SSO_REDIRECT_HOST || uri.path != SSO_REDIRECT_PATH) { + return Result.failure(AuthException("Unexpected SSO callback")) + } + + uri.getQueryParameter("error")?.let { error -> + return Result.failure(AuthException("SSO error: $error")) + } + + val code = uri.getQueryParameter("code") + if (code.isNullOrBlank()) { + return Result.failure(AuthException("SSO callback does not contain an authorization code")) + } + + return Result.success(SsoCallback(code)) + } + + fun isLoginOrRefreshPath(path: String): Boolean { + return path.endsWith(PASSWORD_LOGIN_PATH) || + path.endsWith(SSO_EXCHANGE_PATH) || + path.endsWith(REFRESH_PATH) + } + + companion object { + const val SSO_REDIRECT_URI = "furumi://auth/callback" + + private const val PASSWORD_LOGIN_PATH = "/api/auth/password" + private const val SSO_EXCHANGE_PATH = "/api/auth/sso/exchange" + private const val REFRESH_PATH = "/api/auth/refresh" + private const val LOGOUT_PATH = "/api/auth/logout" + private const val SSO_START_PATH = "/auth/mobile/oidc/start" + private const val SSO_REDIRECT_SCHEME = "furumi" + private const val SSO_REDIRECT_HOST = "auth" + private const val SSO_REDIRECT_PATH = "/callback" + } +} + +data class SsoCallback( + val code: String +) diff --git a/app/src/main/java/com/example/furumi_android/data/remote/PlayerEndpoints.kt b/app/src/main/java/com/example/furumi_android/data/remote/PlayerEndpoints.kt new file mode 100644 index 0000000..5725bc1 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/data/remote/PlayerEndpoints.kt @@ -0,0 +1,19 @@ +package com.example.furumi_android.data.remote + +import javax.inject.Inject + +class PlayerEndpoints @Inject constructor() { + fun artists(baseUrl: String): String = "$baseUrl$ARTISTS_PATH" + + fun artistDetail(baseUrl: String, artistId: Long): String = "$baseUrl$ARTISTS_PATH/$artistId" + + fun releaseDetail(baseUrl: String, releaseId: Long): String = "$baseUrl$RELEASES_PATH/$releaseId" + + fun history(baseUrl: String): String = "$baseUrl$HISTORY_PATH" + + companion object { + private const val ARTISTS_PATH = "/api/player/artists" + private const val RELEASES_PATH = "/api/player/releases" + private const val HISTORY_PATH = "/api/player/history" + } +} diff --git a/app/src/main/java/com/example/furumi_android/data/remote/api/AuthApi.kt b/app/src/main/java/com/example/furumi_android/data/remote/api/AuthApi.kt new file mode 100644 index 0000000..25bdaa6 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/data/remote/api/AuthApi.kt @@ -0,0 +1,46 @@ +package com.example.furumi_android.data.remote.api + +import com.example.furumi_android.data.remote.model.LoginRequest +import com.example.furumi_android.data.remote.model.LoginResponse +import com.example.furumi_android.data.remote.model.LogoutRequest +import com.example.furumi_android.data.remote.model.LogoutResponse +import com.example.furumi_android.data.remote.model.RefreshRequest +import com.example.furumi_android.data.remote.model.SsoExchangeRequest +import com.example.furumi_android.data.remote.model.TokenResponse +import retrofit2.Response +import retrofit2.http.Body +import retrofit2.http.Header +import retrofit2.http.Headers +import retrofit2.http.POST +import retrofit2.http.Url + +interface AuthApi { + @Headers("Accept: application/json") + @POST + suspend fun login( + @Url url: String, + @Body request: LoginRequest + ): Response + + @Headers("Accept: application/json") + @POST + suspend fun ssoExchange( + @Url url: String, + @Body request: SsoExchangeRequest + ): Response + + @Headers("Accept: application/json") + @POST + suspend fun refresh( + @Url url: String, + @Body request: RefreshRequest + ): Response + + @Headers("Accept: application/json") + @POST + suspend fun logout( + @Url url: String, + @Header("Authorization") bearerToken: String, + @Body request: LogoutRequest + ): Response +} diff --git a/app/src/main/java/com/example/furumi_android/data/remote/api/PlayerApi.kt b/app/src/main/java/com/example/furumi_android/data/remote/api/PlayerApi.kt new file mode 100644 index 0000000..e739a2a --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/data/remote/api/PlayerApi.kt @@ -0,0 +1,42 @@ +package com.example.furumi_android.data.remote.api + +import com.example.furumi_android.data.remote.model.ArtistPageResponse +import com.example.furumi_android.data.remote.model.ArtistDetailResponse +import com.example.furumi_android.data.remote.model.PlayHistoryPageResponse +import com.example.furumi_android.data.remote.model.ReleaseDetailResponse +import retrofit2.Response +import retrofit2.http.GET +import retrofit2.http.Headers +import retrofit2.http.Query +import retrofit2.http.Url + +interface PlayerApi { + @Headers("Accept: application/json") + @GET + suspend fun artists( + @Url url: String, + @Query("page") page: Int, + @Query("limit") limit: Int, + @Query("mine") mine: Boolean + ): Response + + @Headers("Accept: application/json") + @GET + suspend fun artistDetail( + @Url url: String + ): Response + + @Headers("Accept: application/json") + @GET + suspend fun releaseDetail( + @Url url: String + ): Response + + @Headers("Accept: application/json") + @GET + suspend fun history( + @Url url: String, + @Query("page") page: Int, + @Query("limit") limit: Int + ): Response +} diff --git a/app/src/main/java/com/example/furumi_android/data/remote/model/AuthMappers.kt b/app/src/main/java/com/example/furumi_android/data/remote/model/AuthMappers.kt new file mode 100644 index 0000000..bc9ee38 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/data/remote/model/AuthMappers.kt @@ -0,0 +1,38 @@ +package com.example.furumi_android.data.remote.model + +import com.example.furumi_android.domain.model.AuthSession +import com.example.furumi_android.domain.model.AuthTokens +import com.example.furumi_android.domain.model.User + +fun LoginResponse.toDomainSession( + serverBaseUrl: String, + receivedAtEpochSeconds: Long +): AuthSession { + return AuthSession( + serverBaseUrl = serverBaseUrl, + user = user.toDomain(), + tokens = tokens.toDomain(receivedAtEpochSeconds) + ) +} + +fun UserResponse.toDomain(): User { + return User( + id = id, + name = name, + role = role + ) +} + +fun TokenResponse.toDomain(receivedAtEpochSeconds: Long): AuthTokens { + return AuthTokens( + accessToken = accessToken, + refreshToken = refreshToken, + tokenType = tokenType, + expiresInSeconds = expiresInSeconds, + expiresAtEpochSeconds = if (expiresInSeconds > 0) { + receivedAtEpochSeconds + expiresInSeconds + } else { + 0L + } + ) +} diff --git a/app/src/main/java/com/example/furumi_android/data/remote/model/AuthRequests.kt b/app/src/main/java/com/example/furumi_android/data/remote/model/AuthRequests.kt new file mode 100644 index 0000000..b6d15c1 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/data/remote/model/AuthRequests.kt @@ -0,0 +1,26 @@ +package com.example.furumi_android.data.remote.model + +import com.squareup.moshi.Json + +data class LoginRequest( + @param:Json(name = "username") val username: String, + @param:Json(name = "password") val password: String, + @param:Json(name = "device_name") val deviceName: String +) + +data class SsoExchangeRequest( + @param:Json(name = "code") val code: String, + @param:Json(name = "device_name") val deviceName: String +) + +data class RefreshRequest( + @param:Json(name = "refresh_token") val refreshToken: String +) + +data class LogoutRequest( + @param:Json(name = "refresh_token") val refreshToken: String +) + +data class LogoutResponse( + @param:Json(name = "revoked") val revoked: Boolean +) diff --git a/app/src/main/java/com/example/furumi_android/data/remote/model/AuthResponses.kt b/app/src/main/java/com/example/furumi_android/data/remote/model/AuthResponses.kt new file mode 100644 index 0000000..448df06 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/data/remote/model/AuthResponses.kt @@ -0,0 +1,25 @@ +package com.example.furumi_android.data.remote.model + +import com.squareup.moshi.Json + +data class LoginResponse( + @param:Json(name = "user") val user: UserResponse, + @param:Json(name = "tokens") val tokens: TokenResponse +) + +data class UserResponse( + @param:Json(name = "id") val id: Int, + @param:Json(name = "name") val name: String, + @param:Json(name = "role") val role: String +) + +data class TokenResponse( + @param:Json(name = "access_token") val accessToken: String, + @param:Json(name = "refresh_token") val refreshToken: String, + @param:Json(name = "token_type") val tokenType: String, + @param:Json(name = "expires_in_seconds") val expiresInSeconds: Int +) + +data class ErrorResponse( + @param:Json(name = "error") val error: String +) diff --git a/app/src/main/java/com/example/furumi_android/data/remote/model/PlayerResponses.kt b/app/src/main/java/com/example/furumi_android/data/remote/model/PlayerResponses.kt new file mode 100644 index 0000000..99953d1 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/data/remote/model/PlayerResponses.kt @@ -0,0 +1,224 @@ +package com.example.furumi_android.data.remote.model + +import com.example.furumi_android.domain.model.ArtistCard +import com.example.furumi_android.domain.model.ArtistDetail +import com.example.furumi_android.domain.model.ArtistPage +import com.example.furumi_android.domain.model.ListeningHistoryItem +import com.example.furumi_android.domain.model.ListeningHistoryPage +import com.example.furumi_android.domain.model.ReleaseCard +import com.example.furumi_android.domain.model.ReleaseDetail +import com.example.furumi_android.domain.model.TrackCard +import com.squareup.moshi.Json + +data class ArtistRefResponse( + @param:Json(name = "id") val id: Long, + @param:Json(name = "name") val name: String +) + +data class TrackItemResponse( + @param:Json(name = "id") val id: Long, + @param:Json(name = "title") val title: String, + @param:Json(name = "track_number") val trackNumber: Int? = null, + @param:Json(name = "disc_number") val discNumber: Int? = null, + @param:Json(name = "duration_seconds") val durationSeconds: Double, + @param:Json(name = "artists") val artists: List = emptyList(), + @param:Json(name = "featured_artists") val featuredArtists: List = emptyList(), + @param:Json(name = "release_title") val releaseTitle: String? = null, + @param:Json(name = "cover_url") val coverUrl: String? = null, + @param:Json(name = "stream_url") val streamUrl: String +) + +data class PlayHistoryItemResponse( + @param:Json(name = "id") val id: Long, + @param:Json(name = "track_id") val trackId: Long, + @param:Json(name = "track_title") val trackTitle: String, + @param:Json(name = "release_title") val releaseTitle: String? = null, + @param:Json(name = "track") val track: TrackItemResponse, + @param:Json(name = "played_at") val playedAt: String, + @param:Json(name = "duration_listened") val durationListened: Double? = null, + @param:Json(name = "completed") val completed: Boolean +) + + +data class PlayHistoryPageResponse( + @param:Json(name = "items") val items: List, + @param:Json(name = "total") val total: Long, + @param:Json(name = "page") val page: Int, + @param:Json(name = "per_page") val perPage: Int +) + +data class ArtistCardResponse( + @param:Json(name = "id") val id: Long, + @param:Json(name = "name") val name: String, + @param:Json(name = "image_url") val imageUrl: String? = null, + @param:Json(name = "release_count") val releaseCount: Int = 0, + @param:Json(name = "track_count") val trackCount: Int = 0 +) + +data class ArtistPageResponse( + @param:Json(name = "items") val items: List, + @param:Json(name = "total") val total: Long, + @param:Json(name = "page") val page: Int, + @param:Json(name = "per_page") val perPage: Int, + @param:Json(name = "has_more") val hasMore: Boolean +) + +data class ReleaseCardResponse( + @param:Json(name = "id") val id: Long, + @param:Json(name = "title") val title: String, + @param:Json(name = "release_type") val releaseType: String? = null, + @param:Json(name = "year") val year: Int? = null, + @param:Json(name = "cover_url") val coverUrl: String? = null, + @param:Json(name = "track_count") val trackCount: Int = 0 +) + +data class ArtistDetailResponse( + @param:Json(name = "artist") val artist: ArtistCardResponse? = null, + @param:Json(name = "id") val id: Long? = null, + @param:Json(name = "name") val name: String? = null, + @param:Json(name = "image_url") val imageUrl: String? = null, + @param:Json(name = "release_count") val releaseCount: Int? = null, + @param:Json(name = "track_count") val trackCount: Int? = null, + @param:Json(name = "top_tracks") val topTracks: List = emptyList(), + @param:Json(name = "releases") val releases: List = emptyList(), + @param:Json(name = "featured_tracks") val featuredTracks: List = emptyList() +) + +data class ReleaseDetailResponse( + @param:Json(name = "release") val release: ReleaseCardResponse? = null, + @param:Json(name = "id") val id: Long? = null, + @param:Json(name = "title") val title: String? = null, + @param:Json(name = "release_type") val releaseType: String? = null, + @param:Json(name = "year") val year: Int? = null, + @param:Json(name = "cover_url") val coverUrl: String? = null, + @param:Json(name = "track_count") val trackCount: Int? = null, + @param:Json(name = "artists") val artists: List = emptyList(), + @param:Json(name = "tracks") val tracks: List = emptyList() +) + +fun PlayHistoryPageResponse.toDomain(): ListeningHistoryPage { + return ListeningHistoryPage( + items = items.map { it.toDomain() }, + total = total, + page = page, + perPage = perPage + ) +} + +private fun PlayHistoryItemResponse.toDomain(): ListeningHistoryItem { + val artistNames = track.artists + .ifEmpty { track.featuredArtists } + .map { it.name } + + return ListeningHistoryItem( + id = id, + trackId = trackId, + title = trackTitle.ifBlank { track.title }, + artists = artistNames, + releaseTitle = releaseTitle ?: track.releaseTitle, + playedAt = playedAt, + durationListenedSeconds = durationListened?.toInt(), + completed = completed + ) + +} + +fun ArtistPageResponse.toDomain(baseUrl: String): ArtistPage { + return ArtistPage( + items = items.map { it.toDomain(baseUrl) }, + total = total, + page = page, + perPage = perPage, + hasMore = hasMore + ) +} + +fun ArtistDetailResponse.toDomain(baseUrl: String): ArtistDetail { + val mappedReleases = releases.map { it.toDomain(baseUrl) } + val mappedTopTracks = topTracks.map { it.toDomain(baseUrl) } + val mappedFeaturedTracks = featuredTracks.map { it.toDomain(baseUrl) } + val mappedArtist = artist?.toDomain(baseUrl) ?: ArtistCard( + id = id ?: 0L, + name = name ?: "Unknown artist", + imageUrl = imageUrl.toAbsoluteMediaUrl(baseUrl), + releaseCount = releaseCount ?: mappedReleases.size, + trackCount = trackCount ?: (mappedTopTracks.size + mappedFeaturedTracks.size) + ) + + return ArtistDetail( + artist = mappedArtist, + releases = mappedReleases, + topTracks = mappedTopTracks, + featuredTracks = mappedFeaturedTracks + ) +} + +fun ReleaseDetailResponse.toDomain(baseUrl: String): ReleaseDetail { + val mappedTracks = tracks.map { it.toDomain(baseUrl) } + val mappedRelease = release?.toDomain(baseUrl) ?: ReleaseCard( + id = id ?: 0L, + title = title ?: "Unknown release", + releaseType = releaseType, + year = year, + coverUrl = coverUrl.toAbsoluteMediaUrl(baseUrl), + trackCount = trackCount ?: mappedTracks.size + ) + + return ReleaseDetail( + release = mappedRelease, + artists = artists.map { it.toDomain(baseUrl) }, + tracks = mappedTracks + ) +} + +private fun ArtistCardResponse.toDomain(baseUrl: String): ArtistCard { + return ArtistCard( + id = id, + name = name, + imageUrl = imageUrl.toAbsoluteMediaUrl(baseUrl), + releaseCount = releaseCount, + trackCount = trackCount + ) +} + +private fun ReleaseCardResponse.toDomain(baseUrl: String): ReleaseCard { + return ReleaseCard( + id = id, + title = title, + releaseType = releaseType, + year = year, + coverUrl = coverUrl.toAbsoluteMediaUrl(baseUrl), + trackCount = trackCount + ) +} + +private fun TrackItemResponse.toDomain(baseUrl: String): TrackCard { + val artistNames = artists + .ifEmpty { featuredArtists } + .map { it.name } + + return TrackCard( + id = id, + title = title, + trackNumber = trackNumber, + discNumber = discNumber, + durationSeconds = durationSeconds.toInt(), + artists = artistNames, + releaseTitle = releaseTitle, + coverUrl = coverUrl.toAbsoluteMediaUrl(baseUrl), + streamUrl = streamUrl.toAbsoluteMediaUrl(baseUrl).orEmpty() + ) +} + +private fun String?.toAbsoluteMediaUrl(baseUrl: String): String? { + val value = this?.trim()?.takeIf { it.isNotBlank() } ?: return null + if (value.startsWith("http://", ignoreCase = true) || + value.startsWith("https://", ignoreCase = true) + ) { + return value + } + + val normalizedBaseUrl = baseUrl.trimEnd('/') + val normalizedPath = if (value.startsWith('/')) value else "/$value" + return "$normalizedBaseUrl$normalizedPath" +} diff --git a/app/src/main/java/com/example/furumi_android/data/repository/AuthDeviceInfo.kt b/app/src/main/java/com/example/furumi_android/data/repository/AuthDeviceInfo.kt new file mode 100644 index 0000000..c74fa9f --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/data/repository/AuthDeviceInfo.kt @@ -0,0 +1,23 @@ +package com.example.furumi_android.data.repository + +import android.os.Build +import javax.inject.Inject + +class AuthDeviceInfo @Inject constructor() { + val deviceName: String + get() { + val manufacturer = Build.MANUFACTURER.orEmpty().trim() + val model = Build.MODEL.orEmpty().trim() + + return when { + model.isBlank() -> DEFAULT_DEVICE_NAME + manufacturer.isBlank() -> model + model.startsWith(manufacturer, ignoreCase = true) -> model + else -> "$manufacturer $model" + } + } + + companion object { + private const val DEFAULT_DEVICE_NAME = "Android Device" + } +} diff --git a/app/src/main/java/com/example/furumi_android/data/repository/AuthRepositoryImpl.kt b/app/src/main/java/com/example/furumi_android/data/repository/AuthRepositoryImpl.kt new file mode 100644 index 0000000..30e744d --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/data/repository/AuthRepositoryImpl.kt @@ -0,0 +1,123 @@ +package com.example.furumi_android.data.repository + +import com.example.furumi_android.data.local.AuthSessionStorage +import com.example.furumi_android.data.remote.AuthApiErrorParser +import com.example.furumi_android.data.remote.AuthEndpoints +import com.example.furumi_android.data.remote.api.AuthApi +import com.example.furumi_android.data.remote.model.LoginRequest +import com.example.furumi_android.data.remote.model.LoginResponse +import com.example.furumi_android.data.remote.model.LogoutRequest +import com.example.furumi_android.data.remote.model.SsoExchangeRequest +import com.example.furumi_android.data.remote.model.toDomainSession +import com.example.furumi_android.domain.model.AuthException +import com.example.furumi_android.domain.model.AuthSession +import com.example.furumi_android.domain.model.ServerConfig +import com.example.furumi_android.domain.repository.AuthRepository +import kotlinx.coroutines.CancellationException +import retrofit2.Response +import javax.inject.Inject + +class AuthRepositoryImpl @Inject constructor( + private val authApi: AuthApi, + private val sessionStorage: AuthSessionStorage, + private val authEndpoints: AuthEndpoints, + private val errorParser: AuthApiErrorParser, + private val deviceInfo: AuthDeviceInfo +) : AuthRepository { + + override suspend fun login(config: ServerConfig, password: String): Result { + return authResult { + val normalizedConfig = ServerConfig.credentials(config.baseUrl, config.login).getOrThrow() + val response = authApi.login( + url = authEndpoints.passwordLogin(normalizedConfig.baseUrl), + request = LoginRequest( + username = normalizedConfig.login, + password = password, + deviceName = deviceInfo.deviceName + ) + ) + saveAuthSession(response, normalizedConfig.baseUrl) + } + } + + override suspend fun createSsoAuthorizationUrl(serverUrl: String): Result { + return authResult { + val config = ServerConfig.sso(serverUrl).getOrThrow() + sessionStorage.savePendingSsoBaseUrl(config.baseUrl) + authEndpoints.ssoStart(config.baseUrl) + } + } + + override suspend fun completeSsoLogin(callbackUrl: String): Result { + return authResult { + val callback = authEndpoints.parseSsoCallback(callbackUrl).getOrThrow() + val baseUrl = sessionStorage.getPendingSsoBaseUrl() + ?: sessionStorage.getBaseUrl() + ?: throw AuthException("SSO server is missing. Start SSO login again.") + val response = authApi.ssoExchange( + url = authEndpoints.ssoExchange(baseUrl), + request = SsoExchangeRequest( + code = callback.code, + deviceName = deviceInfo.deviceName + ) + ) + saveAuthSession(response, baseUrl) + } + } + + private fun saveAuthSession(response: Response, baseUrl: String): AuthSession { + if (!response.isSuccessful) { + throw AuthException(errorParser.messageFrom(response)) + } + + val body = response.body() ?: throw AuthException("Auth response is empty") + val session = body.toDomainSession( + serverBaseUrl = baseUrl, + receivedAtEpochSeconds = nowEpochSeconds() + ) + sessionStorage.saveSession(session) + return session + } + + override suspend fun logout(): Result { + return try { + val session = sessionStorage.getSession() + if (session != null) { + authApi.logout( + url = authEndpoints.logout(session.serverBaseUrl), + bearerToken = session.tokens.authorizationHeader, + request = LogoutRequest(session.tokens.refreshToken) + ) + } + Result.success(Unit) + } catch (e: CancellationException) { + throw e + } catch (e: Exception) { + Result.failure(AuthException("Logged out locally, but server logout failed", e)) + } finally { + sessionStorage.clearSession() + } + } + + override fun getCurrentSession(): AuthSession? { + return sessionStorage.getSession() + } + + override fun getSavedServerUrl(): String? { + return sessionStorage.getBaseUrl() ?: sessionStorage.getPendingSsoBaseUrl() + } + + private suspend fun authResult(block: suspend () -> T): Result { + return try { + Result.success(block()) + } catch (e: CancellationException) { + throw e + } catch (e: Exception) { + Result.failure(e) + } + } + + private fun nowEpochSeconds(): Long { + return System.currentTimeMillis() / 1000L + } +} diff --git a/app/src/main/java/com/example/furumi_android/data/repository/MediaImageLoader.kt b/app/src/main/java/com/example/furumi_android/data/repository/MediaImageLoader.kt new file mode 100644 index 0000000..3828752 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/data/repository/MediaImageLoader.kt @@ -0,0 +1,38 @@ +package com.example.furumi_android.data.repository + +import android.graphics.Bitmap +import android.graphics.BitmapFactory +import java.io.IOException +import javax.inject.Inject +import javax.inject.Singleton +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.withContext +import okhttp3.OkHttpClient +import okhttp3.Request + +@Singleton +class MediaImageLoader @Inject constructor( + private val okHttpClient: OkHttpClient +) { + suspend fun load(url: String): Result { + return withContext(Dispatchers.IO) { + runCatching { + val request = Request.Builder() + .url(url) + .build() + + okHttpClient.newCall(request).execute().use { response -> + if (!response.isSuccessful) { + throw IOException("Image request failed with HTTP ${response.code}") + } + + val bytes = response.body?.bytes() + ?: throw IOException("Image response is empty") + + BitmapFactory.decodeByteArray(bytes, 0, bytes.size) + ?: throw IOException("Image response is not a supported bitmap") + } + } + } + } +} diff --git a/app/src/main/java/com/example/furumi_android/data/repository/PlayerRepositoryImpl.kt b/app/src/main/java/com/example/furumi_android/data/repository/PlayerRepositoryImpl.kt new file mode 100644 index 0000000..7da52e3 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/data/repository/PlayerRepositoryImpl.kt @@ -0,0 +1,112 @@ +package com.example.furumi_android.data.repository + +import com.example.furumi_android.data.local.AuthSessionStorage +import com.example.furumi_android.data.remote.AuthApiErrorParser +import com.example.furumi_android.data.remote.PlayerEndpoints +import com.example.furumi_android.data.remote.api.PlayerApi +import com.example.furumi_android.data.remote.model.toDomain +import com.example.furumi_android.domain.model.ArtistDetail +import com.example.furumi_android.domain.model.ArtistPage +import com.example.furumi_android.domain.model.AuthException +import com.example.furumi_android.domain.model.ListeningHistoryPage +import com.example.furumi_android.domain.model.ReleaseDetail +import com.example.furumi_android.domain.repository.PlayerRepository +import kotlinx.coroutines.CancellationException +import javax.inject.Inject + +class PlayerRepositoryImpl @Inject constructor( + private val playerApi: PlayerApi, + private val sessionStorage: AuthSessionStorage, + private val playerEndpoints: PlayerEndpoints, + private val errorParser: AuthApiErrorParser +) : PlayerRepository { + + override suspend fun getArtists(page: Int, limit: Int, mine: Boolean): Result { + return try { + val baseUrl = sessionStorage.getBaseUrl() + ?: throw AuthException("Server URL is missing") + val response = playerApi.artists( + url = playerEndpoints.artists(baseUrl), + page = page, + limit = limit, + mine = mine + ) + + if (!response.isSuccessful) { + throw AuthException(errorParser.messageFrom(response)) + } + + val body = response.body() ?: throw AuthException("Artists response is empty") + Result.success(body.toDomain(baseUrl)) + } catch (e: CancellationException) { + throw e + } catch (e: Exception) { + Result.failure(e) + } + } + + override suspend fun getArtistDetail(artistId: Long): Result { + return try { + val baseUrl = sessionStorage.getBaseUrl() + ?: throw AuthException("Server URL is missing") + val response = playerApi.artistDetail( + url = playerEndpoints.artistDetail(baseUrl, artistId) + ) + + if (!response.isSuccessful) { + throw AuthException(errorParser.messageFrom(response)) + } + + val body = response.body() ?: throw AuthException("Artist response is empty") + Result.success(body.toDomain(baseUrl)) + } catch (e: CancellationException) { + throw e + } catch (e: Exception) { + Result.failure(e) + } + } + + override suspend fun getReleaseDetail(releaseId: Long): Result { + return try { + val baseUrl = sessionStorage.getBaseUrl() + ?: throw AuthException("Server URL is missing") + val response = playerApi.releaseDetail( + url = playerEndpoints.releaseDetail(baseUrl, releaseId) + ) + + if (!response.isSuccessful) { + throw AuthException(errorParser.messageFrom(response)) + } + + val body = response.body() ?: throw AuthException("Release response is empty") + Result.success(body.toDomain(baseUrl)) + } catch (e: CancellationException) { + throw e + } catch (e: Exception) { + Result.failure(e) + } + } + + override suspend fun getListeningHistory(page: Int, limit: Int): Result { + return try { + val baseUrl = sessionStorage.getBaseUrl() + ?: throw AuthException("Server URL is missing") + val response = playerApi.history( + url = playerEndpoints.history(baseUrl), + page = page, + limit = limit + ) + + if (!response.isSuccessful) { + throw AuthException(errorParser.messageFrom(response)) + } + + val body = response.body() ?: throw AuthException("Listening history response is empty") + Result.success(body.toDomain()) + } catch (e: CancellationException) { + throw e + } catch (e: Exception) { + Result.failure(e) + } + } +} diff --git a/app/src/main/java/com/example/furumi_android/di/NetworkModule.kt b/app/src/main/java/com/example/furumi_android/di/NetworkModule.kt new file mode 100644 index 0000000..e6a42db --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/di/NetworkModule.kt @@ -0,0 +1,90 @@ +package com.example.furumi_android.di + +import com.example.furumi_android.data.local.AuthSessionStorage +import com.example.furumi_android.data.remote.AccessTokenInterceptor +import com.example.furumi_android.data.remote.AuthAuthenticator +import com.example.furumi_android.data.remote.AuthEndpoints +import com.example.furumi_android.data.remote.api.AuthApi +import com.example.furumi_android.data.remote.api.PlayerApi +import com.squareup.moshi.Moshi +import dagger.Lazy +import dagger.Module +import dagger.Provides +import dagger.hilt.InstallIn +import dagger.hilt.components.SingletonComponent +import okhttp3.OkHttpClient +import okhttp3.logging.HttpLoggingInterceptor +import retrofit2.Retrofit +import retrofit2.converter.moshi.MoshiConverterFactory +import javax.inject.Singleton + +@Module +@InstallIn(SingletonComponent::class) +object NetworkModule { + + @Provides + @Singleton + fun provideAccessTokenInterceptor( + sessionStorage: AuthSessionStorage, + authEndpoints: AuthEndpoints + ): AccessTokenInterceptor { + return AccessTokenInterceptor(sessionStorage, authEndpoints) + } + + @Provides + @Singleton + fun provideAuthAuthenticator( + sessionStorage: AuthSessionStorage, + authEndpoints: AuthEndpoints, + authApiLazy: Lazy + ): AuthAuthenticator { + return AuthAuthenticator(sessionStorage, authEndpoints, authApiLazy) + } + + @Provides + @Singleton + fun provideOkHttpClient( + accessTokenInterceptor: AccessTokenInterceptor, + authAuthenticator: AuthAuthenticator + ): OkHttpClient { + val logging = HttpLoggingInterceptor().apply { + level = HttpLoggingInterceptor.Level.BASIC + } + + return OkHttpClient.Builder() + .addInterceptor(accessTokenInterceptor) + .addInterceptor(logging) + .authenticator(authAuthenticator) + .build() + } + + @Provides + @Singleton + fun provideAuthApi( + okHttpClient: OkHttpClient, + moshi: Moshi + ): AuthApi { + // Auth requests use @Url, but Retrofit still requires an initial base URL. + // The actual server URL is supplied per request. + return Retrofit.Builder() + .baseUrl("https://placeholder.com/") + .client(okHttpClient) + .addConverterFactory(MoshiConverterFactory.create(moshi)) + .build() + .create(AuthApi::class.java) + } + + @Provides + @Singleton + fun providePlayerApi( + okHttpClient: OkHttpClient, + moshi: Moshi + ): PlayerApi { + return Retrofit.Builder() + .baseUrl("https://placeholder.com/") + .client(okHttpClient) + .addConverterFactory(MoshiConverterFactory.create(moshi)) + .build() + .create(PlayerApi::class.java) + } +} diff --git a/app/src/main/java/com/example/furumi_android/di/RepositoryModule.kt b/app/src/main/java/com/example/furumi_android/di/RepositoryModule.kt new file mode 100644 index 0000000..3a1a486 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/di/RepositoryModule.kt @@ -0,0 +1,28 @@ +package com.example.furumi_android.di + +import com.example.furumi_android.data.repository.AuthRepositoryImpl +import com.example.furumi_android.data.repository.PlayerRepositoryImpl +import com.example.furumi_android.domain.repository.AuthRepository +import com.example.furumi_android.domain.repository.PlayerRepository +import dagger.Binds +import dagger.Module +import dagger.hilt.InstallIn +import dagger.hilt.components.SingletonComponent +import javax.inject.Singleton + +@Module +@InstallIn(SingletonComponent::class) +abstract class RepositoryModule { + + @Binds + @Singleton + abstract fun bindAuthRepository( + impl: AuthRepositoryImpl + ): AuthRepository + + @Binds + @Singleton + abstract fun bindPlayerRepository( + impl: PlayerRepositoryImpl + ): PlayerRepository +} diff --git a/app/src/main/java/com/example/furumi_android/di/StorageModule.kt b/app/src/main/java/com/example/furumi_android/di/StorageModule.kt new file mode 100644 index 0000000..90025b0 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/di/StorageModule.kt @@ -0,0 +1,34 @@ +package com.example.furumi_android.di + +import android.content.Context +import com.example.furumi_android.data.local.AuthSessionStorage +import com.squareup.moshi.Moshi +import com.squareup.moshi.kotlin.reflect.KotlinJsonAdapterFactory +import dagger.Module +import dagger.Provides +import dagger.hilt.InstallIn +import dagger.hilt.android.qualifiers.ApplicationContext +import dagger.hilt.components.SingletonComponent +import javax.inject.Singleton + +@Module +@InstallIn(SingletonComponent::class) +object StorageModule { + + @Provides + @Singleton + fun provideMoshi(): Moshi { + return Moshi.Builder() + .add(KotlinJsonAdapterFactory()) + .build() + } + + @Provides + @Singleton + fun provideAuthSessionStorage( + @ApplicationContext context: Context, + moshi: Moshi + ): AuthSessionStorage { + return AuthSessionStorage(context, moshi) + } +} diff --git a/app/src/main/java/com/example/furumi_android/domain/model/Artist.kt b/app/src/main/java/com/example/furumi_android/domain/model/Artist.kt new file mode 100644 index 0000000..dfdd3c8 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/domain/model/Artist.kt @@ -0,0 +1,51 @@ +package com.example.furumi_android.domain.model + +data class ArtistCard( + val id: Long, + val name: String, + val imageUrl: String?, + val releaseCount: Int, + val trackCount: Int +) + +data class ArtistPage( + val items: List, + val total: Long, + val page: Int, + val perPage: Int, + val hasMore: Boolean +) + +data class ReleaseCard( + val id: Long, + val title: String, + val releaseType: String?, + val year: Int?, + val coverUrl: String?, + val trackCount: Int +) + +data class TrackCard( + val id: Long, + val title: String, + val trackNumber: Int?, + val discNumber: Int?, + val durationSeconds: Int?, + val artists: List, + val releaseTitle: String?, + val coverUrl: String?, + val streamUrl: String +) + +data class ArtistDetail( + val artist: ArtistCard, + val releases: List, + val topTracks: List, + val featuredTracks: List +) + +data class ReleaseDetail( + val release: ReleaseCard, + val artists: List, + val tracks: List +) diff --git a/app/src/main/java/com/example/furumi_android/domain/model/AuthException.kt b/app/src/main/java/com/example/furumi_android/domain/model/AuthException.kt new file mode 100644 index 0000000..82337a7 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/domain/model/AuthException.kt @@ -0,0 +1,6 @@ +package com.example.furumi_android.domain.model + +class AuthException( + message: String, + cause: Throwable? = null +) : Exception(message, cause) diff --git a/app/src/main/java/com/example/furumi_android/domain/model/AuthSession.kt b/app/src/main/java/com/example/furumi_android/domain/model/AuthSession.kt new file mode 100644 index 0000000..f1b475d --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/domain/model/AuthSession.kt @@ -0,0 +1,7 @@ +package com.example.furumi_android.domain.model + +data class AuthSession( + val serverBaseUrl: String, + val user: User, + val tokens: AuthTokens +) diff --git a/app/src/main/java/com/example/furumi_android/domain/model/AuthTokens.kt b/app/src/main/java/com/example/furumi_android/domain/model/AuthTokens.kt new file mode 100644 index 0000000..e531de1 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/domain/model/AuthTokens.kt @@ -0,0 +1,21 @@ +package com.example.furumi_android.domain.model + +data class AuthTokens( + val accessToken: String, + val refreshToken: String, + val tokenType: String = DEFAULT_TOKEN_TYPE, + val expiresInSeconds: Int, + val expiresAtEpochSeconds: Long = 0L +) { + val authorizationHeader: String + get() = "${tokenType.ifBlank { DEFAULT_TOKEN_TYPE }} $accessToken" + + fun isExpired(nowEpochSeconds: Long, skewSeconds: Long = DEFAULT_EXPIRY_SKEW_SECONDS): Boolean { + return expiresAtEpochSeconds > 0 && nowEpochSeconds >= expiresAtEpochSeconds - skewSeconds + } + + companion object { + const val DEFAULT_TOKEN_TYPE = "Bearer" + private const val DEFAULT_EXPIRY_SKEW_SECONDS = 60L + } +} diff --git a/app/src/main/java/com/example/furumi_android/domain/model/ListeningHistory.kt b/app/src/main/java/com/example/furumi_android/domain/model/ListeningHistory.kt new file mode 100644 index 0000000..df1552e --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/domain/model/ListeningHistory.kt @@ -0,0 +1,19 @@ +package com.example.furumi_android.domain.model + +data class ListeningHistoryPage( + val items: List, + val total: Long, + val page: Int, + val perPage: Int +) + +data class ListeningHistoryItem( + val id: Long, + val trackId: Long, + val title: String, + val artists: List, + val releaseTitle: String?, + val playedAt: String, + val durationListenedSeconds: Int?, + val completed: Boolean +) diff --git a/app/src/main/java/com/example/furumi_android/domain/model/ServerConfig.kt b/app/src/main/java/com/example/furumi_android/domain/model/ServerConfig.kt new file mode 100644 index 0000000..e18fd47 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/domain/model/ServerConfig.kt @@ -0,0 +1,74 @@ +package com.example.furumi_android.domain.model + +import java.net.URI +import java.util.Locale + +data class ServerConfig( + val baseUrl: String, + val login: String = "" +) { + val url: String + get() = baseUrl + + companion object { + fun credentials(serverUrl: String, login: String): Result { + val normalizedBaseUrl = normalizeBaseUrl(serverUrl).getOrElse { return Result.failure(it) } + val trimmedLogin = login.trim() + + if (trimmedLogin.isBlank()) { + return Result.failure(IllegalArgumentException("Username is required")) + } + + return Result.success(ServerConfig(normalizedBaseUrl, trimmedLogin)) + } + + fun sso(serverUrl: String): Result { + val normalizedBaseUrl = normalizeBaseUrl(serverUrl).getOrElse { return Result.failure(it) } + return Result.success(ServerConfig(normalizedBaseUrl)) + } + + fun normalizeBaseUrl(rawUrl: String): Result { + val trimmedUrl = rawUrl.trim().trimEnd('/') + + if (trimmedUrl.isBlank()) { + return Result.failure(IllegalArgumentException("Server URL is required")) + } + + val candidate = if (trimmedUrl.hasScheme()) trimmedUrl else "https://$trimmedUrl" + val uri = runCatching { URI(candidate) }.getOrElse { + return Result.failure(IllegalArgumentException("Enter a valid server URL")) + } + + val scheme = uri.scheme?.lowercase(Locale.US) + val rawAuthority = uri.rawAuthority + val host = uri.host + if (scheme != "http" && scheme != "https") { + return Result.failure(IllegalArgumentException("Server URL must start with http:// or https://")) + } + + if (rawAuthority.isNullOrBlank() || host.isNullOrBlank()) { + return Result.failure(IllegalArgumentException("Enter a valid server URL")) + } + + if (uri.userInfo != null) { + return Result.failure(IllegalArgumentException("Server URL must not contain user info")) + } + + if (uri.rawQuery != null || uri.rawFragment != null) { + return Result.failure(IllegalArgumentException("Server URL must not contain query or fragment")) + } + + val normalizedPath = uri.rawPath + ?.takeUnless { it == "/" } + ?.trimEnd('/') + .orEmpty() + + val authority = rawAuthority.lowercase(Locale.US) + return Result.success("$scheme://$authority$normalizedPath") + } + + private fun String.hasScheme(): Boolean { + return startsWith("http://", ignoreCase = true) || startsWith("https://", ignoreCase = true) + } + } +} diff --git a/app/src/main/java/com/example/furumi_android/domain/model/User.kt b/app/src/main/java/com/example/furumi_android/domain/model/User.kt new file mode 100644 index 0000000..2ad5c44 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/domain/model/User.kt @@ -0,0 +1,7 @@ +package com.example.furumi_android.domain.model + +data class User( + val id: Int, + val name: String, + val role: String +) diff --git a/app/src/main/java/com/example/furumi_android/domain/repository/AuthRepository.kt b/app/src/main/java/com/example/furumi_android/domain/repository/AuthRepository.kt new file mode 100644 index 0000000..ca21ca8 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/domain/repository/AuthRepository.kt @@ -0,0 +1,18 @@ +package com.example.furumi_android.domain.repository + +import com.example.furumi_android.domain.model.AuthSession +import com.example.furumi_android.domain.model.ServerConfig +import com.example.furumi_android.domain.model.User + +interface AuthRepository { + suspend fun login(config: ServerConfig, password: String): Result + suspend fun createSsoAuthorizationUrl(serverUrl: String): Result + suspend fun completeSsoLogin(callbackUrl: String): Result + suspend fun logout(): Result + fun getCurrentSession(): AuthSession? + fun getSavedServerUrl(): String? + + fun getCurrentUser(): User? { + return getCurrentSession()?.user + } +} diff --git a/app/src/main/java/com/example/furumi_android/domain/repository/PlayerRepository.kt b/app/src/main/java/com/example/furumi_android/domain/repository/PlayerRepository.kt new file mode 100644 index 0000000..aff3d44 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/domain/repository/PlayerRepository.kt @@ -0,0 +1,20 @@ +package com.example.furumi_android.domain.repository + +import com.example.furumi_android.domain.model.ArtistPage +import com.example.furumi_android.domain.model.ArtistDetail +import com.example.furumi_android.domain.model.ListeningHistoryPage +import com.example.furumi_android.domain.model.ReleaseDetail + +interface PlayerRepository { + suspend fun getArtists( + page: Int = 1, + limit: Int = 60, + mine: Boolean = false + ): Result + + suspend fun getArtistDetail(artistId: Long): Result + + suspend fun getReleaseDetail(releaseId: Long): Result + + suspend fun getListeningHistory(page: Int = 1, limit: Int = 20): Result +} diff --git a/app/src/main/java/com/example/furumi_android/playback/FurumiPlaybackService.kt b/app/src/main/java/com/example/furumi_android/playback/FurumiPlaybackService.kt new file mode 100644 index 0000000..29325ae --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/playback/FurumiPlaybackService.kt @@ -0,0 +1,126 @@ +package com.example.furumi_android.playback + +import android.app.NotificationChannel +import android.app.NotificationManager +import android.app.PendingIntent +import android.content.Context +import android.content.Intent +import android.content.pm.ServiceInfo +import android.os.Build +import android.util.Log +import androidx.annotation.OptIn +import androidx.core.app.NotificationCompat +import androidx.media3.common.util.UnstableApi +import androidx.media3.session.DefaultMediaNotificationProvider +import androidx.media3.session.MediaSession +import androidx.media3.session.MediaSessionService +import androidx.media3.session.MediaStyleNotificationHelper +import com.example.furumi_android.MainActivity +import dagger.hilt.android.AndroidEntryPoint +import javax.inject.Inject + +@AndroidEntryPoint +class FurumiPlaybackService : MediaSessionService() { + + @Inject lateinit var playbackController: PlaybackController + + private var mediaSession: MediaSession? = null + + companion object { + const val NOTIFICATION_ID = 1001 + const val CHANNEL_ID = "furumi_playback_channel" + } + + @OptIn(UnstableApi::class) + override fun onCreate() { + super.onCreate() + Log.d("FurumiPlaybackService", "onCreate") + + createNotificationChannel() + + val sessionActivity = PendingIntent.getActivity( + this, + 0, + Intent(this, MainActivity::class.java), + PendingIntent.FLAG_IMMUTABLE or PendingIntent.FLAG_UPDATE_CURRENT + ) + + mediaSession = MediaSession.Builder(this, playbackController.player) + .setId("FurumiPlaybackSession") + .setSessionActivity(sessionActivity) + .build() + + // Настройка провайдера с иконкой + val notificationProvider = DefaultMediaNotificationProvider.Builder(this) + .setChannelId(CHANNEL_ID) + .setNotificationId(NOTIFICATION_ID) + .build() + + // Это важно для появления иконки в статус-баре при автоматических обновлениях Media3 + notificationProvider.setSmallIcon(android.R.drawable.ic_media_play) + setMediaNotificationProvider(notificationProvider) + } + + @OptIn(UnstableApi::class) + override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int { + val session = mediaSession + if (session != null) { + val metadata = playbackController.player.currentMediaItem?.mediaMetadata + + val notification = NotificationCompat.Builder(this, CHANNEL_ID) + .setSmallIcon(android.R.drawable.ic_media_play) // Иконка для трея + .setContentTitle(metadata?.title ?: "Furumi") + .setContentText(metadata?.artist ?: "Playing...") + .setOngoing(true) + .setCategory(NotificationCompat.CATEGORY_TRANSPORT) // Сообщаем системе, что это транспорт управления + .setPriority(NotificationCompat.PRIORITY_MAX) // Максимальный приоритет для трея + .setVisibility(NotificationCompat.VISIBILITY_PUBLIC) + .setStyle(MediaStyleNotificationHelper.MediaStyle(session)) + .build() + + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) { + startForeground(NOTIFICATION_ID, notification, ServiceInfo.FOREGROUND_SERVICE_TYPE_MEDIA_PLAYBACK) + } else { + startForeground(NOTIFICATION_ID, notification) + } + } + + super.onStartCommand(intent, flags, startId) + return START_STICKY + } + + override fun onGetSession(controllerInfo: MediaSession.ControllerInfo): MediaSession? { + return mediaSession + } + + private fun createNotificationChannel() { + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { + val name = "Furumi Playback" + // Повышаем важность до HIGH, чтобы иконка не пропадала + val importance = NotificationManager.IMPORTANCE_HIGH + val channel = NotificationChannel(CHANNEL_ID, name, importance).apply { + setShowBadge(true) + lockscreenVisibility = android.app.Notification.VISIBILITY_PUBLIC + } + val notificationManager = getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager + notificationManager.createNotificationChannel(channel) + } + } + + @OptIn(UnstableApi::class) + override fun onTaskRemoved(rootIntent: Intent?) { + val player = playbackController.player + if (!player.playWhenReady || player.mediaItemCount == 0) { + stopSelf() + } + } + + override fun onDestroy() { + Log.d("FurumiPlaybackService", "onDestroy") + mediaSession?.run { + release() + mediaSession = null + } + super.onDestroy() + } +} diff --git a/app/src/main/java/com/example/furumi_android/playback/PlaybackController.kt b/app/src/main/java/com/example/furumi_android/playback/PlaybackController.kt new file mode 100644 index 0000000..38441f8 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/playback/PlaybackController.kt @@ -0,0 +1,263 @@ +package com.example.furumi_android.playback + +import android.content.Context +import android.content.Intent +import android.graphics.Bitmap +import android.net.Uri +import androidx.annotation.OptIn +import androidx.media3.common.AudioAttributes +import androidx.media3.common.C +import androidx.media3.common.MediaItem +import androidx.media3.common.MediaMetadata +import androidx.media3.common.PlaybackException +import androidx.media3.common.Player +import androidx.media3.common.util.UnstableApi +import androidx.core.content.ContextCompat +import androidx.media3.datasource.okhttp.OkHttpDataSource +import androidx.media3.exoplayer.ExoPlayer +import androidx.media3.exoplayer.source.DefaultMediaSourceFactory +import com.example.furumi_android.data.repository.MediaImageLoader +import com.example.furumi_android.domain.model.TrackCard +import dagger.hilt.android.qualifiers.ApplicationContext +import javax.inject.Inject +import javax.inject.Singleton +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.SupervisorJob +import kotlinx.coroutines.cancel +import kotlinx.coroutines.delay +import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.StateFlow +import kotlinx.coroutines.flow.asStateFlow +import kotlinx.coroutines.isActive +import kotlinx.coroutines.launch +import okhttp3.OkHttpClient +import java.io.ByteArrayOutputStream + +data class AudioPlaybackState( + val queue: List = emptyList(), + val currentIndex: Int = -1, + val isPlaying: Boolean = false, + val isBuffering: Boolean = false, + val positionMs: Long = 0L, + val durationMs: Long = 0L, + val errorMessage: String? = null +) { + val currentTrack: TrackCard? + get() = queue.getOrNull(currentIndex) + + val progress: Float + get() = if (durationMs > 0L) { + (positionMs.toFloat() / durationMs.toFloat()).coerceIn(0f, 1f) + } else { + 0f + } + + val canGoPrevious: Boolean + get() = currentIndex > 0 || positionMs > RESTART_WINDOW_MS + + val canGoNext: Boolean + get() = currentIndex in 0 until queue.lastIndex + + private companion object { + const val RESTART_WINDOW_MS = 3_000L + } +} + +@OptIn(UnstableApi::class) +@Singleton +class PlaybackController @Inject constructor( + @param:ApplicationContext private val context: Context, + private val mediaImageLoader: MediaImageLoader, + okHttpClient: OkHttpClient +) { + private val scope = CoroutineScope(SupervisorJob() + Dispatchers.Main.immediate) + private val _state = MutableStateFlow(AudioPlaybackState()) + + val player: ExoPlayer + val state: StateFlow = _state.asStateFlow() + + init { + val dataSourceFactory = OkHttpDataSource.Factory(okHttpClient) + val mediaSourceFactory = DefaultMediaSourceFactory(dataSourceFactory) + + player = ExoPlayer.Builder(context) + .setMediaSourceFactory(mediaSourceFactory) + .build() + .apply { + setAudioAttributes( + AudioAttributes.Builder() + .setContentType(C.AUDIO_CONTENT_TYPE_MUSIC) + .setUsage(C.USAGE_MEDIA) + .build(), + true + ) + setHandleAudioBecomingNoisy(true) + setWakeMode(C.WAKE_MODE_LOCAL) + } + + player.addListener( + object : Player.Listener { + override fun onIsPlayingChanged(isPlaying: Boolean) { + publishState() + } + + override fun onPlaybackStateChanged(playbackState: Int) { + publishState() + } + + override fun onMediaItemTransition(mediaItem: MediaItem?, reason: Int) { + publishState() + // Просто уведомляем систему, что состояние изменилось. + // MediaSessionService сам обновит уведомление, так как он слушает плеер. + } + + override fun onPlayerError(error: PlaybackException) { + _state.value = _state.value.copy( + isPlaying = false, + isBuffering = false, + errorMessage = error.message ?: "Playback failed" + ) + } + } + ) + + scope.launch { + while (isActive) { + publishState() + delay(PROGRESS_TICK_MS) + } + } + } + + fun playQueue(queue: List, startIndex: Int = 0) { + val playableQueue = queue.filter { it.streamUrl.isNotBlank() } + if (playableQueue.isEmpty()) { + _state.value = AudioPlaybackState(errorMessage = "No playable tracks") + return + } + + val safeStartIndex = startIndex.coerceIn(0, playableQueue.lastIndex) + val mediaItems = playableQueue.map { track -> + MediaItem.Builder() + .setMediaId(track.id.toString()) + .setUri(track.streamUrl) + .setMediaMetadata(track.toMediaMetadata()) + .build() + } + + _state.value = AudioPlaybackState( + queue = playableQueue, + currentIndex = safeStartIndex, + isBuffering = true + ) + + player.setMediaItems(mediaItems, safeStartIndex, 0L) + player.prepare() + player.playWhenReady = true + + startPlaybackService() + } + + fun playTrack(track: TrackCard, queue: List) { + val playableQueue = queue + .ifEmpty { listOf(track) } + .filter { it.streamUrl.isNotBlank() } + .ifEmpty { listOf(track).filter { it.streamUrl.isNotBlank() } } + + val startIndex = playableQueue.indexOfFirst { it.id == track.id } + .takeIf { it >= 0 } + ?: 0 + + playQueue(playableQueue, startIndex) + } + + fun togglePlayPause() { + if (_state.value.currentTrack == null) return + + if (player.isPlaying) { + player.pause() + } else { + player.play() + } + publishState() + } + + fun next() { + if (player.hasNextMediaItem()) { + player.seekToNextMediaItem() + player.play() + } + publishState() + } + + fun previous() { + if (player.currentPosition > RESTART_WINDOW_MS || !player.hasPreviousMediaItem()) { + player.seekTo(0L) + } else { + player.seekToPreviousMediaItem() + } + player.play() + publishState() + } + + fun seekToProgress(progress: Float) { + val duration = player.duration.takeIf { it > 0L } ?: return + player.seekTo((duration * progress.coerceIn(0f, 1f)).toLong()) + publishState() + } + + fun release() { + scope.cancel() + player.release() + } + + fun stop() { + player.stop() + _state.value = AudioPlaybackState() + } + + private fun publishState() { + val existingErrorMessage = _state.value.errorMessage + val currentQueue = _state.value.queue + val currentIndex = player.currentMediaItemIndex.takeIf { it >= 0 } ?: _state.value.currentIndex + val duration = player.duration.takeIf { it > 0L } ?: currentQueue + .getOrNull(currentIndex) + ?.durationSeconds + ?.times(1_000L) + ?: 0L + + _state.value = _state.value.copy( + currentIndex = currentIndex, + isPlaying = player.isPlaying, + isBuffering = player.playbackState == Player.STATE_BUFFERING, + positionMs = player.currentPosition.coerceAtLeast(0L), + durationMs = duration, + errorMessage = existingErrorMessage + ) + } + + private companion object { + const val PROGRESS_TICK_MS = 500L + const val RESTART_WINDOW_MS = 3_000L + } + + private fun startPlaybackService() { + val intent = Intent(context, FurumiPlaybackService::class.java) + ContextCompat.startForegroundService(context, intent) + } +} + +private fun TrackCard.toMediaMetadata(): MediaMetadata { + val artist = artists.joinToString(", ") + .ifBlank { releaseTitle.orEmpty() } + .ifBlank { "Unknown artist" } + + return MediaMetadata.Builder() + .setTitle(title) + .setArtist(artist) + .setAlbumTitle(releaseTitle) + .setArtworkUri(coverUrl?.let(Uri::parse)) + .setMediaType(MediaMetadata.MEDIA_TYPE_MUSIC) + .build() +} diff --git a/app/src/main/java/com/example/furumi_android/ui/login/LoginScreen.kt b/app/src/main/java/com/example/furumi_android/ui/login/LoginScreen.kt new file mode 100644 index 0000000..fcca761 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/ui/login/LoginScreen.kt @@ -0,0 +1,378 @@ +package com.example.furumi_android.ui.login + +import android.net.Uri +import androidx.browser.customtabs.CustomTabsIntent +import androidx.compose.foundation.BorderStroke +import androidx.compose.foundation.background +import androidx.compose.foundation.border +import androidx.compose.foundation.rememberScrollState +import androidx.compose.foundation.verticalScroll +import androidx.compose.foundation.layout.* +import androidx.compose.foundation.shape.CircleShape +import androidx.compose.foundation.shape.RoundedCornerShape +import androidx.compose.foundation.text.KeyboardActions +import androidx.compose.foundation.text.KeyboardOptions +import androidx.compose.material3.* +import androidx.compose.runtime.* +import androidx.compose.runtime.saveable.rememberSaveable +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.draw.clip +import androidx.compose.ui.focus.FocusRequester +import androidx.compose.ui.focus.focusRequester +import androidx.compose.ui.graphics.Brush +import androidx.compose.ui.graphics.Color +import androidx.compose.ui.platform.LocalContext +import androidx.compose.ui.platform.LocalFocusManager +import androidx.compose.ui.text.font.FontWeight +import androidx.compose.ui.text.input.ImeAction +import androidx.compose.ui.text.input.KeyboardType +import androidx.compose.ui.text.input.PasswordVisualTransformation +import androidx.compose.ui.unit.dp +import androidx.hilt.navigation.compose.hiltViewModel +import com.example.furumi_android.ui.theme.FurumiElectricCyan +import com.example.furumi_android.ui.theme.FurumiHotOrange +import com.example.furumi_android.ui.theme.FurumiInk +import com.example.furumi_android.ui.theme.FurumiLine +import com.example.furumi_android.ui.theme.FurumiNeonPink +import com.example.furumi_android.ui.theme.FurumiNeonViolet +import com.example.furumi_android.ui.theme.FurumiSurface +import com.example.furumi_android.ui.theme.FurumiSurfaceHigh +import com.example.furumi_android.ui.theme.FurumiTextMuted + +@Composable +fun LoginScreen( + viewModel: LoginViewModel = hiltViewModel(), + onLoginSuccess: () -> Unit +) { + val uiState by viewModel.uiState.collectAsState() + val context = LocalContext.current + val focusManager = LocalFocusManager.current + val loginFocusRequester = remember { FocusRequester() } + val passwordFocusRequester = remember { FocusRequester() } + val isSsoLoading = uiState.loadingAction == LoginLoadingAction.Sso + val isPasswordLoading = uiState.loadingAction == LoginLoadingAction.Password + var isPasswordLoginVisible by rememberSaveable { mutableStateOf(false) } + + LaunchedEffect(uiState.isAuthenticated) { + if (uiState.isAuthenticated) { + onLoginSuccess() + } + } + + LaunchedEffect(uiState.externalAuthUrl) { + uiState.externalAuthUrl?.let { authUrl -> + CustomTabsIntent.Builder() + .build() + .launchUrl(context, Uri.parse(authUrl)) + viewModel.onExternalAuthUrlOpened() + } + } + + Box( + modifier = Modifier + .fillMaxSize() + .background(MaterialTheme.colorScheme.background) + .windowInsetsPadding(WindowInsets.safeDrawing) + .imePadding() + ) { + Column( + modifier = Modifier + .fillMaxSize() + .verticalScroll(rememberScrollState()) + .padding(horizontal = 24.dp, vertical = 28.dp), + verticalArrangement = Arrangement.SpaceBetween + ) { + Column { + BrandHeader() + Spacer(modifier = Modifier.height(34.dp)) + AlbumTileStrip() + } + + Spacer(modifier = Modifier.height(32.dp)) + + LoginControls( + uiState = uiState, + isPasswordLoginVisible = isPasswordLoginVisible, + isSsoLoading = isSsoLoading, + isPasswordLoading = isPasswordLoading, + loginFocusRequester = loginFocusRequester, + passwordFocusRequester = passwordFocusRequester, + onPasswordLoginVisibilityChange = { + isPasswordLoginVisible = !isPasswordLoginVisible + if (!isPasswordLoginVisible) { + focusManager.clearFocus() + } + }, + onServerUrlChange = viewModel::onServerUrlChange, + onLoginChange = viewModel::onLoginChange, + onPasswordChange = viewModel::onPasswordChange, + onSsoClick = { + focusManager.clearFocus() + viewModel.loginWithSSO() + }, + onPasswordLoginClick = { + focusManager.clearFocus() + viewModel.login() + } + ) + } + } +} + +@Composable +private fun BrandHeader() { + Row( + verticalAlignment = Alignment.CenterVertically + ) { + Box( + modifier = Modifier + .size(34.dp) + .clip(CircleShape) + .background(FurumiNeonPink), + contentAlignment = Alignment.Center + ) { + Text( + text = "F", + style = MaterialTheme.typography.titleMedium, + color = MaterialTheme.colorScheme.onPrimary, + fontWeight = FontWeight.ExtraBold + ) + } + + Spacer(modifier = Modifier.width(10.dp)) + + Text( + text = "Furumi", + style = MaterialTheme.typography.titleMedium, + color = MaterialTheme.colorScheme.onBackground + ) + } +} + +@Composable +private fun AlbumTileStrip() { + Row( + modifier = Modifier.fillMaxWidth(), + horizontalArrangement = Arrangement.spacedBy(10.dp) + ) { + AlbumTile( + modifier = Modifier.weight(1f), + brush = Brush.linearGradient(listOf(FurumiNeonPink, FurumiSurfaceHigh)) + ) + AlbumTile( + modifier = Modifier.weight(1f), + brush = Brush.linearGradient(listOf(FurumiElectricCyan, FurumiInk)) + ) + AlbumTile( + modifier = Modifier.weight(1f), + brush = Brush.linearGradient(listOf(FurumiNeonViolet, FurumiHotOrange, FurumiSurface)) + ) + } +} + +@Composable +private fun AlbumTile( + modifier: Modifier, + brush: Brush +) { + Box( + modifier = modifier + .aspectRatio(1f) + .clip(RoundedCornerShape(8.dp)) + .background(brush) + .border(1.dp, Color.White.copy(alpha = 0.08f), RoundedCornerShape(8.dp)) + ) +} + +@Composable +private fun LoginControls( + uiState: LoginUiState, + isPasswordLoginVisible: Boolean, + isSsoLoading: Boolean, + isPasswordLoading: Boolean, + loginFocusRequester: FocusRequester, + passwordFocusRequester: FocusRequester, + onPasswordLoginVisibilityChange: () -> Unit, + onServerUrlChange: (String) -> Unit, + onLoginChange: (String) -> Unit, + onPasswordChange: (String) -> Unit, + onSsoClick: () -> Unit, + onPasswordLoginClick: () -> Unit +) { + Column { + LoginTextField( + value = uiState.serverUrl, + onValueChange = onServerUrlChange, + label = "Server URL", + placeholder = "https://media.example.com", + enabled = !uiState.isLoading, + keyboardType = KeyboardType.Uri, + imeAction = if (isPasswordLoginVisible) ImeAction.Next else ImeAction.Go, + keyboardActions = KeyboardActions( + onGo = { onSsoClick() }, + onNext = { loginFocusRequester.requestFocus() } + ) + ) + + if (uiState.error != null) { + Spacer(modifier = Modifier.height(12.dp)) + Text( + text = uiState.error, + color = MaterialTheme.colorScheme.error, + style = MaterialTheme.typography.bodyMedium + ) + } + + Spacer(modifier = Modifier.height(16.dp)) + + Button( + onClick = onSsoClick, + modifier = Modifier + .fillMaxWidth() + .height(56.dp), + enabled = !uiState.isLoading, + shape = CircleShape, + colors = ButtonDefaults.buttonColors( + containerColor = FurumiNeonPink, + contentColor = MaterialTheme.colorScheme.onPrimary, + disabledContainerColor = FurumiSurfaceHigh, + disabledContentColor = FurumiTextMuted + ) + ) { + if (isSsoLoading) { + CircularProgressIndicator( + modifier = Modifier.size(24.dp), + color = MaterialTheme.colorScheme.onPrimary, + strokeWidth = 2.dp + ) + } else { + Text("Continue with SSO") + } + } + + Spacer(modifier = Modifier.height(14.dp)) + + TextButton( + onClick = onPasswordLoginVisibilityChange, + modifier = Modifier.align(Alignment.CenterHorizontally), + enabled = !uiState.isLoading, + colors = ButtonDefaults.textButtonColors(contentColor = MaterialTheme.colorScheme.onBackground) + ) { + Text( + text = if (isPasswordLoginVisible) "Hide password login" else "Sign in with password", + style = MaterialTheme.typography.labelSmall + ) + } + + if (isPasswordLoginVisible) { + Spacer(modifier = Modifier.height(8.dp)) + HorizontalDivider(color = FurumiLine) + Spacer(modifier = Modifier.height(18.dp)) + + LoginTextField( + value = uiState.login, + onValueChange = onLoginChange, + label = "Username or email", + placeholder = "name@example.com", + enabled = !uiState.isLoading, + keyboardType = KeyboardType.Email, + imeAction = ImeAction.Next, + keyboardActions = KeyboardActions( + onNext = { passwordFocusRequester.requestFocus() } + ), + modifier = Modifier.focusRequester(loginFocusRequester) + ) + + Spacer(modifier = Modifier.height(10.dp)) + + LoginTextField( + value = uiState.password, + onValueChange = onPasswordChange, + label = "Password", + enabled = !uiState.isLoading, + keyboardType = KeyboardType.Password, + imeAction = ImeAction.Done, + keyboardActions = KeyboardActions( + onDone = { onPasswordLoginClick() } + ), + visualTransformation = PasswordVisualTransformation(), + modifier = Modifier.focusRequester(passwordFocusRequester) + ) + + Spacer(modifier = Modifier.height(16.dp)) + + OutlinedButton( + onClick = onPasswordLoginClick, + modifier = Modifier + .fillMaxWidth() + .height(54.dp), + enabled = !uiState.isLoading, + shape = CircleShape, + border = BorderStroke(1.dp, MaterialTheme.colorScheme.onBackground), + colors = ButtonDefaults.outlinedButtonColors( + contentColor = MaterialTheme.colorScheme.onBackground, + disabledContentColor = FurumiTextMuted + ) + ) { + if (isPasswordLoading) { + CircularProgressIndicator( + modifier = Modifier.size(24.dp), + color = MaterialTheme.colorScheme.onBackground, + strokeWidth = 2.dp + ) + } else { + Text("Sign in") + } + } + } + } +} + +@Composable +private fun LoginTextField( + value: String, + onValueChange: (String) -> Unit, + label: String, + enabled: Boolean, + keyboardType: KeyboardType, + imeAction: ImeAction, + keyboardActions: KeyboardActions, + modifier: Modifier = Modifier, + placeholder: String? = null, + visualTransformation: PasswordVisualTransformation? = null +) { + OutlinedTextField( + value = value, + onValueChange = onValueChange, + label = { Text(label) }, + modifier = modifier.fillMaxWidth(), + placeholder = placeholder?.let { { Text(it) } }, + enabled = enabled, + singleLine = true, + maxLines = 1, + visualTransformation = visualTransformation ?: androidx.compose.ui.text.input.VisualTransformation.None, + keyboardOptions = KeyboardOptions( + keyboardType = keyboardType, + imeAction = imeAction + ), + keyboardActions = keyboardActions, + shape = RoundedCornerShape(8.dp), + colors = OutlinedTextFieldDefaults.colors( + focusedTextColor = MaterialTheme.colorScheme.onBackground, + unfocusedTextColor = MaterialTheme.colorScheme.onBackground, + disabledTextColor = FurumiTextMuted, + focusedContainerColor = FurumiSurface, + unfocusedContainerColor = FurumiSurface, + disabledContainerColor = FurumiSurface, + focusedBorderColor = FurumiNeonPink, + unfocusedBorderColor = FurumiLine, + disabledBorderColor = FurumiLine, + focusedLabelColor = FurumiNeonPink, + unfocusedLabelColor = FurumiTextMuted, + disabledLabelColor = FurumiTextMuted, + focusedPlaceholderColor = FurumiTextMuted, + unfocusedPlaceholderColor = FurumiTextMuted + ) + ) +} diff --git a/app/src/main/java/com/example/furumi_android/ui/login/LoginViewModel.kt b/app/src/main/java/com/example/furumi_android/ui/login/LoginViewModel.kt new file mode 100644 index 0000000..57b7064 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/ui/login/LoginViewModel.kt @@ -0,0 +1,161 @@ +package com.example.furumi_android.ui.login + +import android.net.Uri +import androidx.lifecycle.ViewModel +import androidx.lifecycle.viewModelScope +import com.example.furumi_android.domain.model.ServerConfig +import com.example.furumi_android.domain.repository.AuthRepository +import dagger.hilt.android.lifecycle.HiltViewModel +import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.StateFlow +import kotlinx.coroutines.flow.asStateFlow +import kotlinx.coroutines.launch +import javax.inject.Inject + +data class LoginUiState( + val serverUrl: String = "", + val login: String = "", + val password: String = "", + val isLoading: Boolean = false, + val loadingAction: LoginLoadingAction? = null, + val error: String? = null, + val isAuthenticated: Boolean = false, + val externalAuthUrl: String? = null +) + +enum class LoginLoadingAction { + Password, + Sso +} + +@HiltViewModel +class LoginViewModel @Inject constructor( + private val authRepository: AuthRepository +) : ViewModel() { + private val _uiState = MutableStateFlow( + LoginUiState( + serverUrl = authRepository.getSavedServerUrl().orEmpty(), + isAuthenticated = authRepository.getCurrentSession() != null + ) + ) + val uiState: StateFlow = _uiState.asStateFlow() + + fun onServerUrlChange(url: String) { + _uiState.value = _uiState.value.copy(serverUrl = url.withoutLineBreaks(), error = null) + } + + fun onLoginChange(login: String) { + _uiState.value = _uiState.value.copy(login = login.withoutLineBreaks(), error = null) + } + + fun onPasswordChange(password: String) { + _uiState.value = _uiState.value.copy(password = password.withoutLineBreaks(), error = null) + } + + fun login() { + val state = _uiState.value + val password = state.password + + if (password.isBlank()) { + _uiState.value = _uiState.value.copy(error = "Password is required") + return + } + + val config = ServerConfig.credentials(state.serverUrl, state.login).getOrElse { + _uiState.value = _uiState.value.copy(error = it.message ?: "Check your login details") + return + } + + viewModelScope.launch { + _uiState.value = _uiState.value.copy( + isLoading = true, + loadingAction = LoginLoadingAction.Password, + error = null + ) + + val result = authRepository.login( + config = config, + password = password + ) + + result.onSuccess { + _uiState.value = _uiState.value.copy( + isLoading = false, + loadingAction = null, + password = "", + serverUrl = it.serverBaseUrl, + isAuthenticated = true + ) + }.onFailure { + _uiState.value = _uiState.value.copy( + isLoading = false, + loadingAction = null, + error = it.message + ) + } + } + } + + fun loginWithSSO() { + val state = _uiState.value + + viewModelScope.launch { + _uiState.value = _uiState.value.copy( + isLoading = true, + loadingAction = LoginLoadingAction.Sso, + error = null + ) + + authRepository.createSsoAuthorizationUrl(state.serverUrl) + .onSuccess { authUrl -> + _uiState.value = _uiState.value.copy( + isLoading = false, + loadingAction = null, + externalAuthUrl = authUrl + ) + } + .onFailure { + _uiState.value = _uiState.value.copy( + isLoading = false, + loadingAction = null, + error = it.message ?: "Unable to start SSO login" + ) + } + } + } + + fun onExternalAuthUrlOpened() { + _uiState.value = _uiState.value.copy(externalAuthUrl = null) + } + + fun handleDeepLink(uri: Uri?) { + val callbackUrl = uri?.toString() ?: return + + viewModelScope.launch { + _uiState.value = _uiState.value.copy( + isLoading = true, + loadingAction = LoginLoadingAction.Sso, + error = null + ) + val result = authRepository.completeSsoLogin(callbackUrl) + result.onSuccess { + _uiState.value = _uiState.value.copy( + isLoading = false, + loadingAction = null, + serverUrl = it.serverBaseUrl, + isAuthenticated = true + ) + }.onFailure { + _uiState.value = _uiState.value.copy( + isLoading = false, + loadingAction = null, + error = it.message + ) + } + } + } + + private fun String.withoutLineBreaks(): String { + return replace("\r", "").replace("\n", "") + } +} diff --git a/app/src/main/java/com/example/furumi_android/ui/player/PlayerScreen.kt b/app/src/main/java/com/example/furumi_android/ui/player/PlayerScreen.kt new file mode 100644 index 0000000..2fe0a11 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/ui/player/PlayerScreen.kt @@ -0,0 +1,2894 @@ +package com.example.furumi_android.ui.player + +import android.graphics.Bitmap +import androidx.activity.compose.BackHandler +import androidx.compose.foundation.Canvas +import androidx.compose.foundation.Image +import androidx.compose.foundation.background +import androidx.compose.foundation.border +import androidx.compose.foundation.clickable +import androidx.compose.foundation.horizontalScroll +import androidx.compose.foundation.layout.* +import androidx.compose.foundation.lazy.grid.GridCells +import androidx.compose.foundation.lazy.grid.GridItemSpan +import androidx.compose.foundation.lazy.grid.LazyVerticalGrid +import androidx.compose.foundation.lazy.grid.items +import androidx.compose.foundation.rememberScrollState +import androidx.compose.foundation.shape.CircleShape +import androidx.compose.foundation.shape.RoundedCornerShape +import androidx.compose.foundation.verticalScroll +import androidx.compose.material3.Button +import androidx.compose.material3.ButtonDefaults +import androidx.compose.material3.CircularProgressIndicator +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.OutlinedButton +import androidx.compose.material3.Slider +import androidx.compose.material3.SliderDefaults +import androidx.compose.material3.Surface +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.runtime.LaunchedEffect +import androidx.compose.runtime.collectAsState +import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.saveable.rememberSaveable +import androidx.compose.runtime.setValue +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.draw.clip +import androidx.compose.ui.geometry.Offset +import androidx.compose.ui.graphics.Brush +import androidx.compose.ui.graphics.Color +import androidx.compose.ui.graphics.Path +import androidx.compose.ui.graphics.StrokeCap +import androidx.compose.ui.graphics.asImageBitmap +import androidx.compose.ui.graphics.drawscope.Stroke +import androidx.compose.ui.layout.ContentScale +import androidx.compose.ui.text.font.FontWeight +import androidx.compose.ui.text.style.TextOverflow +import androidx.compose.ui.unit.dp +import androidx.hilt.navigation.compose.hiltViewModel +import com.example.furumi_android.domain.model.ArtistCard +import com.example.furumi_android.domain.model.ReleaseCard +import com.example.furumi_android.domain.model.TrackCard +import com.example.furumi_android.domain.model.ListeningHistoryItem +import com.example.furumi_android.playback.AudioPlaybackState +import com.example.furumi_android.ui.theme.FurumiBlack +import com.example.furumi_android.ui.theme.FurumiElectricCyan +import com.example.furumi_android.ui.theme.FurumiHotOrange +import com.example.furumi_android.ui.theme.FurumiLine +import com.example.furumi_android.ui.theme.FurumiNeonPink +import com.example.furumi_android.ui.theme.FurumiNeonViolet +import com.example.furumi_android.ui.theme.FurumiSurface +import com.example.furumi_android.ui.theme.FurumiSurfaceHigh +import com.example.furumi_android.ui.theme.FurumiTextMuted + +private enum class PlayerTab( + val label: String +) { + Global("Global"), + Search("Search"), + Library("Library") +} + +private data class MockTrack( + val title: String, + val artist: String, + val duration: String, + val colors: List +) + +private data class MockPlaylist( + val title: String, + val subtitle: String, + val description: String, + val colors: List, + val tracks: List +) + +@Composable +fun PlayerScreen( + viewModel: PlayerViewModel = hiltViewModel(), + onLoggedOut: () -> Unit +) { + val uiState by viewModel.uiState.collectAsState() + var selectedTabName by rememberSaveable { mutableStateOf(PlayerTab.Global.name) } + val selectedTab = PlayerTab.entries.firstOrNull { it.name == selectedTabName } ?: PlayerTab.Global + var isProfileMenuOpen by rememberSaveable { mutableStateOf(false) } + var selectedPlaylistTitle by rememberSaveable { mutableStateOf(null) } + var isFullPlayerOpen by rememberSaveable { mutableStateOf(false) } + val selectedPlaylist = selectedPlaylistTitle?.let { title -> + mockPlaylists.firstOrNull { it.title == title } + } + + BackHandler( + enabled = selectedPlaylist != null || + uiState.selectedRelease != null || + uiState.selectedArtist != null || + isFullPlayerOpen || + isProfileMenuOpen || + uiState.isListeningHistoryVisible + ) { + when { + uiState.isListeningHistoryVisible -> viewModel.closeListeningHistory() + isFullPlayerOpen -> isFullPlayerOpen = false + selectedPlaylist != null -> selectedPlaylistTitle = null + uiState.selectedRelease != null -> viewModel.closeReleaseDetail() + uiState.selectedArtist != null -> viewModel.closeArtistDetail() + isProfileMenuOpen -> isProfileMenuOpen = false + } + } + + LaunchedEffect(uiState.isLoggedOut) { + if (uiState.isLoggedOut) { + onLoggedOut() + } + } + + Box( + modifier = Modifier + .fillMaxSize() + .background(MaterialTheme.colorScheme.background) + .windowInsetsPadding(WindowInsets.safeDrawing) + ) { + Column(modifier = Modifier.fillMaxSize()) { + Box( + modifier = Modifier + .weight(1f) + .fillMaxWidth() + ) { + val onProfileClick = { isProfileMenuOpen = !isProfileMenuOpen } + if (selectedPlaylist != null) { + Box( + modifier = Modifier + .fillMaxSize() + .verticalScroll(rememberScrollState()) + ) { + PlaylistDetailContent( + playlist = selectedPlaylist, + onBack = { selectedPlaylistTitle = null }, + onTrackClick = { isFullPlayerOpen = true }, + onPlayClick = { isFullPlayerOpen = true } + ) + } + } else if (uiState.selectedRelease != null) { + Box( + modifier = Modifier + .fillMaxSize() + .verticalScroll(rememberScrollState()) + ) { + ReleaseDetailContent( + uiState = uiState, + onBack = viewModel::closeReleaseDetail, + onRetry = viewModel::retryReleaseDetail, + onMediaImageNeeded = viewModel::loadMediaImage, + onPlayClick = { + if (viewModel.playReleaseTracks(shuffle = false)) { + isFullPlayerOpen = true + } + }, + onShuffleClick = { + if (viewModel.playReleaseTracks(shuffle = true)) { + isFullPlayerOpen = true + } + }, + onTrackClick = { track -> + if (viewModel.playTrack(track, uiState.releaseDetail?.tracks.orEmpty())) { + isFullPlayerOpen = true + } + } + ) + } + } else if (uiState.selectedArtist != null) { + Box( + modifier = Modifier + .fillMaxSize() + .verticalScroll(rememberScrollState()) + ) { + ArtistDetailContent( + uiState = uiState, + onBack = viewModel::closeArtistDetail, + onRetry = viewModel::retryArtistDetail, + onMediaImageNeeded = viewModel::loadMediaImage, + onPlayClick = { + if (viewModel.playArtistTracks(shuffle = false)) { + isFullPlayerOpen = true + } + }, + onRadioClick = { + if (viewModel.playArtistTracks(shuffle = true)) { + isFullPlayerOpen = true + } + }, + onTrackClick = { track -> + if (viewModel.playTrack(track, uiState.artistDetail?.topTracks.orEmpty())) { + isFullPlayerOpen = true + } + }, + onReleaseClick = viewModel::openRelease + ) + } + } else { + when (selectedTab) { + PlayerTab.Global -> GlobalContent( + uiState = uiState, + onProfileClick = onProfileClick, + onRetry = viewModel::retryGlobalArtists, + onLoadMore = viewModel::loadMoreGlobalArtists, + onArtistImageNeeded = viewModel::loadMediaImage, + onArtistClick = viewModel::openArtist + ) + PlayerTab.Search -> Box( + modifier = Modifier + .fillMaxSize() + .verticalScroll(rememberScrollState()) + ) { + SearchContent(onProfileClick) + } + PlayerTab.Library -> Box( + modifier = Modifier + .fillMaxSize() + .verticalScroll(rememberScrollState()) + ) { + LibraryContent( + onProfileClick = onProfileClick, + onPlaylistClick = { selectedPlaylistTitle = it.title } + ) + } + } + } + } + + uiState.playback.currentTrack?.let { track -> + NowPlayingBar( + playback = uiState.playback, + coverBitmap = track.coverUrl?.let { uiState.mediaImages[it] }, + onMediaImageNeeded = viewModel::loadMediaImage, + onClick = { isFullPlayerOpen = true }, + onPlayPause = viewModel::togglePlayPause + ) + } + BottomPlayerNav( + selectedTab = selectedTab, + onTabSelected = { + selectedTabName = it.name + selectedPlaylistTitle = null + } + ) + } + + if (isProfileMenuOpen) { + ProfileMenu( + uiState = uiState, + onHistoryClick = { + isProfileMenuOpen = false + viewModel.openListeningHistory() + }, + onLogout = viewModel::logout, + modifier = Modifier + .align(Alignment.TopEnd) + .padding(top = 72.dp, end = 20.dp) + ) + } + + if (isFullPlayerOpen && uiState.playback.currentTrack != null) { + FullPlayerOverlay( + playback = uiState.playback, + coverBitmap = uiState.playback.currentTrack?.coverUrl?.let { uiState.mediaImages[it] }, + onMediaImageNeeded = viewModel::loadMediaImage, + onDismiss = { isFullPlayerOpen = false }, + onPlayPause = viewModel::togglePlayPause, + onPrevious = viewModel::previousTrack, + onNext = viewModel::nextTrack, + onSeekToProgress = viewModel::seekToPlaybackProgress, + onQueueTrackClick = { track -> + viewModel.playTrack(track, uiState.playback.queue) + } + ) + } + + if (uiState.isListeningHistoryVisible) { + ListeningHistoryPanel( + uiState = uiState, + onDismiss = viewModel::closeListeningHistory, + onRetry = viewModel::retryListeningHistory + ) + } + } +} + +@Composable +private fun GlobalContent( + uiState: PlayerUiState, + onProfileClick: () -> Unit, + onRetry: () -> Unit, + onLoadMore: () -> Unit, + onArtistImageNeeded: (String?) -> Unit, + onArtistClick: (ArtistCard) -> Unit +) { + LazyVerticalGrid( + columns = GridCells.Adaptive(minSize = 148.dp), + modifier = Modifier.fillMaxSize(), + contentPadding = PaddingValues(horizontal = 20.dp, vertical = 24.dp), + horizontalArrangement = Arrangement.spacedBy(14.dp), + verticalArrangement = Arrangement.spacedBy(18.dp) + ) { + item(span = { GridItemSpan(maxLineSpan) }) { + PlayerHeader( + title = "Global", + subtitle = artistsSubtitle(uiState), + onProfileClick = onProfileClick + ) + } + + when { + uiState.isGlobalArtistsLoading && uiState.globalArtists.isEmpty() -> { + item(span = { GridItemSpan(maxLineSpan) }) { GlobalArtistsLoading() } + } + + uiState.globalArtistsError != null && uiState.globalArtists.isEmpty() -> { + item(span = { GridItemSpan(maxLineSpan) }) { + GlobalArtistsError( + message = uiState.globalArtistsError, + onRetry = onRetry + ) + } + } + + uiState.globalArtists.isEmpty() -> { + item(span = { GridItemSpan(maxLineSpan) }) { GlobalArtistsEmpty() } + } + + else -> { + item(span = { GridItemSpan(maxLineSpan) }) { + Text( + text = "Artists", + style = MaterialTheme.typography.titleMedium, + color = MaterialTheme.colorScheme.onBackground + ) + } + + items( + items = uiState.globalArtists, + key = { artist -> artist.id } + ) { artist -> + ArtistTile( + artist = artist, + bitmap = artist.imageUrl?.let { uiState.mediaImages[it] }, + onArtistImageNeeded = onArtistImageNeeded, + onClick = { onArtistClick(artist) } + ) + } + + item(span = { GridItemSpan(maxLineSpan) }) { + GlobalArtistsFooter( + uiState = uiState, + onLoadMore = onLoadMore + ) + } + } + } + } +} + +@Composable +private fun ArtistTile( + artist: ArtistCard, + bitmap: Bitmap?, + onArtistImageNeeded: (String?) -> Unit, + onClick: () -> Unit +) { + LaunchedEffect(artist.imageUrl) { + onArtistImageNeeded(artist.imageUrl) + } + + Column( + modifier = Modifier + .fillMaxWidth() + .clip(RoundedCornerShape(8.dp)) + .clickable(onClick = onClick) + ) { + ArtistArtwork( + artist = artist, + bitmap = bitmap, + modifier = Modifier + .fillMaxWidth() + .aspectRatio(1f) + ) + Spacer(modifier = Modifier.height(10.dp)) + Text( + text = artist.name, + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Spacer(modifier = Modifier.height(2.dp)) + Text( + text = "${pluralCount(artist.releaseCount, "release")} / ${pluralCount(artist.trackCount, "track")}", + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + } +} + +@Composable +private fun ArtistArtwork( + artist: ArtistCard, + bitmap: Bitmap?, + modifier: Modifier +) { + MediaArtwork( + title = artist.name, + seedId = artist.id, + bitmap = bitmap, + modifier = modifier, + cornerRadius = 8 + ) +} + +@Composable +private fun MediaArtwork( + title: String, + seedId: Long, + bitmap: Bitmap?, + modifier: Modifier, + cornerRadius: Int +) { + Box( + modifier = modifier + .clip(RoundedCornerShape(cornerRadius.dp)) + .background(Brush.linearGradient(artistArtworkColors(seedId))) + .border(1.dp, Color.White.copy(alpha = 0.10f), RoundedCornerShape(cornerRadius.dp)), + contentAlignment = Alignment.Center + ) { + if (bitmap != null) { + Image( + bitmap = bitmap.asImageBitmap(), + contentDescription = "$title image", + modifier = Modifier.fillMaxSize(), + contentScale = ContentScale.Crop + ) + } else { + Text( + text = title.firstOrNull()?.uppercaseChar()?.toString() ?: "#", + style = MaterialTheme.typography.titleMedium, + color = Color.White, + fontWeight = FontWeight.ExtraBold + ) + } + } +} + +@Composable +private fun ArtistDetailContent( + uiState: PlayerUiState, + onBack: () -> Unit, + onRetry: () -> Unit, + onMediaImageNeeded: (String?) -> Unit, + onPlayClick: () -> Unit, + onRadioClick: () -> Unit, + onTrackClick: (TrackCard) -> Unit, + onReleaseClick: (ReleaseCard) -> Unit +) { + val artist = uiState.artistDetail?.artist ?: uiState.selectedArtist ?: return + val artistImage = artist.imageUrl?.let { uiState.mediaImages[it] } + + LaunchedEffect(artist.imageUrl) { + onMediaImageNeeded(artist.imageUrl) + } + + Column( + modifier = Modifier + .fillMaxWidth() + .padding(horizontal = 20.dp, vertical = 20.dp) + ) { + Row( + modifier = Modifier + .clip(RoundedCornerShape(8.dp)) + .clickable(onClick = onBack) + .padding(vertical = 8.dp, horizontal = 2.dp), + verticalAlignment = Alignment.CenterVertically + ) { + BackGlyph( + modifier = Modifier.size(20.dp), + color = MaterialTheme.colorScheme.onBackground + ) + Spacer(modifier = Modifier.width(8.dp)) + Text( + text = "Artist", + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground + ) + } + + Spacer(modifier = Modifier.height(20.dp)) + + ArtistDetailHero( + artist = artist, + bitmap = artistImage, + onPlayClick = onPlayClick, + onRadioClick = onRadioClick + ) + + Spacer(modifier = Modifier.height(28.dp)) + + when { + uiState.isArtistDetailLoading && uiState.artistDetail == null -> ArtistDetailLoading() + uiState.artistDetailError != null && uiState.artistDetail == null -> ArtistDetailError( + message = uiState.artistDetailError, + onRetry = onRetry + ) + uiState.artistDetail != null -> { + val detail = uiState.artistDetail + if (detail.topTracks.isNotEmpty()) { + PopularTracksSection( + tracks = detail.topTracks, + mediaImages = uiState.mediaImages, + onMediaImageNeeded = onMediaImageNeeded, + onTrackClick = onTrackClick + ) + Spacer(modifier = Modifier.height(28.dp)) + } + + ReleasesSection( + releases = detail.releases, + mediaImages = uiState.mediaImages, + onMediaImageNeeded = onMediaImageNeeded, + onReleaseClick = onReleaseClick + ) + } + } + + Spacer(modifier = Modifier.height(24.dp)) + } +} + +@Composable +private fun ArtistDetailHero( + artist: ArtistCard, + bitmap: Bitmap?, + onPlayClick: () -> Unit, + onRadioClick: () -> Unit +) { + Column { + MediaArtwork( + title = artist.name, + seedId = artist.id, + bitmap = bitmap, + modifier = Modifier + .size(210.dp) + .align(Alignment.CenterHorizontally), + cornerRadius = 10 + ) + Spacer(modifier = Modifier.height(22.dp)) + Text( + text = artist.name, + style = MaterialTheme.typography.headlineMedium, + color = MaterialTheme.colorScheme.onBackground, + maxLines = 2, + overflow = TextOverflow.Ellipsis + ) + Spacer(modifier = Modifier.height(8.dp)) + Text( + text = "${pluralCount(artist.releaseCount, "release")} / ${pluralCount(artist.trackCount, "track")}", + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + Spacer(modifier = Modifier.height(18.dp)) + Row( + modifier = Modifier.fillMaxWidth(), + verticalAlignment = Alignment.CenterVertically + ) { + PillAction( + label = "Play", + selected = true, + modifier = Modifier.weight(1f), + onClick = onPlayClick + ) + Spacer(modifier = Modifier.width(12.dp)) + PillAction( + label = "Radio", + selected = false, + modifier = Modifier.weight(1f), + onClick = onRadioClick + ) + } + } +} + +@Composable +private fun ArtistDetailLoading() { + Box( + modifier = Modifier + .fillMaxWidth() + .height(180.dp), + contentAlignment = Alignment.Center + ) { + CircularProgressIndicator( + color = FurumiNeonPink, + strokeWidth = 2.dp + ) + } +} + +@Composable +private fun ArtistDetailError( + message: String, + onRetry: () -> Unit +) { + Surface( + modifier = Modifier.fillMaxWidth(), + shape = RoundedCornerShape(12.dp), + color = FurumiSurface, + border = androidx.compose.foundation.BorderStroke(1.dp, FurumiLine) + ) { + Column(modifier = Modifier.padding(16.dp)) { + Text( + text = "Could not load artist", + style = MaterialTheme.typography.titleMedium, + color = MaterialTheme.colorScheme.onBackground + ) + Spacer(modifier = Modifier.height(6.dp)) + Text( + text = message, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + Spacer(modifier = Modifier.height(14.dp)) + PillAction( + label = "Retry", + selected = true, + onClick = onRetry + ) + } + } +} + +@Composable +private fun PopularTracksSection( + tracks: List, + mediaImages: Map, + onMediaImageNeeded: (String?) -> Unit, + onTrackClick: (TrackCard) -> Unit +) { + SectionTitle("Popular") + Spacer(modifier = Modifier.height(12.dp)) + Column(verticalArrangement = Arrangement.spacedBy(12.dp)) { + tracks.take(5).forEachIndexed { index, track -> + ArtistDetailTrackRow( + index = index + 1, + track = track, + bitmap = track.coverUrl?.let { mediaImages[it] }, + onMediaImageNeeded = onMediaImageNeeded, + onTrackClick = onTrackClick + ) + } + } +} + +@Composable +private fun ArtistDetailTrackRow( + index: Int, + track: TrackCard, + bitmap: Bitmap?, + onMediaImageNeeded: (String?) -> Unit, + onTrackClick: (TrackCard) -> Unit +) { + LaunchedEffect(track.coverUrl) { + onMediaImageNeeded(track.coverUrl) + } + + Row( + modifier = Modifier + .fillMaxWidth() + .clip(RoundedCornerShape(8.dp)) + .clickable { onTrackClick(track) } + .padding(vertical = 2.dp), + verticalAlignment = Alignment.CenterVertically + ) { + Text( + text = index.toString(), + modifier = Modifier.width(28.dp), + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + MediaArtwork( + title = track.title, + seedId = track.id, + bitmap = bitmap, + modifier = Modifier.size(48.dp), + cornerRadius = 6 + ) + Spacer(modifier = Modifier.width(12.dp)) + Column(modifier = Modifier.weight(1f)) { + Text( + text = track.title, + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Text( + text = track.artists.joinToString(", ") + .ifBlank { track.releaseTitle.orEmpty() } + .ifBlank { "Unknown artist" }, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + } + Text( + text = formatDuration(track.durationSeconds), + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + } +} + +@Composable +private fun ReleasesSection( + releases: List, + mediaImages: Map, + onMediaImageNeeded: (String?) -> Unit, + onReleaseClick: (ReleaseCard) -> Unit +) { + SectionTitle("Releases") + Spacer(modifier = Modifier.height(12.dp)) + + if (releases.isEmpty()) { + Surface( + modifier = Modifier.fillMaxWidth(), + shape = RoundedCornerShape(12.dp), + color = FurumiSurface, + border = androidx.compose.foundation.BorderStroke(1.dp, FurumiLine) + ) { + Text( + text = "No releases yet", + modifier = Modifier.padding(16.dp), + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + } + return + } + + Column(verticalArrangement = Arrangement.spacedBy(18.dp)) { + releases.chunked(2).forEach { row -> + Row( + modifier = Modifier.fillMaxWidth(), + horizontalArrangement = Arrangement.spacedBy(14.dp) + ) { + row.forEach { release -> + ReleaseTile( + release = release, + bitmap = release.coverUrl?.let { mediaImages[it] }, + onMediaImageNeeded = onMediaImageNeeded, + onClick = { onReleaseClick(release) }, + modifier = Modifier.weight(1f) + ) + } + + if (row.size == 1) { + Spacer(modifier = Modifier.weight(1f)) + } + } + } + } +} + +@Composable +private fun ReleaseTile( + release: ReleaseCard, + bitmap: Bitmap?, + onMediaImageNeeded: (String?) -> Unit, + onClick: () -> Unit, + modifier: Modifier = Modifier +) { + LaunchedEffect(release.coverUrl) { + onMediaImageNeeded(release.coverUrl) + } + + Column( + modifier = modifier + .clip(RoundedCornerShape(8.dp)) + .clickable(onClick = onClick) + ) { + MediaArtwork( + title = release.title, + seedId = release.id, + bitmap = bitmap, + modifier = Modifier + .fillMaxWidth() + .aspectRatio(1f), + cornerRadius = 8 + ) + Spacer(modifier = Modifier.height(10.dp)) + Text( + text = release.title, + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Spacer(modifier = Modifier.height(2.dp)) + Text( + text = releaseMeta(release), + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + } +} + +@Composable +private fun ReleaseDetailContent( + uiState: PlayerUiState, + onBack: () -> Unit, + onRetry: () -> Unit, + onMediaImageNeeded: (String?) -> Unit, + onPlayClick: () -> Unit, + onShuffleClick: () -> Unit, + onTrackClick: (TrackCard) -> Unit +) { + val release = uiState.releaseDetail?.release ?: uiState.selectedRelease ?: return + val releaseImage = release.coverUrl?.let { uiState.mediaImages[it] } + val artists = uiState.releaseDetail?.artists.orEmpty() + + LaunchedEffect(release.coverUrl) { + onMediaImageNeeded(release.coverUrl) + } + + Column( + modifier = Modifier + .fillMaxWidth() + .padding(horizontal = 20.dp, vertical = 20.dp) + ) { + Row( + modifier = Modifier + .clip(RoundedCornerShape(8.dp)) + .clickable(onClick = onBack) + .padding(vertical = 8.dp, horizontal = 2.dp), + verticalAlignment = Alignment.CenterVertically + ) { + BackGlyph( + modifier = Modifier.size(20.dp), + color = MaterialTheme.colorScheme.onBackground + ) + Spacer(modifier = Modifier.width(8.dp)) + Text( + text = "Release", + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground + ) + } + + Spacer(modifier = Modifier.height(20.dp)) + + ReleaseDetailHero( + release = release, + artistsText = releaseArtistsText(artists), + bitmap = releaseImage, + onPlayClick = onPlayClick, + onShuffleClick = onShuffleClick + ) + + Spacer(modifier = Modifier.height(28.dp)) + + when { + uiState.isReleaseDetailLoading && uiState.releaseDetail == null -> ReleaseDetailLoading() + uiState.releaseDetailError != null && uiState.releaseDetail == null -> ReleaseDetailError( + message = uiState.releaseDetailError, + onRetry = onRetry + ) + uiState.releaseDetail != null -> ReleaseTracksSection( + tracks = uiState.releaseDetail.tracks, + onTrackClick = onTrackClick + ) + } + + Spacer(modifier = Modifier.height(24.dp)) + } +} + +@Composable +private fun ReleaseDetailHero( + release: ReleaseCard, + artistsText: String, + bitmap: Bitmap?, + onPlayClick: () -> Unit, + onShuffleClick: () -> Unit +) { + Column { + MediaArtwork( + title = release.title, + seedId = release.id, + bitmap = bitmap, + modifier = Modifier + .size(210.dp) + .align(Alignment.CenterHorizontally), + cornerRadius = 10 + ) + Spacer(modifier = Modifier.height(22.dp)) + Text( + text = release.title, + style = MaterialTheme.typography.headlineMedium, + color = MaterialTheme.colorScheme.onBackground, + maxLines = 2, + overflow = TextOverflow.Ellipsis + ) + Spacer(modifier = Modifier.height(8.dp)) + Text( + text = artistsText, + style = MaterialTheme.typography.bodyLarge, + color = MaterialTheme.colorScheme.onBackground, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Spacer(modifier = Modifier.height(4.dp)) + Text( + text = releaseMeta(release), + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Spacer(modifier = Modifier.height(18.dp)) + Row( + modifier = Modifier.fillMaxWidth(), + verticalAlignment = Alignment.CenterVertically + ) { + PillAction( + label = "Play", + selected = true, + modifier = Modifier.weight(1f), + onClick = onPlayClick + ) + Spacer(modifier = Modifier.width(12.dp)) + PillAction( + label = "Shuffle", + selected = false, + modifier = Modifier.weight(1f), + onClick = onShuffleClick + ) + } + } +} + +@Composable +private fun ReleaseDetailLoading() { + Box( + modifier = Modifier + .fillMaxWidth() + .height(180.dp), + contentAlignment = Alignment.Center + ) { + CircularProgressIndicator( + color = FurumiNeonPink, + strokeWidth = 2.dp + ) + } +} + +@Composable +private fun ReleaseDetailError( + message: String, + onRetry: () -> Unit +) { + Surface( + modifier = Modifier.fillMaxWidth(), + shape = RoundedCornerShape(12.dp), + color = FurumiSurface, + border = androidx.compose.foundation.BorderStroke(1.dp, FurumiLine) + ) { + Column(modifier = Modifier.padding(16.dp)) { + Text( + text = "Could not load release", + style = MaterialTheme.typography.titleMedium, + color = MaterialTheme.colorScheme.onBackground + ) + Spacer(modifier = Modifier.height(6.dp)) + Text( + text = message, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + Spacer(modifier = Modifier.height(14.dp)) + PillAction( + label = "Retry", + selected = true, + onClick = onRetry + ) + } + } +} + +@Composable +private fun ReleaseTracksSection( + tracks: List, + onTrackClick: (TrackCard) -> Unit +) { + SectionTitle("Tracks") + Spacer(modifier = Modifier.height(12.dp)) + + if (tracks.isEmpty()) { + Surface( + modifier = Modifier.fillMaxWidth(), + shape = RoundedCornerShape(12.dp), + color = FurumiSurface, + border = androidx.compose.foundation.BorderStroke(1.dp, FurumiLine) + ) { + Text( + text = "No tracks yet", + modifier = Modifier.padding(16.dp), + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + } + return + } + + val groupedTracks = tracks.groupBy { it.discNumber ?: 1 }.toSortedMap() + + Column(verticalArrangement = Arrangement.spacedBy(12.dp)) { + groupedTracks.forEach { (discNumber, discTracks) -> + if (groupedTracks.size > 1) { + Text( + text = "Disc $discNumber", + style = MaterialTheme.typography.labelLarge, + color = FurumiTextMuted + ) + } + + discTracks.forEachIndexed { index, track -> + ReleaseTrackRow( + fallbackIndex = index + 1, + track = track, + onTrackClick = onTrackClick + ) + } + } + } +} + +@Composable +private fun ReleaseTrackRow( + fallbackIndex: Int, + track: TrackCard, + onTrackClick: (TrackCard) -> Unit +) { + Row( + modifier = Modifier + .fillMaxWidth() + .clip(RoundedCornerShape(8.dp)) + .clickable { onTrackClick(track) } + .padding(vertical = 4.dp), + verticalAlignment = Alignment.CenterVertically + ) { + Text( + text = (track.trackNumber ?: fallbackIndex).toString(), + modifier = Modifier.width(32.dp), + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + Column(modifier = Modifier.weight(1f)) { + Text( + text = track.title, + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Text( + text = track.artists.joinToString(", ").ifBlank { "Unknown artist" }, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + } + Spacer(modifier = Modifier.width(10.dp)) + Text( + text = formatDuration(track.durationSeconds), + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + } +} + +@Composable +private fun GlobalArtistsFooter( + uiState: PlayerUiState, + onLoadMore: () -> Unit +) { + when { + uiState.isGlobalArtistsLoadingMore -> { + Box( + modifier = Modifier + .fillMaxWidth() + .height(72.dp), + contentAlignment = Alignment.Center + ) { + CircularProgressIndicator( + color = FurumiNeonPink, + strokeWidth = 2.dp, + modifier = Modifier.size(24.dp) + ) + } + } + + uiState.globalArtistsError != null -> { + Surface( + modifier = Modifier.fillMaxWidth(), + shape = RoundedCornerShape(10.dp), + color = FurumiSurface, + border = androidx.compose.foundation.BorderStroke(1.dp, FurumiLine) + ) { + Row( + modifier = Modifier.padding(12.dp), + verticalAlignment = Alignment.CenterVertically + ) { + Text( + text = "Could not load more artists", + modifier = Modifier.weight(1f), + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Text( + text = "Retry", + modifier = Modifier + .clip(RoundedCornerShape(8.dp)) + .clickable(onClick = onLoadMore) + .padding(horizontal = 10.dp, vertical = 8.dp), + style = MaterialTheme.typography.labelLarge, + color = FurumiNeonPink + ) + } + } + } + + uiState.globalArtistsHasMore -> { + LaunchedEffect(uiState.globalArtists.size, uiState.globalArtistsPage) { + onLoadMore() + } + + Box( + modifier = Modifier + .fillMaxWidth() + .height(72.dp), + contentAlignment = Alignment.Center + ) { + CircularProgressIndicator( + color = FurumiNeonPink, + strokeWidth = 2.dp, + modifier = Modifier.size(24.dp) + ) + } + } + + else -> { + Text( + text = "${uiState.globalArtists.size} artists loaded", + modifier = Modifier + .fillMaxWidth() + .padding(vertical = 18.dp), + style = MaterialTheme.typography.labelSmall, + color = FurumiTextMuted + ) + } + } +} + +@Composable +private fun GlobalArtistsLoading() { + Box( + modifier = Modifier + .fillMaxWidth() + .height(240.dp), + contentAlignment = Alignment.Center + ) { + CircularProgressIndicator( + color = FurumiNeonPink, + strokeWidth = 2.dp + ) + } +} + +@Composable +private fun GlobalArtistsError( + message: String, + onRetry: () -> Unit +) { + Surface( + modifier = Modifier.fillMaxWidth(), + shape = RoundedCornerShape(12.dp), + color = FurumiSurface, + border = androidx.compose.foundation.BorderStroke(1.dp, FurumiLine) + ) { + Column(modifier = Modifier.padding(16.dp)) { + Text( + text = "Could not load artists", + style = MaterialTheme.typography.titleMedium, + color = MaterialTheme.colorScheme.onBackground + ) + Spacer(modifier = Modifier.height(6.dp)) + Text( + text = message, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + Spacer(modifier = Modifier.height(14.dp)) + PillAction( + label = "Retry", + selected = true, + onClick = onRetry + ) + } + } +} + +@Composable +private fun GlobalArtistsEmpty() { + Surface( + modifier = Modifier.fillMaxWidth(), + shape = RoundedCornerShape(12.dp), + color = FurumiSurface, + border = androidx.compose.foundation.BorderStroke(1.dp, FurumiLine) + ) { + Column(modifier = Modifier.padding(16.dp)) { + Text( + text = "No artists yet", + style = MaterialTheme.typography.titleMedium, + color = MaterialTheme.colorScheme.onBackground + ) + Spacer(modifier = Modifier.height(6.dp)) + Text( + text = "Artists uploaded to this server will appear here.", + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + } + } +} + +@Composable +private fun SearchContent(onProfileClick: () -> Unit) { + Column( + modifier = Modifier + .fillMaxWidth() + .padding(horizontal = 20.dp, vertical = 24.dp) + ) { + PlayerHeader(title = "Search", subtitle = "Find music on your server", onProfileClick = onProfileClick) + Spacer(modifier = Modifier.height(18.dp)) + Surface( + modifier = Modifier + .fillMaxWidth() + .height(54.dp), + shape = RoundedCornerShape(8.dp), + color = MaterialTheme.colorScheme.onBackground + ) { + Row( + modifier = Modifier.padding(horizontal = 16.dp), + verticalAlignment = Alignment.CenterVertically + ) { + SearchGlyph( + modifier = Modifier.size(22.dp), + color = FurumiBlack + ) + Spacer(modifier = Modifier.width(12.dp)) + Text( + text = "Artists, albums, tracks", + style = MaterialTheme.typography.bodyLarge, + color = FurumiBlack, + fontWeight = FontWeight.SemiBold + ) + } + } + + Spacer(modifier = Modifier.height(28.dp)) + SectionTitle("Browse") + Spacer(modifier = Modifier.height(12.dp)) + GenreGrid() + } +} + +@Composable +private fun LibraryContent( + onProfileClick: () -> Unit, + onPlaylistClick: (MockPlaylist) -> Unit +) { + Column( + modifier = Modifier + .fillMaxWidth() + .padding(horizontal = 20.dp, vertical = 24.dp) + ) { + PlayerHeader(title = "Your Library", subtitle = "Playlists and saved albums", onProfileClick = onProfileClick) + Spacer(modifier = Modifier.height(16.dp)) + FilterChips(listOf("Playlists", "Albums", "Artists")) + Spacer(modifier = Modifier.height(18.dp)) + LibraryRows(onPlaylistClick) + } +} + +@Composable +private fun PlaylistDetailContent( + playlist: MockPlaylist, + onBack: () -> Unit, + onTrackClick: (MockTrack) -> Unit, + onPlayClick: () -> Unit +) { + Column( + modifier = Modifier + .fillMaxWidth() + .padding(horizontal = 20.dp, vertical = 20.dp) + ) { + Row( + modifier = Modifier + .clip(RoundedCornerShape(8.dp)) + .clickable(onClick = onBack) + .padding(vertical = 8.dp, horizontal = 2.dp), + verticalAlignment = Alignment.CenterVertically + ) { + BackGlyph( + modifier = Modifier.size(20.dp), + color = MaterialTheme.colorScheme.onBackground + ) + Spacer(modifier = Modifier.width(8.dp)) + Text( + text = "Playlist", + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground + ) + } + + Spacer(modifier = Modifier.height(20.dp)) + + AlbumArtwork( + colors = playlist.colors, + modifier = Modifier + .size(190.dp) + .align(Alignment.CenterHorizontally), + cornerRadius = 10 + ) + + Spacer(modifier = Modifier.height(22.dp)) + + Text( + text = playlist.title, + style = MaterialTheme.typography.headlineMedium, + color = MaterialTheme.colorScheme.onBackground + ) + Spacer(modifier = Modifier.height(6.dp)) + Text( + text = playlist.description, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + Spacer(modifier = Modifier.height(10.dp)) + Text( + text = "${playlist.tracks.size} tracks", + style = MaterialTheme.typography.labelSmall, + color = FurumiTextMuted + ) + + Spacer(modifier = Modifier.height(18.dp)) + + Row( + modifier = Modifier.fillMaxWidth(), + verticalAlignment = Alignment.CenterVertically + ) { + PillAction( + label = "Play", + selected = true, + modifier = Modifier.weight(1f), + onClick = onPlayClick + ) + Spacer(modifier = Modifier.width(12.dp)) + PillAction( + label = "Shuffle", + selected = false, + modifier = Modifier.weight(1f), + onClick = onPlayClick + ) + } + + Spacer(modifier = Modifier.height(24.dp)) + + playlist.tracks.forEachIndexed { index, track -> + PlaylistTrackRow( + index = index + 1, + track = track, + onTrackClick = onTrackClick + ) + if (index < playlist.tracks.lastIndex) { + Spacer(modifier = Modifier.height(12.dp)) + } + } + + Spacer(modifier = Modifier.height(24.dp)) + } +} + +@Composable +private fun ListeningHistoryPanel( + uiState: PlayerUiState, + onDismiss: () -> Unit, + onRetry: () -> Unit +) { + Surface( + modifier = Modifier + .fillMaxSize() + .windowInsetsPadding(WindowInsets.safeDrawing), + color = MaterialTheme.colorScheme.background + ) { + Column( + modifier = Modifier + .fillMaxSize() + .verticalScroll(rememberScrollState()) + .padding(horizontal = 20.dp, vertical = 20.dp) + ) { + Row( + modifier = Modifier.fillMaxWidth(), + verticalAlignment = Alignment.CenterVertically, + horizontalArrangement = Arrangement.SpaceBetween + ) { + Column { + Text( + text = "Listening history", + style = MaterialTheme.typography.headlineMedium, + color = MaterialTheme.colorScheme.onBackground + ) + Text( + text = if (uiState.listeningHistoryTotal > 0) { + "${uiState.listeningHistoryTotal} plays" + } else { + "Recently played tracks" + }, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + } + + Text( + text = "Close", + modifier = Modifier + .clip(RoundedCornerShape(8.dp)) + .clickable(onClick = onDismiss) + .padding(horizontal = 10.dp, vertical = 8.dp), + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground + ) + } + + Spacer(modifier = Modifier.height(22.dp)) + + when { + uiState.isListeningHistoryLoading -> ListeningHistoryLoading() + uiState.listeningHistoryError != null -> ListeningHistoryError( + message = uiState.listeningHistoryError, + onRetry = onRetry + ) + uiState.listeningHistory.isEmpty() -> ListeningHistoryEmpty() + else -> ListeningHistoryList(uiState.listeningHistory) + } + } + } +} + +@Composable +private fun ListeningHistoryLoading() { + Box( + modifier = Modifier + .fillMaxWidth() + .height(220.dp), + contentAlignment = Alignment.Center + ) { + CircularProgressIndicator( + color = FurumiNeonPink, + strokeWidth = 2.dp + ) + } +} + +@Composable +private fun ListeningHistoryError( + message: String, + onRetry: () -> Unit +) { + Surface( + modifier = Modifier.fillMaxWidth(), + shape = RoundedCornerShape(12.dp), + color = FurumiSurface, + border = androidx.compose.foundation.BorderStroke(1.dp, FurumiLine) + ) { + Column(modifier = Modifier.padding(16.dp)) { + Text( + text = "Could not load history", + style = MaterialTheme.typography.titleMedium, + color = MaterialTheme.colorScheme.onBackground + ) + Spacer(modifier = Modifier.height(6.dp)) + Text( + text = message, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + Spacer(modifier = Modifier.height(14.dp)) + PillAction( + label = "Retry", + selected = true, + onClick = onRetry + ) + } + } +} + +@Composable +private fun ListeningHistoryEmpty() { + Surface( + modifier = Modifier.fillMaxWidth(), + shape = RoundedCornerShape(12.dp), + color = FurumiSurface, + border = androidx.compose.foundation.BorderStroke(1.dp, FurumiLine) + ) { + Column(modifier = Modifier.padding(16.dp)) { + Text( + text = "No plays yet", + style = MaterialTheme.typography.titleMedium, + color = MaterialTheme.colorScheme.onBackground + ) + Spacer(modifier = Modifier.height(6.dp)) + Text( + text = "Tracks you play will appear here.", + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + } + } +} + +@Composable +private fun ListeningHistoryList(items: List) { + Column(verticalArrangement = Arrangement.spacedBy(14.dp)) { + items.forEach { item -> + ListeningHistoryRow(item) + } + } +} + +@Composable +private fun ListeningHistoryRow(item: ListeningHistoryItem) { + Surface( + modifier = Modifier.fillMaxWidth(), + shape = RoundedCornerShape(10.dp), + color = FurumiSurface, + border = androidx.compose.foundation.BorderStroke(1.dp, FurumiLine) + ) { + Row( + modifier = Modifier.padding(12.dp), + verticalAlignment = Alignment.CenterVertically + ) { + AlbumArtwork( + colors = historyArtworkColors(item.trackId), + modifier = Modifier.size(54.dp), + cornerRadius = 8 + ) + Spacer(modifier = Modifier.width(12.dp)) + Column(modifier = Modifier.weight(1f)) { + Text( + text = item.title, + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Text( + text = item.artists.joinToString(", ").ifBlank { "Unknown artist" }, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Text( + text = item.releaseTitle ?: "Unknown release", + style = MaterialTheme.typography.labelSmall, + color = FurumiTextMuted, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + } + Spacer(modifier = Modifier.width(10.dp)) + Column(horizontalAlignment = Alignment.End) { + Text( + text = formatListenedDuration(item.durationListenedSeconds), + style = MaterialTheme.typography.labelSmall, + color = MaterialTheme.colorScheme.onBackground + ) + Text( + text = if (item.completed) "Completed" else "Partial", + style = MaterialTheme.typography.labelSmall, + color = if (item.completed) FurumiNeonPink else FurumiTextMuted + ) + } + } + } +} + +@Composable +private fun PlaylistTrackRow( + index: Int, + track: MockTrack, + onTrackClick: (MockTrack) -> Unit +) { + Row( + modifier = Modifier + .fillMaxWidth() + .clip(RoundedCornerShape(8.dp)) + .clickable { onTrackClick(track) } + .padding(vertical = 4.dp), + verticalAlignment = Alignment.CenterVertically + ) { + Text( + text = index.toString(), + modifier = Modifier.width(28.dp), + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + Column(modifier = Modifier.weight(1f)) { + Text( + text = track.title, + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Text( + text = track.artist, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + } + Text( + text = track.duration, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + } +} + +@Composable +private fun PlayerHeader( + title: String, + subtitle: String, + onProfileClick: () -> Unit +) { + Row( + modifier = Modifier.fillMaxWidth(), + verticalAlignment = Alignment.CenterVertically, + horizontalArrangement = Arrangement.SpaceBetween + ) { + Column { + Text( + text = title, + style = MaterialTheme.typography.headlineMedium, + color = MaterialTheme.colorScheme.onBackground + ) + Text( + text = subtitle, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + } + + Row( + modifier = Modifier + .height(38.dp) + .clip(CircleShape) + .background(FurumiSurfaceHigh) + .border(1.dp, FurumiLine, CircleShape) + .clickable(onClick = onProfileClick) + .padding(start = 5.dp, end = 12.dp), + verticalAlignment = Alignment.CenterVertically + ) { + Box( + modifier = Modifier + .size(28.dp) + .clip(CircleShape) + .background(FurumiNeonPink), + contentAlignment = Alignment.Center + ) { + Text( + text = "F", + style = MaterialTheme.typography.labelSmall, + color = FurumiBlack, + fontWeight = FontWeight.ExtraBold + ) + } + Spacer(modifier = Modifier.width(8.dp)) + Text( + text = "Profile", + style = MaterialTheme.typography.labelSmall, + color = MaterialTheme.colorScheme.onBackground + ) + } + } +} + +@Composable +private fun ProfileMenu( + uiState: PlayerUiState, + onHistoryClick: () -> Unit, + onLogout: () -> Unit, + modifier: Modifier = Modifier +) { + Surface( + modifier = modifier.width(284.dp), + shape = RoundedCornerShape(12.dp), + color = FurumiSurfaceHigh, + border = androidx.compose.foundation.BorderStroke(1.dp, FurumiLine), + shadowElevation = 8.dp + ) { + Column( + modifier = Modifier.padding(16.dp) + ) { + Text( + text = "Profile", + style = MaterialTheme.typography.titleMedium, + color = MaterialTheme.colorScheme.onBackground + ) + Spacer(modifier = Modifier.height(14.dp)) + Text( + text = uiState.userName, + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Text( + text = uiState.serverUrl.ifBlank { "No server selected" }, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Spacer(modifier = Modifier.height(16.dp)) + OutlinedButton( + onClick = onHistoryClick, + modifier = Modifier + .fillMaxWidth() + .height(46.dp), + shape = CircleShape, + border = androidx.compose.foundation.BorderStroke(1.dp, FurumiLine), + colors = ButtonDefaults.outlinedButtonColors( + contentColor = MaterialTheme.colorScheme.onBackground, + disabledContentColor = FurumiTextMuted + ) + ) { + Text("Listening history") + } + Spacer(modifier = Modifier.height(10.dp)) + Button( + onClick = onLogout, + modifier = Modifier + .fillMaxWidth() + .height(48.dp), + enabled = !uiState.isLoggingOut, + shape = CircleShape, + colors = ButtonDefaults.buttonColors( + containerColor = FurumiNeonPink, + contentColor = FurumiBlack, + disabledContainerColor = FurumiSurface, + disabledContentColor = FurumiTextMuted + ) + ) { + if (uiState.isLoggingOut) { + CircularProgressIndicator( + modifier = Modifier.size(20.dp), + color = FurumiBlack, + strokeWidth = 2.dp + ) + } else { + Text("Log out") + } + } + } + } +} + +@Composable +private fun QuickAccessGrid(onPlaylistClick: (MockPlaylist) -> Unit) { + Column(verticalArrangement = Arrangement.spacedBy(10.dp)) { + Row(horizontalArrangement = Arrangement.spacedBy(10.dp)) { + QuickAccessItem(mockPlaylists[0], Modifier.weight(1f), onPlaylistClick) + QuickAccessItem(mockPlaylists[1], Modifier.weight(1f), onPlaylistClick) + } + Row(horizontalArrangement = Arrangement.spacedBy(10.dp)) { + QuickAccessItem(mockPlaylists[2], Modifier.weight(1f), onPlaylistClick) + QuickAccessItem(mockPlaylists[3], Modifier.weight(1f), onPlaylistClick) + } + } +} + +@Composable +private fun QuickAccessItem( + playlist: MockPlaylist, + modifier: Modifier, + onPlaylistClick: (MockPlaylist) -> Unit +) { + Surface( + modifier = modifier + .height(64.dp) + .clip(RoundedCornerShape(8.dp)) + .clickable { onPlaylistClick(playlist) }, + shape = RoundedCornerShape(8.dp), + color = FurumiSurface + ) { + Row(verticalAlignment = Alignment.CenterVertically) { + AlbumArtwork( + colors = playlist.colors, + modifier = Modifier.size(64.dp), + cornerRadius = 8 + ) + Text( + text = playlist.title, + modifier = Modifier.padding(horizontal = 10.dp), + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground, + maxLines = 2, + overflow = TextOverflow.Ellipsis + ) + } + } +} + +@Composable +private fun SectionTitle(title: String) { + Text( + text = title, + style = MaterialTheme.typography.titleMedium, + color = MaterialTheme.colorScheme.onBackground + ) +} + +@Composable +private fun PlaylistRail( + playlists: List, + onPlaylistClick: (MockPlaylist) -> Unit +) { + Row( + modifier = Modifier.horizontalScroll(rememberScrollState()), + horizontalArrangement = Arrangement.spacedBy(14.dp) + ) { + playlists.forEach { playlist -> + Column( + modifier = Modifier + .width(142.dp) + .clip(RoundedCornerShape(8.dp)) + .clickable { onPlaylistClick(playlist) } + ) { + AlbumArtwork( + colors = playlist.colors, + modifier = Modifier + .size(142.dp) + .clip(RoundedCornerShape(8.dp)), + cornerRadius = 8 + ) + Spacer(modifier = Modifier.height(10.dp)) + Text( + text = playlist.title, + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Text( + text = playlist.subtitle, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + } + } + } +} + +@Composable +private fun TrackList( + tracks: List, + onTrackClick: (MockTrack) -> Unit +) { + Column(verticalArrangement = Arrangement.spacedBy(12.dp)) { + tracks.forEach { track -> + TrackRow(track, onTrackClick) + } + } +} + +@Composable +private fun TrackRow( + track: MockTrack, + onTrackClick: (MockTrack) -> Unit +) { + Row( + modifier = Modifier + .fillMaxWidth() + .clip(RoundedCornerShape(8.dp)) + .clickable { onTrackClick(track) } + .padding(vertical = 2.dp), + verticalAlignment = Alignment.CenterVertically + ) { + AlbumArtwork( + colors = track.colors, + modifier = Modifier.size(48.dp), + cornerRadius = 6 + ) + Spacer(modifier = Modifier.width(12.dp)) + Column(modifier = Modifier.weight(1f)) { + Text( + text = track.title, + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Text( + text = track.artist, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + } + Text( + text = track.duration, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + } +} + +@Composable +private fun FilterChips(filters: List) { + Row( + modifier = Modifier.horizontalScroll(rememberScrollState()), + horizontalArrangement = Arrangement.spacedBy(10.dp) + ) { + filters.forEachIndexed { index, filter -> + Surface( + shape = CircleShape, + color = if (index == 0) FurumiNeonPink else FurumiSurface, + border = if (index == 0) null else androidx.compose.foundation.BorderStroke(1.dp, FurumiLine) + ) { + Text( + text = filter, + modifier = Modifier.padding(horizontal = 14.dp, vertical = 8.dp), + style = MaterialTheme.typography.labelSmall, + color = if (index == 0) FurumiBlack else MaterialTheme.colorScheme.onBackground + ) + } + } + } +} + +@Composable +private fun LibraryRows(onPlaylistClick: (MockPlaylist) -> Unit) { + Column(verticalArrangement = Arrangement.spacedBy(14.dp)) { + mockPlaylists.forEach { playlist -> + Row( + modifier = Modifier + .fillMaxWidth() + .clip(RoundedCornerShape(8.dp)) + .clickable { onPlaylistClick(playlist) }, + verticalAlignment = Alignment.CenterVertically + ) { + AlbumArtwork( + colors = playlist.colors, + modifier = Modifier.size(56.dp), + cornerRadius = 8 + ) + Spacer(modifier = Modifier.width(12.dp)) + Column(modifier = Modifier.weight(1f)) { + Text( + text = playlist.title, + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground + ) + Text( + text = playlist.subtitle, + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + } + } + } + } +} + +@Composable +private fun GenreGrid() { + val genres = listOf( + "Synthwave" to listOf(FurumiNeonPink, FurumiNeonViolet), + "Electronic" to listOf(FurumiElectricCyan, FurumiSurfaceHigh), + "Indie" to listOf(FurumiHotOrange, FurumiNeonPink), + "Focus" to listOf(FurumiNeonViolet, FurumiSurface) + ) + + Column(verticalArrangement = Arrangement.spacedBy(12.dp)) { + genres.chunked(2).forEach { row -> + Row(horizontalArrangement = Arrangement.spacedBy(12.dp)) { + row.forEach { genre -> + GenreTile( + title = genre.first, + colors = genre.second, + modifier = Modifier.weight(1f) + ) + } + } + } + } +} + +@Composable +private fun GenreTile( + title: String, + colors: List, + modifier: Modifier +) { + Box( + modifier = modifier + .height(110.dp) + .clip(RoundedCornerShape(8.dp)) + .background(Brush.linearGradient(colors)) + .border(1.dp, Color.White.copy(alpha = 0.12f), RoundedCornerShape(8.dp)) + .padding(14.dp) + ) { + Text( + text = title, + style = MaterialTheme.typography.titleMedium, + color = Color.White + ) + } +} + +@Composable +private fun PillAction( + label: String, + selected: Boolean, + modifier: Modifier = Modifier, + onClick: () -> Unit +) { + Surface( + modifier = modifier + .height(48.dp) + .clip(CircleShape) + .clickable(onClick = onClick), + shape = CircleShape, + color = if (selected) FurumiNeonPink else FurumiSurface, + border = if (selected) null else androidx.compose.foundation.BorderStroke(1.dp, FurumiLine) + ) { + Box(contentAlignment = Alignment.Center) { + Text( + text = label, + style = MaterialTheme.typography.labelLarge, + color = if (selected) FurumiBlack else MaterialTheme.colorScheme.onBackground + ) + } + } +} + +@Composable +private fun NowPlayingBar( + playback: AudioPlaybackState, + coverBitmap: Bitmap?, + onMediaImageNeeded: (String?) -> Unit, + onClick: () -> Unit, + onPlayPause: () -> Unit +) { + val track = playback.currentTrack ?: return + + LaunchedEffect(track.coverUrl) { + onMediaImageNeeded(track.coverUrl) + } + + Surface( + modifier = Modifier + .fillMaxWidth() + .padding(horizontal = 10.dp, vertical = 6.dp) + .clip(RoundedCornerShape(8.dp)) + .clickable(onClick = onClick), + shape = RoundedCornerShape(8.dp), + color = FurumiSurfaceHigh, + border = androidx.compose.foundation.BorderStroke(1.dp, FurumiLine) + ) { + Column { + Row( + modifier = Modifier.padding(horizontal = 10.dp, vertical = 8.dp), + verticalAlignment = Alignment.CenterVertically + ) { + MediaArtwork( + title = track.title, + seedId = track.id, + bitmap = coverBitmap, + modifier = Modifier.size(42.dp), + cornerRadius = 6 + ) + Spacer(modifier = Modifier.width(10.dp)) + Column(modifier = Modifier.weight(1f)) { + Text( + text = track.title, + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Text( + text = playback.errorMessage ?: trackSubtitle(track), + style = MaterialTheme.typography.bodyMedium, + color = if (playback.errorMessage == null) FurumiTextMuted else FurumiHotOrange, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + } + Box( + modifier = Modifier + .size(38.dp) + .clip(CircleShape) + .background(FurumiNeonPink) + .clickable(onClick = onPlayPause), + contentAlignment = Alignment.Center + ) { + PlaybackGlyph( + playback = playback, + playSize = 16, + pauseSize = 16 + ) + } + } + Box( + modifier = Modifier + .fillMaxWidth() + .height(3.dp) + .background(FurumiLine) + ) { + Box( + modifier = Modifier + .fillMaxWidth(playback.progress) + .fillMaxHeight() + .background(FurumiNeonPink) + ) + } + } + } +} + +@Composable +private fun FullPlayerOverlay( + playback: AudioPlaybackState, + coverBitmap: Bitmap?, + onMediaImageNeeded: (String?) -> Unit, + onDismiss: () -> Unit, + onPlayPause: () -> Unit, + onPrevious: () -> Unit, + onNext: () -> Unit, + onSeekToProgress: (Float) -> Unit, + onQueueTrackClick: (TrackCard) -> Unit +) { + val track = playback.currentTrack ?: return + val upcomingQueue = playback.queue.drop(playback.currentIndex + 1) + + LaunchedEffect(track.coverUrl) { + onMediaImageNeeded(track.coverUrl) + } + + Box( + modifier = Modifier + .fillMaxSize() + .background(MaterialTheme.colorScheme.background) + .windowInsetsPadding(WindowInsets.safeDrawing) + ) { + Column( + modifier = Modifier + .fillMaxSize() + .verticalScroll(rememberScrollState()) + .padding(horizontal = 24.dp, vertical = 18.dp) + ) { + Row( + modifier = Modifier.fillMaxWidth(), + verticalAlignment = Alignment.CenterVertically, + horizontalArrangement = Arrangement.SpaceBetween + ) { + Text( + text = "Now playing", + style = MaterialTheme.typography.labelLarge, + color = FurumiTextMuted + ) + Text( + text = "Done", + modifier = Modifier + .clip(RoundedCornerShape(8.dp)) + .clickable(onClick = onDismiss) + .padding(horizontal = 10.dp, vertical = 8.dp), + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground + ) + } + + Spacer(modifier = Modifier.height(36.dp)) + + MediaArtwork( + title = track.title, + seedId = track.id, + bitmap = coverBitmap, + modifier = Modifier + .fillMaxWidth() + .aspectRatio(1f), + cornerRadius = 12 + ) + + Spacer(modifier = Modifier.height(28.dp)) + + Text( + text = track.title, + style = MaterialTheme.typography.headlineMedium, + color = MaterialTheme.colorScheme.onBackground, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Text( + text = playback.errorMessage ?: trackSubtitle(track), + style = MaterialTheme.typography.bodyLarge, + color = if (playback.errorMessage == null) FurumiTextMuted else FurumiHotOrange, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + + Spacer(modifier = Modifier.height(26.dp)) + + PlayerProgress( + playback = playback, + onSeekToProgress = onSeekToProgress + ) + + Spacer(modifier = Modifier.height(24.dp)) + + Row( + modifier = Modifier.fillMaxWidth(), + horizontalArrangement = Arrangement.SpaceEvenly, + verticalAlignment = Alignment.CenterVertically + ) { + RoundControlButton( + size = 48, + background = FurumiSurface, + onClick = onPrevious + ) { + PreviousGlyph(Modifier.size(20.dp), MaterialTheme.colorScheme.onBackground) + } + RoundControlButton( + size = 72, + background = FurumiNeonPink, + onClick = onPlayPause + ) { + PlaybackGlyph( + playback = playback, + playSize = 24, + pauseSize = 24 + ) + } + RoundControlButton( + size = 48, + background = FurumiSurface, + onClick = onNext + ) { + NextGlyph(Modifier.size(20.dp), MaterialTheme.colorScheme.onBackground) + } + } + + Spacer(modifier = Modifier.height(34.dp)) + + SectionTitle("Up next") + Spacer(modifier = Modifier.height(12.dp)) + if (upcomingQueue.isEmpty()) { + Text( + text = "Queue is empty", + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + } else { + upcomingQueue.take(8).forEachIndexed { index, queueTrack -> + QueueTrackRow( + track = queueTrack, + onClick = { onQueueTrackClick(queueTrack) } + ) + if (index < upcomingQueue.take(8).lastIndex) { + Spacer(modifier = Modifier.height(12.dp)) + } + } + } + } + } +} + +@Composable +private fun QueueTrackRow( + track: TrackCard, + onClick: () -> Unit +) { + Row( + modifier = Modifier + .fillMaxWidth() + .clip(RoundedCornerShape(8.dp)) + .clickable(onClick = onClick) + .padding(vertical = 2.dp), + verticalAlignment = Alignment.CenterVertically + ) { + MediaArtwork( + title = track.title, + seedId = track.id, + bitmap = null, + modifier = Modifier.size(48.dp), + cornerRadius = 6 + ) + Spacer(modifier = Modifier.width(12.dp)) + Column(modifier = Modifier.weight(1f)) { + Text( + text = track.title, + style = MaterialTheme.typography.labelLarge, + color = MaterialTheme.colorScheme.onBackground, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + Text( + text = trackSubtitle(track), + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted, + maxLines = 1, + overflow = TextOverflow.Ellipsis + ) + } + Text( + text = formatDuration(track.durationSeconds), + style = MaterialTheme.typography.bodyMedium, + color = FurumiTextMuted + ) + } +} + +@Composable +private fun PlayerProgress( + playback: AudioPlaybackState, + onSeekToProgress: (Float) -> Unit +) { + Column { + Slider( + value = playback.progress, + onValueChange = onSeekToProgress, + enabled = playback.durationMs > 0L, + colors = SliderDefaults.colors( + thumbColor = FurumiNeonPink, + activeTrackColor = FurumiNeonPink, + inactiveTrackColor = FurumiLine, + disabledThumbColor = FurumiTextMuted, + disabledActiveTrackColor = FurumiLine, + disabledInactiveTrackColor = FurumiLine + ) + ) + Row( + modifier = Modifier.fillMaxWidth(), + horizontalArrangement = Arrangement.SpaceBetween + ) { + Text( + text = formatPlaybackTime(playback.positionMs), + style = MaterialTheme.typography.labelSmall, + color = FurumiTextMuted + ) + Text( + text = formatPlaybackTime(playback.durationMs), + style = MaterialTheme.typography.labelSmall, + color = FurumiTextMuted + ) + } + } +} + +@Composable +private fun RoundControlButton( + size: Int, + background: Color, + onClick: () -> Unit, + content: @Composable BoxScope.() -> Unit +) { + Box( + modifier = Modifier + .size(size.dp) + .clip(CircleShape) + .background(background) + .clickable(onClick = onClick), + contentAlignment = Alignment.Center, + content = content + ) +} + +@Composable +private fun BottomPlayerNav( + selectedTab: PlayerTab, + onTabSelected: (PlayerTab) -> Unit +) { + Surface( + color = FurumiBlack + ) { + Row( + modifier = Modifier + .fillMaxWidth() + .height(66.dp) + .padding(horizontal = 14.dp), + horizontalArrangement = Arrangement.SpaceAround, + verticalAlignment = Alignment.CenterVertically + ) { + PlayerTab.entries.forEach { tab -> + BottomNavItem( + tab = tab, + selected = tab == selectedTab, + onClick = { onTabSelected(tab) } + ) + } + } + } +} + +@Composable +private fun BottomNavItem( + tab: PlayerTab, + selected: Boolean, + onClick: () -> Unit +) { + val color = if (selected) FurumiNeonPink else FurumiTextMuted + Column( + modifier = Modifier + .width(86.dp) + .clip(RoundedCornerShape(8.dp)) + .clickable(onClick = onClick) + .padding(vertical = 6.dp), + horizontalAlignment = Alignment.CenterHorizontally + ) { + when (tab) { + PlayerTab.Global -> GlobalGlyph(Modifier.size(22.dp), color) + PlayerTab.Search -> SearchGlyph(Modifier.size(22.dp), color) + PlayerTab.Library -> LibraryGlyph(Modifier.size(22.dp), color) + } + Spacer(modifier = Modifier.height(4.dp)) + Text( + text = tab.label, + style = MaterialTheme.typography.labelSmall, + color = color + ) + } +} + +@Composable +private fun AlbumArtwork( + colors: List, + modifier: Modifier, + cornerRadius: Int +) { + Box( + modifier = modifier + .clip(RoundedCornerShape(cornerRadius.dp)) + .background(Brush.linearGradient(colors)) + .border(1.dp, Color.White.copy(alpha = 0.10f), RoundedCornerShape(cornerRadius.dp)) + ) { + Canvas(modifier = Modifier.fillMaxSize()) { + drawCircle( + color = Color.White.copy(alpha = 0.16f), + radius = size.minDimension * 0.26f, + center = Offset(size.width * 0.72f, size.height * 0.24f) + ) + drawLine( + color = Color.White.copy(alpha = 0.18f), + start = Offset(size.width * 0.10f, size.height * 0.78f), + end = Offset(size.width * 0.92f, size.height * 0.34f), + strokeWidth = size.minDimension * 0.05f, + cap = StrokeCap.Round + ) + } + } +} + +@Composable +private fun PlayGlyph( + modifier: Modifier, + color: Color +) { + Canvas(modifier = modifier) { + val path = Path().apply { + moveTo(size.width * 0.28f, size.height * 0.18f) + lineTo(size.width * 0.28f, size.height * 0.82f) + lineTo(size.width * 0.82f, size.height * 0.50f) + close() + } + drawPath(path, color) + } +} + +@Composable +private fun PauseGlyph( + modifier: Modifier, + color: Color +) { + Canvas(modifier = modifier) { + val barWidth = size.width * 0.18f + drawRoundRect( + color = color, + topLeft = Offset(size.width * 0.24f, size.height * 0.18f), + size = androidx.compose.ui.geometry.Size(barWidth, size.height * 0.64f) + ) + drawRoundRect( + color = color, + topLeft = Offset(size.width * 0.58f, size.height * 0.18f), + size = androidx.compose.ui.geometry.Size(barWidth, size.height * 0.64f) + ) + } +} + +@Composable +private fun PlaybackGlyph( + playback: AudioPlaybackState, + playSize: Int, + pauseSize: Int +) { + when { + playback.isBuffering -> CircularProgressIndicator( + modifier = Modifier.size(pauseSize.dp), + color = FurumiBlack, + strokeWidth = 2.dp + ) + playback.isPlaying -> PauseGlyph(Modifier.size(pauseSize.dp), FurumiBlack) + else -> PlayGlyph(Modifier.size(playSize.dp), FurumiBlack) + } +} + +@Composable +private fun BackGlyph( + modifier: Modifier, + color: Color +) { + Canvas(modifier = modifier) { + drawLine( + color = color, + start = Offset(size.width * 0.72f, size.height * 0.16f), + end = Offset(size.width * 0.28f, size.height * 0.50f), + strokeWidth = size.minDimension * 0.12f, + cap = StrokeCap.Round + ) + drawLine( + color = color, + start = Offset(size.width * 0.28f, size.height * 0.50f), + end = Offset(size.width * 0.72f, size.height * 0.84f), + strokeWidth = size.minDimension * 0.12f, + cap = StrokeCap.Round + ) + } +} + +@Composable +private fun PreviousGlyph( + modifier: Modifier, + color: Color +) { + Canvas(modifier = modifier) { + val first = Path().apply { + moveTo(size.width * 0.48f, size.height * 0.18f) + lineTo(size.width * 0.48f, size.height * 0.82f) + lineTo(size.width * 0.16f, size.height * 0.50f) + close() + } + val second = Path().apply { + moveTo(size.width * 0.84f, size.height * 0.18f) + lineTo(size.width * 0.84f, size.height * 0.82f) + lineTo(size.width * 0.52f, size.height * 0.50f) + close() + } + drawPath(first, color) + drawPath(second, color) + } +} + +@Composable +private fun NextGlyph( + modifier: Modifier, + color: Color +) { + Canvas(modifier = modifier) { + val first = Path().apply { + moveTo(size.width * 0.16f, size.height * 0.18f) + lineTo(size.width * 0.16f, size.height * 0.82f) + lineTo(size.width * 0.48f, size.height * 0.50f) + close() + } + val second = Path().apply { + moveTo(size.width * 0.52f, size.height * 0.18f) + lineTo(size.width * 0.52f, size.height * 0.82f) + lineTo(size.width * 0.84f, size.height * 0.50f) + close() + } + drawPath(first, color) + drawPath(second, color) + } +} + +@Composable +private fun GlobalGlyph( + modifier: Modifier, + color: Color +) { + Canvas(modifier = modifier) { + val strokeWidth = size.minDimension * 0.09f + drawCircle( + color = color, + radius = size.minDimension * 0.38f, + center = Offset(size.width * 0.50f, size.height * 0.50f), + style = Stroke(width = strokeWidth) + ) + drawLine( + color = color, + start = Offset(size.width * 0.50f, size.height * 0.16f), + end = Offset(size.width * 0.50f, size.height * 0.84f), + strokeWidth = strokeWidth, + cap = StrokeCap.Round + ) + drawLine( + color = color, + start = Offset(size.width * 0.16f, size.height * 0.50f), + end = Offset(size.width * 0.84f, size.height * 0.50f), + strokeWidth = strokeWidth, + cap = StrokeCap.Round + ) + drawLine( + color = color.copy(alpha = 0.74f), + start = Offset(size.width * 0.24f, size.height * 0.34f), + end = Offset(size.width * 0.76f, size.height * 0.34f), + strokeWidth = strokeWidth * 0.78f, + cap = StrokeCap.Round + ) + drawLine( + color = color.copy(alpha = 0.74f), + start = Offset(size.width * 0.24f, size.height * 0.66f), + end = Offset(size.width * 0.76f, size.height * 0.66f), + strokeWidth = strokeWidth * 0.78f, + cap = StrokeCap.Round + ) + } +} + +@Composable +private fun SearchGlyph( + modifier: Modifier, + color: Color +) { + Canvas(modifier = modifier) { + drawCircle( + color = color, + radius = size.minDimension * 0.30f, + center = Offset(size.width * 0.42f, size.height * 0.42f), + style = Stroke(width = size.minDimension * 0.10f) + ) + drawLine( + color = color, + start = Offset(size.width * 0.64f, size.height * 0.64f), + end = Offset(size.width * 0.88f, size.height * 0.88f), + strokeWidth = size.minDimension * 0.10f, + cap = StrokeCap.Round + ) + } +} + +@Composable +private fun LibraryGlyph( + modifier: Modifier, + color: Color +) { + Canvas(modifier = modifier) { + val strokeWidth = size.minDimension * 0.10f + drawLine( + color = color, + start = Offset(size.width * 0.18f, size.height * 0.16f), + end = Offset(size.width * 0.18f, size.height * 0.86f), + strokeWidth = strokeWidth, + cap = StrokeCap.Round + ) + drawLine( + color = color, + start = Offset(size.width * 0.46f, size.height * 0.16f), + end = Offset(size.width * 0.46f, size.height * 0.86f), + strokeWidth = strokeWidth, + cap = StrokeCap.Round + ) + drawLine( + color = color, + start = Offset(size.width * 0.70f, size.height * 0.22f), + end = Offset(size.width * 0.86f, size.height * 0.82f), + strokeWidth = strokeWidth, + cap = StrokeCap.Round + ) + } +} + +private fun artistsSubtitle(uiState: PlayerUiState): String { + return if (uiState.globalArtistsTotal > 0) { + pluralCount(uiState.globalArtistsTotal, "artist") + } else { + "Artists on this server" + } +} + +private fun pluralCount(count: Long, singular: String): String { + val label = if (count == 1L) singular else "${singular}s" + return "$count $label" +} + +private fun pluralCount(count: Int, singular: String): String { + return pluralCount(count.toLong(), singular) +} + +private fun formatListenedDuration(seconds: Int?): String { + if (seconds == null || seconds <= 0) return "--" + val minutes = seconds / 60 + val remainingSeconds = seconds % 60 + return "$minutes:${remainingSeconds.toString().padStart(2, '0')}" +} + +private fun formatDuration(seconds: Int?): String { + if (seconds == null || seconds <= 0) return "--" + val minutes = seconds / 60 + val remainingSeconds = seconds % 60 + return "$minutes:${remainingSeconds.toString().padStart(2, '0')}" +} + +private fun formatPlaybackTime(milliseconds: Long): String { + if (milliseconds <= 0L) return "0:00" + val totalSeconds = milliseconds / 1_000L + val minutes = totalSeconds / 60L + val seconds = totalSeconds % 60L + return "$minutes:${seconds.toString().padStart(2, '0')}" +} + +private fun trackSubtitle(track: TrackCard): String { + return track.artists.joinToString(", ") + .ifBlank { track.releaseTitle.orEmpty() } + .ifBlank { "Unknown artist" } +} + +private fun releaseMeta(release: ReleaseCard): String { + val parts = buildList { + release.year?.let { add(it.toString()) } + release.releaseType + ?.replace('_', ' ') + ?.takeIf { it.isNotBlank() } + ?.let { add(it) } + add(pluralCount(release.trackCount, "track")) + } + + return parts.joinToString(" / ") +} + +private fun releaseArtistsText(artists: List): String { + return artists.joinToString(", ") { it.name }.ifBlank { "Unknown artist" } +} + +private fun artistArtworkColors(artistId: Long): List { + val palettes = listOf( + listOf(FurumiNeonPink, FurumiNeonViolet, FurumiBlack), + listOf(FurumiElectricCyan, FurumiSurfaceHigh), + listOf(FurumiHotOrange, FurumiNeonPink), + listOf(FurumiNeonViolet, FurumiSurface), + listOf(FurumiSurfaceHigh, FurumiNeonPink) + ) + return palettes[(artistId % palettes.size).toInt()] +} + +private fun historyArtworkColors(trackId: Long): List { + val palettes = listOf( + listOf(FurumiNeonPink, FurumiNeonViolet, FurumiBlack), + listOf(FurumiElectricCyan, FurumiSurfaceHigh), + listOf(FurumiHotOrange, FurumiNeonPink), + listOf(FurumiNeonViolet, FurumiSurface) + ) + return palettes[(trackId % palettes.size).toInt()] +} + +private val mockTracks = listOf( + MockTrack( + title = "Neon Drift", + artist = "Vanta Sequence", + duration = "3:42", + colors = listOf(FurumiNeonPink, FurumiNeonViolet, FurumiBlack) + ), + MockTrack( + title = "Glass City", + artist = "Afterimage", + duration = "4:08", + colors = listOf(FurumiElectricCyan, FurumiSurfaceHigh) + ), + MockTrack( + title = "Low Orbit", + artist = "Signal Room", + duration = "2:58", + colors = listOf(FurumiHotOrange, FurumiNeonPink) + ), + MockTrack( + title = "Night Cache", + artist = "Mirror Index", + duration = "5:16", + colors = listOf(FurumiNeonViolet, FurumiSurface) + ), + MockTrack( + title = "Pink Voltage", + artist = "Kairo Unit", + duration = "3:19", + colors = listOf(FurumiNeonPink, FurumiHotOrange) + ), + MockTrack( + title = "After Hours Relay", + artist = "North Terminal", + duration = "4:31", + colors = listOf(FurumiElectricCyan, FurumiNeonViolet) + ), + MockTrack( + title = "Static Bloom", + artist = "Velvet Protocol", + duration = "3:07", + colors = listOf(FurumiSurfaceHigh, FurumiNeonPink) + ), + MockTrack( + title = "Server Room Lullaby", + artist = "Echo Archive", + duration = "5:04", + colors = listOf(FurumiNeonViolet, FurumiBlack) + ) +) + +private val mockPlaylists = listOf( + MockPlaylist( + title = "Daily Pulse", + subtitle = "Playlist", + description = "A fast pass through the tracks that fit today.", + colors = listOf(FurumiNeonPink, FurumiNeonViolet), + tracks = mockTracks + ), + MockPlaylist( + title = "Server Favorites", + subtitle = "74 tracks", + description = "Most replayed music from this Furumi server.", + colors = listOf(FurumiElectricCyan, FurumiSurfaceHigh), + tracks = mockTracks.drop(1) + mockTracks.take(1) + ), + MockPlaylist( + title = "Late Drive", + subtitle = "Playlist", + description = "Night-road energy with neon edges.", + colors = listOf(FurumiHotOrange, FurumiNeonPink), + tracks = mockTracks.drop(2) + mockTracks.take(2) + ), + MockPlaylist( + title = "Albums to Replay", + subtitle = "Collection", + description = "Albums and long listens worth returning to.", + colors = listOf(FurumiNeonViolet, FurumiSurface), + tracks = mockTracks.drop(3) + mockTracks.take(3) + ), + MockPlaylist( + title = "Focus Cache", + subtitle = "32 tracks", + description = "Lower-friction tracks for coding and reading.", + colors = listOf(FurumiSurfaceHigh, FurumiElectricCyan), + tracks = mockTracks.drop(4) + mockTracks.take(4) + ) +) diff --git a/app/src/main/java/com/example/furumi_android/ui/player/PlayerViewModel.kt b/app/src/main/java/com/example/furumi_android/ui/player/PlayerViewModel.kt new file mode 100644 index 0000000..fe7e26d --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/ui/player/PlayerViewModel.kt @@ -0,0 +1,411 @@ +package com.example.furumi_android.ui.player + +import android.graphics.Bitmap +import androidx.lifecycle.ViewModel +import androidx.lifecycle.viewModelScope +import com.example.furumi_android.data.repository.MediaImageLoader +import com.example.furumi_android.domain.model.ArtistCard +import com.example.furumi_android.domain.model.ArtistDetail +import com.example.furumi_android.domain.model.ListeningHistoryItem +import com.example.furumi_android.domain.model.ReleaseCard +import com.example.furumi_android.domain.model.ReleaseDetail +import com.example.furumi_android.domain.model.TrackCard +import com.example.furumi_android.domain.repository.AuthRepository +import com.example.furumi_android.domain.repository.PlayerRepository +import com.example.furumi_android.playback.AudioPlaybackState +import com.example.furumi_android.playback.PlaybackController +import dagger.hilt.android.lifecycle.HiltViewModel +import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.StateFlow +import kotlinx.coroutines.flow.asStateFlow +import kotlinx.coroutines.launch +import javax.inject.Inject + +data class PlayerUiState( + val userName: String = "Listener", + val serverUrl: String = "", + val isLoggingOut: Boolean = false, + val isLoggedOut: Boolean = false, + val isListeningHistoryVisible: Boolean = false, + val isListeningHistoryLoading: Boolean = false, + val listeningHistory: List = emptyList(), + val listeningHistoryTotal: Long = 0, + val listeningHistoryError: String? = null, + val globalArtists: List = emptyList(), + val globalArtistsTotal: Long = 0, + val globalArtistsPage: Int = 0, + val globalArtistsHasMore: Boolean = true, + val isGlobalArtistsLoading: Boolean = false, + val isGlobalArtistsLoadingMore: Boolean = false, + val globalArtistsError: String? = null, + val selectedArtist: ArtistCard? = null, + val artistDetail: ArtistDetail? = null, + val isArtistDetailLoading: Boolean = false, + val artistDetailError: String? = null, + val selectedRelease: ReleaseCard? = null, + val releaseDetail: ReleaseDetail? = null, + val isReleaseDetailLoading: Boolean = false, + val releaseDetailError: String? = null, + val mediaImages: Map = emptyMap(), + val loadingMediaImageUrls: Set = emptySet(), + val playback: AudioPlaybackState = AudioPlaybackState() +) + +@HiltViewModel +class PlayerViewModel @Inject constructor( + private val authRepository: AuthRepository, + private val playerRepository: PlayerRepository, + private val mediaImageLoader: MediaImageLoader, + private val playbackController: PlaybackController +) : ViewModel() { + + private val artistDetailCache = mutableMapOf() + private val releaseDetailCache = mutableMapOf() + + private val _uiState = MutableStateFlow( + authRepository.getCurrentSession()?.let { session -> + PlayerUiState( + userName = session.user.name, + serverUrl = session.serverBaseUrl + ) + } ?: PlayerUiState( + serverUrl = authRepository.getSavedServerUrl().orEmpty() + ) + ) + val uiState: StateFlow = _uiState.asStateFlow() + + init { + loadGlobalArtistsIfNeeded() + observePlayback() + } + + fun loadGlobalArtistsIfNeeded() { + val state = _uiState.value + if (state.globalArtists.isNotEmpty() || + state.isGlobalArtistsLoading || + state.isGlobalArtistsLoadingMore + ) { + return + } + + loadGlobalArtists(page = 1, append = false) + } + + fun retryGlobalArtists() { + loadGlobalArtists(page = 1, append = false) + } + + fun loadMoreGlobalArtists() { + val state = _uiState.value + if (!state.globalArtistsHasMore || + state.isGlobalArtistsLoading || + state.isGlobalArtistsLoadingMore + ) { + return + } + + loadGlobalArtists(page = state.globalArtistsPage + 1, append = true) + } + + fun openArtist(artist: ArtistCard) { + val cachedDetail = artistDetailCache[artist.id] + + _uiState.value = _uiState.value.copy( + selectedArtist = cachedDetail?.artist ?: artist, + artistDetail = cachedDetail, + isArtistDetailLoading = cachedDetail == null, + artistDetailError = null + ) + + if (cachedDetail == null) { + loadArtistDetail(artist.id) + } + } + + fun closeArtistDetail() { + _uiState.value = _uiState.value.copy( + selectedArtist = null, + artistDetail = null, + isArtistDetailLoading = false, + artistDetailError = null, + selectedRelease = null, + releaseDetail = null, + isReleaseDetailLoading = false, + releaseDetailError = null + ) + } + + fun retryArtistDetail() { + val artist = _uiState.value.selectedArtist ?: return + loadArtistDetail(artist.id) + } + + fun openRelease(release: ReleaseCard) { + val cachedDetail = releaseDetailCache[release.id] + + _uiState.value = _uiState.value.copy( + selectedRelease = cachedDetail?.release ?: release, + releaseDetail = cachedDetail, + isReleaseDetailLoading = cachedDetail == null, + releaseDetailError = null + ) + + if (cachedDetail == null) { + loadReleaseDetail(release.id) + } + } + + fun closeReleaseDetail() { + _uiState.value = _uiState.value.copy( + selectedRelease = null, + releaseDetail = null, + isReleaseDetailLoading = false, + releaseDetailError = null + ) + } + + fun retryReleaseDetail() { + val release = _uiState.value.selectedRelease ?: return + loadReleaseDetail(release.id) + } + + fun playTrack(track: TrackCard, queue: List): Boolean { + if (track.streamUrl.isBlank()) return false + + playbackController.playTrack(track, queue) + return true + } + + fun playArtistTracks(shuffle: Boolean = false): Boolean { + val detail = _uiState.value.artistDetail ?: return false + val tracks = detail.topTracks + .ifEmpty { detail.featuredTracks } + .filter { it.streamUrl.isNotBlank() } + .let { if (shuffle) it.shuffled() else it } + + if (tracks.isEmpty()) return false + + playbackController.playQueue(tracks) + return true + } + + fun playReleaseTracks(shuffle: Boolean = false): Boolean { + val tracks = _uiState.value.releaseDetail + ?.tracks + .orEmpty() + .filter { it.streamUrl.isNotBlank() } + .let { if (shuffle) it.shuffled() else it } + + if (tracks.isEmpty()) return false + + playbackController.playQueue(tracks) + return true + } + + fun togglePlayPause() { + playbackController.togglePlayPause() + } + + fun nextTrack() { + playbackController.next() + } + + fun previousTrack() { + playbackController.previous() + } + + fun seekToPlaybackProgress(progress: Float) { + playbackController.seekToProgress(progress) + } + + fun loadMediaImage(url: String?) { + val imageUrl = url?.takeIf { it.isNotBlank() } ?: return + val state = _uiState.value + + if (state.mediaImages.containsKey(imageUrl) || + state.loadingMediaImageUrls.contains(imageUrl) + ) { + return + } + + _uiState.value = state.copy( + loadingMediaImageUrls = state.loadingMediaImageUrls + imageUrl + ) + + viewModelScope.launch { + mediaImageLoader.load(imageUrl) + .onSuccess { bitmap -> + val currentState = _uiState.value + _uiState.value = currentState.copy( + mediaImages = currentState.mediaImages + (imageUrl to bitmap) + ) + } + + val currentState = _uiState.value + _uiState.value = currentState.copy( + loadingMediaImageUrls = currentState.loadingMediaImageUrls - imageUrl + ) + } + } + + fun openListeningHistory() { + _uiState.value = _uiState.value.copy(isListeningHistoryVisible = true) + + if (_uiState.value.listeningHistory.isEmpty() && !_uiState.value.isListeningHistoryLoading) { + loadListeningHistory() + } + } + + fun closeListeningHistory() { + _uiState.value = _uiState.value.copy(isListeningHistoryVisible = false) + } + + fun retryListeningHistory() { + loadListeningHistory() + } + + fun logout() { + if (_uiState.value.isLoggingOut) return + + viewModelScope.launch { + _uiState.value = _uiState.value.copy(isLoggingOut = true) + authRepository.logout() + _uiState.value = _uiState.value.copy( + isLoggingOut = false, + isLoggedOut = true + ) + } + } + + private fun loadListeningHistory() { + viewModelScope.launch { + _uiState.value = _uiState.value.copy( + isListeningHistoryLoading = true, + listeningHistoryError = null + ) + + playerRepository.getListeningHistory(page = 1, limit = 20) + .onSuccess { page -> + _uiState.value = _uiState.value.copy( + isListeningHistoryLoading = false, + listeningHistory = page.items, + listeningHistoryTotal = page.total, + listeningHistoryError = null + ) + } + .onFailure { error -> + _uiState.value = _uiState.value.copy( + isListeningHistoryLoading = false, + listeningHistoryError = error.message ?: "Unable to load listening history" + ) + } + } + } + + private fun observePlayback() { + viewModelScope.launch { + playbackController.state.collect { playbackState -> + _uiState.value = _uiState.value.copy(playback = playbackState) + } + } + } + + private fun loadGlobalArtists(page: Int, append: Boolean) { + viewModelScope.launch { + val state = _uiState.value + _uiState.value = if (append) { + state.copy( + isGlobalArtistsLoadingMore = true, + globalArtistsError = null + ) + } else { + state.copy( + isGlobalArtistsLoading = true, + isGlobalArtistsLoadingMore = false, + globalArtistsError = null, + globalArtistsHasMore = true + ) + } + + playerRepository.getArtists(page = page, limit = ARTIST_PAGE_SIZE, mine = false) + .onSuccess { artistPage -> + val currentArtists = if (append) _uiState.value.globalArtists else emptyList() + val mergedArtists = (currentArtists + artistPage.items).distinctBy { it.id } + + _uiState.value = _uiState.value.copy( + globalArtists = mergedArtists, + globalArtistsTotal = artistPage.total, + globalArtistsPage = artistPage.page, + globalArtistsHasMore = artistPage.hasMore, + isGlobalArtistsLoading = false, + isGlobalArtistsLoadingMore = false, + globalArtistsError = null + ) + } + .onFailure { error -> + _uiState.value = _uiState.value.copy( + isGlobalArtistsLoading = false, + isGlobalArtistsLoadingMore = false, + globalArtistsError = error.message ?: "Unable to load artists" + ) + } + } + } + + private fun loadArtistDetail(artistId: Long) { + viewModelScope.launch { + _uiState.value = _uiState.value.copy( + isArtistDetailLoading = true, + artistDetailError = null + ) + + playerRepository.getArtistDetail(artistId) + .onSuccess { detail -> + artistDetailCache[artistId] = detail + _uiState.value = _uiState.value.copy( + selectedArtist = detail.artist, + artistDetail = detail, + isArtistDetailLoading = false, + artistDetailError = null + ) + } + .onFailure { error -> + _uiState.value = _uiState.value.copy( + isArtistDetailLoading = false, + artistDetailError = error.message ?: "Unable to load artist" + ) + } + } + } + + private fun loadReleaseDetail(releaseId: Long) { + viewModelScope.launch { + _uiState.value = _uiState.value.copy( + isReleaseDetailLoading = true, + releaseDetailError = null + ) + + playerRepository.getReleaseDetail(releaseId) + .onSuccess { detail -> + releaseDetailCache[releaseId] = detail + _uiState.value = _uiState.value.copy( + selectedRelease = detail.release, + releaseDetail = detail, + isReleaseDetailLoading = false, + releaseDetailError = null + ) + } + .onFailure { error -> + _uiState.value = _uiState.value.copy( + isReleaseDetailLoading = false, + releaseDetailError = error.message ?: "Unable to load release" + ) + } + } + } + + private companion object { + const val ARTIST_PAGE_SIZE = 60 + } + +} diff --git a/app/src/main/java/com/example/furumi_android/ui/profile/ProfileScreen.kt b/app/src/main/java/com/example/furumi_android/ui/profile/ProfileScreen.kt new file mode 100644 index 0000000..3fd449a --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/ui/profile/ProfileScreen.kt @@ -0,0 +1,51 @@ +package com.example.furumi_android.ui.profile + +import androidx.compose.foundation.layout.* +import androidx.compose.material3.* +import androidx.compose.runtime.* +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.unit.dp +import androidx.hilt.navigation.compose.hiltViewModel + +@Composable +fun ProfileScreen( + viewModel: ProfileViewModel = hiltViewModel(), + onLoggedOut: () -> Unit +) { + val uiState by viewModel.uiState.collectAsState() + + LaunchedEffect(uiState.isLoggedOut) { + if (uiState.isLoggedOut) { + onLoggedOut() + } + } + + Column( + modifier = Modifier + .fillMaxSize() + .padding(16.dp), + horizontalAlignment = Alignment.CenterHorizontally, + verticalArrangement = Arrangement.Center + ) { + Text(text = "Profile", style = MaterialTheme.typography.headlineMedium) + + Spacer(modifier = Modifier.height(32.dp)) + + uiState.user?.let { user -> + Text(text = "Name: ${user.name}", style = MaterialTheme.typography.bodyLarge) + Text(text = "ID: ${user.id}", style = MaterialTheme.typography.bodyMedium) + Text(text = "Role: ${user.role}", style = MaterialTheme.typography.bodyMedium) + } ?: Text(text = "Loading user info...") + + Spacer(modifier = Modifier.height(48.dp)) + + Button( + onClick = { viewModel.logout() }, + modifier = Modifier.fillMaxWidth(), + colors = ButtonDefaults.buttonColors(containerColor = MaterialTheme.colorScheme.error) + ) { + Text("Logout") + } + } +} diff --git a/app/src/main/java/com/example/furumi_android/ui/profile/ProfileViewModel.kt b/app/src/main/java/com/example/furumi_android/ui/profile/ProfileViewModel.kt new file mode 100644 index 0000000..9126a8e --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/ui/profile/ProfileViewModel.kt @@ -0,0 +1,37 @@ +package com.example.furumi_android.ui.profile + +import androidx.lifecycle.ViewModel +import androidx.lifecycle.viewModelScope +import com.example.furumi_android.domain.model.User +import com.example.furumi_android.domain.repository.AuthRepository +import dagger.hilt.android.lifecycle.HiltViewModel +import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.StateFlow +import kotlinx.coroutines.flow.asStateFlow +import kotlinx.coroutines.launch +import javax.inject.Inject + +data class ProfileUiState( + val user: User? = null, + val isLoggedOut: Boolean = false +) + +@HiltViewModel +class ProfileViewModel @Inject constructor( + private val authRepository: AuthRepository +) : ViewModel() { + + private val _uiState = MutableStateFlow(ProfileUiState()) + val uiState: StateFlow = _uiState.asStateFlow() + + init { + _uiState.value = _uiState.value.copy(user = authRepository.getCurrentUser()) + } + + fun logout() { + viewModelScope.launch { + authRepository.logout() + _uiState.value = _uiState.value.copy(isLoggedOut = true) + } + } +} diff --git a/app/src/main/java/com/example/furumi_android/ui/theme/Color.kt b/app/src/main/java/com/example/furumi_android/ui/theme/Color.kt new file mode 100644 index 0000000..7649325 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/ui/theme/Color.kt @@ -0,0 +1,16 @@ +package com.example.furumi_android.ui.theme + +import androidx.compose.ui.graphics.Color + +val FurumiBlack = Color(0xFF080A0E) +val FurumiInk = Color(0xFF111019) +val FurumiSurface = Color(0xFF1A1524) +val FurumiSurfaceHigh = Color(0xFF2A2036) +val FurumiLine = Color(0xFF4A385D) +val FurumiText = Color(0xFFFFF7FB) +val FurumiTextMuted = Color(0xFFC9B8D6) +val FurumiNeonPink = Color(0xFFFF3EA5) +val FurumiNeonPinkPressed = Color(0xFFE12A8E) +val FurumiElectricCyan = Color(0xFF37F3FF) +val FurumiNeonViolet = Color(0xFFA855F7) +val FurumiHotOrange = Color(0xFFFF7A3D) diff --git a/app/src/main/java/com/example/furumi_android/ui/theme/Theme.kt b/app/src/main/java/com/example/furumi_android/ui/theme/Theme.kt new file mode 100644 index 0000000..88596df --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/ui/theme/Theme.kt @@ -0,0 +1,54 @@ +package com.example.furumi_android.ui.theme + +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.darkColorScheme +import androidx.compose.material3.lightColorScheme +import androidx.compose.runtime.Composable +import androidx.compose.ui.graphics.Color + +private val DarkColorScheme = darkColorScheme( + primary = FurumiNeonPink, + onPrimary = FurumiBlack, + primaryContainer = FurumiNeonPinkPressed, + onPrimaryContainer = FurumiBlack, + secondary = FurumiElectricCyan, + onSecondary = FurumiBlack, + tertiary = FurumiNeonViolet, + background = FurumiBlack, + onBackground = FurumiText, + surface = FurumiInk, + onSurface = FurumiText, + surfaceVariant = FurumiSurface, + onSurfaceVariant = FurumiTextMuted, + outline = FurumiLine, + error = FurumiHotOrange, + onError = Color.White +) + +private val LightColorScheme = lightColorScheme( + primary = Color(0xFFB90068), + onPrimary = Color.White, + secondary = Color(0xFF006B73), + tertiary = Color(0xFF6D27B8), + background = Color(0xFFFFF7FB), + onBackground = Color(0xFF17101F), + surface = Color.White, + onSurface = Color(0xFF17101F), + surfaceVariant = Color(0xFFF2E6F6), + onSurfaceVariant = Color(0xFF67536F), + outline = Color(0xFFD8BFE2) +) + +@Composable +fun FurumiandroidTheme( + darkTheme: Boolean = true, + content: @Composable () -> Unit +) { + val colorScheme = if (darkTheme) DarkColorScheme else LightColorScheme + + MaterialTheme( + colorScheme = colorScheme, + typography = Typography, + content = content + ) +} diff --git a/app/src/main/java/com/example/furumi_android/ui/theme/Type.kt b/app/src/main/java/com/example/furumi_android/ui/theme/Type.kt new file mode 100644 index 0000000..46711d9 --- /dev/null +++ b/app/src/main/java/com/example/furumi_android/ui/theme/Type.kt @@ -0,0 +1,59 @@ +package com.example.furumi_android.ui.theme + +import androidx.compose.material3.Typography +import androidx.compose.ui.text.TextStyle +import androidx.compose.ui.text.font.FontFamily +import androidx.compose.ui.text.font.FontWeight +import androidx.compose.ui.unit.sp + +val Typography = Typography( + headlineLarge = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.ExtraBold, + fontSize = 34.sp, + lineHeight = 40.sp, + letterSpacing = 0.sp + ), + headlineMedium = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.ExtraBold, + fontSize = 28.sp, + lineHeight = 34.sp, + letterSpacing = 0.sp + ), + titleMedium = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.SemiBold, + fontSize = 18.sp, + lineHeight = 24.sp, + letterSpacing = 0.sp + ), + bodyLarge = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Normal, + fontSize = 16.sp, + lineHeight = 24.sp, + letterSpacing = 0.sp + ), + bodyMedium = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Normal, + fontSize = 14.sp, + lineHeight = 20.sp, + letterSpacing = 0.sp + ), + labelLarge = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Bold, + fontSize = 15.sp, + lineHeight = 20.sp, + letterSpacing = 0.sp + ), + labelSmall = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.SemiBold, + fontSize = 12.sp, + lineHeight = 16.sp, + letterSpacing = 0.sp + ) +) diff --git a/app/src/main/keepRules/rules.keep b/app/src/main/keepRules/rules.keep new file mode 100644 index 0000000..d7e081a --- /dev/null +++ b/app/src/main/keepRules/rules.keep @@ -0,0 +1,12 @@ +# Add project specific R8 rules here. +# AGP will combine all keep rule files in src/main/keepRules to pass to R8 +# +# For more details, see +# https://d.android.com/r/tools/r8/keep-rules + +# If your project uses WebView with JS, uncomment the following +# and specify the fully qualified class name to the JavaScript interface +# class: +#-keepclassmembers class fqcn.of.javascript.interface.for.webview { +# public *; +#} \ No newline at end of file diff --git a/app/src/main/res/drawable/ic_launcher_background.xml b/app/src/main/res/drawable/ic_launcher_background.xml new file mode 100644 index 0000000..07d5da9 --- /dev/null +++ b/app/src/main/res/drawable/ic_launcher_background.xml @@ -0,0 +1,170 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/app/src/main/res/drawable/ic_launcher_foreground.xml b/app/src/main/res/drawable/ic_launcher_foreground.xml new file mode 100644 index 0000000..2b068d1 --- /dev/null +++ b/app/src/main/res/drawable/ic_launcher_foreground.xml @@ -0,0 +1,30 @@ + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml b/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml new file mode 100644 index 0000000..6f3b755 --- /dev/null +++ b/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml b/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml new file mode 100644 index 0000000..6f3b755 --- /dev/null +++ b/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/app/src/main/res/mipmap-hdpi/ic_launcher.webp b/app/src/main/res/mipmap-hdpi/ic_launcher.webp new file mode 100644 index 0000000..c209e78 Binary files /dev/null and b/app/src/main/res/mipmap-hdpi/ic_launcher.webp differ diff --git a/app/src/main/res/mipmap-hdpi/ic_launcher_round.webp b/app/src/main/res/mipmap-hdpi/ic_launcher_round.webp new file mode 100644 index 0000000..b2dfe3d Binary files /dev/null and b/app/src/main/res/mipmap-hdpi/ic_launcher_round.webp differ diff --git a/app/src/main/res/mipmap-mdpi/ic_launcher.webp b/app/src/main/res/mipmap-mdpi/ic_launcher.webp new file mode 100644 index 0000000..4f0f1d6 Binary files /dev/null and b/app/src/main/res/mipmap-mdpi/ic_launcher.webp differ diff --git a/app/src/main/res/mipmap-mdpi/ic_launcher_round.webp b/app/src/main/res/mipmap-mdpi/ic_launcher_round.webp new file mode 100644 index 0000000..62b611d Binary files /dev/null and b/app/src/main/res/mipmap-mdpi/ic_launcher_round.webp differ diff --git a/app/src/main/res/mipmap-xhdpi/ic_launcher.webp b/app/src/main/res/mipmap-xhdpi/ic_launcher.webp new file mode 100644 index 0000000..948a307 Binary files /dev/null and b/app/src/main/res/mipmap-xhdpi/ic_launcher.webp differ diff --git a/app/src/main/res/mipmap-xhdpi/ic_launcher_round.webp b/app/src/main/res/mipmap-xhdpi/ic_launcher_round.webp new file mode 100644 index 0000000..1b9a695 Binary files /dev/null and b/app/src/main/res/mipmap-xhdpi/ic_launcher_round.webp differ diff --git a/app/src/main/res/mipmap-xxhdpi/ic_launcher.webp b/app/src/main/res/mipmap-xxhdpi/ic_launcher.webp new file mode 100644 index 0000000..28d4b77 Binary files /dev/null and b/app/src/main/res/mipmap-xxhdpi/ic_launcher.webp differ diff --git a/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.webp b/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.webp new file mode 100644 index 0000000..9287f50 Binary files /dev/null and b/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.webp differ diff --git a/app/src/main/res/mipmap-xxxhdpi/ic_launcher.webp b/app/src/main/res/mipmap-xxxhdpi/ic_launcher.webp new file mode 100644 index 0000000..aa7d642 Binary files /dev/null and b/app/src/main/res/mipmap-xxxhdpi/ic_launcher.webp differ diff --git a/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.webp b/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.webp new file mode 100644 index 0000000..9126ae3 Binary files /dev/null and b/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.webp differ diff --git a/app/src/main/res/values/colors.xml b/app/src/main/res/values/colors.xml new file mode 100644 index 0000000..f8c6127 --- /dev/null +++ b/app/src/main/res/values/colors.xml @@ -0,0 +1,10 @@ + + + #FFBB86FC + #FF6200EE + #FF3700B3 + #FF03DAC5 + #FF018786 + #FF000000 + #FFFFFFFF + \ No newline at end of file diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml new file mode 100644 index 0000000..a636bf5 --- /dev/null +++ b/app/src/main/res/values/strings.xml @@ -0,0 +1,3 @@ + + furumi-android + \ No newline at end of file diff --git a/app/src/main/res/values/themes.xml b/app/src/main/res/values/themes.xml new file mode 100644 index 0000000..a5d1732 --- /dev/null +++ b/app/src/main/res/values/themes.xml @@ -0,0 +1,5 @@ + + + +