<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>Forem: James Eneh</title>
    <description>The latest articles on Forem by James Eneh (@erozonachi).</description>
    <link>https://forem.com/erozonachi</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F200818%2F4884eef7-37f0-4b85-9ecf-9638c8e11b4c.jpeg</url>
      <title>Forem: James Eneh</title>
      <link>https://forem.com/erozonachi</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://forem.com/feed/erozonachi"/>
    <language>en</language>
    <item>
      <title>Create and Handle Flutter Input Form like a Pro</title>
      <dc:creator>James Eneh</dc:creator>
      <pubDate>Fri, 26 Feb 2021 01:50:22 +0000</pubDate>
      <link>https://forem.com/erozonachi/create-and-handle-flutter-input-form-like-a-pro-2k9c</link>
      <guid>https://forem.com/erozonachi/create-and-handle-flutter-input-form-like-a-pro-2k9c</guid>
      <description>&lt;p&gt;Input forms are used in software applications to collect data from users, they're an indispensable component that makes up the user interfaces of any application. We see them play crucial roles in scenarios like; user signup &amp;amp; login, sending a message on an instant messaging app, checking out orders on an e-commerce app, and so on.  That being said, it's important to make input forms readily accessible and easy to navigate by users.&lt;/p&gt;

&lt;p&gt;The aim of this article is to demonstrate how to create and, handle a  Flutter Form with TextFormField widgets for a better user experience. What will be covered here include;&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Explain what Flutter Form and TextFormField classes are.&lt;/li&gt;
&lt;li&gt;Discuss and show how some properties of a TextFormField widget can be used to achieve ease of navigation from one TextFormField to another.&lt;/li&gt;
&lt;li&gt;Discuss and show how some Form and TextFormField widgets’ properties can be used for a basic form field validation.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;em&gt;Note: This article is intended for readers with a basic knowledge of Flutter (i.e., readers who can create and run a Flutter application on an Android or iOS device, and also understand what widgets (both stateful and stateless) are, as used in Flutter).&lt;/em&gt;&lt;/p&gt;

&lt;p&gt;A Flutter Form class is a widget that can be used to wrap/group form fields for easy save, reset, or validation of the grouped form fields. A TextFormField class, on the other hand, is a widget that wraps a TextField widget in a FormField widget.&lt;/p&gt;

&lt;p&gt;Let's dive into how Form and TextFormField widgets, and some of their properties can be used to accomplish ease of navigation from one input field to another, and form validation. A user signup form with four input fields for capturing the user's name, phone, email, and password will be used as a case study.&lt;br&gt;
&lt;em&gt;At this point, you must have created a Flutter project with any valid name of your choice.&lt;/em&gt;&lt;/p&gt;

&lt;h5&gt;
  
  
  Step 1: Let's Create an Instance of a Form Widget with a Global Key that holds the Form's State.
&lt;/h5&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight dart"&gt;&lt;code&gt;

&lt;span class="kd"&gt;final&lt;/span&gt; &lt;span class="n"&gt;_formKey&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="n"&gt;GlobalKey&lt;/span&gt;&lt;span class="p"&gt;&amp;lt;&lt;/span&gt;&lt;span class="n"&gt;FormState&lt;/span&gt;&lt;span class="p"&gt;&amp;gt;();&lt;/span&gt;

&lt;span class="n"&gt;Form&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;
  &lt;span class="nl"&gt;key:&lt;/span&gt; &lt;span class="n"&gt;_formKey&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="nl"&gt;child:&lt;/span&gt; &lt;span class="kc"&gt;null&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
&lt;span class="p"&gt;)&lt;/span&gt;


&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;The private variable identifiable with the name &lt;code&gt;_formKey&lt;/code&gt; was set to a global key holding the form state, and it's passed to the Form widget as a key. This key will allow the form to validate all its descendant input fields.&lt;/p&gt;

&lt;h5&gt;
  
  
  Step 2: Add Four TextFormField Widgets with some Basic Properties to the above Form.
&lt;/h5&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight diff"&gt;&lt;code&gt;&lt;span class="err"&gt;

