restucturization of recipients payment methods
All checks were successful
ci/woodpecker/push/billing_fees Pipeline was successful
ci/woodpecker/push/bff Pipeline was successful
ci/woodpecker/push/db Pipeline was successful
ci/woodpecker/push/chain_gateway Pipeline was successful
ci/woodpecker/push/fx_ingestor Pipeline was successful
ci/woodpecker/push/fx_oracle Pipeline was successful
ci/woodpecker/push/frontend Pipeline was successful
ci/woodpecker/push/nats Pipeline was successful
ci/woodpecker/push/ledger Pipeline was successful
ci/woodpecker/push/notification Pipeline was successful
ci/woodpecker/push/payments_orchestrator Pipeline was successful

This commit is contained in:
Stephan D
2025-12-04 14:40:21 +01:00
parent 3b04753f4e
commit bf85ca062c
120 changed files with 1415 additions and 538 deletions

View File

@@ -0,0 +1,20 @@
import 'package:json_annotation/json_annotation.dart';
part 'card.g.dart';
@JsonSerializable()
class CardPaymentDataDTO {
final String pan;
final String firstName;
final String lastName;
const CardPaymentDataDTO({
required this.pan,
required this.firstName,
required this.lastName,
});
factory CardPaymentDataDTO.fromJson(Map<String, dynamic> json) => _$CardPaymentDataDTOFromJson(json);
Map<String, dynamic> toJson() => _$CardPaymentDataDTOToJson(this);
}

View File

@@ -0,0 +1,20 @@
import 'package:json_annotation/json_annotation.dart';
part 'crypto_address.g.dart';
@JsonSerializable()
class CryptoAddressPaymentDataDTO {
final String address;
final String network;
final String? destinationTag;
const CryptoAddressPaymentDataDTO({
required this.address,
required this.network,
this.destinationTag,
});
factory CryptoAddressPaymentDataDTO.fromJson(Map<String, dynamic> json) => _$CryptoAddressPaymentDataDTOFromJson(json);
Map<String, dynamic> toJson() => _$CryptoAddressPaymentDataDTOToJson(this);
}

View File

@@ -0,0 +1,22 @@
import 'package:json_annotation/json_annotation.dart';
part 'iban.g.dart';
@JsonSerializable()
class IbanPaymentDataDTO {
final String iban;
final String accountHolder;
final String? bic;
final String? bankName;
const IbanPaymentDataDTO({
required this.iban,
required this.accountHolder,
this.bic,
this.bankName,
});
factory IbanPaymentDataDTO.fromJson(Map<String, dynamic> json) => _$IbanPaymentDataDTOFromJson(json);
Map<String, dynamic> toJson() => _$IbanPaymentDataDTOToJson(this);
}

View File

@@ -0,0 +1,33 @@
import 'package:json_annotation/json_annotation.dart';
import 'package:pshared/data/dto/date_time.dart';
import 'package:pshared/data/dto/permissions/bound.dart';
part 'method.g.dart';
@JsonSerializable()
class PaymentMethodDTO extends PermissionBoundDTO {
final String recipientRef;
final String type;
final Map<String, dynamic> data;
@JsonKey(defaultValue: false)
final bool isArchived;
const PaymentMethodDTO({
required super.id,
required super.createdAt,
required super.updatedAt,
required super.permissionRef,
required super.organizationRef,
required this.recipientRef,
required this.type,
required this.data,
this.isArchived = false,
});
factory PaymentMethodDTO.fromJson(Map<String, dynamic> json) => _$PaymentMethodDTOFromJson(json);
@override
Map<String, dynamic> toJson() => _$PaymentMethodDTOToJson(this);
}

View File

@@ -0,0 +1,28 @@
import 'package:json_annotation/json_annotation.dart';
part 'russian_bank.g.dart';
@JsonSerializable()
class RussianBankAccountPaymentDataDTO {
final String recipientName;
final String inn;
final String kpp;
final String bankName;
final String bik;
final String accountNumber;
final String correspondentAccount;
const RussianBankAccountPaymentDataDTO({
required this.recipientName,
required this.inn,
required this.kpp,
required this.bankName,
required this.bik,
required this.accountNumber,
required this.correspondentAccount,
});
factory RussianBankAccountPaymentDataDTO.fromJson(Map<String, dynamic> json) => _$RussianBankAccountPaymentDataDTOFromJson(json);
Map<String, dynamic> toJson() => _$RussianBankAccountPaymentDataDTOToJson(this);
}

