card capture renew

This commit is contained in:
moon 2026-05-16 12:08:41 +06:30
parent 283d2d64ce
commit 79ab7c0bae
7 changed files with 483 additions and 191 deletions

View File

@ -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()
// } }
) )
} }

View File

@ -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 = {}
)
}

View File

@ -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
) )

View File

@ -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
)
} }
} }

View File

@ -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)
}

View 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>

View 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>