&lt;/span&gt;&lt;span class="p"&gt;final _formKey = GlobalKey&amp;lt;FormState&amp;gt;();
&lt;/span&gt;&lt;span class="err"&gt;
&lt;/span&gt;&lt;span class="p"&gt;Form(
&lt;/span&gt;  key: _formKey,
&lt;span class="gd"&gt;-  child: null,
&lt;/span&gt;&lt;span class="gi"&gt;+  child: Column(
+    children: &amp;lt;Widget&amp;gt;[
+      TextFormField(
+        keyboardType: TextInputType.name,
+        textInputAction: TextInputAction.next,
+        decoration: InputDecoration(
+          hintText: 'Enter your full name',
+          labelText: 'Full Name',
+        ),
+      ),
+      TextFormField(
+        keyboardType: TextInputType.phone,
+        textInputAction: TextInputAction.next,
+        decoration: InputDecoration(
+          hintText: 'Enter your phone number',
+          labelText: 'Phone Number',
+        ),
+      ),
+      TextFormField(
+        keyboardType: TextInputType.emailAddress,
+        textInputAction: TextInputAction.next,
+        decoration: InputDecoration(
+          hintText: 'Enter your email address',
+          labelText: 'Email Address',
+        ),
+      ),
+      TextFormField(
+        keyboardType: TextInputType.text,
+        textInputAction: TextInputAction.done,
+        obscureText: true,
+        decoration: InputDecoration(
+          hintText: 'Enter your password',
+          labelText: 'Password',
+        ),
+      ),
+    ],
+  ),
&lt;/span&gt;)
&lt;span class="err"&gt;

&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;The four TextFormFields widgets are for capturing the user's name, phone number, email address, and password. Let's discuss the properties passed to the TextFormField widgets;&lt;/p&gt;

&lt;p&gt;&lt;code&gt;keyboardType&lt;/code&gt;: the type of soft keyboard to pop up when the text field gains focus, passing an appropriate text input type value to this property will add to a better user experience. For example, passing &lt;code&gt;TextInputType.emailAddress&lt;/code&gt; to the &lt;code&gt;keyboardType&lt;/code&gt; property of the email address TextFormField widget will display the appropriate keyboard for typing an email address with &lt;code&gt;@&lt;/code&gt; and &lt;code&gt;.&lt;/code&gt; characters present when the text field gains focus.&lt;/p&gt;

&lt;p&gt;&lt;code&gt;textInputAction&lt;/code&gt;: its value determines the icon to appear as the keyboard’s action key. When set to &lt;code&gt;TextInputAction.next&lt;/code&gt;; the &lt;code&gt;next&lt;/code&gt; icon appears as the action key. We will see how intuitive and descriptive this is for navigating from one text field to another in the next step.&lt;/p&gt;

&lt;p&gt;&lt;code&gt;obscureText&lt;/code&gt;: it takes a boolean value (with &lt;code&gt;false&lt;/code&gt; as default) that determines whether or not the data supplied by the user in the text field is to be obscured. The data is obscured when it's set to &lt;code&gt;true&lt;/code&gt;, which applies to the password TextFormField widget.&lt;/p&gt;

&lt;p&gt;&lt;code&gt;decoration&lt;/code&gt;: It takes an instance of the InputDecoration class whose properties describe how the text field is decorated, as its value. Here, only two of its properties are in use;&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;
&lt;code&gt;hintText&lt;/code&gt;: a placeholder text, shown when the text field is empty, and&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;labelText&lt;/code&gt;: a descriptive label text, shown when the text field has focus or not empty.&lt;/li&gt;
&lt;/ol&gt;

&lt;h5&gt;
  
  
  Step 3: Make the Form Navigable and Submittable via the Keyboard’s Action Key.
&lt;/h5&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight diff"&gt;&lt;code&gt;&lt;span class="err"&gt;