View File

@@ -0,0 +1,16 @@
import 'package:json_annotation/json_annotation.dart';
part 'wallet.g.dart';
@JsonSerializable()
class WalletPaymentDataDTO {
final String walletId;
const WalletPaymentDataDTO({
required this.walletId,
});
factory WalletPaymentDataDTO.fromJson(Map<String, dynamic> json) => _$WalletPaymentDataDTOFromJson(json);
Map<String, dynamic> toJson() => _$WalletPaymentDataDTOToJson(this);
}

View File

@@ -0,0 +1,39 @@
import 'package:json_annotation/json_annotation.dart';
import 'package:pshared/data/dto/date_time.dart';
import 'package:pshared/data/dto/permissions/bound.dart';
part 'recipient.g.dart';
@JsonSerializable()
class RecipientDTO extends PermissionBoundDTO {
final String name;
final String? description;
final String email;
final String? avatarUrl;
final String status;
final String type;
@JsonKey(defaultValue: false)
final bool isArchived;
const RecipientDTO({
required super.id,
required super.createdAt,
required super.updatedAt,
required super.permissionRef,
required super.organizationRef,
required this.name,
required this.email,
required this.status,
required this.type,
this.description,
this.avatarUrl,
this.isArchived = false,
});
factory RecipientDTO.fromJson(Map<String, dynamic> json) => _$RecipientDTOFromJson(json);
@override
Map<String, dynamic> toJson() => _$RecipientDTOToJson(this);
}

View File

@@ -0,0 +1,19 @@
import 'package:pshared/data/dto/payment/card.dart';
import 'package:pshared/models/payment/methods/card.dart';
extension CardPaymentMethodMapper on CardPaymentMethod {
CardPaymentDataDTO toDTO() => CardPaymentDataDTO(
pan: pan,
firstName: firstName,
lastName: lastName,
);
}
extension CardPaymentDataDTOMapper on CardPaymentDataDTO {
CardPaymentMethod toDomain() => CardPaymentMethod(
pan: pan,
firstName: firstName,
lastName: lastName,
);
}

View File

@@ -0,0 +1,19 @@
import 'package:pshared/data/dto/payment/crypto_address.dart';
import 'package:pshared/models/payment/methods/crypto_address.dart';
extension CryptoAddressPaymentMethodMapper on CryptoAddressPaymentMethod {
CryptoAddressPaymentDataDTO toDTO() => CryptoAddressPaymentDataDTO(
address: address,
network: network,
destinationTag: destinationTag,
);
}
extension CryptoAddressPaymentDataDTOMapper on CryptoAddressPaymentDataDTO {
CryptoAddressPaymentMethod toDomain() => CryptoAddressPaymentMethod(
address: address,
network: network,
destinationTag: destinationTag,
);
}

View File

@@ -0,0 +1,21 @@
import 'package:pshared/data/dto/payment/iban.dart';
import 'package:pshared/models/payment/methods/iban.dart';
extension IbanPaymentMethodMapper on IbanPaymentMethod {
IbanPaymentDataDTO toDTO() => IbanPaymentDataDTO(
iban: iban,
accountHolder: accountHolder,
bic: bic,
bankName: bankName,
);
}
extension IbanPaymentDataDTOMapper on IbanPaymentDataDTO {
IbanPaymentMethod toDomain() => IbanPaymentMethod(
iban: iban,
accountHolder: accountHolder,
bic: bic,
bankName: bankName,
);
}

View File

