mirror of
https://github.com/flutter/samples.git
synced 2025-11-09 06:18:49 +00:00
* Update `simplistic_editor` for Flutter 3.4 beta * Re-enable beta and master CI * Disable on master * Added sample code for using plugins and channels from background isolates. * goderbauer feedback 1 * goderbauer feedback2 * goderbauer feedback 3 * Add `background_isolate_channels` to CI * Enable beta CI * Enable all `stable` CI projects * `dart fix --apply` * `print` -> `denugPrint` * Make deps min version not pinned * Drop `_isDebug` * Remove unused import * `dart format` * Fixup `linting_tool` * Fixup `form_app` * Enable all `master` CI * Basic fixes * Patch `simplistic_editor` * Fix nl at eol * Comment out `simplistic_editor` * Incorporating @bleroux's latest changes * Clean up CI scripts * Copy `experimental/material_3_demo` to top level * Update `game_template` * Update `animations` * Update `desktop_photo_search` * Update `flutter_maps_firestore` * Update `form_app` * Update `infinite_list` * Update `isolate_example` * Update `jsonexample` * Update `navigation_and_routing` * Update `place_tracker` * Update `platform_channels` * Update `platform_design` * Update `provider_shopper` * Fixup `context_menus` * `dart format` * Update the main `material_3_demo` * Make `tool/flutter_ci_script_stable.sh` executable again Co-authored-by: Bruno Leroux <bruno.leroux@gmail.com> Co-authored-by: Aaron Clarke <aaclarke@google.com>
170 lines
6.0 KiB
Dart
170 lines
6.0 KiB
Dart
// Copyright 2020, the Flutter project authors. Please see the AUTHORS file
|
|
// for details. All rights reserved. Use of this source code is governed by a
|
|
// BSD-style license that can be found in the LICENSE file.
|
|
|
|
import 'package:english_words/english_words.dart' as english_words;
|
|
import 'package:flutter/material.dart';
|
|
|
|
class FormValidationDemo extends StatefulWidget {
|
|
const FormValidationDemo({super.key});
|
|
|
|
@override
|
|
State<FormValidationDemo> createState() => _FormValidationDemoState();
|
|
}
|
|
|
|
class _FormValidationDemoState extends State<FormValidationDemo> {
|
|
final _formKey = GlobalKey<FormState>();
|
|
String? adjective;
|
|
String? noun;
|
|
bool? agreedToTerms = false;
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return Scaffold(
|
|
appBar: AppBar(
|
|
title: const Text('📖 Story Generator'),
|
|
actions: [
|
|
Padding(
|
|
padding: const EdgeInsets.all(8),
|
|
child: TextButton(
|
|
style: TextButton.styleFrom(foregroundColor: Colors.white),
|
|
child: const Text('Submit'),
|
|
onPressed: () {
|
|
// Validate the form by getting the FormState from the GlobalKey
|
|
// and calling validate() on it.
|
|
var valid = _formKey.currentState!.validate();
|
|
if (!valid) {
|
|
return;
|
|
}
|
|
|
|
showDialog<void>(
|
|
context: context,
|
|
builder: (context) => AlertDialog(
|
|
title: const Text('Your story'),
|
|
content: Text('The $adjective developer saw a $noun'),
|
|
actions: [
|
|
TextButton(
|
|
child: const Text('Done'),
|
|
onPressed: () {
|
|
Navigator.of(context).pop();
|
|
},
|
|
),
|
|
],
|
|
),
|
|
);
|
|
},
|
|
),
|
|
),
|
|
],
|
|
),
|
|
body: Form(
|
|
key: _formKey,
|
|
child: Scrollbar(
|
|
child: SingleChildScrollView(
|
|
padding: const EdgeInsets.all(16),
|
|
child: Column(
|
|
children: [
|
|
// A text field that validates that the text is an adjective.
|
|
TextFormField(
|
|
autofocus: true,
|
|
textInputAction: TextInputAction.next,
|
|
validator: (value) {
|
|
if (value!.isEmpty) {
|
|
return 'Please enter an adjective.';
|
|
}
|
|
if (english_words.adjectives.contains(value)) {
|
|
return null;
|
|
}
|
|
return 'Not a valid adjective.';
|
|
},
|
|
decoration: const InputDecoration(
|
|
filled: true,
|
|
hintText: 'e.g. quick, beautiful, interesting',
|
|
labelText: 'Enter an adjective',
|
|
),
|
|
onChanged: (value) {
|
|
adjective = value;
|
|
},
|
|
),
|
|
const SizedBox(
|
|
height: 24,
|
|
),
|
|
// A text field that validates that the text is a noun.
|
|
TextFormField(
|
|
validator: (value) {
|
|
if (value!.isEmpty) {
|
|
return 'Please enter a noun.';
|
|
}
|
|
if (english_words.nouns.contains(value)) {
|
|
return null;
|
|
}
|
|
return 'Not a valid noun.';
|
|
},
|
|
decoration: const InputDecoration(
|
|
filled: true,
|
|
hintText: 'i.e. a person, place or thing',
|
|
labelText: 'Enter a noun',
|
|
),
|
|
onChanged: (value) {
|
|
noun = value;
|
|
},
|
|
),
|
|
const SizedBox(
|
|
height: 24,
|
|
),
|
|
// A custom form field that requires the user to check a
|
|
// checkbox.
|
|
FormField<bool>(
|
|
initialValue: false,
|
|
validator: (value) {
|
|
if (value == false) {
|
|
return 'You must agree to the terms of service.';
|
|
}
|
|
return null;
|
|
},
|
|
builder: (formFieldState) {
|
|
return Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Row(
|
|
children: [
|
|
Checkbox(
|
|
value: agreedToTerms,
|
|
onChanged: (value) {
|
|
// When the value of the checkbox changes,
|
|
// update the FormFieldState so the form is
|
|
// re-validated.
|
|
formFieldState.didChange(value);
|
|
setState(() {
|
|
agreedToTerms = value;
|
|
});
|
|
},
|
|
),
|
|
Text(
|
|
'I agree to the terms of service.',
|
|
style: Theme.of(context).textTheme.titleMedium,
|
|
),
|
|
],
|
|
),
|
|
if (!formFieldState.isValid)
|
|
Text(
|
|
formFieldState.errorText ?? "",
|
|
style: Theme.of(context)
|
|
.textTheme
|
|
.bodySmall!
|
|
.copyWith(
|
|
color: Theme.of(context).colorScheme.error),
|
|
),
|
|
],
|
|
);
|
|
},
|
|
),
|
|
],
|
|
),
|
|
),
|
|
),
|
|
),
|
|
);
|
|
}
|
|
}
|