invoice/lib/ui/settings/account_management.dart

354 lines
13 KiB
Dart

import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter_redux/flutter_redux.dart';
import 'package:invoiceninja_flutter/constants.dart';
import 'package:invoiceninja_flutter/data/models/entities.dart';
import 'package:invoiceninja_flutter/data/web_client.dart';
import 'package:invoiceninja_flutter/redux/app/app_state.dart';
import 'package:invoiceninja_flutter/redux/settings/settings_actions.dart';
import 'package:invoiceninja_flutter/ui/app/app_header.dart';
import 'package:invoiceninja_flutter/ui/app/buttons/elevated_button.dart';
import 'package:invoiceninja_flutter/ui/app/dialogs/loading_dialog.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/edit_scaffold.dart';
import 'package:invoiceninja_flutter/ui/app/lists/list_divider.dart';
import 'package:invoiceninja_flutter/ui/settings/account_management_vm.dart';
import 'package:invoiceninja_flutter/utils/dialogs.dart';
import 'package:invoiceninja_flutter/utils/formatting.dart';
import 'package:invoiceninja_flutter/utils/icons.dart';
import 'package:invoiceninja_flutter/utils/localization.dart';
import 'package:material_design_icons_flutter/material_design_icons_flutter.dart';
import 'package:url_launcher/url_launcher.dart';
class AccountManagement extends StatefulWidget {
const AccountManagement({
Key key,
@required this.viewModel,
}) : super(key: key);
final AccountManagementVM viewModel;
@override
_AccountManagementState createState() => _AccountManagementState();
}
class _AccountManagementState extends State<AccountManagement>
with SingleTickerProviderStateMixin {
static final GlobalKey<FormState> _formKey =
GlobalKey<FormState>(debugLabel: '_accountManagement');
FocusScopeNode _focusNode;
TabController _controller;
@override
void initState() {
super.initState();
_focusNode = FocusScopeNode();
final settingsUIState = widget.viewModel.state.settingsUIState;
_controller = TabController(
vsync: this, length: 2, initialIndex: settingsUIState.tabIndex);
_controller.addListener(_onTabChanged);
}
void _onTabChanged() {
final store = StoreProvider.of<AppState>(context);
store.dispatch(UpdateSettingsTab(tabIndex: _controller.index));
}
@override
void dispose() {
_focusNode.dispose();
_controller.removeListener(_onTabChanged);
_controller.dispose();
super.dispose();
}
@override
Widget build(BuildContext context) {
final localization = AppLocalization.of(context);
final viewModel = widget.viewModel;
final state = viewModel.state;
final company = viewModel.company;
return EditScaffold(
title: localization.accountManagement,
onSavePressed: viewModel.onSavePressed,
appBarBottom: TabBar(
key: ValueKey(state.settingsUIState.updatedAt),
controller: _controller,
tabs: [
Tab(
text: localization.overview,
),
Tab(
text: localization.enabledModules,
),
],
),
body: AppTabForm(
formKey: _formKey,
focusNode: _focusNode,
tabController: _controller,
children: <Widget>[
_AccountOverview(viewModel: viewModel),
ListView(
children: <Widget>[
FormCard(
// TODO change to kModules.keys
children: kModules.keys.map((module) {
return CheckboxListTile(
controlAffinity: ListTileControlAffinity.leading,
title: Text(localization.lookup(kModules[module])),
value: company.enabledModules & module != 0,
activeColor: Theme.of(context).accentColor,
onChanged: (value) {
int enabledModules = company.enabledModules;
if (value) {
enabledModules = enabledModules | module;
} else {
enabledModules = enabledModules ^ module;
}
viewModel.onCompanyChanged(company
.rebuild((b) => b..enabledModules = enabledModules));
},
);
}).toList()),
],
),
],
),
);
}
}
class _AccountOverview extends StatelessWidget {
const _AccountOverview({
Key key,
@required this.viewModel,
}) : super(key: key);
final AccountManagementVM viewModel;
@override
Widget build(BuildContext context) {
final store = StoreProvider.of<AppState>(context);
final localization = AppLocalization.of(context);
final state = viewModel.state;
final account = state.account;
final company = viewModel.company;
final companies = state.companies;
return ListView(
shrinkWrap: true,
children: <Widget>[
AppHeader(
label: localization.plan,
value: account.plan.isEmpty
? localization.free
: localization.lookup(account.plan),
secondLabel: localization.expiresOn,
secondValue: formatDate(account.planExpires, context),
),
if (state.company.isDisabled)
FormCard(
children: [
SwitchListTile(
value: !company.isDisabled,
onChanged: (value) {
viewModel.onCompanyChanged(
company.rebuild((b) => b..isDisabled = !value));
},
// TODO change to localization
title: Text('Company Activated'),
subtitle: Text('Enable recurring invoices and notifications'),
activeColor: Theme.of(context).accentColor,
)
],
),
Padding(
padding: const EdgeInsets.all(16),
child: Row(
children: [
Expanded(
child: AppButton(
label: localization.purchaseLicense.toUpperCase(),
iconData: Icons.cloud_download,
onPressed: () async {
if (await canLaunch(kWhiteLabelUrl)) {
launch(kWhiteLabelUrl, forceSafariVC: false);
}
},
),
),
SizedBox(width: kGutterWidth),
Expanded(
child: AppButton(
label: localization.applyLicense.toUpperCase(),
iconData: Icons.cloud_done,
onPressed: () {
fieldCallback(
context: context,
title: localization.applyLicense,
field: localization.license,
maxLength: 24,
callback: (value) {
final state = viewModel.state;
final credentials = state.credentials;
final url =
'${credentials.url}/claim_license?license_key=$value';
showDialog<AlertDialog>(
context: context,
barrierDismissible: false,
builder: (BuildContext context) => SimpleDialog(
children: <Widget>[LoadingDialog()],
));
WebClient()
.post(
url,
credentials.token,
)
.then((dynamic response) {
if (Navigator.of(context).canPop()) {
Navigator.of(context).pop();
}
viewModel.onAppliedLicense();
}).catchError((dynamic error) {
if (Navigator.of(context).canPop()) {
Navigator.of(context).pop();
}
showErrorDialog(
context: context, message: '$error');
});
});
},
),
),
],
),
),
Padding(
padding: const EdgeInsets.symmetric(vertical: 16, horizontal: 16),
child: ListDivider(),
),
Padding(
padding: const EdgeInsets.all(16),
child: Row(children: [
Expanded(
child: AppButton(
label: localization.apiTokens.toUpperCase(),
iconData: getEntityIcon(EntityType.token),
onPressed: () {
store.dispatch(ViewSettings(
navigator: Navigator.of(context),
section: kSettingsTokens,
));
},
),
),
SizedBox(width: kGutterWidth),
Expanded(
child: AppButton(
label: localization.apiWebhooks.toUpperCase(),
iconData: getEntityIcon(EntityType.webhook),
onPressed: () {
store.dispatch(ViewSettings(
navigator: Navigator.of(context),
section: kSettingsWebhooks,
));
},
),
),
])),
Padding(
padding: const EdgeInsets.all(16),
child: Row(children: [
Expanded(
child: AppButton(
label: localization.apiDocs.toUpperCase(),
iconData: MdiIcons.bookshelf,
onPressed: () => launch(kApiDocsURL),
),
),
SizedBox(width: kGutterWidth),
Expanded(
child: AppButton(
label: 'Zapier',
iconData: MdiIcons.cloud,
onPressed: () => launch(kZapierURL),
),
),
])),
Padding(
padding: const EdgeInsets.symmetric(vertical: 16, horizontal: 16),
child: ListDivider(),
),
Padding(
padding: const EdgeInsets.all(16),
child: Row(
children: [
Expanded(
child: AppButton(
label: localization.purgeData.toUpperCase(),
color: Colors.red,
iconData: Icons.delete,
onPressed: () {
confirmCallback(
context: context,
message: localization.purgeDataMessage,
typeToConfirm: localization.purge.toLowerCase(),
callback: () {
passwordCallback(
alwaysRequire: true,
context: context,
callback: (password) {
viewModel.onPurgeData(context, password);
});
});
},
),
),
SizedBox(width: kGutterWidth),
Expanded(
child: AppButton(
label: companies.length == 1
? localization.cancelAccount.toUpperCase()
: localization.deleteCompany.toUpperCase(),
color: Colors.red,
iconData: Icons.delete,
onPressed: () {
String message = companies.length == 1
? localization.cancelAccountMessage
: localization.deleteCompanyMessage;
message = message.replaceFirst(
':company',
company.displayName.isEmpty
? localization.untitledCompany
: company.displayName);
confirmCallback(
context: context,
message: message,
typeToConfirm: localization.delete.toLowerCase(),
callback: () {
passwordCallback(
alwaysRequire: true,
context: context,
callback: (password) {
viewModel.onCompanyDelete(context, password);
});
});
},
),
),
],
),
),
],
);
}
}