@@ -0,0 +1,81 @@
import 'package:pshared/data/dto/payment/card.dart';
import 'package:pshared/data/dto/payment/crypto_address.dart';
import 'package:pshared/data/dto/payment/iban.dart';
import 'package:pshared/data/dto/payment/method.dart';
import 'package:pshared/data/dto/payment/russian_bank.dart';
import 'package:pshared/data/dto/payment/wallet.dart';
import 'package:pshared/data/mapper/payment/crypto_address.dart';
import 'package:pshared/data/mapper/payment/card.dart';
import 'package:pshared/data/mapper/payment/iban.dart';
import 'package:pshared/data/mapper/payment/russian_bank.dart';
import 'package:pshared/data/mapper/payment/type.dart';
import 'package:pshared/data/mapper/payment/wallet.dart';
import 'package:pshared/models/organization/bound.dart';
import 'package:pshared/models/payment/methods/card.dart';
import 'package:pshared/models/payment/methods/crypto_address.dart';
import 'package:pshared/models/payment/methods/data.dart';
import 'package:pshared/models/payment/methods/iban.dart';
import 'package:pshared/models/payment/methods/russian_bank.dart';
import 'package:pshared/models/payment/methods/wallet.dart';
import 'package:pshared/models/payment/payment_method.dart';
import 'package:pshared/models/payment/type.dart';
import 'package:pshared/models/permissions/bound.dart';
import 'package:pshared/models/storable.dart';
extension PaymentMethodModelMapper on PaymentMethodModel {
PaymentMethodDTO toDTO() => PaymentMethodDTO(
id: storable.id,
createdAt: storable.createdAt,
updatedAt: storable.updatedAt,
permissionRef: permissionBound.permissionRef,
organizationRef: permissionBound.organizationRef,
recipientRef: recipientRef,
type: paymentTypeToValue(type),
data: _dataToJson(data),
isArchived: isArchived,
);
Map<String, dynamic> _dataToJson(PaymentMethodData data) {
switch (data.type) {
case PaymentType.card:
return (data as CardPaymentMethod).toDTO().toJson();
case PaymentType.iban:
return (data as IbanPaymentMethod).toDTO().toJson();
case PaymentType.bankAccount:
return (data as RussianBankAccountPaymentMethod).toDTO().toJson();
case PaymentType.wallet:
return (data as WalletPaymentMethod).toDTO().toJson();
case PaymentType.cryptoAddress:
return (data as CryptoAddressPaymentMethod).toDTO().toJson();
}
}
}
extension PaymentMethodDTOMapper on PaymentMethodDTO {
PaymentMethodModel toDomain() => PaymentMethodModel(
storable: newStorable(id: id, createdAt: createdAt, updatedAt: updatedAt),
permissionBound: newPermissionBound(
organizationBound: newOrganizationBound(organizationRef: organizationRef),
permissionRef: permissionRef,
),
recipientRef: recipientRef,
data: _dataToDomain(paymentTypeFromValue(type), data),
isArchived: isArchived,
);
PaymentMethodData _dataToDomain(PaymentType paymentType, Map<String, dynamic> payload) {
switch (paymentType) {
case PaymentType.card:
return CardPaymentDataDTO.fromJson(payload).toDomain();
case PaymentType.iban:
return IbanPaymentDataDTO.fromJson(payload).toDomain();
case PaymentType.bankAccount:
return RussianBankAccountPaymentDataDTO.fromJson(payload).toDomain();
case PaymentType.wallet:
return WalletPaymentDataDTO.fromJson(payload).toDomain();
case PaymentType.cryptoAddress:
return CryptoAddressPaymentDataDTO.fromJson(payload).toDomain();
}
}
}

View File

@@ -0,0 +1,27 @@
import 'package:pshared/data/dto/payment/russian_bank.dart';
import 'package:pshared/models/payment/methods/russian_bank.dart';
extension RussianBankPaymentMethodMapper on RussianBankAccountPaymentMethod {
RussianBankAccountPaymentDataDTO toDTO() => RussianBankAccountPaymentDataDTO(
recipientName: recipientName,
inn: inn,
kpp: kpp,
bankName: bankName,
bik: bik,
accountNumber: accountNumber,
correspondentAccount: correspondentAccount,
);
}
extension RussianBankAccountPaymentDataDTOMapper on RussianBankAccountPaymentDataDTO {
RussianBankAccountPaymentMethod toDomain() => RussianBankAccountPaymentMethod(
recipientName: recipientName,
inn: inn,
kpp: kpp,
bankName: bankName,
bik: bik,
accountNumber: accountNumber,
correspondentAccount: correspondentAccount,
);
}

View File

