Showing suggestions for payment reference based on previous payments to this account / IBAN
This commit is contained in:
parent
a41a9bd13a
commit
4761876cf2
|
@ -9,6 +9,7 @@ import androidx.compose.ui.Alignment
|
||||||
import androidx.compose.ui.Modifier
|
import androidx.compose.ui.Modifier
|
||||||
import androidx.compose.ui.text.input.ImeAction
|
import androidx.compose.ui.text.input.ImeAction
|
||||||
import androidx.compose.ui.text.input.KeyboardType
|
import androidx.compose.ui.text.input.KeyboardType
|
||||||
|
import androidx.compose.ui.text.style.TextAlign
|
||||||
import androidx.compose.ui.text.style.TextOverflow
|
import androidx.compose.ui.text.style.TextOverflow
|
||||||
import androidx.compose.ui.unit.dp
|
import androidx.compose.ui.unit.dp
|
||||||
import kotlinx.coroutines.Dispatchers
|
import kotlinx.coroutines.Dispatchers
|
||||||
|
@ -30,6 +31,8 @@ private val uiState = DI.uiState
|
||||||
|
|
||||||
private val bankingService = DI.bankingService
|
private val bankingService = DI.bankingService
|
||||||
|
|
||||||
|
private val formatUtil = DI.formatUtil
|
||||||
|
|
||||||
@Composable
|
@Composable
|
||||||
fun TransferMoneyDialog(
|
fun TransferMoneyDialog(
|
||||||
onDismiss: () -> Unit,
|
onDismiss: () -> Unit,
|
||||||
|
@ -130,6 +133,7 @@ fun TransferMoneyDialog(
|
||||||
dropdownMaxHeight = 350.dp,
|
dropdownMaxHeight = 350.dp,
|
||||||
minTextLengthForSearch = 0,
|
minTextLengthForSearch = 0,
|
||||||
onEnteredTextChanged = { recipientName = it },
|
onEnteredTextChanged = { recipientName = it },
|
||||||
|
getItemTitle = { suggestion -> suggestion.name },
|
||||||
onSelectedItemChanged = {
|
onSelectedItemChanged = {
|
||||||
if (it != null) {
|
if (it != null) {
|
||||||
recipientName = it.name
|
recipientName = it.name
|
||||||
|
@ -179,13 +183,20 @@ fun TransferMoneyDialog(
|
||||||
|
|
||||||
Spacer(modifier = Modifier.height(verticalSpace))
|
Spacer(modifier = Modifier.height(verticalSpace))
|
||||||
|
|
||||||
OutlinedTextField(
|
AutocompleteTextFieldNew(
|
||||||
value = paymentReference,
|
"Verwendungszweck",
|
||||||
onValueChange = { paymentReference = it },
|
dropdownMaxHeight = 250.dp,
|
||||||
label = { Text("Verwendungszweck") },
|
minTextLengthForSearch = 0,
|
||||||
modifier = Modifier.fillMaxWidth(),
|
onEnteredTextChanged = { paymentReference = it },
|
||||||
keyboardOptions = KeyboardOptions(imeAction = ImeAction.Next)
|
onSelectedItemChanged = { paymentReference = it?.reference ?: "" },
|
||||||
)
|
fetchSuggestions = { recipientFinder.findPaymentDataForIban(recipientAccountIdentifier) }
|
||||||
|
) { paymentDataSuggestion ->
|
||||||
|
Row(Modifier.fillMaxWidth(), verticalAlignment = Alignment.CenterVertically) {
|
||||||
|
Text(formatUtil.formatAmount(paymentDataSuggestion.amount, paymentDataSuggestion.currency), Modifier.widthIn(min = 60.dp), textAlign = TextAlign.End)
|
||||||
|
|
||||||
|
Text(paymentDataSuggestion.reference, Modifier.padding(start = 6.dp), maxLines = 1, overflow = TextOverflow.Ellipsis)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Row(Modifier.padding(top = verticalSpace), verticalAlignment = Alignment.CenterVertically) {
|
Row(Modifier.padding(top = verticalSpace), verticalAlignment = Alignment.CenterVertically) {
|
||||||
Switch(
|
Switch(
|
||||||
|
|
|
@ -0,0 +1,11 @@
|
||||||
|
package net.codinux.banking.ui.model
|
||||||
|
|
||||||
|
import kotlinx.datetime.LocalDate
|
||||||
|
import net.codinux.banking.client.model.Amount
|
||||||
|
|
||||||
|
data class PaymentDataSuggestion(
|
||||||
|
val reference: String,
|
||||||
|
val amount: Amount,
|
||||||
|
val currency: String,
|
||||||
|
val valueDate: LocalDate // only needed for sorting
|
||||||
|
)
|
|
@ -1,12 +1,16 @@
|
||||||
package net.codinux.banking.ui.service
|
package net.codinux.banking.ui.service
|
||||||
|
|
||||||
import net.codinux.banking.client.model.AccountTransaction
|
import net.codinux.banking.client.model.AccountTransaction
|
||||||
|
import net.codinux.banking.client.model.Amount
|
||||||
|
import net.codinux.banking.ui.model.PaymentDataSuggestion
|
||||||
import net.codinux.banking.ui.model.RecipientSuggestion
|
import net.codinux.banking.ui.model.RecipientSuggestion
|
||||||
|
|
||||||
class RecipientFinder(private val bankFinder: BankFinder) {
|
class RecipientFinder(private val bankFinder: BankFinder) {
|
||||||
|
|
||||||
private var availableRecipients: Set<RecipientSuggestion> = emptySet()
|
private var availableRecipients: Set<RecipientSuggestion> = emptySet()
|
||||||
|
|
||||||
|
private var transactionsByIban: Map<String, List<PaymentDataSuggestion>> = emptyMap()
|
||||||
|
|
||||||
|
|
||||||
fun findRecipients(query: String): Collection<RecipientSuggestion> {
|
fun findRecipients(query: String): Collection<RecipientSuggestion> {
|
||||||
if (query.isBlank()) {
|
if (query.isBlank()) {
|
||||||
|
@ -38,6 +42,14 @@ class RecipientFinder(private val bankFinder: BankFinder) {
|
||||||
.onEach { // do this after toSet() so that we don't have to call findBankByBicOrIban() for all the duplicates
|
.onEach { // do this after toSet() so that we don't have to call findBankByBicOrIban() for all the duplicates
|
||||||
it.bankInfo = bankFinder.findBankByBicOrIban(it.bankIdentifier, it.accountIdentifier)
|
it.bankInfo = bankFinder.findBankByBicOrIban(it.bankIdentifier, it.accountIdentifier)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
transactionsByIban = transactions.filter { it.otherPartyAccountId != null }.groupBy { it.otherPartyAccountId!! }
|
||||||
|
.mapValues { it.value.map {
|
||||||
|
PaymentDataSuggestion(it.reference, Amount(it.amount.amount.replace("-", "")), it.currency, it.valueDate)
|
||||||
|
}.toSet().sortedByDescending { it.valueDate } }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun findPaymentDataForIban(iban: String): Collection<PaymentDataSuggestion> =
|
||||||
|
transactionsByIban[iban] ?: emptySet()
|
||||||
|
|
||||||
}
|
}
|
Loading…
Reference in New Issue