2024-02-14 16:58:45 +06:30
|
|
|
|
import 'package:fcs/domain/entities/user.dart';
|
|
|
|
|
|
import 'package:fcs/helpers/theme.dart';
|
|
|
|
|
|
import 'package:fcs/pages/widgets/local_app_bar.dart';
|
|
|
|
|
|
import 'package:fcs/pages/widgets/progress.dart';
|
|
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
|
|
import 'package:pin_input_text_field/pin_input_text_field.dart';
|
2024-02-21 17:05:20 +06:30
|
|
|
|
import 'package:provider/provider.dart';
|
2024-02-14 16:58:45 +06:30
|
|
|
|
|
|
|
|
|
|
import '../main/util.dart';
|
|
|
|
|
|
import '../widgets/local_text.dart';
|
2024-02-21 17:05:20 +06:30
|
|
|
|
import 'model/staff_model.dart';
|
2024-02-14 16:58:45 +06:30
|
|
|
|
|
|
|
|
|
|
class StaffPinEditor extends StatefulWidget {
|
|
|
|
|
|
final User staff;
|
|
|
|
|
|
const StaffPinEditor({required this.staff});
|
|
|
|
|
|
@override
|
|
|
|
|
|
_StaffPinEditorState createState() => _StaffPinEditorState();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class _StaffPinEditorState extends State<StaffPinEditor> {
|
2024-02-19 17:06:36 +06:30
|
|
|
|
final _formKey = GlobalKey<FormState>();
|
2024-02-14 16:58:45 +06:30
|
|
|
|
bool _isLoading = false;
|
|
|
|
|
|
late User _staff;
|
|
|
|
|
|
bool _enablePinLogin = false;
|
|
|
|
|
|
String _newPin = "";
|
|
|
|
|
|
String _confirmPin = "";
|
2024-02-21 17:05:20 +06:30
|
|
|
|
FocusNode _newPinFocus = FocusNode();
|
|
|
|
|
|
FocusNode _confirmPinFocus = FocusNode();
|
|
|
|
|
|
TextEditingController _newPinCtl = TextEditingController();
|
|
|
|
|
|
TextEditingController _confirmPinCtl = TextEditingController();
|
2024-02-14 16:58:45 +06:30
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
|
void initState() {
|
2024-02-21 17:05:20 +06:30
|
|
|
|
_init();
|
|
|
|
|
|
super.initState();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
_init() {
|
2024-02-14 16:58:45 +06:30
|
|
|
|
_staff = widget.staff;
|
|
|
|
|
|
_enablePinLogin = _staff.enablePinLogin;
|
|
|
|
|
|
_newPin = _staff.pinDigit ?? "";
|
|
|
|
|
|
_confirmPin = _staff.confirmPinDigit ?? "";
|
2024-02-21 17:05:20 +06:30
|
|
|
|
_newPinCtl.text = _newPin;
|
|
|
|
|
|
_confirmPinCtl.text = _confirmPin;
|
|
|
|
|
|
_checkFocusNode();
|
|
|
|
|
|
|
|
|
|
|
|
if (mounted) {
|
|
|
|
|
|
setState(() {});
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
|
void dispose() {
|
|
|
|
|
|
_newPinFocus.dispose();
|
|
|
|
|
|
_confirmPinFocus.dispose();
|
|
|
|
|
|
super.dispose();
|
2024-02-14 16:58:45 +06:30
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
|
Widget build(BuildContext context) {
|
2024-02-21 17:05:20 +06:30
|
|
|
|
final enablePinBox = InkWell(
|
|
|
|
|
|
onTap: () {
|
|
|
|
|
|
setState(() {
|
|
|
|
|
|
_enablePinLogin = !_enablePinLogin;
|
|
|
|
|
|
});
|
|
|
|
|
|
_checkFocusNode();
|
|
|
|
|
|
},
|
|
|
|
|
|
child: Row(
|
|
|
|
|
|
children: [
|
|
|
|
|
|
Checkbox(
|
|
|
|
|
|
materialTapTargetSize: MaterialTapTargetSize.shrinkWrap,
|
|
|
|
|
|
visualDensity: const VisualDensity(
|
2024-02-14 16:58:45 +06:30
|
|
|
|
horizontal: VisualDensity.minimumDensity,
|
2024-02-21 17:05:20 +06:30
|
|
|
|
),
|
|
|
|
|
|
activeColor: primaryColor,
|
|
|
|
|
|
side: BorderSide(color: Colors.black38, width: 2),
|
|
|
|
|
|
value: _enablePinLogin,
|
|
|
|
|
|
onChanged: (value) {
|
|
|
|
|
|
setState(() {
|
|
|
|
|
|
_enablePinLogin = value ?? false;
|
|
|
|
|
|
});
|
|
|
|
|
|
_checkFocusNode();
|
|
|
|
|
|
},
|
|
|
|
|
|
),
|
|
|
|
|
|
const SizedBox(width: 15),
|
|
|
|
|
|
LocalText(context, 'staff.enable_pin',
|
|
|
|
|
|
fontSize: 15, color: Colors.black)
|
|
|
|
|
|
],
|
|
|
|
|
|
),
|
2024-02-14 16:58:45 +06:30
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
final newPinBox = Column(
|
|
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
|
|
children: [
|
|
|
|
|
|
LocalText(context, 'staff.new_pin',
|
|
|
|
|
|
color: Colors.black54, fontSize: 15),
|
|
|
|
|
|
const SizedBox(height: 8),
|
|
|
|
|
|
PinInputTextField(
|
2024-02-21 17:05:20 +06:30
|
|
|
|
controller: _newPinCtl,
|
|
|
|
|
|
enabled: _enablePinLogin,
|
2024-02-14 16:58:45 +06:30
|
|
|
|
cursor: Cursor(
|
2024-02-14 17:19:55 +06:30
|
|
|
|
color: primaryColor, enabled: true, width: 2, height: 23),
|
2024-02-14 16:58:45 +06:30
|
|
|
|
pinLength: 6,
|
|
|
|
|
|
decoration: BoxLooseDecoration(
|
2024-02-21 17:05:20 +06:30
|
|
|
|
strokeColorBuilder: PinListenColorBuilder(
|
|
|
|
|
|
_enablePinLogin ? primaryColor : Colors.grey.shade400,
|
|
|
|
|
|
Colors.grey.shade400)),
|
2024-02-14 16:58:45 +06:30
|
|
|
|
textInputAction: TextInputAction.done,
|
2024-02-21 17:05:20 +06:30
|
|
|
|
focusNode: _newPinFocus,
|
2024-02-14 16:58:45 +06:30
|
|
|
|
onChanged: _newPinChange),
|
|
|
|
|
|
],
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
final confirmPinBox = Column(
|
|
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
|
|
children: [
|
|
|
|
|
|
LocalText(context, 'staff.confirm_pin',
|
|
|
|
|
|
color: Colors.black54, fontSize: 15),
|
|
|
|
|
|
const SizedBox(height: 8),
|
|
|
|
|
|
PinInputTextField(
|
2024-02-21 17:05:20 +06:30
|
|
|
|
controller: _confirmPinCtl,
|
|
|
|
|
|
enabled: _enablePinLogin,
|
2024-02-14 16:58:45 +06:30
|
|
|
|
pinLength: 6,
|
2024-02-14 17:19:55 +06:30
|
|
|
|
cursor: Cursor(
|
|
|
|
|
|
color: primaryColor, enabled: true, width: 2, height: 23),
|
2024-02-14 16:58:45 +06:30
|
|
|
|
decoration: BoxLooseDecoration(
|
2024-02-21 17:05:20 +06:30
|
|
|
|
strokeColorBuilder: PinListenColorBuilder(
|
|
|
|
|
|
_enablePinLogin ? primaryColor : Colors.grey.shade400,
|
|
|
|
|
|
Colors.grey.shade400)),
|
2024-02-14 16:58:45 +06:30
|
|
|
|
textInputAction: TextInputAction.done,
|
2024-02-21 17:05:20 +06:30
|
|
|
|
focusNode: _confirmPinFocus,
|
2024-02-14 16:58:45 +06:30
|
|
|
|
onChanged: _confirmPinChange),
|
|
|
|
|
|
],
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
final saveButton = Padding(
|
|
|
|
|
|
padding: const EdgeInsets.symmetric(horizontal: 30),
|
|
|
|
|
|
child: fcsButton(
|
|
|
|
|
|
context,
|
|
|
|
|
|
getLocalString(context, 'btn.save'),
|
|
|
|
|
|
callack: _save,
|
|
|
|
|
|
),
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
return LocalProgress(
|
|
|
|
|
|
inAsyncCall: _isLoading,
|
|
|
|
|
|
child: Scaffold(
|
|
|
|
|
|
appBar: LocalAppBar(
|
|
|
|
|
|
labelKey: "staff.pin.title",
|
|
|
|
|
|
backgroundColor: Colors.white,
|
|
|
|
|
|
labelColor: primaryColor,
|
|
|
|
|
|
arrowColor: primaryColor),
|
2024-02-19 17:06:36 +06:30
|
|
|
|
body: Form(
|
|
|
|
|
|
key: _formKey,
|
|
|
|
|
|
child: Padding(
|
|
|
|
|
|
padding: const EdgeInsets.only(left: 15.0, right: 15),
|
|
|
|
|
|
child: ListView(
|
|
|
|
|
|
children: <Widget>[
|
|
|
|
|
|
Column(
|
|
|
|
|
|
children: [
|
|
|
|
|
|
Text(_staff.name ?? "",
|
|
|
|
|
|
style: TextStyle(color: Colors.black, fontSize: 18)),
|
|
|
|
|
|
Text(_staff.fcsID ?? "",
|
|
|
|
|
|
style: TextStyle(color: Colors.black, fontSize: 15)),
|
|
|
|
|
|
],
|
|
|
|
|
|
),
|
2024-02-21 17:05:20 +06:30
|
|
|
|
const SizedBox(height: 10),
|
2024-02-19 17:06:36 +06:30
|
|
|
|
enablePinBox,
|
2024-02-21 17:05:20 +06:30
|
|
|
|
const SizedBox(height: 20),
|
2024-02-19 17:06:36 +06:30
|
|
|
|
FormField(
|
|
|
|
|
|
builder: (FormFieldState<bool> state) {
|
|
|
|
|
|
return Column(
|
|
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
|
|
children: [
|
|
|
|
|
|
newPinBox,
|
|
|
|
|
|
Padding(
|
|
|
|
|
|
padding: const EdgeInsets.only(top: 8.0),
|
|
|
|
|
|
child: _newPin == '' || _newPin.length < 6
|
|
|
|
|
|
? SizedBox(
|
|
|
|
|
|
height: 20,
|
|
|
|
|
|
child: Text(
|
|
|
|
|
|
state.errorText ?? '',
|
|
|
|
|
|
style: const TextStyle(
|
|
|
|
|
|
color: dangerColor, fontSize: 12),
|
|
|
|
|
|
),
|
|
|
|
|
|
)
|
|
|
|
|
|
: const SizedBox(height: 20),
|
|
|
|
|
|
),
|
|
|
|
|
|
],
|
|
|
|
|
|
);
|
|
|
|
|
|
},
|
|
|
|
|
|
validator: (value) {
|
|
|
|
|
|
if (_newPin == "") {
|
|
|
|
|
|
return 'Enter new PIN';
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (_newPin.length < 6) {
|
|
|
|
|
|
return 'New PIN must be 6 digits';
|
|
|
|
|
|
}
|
|
|
|
|
|
return null;
|
|
|
|
|
|
},
|
|
|
|
|
|
),
|
|
|
|
|
|
SizedBox(height: 10),
|
|
|
|
|
|
FormField(
|
|
|
|
|
|
builder: (FormFieldState<bool> state) {
|
|
|
|
|
|
return Column(
|
|
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
|
|
children: [
|
|
|
|
|
|
confirmPinBox,
|
|
|
|
|
|
Padding(
|
|
|
|
|
|
padding: const EdgeInsets.only(top: 8.0),
|
|
|
|
|
|
child: _confirmPin == '' ||
|
|
|
|
|
|
_confirmPin.length < 6 ||
|
|
|
|
|
|
_confirmPin != _newPin
|
|
|
|
|
|
? SizedBox(
|
|
|
|
|
|
height: 20,
|
|
|
|
|
|
child: Text(
|
|
|
|
|
|
state.errorText ?? '',
|
|
|
|
|
|
style: const TextStyle(
|
|
|
|
|
|
color: dangerColor, fontSize: 12),
|
|
|
|
|
|
),
|
|
|
|
|
|
)
|
|
|
|
|
|
: const SizedBox(height: 20)),
|
|
|
|
|
|
],
|
|
|
|
|
|
);
|
|
|
|
|
|
},
|
|
|
|
|
|
validator: (value) {
|
|
|
|
|
|
if (_confirmPin == "") {
|
|
|
|
|
|
return 'Enter confirm PIN';
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (_confirmPin.length < 6) {
|
|
|
|
|
|
return 'Confirm PIN must be 6 digits';
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (_confirmPin != _newPin) {
|
|
|
|
|
|
return "Those pins didn’t match. Try again.";
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return null;
|
|
|
|
|
|
},
|
|
|
|
|
|
),
|
|
|
|
|
|
SizedBox(height: 20),
|
|
|
|
|
|
saveButton,
|
|
|
|
|
|
SizedBox(height: 30)
|
|
|
|
|
|
],
|
|
|
|
|
|
),
|
2024-02-14 16:58:45 +06:30
|
|
|
|
),
|
|
|
|
|
|
),
|
|
|
|
|
|
));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-02-21 17:05:20 +06:30
|
|
|
|
_checkFocusNode() {
|
|
|
|
|
|
if (_enablePinLogin) {
|
|
|
|
|
|
_onChangeFocusNode();
|
|
|
|
|
|
} else {
|
|
|
|
|
|
_cancelFoucsNode();
|
|
|
|
|
|
}
|
|
|
|
|
|
if (mounted) {
|
|
|
|
|
|
setState(() {});
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
_onChangeFocusNode() {
|
|
|
|
|
|
if (this._newPin.length == 6) {
|
|
|
|
|
|
_newPinFocus.unfocus();
|
|
|
|
|
|
Future.delayed(const Duration(milliseconds: 300), () {
|
|
|
|
|
|
_confirmPinFocus.requestFocus();
|
|
|
|
|
|
});
|
|
|
|
|
|
} else {
|
|
|
|
|
|
_confirmPinFocus.unfocus();
|
|
|
|
|
|
Future.delayed(const Duration(milliseconds: 300), () {
|
|
|
|
|
|
_newPinFocus.requestFocus();
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
_cancelFoucsNode() {
|
|
|
|
|
|
_newPinFocus.unfocus();
|
|
|
|
|
|
Future.delayed(const Duration(milliseconds: 300), () {
|
|
|
|
|
|
_confirmPinFocus.unfocus();
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-02-14 16:58:45 +06:30
|
|
|
|
_newPinChange(pin) {
|
|
|
|
|
|
setState(() {
|
|
|
|
|
|
this._newPin = pin;
|
2024-02-21 17:05:20 +06:30
|
|
|
|
if (this._newPin.length == 6) {
|
|
|
|
|
|
_confirmPinFocus.requestFocus();
|
|
|
|
|
|
}
|
2024-02-14 16:58:45 +06:30
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
_confirmPinChange(pin) {
|
|
|
|
|
|
setState(() {
|
|
|
|
|
|
this._confirmPin = pin;
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
_save() async {
|
2024-02-19 17:06:36 +06:30
|
|
|
|
if (!_formKey.currentState!.validate()) return;
|
2024-02-14 16:58:45 +06:30
|
|
|
|
|
|
|
|
|
|
setState(() {
|
|
|
|
|
|
_isLoading = true;
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
try {
|
2024-02-21 17:05:20 +06:30
|
|
|
|
await context.read<StaffModel>().updatePin(
|
|
|
|
|
|
userID: _staff.id!,
|
|
|
|
|
|
enablePin: _enablePinLogin,
|
|
|
|
|
|
pin: int.parse(_confirmPin));
|
2024-02-14 16:58:45 +06:30
|
|
|
|
Navigator.pop(context, true);
|
|
|
|
|
|
} catch (e) {
|
|
|
|
|
|
showMsgDialog(context, "Error", e.toString());
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
setState(() {
|
|
|
|
|
|
_isLoading = false;
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|