@@ -0,0 +1,34 @@
import 'package:pshared/models/payment/type.dart';
PaymentType paymentTypeFromValue(String value) {
switch (value) {
case 'iban':
return PaymentType.iban;
case 'card':
return PaymentType.card;
case 'bankAccount':
return PaymentType.bankAccount;
case 'wallet':
return PaymentType.wallet;
case 'cryptoAddress':
return PaymentType.cryptoAddress;
default:
return PaymentType.iban;
}
}
String paymentTypeToValue(PaymentType type) {
switch (type) {
case PaymentType.iban:
return 'iban';
case PaymentType.card:
return 'card';
case PaymentType.bankAccount:
return 'bankAccount';
case PaymentType.wallet:
return 'wallet';
case PaymentType.cryptoAddress:
return 'cryptoAddress';
}
}

View File

@@ -0,0 +1,13 @@
import 'package:pshared/data/dto/payment/wallet.dart';
import 'package:pshared/models/payment/methods/wallet.dart';
extension WalletPaymentMethodMapper on WalletPaymentMethod {
WalletPaymentDataDTO toDTO() => WalletPaymentDataDTO(
walletId: walletId,
);
}
extension WalletPaymentDataDTOMapper on WalletPaymentDataDTO {
WalletPaymentMethod toDomain() => WalletPaymentMethod(walletId: walletId);
}

View File

@@ -0,0 +1,85 @@
import 'package:pshared/data/dto/recipient/recipient.dart';
import 'package:pshared/models/describable.dart';
import 'package:pshared/models/organization/bound.dart';
import 'package:pshared/models/permissions/bound.dart';
import 'package:pshared/models/recipient/recipient_model.dart';
import 'package:pshared/models/recipient/status.dart';
import 'package:pshared/models/recipient/type.dart';
import 'package:pshared/models/storable.dart';
extension RecipientModelMapper on RecipientModel {
RecipientDTO toDTO() => RecipientDTO(
id: storable.id,
createdAt: storable.createdAt,
updatedAt: storable.updatedAt,
permissionRef: permissionBound.permissionRef,
organizationRef: permissionBound.organizationRef,
name: name,
description: description,
email: email,
avatarUrl: avatarUrl,
status: _recipientStatusToValue(status),
type: _recipientTypeToValue(type),
isArchived: isArchived,
);
}
extension RecipientDTOMapper on RecipientDTO {
RecipientModel toDomain() => RecipientModel(
storable: newStorable(id: id, createdAt: createdAt, updatedAt: updatedAt),
permissionBound: newPermissionBound(
organizationBound: newOrganizationBound(organizationRef: organizationRef),
permissionRef: permissionRef,
),
describable: newDescribable(name: name, description: description),
email: email,
avatarUrl: avatarUrl,
status: _recipientStatusFromValue(status),
type: _recipientTypeFromValue(type),
isArchived: isArchived,
);
}
RecipientStatus _recipientStatusFromValue(String value) {
switch (value) {
case 'ready':
return RecipientStatus.ready;
case 'registered':
return RecipientStatus.registered;
case 'notRegistered':
return RecipientStatus.notRegistered;
default:
return RecipientStatus.ready;
}
}
String _recipientStatusToValue(RecipientStatus status) {
switch (status) {
case RecipientStatus.ready:
return 'ready';
case RecipientStatus.registered:
return 'registered';
case RecipientStatus.notRegistered:
return 'notRegistered';
}
}
RecipientType _recipientTypeFromValue(String value) {
switch (value) {
case 'external':
return RecipientType.external;
case 'internal':
default:
return RecipientType.internal;
}
}
String _recipientTypeToValue(RecipientType type) {
switch (type) {
case RecipientType.internal:
return 'internal';
case RecipientType.external:
return 'external';
}
}

View File

@@ -0,0 +1,18 @@
import 'package:pshared/models/payment/methods/data.dart';
import 'package:pshared/models/payment/type.dart';
class CryptoAddressPaymentMethod extends PaymentMethodData {
@override
final PaymentType type = PaymentType.cryptoAddress;
final String address;
final String network;
final String? destinationTag;
CryptoAddressPaymentMethod({
required this.address,
required this.network,
this.destinationTag,
});
}

View File

