import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:font_awesome_flutter/font_awesome_flutter.dart'; import 'package:invoiceninja_flutter/ui/app/form_card.dart'; import 'package:invoiceninja_flutter/ui/app/forms/app_form.dart'; import 'package:invoiceninja_flutter/ui/app/forms/bool_dropdown_button.dart'; import 'package:invoiceninja_flutter/ui/app/forms/decorated_form_field.dart'; import 'package:invoiceninja_flutter/ui/app/lists/list_divider.dart'; import 'package:invoiceninja_flutter/ui/settings/client_portal_vm.dart'; import 'package:invoiceninja_flutter/ui/app/edit_scaffold.dart'; import 'package:invoiceninja_flutter/utils/completers.dart'; import 'package:invoiceninja_flutter/utils/localization.dart'; class ClientPortal extends StatefulWidget { const ClientPortal({ Key key, @required this.viewModel, }) : super(key: key); final ClientPortalVM viewModel; @override _ClientPortalState createState() => _ClientPortalState(); } class _ClientPortalState extends State with SingleTickerProviderStateMixin { static final GlobalKey _formKey = GlobalKey(debugLabel: '_clientPortal'); final FocusScopeNode _focusNode = FocusScopeNode(); TabController _controller; bool autoValidate = false; final _debouncer = Debouncer(); final _subdomainController = TextEditingController(); final _portalDomainController = TextEditingController(); final _customCssController = TextEditingController(); final _customJavaScriptController = TextEditingController(); final _customMessageDashboard = TextEditingController(); final _customMessageUnpaidInvoice = TextEditingController(); final _customMessagePaidInvoice = TextEditingController(); final _customMessageUnapprovedQuote = TextEditingController(); final _termsController = TextEditingController(); final _privacyController = TextEditingController(); List _controllers = []; @override void initState() { super.initState(); _controller = TabController(vsync: this, length: 2); } @override void dispose() { _controller.dispose(); _controllers.forEach((dynamic controller) { controller.removeListener(_onChanged); controller.dispose(); }); super.dispose(); } @override void didChangeDependencies() { _controllers = [ _subdomainController, _portalDomainController, _customCssController, _customJavaScriptController, _customMessageDashboard, _customMessageUnpaidInvoice, _customMessagePaidInvoice, _customMessageUnapprovedQuote, _termsController, _privacyController, ]; _controllers .forEach((dynamic controller) => controller.removeListener(_onChanged)); final company = widget.viewModel.company; final settings = widget.viewModel.settings; _portalDomainController.text = company.portalDomain; _subdomainController.text = company.subdomain; _customMessageDashboard.text = settings.customMessageDashboard; _customMessagePaidInvoice.text = settings.customMessagePaidInvoice; _customMessageUnpaidInvoice.text = settings.customMessageUnpaidInvoice; _customMessageUnapprovedQuote.text = settings.customMessageUnapprovedQuote; _privacyController.text = settings.clientPortalPrivacy; _termsController.text = settings.clientPortalTerms; _controllers .forEach((dynamic controller) => controller.addListener(_onChanged)); super.didChangeDependencies(); } void _onChanged() { _debouncer.run(() { final company = widget.viewModel.company.rebuild((b) => b ..portalDomain = _portalDomainController.text.trim() ..subdomain = _subdomainController.text.trim()); if (company != widget.viewModel.company) { widget.viewModel.onCompanyChanged(company); } final settings = widget.viewModel.settings.rebuild((b) => b ..customMessageDashboard = _customMessageDashboard.text.trim() ..customMessageUnpaidInvoice = _customMessageUnpaidInvoice.text.trim() ..customMessagePaidInvoice = _customMessagePaidInvoice.text.trim() ..customMessageUnapprovedQuote = _customMessageUnapprovedQuote.text.trim() ..clientPortalTerms = _termsController.text.trim() ..clientPortalPrivacy = _privacyController.text.trim()); if (settings != widget.viewModel.settings) { widget.viewModel.onSettingsChanged(settings); } }); } void _onSavePressed(BuildContext context) { final bool isValid = _formKey.currentState.validate(); setState(() { autoValidate = !isValid; }); if (!isValid) { return; } widget.viewModel.onSavePressed(context); } @override Widget build(BuildContext context) { final localization = AppLocalization.of(context); final viewModel = widget.viewModel; final state = viewModel.state; final company = viewModel.company; final settings = viewModel.settings; return EditScaffold( title: localization.clientPortal, onSavePressed: (context) => _onSavePressed(context), appBarBottom: TabBar( key: ValueKey(state.settingsUIState.updatedAt), controller: _controller, tabs: [ Tab( text: localization.settings, ), Tab( text: localization.authorization, ), /* Tab( text: localization.messages, ), Tab( text: localization.customize, ), */ ], ), body: AppTabForm( tabController: _controller, formKey: _formKey, focusNode: _focusNode, children: [ ListView( children: [ /* if (!state.settingsUIState.isFiltered) FormCard( children: [ AppDropdownButton( labelText: localization.portalMode, value: viewModel.company.portalMode, onChanged: (dynamic value) => viewModel.onCompanyChanged( viewModel.company .rebuild((b) => b..portalMode = value)), items: [ DropdownMenuItem( child: Text(localization.subdomain), value: kClientPortalModeSubdomain, ), if (company.isEnterprisePlan) DropdownMenuItem( child: Text(localization.domain), value: kClientPortalModeDomain, ), DropdownMenuItem( child: Text('iFrame'), value: kClientPortalModeIFrame, ), ], ), if (company.portalMode != kClientPortalModeSubdomain) DecoratedFormField( label: company.portalMode == kClientPortalModeDomain ? localization.domainUrl : localization.iFrameUrl, controller: _portalDomainController, keyboardType: TextInputType.url, validator: (val) => val.isEmpty || val.trim().isEmpty ? localization.pleaseEnterAValue : null, ), DecoratedFormField( label: localization.subdomain, controller: _subdomainController, ), ], ), */ FormCard( children: [ BoolDropdownButton( label: localization.clientPortal, value: settings.enablePortal, iconData: FontAwesomeIcons.cloud, onChanged: (value) => viewModel.onSettingsChanged( settings.rebuild((b) => b..enablePortal = value)), ), BoolDropdownButton( label: localization.dashboard, value: settings.enablePortalDashboard, iconData: FontAwesomeIcons.tachometerAlt, onChanged: (value) => viewModel.onSettingsChanged(settings .rebuild((b) => b..enablePortalDashboard = value)), ), /* Re-enable with tasks BoolDropdownButton( label: localization.tasks, value: settings.enablePortalTasks, iconData: getEntityIcon(EntityType.task), onChanged: (value) => viewModel.onSettingsChanged( settings.rebuild((b) => b..enablePortalTasks = value)), ), */ ], ), FormCard( crossAxisAlignment: CrossAxisAlignment.start, children: [ BoolDropdownButton( label: localization.clientRegistration, helpLabel: localization.clientRegistrationHelp, value: company.clientCanRegister, iconData: FontAwesomeIcons.userPlus, onChanged: (value) => viewModel.onCompanyChanged( company.rebuild((b) => b..clientCanRegister = value)), ), BoolDropdownButton( label: localization.storefront, helpLabel: localization.storefrontHelp, value: company.enableShopApi, iconData: FontAwesomeIcons.shoppingCart, onChanged: (value) => viewModel.onCompanyChanged( company.rebuild((b) => b..enableShopApi = value)), ), if (state.company.enableShopApi ?? false) ...[ SizedBox(height: 16), ListDivider(), Builder(builder: (BuildContext context) { return ListTile( title: Padding( padding: const EdgeInsets.all(16), child: Text( '${localization.companyKey}: ${company.companyKey.substring(0, 16)}...', style: Theme.of(context).textTheme.headline6, overflow: TextOverflow.ellipsis, ), ), trailing: Icon(Icons.content_copy), onTap: () { Clipboard.setData( ClipboardData(text: company.companyKey)); Scaffold.of(context).showSnackBar(SnackBar( content: Text(localization.copiedToClipboard .replaceFirst( ':value ', company.companyKey)))); }, ); }), ListDivider(), ], DecoratedFormField( controller: _termsController, label: localization.termsOfService, maxLines: 6, ), DecoratedFormField( controller: _privacyController, label: localization.privacyPolicy, maxLines: 6, ), ], ) ], ), ListView( children: [ FormCard( children: [ BoolDropdownButton( label: localization.enablePortalPassword, helpLabel: localization.enablePortalPasswordHelp, value: settings.enablePortalPassword, iconData: FontAwesomeIcons.shieldAlt, onChanged: (value) => viewModel.onSettingsChanged(settings .rebuild((b) => b..enablePortalPassword = value)), ), ], ), FormCard( children: [ BoolDropdownButton( label: localization.showAcceptInvoiceTerms, helpLabel: localization.showAcceptInvoiceTermsHelp, value: settings.showAcceptInvoiceTerms, iconData: FontAwesomeIcons.checkSquare, onChanged: (value) => viewModel.onSettingsChanged(settings .rebuild((b) => b..showAcceptInvoiceTerms = value)), ), BoolDropdownButton( label: localization.showAcceptQuoteTerms, helpLabel: localization.showAcceptQuoteTermsHelp, value: settings.showAcceptQuoteTerms, iconData: FontAwesomeIcons.checkSquare, onChanged: (value) => viewModel.onSettingsChanged(settings .rebuild((b) => b..showAcceptQuoteTerms = value)), ), ], ), FormCard( children: [ BoolDropdownButton( label: localization.requireInvoiceSignature, helpLabel: localization.requireInvoiceSignatureHelp, value: settings.requireInvoiceSignature, iconData: FontAwesomeIcons.fileSignature, onChanged: (value) => viewModel.onSettingsChanged(settings .rebuild((b) => b..requireInvoiceSignature = value)), ), BoolDropdownButton( label: localization.requireQuoteSignature, helpLabel: localization.requireInvoiceSignatureHelp, value: settings.requireQuoteSignature, iconData: FontAwesomeIcons.fileSignature, onChanged: (value) => viewModel.onSettingsChanged(settings .rebuild((b) => b..requireQuoteSignature = value)), ), /* TODO Re-enable BoolDropdownButton( label: localization.signatureOnPdf, helpLabel: localization.signatureOnPdfHelp, value: settings.signatureOnPdf, iconData: FontAwesomeIcons.fileContract, onChanged: (value) => viewModel.onSettingsChanged( settings.rebuild((b) => b..signatureOnPdf = value)), ), */ ], ), ], ), /* ListView( children: [ FormCard( children: [ DecoratedFormField( controller: _customMessageDashboard, label: localization.dashboard, maxLines: 6, ), DecoratedFormField( controller: _customMessageUnpaidInvoice, label: localization.unpaidInvoice, maxLines: 6, ), DecoratedFormField( controller: _customMessagePaidInvoice, label: localization.paidInvoice, maxLines: 6, ), DecoratedFormField( controller: _customMessageUnapprovedQuote, label: localization.unapprovedQuote, maxLines: 6, ), ], ), ], ), ListView( children: [ FormCard( children: [ DecoratedFormField( label: localization.customCss, controller: _customCssController, maxLines: 8, ), if (isSelfHosted(context)) DecoratedFormField( label: localization.customJavascript, controller: _customJavaScriptController, maxLines: 8, ), ], ) ], ), */ ], ), ); } }