formdator 0.4.1 formdator: ^0.4.1 copied to clipboard
A formidable collection of form field validators that can be selected and grouped into various combinations through composition — Decorator Pattern.
import 'package:flutter/material.dart';
import 'package:formdator/formdator.dart';
void main() {
runApp(const _DemoApp());
}
/// Demo purposes app widget.
///
/// The widget displays a simple registration form in which a user enters an
/// email, a password and password confirmation.
class _DemoApp extends StatelessWidget {
/// Ctor.
const _DemoApp({Key? key}) : super(key: key);
@override
Widget build(BuildContext context) {
return MaterialApp(
debugShowCheckedModeBanner: false,
title: 'Simple form demo',
home: SafeArea(
child: _RegForm(
title: 'Enter an email, a password and password confirmation',
),
),
);
}
}
/// A simple registration form widget.
class _RegForm extends StatelessWidget {
/// Ctor.
_RegForm({required String title, Key? key})
: _title = title,
super(key: key);
final String _title;
// the input data.
final Map<String, dynamic> _data = {};
// the form's mandatory key.
final _fkey = GlobalKey<FormState>();
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
elevation: 0.0,
title: Text(_title),
),
body: Center(
child: SingleChildScrollView(
padding: const EdgeInsets.all(16),
child: SizedBox(
width: 500,
child: Form(
key: _fkey,
child: Column(
children: [
_EmailField(onSaved: _saveEmail),
_SecretField(
label: 'Password',
onChanged: _refreshPass,
),
_SecretField(
label: 'Confirm password',
onChanged: _refreshConfirm,
onSaved: _savePass,
extra: Equal(_enteredPass, diff: 'does not match.'),
),
_ClearSubmitBar(fkey: _fkey, data: _data),
],
),
),
),
),
),
);
}
/// The current entered password value.
String _enteredPass() => (_data['pass'] as String?) ?? '';
/// Refreshes the password value.
void _refreshPass(String value) => _data['pass'] = value;
/// Refreshes the password confirmation value.
void _refreshConfirm(String value) => _data['confirm'] = value;
void _saveEmail(String? email) {
if (email != null && email.isNotEmpty) {
_data['email'] = email;
}
}
void _savePass(String? pass) {
if (pass != null && pass.isNotEmpty) {
_data['pass'] = pass;
}
}
}
/// Callback form onChange event.
typedef OnChanged = void Function(String);
/// Callback for onSave event.
typedef OnSaved = void Function(String?);
/// Convenient form field for sensitive data.
class _SecretField extends StatelessWidget {
/// A form field suitable for entering sensitive data.
///
/// [label] descriptive text for the field.
/// [onChange] callback for data chage event.
/// [onSaved] callback for data saved event.
/// [extra] an optional extra validation step.
_SecretField({
required String label,
OnChanged? onChanged,
OnSaved? onSaved,
StrVal? extra,
Key? key,
}) : _label = label,
_onChanged = onChanged,
_onSaved = onSaved,
_extra = extra ?? const Ok().call, // dummy validator as default
super(key: key);
final String _label;
final OnChanged? _onChanged;
final OnSaved? _onSaved;
final StrVal _extra;
@override
Widget build(BuildContext context) {
return TextFormField(
onSaved: _onSaved,
onChanged: _onChanged,
validator: Rules<String>([
const Req(blank: 'cannot be blank.'),
Len.range(
min: 4,
short: 'too short; min length is 4.',
max: 8,
long: 'too long; max length is 8.',
),
_extra,
]),
keyboardType: TextInputType.visiblePassword,
decoration: InputDecoration(
labelText: _label,
filled: true,
icon: const Icon(Icons.password),
hintText: 'between 4 and 8 characters',
),
);
}
}
/// Email field widget — it trims the entered email value and validate it so
/// that it is not blank and is well-formed.
class _EmailField extends StatelessWidget {
/// Non-blank well-formed email with an optional [extra] validation step.
///
/// [onSaved] callback for email saved event.
_EmailField({OnSaved? onSaved, StrVal? extra, Key? key})
: _onSaved = onSaved,
_extra = extra ?? const Ok().call, // defaults to dummy validator.
super(key: key);
final OnSaved? _onSaved;
final StrVal _extra;
@override
Widget build(BuildContext context) {
return TextFormField(
onSaved: _onSaved,
validator: Rules<String>([
Trim(ReqEmail()),
_extra,
]),
keyboardType: TextInputType.emailAddress,
decoration: const InputDecoration(
labelText: 'Email',
filled: true,
icon: Icon(Icons.email),
hintText: 'a_user@aprovider.com',
),
);
}
}
/// Row with clear and submit buttons.
class _ClearSubmitBar extends StatelessWidget {
/// Encapsulates the form state.
const _ClearSubmitBar({
required GlobalKey<FormState> fkey,
required Map<String, dynamic> data,
Key? key,
}) : _fkey = fkey,
_data = data,
super(key: key);
final GlobalKey<FormState> _fkey;
final Map<String, dynamic> _data;
@override
Widget build(BuildContext context) {
return ButtonBar(
children: <Widget>[
TextButton(
onPressed: () {
_data.clear();
_fkey.currentState?.reset();
},
child: const Text('Clear'),
),
ElevatedButton(
onPressed: () {
if (_fkey.currentState?.validate() ?? false) {
_fkey.currentState?.save();
_showDialog(context);
}
},
child: const Text('Submit'),
),
],
);
}
/// Displays a success dialog.
void _showDialog(BuildContext context) {
showDialog<String>(
context: context,
builder: (BuildContext context) => AlertDialog(
title: const Text('Success'),
content: const Text('Your information has been submitted.'),
actions: <Widget>[
TextButton(
onPressed: () => Navigator.pop(context, 'Ok'),
child: const Text('Thanks'),
),
],
),
);
}
}