@@ -0,0 +1,46 @@
import 'package:pshared/models/payment/methods/data.dart';
import 'package:pshared/models/payment/type.dart';
import 'package:pshared/models/permissions/bound.dart';
import 'package:pshared/models/storable.dart';
class PaymentMethodModel implements PermissionBound, Storable {
final Storable storable;
final PermissionBound permissionBound;
final String recipientRef;
final PaymentMethodData data;
final bool isArchived;
const PaymentMethodModel({
required this.storable,
required this.permissionBound,
required this.recipientRef,
required this.data,
this.isArchived = false,
});
PaymentType get type => data.type;
@override
String get id => storable.id;
@override
DateTime get createdAt => storable.createdAt;
@override
DateTime get updatedAt => storable.updatedAt;
@override
String get organizationRef => permissionBound.organizationRef;
@override
String get permissionRef => permissionBound.permissionRef;
PaymentMethodModel copyWith({
PaymentMethodData? data,
bool? isArchived,
}) => PaymentMethodModel(
storable: storable,
permissionBound: permissionBound,
recipientRef: recipientRef,
data: data ?? this.data,
isArchived: isArchived ?? this.isArchived,
);
}

View File

@@ -3,4 +3,5 @@ enum PaymentType {
iban,
wallet,
card,
cryptoAddress,
}

View File

@@ -1,5 +1,6 @@
import 'package:pshared/models/payment/methods/card.dart';
import 'package:pshared/models/payment/methods/iban.dart';
import 'package:pshared/models/payment/methods/crypto_address.dart';
import 'package:pshared/models/payment/methods/russian_bank.dart';
import 'package:pshared/models/payment/methods/wallet.dart';
import 'package:pshared/models/recipient/status.dart';
@@ -16,6 +17,7 @@ class Recipient {
final IbanPaymentMethod? iban;
final RussianBankAccountPaymentMethod? bank;
final WalletPaymentMethod? wallet;
final CryptoAddressPaymentMethod? cryptoAddress;
const Recipient({
this.avatarUrl,
@@ -27,6 +29,7 @@ class Recipient {
this.iban,
this.bank,
this.wallet,
this.cryptoAddress,
});
/// Convenience factory for quickly creating mock recipients.
@@ -39,6 +42,7 @@ class Recipient {
IbanPaymentMethod? iban,
RussianBankAccountPaymentMethod? bank,
WalletPaymentMethod? wallet,
CryptoAddressPaymentMethod? cryptoAddress,
}) =>
Recipient(
avatarUrl: null,
@@ -50,6 +54,7 @@ class Recipient {
iban: iban,
bank: bank,
wallet: wallet,
cryptoAddress: cryptoAddress,
);
bool matchesQuery(String q) {
@@ -71,6 +76,9 @@ class Recipient {
bank?.bik,
bank?.correspondentAccount,
wallet?.walletId,
cryptoAddress?.address,
cryptoAddress?.network,
cryptoAddress?.destinationTag,
];
return searchable.any((field) => field?.toLowerCase().contains(q) ?? false);

View File

@@ -0,0 +1,64 @@
import 'package:pshared/models/describable.dart';
import 'package:pshared/models/permissions/bound/describable.dart';
import 'package:pshared/models/permissions/bound.dart';
import 'package:pshared/models/recipient/status.dart';
import 'package:pshared/models/recipient/type.dart';
import 'package:pshared/models/storable.dart';
class RecipientModel implements PermissionBoundStorableDescribable {
final Storable storable;
final PermissionBound permissionBound;
final Describable describable;
final String email;
final String? avatarUrl;
final RecipientStatus status;
final RecipientType type;
final bool isArchived;
const RecipientModel({
required this.storable,
required this.permissionBound,
required this.describable,
required this.email,
required this.status,
required this.type,
this.avatarUrl,
this.isArchived = false,
});
@override
String get id => storable.id;
@override
DateTime get createdAt => storable.createdAt;
@override
DateTime get updatedAt => storable.updatedAt;
@override
String get organizationRef => permissionBound.organizationRef;
@override
String get permissionRef => permissionBound.permissionRef;
@override
String get name => describable.name;
@override
String? get description => describable.description;
RecipientModel copyWith({
Describable? describable,
String? email,
String? Function()? avatarUrl,
RecipientStatus? status,
RecipientType? type,
bool? isArchived,
}) => RecipientModel(
storable: storable,
permissionBound: permissionBound,
describable: describableCopyWithOther(this.describable, describable),
email: email ?? this.email,
avatarUrl: avatarUrl != null ? avatarUrl() : this.avatarUrl,
status: status ?? this.status,
type: type ?? this.type,
isArchived: isArchived ?? this.isArchived,
);
}

View File

@@ -1,6 +1,7 @@
import 'package:flutter/material.dart';
import 'package:pshared/models/payment/methods/card.dart';
import 'package:pshared/models/payment/methods/crypto_address.dart';
import 'package:pshared/models/payment/methods/iban.dart';
import 'package:pshared/models/payment/methods/russian_bank.dart';
import 'package:pshared/models/payment/methods/wallet.dart';
@@ -46,6 +47,7 @@ class _AdressBookRecipientFormState extends State<AdressBookRecipientForm> {
if (r?.iban != null) _methods[PaymentType.iban] = r!.iban;
if (r?.wallet != null) _methods[PaymentType.wallet] = r!.wallet;
if (r?.bank != null) _methods[PaymentType.bankAccount] = r!.bank;
if (r?.cryptoAddress != null) _methods[PaymentType.cryptoAddress] = r!.cryptoAddress;
}
//TODO Change when registration is ready
@@ -74,6 +76,7 @@ class _AdressBookRecipientFormState extends State<AdressBookRecipientForm> {
iban: _methods[PaymentType.iban] as IbanPaymentMethod?,
wallet: _methods[PaymentType.wallet] as WalletPaymentMethod?,
bank: _methods[PaymentType.bankAccount] as RussianBankAccountPaymentMethod?,
cryptoAddress: _methods[PaymentType.cryptoAddress] as CryptoAddressPaymentMethod?,
);
widget.onSaved?.call(recipient);
@@ -106,4 +109,4 @@ class _AdressBookRecipientFormState extends State<AdressBookRecipientForm> {
},
);
}
}
}

