vibrate_button 0.0.2
vibrate_button: ^0.0.2 copied to clipboard
A premium, interactive Flutter vibrate button package featuring validation-triggered shake animations and a smooth wave-based loading indicator.
example/lib/main.dart
import 'package:flutter/material.dart';
import 'package:get/get.dart';
import 'package:vibrate_button/vibrate_button.dart';
void main() {
runApp(const MyApp());
}
class MyApp extends StatelessWidget {
const MyApp({super.key});
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Vibrate Button Login',
theme: ThemeData(
useMaterial3: true,
colorSchemeSeed: Colors.deepPurple,
inputDecorationTheme: InputDecorationTheme(
border: OutlineInputBorder(
borderRadius: BorderRadius.circular(12),
),
filled: true,
fillColor: Colors.grey.shade50,
),
),
home: const LoginPage(),
);
}
}
class LoginPage extends StatefulWidget {
const LoginPage({super.key});
@override
State<LoginPage> createState() => _LoginPageState();
}
class _LoginPageState extends State<LoginPage> {
final GlobalKey<FormState> _formKey = GlobalKey<FormState>();
final TextEditingController _emailController = TextEditingController();
final TextEditingController _passwordController = TextEditingController();
bool _isLoading = false;
bool _obscurePassword = true;
@override
void dispose() {
_emailController.dispose();
_passwordController.dispose();
super.dispose();
}
void _handleLogin() async {
setState(() {
_isLoading = true;
});
// Simulate network delay
await Future.delayed(const Duration(seconds: 2));
if (mounted) {
setState(() {
_isLoading = false;
});
Get.snackbar(
'Success',
'Logged in successfully',
snackPosition: SnackPosition.BOTTOM,
backgroundColor: Colors.green.withOpacity(0.8),
colorText: Colors.white,
margin: const EdgeInsets.all(16),
);
}
}
@override
Widget build(BuildContext context) {
return Scaffold(
body: Container(
decoration: BoxDecoration(
gradient: LinearGradient(
begin: Alignment.topLeft,
end: Alignment.bottomRight,
colors: [
Colors.deepPurple.shade50,
Colors.white,
],
),
),
child: SafeArea(
child: Center(
child: SingleChildScrollView(
padding: const EdgeInsets.all(24.0),
child: Form(
key: _formKey,
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
crossAxisAlignment: CrossAxisAlignment.stretch,
children: [
const Icon(
Icons.lock_person_rounded,
size: 80,
color: Colors.deepPurple,
),
const SizedBox(height: 32),
Text(
'Welcome Back',
style: Theme.of(context).textTheme.headlineMedium?.copyWith(
fontWeight: FontWeight.bold,
color: Colors.deepPurple.shade900,
),
textAlign: TextAlign.center,
),
const SizedBox(height: 8),
Text(
'Sign in to your account',
style: Theme.of(context).textTheme.bodyLarge?.copyWith(
color: Colors.grey.shade600,
),
textAlign: TextAlign.center,
),
const SizedBox(height: 48),
TextFormField(
controller: _emailController,
keyboardType: TextInputType.emailAddress,
decoration: const InputDecoration(
labelText: 'Email',
prefixIcon: Icon(Icons.email_outlined),
hintText: 'user@example.com',
),
validator: (value) {
if (value == null || value.isEmpty) {
return 'Please enter your email';
}
if (!GetUtils.isEmail(value)) {
return 'Please enter a valid email address';
}
return null;
},
),
const SizedBox(height: 20),
TextFormField(
controller: _passwordController,
obscureText: _obscurePassword,
decoration: InputDecoration(
labelText: 'Password',
prefixIcon: const Icon(Icons.lock_outline),
suffixIcon: IconButton(
icon: Icon(
_obscurePassword
? Icons.visibility_outlined
: Icons.visibility_off_outlined,
),
onPressed: () {
setState(() {
_obscurePassword = !_obscurePassword;
});
},
),
),
validator: (value) {
if (value == null || value.isEmpty) {
return 'Please enter your password';
}
if (value.length < 6) {
return 'Password must be at least 6 characters';
}
return null;
},
),
const SizedBox(height: 32),
VibrateButton(
loading: _isLoading,
formState: _formKey,
text: 'Login',
onTap: _handleLogin,
radius: 12,
height: 54,
color: Colors.deepPurple,
textStyle: const TextStyle(
color: Colors.white,
fontSize: 18,
fontWeight: FontWeight.w600,
),
),
const SizedBox(height: 24),
TextButton(
onPressed: () {},
child: Text(
'Forgot Password?',
style: TextStyle(color: Colors.deepPurple.shade700),
),
),
],
),
),
),
),
),
),
);
}
}