&lt;/span&gt;&lt;span class="p"&gt;final _formKey = GlobalKey&amp;lt;FormState&amp;gt;();
&lt;/span&gt;&lt;span class="gi"&gt;+
+ final FocusNode _nameFocusNode = FocusNode();
+ final FocusNode _phoneFocusNode = FocusNode();
+ final FocusNode _emailFocusNode = FocusNode();
+ final FocusNode _passwordFocusNode = FocusNode();
+
+ _nextFocus(FocusNode focusNode) {
+   FocusScope.of(context).requestFocus(focusNode);
+ }
+
+ _submitForm() {
+   Scaffold.of(context).showSnackBar(SnackBar(content: 
+   Text('Registration sent')));
+ }
&lt;/span&gt;&lt;span class="err"&gt;
&lt;/span&gt;&lt;span class="p"&gt;Form(
&lt;/span&gt;  key: _formKey,
  child: Column(
    children: &amp;lt;Widget&amp;gt;[
      TextFormField(
         keyboardType: TextInputType.name,
         textInputAction: TextInputAction.next,
&lt;span class="gi"&gt;+        focusNode: _nameFocusNode,
+        onFieldSubmitted: (String value) {
+          //Do anything with value
+          _nextFocus(_phoneFocusNode);
+        },
&lt;/span&gt;         decoration: InputDecoration(
           hintText: 'Enter your full name',
           labelText: 'Full Name',
         ),
      ),
      TextFormField(
         keyboardType: TextInputType.phone,
         textInputAction: TextInputAction.next,
&lt;span class="gi"&gt;+        focusNode: _phoneFocusNode,
+        onFieldSubmitted: (String value) {
+          //Do anything with value
+          _nextFocus(_emailFocusNode);
+        },
&lt;/span&gt;         decoration: InputDecoration(
           hintText: 'Enter your phone number',
           labelText: 'Phone Number',
         ),
      ),
      TextFormField(
         keyboardType: TextInputType.emailAddress,
         textInputAction: TextInputAction.next,
&lt;span class="gi"&gt;+        focusNode: _emailFocusNode,
+        onFieldSubmitted: (String value) {
+          //Do anything with value
+          _nextFocus(_passwordFocusNode);
+        },
&lt;/span&gt;         decoration: InputDecoration(
           hintText: 'Enter your email address',
           labelText: 'Email Address',
         ),
      ),
      TextFormField(
         keyboardType: TextInputType.text,
         textInputAction: TextInputAction.done,
         obscureText: true,
&lt;span class="gi"&gt;+        focusNode: _passwordFocusNode,
+        onFieldSubmitted: (String value) {
+          //Do anything with value
+          _submitForm();
+        },
&lt;/span&gt;         decoration: InputDecoration(
           hintText: 'Enter your password',
           labelText: 'Password',
         ),
      ),
    ],
  ),
)
&lt;span class="err"&gt;

&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;The &lt;code&gt;focusNode&lt;/code&gt; property of every TextFormField widget in the &lt;code&gt;Form&lt;/code&gt; is assigned a FocusNode object to make the form easily navigable from one input field to another. Here, a FocusNode object allows keyboard focus to be obtained and pass around amongst input fields. We have four FocusNode objects (&lt;code&gt;_nameFocusNode&lt;/code&gt;, &lt;code&gt;_phoneFocusNode&lt;/code&gt;, &lt;code&gt;_emailFocusNode&lt;/code&gt;, and &lt;code&gt;_passwordFocusNode&lt;/code&gt;) declared, initialized and passed to the four TextFormField widgets in our form respectively.&lt;/p&gt;

&lt;p&gt;The above code update added the &lt;code&gt;onFieldSubmitted&lt;/code&gt; function property to the TextFormField widgets. This function takes a string argument which is the current value of its associated input field, and it's called when a user clicks the keyboard action key.&lt;/p&gt;

&lt;p&gt;For &lt;code&gt;name&lt;/code&gt;, &lt;code&gt;phone&lt;/code&gt;, and &lt;code&gt;email&lt;/code&gt; input fields, the &lt;code&gt;onFieldSubmitted&lt;/code&gt; function is used to pass keyboard focus to the next input field when a user clicks the action key. Thus, their &lt;code&gt;textInputAction&lt;/code&gt; properties are set to &lt;code&gt;TextInputAction.next&lt;/code&gt; to make the action key's behaviour intuitive. The &lt;code&gt;onFieldSubmitted&lt;/code&gt; function calls the &lt;code&gt;_nextFocus()&lt;/code&gt; method with the next &lt;code&gt;FocusNode&lt;/code&gt; object as a parameter to pass the keyboard focus to the next input field. The &lt;code&gt;_nextFocus()&lt;/code&gt; does the passing of keyboard focus to the received FocusNode object.&lt;/p&gt;

&lt;p&gt;The &lt;code&gt;onFieldSubmitted&lt;/code&gt; property of the password &lt;code&gt;TextFormField&lt;/code&gt; widget submits the form, and as such, the widget's &lt;code&gt;textInputAction&lt;/code&gt; property has the value &lt;code&gt;TextInputAction.done&lt;/code&gt; to depict the action key's behaviour. The &lt;code&gt;onFieldSubmitted&lt;/code&gt; function calls the &lt;code&gt;_submitForm()&lt;/code&gt; method, which flashes the message, &lt;code&gt;Registration sent&lt;/code&gt; on screen. The &lt;code&gt;_submitForm()&lt;/code&gt; method will do much more than flashing a message in the next step.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://media.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fy4gec30snpmtdktz3ti4.png" class="article-body-image-wrapper"&gt;&lt;img src="https://media.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fy4gec30snpmtdktz3ti4.png" alt="Next and Done Action Keys"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;h6&gt;
  
  
  &lt;code&gt;Next&lt;/code&gt; and &lt;code&gt;Done&lt;/code&gt; Action Keys on Android
