import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:invoiceninja_flutter/constants.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/decorated_form_field.dart'; import 'package:invoiceninja_flutter/ui/settings/custom_fields_vm.dart'; import 'package:invoiceninja_flutter/ui/settings/settings_scaffold.dart'; import 'package:invoiceninja_flutter/utils/localization.dart'; class CustomFields extends StatefulWidget { const CustomFields({ Key key, @required this.viewModel, }) : super(key: key); final CustomFieldsVM viewModel; @override _CustomFieldsState createState() => _CustomFieldsState(); } class _CustomFieldsState extends State with SingleTickerProviderStateMixin { static final GlobalKey _formKey = GlobalKey(); FocusScopeNode _focusNode; TabController _controller; bool autoValidate = false; final _recurringPrefixController = TextEditingController(); List _controllers = []; @override void initState() { super.initState(); _focusNode = FocusScopeNode(); _controller = TabController(vsync: this, length: 5); } @override void dispose() { _focusNode.dispose(); _controller.dispose(); _controllers.forEach((dynamic controller) { controller.removeListener(_onChanged); controller.dispose(); }); super.dispose(); } @override void didChangeDependencies() { _controllers = [ _recurringPrefixController, ]; _controllers .forEach((dynamic controller) => controller.removeListener(_onChanged)); final settings = widget.viewModel.settings; _recurringPrefixController.text = settings.recurringInvoiceNumberPrefix; _controllers .forEach((dynamic controller) => controller.addListener(_onChanged)); super.didChangeDependencies(); } void _onChanged() { final settings = widget.viewModel.settings.rebuild((b) => b ..recurringInvoiceNumberPrefix = _recurringPrefixController.text.trim()); if (settings != widget.viewModel.settings) { widget.viewModel.onSettingsChanged(settings); } } @override Widget build(BuildContext context) { final localization = AppLocalization.of(context); final viewModel = widget.viewModel; final state = viewModel.state; return SettingsScaffold( title: localization.customFields, onSavePressed: viewModel.onSavePressed, appBarBottom: TabBar( key: ValueKey(state.settingsUIState.updatedAt), controller: _controller, isScrollable: true, tabs: [ Tab( text: localization.company, ), Tab( text: localization.clients, ), Tab( text: localization.invoices, ), Tab( text: localization.credits, ), Tab( text: localization.payments, ), ], ), body: AppTabForm( tabController: _controller, formKey: _formKey, focusNode: _focusNode, children: [ ListView( children: [ CustomFieldsSettings( viewModel: viewModel, fieldLabel: localization.customCompanyField, valueLabel: localization.customCompanyValue, showValues: true, ), ], ), ListView(children: [ CustomFieldsSettings( viewModel: viewModel, fieldLabel: localization.customClientField, ), CustomFieldsSettings( viewModel: viewModel, fieldLabel: localization.customContactField, ), ]), ListView(children: [ CustomFieldsSettings( viewModel: viewModel, fieldLabel: localization.customInvoiceField, ), CustomFieldsSettings( viewModel: viewModel, fieldLabel: localization.customInvoiceSurcharge, showChargeTaxes: true, ), ]), ListView(children: [ CustomFieldsSettings( viewModel: viewModel, fieldLabel: localization.customCreditField, ), ]), ListView(children: [ CustomFieldsSettings( viewModel: viewModel, fieldLabel: localization.customPaymentField, ), ]), ], ), ); } } class CustomFieldsSettings extends StatefulWidget { const CustomFieldsSettings( {@required this.fieldLabel, @required this.viewModel, this.showChargeTaxes = false, this.showValues = false, this.valueLabel}); final CustomFieldsVM viewModel; final String fieldLabel; final String valueLabel; final bool showChargeTaxes; final bool showValues; @override _CustomFieldsSettingsState createState() => _CustomFieldsSettingsState(); } class _CustomFieldsSettingsState extends State { final _customField1Controller = TextEditingController(); final _customField2Controller = TextEditingController(); final _customField3Controller = TextEditingController(); final _customField4Controller = TextEditingController(); final _customValue1Controller = TextEditingController(); final _customValue2Controller = TextEditingController(); final _customValue3Controller = TextEditingController(); final _customValue4Controller = TextEditingController(); List _controllers = []; @override void dispose() { _controllers.forEach((dynamic controller) { controller.removeListener(_onChanged); controller.dispose(); }); super.dispose(); } @override void didChangeDependencies() { _controllers = [ _customField1Controller, _customField2Controller, _customField3Controller, _customField4Controller, _customValue1Controller, _customValue2Controller, _customValue3Controller, _customValue4Controller, ]; _controllers .forEach((dynamic controller) => controller.removeListener(_onChanged)); /* final product = widget.viewModel.product; _productKeyController.text = product.productKey; */ _controllers .forEach((dynamic controller) => controller.addListener(_onChanged)); super.didChangeDependencies(); } void _onChanged() { /* final product = widget.viewModel.product.rebuild((b) => b ..customValue2 = _custom2Controller.text.trim()); if (product != widget.viewModel.product) { widget.viewModel.onChanged(product); } */ } @override Widget build(BuildContext context) { final viewModel = widget.viewModel; final company = viewModel.company; return FormCard( children: [ Row( children: [ Expanded( child: CustomFormField( label: widget.fieldLabel, controller: _customField1Controller, showTaxes: widget.showChargeTaxes, taxesEnabled: company.enableCustomSurchargeTaxes1, onTaxesChanged: (value) => viewModel.onCompanyChanged(company .rebuild((b) => b..enableCustomSurchargeTaxes1 = value)), ), ), if (widget.showValues) ...[ SizedBox(width: kGutterWidth), Expanded( child: CustomFormField( label: widget.valueLabel, controller: _customValue1Controller, ), ), ] ], ), Row( children: [ Expanded( child: CustomFormField( label: widget.fieldLabel, controller: _customField2Controller, showTaxes: widget.showChargeTaxes, taxesEnabled: company.enableCustomSurchargeTaxes2, onTaxesChanged: (value) => viewModel.onCompanyChanged(company .rebuild((b) => b..enableCustomSurchargeTaxes2 = value)), ), ), if (widget.showValues) ...[ SizedBox(width: kGutterWidth), Expanded( child: CustomFormField( label: widget.valueLabel, controller: _customValue2Controller, ), ), ] ], ), Row( children: [ Expanded( child: CustomFormField( label: widget.fieldLabel, controller: _customField3Controller, showTaxes: widget.showChargeTaxes, taxesEnabled: company.enableCustomSurchargeTaxes3, onTaxesChanged: (value) => viewModel.onCompanyChanged(company .rebuild((b) => b..enableCustomSurchargeTaxes3 = value)), ), ), if (widget.showValues) ...[ SizedBox(width: kGutterWidth), Expanded( child: CustomFormField( label: widget.valueLabel, controller: _customValue3Controller, ), ), ] ], ), Row( children: [ Expanded( child: CustomFormField( label: widget.fieldLabel, controller: _customField4Controller, showTaxes: widget.showChargeTaxes, taxesEnabled: company.enableCustomSurchargeTaxes4, onTaxesChanged: (value) => viewModel.onCompanyChanged(company .rebuild((b) => b..enableCustomSurchargeTaxes4 = value)), ), ), if (widget.showValues) ...[ SizedBox(width: kGutterWidth), Expanded( child: CustomFormField( label: widget.valueLabel, controller: _customValue4Controller, ), ), ] ], ), ], ); } } class CustomFormField extends StatelessWidget { const CustomFormField({ @required this.label, @required this.controller, this.showTaxes = false, this.taxesEnabled, this.onTaxesChanged, }); final String label; final TextEditingController controller; final bool showTaxes; final bool taxesEnabled; final Function(bool) onTaxesChanged; @override Widget build(BuildContext context) { final localization = AppLocalization.of(context); return Row( children: [ Expanded( child: DecoratedFormField( label: label, controller: controller, ), ), if (showTaxes) ...[ Checkbox( activeColor: Theme.of(context).accentColor, value: taxesEnabled ?? false, onChanged: onTaxesChanged, ), GestureDetector( child: Text(localization.chargeTaxes), onTap: () => onTaxesChanged(!taxesEnabled), ), ] ], ); } }