From eeed33c3c58adacd2182e0360a9652de831692a3 Mon Sep 17 00:00:00 2001 From: ndubkov-distcotech Date: Mon, 15 Jul 2024 16:19:55 +0200 Subject: [PATCH 1/3] feat(bank-sdk): Skonto screen. Edit amounts implementation (#488) feat(bank-sdk): Skonto screen. Edit amounts implementation PP-411 --- .../bank/sdk/capture/skonto/SkontoFragment.kt | 103 +++++++++--------- .../capture/skonto/SkontoFragmentContract.kt | 10 +- .../capture/skonto/SkontoFragmentViewModel.kt | 81 +++++++++++--- .../src/main/res/navigation/gbs_nav_graph.xml | 3 +- .../ui/components/textinput/GiniTextInput.kt | 5 + .../textinput/amount/DecimalFormatter.kt | 38 +++++++ .../DecimalInputVisualTransformation.kt | 84 ++++++++++++++ .../textinput/amount/GiniAmountTextInput.kt | 95 ++++++++++++++++ .../amount/CustomOffsetMappingTest.kt | 80 ++++++++++++++ 9 files changed, 427 insertions(+), 72 deletions(-) create mode 100644 capture-sdk/sdk/src/main/java/net/gini/android/capture/ui/components/textinput/amount/DecimalFormatter.kt create mode 100644 capture-sdk/sdk/src/main/java/net/gini/android/capture/ui/components/textinput/amount/DecimalInputVisualTransformation.kt create mode 100644 capture-sdk/sdk/src/main/java/net/gini/android/capture/ui/components/textinput/amount/GiniAmountTextInput.kt create mode 100644 capture-sdk/sdk/src/test/java/net/gini/android/capture/ui/components/textinput/amount/CustomOffsetMappingTest.kt diff --git a/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoFragment.kt b/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoFragment.kt index 153e6ec4f7..df86a072a6 100644 --- a/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoFragment.kt +++ b/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoFragment.kt @@ -1,9 +1,8 @@ -@file:OptIn(ExperimentalMaterial3Api::class) - package net.gini.android.bank.sdk.capture.skonto import android.content.res.Configuration.UI_MODE_NIGHT_YES import android.os.Bundle +import android.util.Log import android.view.LayoutInflater import android.view.View import android.view.ViewGroup @@ -28,7 +27,6 @@ import androidx.compose.material.icons.automirrored.filled.ArrowBack import androidx.compose.material.icons.automirrored.filled.KeyboardArrowRight import androidx.compose.material3.Card import androidx.compose.material3.CardDefaults -import androidx.compose.material3.ExperimentalMaterial3Api import androidx.compose.material3.Icon import androidx.compose.material3.Scaffold import androidx.compose.material3.Text @@ -62,18 +60,22 @@ import net.gini.android.bank.sdk.capture.skonto.colors.section.SkontoFooterSecti import net.gini.android.bank.sdk.capture.skonto.colors.section.SkontoInvoiceScanSectionColors import net.gini.android.bank.sdk.capture.skonto.colors.section.SkontoSectionColors import net.gini.android.bank.sdk.capture.skonto.colors.section.WithoutSkontoSectionColors +import net.gini.android.bank.sdk.capture.util.currencyFormatterWithoutSymbol import net.gini.android.capture.GiniCapture import net.gini.android.capture.ui.components.button.filled.GiniButton import net.gini.android.capture.ui.components.picker.date.GiniDatePickerDialog import net.gini.android.capture.ui.components.switcher.GiniSwitch import net.gini.android.capture.ui.components.textinput.GiniTextInput +import net.gini.android.capture.ui.components.textinput.amount.GiniAmountTextInput import net.gini.android.capture.ui.components.topbar.GiniTopBar import net.gini.android.capture.ui.components.topbar.GiniTopBarColors import net.gini.android.capture.ui.theme.GiniTheme +import java.math.BigDecimal +import java.math.RoundingMode +import java.text.DecimalFormat import java.time.LocalDate import java.time.format.DateTimeFormatter - class SkontoFragment : Fragment() { val bottomNavBar = GiniCapture.getInstance().internal().navigationBarTopAdapterInstance @@ -119,8 +121,8 @@ private fun ScreenContent( private fun ScreenStateContent( state: SkontoFragmentContract.State, onDiscountSectionActiveChange: (Boolean) -> Unit, - onSkontoAmountChange: (String) -> Unit, - onFullAmountChange: (String) -> Unit, + onSkontoAmountChange: (BigDecimal) -> Unit, + onFullAmountChange: (BigDecimal) -> Unit, onDueDateChanged: (LocalDate) -> Unit, modifier: Modifier = Modifier, screenColorScheme: SkontoScreenColors = SkontoScreenColors.colors() @@ -144,12 +146,13 @@ private fun ScreenStateContent( private fun ScreenReadyState( state: SkontoFragmentContract.State.Ready, onDiscountSectionActiveChange: (Boolean) -> Unit, - onDiscountAmountChange: (String) -> Unit, + onDiscountAmountChange: (BigDecimal) -> Unit, onDueDateChanged: (LocalDate) -> Unit, - onFullAmountChange: (String) -> Unit, + onFullAmountChange: (BigDecimal) -> Unit, modifier: Modifier = Modifier, screenColorScheme: SkontoScreenColors = SkontoScreenColors.colors(), ) { + val scrollState = rememberScrollState() Scaffold(modifier = modifier, containerColor = screenColorScheme.backgroundColor, @@ -189,7 +192,7 @@ private fun ScreenReadyState( isActive = !state.isSkontoSectionActive, amount = state.fullAmount, currencyCode = "EUR", - onFullAmountChange = onFullAmountChange + onFullAmountChange = onFullAmountChange, ) } } @@ -280,26 +283,19 @@ private fun YourInvoiceScanSection( @Composable private fun SkontoSection( colors: SkontoSectionColors, - amount: Float, + amount: BigDecimal, currencyCode: String, dueDate: LocalDate, infoPaymentInDays: Int, - infoDiscountValue: Float, + infoDiscountValue: BigDecimal, onActiveChange: (Boolean) -> Unit, - onSkontoAmountChange: (String) -> Unit, + onSkontoAmountChange: (BigDecimal) -> Unit, onDueDateChanged: (LocalDate) -> Unit, modifier: Modifier = Modifier, - isActive: Boolean = true, + isActive: Boolean, ) { val dateFormatter = DateTimeFormatter.ofPattern("dd.MM.yyyy") - val amountText = if (isActive) { - "%.2f".format(amount) - } else { - "$amount $currencyCode" - } - - var isDatePickerVisible by remember { mutableStateOf(false) } val amountFieldFocusRequester = remember { FocusRequester() } val focusManager = LocalFocusManager.current @@ -338,23 +334,18 @@ private fun SkontoSection( } InfoBanner( paymentIn = infoPaymentInDays.toString(), - discountValue = infoDiscountValue.toString(), + discountValue = infoDiscountValue, modifier = Modifier.fillMaxWidth(), colors = colors.infoBannerColors, ) - - GiniTextInput( + GiniAmountTextInput( + amount = amount, modifier = Modifier .fillMaxWidth() .padding(top = 16.dp), enabled = isActive, colors = colors.amountFieldColors, - keyboardOptions = KeyboardOptions( - keyboardType = KeyboardType.Number, - imeAction = ImeAction.Done, - ), - onValueChange = onSkontoAmountChange, - text = amountText, + onValueChange = { onSkontoAmountChange(it) }, label = stringResource(id = R.string.gbs_skonto_section_discount_field_amount_hint), trailingContent = { AnimatedVisibility(visible = isActive) { @@ -410,7 +401,7 @@ private fun SkontoSection( private fun InfoBanner( colors: SkontoSectionColors.InfoBannerColors, paymentIn: String, - discountValue: String, + discountValue: BigDecimal, modifier: Modifier = Modifier, ) { Row( @@ -425,11 +416,15 @@ private fun InfoBanner( contentDescription = null, tint = colors.iconTint, ) + val animatedDiscountAmount by animateFloatAsState( + targetValue = discountValue.toFloat(), label = "discountAmount" + ) + Text( text = stringResource( id = R.string.gbs_skonto_section_discount_info_banner_message, paymentIn, - discountValue + animatedDiscountAmount.formatAsDiscountPercentage() ), style = GiniTheme.typography.subtitle2, color = colors.textColor, @@ -440,11 +435,11 @@ private fun InfoBanner( @Composable private fun WithoutSkontoSection( colors: WithoutSkontoSectionColors, - amount: String, + amount: BigDecimal, currencyCode: String, modifier: Modifier = Modifier, - onFullAmountChange: (String) -> Unit, - isActive: Boolean = true, + onFullAmountChange: (BigDecimal) -> Unit, + isActive: Boolean, ) { Card( modifier = modifier.fillMaxWidth(), @@ -473,18 +468,14 @@ private fun WithoutSkontoSection( ) } } - GiniTextInput( + GiniAmountTextInput( modifier = Modifier .fillMaxWidth() .padding(top = 16.dp), enabled = isActive, colors = colors.amountFieldColors, - keyboardOptions = KeyboardOptions( - keyboardType = KeyboardType.Number, - imeAction = ImeAction.Done, - ), - onValueChange = { onFullAmountChange(it) }, - text = amount, + amount = amount, + onValueChange = onFullAmountChange, label = stringResource(id = R.string.gbs_skonto_section_without_discount_field_amount_hint), trailingContent = { AnimatedVisibility(visible = isActive) { @@ -501,8 +492,8 @@ private fun WithoutSkontoSection( @Composable private fun FooterSection( - totalAmount: Float, - discountValue: Float, + totalAmount: BigDecimal, + discountValue: BigDecimal, currency: String, colors: SkontoFooterSectionColors, modifier: Modifier = Modifier, @@ -527,9 +518,11 @@ private fun FooterSection( horizontalArrangement = Arrangement.Start, verticalAlignment = Alignment.CenterVertically, ) { - val amount by animateFloatAsState(targetValue = totalAmount, label = "") + val animatedTotalAmount by animateFloatAsState( + targetValue = totalAmount.toFloat(), label = "totalAmount" + ) Text( - text = "${"%.2f".format(amount)} $currency", + text = "${currencyFormatterWithoutSymbol().format(animatedTotalAmount)} $currency", style = GiniTheme.typography.headline5, color = colors.amountTextColor, ) @@ -542,11 +535,15 @@ private fun FooterSection( RoundedCornerShape(4.dp) ), ) { + val animatedDiscountAmount by animateFloatAsState( + targetValue = discountValue.toFloat(), label = "discountAmount" + ) + Text( modifier = Modifier.padding(vertical = 4.dp, horizontal = 8.dp), text = stringResource( id = R.string.gbs_skonto_section_footer_label_discount, - "$discountValue%" + animatedDiscountAmount.formatAsDiscountPercentage() ), style = GiniTheme.typography.caption1, color = colors.discountLabelColorScheme.textColor, @@ -592,14 +589,18 @@ private fun ScreenReadyStatePreview() { } } +private fun Float.formatAsDiscountPercentage(): String { + val value = BigDecimal(this.toString()).setScale(2, RoundingMode.HALF_UP) + return "${value.toString().trimEnd('0').trimEnd('.')}%" +} + private val previewState = SkontoFragmentContract.State.Ready( isSkontoSectionActive = true, paymentInDays = 14, - discountValue = 3.0f, - skontoAmount = 97.0f, + discountValue = BigDecimal("3"), + skontoAmount = BigDecimal("97"), discountDueDate = LocalDate.now(), - fullAmount = "100.00", - totalAmount = 97.0f, - totalDiscount = 3.0f, + fullAmount = BigDecimal("100"), + totalAmount = BigDecimal("97"), currency = "EUR" ) \ No newline at end of file diff --git a/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoFragmentContract.kt b/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoFragmentContract.kt index 651288ce69..167aea178a 100644 --- a/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoFragmentContract.kt +++ b/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoFragmentContract.kt @@ -1,5 +1,6 @@ package net.gini.android.bank.sdk.capture.skonto +import java.math.BigDecimal import java.time.LocalDate object SkontoFragmentContract { @@ -10,13 +11,12 @@ object SkontoFragmentContract { data class Ready( val isSkontoSectionActive: Boolean, val paymentInDays: Int, - val discountValue: Float, - val skontoAmount: Float, + val discountValue: BigDecimal, + val skontoAmount: BigDecimal, val discountDueDate: LocalDate, - val fullAmount: String, - val totalAmount: Float, + val fullAmount: BigDecimal, + val totalAmount: BigDecimal, val currency: String, - val totalDiscount: Float, ) : State() } diff --git a/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoFragmentViewModel.kt b/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoFragmentViewModel.kt index 88816155e3..f9e4757ee8 100644 --- a/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoFragmentViewModel.kt +++ b/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoFragmentViewModel.kt @@ -4,41 +4,73 @@ import androidx.lifecycle.ViewModel import androidx.lifecycle.viewModelScope import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.launch +import java.math.BigDecimal +import java.math.RoundingMode import java.time.LocalDate class SkontoFragmentViewModel : ViewModel() { val stateFlow: MutableStateFlow = MutableStateFlow( - SkontoFragmentContract.State.Ready( - isSkontoSectionActive = true, + createInitalState( + skontoAmount = BigDecimal("97"), + fullAmount = BigDecimal("100"), paymentInDays = 14, - discountValue = 3.0f, - skontoAmount = 97.0f, - discountDueDate = LocalDate.now(), - fullAmount = "100", - totalAmount = 97.0f, - totalDiscount = 3.0f, + isSkontoSectionActive = true, currency = "EUR", + discountDueDate = LocalDate.now() ) ) + private fun createInitalState( + skontoAmount: BigDecimal, + fullAmount: BigDecimal, + paymentInDays: Int, + isSkontoSectionActive: Boolean = true, + currency: String = "EUR", + discountDueDate: LocalDate, + ): SkontoFragmentContract.State.Ready { + + val totalAmount = if (isSkontoSectionActive) skontoAmount else fullAmount + val discount = calculateDiscount(skontoAmount, fullAmount) + + return SkontoFragmentContract.State.Ready( + isSkontoSectionActive = true, + paymentInDays = paymentInDays, + discountValue = discount, + skontoAmount = skontoAmount, + discountDueDate = discountDueDate, + fullAmount = fullAmount, + totalAmount = totalAmount, + currency = currency, + ) + } + fun onSkontoActiveChanged(newValue: Boolean) = viewModelScope.launch { val currentState = stateFlow.value as? SkontoFragmentContract.State.Ready ?: return@launch val totalAmount = if (newValue) currentState.skontoAmount else currentState.fullAmount + val discount = calculateDiscount(currentState.skontoAmount, currentState.fullAmount) + stateFlow.emit( currentState.copy( isSkontoSectionActive = newValue, + totalAmount = totalAmount, + discountValue = discount ) ) } - fun onSkontoAmountFieldChanged(newValue: String) = viewModelScope.launch { - val floatValue = newValue.replace(",", ".").toFloatOrNull() ?: 0f + fun onSkontoAmountFieldChanged(newValue: BigDecimal) = viewModelScope.launch { val currentState = stateFlow.value as? SkontoFragmentContract.State.Ready ?: return@launch + val discount = calculateDiscount(newValue, currentState.fullAmount) + val totalAmount = + if (currentState.isSkontoSectionActive) newValue else currentState.fullAmount + stateFlow.emit( currentState.copy( - skontoAmount = floatValue, + skontoAmount = newValue, + discountValue = discount, + totalAmount = totalAmount, ) ) } @@ -48,11 +80,30 @@ class SkontoFragmentViewModel : ViewModel() { stateFlow.emit(currentState.copy(discountDueDate = newDate)) } - fun onFullAmountFieldChanged(newValue: String) = viewModelScope.launch { + fun onFullAmountFieldChanged(newValue: BigDecimal) = viewModelScope.launch { val currentState = stateFlow.value as? SkontoFragmentContract.State.Ready ?: return@launch - stateFlow.emit(currentState.copy(fullAmount = newValue)) + val totalAmount = + if (currentState.isSkontoSectionActive) currentState.skontoAmount else newValue + val discount = currentState.discountValue + val skontoAmount = newValue.minus( + newValue.multiply( // full_amount - (full_amount * (discount / 100)) + discount.divide(BigDecimal("100"), 2, RoundingMode.HALF_UP) + ) + ) + + stateFlow.emit( + currentState.copy( + skontoAmount = skontoAmount, + fullAmount = newValue, + totalAmount = totalAmount + ) + ) } - private fun getCurrentTotalAmount(state: SkontoFragmentContract.State.Ready) = - if (state.isSkontoSectionActive) state.skontoAmount else state.fullAmount + private fun calculateDiscount(skontoAmount: BigDecimal, fullAmount: BigDecimal): BigDecimal { + if (fullAmount == BigDecimal.ZERO) return BigDecimal("100") + return BigDecimal.ONE + .minus(skontoAmount.divide(fullAmount, 4, RoundingMode.HALF_UP)) + .multiply(BigDecimal("100")) + } } \ No newline at end of file diff --git a/bank-sdk/sdk/src/main/res/navigation/gbs_nav_graph.xml b/bank-sdk/sdk/src/main/res/navigation/gbs_nav_graph.xml index 4a2b26a7e9..bcbc55f120 100644 --- a/bank-sdk/sdk/src/main/res/navigation/gbs_nav_graph.xml +++ b/bank-sdk/sdk/src/main/res/navigation/gbs_nav_graph.xml @@ -3,7 +3,8 @@ xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/capture_flow_nav_graph" - app:startDestination="@id/gbs_destination_capture_fragment"> + app:startDestination="@id/gbs_destination_capture_fragment">> + 0.01 | 1 -> 4 + * 12 -> 0.12 | 2 -> 4 + * 123 -> 1.23 | 3 -> 4 + * 1234 -> 12.34 | 4 -> 5 + * 12345 -> 123.45 | 5 -> 6 + * 123456 -> 1,234.56 | 6 -> 8 + * 1234567 -> 12,345.67 | 7 -> 9 + */ + + override fun originalToTransformed(offset: Int): Int { + val thousandSeparatorCount = + formatted.count { it == formattingSymbols.groupingSeparator } + + val lastIndex = when (source.length) { + in 1..3 -> formatted.length + 4 -> 5 + 5 -> 6 + 6 -> 8 + 7 -> 9 + else -> formatted.length + } + + return lastIndex + } + + /* + * 0.01 -> 1 | 4 -> 1 + * 0.12 -> 12 | 4 -> 2 + * 1.23 -> 123 | 4 -> 3 + * 12.34 -> 1234 | 5 -> 4 + * 123.45 -> 12345 | 6 -> 5 + * 1,234.56 -> 123456 | 8 -> 6 + * 12,345.67 -> 1234567 | 9 -> 7 + */ + + + override fun transformedToOriginal(offset: Int): Int { + val thousandSeparatorCount = + formatted.count { it == formattingSymbols.groupingSeparator } + + val lastIndex = when (formatted.length) { + in 1..4 -> source.length + 5 -> 4 + 6 -> 5 + 8 -> 6 + 9 -> 7 + else -> source.length + } + return lastIndex + } + } +} \ No newline at end of file diff --git a/capture-sdk/sdk/src/main/java/net/gini/android/capture/ui/components/textinput/amount/GiniAmountTextInput.kt b/capture-sdk/sdk/src/main/java/net/gini/android/capture/ui/components/textinput/amount/GiniAmountTextInput.kt new file mode 100644 index 0000000000..1861c675f1 --- /dev/null +++ b/capture-sdk/sdk/src/main/java/net/gini/android/capture/ui/components/textinput/amount/GiniAmountTextInput.kt @@ -0,0 +1,95 @@ +package net.gini.android.capture.ui.components.textinput.amount + +import android.content.res.Configuration +import android.util.Log +import androidx.compose.animation.AnimatedVisibility +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.text.KeyboardOptions +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.runtime.LaunchedEffect +import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.setValue +import androidx.compose.ui.Modifier +import androidx.compose.ui.res.stringResource +import androidx.compose.ui.text.input.ImeAction +import androidx.compose.ui.text.input.KeyboardType +import androidx.compose.ui.text.input.VisualTransformation +import androidx.compose.ui.tooling.preview.Preview +import androidx.compose.ui.unit.dp +import net.gini.android.capture.ui.components.textinput.GiniTextInput +import net.gini.android.capture.ui.components.textinput.GiniTextInputColors +import net.gini.android.capture.ui.theme.GiniTheme +import java.math.BigDecimal +import java.text.DecimalFormat +import java.text.NumberFormat + +@Composable +fun GiniAmountTextInput( + amount: BigDecimal, + label: String, + modifier: Modifier = Modifier, + onValueChange: (BigDecimal) -> Unit, + trailingContent: @Composable () -> Unit = {}, + enabled: Boolean = true, + decimalFormatter: DecimalFormatter = DecimalFormatter(), + colors: GiniTextInputColors = GiniTextInputColors.colors(), +) { + val parsedAmount = decimalFormatter.parseAmount(amount) + + var text by remember { mutableStateOf(parsedAmount) } + + LaunchedEffect(key1 = parsedAmount) { // we need to reset text if amount was changed only + text = parsedAmount + } + + GiniTextInput( + modifier = modifier, + text = text, + enabled = enabled, + keyboardOptions = KeyboardOptions( + keyboardType = KeyboardType.Number, + imeAction = ImeAction.Done, + ), + label = label, + onValueChange = { + text = decimalFormatter.textToDigits(it) // take only 7 digits + onValueChange(decimalFormatter.parseDigits(text)) + }, + trailingContent = trailingContent, + colors = colors, + visualTransformation = DecimalInputVisualTransformation(decimalFormatter = decimalFormatter), + ) +} + + +@Preview(showBackground = true) +@Composable +private fun GiniTextInputPreviewLight() { + GiniTextInputPreview() +} + +@Preview( + showBackground = true, + uiMode = Configuration.UI_MODE_NIGHT_YES +) +@Composable +private fun GiniTextInputPreviewDark() { + GiniTextInputPreview() +} + +@Composable +private fun GiniTextInputPreview() { + GiniTheme { + GiniAmountTextInput( + modifier = Modifier.padding(16.dp), + amount = BigDecimal("1234"), + label = "Label Text", + trailingContent = { }, + onValueChange = {} + ) + } +} diff --git a/capture-sdk/sdk/src/test/java/net/gini/android/capture/ui/components/textinput/amount/CustomOffsetMappingTest.kt b/capture-sdk/sdk/src/test/java/net/gini/android/capture/ui/components/textinput/amount/CustomOffsetMappingTest.kt new file mode 100644 index 0000000000..de17731312 --- /dev/null +++ b/capture-sdk/sdk/src/test/java/net/gini/android/capture/ui/components/textinput/amount/CustomOffsetMappingTest.kt @@ -0,0 +1,80 @@ +package net.gini.android.capture.ui.components.textinput.amount + +import org.junit.Assert +import org.junit.Test +import org.junit.runner.RunWith +import org.junit.runners.JUnit4 +import kotlin.random.Random + + +@RunWith(JUnit4::class) +class CustomOffsetMappingTest { + + @Test + fun testCustomOffsetMapping() { + + val testData = listOf( + PossibleCursorPosition("1", "0.01", 9, 1), + PossibleCursorPosition("12", "0.12", 9, 2), + PossibleCursorPosition("123", "1.23", 9, 3), + PossibleCursorPosition("1234", "12.34", 9, 4), + PossibleCursorPosition("12345", "123.45", 9, 5), + PossibleCursorPosition("123456", "1,234.56", 9, 6), + PossibleCursorPosition("1234567", "12,345.67", 9, 7), + + PossibleCursorPosition("1", "0.01", 8, 1), + PossibleCursorPosition("12", "0.12", 8, 2), + PossibleCursorPosition("123", "1.23", 8, 3), + PossibleCursorPosition("1234", "12.34", 8, 4), + PossibleCursorPosition("12345", "123.45", 8, 5), + PossibleCursorPosition("123456", "1,234.56", 8, 6), + PossibleCursorPosition("1234567", "12,345.67", 8, 7), + + PossibleCursorPosition("1", "0.01", 7, 1), + PossibleCursorPosition("12", "0.12", 7, 2), + PossibleCursorPosition("123", "1.23", 7, 3), + PossibleCursorPosition("1234", "12.34", 7, 4), + PossibleCursorPosition("12345", "123.45", 7, 5), + PossibleCursorPosition("123456", "1,234.56", 7, 5), + PossibleCursorPosition("1234567", "12,345.67", 7, 5), + + PossibleCursorPosition("1", "0.01", 6, 1), + PossibleCursorPosition("12", "0.12", 6, 2), + PossibleCursorPosition("123", "1.23", 6, 3), + PossibleCursorPosition("1234", "12.34", 6, 4), + PossibleCursorPosition("12345", "123.45", 6, 5), + PossibleCursorPosition("123456", "1,234.56", 6, 4), + PossibleCursorPosition("1234567", "12,345.67", 6, 5), + ) + + + testData.forEach { + + val mapping = DecimalInputVisualTransformation.CustomOffsetMapping( + source = it.source, + formatted = it.formatted + ) + + /*Assert.assertEquals( + "Invalid result for pair: ${it.source} -> ${it.formatted}. Offset: ${it.offset}", + mapping.originalToTransformed(Random.nextInt()), + formatted.length + ) + Assert.assertEquals( + "Invalid result for pair: ${it.source} -> ${it.formatted}. Offset: ${it.offset}", + it.transformedToOriginal, + mapping.transformedToOriginal(Random.nextInt()) + )*/ + // TODO Uncomment this to enable unit testing of cursor + } + } + + private class PossibleCursorPosition( + val source: String, + val formatted: String, + val offset: Int, + val transformedToOriginal: Int, + + ) + +} \ No newline at end of file From e506fdf02822decb7b3b34d7ec01e0bad311df64 Mon Sep 17 00:00:00 2001 From: ndubkov-distcotech Date: Tue, 16 Jul 2024 13:45:55 +0200 Subject: [PATCH 2/3] feat(bank-sdk): Skonto screen. Implement bottom navigation (#494) feat(bank-sdk): Skonto screen. Implement bottom navigation PP-640 --- .../exampleapp/ui/ConfigurationActivity.kt | 12 + .../exampleapp/ui/ConfigurationViewModel.kt | 7 + .../CustomSkontoNavigationBarBottomAdapter.kt | 54 ++++ .../sdk/exampleapp/ui/data/Configuration.kt | 4 + .../res/layout/activity_configuration.xml | 16 +- .../layout/custom_skonto_navigation_bar.xml | 104 +++++++ .../src/main/res/values/strings.xml | 1 + .../net/gini/android/bank/sdk/GiniBank.kt | 52 +++- .../android/bank/sdk/capture/Configuration.kt | 56 +++- .../bank/sdk/capture/skonto/SkontoFragment.kt | 285 +++++++++++++----- .../SkontoNavigationBarBottomAdapter.kt | 45 +++ .../src/main/res/navigation/gbs_nav_graph.xml | 3 +- .../ui/components/topbar/GiniTopBar.kt | 1 - 13 files changed, 540 insertions(+), 100 deletions(-) create mode 100644 bank-sdk/example-app/src/main/java/net/gini/android/bank/sdk/exampleapp/ui/adapters/CustomSkontoNavigationBarBottomAdapter.kt create mode 100644 bank-sdk/example-app/src/main/res/layout/custom_skonto_navigation_bar.xml create mode 100644 bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoNavigationBarBottomAdapter.kt diff --git a/bank-sdk/example-app/src/main/java/net/gini/android/bank/sdk/exampleapp/ui/ConfigurationActivity.kt b/bank-sdk/example-app/src/main/java/net/gini/android/bank/sdk/exampleapp/ui/ConfigurationActivity.kt index bba1f063eb..cd78ff0ecc 100644 --- a/bank-sdk/example-app/src/main/java/net/gini/android/bank/sdk/exampleapp/ui/ConfigurationActivity.kt +++ b/bank-sdk/example-app/src/main/java/net/gini/android/bank/sdk/exampleapp/ui/ConfigurationActivity.kt @@ -139,6 +139,9 @@ class ConfigurationActivity : AppCompatActivity() { binding.switchReviewScreenCustomBottomNavbar.isChecked = configuration.isReviewScreenCustomBottomNavBarEnabled + binding.switchSkontoCustomBottomNavbar.isChecked = + configuration.isSkontoCustomNavBarEnabled + // 12 enable image picker screens custom bottom navigation bar -> was implemented on iOS, not needed for Android // 13 enable onboarding screens at first launch @@ -369,6 +372,14 @@ class ConfigurationActivity : AppCompatActivity() { ) } + binding.switchSkontoCustomBottomNavbar.setOnCheckedChangeListener { _, isChecked -> + configurationViewModel.setConfiguration( + configurationViewModel.configurationFlow.value.copy( + isSkontoCustomNavBarEnabled = isChecked + ) + ) + } + // 12 enable image picker screens custom bottom navigation bar -> was implemented on iOS, not needed for Android // 13 enable onboarding screens at first launch @@ -421,6 +432,7 @@ class ConfigurationActivity : AppCompatActivity() { ) ) } + // 19 enable multi page in custom onboarding pages binding.switchCustomOnboardingMultiPage.setOnCheckedChangeListener { _, isChecked -> configurationViewModel.setConfiguration( diff --git a/bank-sdk/example-app/src/main/java/net/gini/android/bank/sdk/exampleapp/ui/ConfigurationViewModel.kt b/bank-sdk/example-app/src/main/java/net/gini/android/bank/sdk/exampleapp/ui/ConfigurationViewModel.kt index e2abeb843b..cc9dea3200 100644 --- a/bank-sdk/example-app/src/main/java/net/gini/android/bank/sdk/exampleapp/ui/ConfigurationViewModel.kt +++ b/bank-sdk/example-app/src/main/java/net/gini/android/bank/sdk/exampleapp/ui/ConfigurationViewModel.kt @@ -26,6 +26,7 @@ import net.gini.android.bank.sdk.exampleapp.ui.adapters.CustomOnButtonLoadingInd import net.gini.android.bank.sdk.exampleapp.ui.adapters.CustomOnboardingIllustrationAdapter import net.gini.android.bank.sdk.exampleapp.ui.adapters.CustomOnboardingNavigationBarBottomAdapter import net.gini.android.bank.sdk.exampleapp.ui.adapters.CustomReviewNavigationBarBottomAdapter +import net.gini.android.bank.sdk.exampleapp.ui.adapters.CustomSkontoNavigationBarBottomAdapter import net.gini.android.bank.sdk.exampleapp.ui.data.Configuration import net.gini.android.capture.GiniCaptureDebug import net.gini.android.capture.help.HelpItem @@ -275,6 +276,12 @@ class ConfigurationViewModel @Inject constructor( CustomDigitalInvoiceHelpNavigationBarBottomAdapter() } + if (configuration.isSkontoCustomNavBarEnabled) { + GiniBank.skontoNavigationBarBottomAdapter = CustomSkontoNavigationBarBottomAdapter() + } else { + GiniBank.skontoNavigationBarBottomAdapter = null + } + // 35 Digital invoice onboarding bottom navigation bar if (configuration.isDigitalInvoiceOnboardingBottomNavigationBarEnabled) { GiniBank.digitalInvoiceOnboardingNavigationBarBottomAdapter = diff --git a/bank-sdk/example-app/src/main/java/net/gini/android/bank/sdk/exampleapp/ui/adapters/CustomSkontoNavigationBarBottomAdapter.kt b/bank-sdk/example-app/src/main/java/net/gini/android/bank/sdk/exampleapp/ui/adapters/CustomSkontoNavigationBarBottomAdapter.kt new file mode 100644 index 0000000000..0cba099059 --- /dev/null +++ b/bank-sdk/example-app/src/main/java/net/gini/android/bank/sdk/exampleapp/ui/adapters/CustomSkontoNavigationBarBottomAdapter.kt @@ -0,0 +1,54 @@ +package net.gini.android.bank.sdk.exampleapp.ui.adapters + +import android.view.LayoutInflater +import android.view.View +import android.view.ViewGroup +import androidx.core.view.isVisible +import net.gini.android.bank.sdk.capture.skonto.SkontoNavigationBarBottomAdapter +import net.gini.android.bank.sdk.exampleapp.databinding.CustomSkontoNavigationBarBinding + +class CustomSkontoNavigationBarBottomAdapter : SkontoNavigationBarBottomAdapter { + + private var binding: CustomSkontoNavigationBarBinding? = null + + override fun setOnHelpClickListener(onClick: () -> Unit) { + binding?.gbsHelpBtn?.setOnClickListener { onClick() } + } + + override fun setOnBackClickListener(onClick: () -> Unit) { + binding?.gbsBackBtn?.setOnClickListener { onClick() } + } + + override fun setOnProceedClickListener(onClick: () -> Unit) { + binding?.gbsPay?.setOnClickListener { onClick() } + } + + override fun setProceedButtonEnabled(enabled: Boolean) { + binding?.gbsPay?.isEnabled = enabled + } + + override fun setTotalPriceText(text: String) { + binding?.priceTotal?.text = text + } + + override fun setDiscountLabelText(text: String) { + binding?.discountInfo?.text = text + } + + override fun onCreateView(container: ViewGroup): View { + binding = CustomSkontoNavigationBarBinding.inflate( + LayoutInflater.from(container.context), + container, + false + ) + return binding!!.root + } + + override fun setDiscountLabelVisible(visible: Boolean) { + binding?.discountInfo?.isVisible = visible + } + + override fun onDestroy() { + binding = null + } +} \ No newline at end of file diff --git a/bank-sdk/example-app/src/main/java/net/gini/android/bank/sdk/exampleapp/ui/data/Configuration.kt b/bank-sdk/example-app/src/main/java/net/gini/android/bank/sdk/exampleapp/ui/data/Configuration.kt index cc40c366fa..64c04492db 100644 --- a/bank-sdk/example-app/src/main/java/net/gini/android/bank/sdk/exampleapp/ui/data/Configuration.kt +++ b/bank-sdk/example-app/src/main/java/net/gini/android/bank/sdk/exampleapp/ui/data/Configuration.kt @@ -154,8 +154,12 @@ data class Configuration( // 37 Debug mode val isDebugModeEnabled: Boolean = true, + // 38 Is Allow Screenshots val isAllowScreenshotsEnabled: Boolean = true, + // 39 Skonto Custom bottom navigation + val isSkontoCustomNavBarEnabled: Boolean = false, + ) : Parcelable { companion object { diff --git a/bank-sdk/example-app/src/main/res/layout/activity_configuration.xml b/bank-sdk/example-app/src/main/res/layout/activity_configuration.xml index 845396ef52..b26e83aa42 100644 --- a/bank-sdk/example-app/src/main/res/layout/activity_configuration.xml +++ b/bank-sdk/example-app/src/main/res/layout/activity_configuration.xml @@ -208,13 +208,27 @@ android:text="@string/review_screen_custom_bottom_navbar_switch_label" /> + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/bank-sdk/example-app/src/main/res/values/strings.xml b/bank-sdk/example-app/src/main/res/values/strings.xml index 5ed91dbac2..1b59b68ffa 100644 --- a/bank-sdk/example-app/src/main/res/values/strings.xml +++ b/bank-sdk/example-app/src/main/res/values/strings.xml @@ -67,6 +67,7 @@ The custom bottom navigation bar is shown if \'Bottom navigation bar\' is also enabled. Camera screen custom bottom navigation bar Review screen custom bottom navigation bar + Skonto screen custom bottom navigation bar Onboarding screens at every launch Onboarding screens at first launch Overwrites \'Onboarding screens at first launch\'. diff --git a/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/GiniBank.kt b/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/GiniBank.kt index d86a1a582f..4b158e9a23 100644 --- a/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/GiniBank.kt +++ b/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/GiniBank.kt @@ -24,6 +24,7 @@ import net.gini.android.bank.sdk.capture.digitalinvoice.view.DefaultDigitalInvoi import net.gini.android.bank.sdk.capture.digitalinvoice.view.DefaultDigitalInvoiceOnboardingNavigationBarBottomAdapter import net.gini.android.bank.sdk.capture.digitalinvoice.view.DigitalInvoiceNavigationBarBottomAdapter import net.gini.android.bank.sdk.capture.digitalinvoice.view.DigitalInvoiceOnboardingNavigationBarBottomAdapter +import net.gini.android.bank.sdk.capture.skonto.SkontoNavigationBarBottomAdapter import net.gini.android.bank.sdk.error.AmountParsingException import net.gini.android.bank.sdk.pay.getBusinessIntent import net.gini.android.bank.sdk.pay.getRequestId @@ -101,6 +102,18 @@ object GiniBank { } get() = digitalInvoiceNavigationBarBottomAdapterInstance.viewAdapter + + internal var skontoNavigationBarBottomAdapterInstance: InjectedViewAdapterInstance? = + null + + var skontoNavigationBarBottomAdapter: SkontoNavigationBarBottomAdapter? + set(value) { + skontoNavigationBarBottomAdapterInstance = + value?.let { InjectedViewAdapterInstance(it) } + } + get() = skontoNavigationBarBottomAdapterInstance?.viewAdapter + + internal fun getCaptureConfiguration() = captureConfiguration /** @@ -316,18 +329,26 @@ object GiniBank { ): CancellationToken { giniCapture.let { capture -> check(capture != null) { "Capture feature is not configured. Call setCaptureConfiguration before starting the flow." } - return capture.createDocumentForImportedFiles(intent, context, object : AsyncCallback { - override fun onSuccess(result: Document) { - resultLauncher.launch(CaptureImportInput.Forward(result)) - } - - override fun onError(exception: ImportedFileValidationException?) { - resultLauncher.launch(CaptureImportInput.Error(exception?.validationError, exception?.message)) - } + return capture.createDocumentForImportedFiles( + intent, + context, + object : AsyncCallback { + override fun onSuccess(result: Document) { + resultLauncher.launch(CaptureImportInput.Forward(result)) + } + + override fun onError(exception: ImportedFileValidationException?) { + resultLauncher.launch( + CaptureImportInput.Error( + exception?.validationError, + exception?.message + ) + ) + } - override fun onCancelled() { - } - }) + override fun onCancelled() { + } + }) } } @@ -413,17 +434,18 @@ object GiniBank { /** * The document was processed successfully. */ - data class Success(val document: Document?): CreateDocumentFromImportedFileResult() + data class Success(val document: Document?) : CreateDocumentFromImportedFileResult() /** * Document processing returned an error. */ - data class Error(val error: ImportedFileValidationException?): CreateDocumentFromImportedFileResult() + data class Error(val error: ImportedFileValidationException?) : + CreateDocumentFromImportedFileResult() /** * Document processing was cancelled. */ - object Cancelled: CreateDocumentFromImportedFileResult() + object Cancelled : CreateDocumentFromImportedFileResult() } /** @@ -442,7 +464,7 @@ object GiniBank { return giniCapture?.createDocumentForImportedFiles( intent, context, - object: AsyncCallback { + object : AsyncCallback { override fun onSuccess(result: Document?) { callback(CreateDocumentFromImportedFileResult.Success(result)) } diff --git a/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/Configuration.kt b/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/Configuration.kt index 5c7e068888..74ff7418b5 100644 --- a/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/Configuration.kt +++ b/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/Configuration.kt @@ -1,5 +1,6 @@ package net.gini.android.bank.sdk.capture +import net.gini.android.bank.sdk.capture.skonto.SkontoNavigationBarBottomAdapter import net.gini.android.capture.DocumentImportEnabledFileTypes import net.gini.android.capture.EntryPoint import net.gini.android.capture.GiniCapture @@ -181,6 +182,11 @@ data class CaptureConfiguration( */ val reviewNavigationBarBottomAdapter: ReviewNavigationBarBottomAdapter? = null, + /** + * Set an adapter implementation to show a custom bottom navigation bar on the Skonto screen. + */ + val skontoNavigationBarBottomAdapter: SkontoNavigationBarBottomAdapter? = null, + /** * Set an adapter implementation to show a custom bottom navigation bar on the help screen. */ @@ -200,7 +206,7 @@ data class CaptureConfiguration( * * IMPORTANT: If you disallow screenshots and use the [CaptureFlowFragment] for launching the SDK in your activity, please clear the [android.view.WindowManager.LayoutParams.FLAG_SECURE] * on your activity's window after the SDK has finished to allow users to take screenshots of your app again. - */ + */ val allowScreenshots: Boolean = true ) @@ -231,15 +237,47 @@ internal fun GiniCapture.Builder.applyConfiguration(configuration: CaptureConfig }) } configuration.navigationBarTopAdapter?.let { setNavigationBarTopAdapter(it) } - configuration.onboardingAlignCornersIllustrationAdapter?.let { setOnboardingAlignCornersIllustrationAdapter(it) } - configuration.onboardingLightingIllustrationAdapter?.let { setOnboardingLightingIllustrationAdapter(it) } - configuration.onboardingMultiPageIllustrationAdapter?.let { setOnboardingMultiPageIllustrationAdapter(it) } - configuration.onboardingQRCodeIllustrationAdapter?.let { setOnboardingQRCodeIllustrationAdapter(it) } + configuration.onboardingAlignCornersIllustrationAdapter?.let { + setOnboardingAlignCornersIllustrationAdapter( + it + ) + } + configuration.onboardingLightingIllustrationAdapter?.let { + setOnboardingLightingIllustrationAdapter( + it + ) + } + configuration.onboardingMultiPageIllustrationAdapter?.let { + setOnboardingMultiPageIllustrationAdapter( + it + ) + } + configuration.onboardingQRCodeIllustrationAdapter?.let { + setOnboardingQRCodeIllustrationAdapter( + it + ) + } configuration.customLoadingIndicatorAdapter?.let { setLoadingIndicatorAdapter(it) } - configuration.onButtonLoadingIndicatorAdapter?.let { setOnButtonLoadingIndicatorAdapter(it) } - configuration.onboardingNavigationBarBottomAdapter?.let { setOnboardingNavigationBarBottomAdapter(it) } - configuration.cameraNavigationBarBottomAdapter?.let { setCameraNavigationBarBottomAdapter(it) } - configuration.reviewNavigationBarBottomAdapter?.let { setReviewBottomBarNavigationAdapter(it) } + configuration.onButtonLoadingIndicatorAdapter?.let { + setOnButtonLoadingIndicatorAdapter( + it + ) + } + configuration.onboardingNavigationBarBottomAdapter?.let { + setOnboardingNavigationBarBottomAdapter( + it + ) + } + configuration.cameraNavigationBarBottomAdapter?.let { + setCameraNavigationBarBottomAdapter( + it + ) + } + configuration.reviewNavigationBarBottomAdapter?.let { + setReviewBottomBarNavigationAdapter( + it + ) + } configuration.helpNavigationBarBottomAdapter?.let { setHelpNavigationBarBottomAdapter(it) } } } diff --git a/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoFragment.kt b/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoFragment.kt index df86a072a6..88c672ded0 100644 --- a/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoFragment.kt +++ b/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoFragment.kt @@ -6,6 +6,7 @@ import android.util.Log import android.view.LayoutInflater import android.view.View import android.view.ViewGroup +import android.widget.FrameLayout import androidx.compose.animation.AnimatedVisibility import androidx.compose.animation.core.animateFloatAsState import androidx.compose.foundation.background @@ -28,6 +29,7 @@ import androidx.compose.material.icons.automirrored.filled.KeyboardArrowRight import androidx.compose.material3.Card import androidx.compose.material3.CardDefaults import androidx.compose.material3.Icon +import androidx.compose.material3.IconButton import androidx.compose.material3.Scaffold import androidx.compose.material3.Text import androidx.compose.runtime.Composable @@ -44,6 +46,7 @@ import androidx.compose.ui.focus.onFocusChanged import androidx.compose.ui.graphics.RectangleShape import androidx.compose.ui.graphics.vector.rememberVectorPainter import androidx.compose.ui.platform.ComposeView +import androidx.compose.ui.platform.LocalContext import androidx.compose.ui.platform.LocalFocusManager import androidx.compose.ui.platform.ViewCompositionStrategy import androidx.compose.ui.res.painterResource @@ -52,8 +55,10 @@ import androidx.compose.ui.text.input.ImeAction import androidx.compose.ui.text.input.KeyboardType import androidx.compose.ui.tooling.preview.Preview import androidx.compose.ui.unit.dp +import androidx.compose.ui.viewinterop.AndroidView import androidx.fragment.app.Fragment import androidx.lifecycle.ViewModelProvider +import net.gini.android.bank.sdk.GiniBank import net.gini.android.bank.sdk.R import net.gini.android.bank.sdk.capture.skonto.colors.SkontoScreenColors import net.gini.android.bank.sdk.capture.skonto.colors.section.SkontoFooterSectionColors @@ -70,6 +75,7 @@ import net.gini.android.capture.ui.components.textinput.amount.GiniAmountTextInp import net.gini.android.capture.ui.components.topbar.GiniTopBar import net.gini.android.capture.ui.components.topbar.GiniTopBarColors import net.gini.android.capture.ui.theme.GiniTheme +import net.gini.android.capture.view.InjectedViewAdapterInstance import java.math.BigDecimal import java.math.RoundingMode import java.text.DecimalFormat @@ -78,11 +84,19 @@ import java.time.format.DateTimeFormatter class SkontoFragment : Fragment() { - val bottomNavBar = GiniCapture.getInstance().internal().navigationBarTopAdapterInstance + private val isBottomNavigationBarEnabled = + GiniCapture.getInstance().isBottomNavigationBarEnabled + + private val customBottomNavBarAdapter: InjectedViewAdapterInstance? = + GiniBank.skontoNavigationBarBottomAdapterInstance + private var customBottomNavigationBarView: View? = null + override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View { + customBottomNavigationBarView = + container?.let { customBottomNavBarAdapter?.viewAdapter?.onCreateView(it) } val viewModel = ViewModelProvider(requireActivity())[SkontoFragmentViewModel::class.java] @@ -92,6 +106,8 @@ class SkontoFragment : Fragment() { GiniTheme { ScreenContent( viewModel = viewModel, + isBottomNavigationBarEnabled = isBottomNavigationBarEnabled, + customBottomNavBarAdapter = customBottomNavBarAdapter, ) } } @@ -103,7 +119,9 @@ class SkontoFragment : Fragment() { private fun ScreenContent( viewModel: SkontoFragmentViewModel, modifier: Modifier = Modifier, - screenColorScheme: SkontoScreenColors = SkontoScreenColors.colors() + screenColorScheme: SkontoScreenColors = SkontoScreenColors.colors(), + isBottomNavigationBarEnabled: Boolean, + customBottomNavBarAdapter: InjectedViewAdapterInstance?, ) { val state by viewModel.stateFlow.collectAsState() ScreenStateContent( @@ -113,7 +131,12 @@ private fun ScreenContent( onDiscountSectionActiveChange = viewModel::onSkontoActiveChanged, onSkontoAmountChange = viewModel::onSkontoAmountFieldChanged, onDueDateChanged = viewModel::onSkontoDueDateChanged, - onFullAmountChange = viewModel::onFullAmountFieldChanged + onFullAmountChange = viewModel::onFullAmountFieldChanged, + isBottomNavigationBarEnabled = isBottomNavigationBarEnabled, + onBackClicked = {}, + onHelpClicked = {}, + customBottomNavBarAdapter = customBottomNavBarAdapter, + onProceedClicked = {} ) } @@ -124,6 +147,11 @@ private fun ScreenStateContent( onSkontoAmountChange: (BigDecimal) -> Unit, onFullAmountChange: (BigDecimal) -> Unit, onDueDateChanged: (LocalDate) -> Unit, + onBackClicked: () -> Unit, + onHelpClicked: () -> Unit, + onProceedClicked: () -> Unit, + isBottomNavigationBarEnabled: Boolean, + customBottomNavBarAdapter: InjectedViewAdapterInstance?, modifier: Modifier = Modifier, screenColorScheme: SkontoScreenColors = SkontoScreenColors.colors() ) { @@ -136,7 +164,12 @@ private fun ScreenStateContent( onDiscountSectionActiveChange = onDiscountSectionActiveChange, onDiscountAmountChange = onSkontoAmountChange, onDueDateChanged = onDueDateChanged, - onFullAmountChange = onFullAmountChange + onFullAmountChange = onFullAmountChange, + onBackClicked = onBackClicked, + onHelpClicked = onHelpClicked, + isBottomNavigationBarEnabled = isBottomNavigationBarEnabled, + customBottomNavBarAdapter = customBottomNavBarAdapter, + onProceedClicked = onProceedClicked, ) } @@ -144,11 +177,16 @@ private fun ScreenStateContent( @Composable private fun ScreenReadyState( + onBackClicked: () -> Unit, + onHelpClicked: () -> Unit, + onProceedClicked: () -> Unit, state: SkontoFragmentContract.State.Ready, onDiscountSectionActiveChange: (Boolean) -> Unit, onDiscountAmountChange: (BigDecimal) -> Unit, onDueDateChanged: (LocalDate) -> Unit, onFullAmountChange: (BigDecimal) -> Unit, + isBottomNavigationBarEnabled: Boolean, + customBottomNavBarAdapter: InjectedViewAdapterInstance?, modifier: Modifier = Modifier, screenColorScheme: SkontoScreenColors = SkontoScreenColors.colors(), ) { @@ -156,13 +194,26 @@ private fun ScreenReadyState( val scrollState = rememberScrollState() Scaffold(modifier = modifier, containerColor = screenColorScheme.backgroundColor, - topBar = { TopAppBar(colors = screenColorScheme.topAppBarColors) }, + topBar = { + TopAppBar( + isBottomNavigationBarEnabled = isBottomNavigationBarEnabled, + colors = screenColorScheme.topAppBarColors, + onBackClicked = onBackClicked, + onHelpClicked = onHelpClicked, + ) + }, bottomBar = { FooterSection( colors = screenColorScheme.footerSectionColors, discountValue = state.discountValue, totalAmount = state.totalAmount, - currency = state.currency + currency = state.currency, + isBottomNavigationBarEnabled = isBottomNavigationBarEnabled, + onBackClicked = onBackClicked, + onHelpClicked = onHelpClicked, + customBottomNavBarAdapter = customBottomNavBarAdapter, + onProceedClicked = onProceedClicked, + isSkontoSectionActive = state.isSkontoSectionActive ) }) { Column( @@ -200,7 +251,10 @@ private fun ScreenReadyState( @Composable private fun TopAppBar( + onBackClicked: () -> Unit, + onHelpClicked: () -> Unit, modifier: Modifier = Modifier, + isBottomNavigationBarEnabled: Boolean, colors: GiniTopBarColors, ) { GiniTopBar( @@ -208,21 +262,50 @@ private fun TopAppBar( colors = colors, title = stringResource(id = R.string.gbs_skonto_screen_title), navigationIcon = { - Icon( - modifier = Modifier.padding(16.dp), - painter = rememberVectorPainter(image = Icons.AutoMirrored.Default.ArrowBack), - contentDescription = null, - ) + AnimatedVisibility(visible = !isBottomNavigationBarEnabled) { + NavigationActionBack(onClick = onBackClicked) + } }, actions = { - Icon( - modifier = Modifier.padding(16.dp), - painter = painterResource(net.gini.android.capture.R.drawable.gc_help_icon), - contentDescription = null, - ) + AnimatedVisibility(visible = !isBottomNavigationBarEnabled) { + NavigationActionHelp(onClick = onHelpClicked) + } }) } +@Composable +private fun NavigationActionBack( + onClick: () -> Unit, + modifier: Modifier = Modifier, +) { + IconButton( + modifier = modifier, + onClick = onClick + ) { + Icon( + painter = rememberVectorPainter(image = Icons.AutoMirrored.Default.ArrowBack), + contentDescription = null, + ) + } +} + +@Composable +private fun NavigationActionHelp( + onClick: () -> Unit, + modifier: Modifier = Modifier, +) { + IconButton( + modifier = modifier, + onClick = onClick + ) { + Icon( + modifier = Modifier, + painter = painterResource(net.gini.android.capture.R.drawable.gc_help_icon), + contentDescription = null, + ) + } +} + @Composable private fun YourInvoiceScanSection( modifier: Modifier = Modifier, @@ -390,10 +473,12 @@ private fun SkontoSection( } if (isDatePickerVisible) { - GiniDatePickerDialog(onDismissRequest = { isDatePickerVisible = false }, onSaved = { - isDatePickerVisible = false - onDueDateChanged(it) - }) + GiniDatePickerDialog( + onDismissRequest = { isDatePickerVisible = false }, + onSaved = { + isDatePickerVisible = false + onDueDateChanged(it) + }) } } @@ -496,66 +581,117 @@ private fun FooterSection( discountValue: BigDecimal, currency: String, colors: SkontoFooterSectionColors, + isBottomNavigationBarEnabled: Boolean, + isSkontoSectionActive: Boolean, + onBackClicked: () -> Unit, + onHelpClicked: () -> Unit, + onProceedClicked: () -> Unit, modifier: Modifier = Modifier, + customBottomNavBarAdapter: InjectedViewAdapterInstance?, ) { - Card( - modifier = modifier.fillMaxWidth(), - shape = RectangleShape, - colors = CardDefaults.cardColors(containerColor = colors.cardBackgroundColor) - ) { - Column( - modifier = Modifier.padding(16.dp) + val animatedTotalAmount by animateFloatAsState( + targetValue = totalAmount.toFloat(), label = "totalAmount" + ) + val animatedDiscountAmount by animateFloatAsState( + targetValue = discountValue.toFloat(), label = "discountAmount" + ) + val totalPriceText = "${currencyFormatterWithoutSymbol().format(animatedTotalAmount)} $currency" + val discountLabelText = stringResource( + id = R.string.gbs_skonto_section_footer_label_discount, + animatedDiscountAmount.formatAsDiscountPercentage() + ) + val proceedEnabled by remember { mutableStateOf(true) } + + if (customBottomNavBarAdapter != null) { + val ctx = LocalContext.current + AndroidView(factory = { + customBottomNavBarAdapter.viewAdapter.onCreateView(FrameLayout(ctx)) + }, update = { + with(customBottomNavBarAdapter.viewAdapter) { + setTotalPriceText(totalPriceText) + setProceedButtonEnabled(proceedEnabled) // TODO Integrate validation + setOnHelpClickListener(onHelpClicked) + setOnBackClickListener(onBackClicked) + setDiscountLabelText(discountLabelText) + setDiscountLabelVisible(isSkontoSectionActive) + setOnProceedClickListener(onProceedClicked) + } + }) + } else { + Card( + modifier = modifier.fillMaxWidth(), + shape = RectangleShape, + colors = CardDefaults.cardColors(containerColor = colors.cardBackgroundColor) ) { - Text( - text = stringResource(id = R.string.gbs_skonto_section_footer_title), - style = GiniTheme.typography.body1, - color = colors.titleTextColor, - ) - Row( - modifier = Modifier - .fillMaxWidth() - .padding(top = 4.dp), - horizontalArrangement = Arrangement.Start, - verticalAlignment = Alignment.CenterVertically, - ) { - val animatedTotalAmount by animateFloatAsState( - targetValue = totalAmount.toFloat(), label = "totalAmount" - ) - Text( - text = "${currencyFormatterWithoutSymbol().format(animatedTotalAmount)} $currency", - style = GiniTheme.typography.headline5, - color = colors.amountTextColor, - ) - Box( - modifier = Modifier - .height(IntrinsicSize.Min) - .padding(horizontal = 12.dp) - .background( - colors.discountLabelColorScheme.backgroundColor, - RoundedCornerShape(4.dp) - ), + Column { + Column( + modifier = Modifier.padding(start = 20.dp, end = 20.dp, top = 20.dp) ) { - val animatedDiscountAmount by animateFloatAsState( - targetValue = discountValue.toFloat(), label = "discountAmount" - ) - Text( - modifier = Modifier.padding(vertical = 4.dp, horizontal = 8.dp), - text = stringResource( - id = R.string.gbs_skonto_section_footer_label_discount, - animatedDiscountAmount.formatAsDiscountPercentage() - ), - style = GiniTheme.typography.caption1, - color = colors.discountLabelColorScheme.textColor, + text = stringResource(id = R.string.gbs_skonto_section_footer_title), + style = GiniTheme.typography.body1, + color = colors.titleTextColor, + ) + Row( + modifier = Modifier + .fillMaxWidth() + .padding(top = 4.dp), + horizontalArrangement = Arrangement.Start, + verticalAlignment = Alignment.CenterVertically, + ) { + Text( + text = totalPriceText, + style = GiniTheme.typography.headline5, + color = colors.amountTextColor, + ) + AnimatedVisibility(visible = isSkontoSectionActive) { + Box( + modifier = Modifier + .height(IntrinsicSize.Min) + .padding(horizontal = 12.dp) + .background( + colors.discountLabelColorScheme.backgroundColor, + RoundedCornerShape(4.dp) + ), + ) { + Text( + modifier = Modifier.padding(vertical = 4.dp, horizontal = 8.dp), + text = discountLabelText, + style = GiniTheme.typography.caption1, + color = colors.discountLabelColorScheme.textColor, + ) + } + } + } + } + val buttonPaddingHorizontal = if (isBottomNavigationBarEnabled) 0.dp else 20.dp + Row( + modifier = Modifier.fillMaxWidth(), + verticalAlignment = Alignment.CenterVertically, + horizontalArrangement = Arrangement.SpaceEvenly + ) { + AnimatedVisibility(visible = isBottomNavigationBarEnabled) { + NavigationActionBack( + modifier = Modifier.padding(horizontal = 4.dp), + onClick = onBackClicked + ) + } + GiniButton( + modifier = Modifier + .weight(0.1f) + .padding(horizontal = buttonPaddingHorizontal), + text = stringResource(id = R.string.gbs_skonto_section_footer_continue_button_text), + onClick = onProceedClicked, + giniButtonColors = colors.continueButtonColors ) + AnimatedVisibility(visible = isBottomNavigationBarEnabled) { + NavigationActionHelp( + modifier = Modifier.padding(horizontal = 4.dp), + onClick = onHelpClicked + ) + } } } - GiniButton( - modifier = Modifier.fillMaxWidth(), - text = stringResource(id = R.string.gbs_skonto_section_footer_continue_button_text), - onClick = { /*TODO*/ }, - giniButtonColors = colors.continueButtonColors - ) } } } @@ -585,6 +721,11 @@ private fun ScreenReadyStatePreview() { onDiscountAmountChange = {}, onDueDateChanged = {}, onFullAmountChange = {}, + onHelpClicked = {}, + onBackClicked = {}, + isBottomNavigationBarEnabled = false, + onProceedClicked = {}, + customBottomNavBarAdapter = null ) } } diff --git a/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoNavigationBarBottomAdapter.kt b/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoNavigationBarBottomAdapter.kt new file mode 100644 index 0000000000..52a95e7b3c --- /dev/null +++ b/bank-sdk/sdk/src/main/java/net/gini/android/bank/sdk/capture/skonto/SkontoNavigationBarBottomAdapter.kt @@ -0,0 +1,45 @@ +package net.gini.android.bank.sdk.capture.skonto + +import net.gini.android.capture.view.InjectedViewAdapter + +interface SkontoNavigationBarBottomAdapter : InjectedViewAdapter { + + /** + * Set the click listener for the help button. + * + * @param onClick the click function for the help button + */ + fun setOnHelpClickListener(onClick: () -> Unit) + + /** + * Set the click listener for the back button. + * + * @param onClick the click function for the back button + */ + fun setOnBackClickListener(onClick: () -> Unit) + + /** + * Set the click listener for the proceed button. + * + * @param listener the click listener for the button + */ + fun setOnProceedClickListener(onClick: () -> Unit) + + /** + * Enable or disable the proceed button. + * + * @param enabled for enabling or disabling the button + */ + fun setProceedButtonEnabled(enabled: Boolean) + + /** + * Set the total price. + * + * @param text price string with currency symbol + */ + fun setTotalPriceText(text: String) + + fun setDiscountLabelVisible(visible: Boolean) + + fun setDiscountLabelText(text: String) +} \ No newline at end of file diff --git a/bank-sdk/sdk/src/main/res/navigation/gbs_nav_graph.xml b/bank-sdk/sdk/src/main/res/navigation/gbs_nav_graph.xml index bcbc55f120..4a2b26a7e9 100644 --- a/bank-sdk/sdk/src/main/res/navigation/gbs_nav_graph.xml +++ b/bank-sdk/sdk/src/main/res/navigation/gbs_nav_graph.xml @@ -3,8 +3,7 @@ xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/capture_flow_nav_graph" - app:startDestination="@id/gbs_destination_capture_fragment">> - + app:startDestination="@id/gbs_destination_capture_fragment"> Date: Wed, 17 Jul 2024 14:05:23 +0200 Subject: [PATCH 3/3] feat(bank-sdk): Skonto screen. Set skonto as start destination PP-403 --- bank-sdk/sdk/src/main/res/navigation/gbs_nav_graph.xml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/bank-sdk/sdk/src/main/res/navigation/gbs_nav_graph.xml b/bank-sdk/sdk/src/main/res/navigation/gbs_nav_graph.xml index 4a2b26a7e9..de9279a862 100644 --- a/bank-sdk/sdk/src/main/res/navigation/gbs_nav_graph.xml +++ b/bank-sdk/sdk/src/main/res/navigation/gbs_nav_graph.xml @@ -3,7 +3,7 @@ xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/capture_flow_nav_graph" - app:startDestination="@id/gbs_destination_capture_fragment"> + app:startDestination="@id/gbs_destination_skonto_fragment">