2023-03-21 23:18:07 +00:00
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
|
|
|
import 'package:isar/isar.dart';
|
|
|
|
import 'package:qr_flutter/qr_flutter.dart';
|
|
|
|
import 'package:stackwallet/db/main_db.dart';
|
|
|
|
import 'package:stackwallet/models/isar/models/isar_models.dart';
|
|
|
|
import 'package:stackwallet/pages/receive_view/addresses/address_tag.dart';
|
2023-03-22 19:28:28 +00:00
|
|
|
import 'package:stackwallet/pages/wallet_view/sub_widgets/no_transactions_found.dart';
|
2023-03-21 23:18:07 +00:00
|
|
|
import 'package:stackwallet/providers/global/wallets_provider.dart';
|
|
|
|
import 'package:stackwallet/utilities/address_utils.dart';
|
|
|
|
import 'package:stackwallet/utilities/text_styles.dart';
|
|
|
|
import 'package:stackwallet/utilities/theme/stack_colors.dart';
|
|
|
|
import 'package:stackwallet/utilities/util.dart';
|
|
|
|
import 'package:stackwallet/widgets/background.dart';
|
|
|
|
import 'package:stackwallet/widgets/conditional_parent.dart';
|
|
|
|
import 'package:stackwallet/widgets/custom_buttons/app_bar_icon_button.dart';
|
|
|
|
import 'package:stackwallet/widgets/custom_buttons/simple_copy_button.dart';
|
|
|
|
import 'package:stackwallet/widgets/custom_buttons/simple_edit_button.dart';
|
|
|
|
import 'package:stackwallet/widgets/rounded_white_container.dart';
|
2023-03-22 19:28:28 +00:00
|
|
|
import 'package:stackwallet/widgets/transaction_card.dart';
|
2023-03-21 23:18:07 +00:00
|
|
|
|
|
|
|
class AddressDetailsView extends ConsumerStatefulWidget {
|
|
|
|
const AddressDetailsView({
|
|
|
|
Key? key,
|
|
|
|
required this.addressId,
|
|
|
|
required this.walletId,
|
|
|
|
}) : super(key: key);
|
|
|
|
|
|
|
|
static const String routeName = "/addressDetailsView";
|
|
|
|
|
|
|
|
final Id addressId;
|
|
|
|
final String walletId;
|
|
|
|
|
|
|
|
@override
|
|
|
|
ConsumerState<AddressDetailsView> createState() => _AddressDetailsViewState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _AddressDetailsViewState extends ConsumerState<AddressDetailsView> {
|
|
|
|
final _qrKey = GlobalKey();
|
|
|
|
final isDesktop = Util.isDesktop;
|
|
|
|
|
|
|
|
late Stream<AddressLabel?> stream;
|
|
|
|
late final Address address;
|
|
|
|
|
|
|
|
AddressLabel? label;
|
|
|
|
|
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
address = MainDB.instance.isar.addresses
|
|
|
|
.where()
|
|
|
|
.idEqualTo(widget.addressId)
|
|
|
|
.findFirstSync()!;
|
|
|
|
|
|
|
|
label = MainDB.instance.getAddressLabelSync(widget.walletId, address.value);
|
|
|
|
Id? id = label?.id;
|
|
|
|
if (id == null) {
|
|
|
|
label = AddressLabel(
|
|
|
|
walletId: widget.walletId,
|
|
|
|
addressString: address.value,
|
|
|
|
value: "",
|
|
|
|
tags: address.subType == AddressSubType.receiving
|
|
|
|
? ["receiving"]
|
|
|
|
: address.subType == AddressSubType.change
|
|
|
|
? ["change"]
|
|
|
|
: null,
|
|
|
|
);
|
|
|
|
id = MainDB.instance.putAddressLabelSync(label!);
|
|
|
|
}
|
|
|
|
stream = MainDB.instance.watchAddressLabel(id: id);
|
|
|
|
|
|
|
|
super.initState();
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
final coin = ref.watch(walletsChangeNotifierProvider
|
|
|
|
.select((value) => value.getManager(widget.walletId).coin));
|
2023-03-22 19:28:28 +00:00
|
|
|
final query = MainDB.instance
|
|
|
|
.getTransactions(widget.walletId)
|
|
|
|
.filter()
|
|
|
|
.address((q) => q.valueEqualTo(address.value));
|
|
|
|
|
2023-03-21 23:18:07 +00:00
|
|
|
return ConditionalParent(
|
|
|
|
condition: !isDesktop,
|
|
|
|
builder: (child) => Background(
|
|
|
|
child: Scaffold(
|
|
|
|
backgroundColor:
|
|
|
|
Theme.of(context).extension<StackColors>()!.background,
|
|
|
|
appBar: AppBar(
|
|
|
|
backgroundColor:
|
|
|
|
Theme.of(context).extension<StackColors>()!.backgroundAppBar,
|
|
|
|
leading: AppBarBackButton(
|
|
|
|
onPressed: () {
|
|
|
|
Navigator.of(context).pop();
|
|
|
|
},
|
|
|
|
),
|
|
|
|
titleSpacing: 0,
|
|
|
|
title: Text(
|
|
|
|
"Wallet addresses",
|
|
|
|
style: STextStyles.navBarTitle(context),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
body: SafeArea(
|
2023-03-22 19:28:28 +00:00
|
|
|
child: LayoutBuilder(
|
|
|
|
builder: (builderContext, constraints) {
|
|
|
|
return SingleChildScrollView(
|
|
|
|
child: ConstrainedBox(
|
|
|
|
constraints: BoxConstraints(
|
|
|
|
minHeight: constraints.maxHeight,
|
|
|
|
),
|
|
|
|
child: Padding(
|
|
|
|
padding: const EdgeInsets.all(16),
|
|
|
|
child: child,
|
2023-03-21 23:18:07 +00:00
|
|
|
),
|
2023-03-22 19:28:28 +00:00
|
|
|
),
|
|
|
|
);
|
2023-03-21 23:18:07 +00:00
|
|
|
},
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
child: StreamBuilder<AddressLabel?>(
|
|
|
|
stream: stream,
|
|
|
|
builder: (context, snapshot) {
|
|
|
|
if (snapshot.hasData) {
|
|
|
|
label = snapshot.data!;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Column(
|
2023-03-22 19:28:28 +00:00
|
|
|
crossAxisAlignment: CrossAxisAlignment.stretch,
|
2023-03-21 23:18:07 +00:00
|
|
|
children: [
|
|
|
|
Center(
|
|
|
|
child: RepaintBoundary(
|
|
|
|
key: _qrKey,
|
|
|
|
child: QrImage(
|
|
|
|
data: AddressUtils.buildUriString(
|
|
|
|
coin,
|
|
|
|
address.value,
|
|
|
|
{},
|
|
|
|
),
|
|
|
|
size: 220,
|
|
|
|
backgroundColor:
|
|
|
|
Theme.of(context).extension<StackColors>()!.background,
|
|
|
|
foregroundColor: Theme.of(context)
|
|
|
|
.extension<StackColors>()!
|
|
|
|
.accentColorDark,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
const SizedBox(
|
|
|
|
height: 16,
|
|
|
|
),
|
|
|
|
_Item(
|
|
|
|
title: "Address",
|
|
|
|
data: address.value,
|
|
|
|
button: SimpleCopyButton(
|
|
|
|
data: address.value,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
const SizedBox(
|
|
|
|
height: 12,
|
|
|
|
),
|
|
|
|
_Item(
|
|
|
|
title: "Label",
|
|
|
|
data: label!.value,
|
|
|
|
button: SimpleEditButton(
|
|
|
|
editValue: label!.value,
|
|
|
|
editLabel: 'label',
|
|
|
|
onValueChanged: (value) {
|
|
|
|
MainDB.instance.putAddressLabel(
|
|
|
|
label!.copyWith(
|
|
|
|
label: value,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
|
|
|
),
|
|
|
|
const SizedBox(
|
|
|
|
height: 12,
|
|
|
|
),
|
|
|
|
_Tags(
|
|
|
|
tags: label!.tags,
|
|
|
|
),
|
|
|
|
if (address.derivationPath != null)
|
|
|
|
const SizedBox(
|
|
|
|
height: 12,
|
|
|
|
),
|
|
|
|
if (address.derivationPath != null)
|
|
|
|
_Item(
|
|
|
|
title: "Derivation path",
|
|
|
|
data: address.derivationPath!.value,
|
2023-03-22 19:28:28 +00:00
|
|
|
button: Container(),
|
2023-03-21 23:18:07 +00:00
|
|
|
),
|
|
|
|
const SizedBox(
|
|
|
|
height: 12,
|
|
|
|
),
|
|
|
|
_Item(
|
|
|
|
title: "Type",
|
|
|
|
data: address.type.readableName,
|
|
|
|
button: Container(),
|
|
|
|
),
|
|
|
|
const SizedBox(
|
|
|
|
height: 12,
|
|
|
|
),
|
|
|
|
_Item(
|
|
|
|
title: "Sub type",
|
|
|
|
data: address.subType.prettyName,
|
|
|
|
button: Container(),
|
|
|
|
),
|
|
|
|
const SizedBox(
|
|
|
|
height: 20,
|
|
|
|
),
|
|
|
|
Text(
|
|
|
|
"Transactions",
|
2023-03-22 19:28:28 +00:00
|
|
|
textAlign: TextAlign.left,
|
2023-03-21 23:18:07 +00:00
|
|
|
style: STextStyles.itemSubtitle(context).copyWith(
|
|
|
|
color: Theme.of(context).extension<StackColors>()!.textDark3,
|
|
|
|
),
|
|
|
|
),
|
2023-03-22 19:28:28 +00:00
|
|
|
const SizedBox(
|
|
|
|
height: 12,
|
|
|
|
),
|
|
|
|
if (query.countSync() == 0) const NoTransActionsFound(),
|
|
|
|
if (query.countSync() > 0)
|
|
|
|
RoundedWhiteContainer(
|
|
|
|
padding: EdgeInsets.zero,
|
|
|
|
child: Column(
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
children: query
|
|
|
|
.findAllSync()
|
|
|
|
.map((e) => TransactionCard(
|
|
|
|
transaction: e, walletId: widget.walletId))
|
|
|
|
.toList(),
|
|
|
|
),
|
|
|
|
),
|
2023-03-21 23:18:07 +00:00
|
|
|
],
|
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class _Tags extends StatelessWidget {
|
|
|
|
const _Tags({
|
|
|
|
Key? key,
|
|
|
|
required this.tags,
|
|
|
|
}) : super(key: key);
|
|
|
|
|
|
|
|
final List<String>? tags;
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return RoundedWhiteContainer(
|
|
|
|
child: Column(
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
children: [
|
|
|
|
Row(
|
|
|
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
|
|
children: [
|
|
|
|
Text(
|
|
|
|
"Tags",
|
|
|
|
style: STextStyles.itemSubtitle(context),
|
|
|
|
),
|
2023-03-22 19:28:28 +00:00
|
|
|
Container(),
|
|
|
|
// SimpleEditButton(
|
|
|
|
// onPressedOverride: () {
|
|
|
|
// // TODO edit tags
|
|
|
|
// },
|
|
|
|
// ),
|
2023-03-21 23:18:07 +00:00
|
|
|
],
|
|
|
|
),
|
|
|
|
const SizedBox(
|
|
|
|
height: 8,
|
|
|
|
),
|
|
|
|
tags != null && tags!.isNotEmpty
|
|
|
|
? Wrap(
|
|
|
|
spacing: 10,
|
|
|
|
runSpacing: 10,
|
|
|
|
children: tags!
|
|
|
|
.map(
|
|
|
|
(e) => AddressTag(
|
|
|
|
tag: e,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
.toList(),
|
|
|
|
)
|
|
|
|
: Text(
|
|
|
|
"Tags will appear here",
|
|
|
|
style: STextStyles.w500_14(context).copyWith(
|
|
|
|
color: Theme.of(context)
|
|
|
|
.extension<StackColors>()!
|
|
|
|
.textSubtitle3,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class _Item extends StatelessWidget {
|
|
|
|
const _Item({
|
|
|
|
Key? key,
|
|
|
|
required this.title,
|
|
|
|
required this.data,
|
|
|
|
required this.button,
|
|
|
|
}) : super(key: key);
|
|
|
|
|
|
|
|
final String title;
|
|
|
|
final String data;
|
|
|
|
final Widget button;
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return RoundedWhiteContainer(
|
|
|
|
child: Column(
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
children: [
|
|
|
|
Row(
|
|
|
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
|
|
children: [
|
|
|
|
Text(
|
|
|
|
title,
|
|
|
|
style: STextStyles.itemSubtitle(context),
|
|
|
|
),
|
|
|
|
button,
|
|
|
|
],
|
|
|
|
),
|
|
|
|
const SizedBox(
|
|
|
|
height: 5,
|
|
|
|
),
|
|
|
|
data.isNotEmpty
|
|
|
|
? SelectableText(
|
|
|
|
data,
|
|
|
|
style: STextStyles.w500_14(context),
|
|
|
|
)
|
|
|
|
: Text(
|
|
|
|
"$title will appear here",
|
|
|
|
style: STextStyles.w500_14(context).copyWith(
|
|
|
|
color: Theme.of(context)
|
|
|
|
.extension<StackColors>()!
|
|
|
|
.textSubtitle3,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|