2024-08-02 19:40:38 +00:00
|
|
|
import 'dart:io';
|
|
|
|
|
2024-05-01 00:41:02 +00:00
|
|
|
import 'package:barcode_scan2/barcode_scan2.dart';
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:flutter/services.dart';
|
2024-05-27 23:56:22 +00:00
|
|
|
|
2024-05-23 00:37:06 +00:00
|
|
|
import '../../themes/stack_colors.dart';
|
|
|
|
import '../../utilities/constants.dart';
|
|
|
|
import '../../utilities/logger.dart';
|
|
|
|
import '../../utilities/text_styles.dart';
|
|
|
|
import '../../utilities/util.dart';
|
|
|
|
import '../conditional_parent.dart';
|
2024-08-02 19:40:38 +00:00
|
|
|
import '../desktop/qr_code_scanner_dialog.dart';
|
2024-05-23 00:37:06 +00:00
|
|
|
import '../icon_widgets/clipboard_icon.dart';
|
|
|
|
import '../icon_widgets/qrcode_icon.dart';
|
|
|
|
import '../icon_widgets/x_icon.dart';
|
|
|
|
import '../textfield_icon_button.dart';
|
2024-05-01 00:41:02 +00:00
|
|
|
|
|
|
|
class FrostStepField extends StatefulWidget {
|
|
|
|
const FrostStepField({
|
|
|
|
super.key,
|
|
|
|
required this.controller,
|
|
|
|
required this.focusNode,
|
|
|
|
this.label,
|
|
|
|
this.hint,
|
2024-05-01 15:13:47 +00:00
|
|
|
required this.onChanged,
|
2024-05-01 00:41:02 +00:00
|
|
|
required this.showQrScanOption,
|
|
|
|
});
|
|
|
|
|
|
|
|
final TextEditingController controller;
|
|
|
|
final FocusNode focusNode;
|
|
|
|
final String? label;
|
|
|
|
final String? hint;
|
2024-05-01 15:13:47 +00:00
|
|
|
final void Function(String) onChanged;
|
2024-05-01 00:41:02 +00:00
|
|
|
final bool showQrScanOption;
|
|
|
|
|
|
|
|
@override
|
|
|
|
State<FrostStepField> createState() => _FrostStepFieldState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _FrostStepFieldState extends State<FrostStepField> {
|
|
|
|
final _xKey = UniqueKey();
|
|
|
|
final _pasteKey = UniqueKey();
|
|
|
|
late final Key? _qrKey;
|
|
|
|
|
|
|
|
bool _isEmpty = true;
|
|
|
|
|
|
|
|
final _inputBorder = OutlineInputBorder(
|
|
|
|
borderSide: const BorderSide(
|
|
|
|
width: 0,
|
|
|
|
color: Colors.transparent,
|
|
|
|
),
|
|
|
|
borderRadius: BorderRadius.circular(
|
|
|
|
Constants.size.circularBorderRadius,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
|
2024-05-01 15:13:47 +00:00
|
|
|
late final void Function(String) _changed;
|
|
|
|
|
2024-05-01 00:41:02 +00:00
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
_qrKey = widget.showQrScanOption ? UniqueKey() : null;
|
|
|
|
_isEmpty = widget.controller.text.isEmpty;
|
2024-05-01 15:13:47 +00:00
|
|
|
|
|
|
|
_changed = (value) {
|
|
|
|
if (context.mounted) {
|
|
|
|
widget.onChanged.call(value);
|
|
|
|
setState(() {
|
|
|
|
_isEmpty = widget.controller.text.isEmpty;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-05-01 00:41:02 +00:00
|
|
|
super.initState();
|
|
|
|
}
|
|
|
|
|
2024-08-02 19:40:38 +00:00
|
|
|
Future<void> scanQr() async {
|
|
|
|
try {
|
|
|
|
if (Platform.isAndroid || Platform.isIOS) {
|
|
|
|
if (FocusScope.of(context).hasFocus) {
|
|
|
|
FocusScope.of(context).unfocus();
|
|
|
|
await Future<void>.delayed(
|
|
|
|
const Duration(milliseconds: 75),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
final qrResult = await BarcodeScanner.scan();
|
|
|
|
|
|
|
|
widget.controller.text = qrResult.rawContent;
|
|
|
|
|
|
|
|
_changed(widget.controller.text);
|
|
|
|
} else {
|
|
|
|
// Platform.isLinux, Platform.isWindows, or Platform.isMacOS.
|
2024-12-11 23:40:22 +00:00
|
|
|
final qrResult = await showDialog<String>(
|
2024-08-02 19:40:38 +00:00
|
|
|
context: context,
|
2024-12-11 23:40:22 +00:00
|
|
|
builder: (context) => const QrCodeScannerDialog(),
|
2024-08-02 19:40:38 +00:00
|
|
|
);
|
2024-12-11 23:40:22 +00:00
|
|
|
|
|
|
|
if (qrResult == null) {
|
|
|
|
Logging.instance.log(
|
|
|
|
"Qr scanning cancelled",
|
|
|
|
level: LogLevel.Info,
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
// TODO [prio=low]: Validate QR code data.
|
|
|
|
widget.controller.text = qrResult;
|
|
|
|
|
|
|
|
_changed(widget.controller.text);
|
|
|
|
}
|
2024-08-02 19:40:38 +00:00
|
|
|
}
|
|
|
|
} on PlatformException catch (e, s) {
|
|
|
|
Logging.instance.log(
|
|
|
|
"Failed to get camera permissions while trying to scan qr code: $e\n$s",
|
|
|
|
level: LogLevel.Warning,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-05-01 00:41:02 +00:00
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return ConditionalParent(
|
|
|
|
condition: widget.label != null,
|
|
|
|
builder: (child) => Column(
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.stretch,
|
|
|
|
children: [
|
|
|
|
Text(
|
|
|
|
widget.label!,
|
|
|
|
style: STextStyles.w500_14(context),
|
|
|
|
),
|
|
|
|
const SizedBox(
|
|
|
|
height: 4,
|
|
|
|
),
|
|
|
|
child,
|
|
|
|
],
|
|
|
|
),
|
|
|
|
child: TextField(
|
|
|
|
controller: widget.controller,
|
|
|
|
focusNode: widget.focusNode,
|
|
|
|
readOnly: false,
|
|
|
|
autocorrect: false,
|
|
|
|
enableSuggestions: false,
|
|
|
|
style: STextStyles.field(context),
|
2024-05-01 15:13:47 +00:00
|
|
|
onChanged: _changed,
|
2024-05-01 00:41:02 +00:00
|
|
|
decoration: InputDecoration(
|
|
|
|
hintText: widget.hint,
|
|
|
|
fillColor: widget.focusNode.hasFocus
|
|
|
|
? Theme.of(context).extension<StackColors>()!.textFieldActiveBG
|
|
|
|
: Theme.of(context).extension<StackColors>()!.textFieldDefaultBG,
|
|
|
|
hintStyle: Util.isDesktop
|
|
|
|
? STextStyles.desktopTextFieldLabel(context)
|
|
|
|
: STextStyles.fieldLabel(context),
|
|
|
|
enabledBorder: _inputBorder,
|
|
|
|
focusedBorder: _inputBorder,
|
|
|
|
errorBorder: _inputBorder,
|
|
|
|
disabledBorder: _inputBorder,
|
|
|
|
focusedErrorBorder: _inputBorder,
|
|
|
|
suffixIcon: Padding(
|
|
|
|
padding: _isEmpty
|
|
|
|
? const EdgeInsets.only(right: 8)
|
|
|
|
: const EdgeInsets.only(right: 0),
|
|
|
|
child: UnconstrainedBox(
|
|
|
|
child: Row(
|
|
|
|
mainAxisAlignment: MainAxisAlignment.spaceAround,
|
|
|
|
children: [
|
|
|
|
!_isEmpty
|
|
|
|
? TextFieldIconButton(
|
|
|
|
semanticsLabel:
|
|
|
|
"Clear Button. Clears The Frost Step Field Input.",
|
|
|
|
key: _xKey,
|
|
|
|
onTap: () {
|
|
|
|
widget.controller.text = "";
|
|
|
|
|
2024-05-01 15:13:47 +00:00
|
|
|
_changed(widget.controller.text);
|
2024-05-01 00:41:02 +00:00
|
|
|
},
|
|
|
|
child: const XIcon(),
|
|
|
|
)
|
|
|
|
: TextFieldIconButton(
|
|
|
|
semanticsLabel:
|
|
|
|
"Paste Button. Pastes From Clipboard To Frost Step Field Input.",
|
|
|
|
key: _pasteKey,
|
|
|
|
onTap: () async {
|
|
|
|
final ClipboardData? data =
|
|
|
|
await Clipboard.getData(Clipboard.kTextPlain);
|
|
|
|
if (data?.text != null && data!.text!.isNotEmpty) {
|
|
|
|
widget.controller.text = data.text!.trim();
|
|
|
|
}
|
|
|
|
|
2024-05-01 15:13:47 +00:00
|
|
|
_changed(widget.controller.text);
|
2024-05-01 00:41:02 +00:00
|
|
|
},
|
|
|
|
child:
|
|
|
|
_isEmpty ? const ClipboardIcon() : const XIcon(),
|
|
|
|
),
|
|
|
|
if (_isEmpty && widget.showQrScanOption)
|
|
|
|
TextFieldIconButton(
|
|
|
|
semanticsLabel:
|
|
|
|
"Scan QR Button. Opens Camera For Scanning QR Code.",
|
|
|
|
key: _qrKey,
|
2024-08-02 19:40:38 +00:00
|
|
|
onTap: scanQr,
|
2024-05-01 00:41:02 +00:00
|
|
|
child: const QrCodeIcon(),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|