View File

@@ -41,7 +41,12 @@ class RecipientPaymentRow extends StatelessWidget {
type: PaymentType.wallet,
value: recipient.wallet!.walletId
),
if (recipient.cryptoAddress?.address.isNotEmpty ?? false)
RecipientAddressBookInfoRow(
type: PaymentType.cryptoAddress,
value: recipient.cryptoAddress!.address,
),
],
);
}
}
}

View File

@@ -73,6 +73,11 @@ class RecipientItem extends StatelessWidget {
label: getPaymentTypeLabel(context, PaymentType.wallet),
value: recipient.wallet!.walletId,
),
if (recipient.cryptoAddress?.address.isNotEmpty == true)
PaymentInfoRow(
label: getPaymentTypeLabel(context, PaymentType.cryptoAddress),
value: recipient.cryptoAddress!.address,
),
],
),
],

View File

@@ -0,0 +1,101 @@
import 'package:flutter/material.dart';
import 'package:pshared/models/payment/methods/crypto_address.dart';
import 'package:pweb/utils/text_field_styles.dart';
class CryptoAddressForm extends StatefulWidget {
final void Function(CryptoAddressPaymentMethod) onChanged;
final CryptoAddressPaymentMethod? initialData;
final bool isEditable;
const CryptoAddressForm({
super.key,
required this.onChanged,
this.initialData,
required this.isEditable,
});
@override
State<CryptoAddressForm> createState() => _CryptoAddressFormState();
}
class _CryptoAddressFormState extends State<CryptoAddressForm> {
late TextEditingController _addressCtrl;
late TextEditingController _networkCtrl;
late TextEditingController _destinationTagCtrl;
@override
void initState() {
super.initState();
_addressCtrl = TextEditingController(text: widget.initialData?.address);
_networkCtrl = TextEditingController(text: widget.initialData?.network);
_destinationTagCtrl = TextEditingController(text: widget.initialData?.destinationTag);
}
void _emit() {
if (_addressCtrl.text.isNotEmpty && _networkCtrl.text.isNotEmpty) {
widget.onChanged(
CryptoAddressPaymentMethod(
address: _addressCtrl.text,
network: _networkCtrl.text,
destinationTag: _destinationTagCtrl.text.isNotEmpty ? _destinationTagCtrl.text : null,
),
);
}
}
@override
void didUpdateWidget(covariant CryptoAddressForm oldWidget) {
super.didUpdateWidget(oldWidget);
final newData = widget.initialData;
final oldData = oldWidget.initialData;
if (newData == null && oldData != null) {
_addressCtrl.clear();
_networkCtrl.clear();
_destinationTagCtrl.clear();
return;
}
if (newData != null && newData != oldData) {
_addressCtrl.text = newData.address;
_networkCtrl.text = newData.network;
_destinationTagCtrl.text = newData.destinationTag ?? '';
}
}
@override
Widget build(BuildContext context) {
return Column(
children: [
TextFormField(
readOnly: !widget.isEditable,
controller: _addressCtrl,
decoration: getInputDecoration(context, 'Crypto address', widget.isEditable),
style: getTextFieldStyle(context, widget.isEditable),
onChanged: (_) => _emit(),
validator: (val) => (val?.isEmpty ?? true) ? 'Enter crypto address' : null,
),
const SizedBox(height: 12),
TextFormField(
readOnly: !widget.isEditable,
controller: _networkCtrl,
decoration: getInputDecoration(context, 'Network', widget.isEditable),
style: getTextFieldStyle(context, widget.isEditable),
onChanged: (_) => _emit(),
validator: (val) => (val?.isEmpty ?? true) ? 'Enter network' : null,
),
const SizedBox(height: 12),
TextFormField(
readOnly: !widget.isEditable,
controller: _destinationTagCtrl,
decoration: getInputDecoration(context, 'Destination tag / memo (optional)', widget.isEditable),
style: getTextFieldStyle(context, widget.isEditable),
onChanged: (_) => _emit(),
),
],
);
}
}

