From 00f6c1d3493396b9874473595da8bbded6fbec4c Mon Sep 17 00:00:00 2001 From: Martin Felber Date: Fri, 27 Dec 2024 20:54:45 +0100 Subject: [PATCH 1/7] Use tum artemis server url as default for unrestricted build flavor --- .../src/main/kotlin/commonConfiguration/KotlinAndroid.kt | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/build-logic/convention/src/main/kotlin/commonConfiguration/KotlinAndroid.kt b/build-logic/convention/src/main/kotlin/commonConfiguration/KotlinAndroid.kt index eafc2d203..40f40ba46 100644 --- a/build-logic/convention/src/main/kotlin/commonConfiguration/KotlinAndroid.kt +++ b/build-logic/convention/src/main/kotlin/commonConfiguration/KotlinAndroid.kt @@ -134,6 +134,8 @@ internal fun Project.configureReleaseTypeFlavors( } } +private const val TUM_ARTEMIS_SERVER_URL = "https://artemis.cit.tum.de" + internal fun Project.configureInstanceSelectionFlavors( commonExtension: CommonExtension<*, *, *, *, *, *>, ) { @@ -153,7 +155,7 @@ internal fun Project.configureInstanceSelectionFlavors( buildConfigField( "String", ProductFlavors.BuildConfigFields.DefaultServerUrl, - "\"\"" + "\"$TUM_ARTEMIS_SERVER_URL\"" ) } @@ -169,7 +171,7 @@ internal fun Project.configureInstanceSelectionFlavors( buildConfigField( "String", ProductFlavors.BuildConfigFields.DefaultServerUrl, - "\"https://artemis.cit.tum.de\"" + "\"$TUM_ARTEMIS_SERVER_URL\"" ) } } From e09dfd0d514cce7a9144d0b37973aa4ef7cb33b9 Mon Sep 17 00:00:00 2001 From: Martin Felber Date: Fri, 27 Dec 2024 20:55:18 +0100 Subject: [PATCH 2/7] Adapt ServerConfigurationService to use default implementations --- .../ServerConfigurationServiceStub.kt | 4 +--- .../datastore/ServerConfigurationService.kt | 1 + .../impl/ServerConfigurationServiceImpl.kt | 22 ------------------- 3 files changed, 2 insertions(+), 25 deletions(-) diff --git a/core/datastore/src/debug/kotlin/de/tum/informatics/www1/artemis/native_app/core/datastore/ServerConfigurationServiceStub.kt b/core/datastore/src/debug/kotlin/de/tum/informatics/www1/artemis/native_app/core/datastore/ServerConfigurationServiceStub.kt index b3ee1415a..41ea73d56 100644 --- a/core/datastore/src/debug/kotlin/de/tum/informatics/www1/artemis/native_app/core/datastore/ServerConfigurationServiceStub.kt +++ b/core/datastore/src/debug/kotlin/de/tum/informatics/www1/artemis/native_app/core/datastore/ServerConfigurationServiceStub.kt @@ -1,12 +1,10 @@ package de.tum.informatics.www1.artemis.native_app.core.datastore -import de.tum.informatics.www1.artemis.native_app.core.datastore.ServerConfigurationService import kotlinx.coroutines.flow.Flow import kotlinx.coroutines.flow.flowOf class ServerConfigurationServiceStub( - override val serverUrl: Flow = flowOf("https://example.com"), - override val hasUserSelectedInstance: Flow = flowOf(true) + override val serverUrl: Flow = flowOf("https://example.com") ) : ServerConfigurationService { override suspend fun updateServerUrl(serverUrl: String) = Unit } diff --git a/core/datastore/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/core/datastore/ServerConfigurationService.kt b/core/datastore/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/core/datastore/ServerConfigurationService.kt index 46aeb6bb2..175e831a8 100644 --- a/core/datastore/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/core/datastore/ServerConfigurationService.kt +++ b/core/datastore/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/core/datastore/ServerConfigurationService.kt @@ -21,6 +21,7 @@ interface ServerConfigurationService { * If [updateServerUrl] has ever been called. */ val hasUserSelectedInstance: Flow + get() = serverUrl.map { it.isNotBlank() } suspend fun updateServerUrl(serverUrl: String) } \ No newline at end of file diff --git a/core/datastore/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/core/datastore/impl/ServerConfigurationServiceImpl.kt b/core/datastore/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/core/datastore/impl/ServerConfigurationServiceImpl.kt index a9a334f28..9bbcc8f64 100644 --- a/core/datastore/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/core/datastore/impl/ServerConfigurationServiceImpl.kt +++ b/core/datastore/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/core/datastore/impl/ServerConfigurationServiceImpl.kt @@ -1,13 +1,11 @@ package de.tum.informatics.www1.artemis.native_app.core.datastore.impl import android.content.Context -import androidx.datastore.preferences.core.booleanPreferencesKey import androidx.datastore.preferences.core.edit import androidx.datastore.preferences.core.stringPreferencesKey import androidx.datastore.preferences.preferencesDataStore import de.tum.informatics.www1.artemis.native_app.core.datastore.BuildConfig import de.tum.informatics.www1.artemis.native_app.core.datastore.ServerConfigurationService -import io.ktor.http.Url import kotlinx.coroutines.DelicateCoroutinesApi import kotlinx.coroutines.GlobalScope import kotlinx.coroutines.flow.Flow @@ -28,7 +26,6 @@ internal class ServerConfigurationServiceImpl( private val Context.serverCommunicationPreferences by preferencesDataStore("server_communication") private val SERVER_URL_KEY = stringPreferencesKey("server_url") - private val HAS_SELECTED_INSTANCE_KEY = booleanPreferencesKey("has_selected_instance") } /** @@ -47,28 +44,9 @@ internal class ServerConfigurationServiceImpl( .shareIn(GlobalScope, SharingStarted.Eagerly, replay = 1) } - override val host: Flow = - serverUrl - .map { Url(it).host } - - /** - * Use to decide if we want to show an instance selection UI to the user. - * If [BuildConfig.hasInstanceRestriction] is set to true, we never want to show such a UI. - */ - override val hasUserSelectedInstance: Flow = - if (BuildConfig.hasInstanceRestriction) flowOf(true) - else { - context - .serverCommunicationPreferences - .data - .map { it[HAS_SELECTED_INSTANCE_KEY] ?: false } - } - - override suspend fun updateServerUrl(serverUrl: String) { context.serverCommunicationPreferences.edit { data -> data[SERVER_URL_KEY] = serverUrl - data[HAS_SELECTED_INSTANCE_KEY] = true } } } \ No newline at end of file From 785b68fd1822f15fb1e57454b138104073855b56 Mon Sep 17 00:00:00 2001 From: Martin Felber Date: Fri, 27 Dec 2024 21:07:43 +0100 Subject: [PATCH 3/7] Adapt code to no longer rely on InstanceSelection --- .../TestServerConfigurationProvider.kt | 2 -- .../datastore/ServerConfigurationService.kt | 6 ---- .../native_app/feature/login/AccountUi.kt | 16 ++-------- .../feature/settings/SettingsScreen.kt | 32 ++++--------------- 4 files changed, 9 insertions(+), 47 deletions(-) diff --git a/core/core-test/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/core/test/test_setup/TestServerConfigurationProvider.kt b/core/core-test/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/core/test/test_setup/TestServerConfigurationProvider.kt index 7be927f5b..999eb67e8 100644 --- a/core/core-test/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/core/test/test_setup/TestServerConfigurationProvider.kt +++ b/core/core-test/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/core/test/test_setup/TestServerConfigurationProvider.kt @@ -8,7 +8,5 @@ import kotlinx.coroutines.flow.flowOf class TestServerConfigurationProvider : ServerConfigurationService { override val serverUrl: Flow = flowOf(testServerUrl) - override val hasUserSelectedInstance: Flow = flowOf(true) - override suspend fun updateServerUrl(serverUrl: String) = Unit } diff --git a/core/datastore/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/core/datastore/ServerConfigurationService.kt b/core/datastore/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/core/datastore/ServerConfigurationService.kt index 175e831a8..4b01fafde 100644 --- a/core/datastore/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/core/datastore/ServerConfigurationService.kt +++ b/core/datastore/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/core/datastore/ServerConfigurationService.kt @@ -17,11 +17,5 @@ interface ServerConfigurationService { val host: Flow get() = serverUrl.map { Url(it).host } - /** - * If [updateServerUrl] has ever been called. - */ - val hasUserSelectedInstance: Flow - get() = serverUrl.map { it.isNotBlank() } - suspend fun updateServerUrl(serverUrl: String) } \ No newline at end of file diff --git a/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt b/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt index 278bdca04..7decef9d6 100644 --- a/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt +++ b/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt @@ -87,8 +87,6 @@ private const val ARG_REMEMBER_ME = "rememberMe" @Serializable private sealed interface NestedDestination { - @Serializable - data object InstanceSelection : NestedDestination @Serializable data object CustomInstanceSelection : NestedDestination @Serializable @@ -212,12 +210,6 @@ private fun LoginUiScreen( val nestedNavController = rememberNavController() val serverConfigurationService: ServerConfigurationService = koinInject() - val hasSelectedInstance = serverConfigurationService - .hasUserSelectedInstance - .collectAsState(initial = null) - .value - ?: return // Display nothing to avoid switching between destinations - // Force recomposition val currentBackStack by nestedNavController.currentBackStackEntryAsState() nestedNavController.currentBackStackEntryAsState().value @@ -271,7 +263,7 @@ private fun LoginUiScreen( .consumeWindowInsets(WindowInsets.systemBars) .padding(top = paddingValues.calculateTopPadding()), navController = nestedNavController, - startDestination = if (hasSelectedInstance) NestedDestination.Home else NestedDestination.InstanceSelection + startDestination = NestedDestination.Home ) { animatedComposable { AccountScreen( @@ -298,11 +290,7 @@ private fun LoginUiScreen( .fillMaxSize() .padding(horizontal = 16.dp) ) { - nestedNavController.navigate(NestedDestination.Home) { - popUpTo { - inclusive = true - } - } + nestedNavController.navigate(NestedDestination.Home) } } diff --git a/feature/settings/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/settings/SettingsScreen.kt b/feature/settings/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/settings/SettingsScreen.kt index fb97d1cc6..7c209f462 100644 --- a/feature/settings/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/settings/SettingsScreen.kt +++ b/feature/settings/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/settings/SettingsScreen.kt @@ -20,7 +20,6 @@ import androidx.compose.material.icons.automirrored.filled.ArrowBack import androidx.compose.material.icons.filled.AlternateEmail import androidx.compose.material.icons.filled.ChevronRight import androidx.compose.material.icons.filled.Mail -import androidx.compose.material3.Button import androidx.compose.material3.Card import androidx.compose.material3.Icon import androidx.compose.material3.IconButton @@ -147,10 +146,6 @@ private fun SettingsScreen( val scope = rememberCoroutineScope() - val hasUserSelectedInstance by serverConfigurationService.hasUserSelectedInstance.collectAsState( - initial = false - ) - val accountDataService: AccountDataService = koinInject() val networkStatusProvider: NetworkStatusProvider = koinInject() @@ -230,7 +225,6 @@ private fun SettingsScreen( AboutSection( modifier = Modifier.fillMaxWidth(), - hasUserSelectedInstance = hasUserSelectedInstance, serverUrl = serverUrl, onOpenPrivacyPolicy = { val link = URLBuilder(serverUrl).appendPathSegments("privacy").buildString() @@ -242,9 +236,6 @@ private fun SettingsScreen( linkOpener.openLink(link) }, onOpenThirdPartyLicenses = onDisplayThirdPartyLicenses, - // it can only be unselected, if the user has navigated to the settings from the instance selection screen. - // Therefore, a simple navigate up will let the user select the server instance. - onRequestSelectServerInstance = onNavigateUp ) BuildInformationSection( @@ -341,9 +332,7 @@ private fun NotificationSection(modifier: Modifier, onOpenNotificationSettings: @Composable private fun AboutSection( modifier: Modifier, - hasUserSelectedInstance: Boolean, serverUrl: String, - onRequestSelectServerInstance: () -> Unit, onOpenPrivacyPolicy: () -> Unit, onOpenImprint: () -> Unit, onOpenThirdPartyLicenses: () -> Unit @@ -362,7 +351,7 @@ private fun AboutSection( ) } - if (hasUserSelectedInstance) { + if (serverUrl.isNotEmpty()) { ServerURLEntry( modifier = Modifier.fillMaxWidth(), text = stringResource(R.string.settings_server_url), @@ -385,19 +374,12 @@ private fun AboutSection( onClick = onOpenImprint ) } else { - Column(modifier = Modifier.padding(horizontal = 16.dp)) { - Text( - text = stringResource(id = R.string.settings_server_specifics_unavailable), - style = MaterialTheme.typography.bodyMedium, - color = MaterialTheme.colorScheme.error - ) - - Button( - onClick = onRequestSelectServerInstance - ) { - Text(text = stringResource(id = R.string.settings_server_specifics_unavailable_select_instance_button)) - } - } + Text( + modifier = Modifier.padding(horizontal = 16.dp), + text = stringResource(id = R.string.settings_server_specifics_unavailable), + style = MaterialTheme.typography.bodyMedium, + color = MaterialTheme.colorScheme.error + ) } ButtonEntry( From 391e6c8f5763295002d6732db6e6bfdd3704e61d Mon Sep 17 00:00:00 2001 From: Martin Felber Date: Mon, 30 Dec 2024 22:47:43 +0100 Subject: [PATCH 4/7] Address PR comments --- .../native_app/feature/login/AccountUi.kt | 49 +++++++++++++++---- .../native_app/feature/login/login/LoginUi.kt | 3 +- 2 files changed, 40 insertions(+), 12 deletions(-) diff --git a/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt b/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt index 7decef9d6..4ffbc57cb 100644 --- a/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt +++ b/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt @@ -7,12 +7,14 @@ import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.PaddingValues +import androidx.compose.foundation.layout.Spacer import androidx.compose.foundation.layout.WindowInsets import androidx.compose.foundation.layout.asPaddingValues import androidx.compose.foundation.layout.consumeWindowInsets import androidx.compose.foundation.layout.fillMaxHeight import androidx.compose.foundation.layout.fillMaxSize import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.height import androidx.compose.foundation.layout.imePadding import androidx.compose.foundation.layout.padding import androidx.compose.foundation.layout.systemBars @@ -252,7 +254,9 @@ private fun LoginUiScreen( ) } ) { paddingValues -> - val sheetState = rememberModalBottomSheetState() + val sheetState = rememberModalBottomSheetState( + skipPartiallyExpanded = true + ) val scope = rememberCoroutineScope() var showBottomSheet by remember { mutableStateOf(false) } @@ -261,7 +265,8 @@ private fun LoginUiScreen( .fillMaxSize() .imePadding() .consumeWindowInsets(WindowInsets.systemBars) - .padding(top = paddingValues.calculateTopPadding()), + .padding(top = paddingValues.calculateTopPadding()) + .padding(horizontal = Spacings.ScreenHorizontalSpacing), navController = nestedNavController, startDestination = NestedDestination.Home ) { @@ -390,10 +395,12 @@ private fun AccountScreen( onClickSaml2Login: (rememberMe: Boolean) -> Unit ) { val serverProfileInfo by viewModel.serverProfileInfo.collectAsState() + val selectedInstance by viewModel.selectedArtemisInstance.collectAsState() AccountUi( modifier = modifier, serverProfileInfo = serverProfileInfo, + host = selectedInstance.host, canSwitchInstance = canSwitchInstance, retryLoadServerProfileInfo = viewModel::requestReloadServerProfileInfo, onNavigateToLoginScreen = onNavigateToLoginScreen, @@ -408,6 +415,7 @@ private fun AccountScreen( private fun AccountUi( modifier: Modifier, serverProfileInfo: DataState, + host: String, canSwitchInstance: Boolean, retryLoadServerProfileInfo: () -> Unit, onNavigateToLoginScreen: () -> Unit, @@ -425,7 +433,10 @@ private fun AccountUi( .fillMaxHeight(0.05f) ) - ArtemisHeader(modifier = Modifier.fillMaxWidth()) + ArtemisHeader( + modifier = Modifier.fillMaxWidth(), + host = host + ) Box( modifier = Modifier @@ -509,7 +520,7 @@ private fun RegisterLoginAccount( //Just for the preview. LoginUi( modifier = loginUiModifier, - accountName = "TUm", + accountName = "TUM", needsToAcceptTerms = true, hasUserAcceptedTerms = true, saml2Config = null, @@ -596,10 +607,16 @@ private fun LoginOrRegister( } @Composable -internal fun ArtemisHeader(modifier: Modifier) { - Column(modifier = modifier) { +internal fun ArtemisHeader( + modifier: Modifier, + host: String +) { + Column( + modifier = modifier, + horizontalAlignment = Alignment.CenterHorizontally, + verticalArrangement = Arrangement.spacedBy(4.dp) + ) { Text( - modifier = Modifier.fillMaxWidth(), text = stringResource(id = R.string.account_screen_title), fontSize = 32.sp, fontWeight = FontWeight.Bold, @@ -607,14 +624,22 @@ internal fun ArtemisHeader(modifier: Modifier) { ) Text( - modifier = Modifier - .fillMaxWidth() - .padding(top = 4.dp), text = stringResource(id = R.string.account_screen_subtitle), fontSize = 20.sp, fontWeight = FontWeight.Normal, textAlign = TextAlign.Center ) + + + if (BuildConfig.DEBUG) { + Spacer(Modifier.height(8.dp)) + + Text( + text = host, + style = MaterialTheme.typography.bodySmall, + color = MaterialTheme.colorScheme.secondary + ) + } } } @@ -625,6 +650,7 @@ fun AccountUiPreviewLoadingProfileInfo() { modifier = Modifier.fillMaxSize(), canSwitchInstance = true, serverProfileInfo = DataState.Loading(), + host = "", retryLoadServerProfileInfo = {}, onNavigateToLoginScreen = {}, onNavigateToRegisterScreen = {}, @@ -641,6 +667,7 @@ fun AccountUiPreviewFailedLoadingProfileInfo() { modifier = Modifier.fillMaxSize(), canSwitchInstance = true, serverProfileInfo = DataState.Failure(IOException()), + host = "", retryLoadServerProfileInfo = {}, onNavigateToLoginScreen = {}, onNavigateToRegisterScreen = {}, @@ -661,6 +688,7 @@ fun AccountUiPreviewWithRegister() { registrationEnabled = true ) ), + host = "", retryLoadServerProfileInfo = {}, onNavigateToLoginScreen = {}, onNavigateToRegisterScreen = {}, @@ -681,6 +709,7 @@ fun AccountUiPreviewWithoutRegister() { registrationEnabled = false ) ), + host = "", retryLoadServerProfileInfo = {}, onNavigateToLoginScreen = {}, onNavigateToRegisterScreen = {}, diff --git a/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/login/LoginUi.kt b/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/login/LoginUi.kt index f38f78582..228b90772 100644 --- a/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/login/LoginUi.kt +++ b/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/login/LoginUi.kt @@ -3,7 +3,6 @@ package de.tum.informatics.www1.artemis.native_app.feature.login.login import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.fillMaxWidth -import androidx.compose.foundation.layout.imePadding import androidx.compose.foundation.layout.padding import androidx.compose.foundation.layout.widthIn import androidx.compose.foundation.rememberScrollState @@ -193,7 +192,7 @@ internal fun LoginUi( ) val loginModifier = Modifier - .padding(16.dp) + .padding(vertical = 16.dp) .widthIn(max = 600.dp) .fillMaxWidth(0.8f) .align(Alignment.CenterHorizontally) From cfe1f381355494fac88d1b8f1666285d1d446da9 Mon Sep 17 00:00:00 2001 From: Martin Felber Date: Mon, 30 Dec 2024 22:53:18 +0100 Subject: [PATCH 5/7] Fix warning --- .../native_app/feature/login/AccountUi.kt | 26 ++++++++----------- 1 file changed, 11 insertions(+), 15 deletions(-) diff --git a/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt b/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt index 4ffbc57cb..a5ef789d0 100644 --- a/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt +++ b/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt @@ -9,7 +9,6 @@ import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.PaddingValues import androidx.compose.foundation.layout.Spacer import androidx.compose.foundation.layout.WindowInsets -import androidx.compose.foundation.layout.asPaddingValues import androidx.compose.foundation.layout.consumeWindowInsets import androidx.compose.foundation.layout.fillMaxHeight import androidx.compose.foundation.layout.fillMaxSize @@ -19,7 +18,6 @@ import androidx.compose.foundation.layout.imePadding import androidx.compose.foundation.layout.padding import androidx.compose.foundation.layout.systemBars import androidx.compose.foundation.rememberScrollState -import androidx.compose.foundation.text.ClickableText import androidx.compose.foundation.verticalScroll import androidx.compose.material.icons.Icons import androidx.compose.material.icons.filled.Close @@ -32,6 +30,7 @@ import androidx.compose.material3.MaterialTheme import androidx.compose.material3.ModalBottomSheet import androidx.compose.material3.Scaffold import androidx.compose.material3.Text +import androidx.compose.material3.TextButton import androidx.compose.material3.TopAppBar import androidx.compose.material3.rememberModalBottomSheetState import androidx.compose.runtime.Composable @@ -46,7 +45,6 @@ import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.platform.LocalInspectionMode import androidx.compose.ui.res.stringResource -import androidx.compose.ui.text.AnnotatedString import androidx.compose.ui.text.font.FontWeight import androidx.compose.ui.text.style.TextAlign import androidx.compose.ui.tooling.preview.Preview @@ -457,19 +455,17 @@ private fun AccountUi( ) if (canSwitchInstance) { - ClickableText( + TextButton( modifier = Modifier - .align(Alignment.CenterHorizontally) - .padding( - bottom = WindowInsets.systemBars - .asPaddingValues() - .calculateBottomPadding() - + 8.dp - ), - text = AnnotatedString(stringResource(id = R.string.account_change_artemis_instance_label)), - style = MaterialTheme.typography.bodyMedium.copy(color = MaterialTheme.colorScheme.linkTextColor), - onClick = { onNavigateToInstanceSelection() } - ) + .padding(vertical = 8.dp) + .align(Alignment.CenterHorizontally), + onClick = onNavigateToInstanceSelection + ) { + Text( + text = stringResource(id = R.string.account_change_artemis_instance_label), + style = MaterialTheme.typography.bodyMedium.copy(color = MaterialTheme.colorScheme.linkTextColor) + ) + } } } } From 6cdb0329731510d4756c1dec8a7e59af3d01e0be Mon Sep 17 00:00:00 2001 From: Martin Felber Date: Thu, 2 Jan 2025 12:57:06 +0100 Subject: [PATCH 6/7] Add top padding to bottomSheet --- .../www1/artemis/native_app/feature/login/AccountUi.kt | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt b/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt index a5ef789d0..1abc6561b 100644 --- a/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt +++ b/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt @@ -9,6 +9,7 @@ import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.PaddingValues import androidx.compose.foundation.layout.Spacer import androidx.compose.foundation.layout.WindowInsets +import androidx.compose.foundation.layout.asPaddingValues import androidx.compose.foundation.layout.consumeWindowInsets import androidx.compose.foundation.layout.fillMaxHeight import androidx.compose.foundation.layout.fillMaxSize @@ -350,6 +351,9 @@ private fun LoginUiScreen( if (showBottomSheet) { ModalBottomSheet( + modifier = Modifier.padding( + top = WindowInsets.systemBars.asPaddingValues().calculateTopPadding(), + ), onDismissRequest = { showBottomSheet = false }, From f01452ac97cc85e7d2f34f801a84628c3a84eb64 Mon Sep 17 00:00:00 2001 From: Martin Felber Date: Thu, 2 Jan 2025 13:09:29 +0100 Subject: [PATCH 7/7] Move bottomSheet to InstanceSelectionScreen.kt; show instanceUrl also in production for selected CustomInstance --- .../native_app/feature/login/AccountUi.kt | 88 ++++++------------- .../InstanceSelectionScreen.kt | 59 +++++++++++++ 2 files changed, 85 insertions(+), 62 deletions(-) diff --git a/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt b/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt index 1abc6561b..cb887b4ed 100644 --- a/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt +++ b/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/AccountUi.kt @@ -9,7 +9,6 @@ import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.PaddingValues import androidx.compose.foundation.layout.Spacer import androidx.compose.foundation.layout.WindowInsets -import androidx.compose.foundation.layout.asPaddingValues import androidx.compose.foundation.layout.consumeWindowInsets import androidx.compose.foundation.layout.fillMaxHeight import androidx.compose.foundation.layout.fillMaxSize @@ -28,12 +27,10 @@ import androidx.compose.material3.HorizontalDivider import androidx.compose.material3.Icon import androidx.compose.material3.IconButton import androidx.compose.material3.MaterialTheme -import androidx.compose.material3.ModalBottomSheet import androidx.compose.material3.Scaffold import androidx.compose.material3.Text import androidx.compose.material3.TextButton import androidx.compose.material3.TopAppBar -import androidx.compose.material3.rememberModalBottomSheetState import androidx.compose.runtime.Composable import androidx.compose.runtime.collectAsState import androidx.compose.runtime.getValue @@ -68,14 +65,13 @@ import de.tum.informatics.www1.artemis.native_app.core.ui.material.colors.linkTe import de.tum.informatics.www1.artemis.native_app.core.ui.navigation.DefaultTransition import de.tum.informatics.www1.artemis.native_app.core.ui.navigation.animatedComposable import de.tum.informatics.www1.artemis.native_app.feature.login.custom_instance_selection.CustomInstanceSelectionScreen -import de.tum.informatics.www1.artemis.native_app.feature.login.instance_selection.InstanceSelectionScreen +import de.tum.informatics.www1.artemis.native_app.feature.login.instance_selection.InstanceSelectionBottomSheet import de.tum.informatics.www1.artemis.native_app.feature.login.login.LoginScreen import de.tum.informatics.www1.artemis.native_app.feature.login.login.LoginUi import de.tum.informatics.www1.artemis.native_app.feature.login.register.RegisterUi import de.tum.informatics.www1.artemis.native_app.feature.login.saml2_login.Saml2LoginScreen import de.tum.informatics.www1.artemis.native_app.feature.login.saml2_login.Saml2LoginViewModel import de.tum.informatics.www1.artemis.native_app.feature.login.service.ServerNotificationStorageService -import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.flow.first import kotlinx.coroutines.launch import kotlinx.serialization.Serializable @@ -253,11 +249,7 @@ private fun LoginUiScreen( ) } ) { paddingValues -> - val sheetState = rememberModalBottomSheetState( - skipPartiallyExpanded = true - ) - val scope = rememberCoroutineScope() - var showBottomSheet by remember { mutableStateOf(false) } + var showInstanceSelectionBottomSheet by remember { mutableStateOf(false) } NavHost( modifier = Modifier @@ -281,7 +273,7 @@ private fun LoginUiScreen( }, onNavigateToInstanceSelection = { onNavigatedToInstanceSelection() - showBottomSheet = true + showInstanceSelectionBottomSheet = true }, onLoggedIn = onLoggedIn, onClickSaml2Login = onClickSaml2Login @@ -336,48 +328,20 @@ private fun LoginUiScreen( } } - val hideBottomSheet: (action: (suspend CoroutineScope.() -> Unit)?) -> Unit = { action -> - scope.launch { - if (action != null) { - action() - } - sheetState.hide() - }.invokeOnCompletion { - if (!sheetState.isVisible) { - showBottomSheet = false - } - } - } - - if (showBottomSheet) { - ModalBottomSheet( - modifier = Modifier.padding( - top = WindowInsets.systemBars.asPaddingValues().calculateTopPadding(), - ), - onDismissRequest = { - showBottomSheet = false + if (showInstanceSelectionBottomSheet) { + InstanceSelectionBottomSheet( + onDismiss = { + showInstanceSelectionBottomSheet = false }, - sheetState = sheetState - ) { - InstanceSelectionScreen( - modifier = Modifier - .fillMaxSize() - .padding(horizontal = Spacings.ScreenHorizontalSpacing), - availableInstances = ArtemisInstances.instances, - onSelectArtemisInstance = { serverUrl -> - hideBottomSheet { - serverConfigurationService.updateServerUrl(serverUrl) - } - }, - onRequestOpenCustomInstanceSelection = { - hideBottomSheet { - nestedNavController.navigate( - NestedDestination.CustomInstanceSelection - ) - } - } - ) - } + onSelectArtemisInstance = { + serverConfigurationService.updateServerUrl(it) + }, + onRequestOpenCustomInstanceSelection = { + nestedNavController.navigate( + NestedDestination.CustomInstanceSelection + ) + } + ) } } } @@ -402,7 +366,7 @@ private fun AccountScreen( AccountUi( modifier = modifier, serverProfileInfo = serverProfileInfo, - host = selectedInstance.host, + selectedInstance = selectedInstance, canSwitchInstance = canSwitchInstance, retryLoadServerProfileInfo = viewModel::requestReloadServerProfileInfo, onNavigateToLoginScreen = onNavigateToLoginScreen, @@ -417,7 +381,7 @@ private fun AccountScreen( private fun AccountUi( modifier: Modifier, serverProfileInfo: DataState, - host: String, + selectedInstance: ArtemisInstances.ArtemisInstance, canSwitchInstance: Boolean, retryLoadServerProfileInfo: () -> Unit, onNavigateToLoginScreen: () -> Unit, @@ -437,7 +401,7 @@ private fun AccountUi( ArtemisHeader( modifier = Modifier.fillMaxWidth(), - host = host + selectedInstance = selectedInstance ) Box( @@ -609,7 +573,7 @@ private fun LoginOrRegister( @Composable internal fun ArtemisHeader( modifier: Modifier, - host: String + selectedInstance: ArtemisInstances.ArtemisInstance, ) { Column( modifier = modifier, @@ -631,11 +595,11 @@ internal fun ArtemisHeader( ) - if (BuildConfig.DEBUG) { + if (BuildConfig.DEBUG || selectedInstance.type == ArtemisInstances.ArtemisInstance.Type.CUSTOM) { Spacer(Modifier.height(8.dp)) Text( - text = host, + text = selectedInstance.host, style = MaterialTheme.typography.bodySmall, color = MaterialTheme.colorScheme.secondary ) @@ -650,7 +614,7 @@ fun AccountUiPreviewLoadingProfileInfo() { modifier = Modifier.fillMaxSize(), canSwitchInstance = true, serverProfileInfo = DataState.Loading(), - host = "", + selectedInstance = ArtemisInstances.TumArtemis, retryLoadServerProfileInfo = {}, onNavigateToLoginScreen = {}, onNavigateToRegisterScreen = {}, @@ -667,7 +631,7 @@ fun AccountUiPreviewFailedLoadingProfileInfo() { modifier = Modifier.fillMaxSize(), canSwitchInstance = true, serverProfileInfo = DataState.Failure(IOException()), - host = "", + selectedInstance = ArtemisInstances.TumArtemis, retryLoadServerProfileInfo = {}, onNavigateToLoginScreen = {}, onNavigateToRegisterScreen = {}, @@ -688,7 +652,7 @@ fun AccountUiPreviewWithRegister() { registrationEnabled = true ) ), - host = "", + selectedInstance = ArtemisInstances.TumArtemis, retryLoadServerProfileInfo = {}, onNavigateToLoginScreen = {}, onNavigateToRegisterScreen = {}, @@ -709,7 +673,7 @@ fun AccountUiPreviewWithoutRegister() { registrationEnabled = false ) ), - host = "", + selectedInstance = ArtemisInstances.TumArtemis, retryLoadServerProfileInfo = {}, onNavigateToLoginScreen = {}, onNavigateToRegisterScreen = {}, diff --git a/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/instance_selection/InstanceSelectionScreen.kt b/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/instance_selection/InstanceSelectionScreen.kt index f7dc4e843..dcb81a09f 100644 --- a/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/instance_selection/InstanceSelectionScreen.kt +++ b/feature/login/src/main/kotlin/de/tum/informatics/www1/artemis/native_app/feature/login/instance_selection/InstanceSelectionScreen.kt @@ -7,6 +7,7 @@ import androidx.compose.foundation.layout.Row import androidx.compose.foundation.layout.WindowInsets import androidx.compose.foundation.layout.asPaddingValues import androidx.compose.foundation.layout.aspectRatio +import androidx.compose.foundation.layout.fillMaxSize import androidx.compose.foundation.layout.fillMaxWidth import androidx.compose.foundation.layout.height import androidx.compose.foundation.layout.padding @@ -17,9 +18,12 @@ import androidx.compose.material.icons.filled.Add import androidx.compose.material3.Card import androidx.compose.material3.Icon import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.ModalBottomSheet import androidx.compose.material3.Text +import androidx.compose.material3.rememberModalBottomSheetState import androidx.compose.runtime.Composable import androidx.compose.runtime.remember +import androidx.compose.runtime.rememberCoroutineScope import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.platform.LocalContext @@ -30,7 +34,62 @@ import androidx.compose.ui.unit.dp import coil3.compose.AsyncImage import coil3.request.ImageRequest import de.tum.informatics.www1.artemis.native_app.core.datastore.defaults.ArtemisInstances +import de.tum.informatics.www1.artemis.native_app.core.ui.Spacings import de.tum.informatics.www1.artemis.native_app.feature.login.R +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.launch + + +@Composable +fun InstanceSelectionBottomSheet( + modifier: Modifier = Modifier, + onDismiss: () -> Unit, + onSelectArtemisInstance: suspend (String) -> Unit, + onRequestOpenCustomInstanceSelection: suspend () -> Unit +) { + val scope = rememberCoroutineScope() + val sheetState = rememberModalBottomSheetState( + skipPartiallyExpanded = true + ) + + val hideBottomSheet: (action: (suspend CoroutineScope.() -> Unit)?) -> Unit = { action -> + scope.launch { + if (action != null) { + action() + } + sheetState.hide() + }.invokeOnCompletion { + if (!sheetState.isVisible) { + onDismiss() + } + } + } + + ModalBottomSheet( + modifier = modifier.padding( + top = WindowInsets.systemBars.asPaddingValues().calculateTopPadding(), + ), + onDismissRequest = onDismiss, + sheetState = sheetState + ) { + InstanceSelectionScreen( + modifier = Modifier + .fillMaxSize() + .padding(horizontal = Spacings.ScreenHorizontalSpacing), + availableInstances = ArtemisInstances.instances, + onSelectArtemisInstance = { serverUrl -> + hideBottomSheet { + onSelectArtemisInstance(serverUrl) + } + }, + onRequestOpenCustomInstanceSelection = { + hideBottomSheet { + onRequestOpenCustomInstanceSelection() + } + } + ) + } +} @Composable internal fun InstanceSelectionScreen(