-
Notifications
You must be signed in to change notification settings - Fork 665
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Introduce CardEditUIHandler for the CardEditUI #10462
Changes from all commits
b73822a
68d6fc1
138da16
22adff3
9c64789
5fa25cf
d7918ea
e0bf36c
662df22
6070548
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,33 @@ | ||
package com.stripe.android.paymentsheet.ui | ||
|
||
import com.stripe.android.paymentsheet.CardUpdateParams | ||
|
||
/** | ||
* Represents the editable details of a card payment method. | ||
* | ||
* @property cardBrandChoice The currently selected card brand choice. | ||
*/ | ||
internal data class CardDetailsEntry( | ||
val cardBrandChoice: CardBrandChoice, | ||
) { | ||
/** | ||
* Determines if the card details have changed compared to the provided values. | ||
* | ||
* @param originalCardBrandChoice The card brand choice to compare against. | ||
* @return True if any of the card details have changed, false otherwise. | ||
*/ | ||
fun hasChanged( | ||
originalCardBrandChoice: CardBrandChoice, | ||
): Boolean { | ||
return originalCardBrandChoice != this.cardBrandChoice | ||
} | ||
} | ||
|
||
/** | ||
* Converts the CardDetailsEntry to CardUpdateParams. | ||
* | ||
* @return CardUpdateParams containing the updated card brand. | ||
*/ | ||
internal fun CardDetailsEntry.toUpdateParams(): CardUpdateParams { | ||
return CardUpdateParams(cardBrand = cardBrandChoice.brand) | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,147 @@ | ||
package com.stripe.android.paymentsheet.ui | ||
|
||
import androidx.compose.runtime.Immutable | ||
import com.stripe.android.CardBrandFilter | ||
import com.stripe.android.DefaultCardBrandFilter | ||
import com.stripe.android.core.utils.DateUtils | ||
import com.stripe.android.model.CardBrand | ||
import com.stripe.android.model.PaymentMethod | ||
import com.stripe.android.paymentsheet.CardUpdateParams | ||
import kotlinx.coroutines.CoroutineScope | ||
import kotlinx.coroutines.Dispatchers | ||
import kotlinx.coroutines.flow.MutableStateFlow | ||
import kotlinx.coroutines.flow.SharingStarted | ||
import kotlinx.coroutines.flow.StateFlow | ||
import kotlinx.coroutines.flow.collectLatest | ||
import kotlinx.coroutines.flow.mapLatest | ||
import kotlinx.coroutines.flow.stateIn | ||
import kotlinx.coroutines.flow.update | ||
import kotlinx.coroutines.launch | ||
|
||
internal typealias CardUpdateParamsCallback = (CardUpdateParams?) -> Unit | ||
|
||
internal typealias CardBrandCallback = (CardBrand) -> Unit | ||
|
||
internal interface EditCardDetailsInteractor { | ||
val state: StateFlow<State> | ||
|
||
val onCardUpdateParamsChanged: CardUpdateParamsCallback | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Does this still need to be exposed? Is the UI going to call it somewhere? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yh, it needs to be exposed. It is used in the
The
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Would passing it through the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Emitting
We could call There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yeah I see what's going on here now. I think exposing an attribute for testing is the wrong solution. Re-introducing the I'm peaceful with keeping this for now! Can we follow-up with the factory? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yh, I can do this in a follow-up. Thanks! |
||
|
||
fun handleViewAction(viewAction: ViewAction) | ||
|
||
@Immutable | ||
data class State( | ||
val card: PaymentMethod.Card, | ||
val selectedCardBrand: CardBrandChoice, | ||
val paymentMethodIcon: Int, | ||
val shouldShowCardBrandDropdown: Boolean, | ||
val availableNetworks: List<CardBrandChoice> | ||
) | ||
|
||
sealed interface ViewAction { | ||
data class BrandChoiceChanged(val cardBrandChoice: CardBrandChoice) : ViewAction | ||
} | ||
|
||
companion object { | ||
fun create( | ||
coroutineScope: CoroutineScope, | ||
isModifiable: Boolean, | ||
cardBrandFilter: CardBrandFilter = DefaultCardBrandFilter, | ||
card: PaymentMethod.Card, | ||
onBrandChoiceChanged: CardBrandCallback, | ||
onCardUpdateParamsChanged: CardUpdateParamsCallback | ||
): EditCardDetailsInteractor { | ||
return DefaultEditCardDetailsInteractor( | ||
card = card, | ||
cardBrandFilter = cardBrandFilter, | ||
coroutineScope = coroutineScope, | ||
onBrandChoiceChanged = onBrandChoiceChanged, | ||
onCardUpdateParamsChanged = onCardUpdateParamsChanged, | ||
isModifiable = isModifiable | ||
) | ||
} | ||
} | ||
} | ||
|
||
private class DefaultEditCardDetailsInteractor( | ||
private val card: PaymentMethod.Card, | ||
private val cardBrandFilter: CardBrandFilter, | ||
private val isModifiable: Boolean, | ||
coroutineScope: CoroutineScope, | ||
private val onBrandChoiceChanged: CardBrandCallback, | ||
override val onCardUpdateParamsChanged: CardUpdateParamsCallback | ||
) : EditCardDetailsInteractor { | ||
private val cardDetailsEntry = MutableStateFlow( | ||
value = buildDefaultCardEntry() | ||
) | ||
|
||
override val state: StateFlow<EditCardDetailsInteractor.State> = cardDetailsEntry.mapLatest { inputState -> | ||
uiState(inputState.cardBrandChoice) | ||
}.stateIn( | ||
scope = coroutineScope, | ||
started = SharingStarted.Eagerly, | ||
initialValue = uiState() | ||
) | ||
|
||
init { | ||
coroutineScope.launch(Dispatchers.Main) { | ||
cardDetailsEntry.collectLatest { state -> | ||
val newParams = state.takeIf { | ||
it.hasChanged( | ||
originalCardBrandChoice = defaultCardBrandChoice(), | ||
) | ||
}?.toUpdateParams() | ||
onCardUpdateParamsChanged(newParams) | ||
} | ||
} | ||
} | ||
|
||
private fun onBrandChoiceChanged(cardBrandChoice: CardBrandChoice) { | ||
if (cardBrandChoice != state.value.selectedCardBrand) { | ||
onBrandChoiceChanged(cardBrandChoice.brand) | ||
} | ||
cardDetailsEntry.update { | ||
it.copy( | ||
cardBrandChoice = cardBrandChoice | ||
) | ||
} | ||
} | ||
|
||
override fun handleViewAction(viewAction: EditCardDetailsInteractor.ViewAction) { | ||
when (viewAction) { | ||
is EditCardDetailsInteractor.ViewAction.BrandChoiceChanged -> { | ||
onBrandChoiceChanged(viewAction.cardBrandChoice) | ||
} | ||
} | ||
} | ||
|
||
private fun buildDefaultCardEntry(): CardDetailsEntry { | ||
return CardDetailsEntry( | ||
cardBrandChoice = defaultCardBrandChoice() | ||
) | ||
} | ||
|
||
private fun defaultCardBrandChoice() = card.getPreferredChoice(cardBrandFilter) | ||
|
||
private fun uiState(cardBrandChoice: CardBrandChoice = defaultCardBrandChoice()): EditCardDetailsInteractor.State { | ||
return EditCardDetailsInteractor.State( | ||
card = card, | ||
selectedCardBrand = cardBrandChoice, | ||
paymentMethodIcon = card.getSavedPaymentMethodIcon(forVerticalMode = true), | ||
shouldShowCardBrandDropdown = isModifiable && isExpired().not(), | ||
availableNetworks = card.getAvailableNetworks(cardBrandFilter) | ||
) | ||
} | ||
|
||
private fun isExpired(): Boolean { | ||
val cardExpiryMonth = card.expiryMonth | ||
val cardExpiryYear = card.expiryYear | ||
// If the card's expiration dates are missing, we can't conclude that it is expired, so we don't want to | ||
// show the user an expired card error. | ||
return cardExpiryMonth != null && cardExpiryYear != null && | ||
!DateUtils.isExpiryDataValid( | ||
expiryMonth = cardExpiryMonth, | ||
expiryYear = cardExpiryYear, | ||
) | ||
} | ||
} | ||
samer-stripe marked this conversation as resolved.
Show resolved
Hide resolved
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Question: Are we expecting to add parameters to
CardDetailsEntry
that we shouldn't compare?equals
ofdata class
instead ofhasChanged
?There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Also should we just compare entries here?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We're not comparing two versions of
CardDetailsEntry
,distinctUntilChanged
would have sufficed in that scenario. We're comparing against the existing card and billing details (taking billingDetailsCollectionMode into account).I have the full logic on my prototype branch here - https://github.com/stripe/stripe-android/pull/10366/files#r2025490382