&lt;/h6&gt;

&lt;h5&gt;
  
  
  Step 4: Non-Empty Validation of Input Fields and Form Submission
&lt;/h5&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight diff"&gt;&lt;code&gt;&lt;span class="err"&gt;

&lt;/span&gt;&lt;span class="p"&gt;final _formKey = GlobalKey&amp;lt;FormState&amp;gt;();
&lt;/span&gt;&lt;span class="err"&gt;
&lt;/span&gt;&lt;span class="p"&gt;final FocusNode _nameFocusNode = FocusNode();
final FocusNode _phoneFocusNode = FocusNode();
final FocusNode _emailFocusNode = FocusNode();
final FocusNode _passwordFocusNode = FocusNode();
&lt;/span&gt;&lt;span class="gi"&gt;+
+ final TextEditingController _nameController = TextEditingController();
+ final TextEditingController _phoneController = TextEditingController();
+ final TextEditingController _emailController = TextEditingController();
+ final TextEditingController _passwordController = TextEditingController();
&lt;/span&gt;&lt;span class="err"&gt;
&lt;/span&gt;&lt;span class="p"&gt;_nextFocus(FocusNode focusNode) {
&lt;/span&gt;  FocusScope.of(context).requestFocus(focusNode);
}
&lt;span class="err"&gt;
&lt;/span&gt;&lt;span class="p"&gt;_submitForm() {
&lt;/span&gt;&lt;span class="gi"&gt;+  if (_formKey.currentState.validate()) {
+    final user = {
+      'name': _nameController.text,
+      'phone': _phoneController.text,
+      'email': _emailController.text,
+      'password': _passwordController.text,
+    };
+    print(user.toString());
&lt;/span&gt;&lt;span class="err"&gt;
&lt;/span&gt;     // If the form passes validation, display a Snackbar.
     Scaffold.of(context).showSnackBar(SnackBar(
       content:Text('Registration sent')));
&lt;span class="gi"&gt;+
+    _formKey.currentState.save();
+    _formKey.currentState.reset();
+    _nextFocus(_nameFocusNode);
+  }
&lt;/span&gt;}
&lt;span class="gi"&gt;+
+ String _validateInput(String value) {
+   if(value.trim().isEmpty) {
+     return 'Field required';
+   }
+   return null;
+ }
&lt;/span&gt;&lt;span class="err"&gt;
&lt;/span&gt;&lt;span class="p"&gt;Form(
&lt;/span&gt;  key: _formKey,
  child: Column(
    children: &amp;lt;Widget&amp;gt;[
      TextFormField(
        keyboardType: TextInputType.name,
        textInputAction: TextInputAction.next,
        focusNode: _nameFocusNode,
        onFieldSubmitted: (String value) {
          //Do anything with value
          _nextFocus(_phoneFocusNode);
        },
&lt;span class="gi"&gt;+       controller: _nameController,
+       validator: _validateInput,
&lt;/span&gt;        decoration: InputDecoration(
          hintText: 'Enter your full name',
          labelText: 'Full Name',
        ),
      ),
      TextFormField(
        keyboardType: TextInputType.phone,
        textInputAction: TextInputAction.next,
        focusNode: _phoneFocusNode,
        onFieldSubmitted: (String value) {
          //Do anything with value
          _nextFocus(_emailFocusNode);
        },
&lt;span class="gi"&gt;+       controller: _phoneController,
+       validator: _validateInput,
&lt;/span&gt;        decoration: InputDecoration(
          hintText: 'Enter your phone number',
          labelText: 'Phone Number',
        ),
      ),
      TextFormField(
        keyboardType: TextInputType.emailAddress,
        textInputAction: TextInputAction.next,
        focusNode: _emailFocusNode,
        onFieldSubmitted: (String value) {
          //Do anything with value
          _nextFocus(_passwordFocusNode);
        },
&lt;span class="gi"&gt;+       controller: _emailController,
+       validator: _validateInput,
&lt;/span&gt;        decoration: InputDecoration(
          hintText: 'Enter your email address',
          labelText: 'Email Address',
        ),
      ),
      TextFormField(
        keyboardType: TextInputType.text,
        textInputAction: TextInputAction.done,
        obscureText: true,
        focusNode: _passwordFocusNode,
        onFieldSubmitted: (String value) {
          //Do anything with value
          _submitForm();
        },
&lt;span class="gi"&gt;+       controller: _passwordController,
+       validator: _validateInput,
&lt;/span&gt;        decoration: InputDecoration(
          hintText: 'Enter your password',
          labelText: 'Password',
        ),
      ),
&lt;span class="gi"&gt;+     ElevatedButton(
+       onPressed: _submitForm,
+       child: Text('Register'),
+     ),
&lt;/span&gt;    ],
  ),
)
&lt;span class="err"&gt;

