import 'package:flutter/material.dart'; import 'package:flutter_mobx/flutter_mobx.dart'; import 'package:flutter_svg/svg.dart'; import 'package:material_design_icons_flutter/material_design_icons_flutter.dart'; import 'package:warp_api/warp_api.dart'; import 'main.dart'; import 'generated/l10n.dart'; import 'settings.dart'; import 'store.dart'; class ContactsTab extends StatefulWidget { ContactsTab({key: Key}): super(key: key); @override State createState() => ContactsState(); } class ContactsState extends State { @override Widget build(BuildContext context) { return Observer(builder: (context) => Padding(padding: EdgeInsets.all(8), child: contacts.contacts.isEmpty ? NoContact() : Column(children: [ if (!settings.coins[active.coin].contactsSaved) OutlinedButton(onPressed: active.canPay ? _onCommit : null, child: Text(S.of(context).saveToBlockchain), style: OutlinedButton.styleFrom( side: BorderSide( width: 1, color: Theme.of(context).primaryColor))), Expanded(child: ListView.builder( itemCount: contacts.contacts.length, itemBuilder: (BuildContext context, int index) { final c = contacts.contacts[index]; return Card( child: Dismissible( key: Key("${c.id}"), child: ListTile( title: Text(c.name, style: Theme.of(context).textTheme.headline5), subtitle: Text(c.address), trailing: Icon(Icons.chevron_right), onTap: () { _onContact(c); }, onLongPress: () { _editContact(c); }, ), confirmDismiss: (_) async { return await _onConfirmDelContact(c); }, onDismissed: (_) { _delContact(c); })); }) )])) ); } _onContact(Contact c) { Navigator.of(context).pushNamed('/send', arguments: SendPageArgs(contact: c)); } _editContact(Contact c) async { final contact = await showContactForm(context, c, true); if (contact != null) { contacts.add(contact); } } Future _onConfirmDelContact(Contact c) async { final confirm = await showMessageBox(context, S.of(context).deleteContact, S.of(context).areYouSureYouWantToDeleteThisContact, S.of(context).delete); return confirm; } _delContact(Contact c) async { await contacts.remove(c); } Future showContactForm(BuildContext context, Contact c, bool edit) async { final key = GlobalKey(); final contact = await showDialog( context: context, builder: (context) => AlertDialog( contentPadding: EdgeInsets.all(16), title: Text(edit ? S.of(context).editContact : S.of(context).addContact), content: ContactForm(c, key: key), actions: confirmButtons(context, () { key.currentState!.onOK(); }), )); return contact; } _onCommit() async { final approve = await showMessageBox(context, S.of(context).saveToBlockchain, S.of(context).areYouSureYouWantToSaveYourContactsIt(activeCoin().ticker), S.of(context).ok); if (approve) { final tx = await WarpApi.commitUnsavedContacts( active.coin, active.id, settings.anchorOffset); final snackBar = SnackBar(content: Text("${S.of(context).txId}: $tx")); rootScaffoldMessengerKey.currentState?.showSnackBar(snackBar); contacts.markContactsSaved(active.coin, true); } } } class NoContact extends StatelessWidget { @override Widget build(BuildContext context) { final Widget contact = SvgPicture.asset('assets/contacts.svg', color: Theme.of(context).primaryColor, semanticsLabel: 'Contacts'); return Column(mainAxisAlignment: MainAxisAlignment.center, children: [ SizedBox(child: contact, height: 150, width: 150), Padding(padding: EdgeInsets.symmetric(vertical: 16)), Text(S.of(context).noContacts, style: Theme.of(context).textTheme.headline5), Padding(padding: EdgeInsets.symmetric(vertical: 8)), Text(S.of(context).createANewContactAndItWillShowUpHere, style: Theme.of(context).textTheme.bodyText1), ]); } } class ContactForm extends StatefulWidget { final Contact contact; ContactForm(this.contact, {Key? key}) : super(key: key); @override State createState() => ContactState(); } class ContactState extends State { final formKey = GlobalKey(); final nameController = TextEditingController(); var address = ""; @override void initState() { super.initState(); nameController.text = widget.contact.name; address = widget.contact.address; } @override Widget build(BuildContext context) { return Form( key: formKey, child: SingleChildScrollView( child: Column(children: [ TextFormField( decoration: InputDecoration(labelText: S.of(context).contactName), controller: nameController, validator: _checkName, ), AddressInput(widget.contact.id, S.of(context).address, address, (addr) { address = addr ?? ""; }) ]))); } onOK() { final state = formKey.currentState!; if (state.validate()) { state.save(); final contact = Contact(widget.contact.id, nameController.text, address); Navigator.of(context).pop(contact); active.update(); } } String? _checkName(String? v) { if (v == null || v.isEmpty) return S.of(context).nameIsEmpty; return null; } } class AddressInput extends StatefulWidget { final int id; final void Function(String?) onSaved; final String labelText; final String initialValue; AddressInput(this.id, this.labelText, this.initialValue, this.onSaved); @override State createState() => AddressState(); } class AddressState extends State { final _addressController = TextEditingController(); @override void initState() { super.initState(); _addressController.text = widget.initialValue; } @override Widget build(BuildContext context) { return Row(children: [ Expanded( child: TextFormField( decoration: InputDecoration(labelText: widget.labelText), minLines: 4, maxLines: null, keyboardType: TextInputType.multiline, controller: _addressController, validator: _checkAddress, onSaved: widget.onSaved, ), ), IconButton(icon: new Icon(MdiIcons.qrcodeScan), onPressed: _onScan) ]); } String? _checkAddress(String? v) { if (v == null || v.isEmpty) return S.of(context).addressIsEmpty; final zaddr = WarpApi.getSaplingFromUA(v); if (zaddr.isNotEmpty) return null; if (!WarpApi.validAddress(active.coin, v)) return S.of(context).invalidAddress; if (contacts.contacts.where((c) => c.address == v && c.id != widget.id).isNotEmpty) return S.of(context).duplicateContact; return null; } void _onScan() async { var address = await scanCode(context); if (address != null) setState(() { _addressController.text = address; }); } }