Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -31,3 +31,8 @@ migrate_working_dir/
.flutter-plugins-dependencies
.packages
build/

# Generated files
**/generated_plugin_registrant.*
**/generated_plugins.*
**/GeneratedPluginRegistrant.*
77 changes: 77 additions & 0 deletions lib/src/components/_supa_password_field.dart
Original file line number Diff line number Diff line change
@@ -0,0 +1,77 @@
import 'package:flutter/material.dart';

/// Internal password field component with visibility toggle
class SupaPasswordField extends StatefulWidget {
/// Controller for the password field
final TextEditingController controller;

/// Validator function for the password field
final String? Function(String?)? validator;

/// Label text for the password field
final String labelText;

/// Prefix icon for the password field
final Widget? prefixIcon;

/// Autofill hints for the password field
final Iterable<String>? autofillHints;

/// Text input action for the password field
final TextInputAction? textInputAction;

/// Callback when field is submitted
final void Function(String)? onFieldSubmitted;

/// Whether the field should auto-validate
final AutovalidateMode? autovalidateMode;

const SupaPasswordField({
super.key,
required this.controller,
this.validator,
required this.labelText,
this.prefixIcon,
this.autofillHints,
this.textInputAction,
this.onFieldSubmitted,
this.autovalidateMode,
});

@override
State<SupaPasswordField> createState() => _SupaPasswordFieldState();
}

class _SupaPasswordFieldState extends State<SupaPasswordField> {
bool _obscureText = true;

@override
Widget build(BuildContext context) {
return TextFormField(
controller: widget.controller,
validator: widget.validator,
obscureText: _obscureText,
autofillHints: widget.autofillHints,
textInputAction: widget.textInputAction,
onFieldSubmitted: widget.onFieldSubmitted,
autovalidateMode: widget.autovalidateMode,
decoration: InputDecoration(
prefixIcon: widget.prefixIcon,
label: Text(widget.labelText),
suffixIcon: Tooltip(
message: _obscureText ? 'Show password' : 'Hide password',
child: IconButton(
icon: Icon(
_obscureText ? Icons.visibility_off : Icons.visibility,
),
onPressed: () {
setState(() {
_obscureText = !_obscureText;
});
},
),
),
),
);
}
}
21 changes: 8 additions & 13 deletions lib/src/components/supa_email_auth.dart
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import 'package:email_validator/email_validator.dart';
import 'package:flutter/material.dart';
import 'package:supabase_auth_ui/src/components/_supa_password_field.dart';
import 'package:supabase_auth_ui/src/localizations/supa_email_auth_localization.dart';
import 'package:supabase_auth_ui/src/utils/constants.dart';
import 'package:supabase_flutter/supabase_flutter.dart';
Expand Down Expand Up @@ -328,7 +329,10 @@ class _SupaEmailAuthState extends State<SupaEmailAuth> {
),
if (!_isRecoveringPassword) ...[
spacer(16),
TextFormField(
SupaPasswordField(
controller: _passwordController,
labelText: localization.enterPassword,
prefixIcon: widget.prefixIconPassword,
autofillHints: _isSigningIn
? [AutofillHints.password]
: [AutofillHints.newPassword],
Expand All @@ -343,12 +347,6 @@ class _SupaEmailAuthState extends State<SupaEmailAuth> {
}
return null;
},
decoration: InputDecoration(
prefixIcon: widget.prefixIconPassword,
label: Text(localization.enterPassword),
),
obscureText: true,
controller: _passwordController,
onFieldSubmitted: (_) {
if (widget.metadataFields == null || _isSigningIn) {
_signInSignUp();
Expand All @@ -357,13 +355,10 @@ class _SupaEmailAuthState extends State<SupaEmailAuth> {
),
if (widget.showConfirmPasswordField && !_isSigningIn) ...[
spacer(16),
TextFormField(
SupaPasswordField(
controller: _confirmPasswordController,
decoration: InputDecoration(
prefixIcon: widget.prefixIconPassword,
label: Text(localization.confirmPassword),
),
obscureText: true,
labelText: localization.confirmPassword,
prefixIcon: widget.prefixIconPassword,
validator: (value) {
if (value != _passwordController.text) {
return localization.confirmPasswordError;
Expand Down
14 changes: 6 additions & 8 deletions lib/src/components/supa_phone_auth.dart
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import 'package:flutter/material.dart';
import 'package:supabase_auth_ui/src/components/_supa_password_field.dart';
import 'package:supabase_auth_ui/src/utils/constants.dart';
import 'package:supabase_auth_ui/supabase_auth_ui.dart';

Expand All @@ -10,7 +11,7 @@ class SupaPhoneAuth extends StatefulWidget {
/// Method to be called when the auth action is success
final void Function(AuthResponse response) onSuccess;

/// Method to be called when the auth action threw an excepction
/// Method to be called when the auth action threw an exception
final void Function(Object error)? onError;

/// Localization for the form
Expand Down Expand Up @@ -71,7 +72,10 @@ class _SupaPhoneAuthState extends State<SupaPhoneAuth> {
controller: _phone,
),
spacer(16),
TextFormField(
SupaPasswordField(
controller: _password,
labelText: localization.enterPassword,
prefixIcon: const Icon(Icons.lock),
autofillHints: isSigningIn
? [AutofillHints.password]
: [AutofillHints.newPassword],
Expand All @@ -82,12 +86,6 @@ class _SupaPhoneAuthState extends State<SupaPhoneAuth> {
}
return null;
},
decoration: InputDecoration(
prefixIcon: const Icon(Icons.lock),
label: Text(localization.enterPassword),
),
obscureText: true,
controller: _password,
),
spacer(16),
ElevatedButton(
Expand Down
11 changes: 5 additions & 6 deletions lib/src/components/supa_reset_password.dart
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import 'package:flutter/material.dart';
import 'package:supabase_auth_ui/src/components/_supa_password_field.dart';
import 'package:supabase_auth_ui/src/localizations/supa_reset_password_localization.dart';
import 'package:supabase_auth_ui/src/utils/constants.dart';
import 'package:supabase_flutter/supabase_flutter.dart';
Expand Down Expand Up @@ -47,19 +48,17 @@ class _SupaResetPasswordState extends State<SupaResetPassword> {
child: Column(
crossAxisAlignment: CrossAxisAlignment.stretch,
children: [
TextFormField(
SupaPasswordField(
controller: _password,
labelText: localization.enterPassword,
prefixIcon: const Icon(Icons.lock),
autofillHints: const [AutofillHints.newPassword],
validator: (value) {
if (value == null || value.isEmpty || value.length < 6) {
return localization.passwordLengthError;
}
return null;
},
decoration: InputDecoration(
prefixIcon: const Icon(Icons.lock),
label: Text(localization.enterPassword),
),
controller: _password,
),
spacer(16),
ElevatedButton(
Expand Down