invoice/lib/ui/settings/user_details.dart

402 lines
14 KiB
Dart

import 'package:built_collection/built_collection.dart';
import 'package:flutter/cupertino.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/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/form_card.dart';
import 'package:invoiceninja_flutter/ui/app/forms/app_form.dart';
import 'package:invoiceninja_flutter/ui/app/forms/color_picker.dart';
import 'package:invoiceninja_flutter/ui/app/forms/decorated_form_field.dart';
import 'package:invoiceninja_flutter/ui/app/edit_scaffold.dart';
import 'package:invoiceninja_flutter/ui/app/forms/notification_settings.dart';
import 'package:invoiceninja_flutter/ui/app/forms/password_field.dart';
import 'package:invoiceninja_flutter/ui/app/loading_indicator.dart';
import 'package:invoiceninja_flutter/ui/app/scrollable_listview.dart';
import 'package:invoiceninja_flutter/ui/settings/user_details_vm.dart';
import 'package:invoiceninja_flutter/utils/completers.dart';
import 'package:invoiceninja_flutter/utils/dialogs.dart';
import 'package:invoiceninja_flutter/utils/localization.dart';
import 'package:url_launcher/url_launcher.dart';
class UserDetails extends StatefulWidget {
const UserDetails({
Key key,
@required this.viewModel,
}) : super(key: key);
final UserDetailsVM viewModel;
@override
_UserDetailsState createState() => _UserDetailsState();
}
class _UserDetailsState extends State<UserDetails>
with SingleTickerProviderStateMixin {
static final GlobalKey<FormState> _formKey =
GlobalKey<FormState>(debugLabel: '_userDetails');
final FocusScopeNode _focusNode = FocusScopeNode();
TabController _controller;
bool autoValidate = false;
final _firstNameController = TextEditingController();
final _lastNameController = TextEditingController();
final _phoneController = TextEditingController();
final _emailController = TextEditingController();
final _passwordController = TextEditingController();
List<TextEditingController> _controllers = [];
final _debouncer = Debouncer();
@override
void initState() {
super.initState();
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();
_controllers.forEach((dynamic controller) {
controller.removeListener(_onChanged);
controller.dispose();
});
super.dispose();
}
@override
void didChangeDependencies() {
_controllers = [
_firstNameController,
_lastNameController,
_emailController,
_phoneController,
_passwordController,
];
_controllers
.forEach((dynamic controller) => controller.removeListener(_onChanged));
final user = widget.viewModel.state.user;
_firstNameController.text = user.firstName;
_lastNameController.text = user.lastName;
_emailController.text = user.email;
_phoneController.text = user.phone;
_passwordController.text = user.password;
_controllers
.forEach((dynamic controller) => controller.addListener(_onChanged));
super.didChangeDependencies();
}
void _onChanged() {
_debouncer.run(() {
final user = widget.viewModel.user.rebuild((b) => b
..firstName = _firstNameController.text.trim()
..lastName = _lastNameController.text.trim()
..email = _emailController.text.trim()
..phone = _phoneController.text.trim()
..password = _passwordController.text.trim());
if (user != widget.viewModel.user) {
widget.viewModel.onChanged(user);
}
});
}
@override
Widget build(BuildContext context) {
final localization = AppLocalization.of(context);
final viewModel = widget.viewModel;
final user = viewModel.user;
final state = viewModel.state;
return EditScaffold(
title: localization.userDetails,
onSavePressed: (context) {
final bool isValid = _formKey.currentState.validate();
setState(() {
autoValidate = !isValid ?? false;
});
if (!isValid) {
return;
}
viewModel.onSavePressed(context);
},
appBarBottom: TabBar(
controller: _controller,
tabs: [
Tab(
text: localization.details,
),
Tab(
text: localization.notifications,
),
],
),
body: AppTabForm(
focusNode: _focusNode,
formKey: _formKey,
tabController: _controller,
children: <Widget>[
ScrollableListView(
children: <Widget>[
FormCard(children: <Widget>[
DecoratedFormField(
label: localization.firstName,
controller: _firstNameController,
validator: (val) => val.isEmpty || val.trim().isEmpty
? localization.pleaseEnterAFirstName
: null,
autovalidate: autoValidate,
),
DecoratedFormField(
label: localization.lastName,
controller: _lastNameController,
validator: (val) => val.isEmpty || val.trim().isEmpty
? localization.pleaseEnterALastName
: null,
autovalidate: autoValidate,
),
DecoratedFormField(
label: localization.email,
controller: _emailController,
validator: (val) => val.isEmpty || val.trim().isEmpty
? localization.pleaseEnterYourEmail
: null,
autovalidate: autoValidate,
),
DecoratedFormField(
label: localization.phone,
controller: _phoneController,
),
PasswordFormField(
controller: _passwordController,
autoValidate: autoValidate,
),
]),
if (state.isHosted)
Padding(
padding: const EdgeInsets.only(
left: 18, top: 20, right: 18, bottom: 10),
child: Row(
children: [
Expanded(
child: OutlineButton(
child: Text((user.isConnectedToGoogle
? localization.disconnectGoogle
: localization.connectGoogle)
.toUpperCase()),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(5)),
onPressed: state.settingsUIState.isChanged
? null
: () {
if (user.isConnectedToGoogle) {
viewModel
.onDisconnectGooglePressed(context);
} else {
viewModel.onConnectGooglePressed(context);
}
},
),
),
SizedBox(width: kTableColumnGap),
Expanded(
child: OutlineButton(
child: Text((user.isTwoFactorEnabled
? localization.disableTwoFactor
: localization.enableTwoFactor)
.toUpperCase()),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(5)),
onPressed: state.settingsUIState.isChanged
? null
: () {
if (user.isTwoFactorEnabled) {
viewModel
.onDisableTwoFactorPressed(context);
} else {
if (state.user.phone.isEmpty ||
user.phone.isEmpty) {
showMessageDialog(
context: context,
message: localization
.enterPhoneToEnableTwoFactor);
return;
}
showDialog<void>(
context: context,
builder: (BuildContext context) =>
_EnableTwoFactor(
state: viewModel.state),
);
}
},
),
),
],
),
),
FormCard(
children: <Widget>[
FormColorPicker(
labelText: localization.accentColor,
initialValue:
viewModel.state.userCompany.settings.accentColor,
onSelected: (value) {
widget.viewModel.onChanged(user.rebuild(
(b) => b..userCompany.settings.accentColor = value));
},
),
],
),
],
),
ScrollableListView(
children: <Widget>[
NotificationSettings(
user: user,
onChanged: (channel, options) {
viewModel.onChanged(user.rebuild((b) => b
..userCompany.notifications[channel] = BuiltList(options)));
},
),
],
)
],
),
);
}
}
class _EnableTwoFactor extends StatefulWidget {
const _EnableTwoFactor({@required this.state});
final AppState state;
@override
_EnableTwoFactorState createState() => _EnableTwoFactorState();
}
class _EnableTwoFactorState extends State<_EnableTwoFactor> {
String _secret = 'test';
//String _secret;
String _oneTimePassword;
String _smsCode;
@override
void initState() {
super.initState();
final webClient = WebClient();
final credentials = widget.state.credentials;
final url = '${credentials.url}/settings/enable_two_factor';
webClient.get(url, credentials.token).then((dynamic response) {
print('## response: $response');
});
}
@override
Widget build(BuildContext context) {
final localzation = AppLocalization.of(context);
return AlertDialog(
title: Text(localzation.enableTwoFactor),
content: _secret == null
? LoadingIndicator(height: 100)
: SizedBox(
width: 280,
child: Column(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.center,
children: [
Text(_secret),
Row(
children: [
Expanded(
child: DecoratedFormField(
label: localzation.oneTimePassword,
onChanged: (value) {
_oneTimePassword = value;
},
),
),
SizedBox(width: kTableColumnGap),
SizedBox(
width: 100,
child: TextButton(
onPressed: () {
launch(
'https://github.com/antonioribeiro/google2fa#google-authenticator-apps');
},
child: Text(localzation.learnMore),
),
),
],
),
Row(
children: [
Expanded(
child: DecoratedFormField(
label: localzation.smsCode,
onChanged: (value) {
_smsCode = value;
},
),
),
SizedBox(width: kTableColumnGap),
SizedBox(
width: 100,
child: TextButton(
onPressed: () {
//
},
child: Text(localzation.sendSms),
),
),
],
),
],
),
),
actions: [
TextButton(
onPressed: () {
Navigator.of(context).pop();
},
child: Text(
localzation.cancel.toUpperCase(),
),
),
TextButton(
onPressed: () {
Navigator.of(context).pop();
},
child: Text(
localzation.save.toUpperCase(),
),
),
],
);
}
}