View File

@@ -1,12 +1,14 @@
import 'package:flutter/material.dart';
import 'package:pshared/models/payment/methods/card.dart';
import 'package:pshared/models/payment/methods/crypto_address.dart';
import 'package:pshared/models/payment/methods/iban.dart';
import 'package:pshared/models/payment/methods/russian_bank.dart';
import 'package:pshared/models/payment/methods/wallet.dart';
import 'package:pshared/models/payment/type.dart';
import 'package:pweb/pages/payment_methods/add/card.dart';
import 'package:pweb/pages/payment_methods/add/crypto_address.dart';
import 'package:pweb/pages/payment_methods/add/iban.dart';
import 'package:pweb/pages/payment_methods/add/russian_bank.dart';
import 'package:pweb/pages/payment_methods/add/wallet.dart';
@@ -49,6 +51,11 @@ class PaymentMethodForm extends StatelessWidget {
initialData: initialData as RussianBankAccountPaymentMethod?,
isEditable: isEditable,
),
PaymentType.cryptoAddress => CryptoAddressForm(
onChanged: onChanged,
initialData: initialData as CryptoAddressPaymentMethod?,
isEditable: isEditable,
),
_ => const SizedBox.shrink(),
};
}

View File

@@ -13,5 +13,7 @@ IconData iconForPaymentType(PaymentType type) {
return Icons.account_balance_wallet;
case PaymentType.card:
return Icons.credit_card;
case PaymentType.cryptoAddress:
return Icons.currency_bitcoin;
}
}

View File

@@ -148,6 +148,7 @@ class PageSelectorProvider extends ChangeNotifier {
if (recipient.iban != null) PaymentType.iban: recipient.iban!,
if (recipient.wallet != null) PaymentType.wallet: recipient.wallet!,
if (recipient.bank != null) PaymentType.bankAccount: recipient.bank!,
if (recipient.cryptoAddress != null) PaymentType.cryptoAddress: recipient.cryptoAddress!,
};
}

View File

@@ -12,5 +12,6 @@ String getPaymentTypeLabel(BuildContext context, PaymentType type) {
PaymentType.bankAccount => l10n.paymentTypeBankAccount,
PaymentType.iban => l10n.paymentTypeIban,
PaymentType.wallet => l10n.paymentTypeWallet,
PaymentType.cryptoAddress => 'Crypto address',
};
}