card capture renew
This commit is contained in:
parent
283d2d64ce
commit
79ab7c0bae
@ -12,20 +12,18 @@ fun AmountRoute(
|
|||||||
onBack: () -> Unit,
|
onBack: () -> Unit,
|
||||||
onNavigateCardWaiting: () -> Unit
|
onNavigateCardWaiting: () -> Unit
|
||||||
) {
|
) {
|
||||||
val canGoBack = !action.equals("Sale", ignoreCase = true)
|
|
||||||
|
|
||||||
AmountScreen(
|
AmountScreen(
|
||||||
onBackClick = onBack,
|
onBackClick = onBack,
|
||||||
// onNextClick = { amount ->
|
onChargeClick = { amount ->
|
||||||
// sharedViewModel.amount.value = amount
|
sharedViewModel.amount.value = amount
|
||||||
// sharedViewModel.setAmountExist(true)
|
sharedViewModel.setAmountExist(true)
|
||||||
// sharedViewModel.setCardDataExist(false)
|
sharedViewModel.setCardDataExist(false)
|
||||||
// sharedViewModel.setTransMenu(null)
|
sharedViewModel.setTransMenu(null)
|
||||||
// sharedViewModel.transactionsType.value = TransactionsType.SALE
|
sharedViewModel.transactionsType.value = TransactionsType.SALE
|
||||||
// sharedViewModel.processCode.value =
|
sharedViewModel.processCode.value =
|
||||||
// ProcessCode.SALE_PURCHASE + ProcessCode.SMART + ProcessCode.TO_ACCOUNT
|
ProcessCode.SALE_PURCHASE + ProcessCode.SMART + ProcessCode.TO_ACCOUNT
|
||||||
//
|
|
||||||
// onNavigateCardWaiting()
|
onNavigateCardWaiting()
|
||||||
// }
|
}
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,8 +1,9 @@
|
|||||||
package com.mob.utsmyanmar.ui.cardwaiting
|
package com.mob.utsmyanmar.ui.cardwaiting
|
||||||
|
|
||||||
import androidx.activity.compose.BackHandler
|
import androidx.activity.compose.BackHandler
|
||||||
import androidx.compose.foundation.Image
|
import androidx.compose.foundation.BorderStroke
|
||||||
import androidx.compose.foundation.background
|
import androidx.compose.foundation.background
|
||||||
|
import androidx.compose.foundation.clickable
|
||||||
import androidx.compose.foundation.layout.Arrangement
|
import androidx.compose.foundation.layout.Arrangement
|
||||||
import androidx.compose.foundation.layout.Box
|
import androidx.compose.foundation.layout.Box
|
||||||
import androidx.compose.foundation.layout.Column
|
import androidx.compose.foundation.layout.Column
|
||||||
@ -11,18 +12,22 @@ import androidx.compose.foundation.layout.Spacer
|
|||||||
import androidx.compose.foundation.layout.fillMaxSize
|
import androidx.compose.foundation.layout.fillMaxSize
|
||||||
import androidx.compose.foundation.layout.fillMaxWidth
|
import androidx.compose.foundation.layout.fillMaxWidth
|
||||||
import androidx.compose.foundation.layout.height
|
import androidx.compose.foundation.layout.height
|
||||||
|
import androidx.compose.foundation.layout.navigationBarsPadding
|
||||||
import androidx.compose.foundation.layout.padding
|
import androidx.compose.foundation.layout.padding
|
||||||
import androidx.compose.foundation.layout.size
|
import androidx.compose.foundation.layout.size
|
||||||
|
import androidx.compose.foundation.layout.statusBarsPadding
|
||||||
|
import androidx.compose.foundation.layout.width
|
||||||
|
import androidx.compose.foundation.shape.CircleShape
|
||||||
import androidx.compose.foundation.shape.RoundedCornerShape
|
import androidx.compose.foundation.shape.RoundedCornerShape
|
||||||
import androidx.compose.material3.Button
|
import androidx.compose.material.icons.Icons
|
||||||
import androidx.compose.material3.ButtonDefaults
|
import androidx.compose.material.icons.rounded.KeyboardArrowLeft
|
||||||
import androidx.compose.material3.CenterAlignedTopAppBar
|
import androidx.compose.material.icons.rounded.Wifi
|
||||||
import androidx.compose.material3.ExperimentalMaterial3Api
|
import androidx.compose.material3.CircularProgressIndicator
|
||||||
|
import androidx.compose.material3.HorizontalDivider
|
||||||
import androidx.compose.material3.Icon
|
import androidx.compose.material3.Icon
|
||||||
import androidx.compose.material3.IconButton
|
import androidx.compose.material3.IconButton
|
||||||
import androidx.compose.material3.Scaffold
|
import androidx.compose.material3.Surface
|
||||||
import androidx.compose.material3.Text
|
import androidx.compose.material3.Text
|
||||||
import androidx.compose.material3.TopAppBarDefaults
|
|
||||||
import androidx.compose.runtime.Composable
|
import androidx.compose.runtime.Composable
|
||||||
import androidx.compose.runtime.DisposableEffect
|
import androidx.compose.runtime.DisposableEffect
|
||||||
import androidx.compose.runtime.LaunchedEffect
|
import androidx.compose.runtime.LaunchedEffect
|
||||||
@ -30,23 +35,22 @@ import androidx.compose.runtime.collectAsState
|
|||||||
import androidx.compose.runtime.getValue
|
import androidx.compose.runtime.getValue
|
||||||
import androidx.compose.ui.Alignment
|
import androidx.compose.ui.Alignment
|
||||||
import androidx.compose.ui.Modifier
|
import androidx.compose.ui.Modifier
|
||||||
|
import androidx.compose.ui.draw.alpha
|
||||||
|
import androidx.compose.ui.draw.rotate
|
||||||
import androidx.compose.ui.res.painterResource
|
import androidx.compose.ui.res.painterResource
|
||||||
import androidx.compose.ui.text.font.FontWeight
|
import androidx.compose.ui.text.font.FontWeight
|
||||||
import androidx.compose.ui.text.style.TextAlign
|
import androidx.compose.ui.text.style.TextAlign
|
||||||
import androidx.compose.ui.unit.dp
|
import androidx.compose.ui.unit.dp
|
||||||
import androidx.compose.ui.unit.sp
|
import androidx.compose.ui.unit.sp
|
||||||
import com.mob.utsmyanmar.R
|
import com.mob.utsmyanmar.R
|
||||||
import com.mob.utsmyanmar.ui.theme.Black
|
import com.mob.utsmyanmar.ui.preview.P2Preview
|
||||||
import com.mob.utsmyanmar.ui.theme.Primary
|
import com.mob.utsmyanmar.ui.theme.Color
|
||||||
import com.mob.utsmyanmar.ui.theme.White
|
|
||||||
|
|
||||||
@OptIn(ExperimentalMaterial3Api::class)
|
|
||||||
@Composable
|
@Composable
|
||||||
fun CardWaitingScreen(
|
fun CardWaitingScreen(
|
||||||
viewModel: CardWaitingViewModel,
|
viewModel: CardWaitingViewModel,
|
||||||
|
amount: String,
|
||||||
onManualEntry: () -> Unit,
|
onManualEntry: () -> Unit,
|
||||||
onProcessingCard: () -> Unit,
|
|
||||||
onTimeout: () -> Unit,
|
|
||||||
onBack: () -> Unit,
|
onBack: () -> Unit,
|
||||||
onMain: () -> Unit
|
onMain: () -> Unit
|
||||||
) {
|
) {
|
||||||
@ -56,10 +60,10 @@ fun CardWaitingScreen(
|
|||||||
viewModel.events.collect { event ->
|
viewModel.events.collect { event ->
|
||||||
when (event) {
|
when (event) {
|
||||||
CardWaitingEvent.GoManualEntry -> onManualEntry()
|
CardWaitingEvent.GoManualEntry -> onManualEntry()
|
||||||
CardWaitingEvent.GoProcessingCard -> onProcessingCard()
|
|
||||||
CardWaitingEvent.GoTimeout -> onTimeout()
|
|
||||||
CardWaitingEvent.GoMain -> onMain()
|
CardWaitingEvent.GoMain -> onMain()
|
||||||
CardWaitingEvent.GoBack -> onBack()
|
CardWaitingEvent.GoBack -> onBack()
|
||||||
|
CardWaitingEvent.GoProcessingCard,
|
||||||
|
CardWaitingEvent.GoTimeout -> Unit
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -78,160 +82,360 @@ fun CardWaitingScreen(
|
|||||||
viewModel.onBackPressed()
|
viewModel.onBackPressed()
|
||||||
}
|
}
|
||||||
|
|
||||||
Scaffold(
|
CardWaitingScreenContent(
|
||||||
topBar = {
|
amount = amount,
|
||||||
CenterAlignedTopAppBar(
|
uiState = uiState,
|
||||||
title = {
|
onBackClick = viewModel::onBackPressed,
|
||||||
Text(
|
onManualEntryClick = viewModel::onManualEntryClick
|
||||||
text = "CARD CAPTURE",
|
|
||||||
color = White,
|
|
||||||
fontSize = 16.sp,
|
|
||||||
fontWeight = FontWeight.SemiBold
|
|
||||||
)
|
|
||||||
},
|
|
||||||
navigationIcon = {
|
|
||||||
if (uiState.canGoBack) {
|
|
||||||
IconButton(onClick = onBack) {
|
|
||||||
Icon(
|
|
||||||
painter = painterResource(R.drawable.ic_left_arrow),
|
|
||||||
contentDescription = "Back",
|
|
||||||
tint = White
|
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
|
||||||
},
|
|
||||||
colors = TopAppBarDefaults.topAppBarColors(
|
|
||||||
containerColor = Primary
|
|
||||||
)
|
|
||||||
)
|
|
||||||
},
|
|
||||||
containerColor = White
|
|
||||||
) { paddingValues ->
|
|
||||||
Column(
|
|
||||||
modifier = Modifier
|
|
||||||
.fillMaxSize()
|
|
||||||
.padding(paddingValues)
|
|
||||||
.background(White)
|
|
||||||
) {
|
|
||||||
|
|
||||||
Box(
|
@Composable
|
||||||
modifier = Modifier
|
private fun CardWaitingScreenContent(
|
||||||
.fillMaxSize()
|
amount: String,
|
||||||
.background(
|
uiState: CardWaitingUiState,
|
||||||
color = Primary,
|
onBackClick: () -> Unit,
|
||||||
shape = RoundedCornerShape(topStart = 18.dp, topEnd = 18.dp)
|
onManualEntryClick: () -> Unit
|
||||||
)
|
|
||||||
.padding(18.dp)
|
|
||||||
) {
|
) {
|
||||||
Column(
|
Column(
|
||||||
modifier = Modifier.fillMaxSize(),
|
modifier = Modifier
|
||||||
|
.fillMaxSize()
|
||||||
|
.background(Color.IvoryBeige)
|
||||||
|
.statusBarsPadding()
|
||||||
|
.navigationBarsPadding()
|
||||||
|
.padding(horizontal = 20.dp),
|
||||||
horizontalAlignment = Alignment.CenterHorizontally
|
horizontalAlignment = Alignment.CenterHorizontally
|
||||||
) {
|
) {
|
||||||
Box(
|
Box(
|
||||||
modifier = Modifier
|
modifier = Modifier
|
||||||
.fillMaxWidth()
|
.fillMaxWidth()
|
||||||
.background(
|
.height(54.dp),
|
||||||
color = White,
|
|
||||||
shape = RoundedCornerShape(24.dp)
|
|
||||||
)
|
|
||||||
.padding(horizontal = 22.dp, vertical = 28.dp)
|
|
||||||
) {
|
|
||||||
Column(
|
|
||||||
modifier = Modifier.fillMaxWidth(),
|
|
||||||
horizontalAlignment = Alignment.CenterHorizontally
|
|
||||||
) {
|
|
||||||
Box(
|
|
||||||
modifier = Modifier
|
|
||||||
.size(92.dp)
|
|
||||||
.background(
|
|
||||||
color = Primary.copy(alpha = 0.08f),
|
|
||||||
shape = RoundedCornerShape(28.dp)
|
|
||||||
),
|
|
||||||
contentAlignment = Alignment.Center
|
contentAlignment = Alignment.Center
|
||||||
) {
|
) {
|
||||||
Image(
|
if (uiState.canGoBack) {
|
||||||
painter = painterResource(R.drawable.buy_ecommerce_finance_payment_pos_shop_svgrepo_com),
|
IconButton(
|
||||||
contentDescription = "Card reader"
|
onClick = onBackClick,
|
||||||
|
modifier = Modifier.align(Alignment.CenterStart)
|
||||||
|
) {
|
||||||
|
Icon(
|
||||||
|
imageVector = Icons.Rounded.KeyboardArrowLeft,
|
||||||
|
contentDescription = "Back",
|
||||||
|
tint = Color.LegacyRed
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Spacer(modifier = Modifier.height(20.dp))
|
Spacer(modifier = Modifier.height(8.dp))
|
||||||
|
|
||||||
Text(
|
Text(
|
||||||
text = if (uiState.isFallback) {
|
text = "AMOUNT TO PAY",
|
||||||
"Swipe Card"
|
color = Color.Black,
|
||||||
} else {
|
fontSize = 10.sp,
|
||||||
"Present Card"
|
fontWeight = FontWeight.Medium
|
||||||
},
|
)
|
||||||
color = Black,
|
|
||||||
fontSize = 24.sp,
|
Spacer(modifier = Modifier.height(6.dp))
|
||||||
|
|
||||||
|
Row(verticalAlignment = Alignment.Bottom) {
|
||||||
|
Text(
|
||||||
|
text = amount,
|
||||||
|
color = Color.LegacyRed,
|
||||||
|
fontSize = 30.sp,
|
||||||
fontWeight = FontWeight.Bold
|
fontWeight = FontWeight.Bold
|
||||||
)
|
)
|
||||||
|
|
||||||
Spacer(modifier = Modifier.height(10.dp))
|
Spacer(modifier = Modifier.width(6.dp))
|
||||||
|
|
||||||
|
Text(
|
||||||
|
text = "MMK",
|
||||||
|
color = Color.LegacyRed,
|
||||||
|
fontSize = 10.sp,
|
||||||
|
fontWeight = FontWeight.Bold,
|
||||||
|
modifier = Modifier.padding(bottom = 6.dp)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
Spacer(modifier = Modifier.height(32.dp))
|
||||||
|
|
||||||
|
ContactlessCircle(isLoading = uiState.isLoading)
|
||||||
|
|
||||||
|
Spacer(modifier = Modifier.height(24.dp))
|
||||||
|
|
||||||
|
Text(
|
||||||
|
text = when {
|
||||||
|
uiState.isCardCaptured -> "Card Detected"
|
||||||
|
uiState.isFallback -> "Swipe Your Card"
|
||||||
|
else -> "Tap Your Card"
|
||||||
|
},
|
||||||
|
color = Color.LegacyRed,
|
||||||
|
fontSize = 13.sp,
|
||||||
|
fontWeight = FontWeight.Bold
|
||||||
|
)
|
||||||
|
|
||||||
|
Text(
|
||||||
|
text = if (uiState.isCardCaptured) {
|
||||||
|
"Reader captured card data"
|
||||||
|
} else {
|
||||||
|
"Hold your card near the reader"
|
||||||
|
},
|
||||||
|
color = Color.Black,
|
||||||
|
fontSize = 9.sp
|
||||||
|
)
|
||||||
|
|
||||||
|
Spacer(modifier = Modifier.height(14.dp))
|
||||||
|
|
||||||
|
Row(
|
||||||
|
modifier = Modifier.fillMaxWidth(),
|
||||||
|
verticalAlignment = Alignment.CenterVertically
|
||||||
|
) {
|
||||||
|
HorizontalDivider(
|
||||||
|
modifier = Modifier.weight(1f),
|
||||||
|
color = Color.Gray.copy(alpha = 0.4f)
|
||||||
|
)
|
||||||
|
|
||||||
|
Text(
|
||||||
|
text = "OR",
|
||||||
|
modifier = Modifier.padding(horizontal = 12.dp),
|
||||||
|
color = Color.Gray,
|
||||||
|
fontSize = 10.sp
|
||||||
|
)
|
||||||
|
|
||||||
|
HorizontalDivider(
|
||||||
|
modifier = Modifier.weight(1f),
|
||||||
|
color = Color.Gray.copy(alpha = 0.4f)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
Spacer(modifier = Modifier.height(12.dp))
|
||||||
|
|
||||||
|
InsertCardRow()
|
||||||
|
|
||||||
|
Spacer(modifier = Modifier.height(18.dp))
|
||||||
|
|
||||||
Text(
|
Text(
|
||||||
text = uiState.alertMessage,
|
text = uiState.alertMessage,
|
||||||
color = Black,
|
color = Color.Black,
|
||||||
fontSize = 18.sp,
|
fontSize = 11.sp,
|
||||||
lineHeight = 26.sp,
|
textAlign = TextAlign.Center,
|
||||||
textAlign = TextAlign.Center
|
lineHeight = 16.sp
|
||||||
|
)
|
||||||
|
|
||||||
|
Spacer(modifier = Modifier.height(18.dp))
|
||||||
|
|
||||||
|
StatusPanel(uiState = uiState)
|
||||||
|
|
||||||
|
Spacer(modifier = Modifier.weight(1f))
|
||||||
|
|
||||||
|
ManualEntryAction(onManualEntryClick = onManualEntryClick)
|
||||||
|
|
||||||
|
Spacer(modifier = Modifier.height(18.dp))
|
||||||
|
|
||||||
|
CardLogoRow()
|
||||||
|
|
||||||
|
Spacer(modifier = Modifier.height(34.dp))
|
||||||
|
|
||||||
|
PoweredByMob()
|
||||||
|
|
||||||
|
Spacer(modifier = Modifier.height(12.dp))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
private fun ContactlessCircle(isLoading: Boolean) {
|
||||||
|
Box(
|
||||||
|
modifier = Modifier.size(190.dp),
|
||||||
|
contentAlignment = Alignment.Center
|
||||||
|
) {
|
||||||
|
CircleBorder(180, 0.1f)
|
||||||
|
CircleBorder(150, 0.20f)
|
||||||
|
|
||||||
|
Surface(
|
||||||
|
modifier = Modifier.size(108.dp),
|
||||||
|
shape = CircleShape,
|
||||||
|
color = Color.White,
|
||||||
|
shadowElevation = 8.dp
|
||||||
|
) {
|
||||||
|
Box(contentAlignment = Alignment.Center) {
|
||||||
|
if (isLoading) {
|
||||||
|
CircularProgressIndicator(
|
||||||
|
color = Color.LegacyRed,
|
||||||
|
strokeWidth = 3.dp,
|
||||||
|
modifier = Modifier.size(40.dp)
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
Icon(
|
||||||
|
imageVector = Icons.Rounded.Wifi,
|
||||||
|
contentDescription = null,
|
||||||
|
tint = Color.LegacyRed,
|
||||||
|
modifier = Modifier
|
||||||
|
.size(72.dp)
|
||||||
|
.rotate(90f)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Spacer(modifier = Modifier.height(16.dp))
|
@Composable
|
||||||
|
private fun CircleBorder(
|
||||||
|
size: Int,
|
||||||
|
alpha: Float
|
||||||
|
) {
|
||||||
|
Surface(
|
||||||
|
modifier = Modifier
|
||||||
|
.size(size.dp)
|
||||||
|
.alpha(alpha),
|
||||||
|
shape = CircleShape,
|
||||||
|
border = BorderStroke(1.dp, Color.LegacyRed)
|
||||||
|
) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
private fun InsertCardRow() {
|
||||||
|
Row(
|
||||||
|
modifier = Modifier.height(44.dp),
|
||||||
|
verticalAlignment = Alignment.CenterVertically
|
||||||
|
) {
|
||||||
|
Icon(
|
||||||
|
painter = painterResource(id = R.drawable.ic_insert_card),
|
||||||
|
contentDescription = null,
|
||||||
|
tint = Color.LegacyRed,
|
||||||
|
modifier = Modifier.size(28.dp)
|
||||||
|
)
|
||||||
|
|
||||||
|
Spacer(modifier = Modifier.width(10.dp))
|
||||||
|
|
||||||
|
Column {
|
||||||
|
Text(
|
||||||
|
text = "Insert Your Card",
|
||||||
|
color = Color.LegacyRed,
|
||||||
|
fontSize = 11.sp,
|
||||||
|
fontWeight = FontWeight.Bold
|
||||||
|
)
|
||||||
|
Text(
|
||||||
|
text = "Chip facing up",
|
||||||
|
color = Color.Black,
|
||||||
|
fontSize = 9.sp
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
private fun StatusPanel(uiState: CardWaitingUiState) {
|
||||||
|
Surface(
|
||||||
|
modifier = Modifier.fillMaxWidth(),
|
||||||
|
color = Color.White,
|
||||||
|
shape = RoundedCornerShape(12.dp)
|
||||||
|
) {
|
||||||
Row(
|
Row(
|
||||||
modifier = Modifier
|
modifier = Modifier
|
||||||
.fillMaxWidth()
|
.fillMaxWidth()
|
||||||
.background(
|
.padding(horizontal = 14.dp, vertical = 12.dp),
|
||||||
color = White.copy(alpha = 0.12f),
|
|
||||||
shape = RoundedCornerShape(22.dp)
|
|
||||||
)
|
|
||||||
.padding(horizontal = 18.dp, vertical = 16.dp),
|
|
||||||
horizontalArrangement = Arrangement.SpaceBetween,
|
horizontalArrangement = Arrangement.SpaceBetween,
|
||||||
verticalAlignment = Alignment.CenterVertically
|
verticalAlignment = Alignment.CenterVertically
|
||||||
) {
|
) {
|
||||||
Text(
|
Text(
|
||||||
text = if (uiState.isLoading) {
|
text = when {
|
||||||
"Reader status"
|
uiState.isCardCaptured -> "Reader status"
|
||||||
} else {
|
uiState.isLoading -> "Reader status"
|
||||||
"Ready for card"
|
else -> "Ready for card"
|
||||||
},
|
},
|
||||||
color = White,
|
color = Color.LegacyRed,
|
||||||
fontSize = 16.sp,
|
fontSize = 12.sp,
|
||||||
fontWeight = FontWeight.SemiBold
|
fontWeight = FontWeight.Bold
|
||||||
)
|
)
|
||||||
|
|
||||||
Text(
|
Text(
|
||||||
text = if (uiState.isLoading) "Initializing" else "Waiting",
|
text = when {
|
||||||
color = White,
|
uiState.isCardCaptured -> "Captured"
|
||||||
fontSize = 15.sp
|
uiState.isLoading -> "Initializing"
|
||||||
|
else -> "Waiting"
|
||||||
|
},
|
||||||
|
color = Color.Gray,
|
||||||
|
fontSize = 11.sp
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Spacer(modifier = Modifier.weight(1f))
|
@Composable
|
||||||
|
private fun ManualEntryAction(onManualEntryClick: () -> Unit) {
|
||||||
Button(
|
Surface(
|
||||||
onClick = viewModel::onManualEntryClick,
|
modifier = Modifier.fillMaxWidth(),
|
||||||
|
color = Color.White,
|
||||||
|
shape = RoundedCornerShape(12.dp)
|
||||||
|
) {
|
||||||
|
Box(
|
||||||
modifier = Modifier
|
modifier = Modifier
|
||||||
.fillMaxWidth()
|
.fillMaxWidth()
|
||||||
.height(64.dp),
|
.background(Color.White)
|
||||||
shape = RoundedCornerShape(18.dp),
|
.clickable(onClick = onManualEntryClick)
|
||||||
colors = ButtonDefaults.buttonColors(
|
.padding(vertical = 16.dp),
|
||||||
containerColor = White,
|
contentAlignment = Alignment.Center
|
||||||
contentColor = Primary
|
|
||||||
)
|
|
||||||
) {
|
) {
|
||||||
Text(
|
Text(
|
||||||
text = "Manual Entry",
|
text = "Manual Entry",
|
||||||
fontSize = 20.sp,
|
color = Color.LegacyRed,
|
||||||
|
fontSize = 14.sp,
|
||||||
fontWeight = FontWeight.Bold
|
fontWeight = FontWeight.Bold
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
private fun CardLogoRow() {
|
||||||
|
Surface(
|
||||||
|
modifier = Modifier.fillMaxWidth(),
|
||||||
|
color = Color.White,
|
||||||
|
shape = RoundedCornerShape(3.dp)
|
||||||
|
) {
|
||||||
|
Row(
|
||||||
|
modifier = Modifier
|
||||||
|
.fillMaxWidth()
|
||||||
|
.height(42.dp)
|
||||||
|
.padding(horizontal = 14.dp),
|
||||||
|
horizontalArrangement = Arrangement.SpaceBetween,
|
||||||
|
verticalAlignment = Alignment.CenterVertically
|
||||||
|
) {
|
||||||
|
Text("VISA", fontSize = 22.sp, fontWeight = FontWeight.Bold)
|
||||||
|
Text("MC", fontSize = 18.sp, fontWeight = FontWeight.Bold)
|
||||||
|
Text("MPU", fontSize = 20.sp, fontWeight = FontWeight.Bold)
|
||||||
|
Text("UnionPay", fontSize = 12.sp, fontWeight = FontWeight.Bold)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
private fun PoweredByMob() {
|
||||||
|
Row(verticalAlignment = Alignment.CenterVertically) {
|
||||||
|
Text(
|
||||||
|
text = "Powered by",
|
||||||
|
color = Color.Gray,
|
||||||
|
fontSize = 8.sp
|
||||||
|
)
|
||||||
|
|
||||||
|
Spacer(modifier = Modifier.width(4.dp))
|
||||||
|
|
||||||
|
Text(
|
||||||
|
text = "MOB",
|
||||||
|
color = Color.Gray,
|
||||||
|
fontSize = 9.sp,
|
||||||
|
fontWeight = FontWeight.Bold
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@P2Preview
|
||||||
|
@Composable
|
||||||
|
fun PreviewCardWaitingScreen() {
|
||||||
|
CardWaitingScreenContent(
|
||||||
|
amount = "50,000",
|
||||||
|
uiState = CardWaitingUiState(),
|
||||||
|
onBackClick = {},
|
||||||
|
onManualEntryClick = {}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|||||||
@ -4,5 +4,6 @@ data class CardWaitingUiState(
|
|||||||
val alertMessage: String = "Please insert, tap, or swipe card",
|
val alertMessage: String = "Please insert, tap, or swipe card",
|
||||||
val isLoading: Boolean = false,
|
val isLoading: Boolean = false,
|
||||||
val isFallback: Boolean = false,
|
val isFallback: Boolean = false,
|
||||||
val canGoBack: Boolean = true
|
val canGoBack: Boolean = true,
|
||||||
|
val isCardCaptured: Boolean = false
|
||||||
)
|
)
|
||||||
|
|||||||
@ -78,7 +78,17 @@ class CardWaitingViewModel(
|
|||||||
_uiState.update {
|
_uiState.update {
|
||||||
it.copy(
|
it.copy(
|
||||||
alertMessage = "Fallback!\nPlease stripe!",
|
alertMessage = "Fallback!\nPlease stripe!",
|
||||||
isFallback = true
|
isFallback = true,
|
||||||
|
isCardCaptured = false
|
||||||
|
)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
_uiState.update {
|
||||||
|
it.copy(
|
||||||
|
alertMessage = "Please insert, tap, or swipe card",
|
||||||
|
isFallback = false,
|
||||||
|
isLoading = false,
|
||||||
|
isCardCaptured = false
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -122,7 +132,8 @@ class CardWaitingViewModel(
|
|||||||
_uiState.update {
|
_uiState.update {
|
||||||
it.copy(
|
it.copy(
|
||||||
alertMessage = "Initializing card reader...",
|
alertMessage = "Initializing card reader...",
|
||||||
isLoading = true
|
isLoading = true,
|
||||||
|
isCardCaptured = false
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -136,7 +147,8 @@ class CardWaitingViewModel(
|
|||||||
} else {
|
} else {
|
||||||
"Please insert, tap, or swipe card"
|
"Please insert, tap, or swipe card"
|
||||||
},
|
},
|
||||||
isLoading = false
|
isLoading = false,
|
||||||
|
isCardCaptured = false
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
setupCardReadProcess(isFallback)
|
setupCardReadProcess(isFallback)
|
||||||
@ -209,8 +221,14 @@ class CardWaitingViewModel(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
viewModelScope.launch {
|
stopCardReading()
|
||||||
_events.send(CardWaitingEvent.GoProcessingCard)
|
|
||||||
|
_uiState.update {
|
||||||
|
it.copy(
|
||||||
|
alertMessage = "Card detected.\nOnline process disabled.",
|
||||||
|
isLoading = false,
|
||||||
|
isCardCaptured = true
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -87,16 +87,8 @@ fun AppNavGraph(
|
|||||||
|
|
||||||
CardWaitingScreen(
|
CardWaitingScreen(
|
||||||
viewModel = cardWaitingViewModel,
|
viewModel = cardWaitingViewModel,
|
||||||
|
amount = formatAmountForDisplay(sharedViewModel.amount.value),
|
||||||
onManualEntry = {},
|
onManualEntry = {},
|
||||||
onProcessingCard = {
|
|
||||||
navController.navigate(Routes.ProcessingCard.route) {
|
|
||||||
popUpTo(Routes.CardWaiting.route) {
|
|
||||||
inclusive = true
|
|
||||||
}
|
|
||||||
launchSingleTop = true
|
|
||||||
}
|
|
||||||
},
|
|
||||||
onTimeout = { navController.popBackStack() },
|
|
||||||
onBack = { navController.popBackStack() },
|
onBack = { navController.popBackStack() },
|
||||||
onMain = {
|
onMain = {
|
||||||
navController.navigate(Routes.Dashboard.route) {
|
navController.navigate(Routes.Dashboard.route) {
|
||||||
@ -207,3 +199,9 @@ fun AppNavGraph(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun formatAmountForDisplay(amount: String?): String {
|
||||||
|
val normalizedAmount = amount.orEmpty()
|
||||||
|
val value = normalizedAmount.toLongOrNull() ?: return normalizedAmount.ifBlank { "0" }
|
||||||
|
return "%,d".format(value)
|
||||||
|
}
|
||||||
|
|||||||
24
app/src/main/res/drawable/ic_insert_card.xml
Normal file
24
app/src/main/res/drawable/ic_insert_card.xml
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
<!--
|
||||||
|
~ Copyright (C) 2026 The Android Open Source Project
|
||||||
|
~
|
||||||
|
~ Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
~ you may not use this file except in compliance with the License.
|
||||||
|
~ You may obtain a copy of the License at
|
||||||
|
~
|
||||||
|
~ http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
~
|
||||||
|
~ Unless required by applicable law or agreed to in writing, software
|
||||||
|
~ distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
~ See the License for the specific language governing permissions and
|
||||||
|
~ limitations under the License.
|
||||||
|
-->
|
||||||
|
<vector xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:width="800dp"
|
||||||
|
android:height="800dp"
|
||||||
|
android:viewportWidth="15.133"
|
||||||
|
android:viewportHeight="15.133">
|
||||||
|
<path
|
||||||
|
android:pathData="M14.487,0H0.647C0.306,0 0.028,0.277 0.028,0.619v2.165c0,0.342 0.277,0.619 0.619,0.619h0.966h0.074h0.902l-0.726,10.54c0,0.654 0.532,1.186 1.186,1.188l9.033,0.002c0.653,0 1.188,-0.53 1.188,-1.187L12.552,3.402h0.891c0.005,0 0.008,-0.002 0.013,-0.002v0.002h1.031c0.341,0 0.618,-0.277 0.618,-0.619V0.619C15.105,0.277 14.828,0 14.487,0zM7.883,14.339L3.05,14.337c-0.218,0 -0.395,-0.178 -0.395,-0.396L3.441,2.518c0,-0.218 0.178,-0.395 0.396,-0.395l4.051,0.001L7.883,14.339zM12.084,14.339h-1.062L10.246,2.125h1.061c0.22,0 0.396,0.177 0.396,0.395l0.776,11.424C12.478,14.164 12.301,14.341 12.084,14.339zM12.524,2.963L12.524,2.963l-0.03,-0.442c0,-0.655 -0.531,-1.187 -1.188,-1.187L3.837,1.331c-0.654,0 -1.187,0.532 -1.187,1.186L2.62,2.965H2.619H1.717l0.006,-0.084c0.02,-1.148 0.96,-2.077 2.113,-2.077l7.47,0.003c0.564,0 1.096,0.22 1.496,0.62c0.391,0.392 0.609,0.908 0.617,1.459l0.006,0.077C13.425,2.963 12.524,2.963 12.524,2.963zM3.387,10.312l1.075,-0.009l0.025,3.125l-1.074,0.009L3.387,10.312z"
|
||||||
|
android:fillColor="#030104"/>
|
||||||
|
</vector>
|
||||||
49
app/src/main/res/drawable/ic_wifi.xml
Normal file
49
app/src/main/res/drawable/ic_wifi.xml
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
<!--
|
||||||
|
~ Copyright (C) 2026 The Android Open Source Project
|
||||||
|
~
|
||||||
|
~ Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
~ you may not use this file except in compliance with the License.
|
||||||
|
~ You may obtain a copy of the License at
|
||||||
|
~
|
||||||
|
~ http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
~
|
||||||
|
~ Unless required by applicable law or agreed to in writing, software
|
||||||
|
~ distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
~ See the License for the specific language governing permissions and
|
||||||
|
~ limitations under the License.
|
||||||
|
-->
|
||||||
|
<vector xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:width="800dp"
|
||||||
|
android:height="800dp"
|
||||||
|
android:viewportWidth="24"
|
||||||
|
android:viewportHeight="24">
|
||||||
|
<path
|
||||||
|
android:pathData="M4.91,11.84C9.21,8.52 14.8,8.52 19.1,11.84"
|
||||||
|
android:strokeLineJoin="round"
|
||||||
|
android:strokeWidth="1.5"
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:strokeColor="#292D32"
|
||||||
|
android:strokeLineCap="round"/>
|
||||||
|
<path
|
||||||
|
android:pathData="M2,8.36C8.06,3.68 15.94,3.68 22,8.36"
|
||||||
|
android:strokeLineJoin="round"
|
||||||
|
android:strokeWidth="1.5"
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:strokeColor="#292D32"
|
||||||
|
android:strokeLineCap="round"/>
|
||||||
|
<path
|
||||||
|
android:pathData="M6.79,15.49C9.94,13.05 14.05,13.05 17.2,15.49"
|
||||||
|
android:strokeLineJoin="round"
|
||||||
|
android:strokeWidth="1.5"
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:strokeColor="#292D32"
|
||||||
|
android:strokeLineCap="round"/>
|
||||||
|
<path
|
||||||
|
android:pathData="M9.4,19.15C10.98,17.93 13.03,17.93 14.61,19.15"
|
||||||
|
android:strokeLineJoin="round"
|
||||||
|
android:strokeWidth="1.5"
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:strokeColor="#292D32"
|
||||||
|
android:strokeLineCap="round"/>
|
||||||
|
</vector>
|
||||||
Loading…
Reference in New Issue
Block a user