&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

&lt;p&gt;The above code update added two new properties to the TextFormField widgets; &lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;the &lt;code&gt;controller&lt;/code&gt; property: a TextEditingController object, and &lt;/li&gt;
&lt;li&gt;the &lt;code&gt;validator&lt;/code&gt; property: a function that takes a string argument (value of the input field) and returns either a string value (validation error message) or &lt;code&gt;null&lt;/code&gt;.&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;The &lt;code&gt;TextEditingController&lt;/code&gt; object allows the value of an input field to be accessed even as a user types into the input field via its &lt;code&gt;text&lt;/code&gt; property. Four &lt;code&gt;TextEditingController&lt;/code&gt; objects have been instantiated and used as controllers in &lt;code&gt;TextFormField&lt;/code&gt; widgets for the &lt;code&gt;name&lt;/code&gt;, &lt;code&gt;phone&lt;/code&gt;, &lt;code&gt;email&lt;/code&gt;, and &lt;code&gt;password&lt;/code&gt; fields.&lt;/p&gt;

&lt;p&gt;A function named &lt;code&gt;_validateInput&lt;/code&gt; has been defined; it takes a string argument &lt;code&gt;value&lt;/code&gt; and either returns a string value '&lt;code&gt;Field required&lt;/code&gt;' if the argument is empty or returns &lt;code&gt;null&lt;/code&gt; if the argument is not empty. The &lt;code&gt;_validateInput()&lt;/code&gt; is used as validators in all the form's input fields.&lt;/p&gt;

&lt;p&gt;The &lt;code&gt;_submitForm()&lt;/code&gt; function has been modified to print values of &lt;code&gt;name&lt;/code&gt;, &lt;code&gt;phone&lt;/code&gt;, &lt;code&gt;email&lt;/code&gt;, and &lt;code&gt;password&lt;/code&gt; input fields to console. Flash the message '&lt;code&gt;Registration sent&lt;/code&gt;' on-screen. Save the form's current state, reset the form's current state and pass keyboard focus to the &lt;code&gt;name&lt;/code&gt; &lt;code&gt;TextFormField&lt;/code&gt; widget on successful form validation.&lt;/p&gt;

&lt;p&gt;The &lt;code&gt;_formKey.currentState.validate()&lt;/code&gt; calls the validators of the form's descendant &lt;code&gt;TextFormField&lt;/code&gt; widgets; if all the validators return &lt;code&gt;null&lt;/code&gt;, then it returns &lt;code&gt;true&lt;/code&gt;, else it returns &lt;code&gt;false&lt;/code&gt;.&lt;br&gt;
The &lt;code&gt;_formKey.currentState.save()&lt;/code&gt; saves all the form's descentant &lt;code&gt;TextFormField&lt;/code&gt; widgets, while the &lt;code&gt;_formKey.currentState.reset()&lt;/code&gt; resets all TextFormField widgets wrapped in the form to their initial values.&lt;/p&gt;

&lt;p&gt;An &lt;code&gt;ElevatedButton&lt;/code&gt; widget has also been added to the form to handle submission when pressed/clicked. It takes the function &lt;code&gt;_submitForm()&lt;/code&gt; as its &lt;code&gt;onPressed&lt;/code&gt; property and has a text that reads '&lt;code&gt;Register&lt;/code&gt;' as its child widget.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://media.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fytx07p0maasjnky89j4j.png" class="article-body-image-wrapper"&gt;&lt;img src="https://media.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fytx07p0maasjnky89j4j.png" alt="User Signup Form"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;h6&gt;
  
  
  The User Signup Form Screen
&lt;/h6&gt;

&lt;p&gt;You can check out the complete code &lt;a href="https://github.com/erozonachi/flutter_form_handling" rel="noopener noreferrer"&gt;here.&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;em&gt;Thanks for reading!&lt;/em&gt; 🙏🏼&lt;/p&gt;

</description>
      <category>flutter</category>
      <category>dart</category>
      <category>android</category>
      <category>ios</category>
    </item>
  </channel>
</rss>
