Eliminar creche_app/lib/features/auth/login_screen.dart
This commit is contained in:
parent
70356f8863
commit
4de3bc12fd
|
|
@ -1,449 +0,0 @@
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:flutter/foundation.dart';
|
|
||||||
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
|
||||||
import 'package:go_router/go_router.dart';
|
|
||||||
import 'package:supabase_flutter/supabase_flutter.dart';
|
|
||||||
import '/core/auth_provider.dart';
|
|
||||||
|
|
||||||
class LoginScreen extends ConsumerStatefulWidget {
|
|
||||||
const LoginScreen({super.key});
|
|
||||||
@override
|
|
||||||
ConsumerState<LoginScreen> createState() => _LoginScreenState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _LoginScreenState extends ConsumerState<LoginScreen>
|
|
||||||
with SingleTickerProviderStateMixin {
|
|
||||||
final _emailCtrl = TextEditingController();
|
|
||||||
final _passCtrl = TextEditingController();
|
|
||||||
final _nameCtrl = TextEditingController(); // só no registo
|
|
||||||
bool _isRegister = false; // toggle login/registo
|
|
||||||
bool _obscure = true;
|
|
||||||
bool _loading = false;
|
|
||||||
bool _biometricLoading = false;
|
|
||||||
String? _error;
|
|
||||||
late AnimationController _animCtrl;
|
|
||||||
late Animation<double> _fadeAnim;
|
|
||||||
late Animation<Offset> _slideAnim;
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
_animCtrl = AnimationController(vsync: this, duration: const Duration(milliseconds: 700));
|
|
||||||
_fadeAnim = CurvedAnimation(parent: _animCtrl, curve: Curves.easeOut);
|
|
||||||
_slideAnim = Tween<Offset>(begin: const Offset(0, 0.07), end: Offset.zero)
|
|
||||||
.animate(CurvedAnimation(parent: _animCtrl, curve: Curves.easeOut));
|
|
||||||
_animCtrl.forward();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
_animCtrl.dispose();
|
|
||||||
_emailCtrl.dispose();
|
|
||||||
_passCtrl.dispose();
|
|
||||||
_nameCtrl.dispose();
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
// ── toggle entre Login e Registo ─────────────────────────────
|
|
||||||
void _toggleMode() {
|
|
||||||
setState(() { _isRegister = !_isRegister; _error = null; });
|
|
||||||
_animCtrl.forward(from: 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
// ── Login ─────────────────────────────────────────────────────
|
|
||||||
Future<void> _signIn() async {
|
|
||||||
if (_emailCtrl.text.trim().isEmpty || _passCtrl.text.isEmpty) {
|
|
||||||
setState(() => _error = 'Preencha o email e a senha.');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
setState(() { _loading = true; _error = null; });
|
|
||||||
try {
|
|
||||||
await ref.read(authNotifierProvider.notifier).signIn(
|
|
||||||
_emailCtrl.text.trim(), _passCtrl.text,
|
|
||||||
);
|
|
||||||
if (mounted) context.go('/home');
|
|
||||||
} on WaitingApprovalException {
|
|
||||||
if (mounted) context.go('/waiting');
|
|
||||||
} catch (e) {
|
|
||||||
setState(() => _error = _friendly(e.toString()));
|
|
||||||
} finally {
|
|
||||||
if (mounted) setState(() => _loading = false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ── Registo ───────────────────────────────────────────────────
|
|
||||||
Future<void> _signUp() async {
|
|
||||||
final email = _emailCtrl.text.trim();
|
|
||||||
final name = _nameCtrl.text.trim();
|
|
||||||
final pass = _passCtrl.text;
|
|
||||||
|
|
||||||
if (email.isEmpty || pass.isEmpty || name.isEmpty) {
|
|
||||||
setState(() => _error = 'Preencha todos os campos.');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (pass.length < 6) {
|
|
||||||
setState(() => _error = 'A senha deve ter pelo menos 6 caracteres.');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
setState(() { _loading = true; _error = null; });
|
|
||||||
try {
|
|
||||||
final sb = Supabase.instance.client;
|
|
||||||
|
|
||||||
// 1. Criar conta no Supabase Auth
|
|
||||||
final res = await sb.auth.signUp(email: email, password: pass);
|
|
||||||
final user = res.user;
|
|
||||||
if (user == null) throw Exception('Erro ao criar conta.');
|
|
||||||
|
|
||||||
// 2. Criar perfil (sem role — será atribuído pelo convite)
|
|
||||||
await sb.from('profiles').upsert({
|
|
||||||
'user_id': user.id,
|
|
||||||
'full_name': name,
|
|
||||||
'role': 'parent', // role temporário; convite vai actualizar
|
|
||||||
});
|
|
||||||
|
|
||||||
// 3. Ir para o splash — ele detecta convites pendentes automaticamente
|
|
||||||
if (mounted) {
|
|
||||||
_showSnack('Conta criada! A verificar convites...', ok: true);
|
|
||||||
await Future.delayed(const Duration(milliseconds: 800));
|
|
||||||
if (mounted) context.go('/splash');
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
setState(() => _error = _friendly(e.toString()));
|
|
||||||
} finally {
|
|
||||||
if (mounted) setState(() => _loading = false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ── Recuperar senha ───────────────────────────────────────────
|
|
||||||
Future<void> _forgotPassword() async {
|
|
||||||
final email = _emailCtrl.text.trim();
|
|
||||||
if (email.isEmpty) {
|
|
||||||
setState(() => _error = 'Digite o seu email primeiro.');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
await Supabase.instance.client.auth.resetPasswordForEmail(email);
|
|
||||||
if (mounted) _showSnack('Email de recuperação enviado para $email.', ok: true);
|
|
||||||
} catch (e) {
|
|
||||||
setState(() => _error = 'Erro ao enviar email: $e');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ── Biometria ────────────────────────────────────────────────
|
|
||||||
Future<void> _biometricSignIn() async {
|
|
||||||
if (kIsWeb) {
|
|
||||||
setState(() => _error = 'Biometria não disponível no Web.');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
setState(() { _biometricLoading = true; _error = null; });
|
|
||||||
try {
|
|
||||||
await ref.read(authNotifierProvider.notifier).biometricSignIn();
|
|
||||||
if (mounted) context.go('/home');
|
|
||||||
} on WaitingApprovalException {
|
|
||||||
if (mounted) context.go('/waiting');
|
|
||||||
} catch (e) {
|
|
||||||
setState(() => _error = _friendly(e.toString()));
|
|
||||||
} finally {
|
|
||||||
if (mounted) setState(() => _biometricLoading = false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void _showSnack(String msg, {bool ok = false}) {
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(SnackBar(
|
|
||||||
content: Text(msg, style: const TextStyle(color: Colors.white)),
|
|
||||||
backgroundColor: ok ? const Color(0xFF2ECC71) : Colors.red,
|
|
||||||
behavior: SnackBarBehavior.floating,
|
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(12)),
|
|
||||||
duration: const Duration(seconds: 3),
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
String _friendly(String e) {
|
|
||||||
if (e.contains('Invalid login')) return 'Email ou senha incorrectos.';
|
|
||||||
if (e.contains('Email not confirmed')) return 'Confirme o seu email primeiro.';
|
|
||||||
if (e.contains('already registered')) return 'Este email já tem conta. Faça login.';
|
|
||||||
if (e.contains('Password should')) return 'A senha deve ter pelo menos 6 caracteres.';
|
|
||||||
if (e.contains('IP')) return e;
|
|
||||||
return 'Erro. Tente novamente.';
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Scaffold(
|
|
||||||
backgroundColor: const Color(0xFF0D1117),
|
|
||||||
body: Stack(children: [
|
|
||||||
Positioned(top: -80, right: -60, child: _Orb(size: 280, color: const Color(0xFF4FC3F7).withOpacity(0.11))),
|
|
||||||
Positioned(bottom: -60, left: -40, child: _Orb(size: 220, color: const Color(0xFFA5D6A7).withOpacity(0.09))),
|
|
||||||
SafeArea(
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 28),
|
|
||||||
child: FadeTransition(
|
|
||||||
opacity: _fadeAnim,
|
|
||||||
child: SlideTransition(
|
|
||||||
position: _slideAnim,
|
|
||||||
child: Column(children: [
|
|
||||||
const SizedBox(height: 44),
|
|
||||||
|
|
||||||
// ── Logo ──────────────────────────────────────
|
|
||||||
Container(
|
|
||||||
padding: const EdgeInsets.all(18),
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
shape: BoxShape.circle,
|
|
||||||
gradient: RadialGradient(colors: [
|
|
||||||
const Color(0xFF4FC3F7).withOpacity(0.18), Colors.transparent,
|
|
||||||
]),
|
|
||||||
border: Border.all(color: const Color(0xFF4FC3F7).withOpacity(0.25), width: 1.5),
|
|
||||||
),
|
|
||||||
child: Image.asset('assets/logo.png', height: 80,
|
|
||||||
errorBuilder: (_, __, ___) => const Icon(Icons.child_care, size: 75, color: Color(0xFF4FC3F7))),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 20),
|
|
||||||
const Text('Creche e Berçário',
|
|
||||||
style: TextStyle(color: Color(0xFF4FC3F7), fontSize: 13, letterSpacing: 2.5, fontWeight: FontWeight.w500)),
|
|
||||||
const SizedBox(height: 3),
|
|
||||||
const Text('SEMENTES DO FUTURO',
|
|
||||||
style: TextStyle(color: Colors.white, fontSize: 21, fontWeight: FontWeight.w900, letterSpacing: 1.2)),
|
|
||||||
const SizedBox(height: 5),
|
|
||||||
Text('"Conforto, cuidado e aprendizagem"',
|
|
||||||
style: TextStyle(color: Colors.white.withOpacity(0.35), fontSize: 11, fontStyle: FontStyle.italic)),
|
|
||||||
const SizedBox(height: 36),
|
|
||||||
|
|
||||||
// ── Card principal ────────────────────────────
|
|
||||||
Container(
|
|
||||||
padding: const EdgeInsets.all(26),
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: const Color(0xFF161B22),
|
|
||||||
borderRadius: BorderRadius.circular(24),
|
|
||||||
border: Border.all(color: Colors.white.withOpacity(0.07)),
|
|
||||||
boxShadow: [BoxShadow(color: Colors.black.withOpacity(0.35), blurRadius: 40, offset: const Offset(0, 16))],
|
|
||||||
),
|
|
||||||
child: Column(crossAxisAlignment: CrossAxisAlignment.start, children: [
|
|
||||||
|
|
||||||
// Título dinâmico
|
|
||||||
Text(
|
|
||||||
_isRegister ? 'Criar conta 📝' : 'Bem-vindo de volta 👋',
|
|
||||||
style: const TextStyle(color: Colors.white, fontSize: 20, fontWeight: FontWeight.bold),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 4),
|
|
||||||
Text(
|
|
||||||
_isRegister
|
|
||||||
? 'Cria a tua conta — um convite vai atribuir o teu acesso'
|
|
||||||
: 'Faz login para continuar',
|
|
||||||
style: TextStyle(color: Colors.white.withOpacity(0.4), fontSize: 12),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 22),
|
|
||||||
|
|
||||||
// Erro
|
|
||||||
if (_error != null) ...[
|
|
||||||
Container(
|
|
||||||
padding: const EdgeInsets.all(12),
|
|
||||||
decoration: BoxDecoration(color: Colors.red.withOpacity(0.1),
|
|
||||||
borderRadius: BorderRadius.circular(10), border: Border.all(color: Colors.red.withOpacity(0.3))),
|
|
||||||
child: Row(children: [
|
|
||||||
const Icon(Icons.error_outline, color: Colors.red, size: 16),
|
|
||||||
const SizedBox(width: 8),
|
|
||||||
Expanded(child: Text(_error!, style: const TextStyle(color: Colors.red, fontSize: 12))),
|
|
||||||
GestureDetector(onTap: () => setState(() => _error = null),
|
|
||||||
child: const Icon(Icons.close, color: Colors.red, size: 14)),
|
|
||||||
]),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 16),
|
|
||||||
],
|
|
||||||
|
|
||||||
// Nome (só no registo)
|
|
||||||
if (_isRegister) ...[
|
|
||||||
_Field(ctrl: _nameCtrl, label: 'Nome completo', icon: Icons.person_outline),
|
|
||||||
const SizedBox(height: 14),
|
|
||||||
],
|
|
||||||
|
|
||||||
// Email
|
|
||||||
_Field(ctrl: _emailCtrl, label: 'Email', icon: Icons.alternate_email, type: TextInputType.emailAddress),
|
|
||||||
const SizedBox(height: 14),
|
|
||||||
|
|
||||||
// Senha
|
|
||||||
_Field(
|
|
||||||
ctrl: _passCtrl, label: 'Senha', icon: Icons.lock_outline, obscure: _obscure,
|
|
||||||
onSubmitted: (_) => _isRegister ? _signUp() : _signIn(),
|
|
||||||
suffix: IconButton(
|
|
||||||
icon: Icon(_obscure ? Icons.visibility_off : Icons.visibility, color: Colors.white38, size: 20),
|
|
||||||
onPressed: () => setState(() => _obscure = !_obscure),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
|
|
||||||
if (!_isRegister) ...[
|
|
||||||
const SizedBox(height: 12),
|
|
||||||
Align(
|
|
||||||
alignment: Alignment.centerRight,
|
|
||||||
child: GestureDetector(
|
|
||||||
onTap: _forgotPassword,
|
|
||||||
child: const Text('Esqueci a senha',
|
|
||||||
style: TextStyle(color: Color(0xFF4FC3F7), fontSize: 12, fontWeight: FontWeight.w500)),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
|
|
||||||
const SizedBox(height: 24),
|
|
||||||
|
|
||||||
// Botão principal
|
|
||||||
_GradButton(
|
|
||||||
label: _isRegister ? 'Criar Conta' : 'Entrar',
|
|
||||||
isLoading: _loading,
|
|
||||||
onTap: _isRegister ? _signUp : _signIn,
|
|
||||||
),
|
|
||||||
|
|
||||||
if (!_isRegister) ...[
|
|
||||||
const SizedBox(height: 14),
|
|
||||||
Row(children: [
|
|
||||||
Expanded(child: Divider(color: Colors.white.withOpacity(0.09))),
|
|
||||||
Padding(padding: const EdgeInsets.symmetric(horizontal: 12),
|
|
||||||
child: Text('ou', style: TextStyle(color: Colors.white.withOpacity(0.25), fontSize: 11))),
|
|
||||||
Expanded(child: Divider(color: Colors.white.withOpacity(0.09))),
|
|
||||||
]),
|
|
||||||
const SizedBox(height: 14),
|
|
||||||
_BiometricBtn(isLoading: _biometricLoading, onTap: _biometricSignIn),
|
|
||||||
],
|
|
||||||
]),
|
|
||||||
),
|
|
||||||
|
|
||||||
const SizedBox(height: 22),
|
|
||||||
|
|
||||||
// ── Toggle Login / Registo ─────────────────────
|
|
||||||
GestureDetector(
|
|
||||||
onTap: _toggleMode,
|
|
||||||
child: Container(
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 20, vertical: 14),
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: const Color(0xFF161B22),
|
|
||||||
borderRadius: BorderRadius.circular(14),
|
|
||||||
border: Border.all(color: Colors.white.withOpacity(0.07)),
|
|
||||||
),
|
|
||||||
child: Row(mainAxisAlignment: MainAxisAlignment.center, children: [
|
|
||||||
Text(
|
|
||||||
_isRegister ? 'Já tens conta? ' : 'Ainda não tens conta? ',
|
|
||||||
style: TextStyle(color: Colors.white.withOpacity(0.45), fontSize: 13),
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
_isRegister ? 'Fazer login' : 'Criar conta',
|
|
||||||
style: const TextStyle(color: Color(0xFF4FC3F7), fontSize: 13, fontWeight: FontWeight.bold),
|
|
||||||
),
|
|
||||||
]),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
|
|
||||||
const SizedBox(height: 20),
|
|
||||||
// Aviso do fluxo de convites
|
|
||||||
if (_isRegister) Container(
|
|
||||||
padding: const EdgeInsets.all(12),
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: const Color(0xFF4FC3F7).withOpacity(0.06),
|
|
||||||
borderRadius: BorderRadius.circular(12),
|
|
||||||
border: Border.all(color: const Color(0xFF4FC3F7).withOpacity(0.2)),
|
|
||||||
),
|
|
||||||
child: const Row(crossAxisAlignment: CrossAxisAlignment.start, children: [
|
|
||||||
Icon(Icons.info_outline, color: Color(0xFF4FC3F7), size: 16),
|
|
||||||
SizedBox(width: 10),
|
|
||||||
Expanded(child: Text(
|
|
||||||
'Após criar a conta, se tiveres um convite da Diretora, verás automaticamente o ecrã de convite para aceitar o teu acesso.',
|
|
||||||
style: TextStyle(color: Color(0xFF888888), fontSize: 11, height: 1.5),
|
|
||||||
)),
|
|
||||||
]),
|
|
||||||
),
|
|
||||||
|
|
||||||
const SizedBox(height: 20),
|
|
||||||
Text('v1.0 · Diário do Candengue',
|
|
||||||
style: TextStyle(color: Colors.white.withOpacity(0.15), fontSize: 10)),
|
|
||||||
const SizedBox(height: 24),
|
|
||||||
]),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
]),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ── Widgets auxiliares ─────────────────────────────────────────────
|
|
||||||
|
|
||||||
class _Field extends StatelessWidget {
|
|
||||||
final TextEditingController ctrl;
|
|
||||||
final String label;
|
|
||||||
final IconData icon;
|
|
||||||
final bool obscure;
|
|
||||||
final TextInputType type;
|
|
||||||
final Widget? suffix;
|
|
||||||
final Function(String)? onSubmitted;
|
|
||||||
const _Field({required this.ctrl, required this.label, required this.icon,
|
|
||||||
this.obscure = false, this.type = TextInputType.text, this.suffix, this.onSubmitted});
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) => TextField(
|
|
||||||
controller: ctrl, obscureText: obscure, keyboardType: type, onSubmitted: onSubmitted,
|
|
||||||
style: const TextStyle(color: Colors.white, fontSize: 14),
|
|
||||||
decoration: InputDecoration(
|
|
||||||
labelText: label,
|
|
||||||
labelStyle: TextStyle(color: Colors.white.withOpacity(0.4), fontSize: 13),
|
|
||||||
prefixIcon: Icon(icon, color: const Color(0xFF4FC3F7).withOpacity(0.7), size: 19),
|
|
||||||
suffixIcon: suffix,
|
|
||||||
filled: true, fillColor: Colors.white.withOpacity(0.05),
|
|
||||||
enabledBorder: OutlineInputBorder(borderRadius: BorderRadius.circular(12), borderSide: BorderSide(color: Colors.white.withOpacity(0.1))),
|
|
||||||
focusedBorder: OutlineInputBorder(borderRadius: BorderRadius.circular(12), borderSide: const BorderSide(color: Color(0xFF4FC3F7), width: 1.5)),
|
|
||||||
contentPadding: const EdgeInsets.symmetric(horizontal: 16, vertical: 15),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
class _GradButton extends StatelessWidget {
|
|
||||||
final String label; final bool isLoading; final VoidCallback onTap;
|
|
||||||
const _GradButton({required this.label, required this.isLoading, required this.onTap});
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) => GestureDetector(
|
|
||||||
onTap: isLoading ? null : onTap,
|
|
||||||
child: AnimatedContainer(
|
|
||||||
duration: const Duration(milliseconds: 200), height: 52,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
gradient: LinearGradient(colors: isLoading
|
|
||||||
? [const Color(0xFF2A6A8A), const Color(0xFF2A6A8A)]
|
|
||||||
: [const Color(0xFF4FC3F7), const Color(0xFF0288D1)]),
|
|
||||||
borderRadius: BorderRadius.circular(14),
|
|
||||||
boxShadow: isLoading ? [] : [BoxShadow(color: const Color(0xFF4FC3F7).withOpacity(0.3), blurRadius: 18, offset: const Offset(0, 6))],
|
|
||||||
),
|
|
||||||
child: Center(child: isLoading
|
|
||||||
? const SizedBox(height: 22, width: 22, child: CircularProgressIndicator(color: Colors.white, strokeWidth: 2.5))
|
|
||||||
: Text(label, style: const TextStyle(color: Colors.white, fontSize: 16, fontWeight: FontWeight.bold))),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
class _BiometricBtn extends StatelessWidget {
|
|
||||||
final bool isLoading; final VoidCallback onTap;
|
|
||||||
const _BiometricBtn({required this.isLoading, required this.onTap});
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) => GestureDetector(
|
|
||||||
onTap: isLoading ? null : onTap,
|
|
||||||
child: Container(
|
|
||||||
height: 50,
|
|
||||||
decoration: BoxDecoration(color: Colors.white.withOpacity(0.04), borderRadius: BorderRadius.circular(14),
|
|
||||||
border: Border.all(color: Colors.white.withOpacity(0.1))),
|
|
||||||
child: Row(mainAxisAlignment: MainAxisAlignment.center, children: [
|
|
||||||
isLoading
|
|
||||||
? const SizedBox(height: 20, width: 20, child: CircularProgressIndicator(color: Color(0xFF4FC3F7), strokeWidth: 2))
|
|
||||||
: const Icon(Icons.fingerprint, color: Color(0xFF4FC3F7), size: 22),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
Text(kIsWeb ? 'Biometria (só mobile)' : 'Entrar com biometria',
|
|
||||||
style: TextStyle(color: kIsWeb ? Colors.white24 : Colors.white60, fontSize: 13)),
|
|
||||||
]),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
class _Orb extends StatelessWidget {
|
|
||||||
final double size; final Color color;
|
|
||||||
const _Orb({required this.size, required this.color});
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) => Container(
|
|
||||||
width: size, height: size,
|
|
||||||
decoration: BoxDecoration(shape: BoxShape.circle, color: color,
|
|
||||||
boxShadow: [BoxShadow(color: color, blurRadius: size / 2)]),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
Loading…
Reference in New Issue