feat(ui): add walk guide, server connect, pairing, auth, and splash screens
This commit is contained in:
parent
c8a1818e97
commit
790db043a9
@ -1 +1,256 @@
|
||||
export '../screens.dart';
|
||||
// ignore_for_file: use_build_context_synchronously
|
||||
|
||||
import 'dart:async';
|
||||
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||
import 'package:go_router/go_router.dart';
|
||||
import 'package:shared_preferences/shared_preferences.dart';
|
||||
|
||||
import '../../app/app_cubit.dart';
|
||||
import '../../app/injection_container.dart';
|
||||
import '../../core/constants/app_constants.dart';
|
||||
import '../../core/network/api_client.dart';
|
||||
import '../../core/services/offline_queue_service.dart';
|
||||
import '../../core/services/tts_service.dart';
|
||||
import '../../core/services/websocket_service.dart';
|
||||
import '../../core/storage/secure_storage.dart';
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// LoginScreen
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
class LoginScreen extends StatefulWidget {
|
||||
const LoginScreen({super.key});
|
||||
|
||||
@override
|
||||
State<LoginScreen> createState() => _LoginScreenState();
|
||||
}
|
||||
|
||||
class _LoginScreenState extends State<LoginScreen> {
|
||||
final _email = TextEditingController();
|
||||
final _password = TextEditingController();
|
||||
bool _loading = false;
|
||||
bool _showPassword = false;
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
_loadPendingLoginEmail();
|
||||
}
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
_email.dispose();
|
||||
_password.dispose();
|
||||
super.dispose();
|
||||
}
|
||||
|
||||
Future<void> _loadPendingLoginEmail() async {
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
final pendingEmail = prefs.getString('pending_login_email');
|
||||
if (!mounted) return;
|
||||
setState(() {
|
||||
if (pendingEmail != null && pendingEmail.isNotEmpty) {
|
||||
_email.text = pendingEmail;
|
||||
}
|
||||
});
|
||||
await prefs.remove('pending_login_email');
|
||||
}
|
||||
|
||||
Future<void> _login() async {
|
||||
if (_email.text.trim().isEmpty || _password.text.isEmpty) {
|
||||
_snack(context, 'Isi email dan password dulu.');
|
||||
return;
|
||||
}
|
||||
setState(() => _loading = true);
|
||||
try {
|
||||
final res = await sl<ApiClient>().dio.post('/auth/login', data: {
|
||||
'email': _email.text.trim(),
|
||||
'password': _password.text,
|
||||
});
|
||||
await _saveAuthAndRoute(
|
||||
context, Map<String, dynamic>.from(res.data['data'] as Map));
|
||||
} on DioException catch (e) {
|
||||
_snack(context, _friendlyDioMessage(e, fallback: 'Login gagal'));
|
||||
} catch (e) {
|
||||
_snack(context, 'Login gagal: $e');
|
||||
} finally {
|
||||
if (mounted) setState(() => _loading = false);
|
||||
}
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return _AuthFrame(
|
||||
title: 'Sign in',
|
||||
subtitle: 'Masuk sebagai Guardian atau User.',
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||
children: [
|
||||
TextField(
|
||||
controller: _email,
|
||||
keyboardType: TextInputType.emailAddress,
|
||||
textInputAction: TextInputAction.next,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Email',
|
||||
prefixIcon: Icon(Icons.email_outlined),
|
||||
)),
|
||||
const SizedBox(height: 12),
|
||||
TextField(
|
||||
controller: _password,
|
||||
obscureText: !_showPassword,
|
||||
textInputAction: TextInputAction.done,
|
||||
onSubmitted: (_) => _login(),
|
||||
decoration: InputDecoration(
|
||||
labelText: 'Password',
|
||||
prefixIcon: const Icon(Icons.lock_outline),
|
||||
suffixIcon: IconButton(
|
||||
icon: Icon(
|
||||
_showPassword ? Icons.visibility_off : Icons.visibility),
|
||||
onPressed: () =>
|
||||
setState(() => _showPassword = !_showPassword),
|
||||
),
|
||||
)),
|
||||
const SizedBox(height: 18),
|
||||
FilledButton.icon(
|
||||
onPressed: _loading ? null : _login,
|
||||
icon: _loading
|
||||
? const SizedBox(
|
||||
width: 18,
|
||||
height: 18,
|
||||
child: CircularProgressIndicator(strokeWidth: 2))
|
||||
: const Icon(Icons.login),
|
||||
label: const Text('Login'),
|
||||
),
|
||||
TextButton(
|
||||
onPressed: () => context.go('/register'),
|
||||
child: const Text('Buat akun baru')),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Shared private widgets
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
class _AuthFrame extends StatelessWidget {
|
||||
final String title;
|
||||
final String subtitle;
|
||||
final Widget child;
|
||||
const _AuthFrame(
|
||||
{required this.title, required this.subtitle, required this.child});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
body: Center(
|
||||
child: SingleChildScrollView(
|
||||
padding: const EdgeInsets.all(24),
|
||||
child: ConstrainedBox(
|
||||
constraints: const BoxConstraints(maxWidth: 460),
|
||||
child: Card(
|
||||
elevation: 0,
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(18),
|
||||
side: const BorderSide(color: Color(0xFFE2E8F0))),
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(24),
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||
children: [
|
||||
const Icon(Icons.navigation_rounded,
|
||||
color: Color(0xFF1A56DB), size: 42),
|
||||
const SizedBox(height: 14),
|
||||
Text(title,
|
||||
textAlign: TextAlign.center,
|
||||
style: Theme.of(context)
|
||||
.textTheme
|
||||
.headlineSmall
|
||||
?.copyWith(fontWeight: FontWeight.w800)),
|
||||
const SizedBox(height: 4),
|
||||
Text(subtitle,
|
||||
textAlign: TextAlign.center,
|
||||
style: const TextStyle(color: Color(0xFF64748B))),
|
||||
const SizedBox(height: 22),
|
||||
child,
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Helpers (shared for login flow)
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
Future<void> _saveAuthAndRoute(
|
||||
BuildContext context, Map<String, dynamic> data) async {
|
||||
await sl<SecureStorage>().saveTokens(
|
||||
accessToken: data['accessToken'],
|
||||
refreshToken: data['refreshToken'],
|
||||
role: data['role'],
|
||||
userId: data['userId'].toString(),
|
||||
displayName: data['displayName'],
|
||||
uniqueUserId: data['uniqueUserId'],
|
||||
);
|
||||
final serverUrl = await AppConstants.getServerUrl();
|
||||
if (serverUrl != null) {
|
||||
context
|
||||
.read<AppCubit>()
|
||||
.setSession(role: data['role'], serverUrl: serverUrl);
|
||||
_startPostLoginServices(serverUrl);
|
||||
}
|
||||
sl<TtsService>().speak('Selamat datang ${data['displayName'] ?? ''}');
|
||||
if (context.mounted) {
|
||||
context.go(data['role'] == 'ROLE_GUARDIAN'
|
||||
? '/guardian/dashboard'
|
||||
: '/user/walkguide');
|
||||
}
|
||||
}
|
||||
|
||||
void _startPostLoginServices(String serverUrl) {
|
||||
Future.microtask(() async {
|
||||
try {
|
||||
await sl<WebSocketService>()
|
||||
.connect(serverUrl)
|
||||
.timeout(const Duration(seconds: 2));
|
||||
await sl<OfflineQueueService>()
|
||||
.syncPending(sl<ApiClient>())
|
||||
.timeout(const Duration(seconds: 3));
|
||||
} catch (e) {
|
||||
debugPrint('Post-login services skipped: $e');
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
void _snack(BuildContext context, String message) {
|
||||
if (context.mounted) {
|
||||
ScaffoldMessenger.of(context)
|
||||
.showSnackBar(SnackBar(content: Text(message)));
|
||||
}
|
||||
}
|
||||
|
||||
String _friendlyDioMessage(DioException e, {required String fallback}) {
|
||||
final data = e.response?.data;
|
||||
if (data is Map && data['message'] != null) return data['message'].toString();
|
||||
if (e.response?.statusCode == 401) {
|
||||
return 'Email atau password salah.';
|
||||
}
|
||||
if (e.type == DioExceptionType.connectionTimeout ||
|
||||
e.type == DioExceptionType.receiveTimeout) {
|
||||
return 'Server terlalu lama merespons. Pastikan backend masih running dan URL server benar.';
|
||||
}
|
||||
if (e.type == DioExceptionType.connectionError) {
|
||||
return 'Tidak bisa ke server. Di Chrome pakai http://localhost:8080. Di HP pakai IP laptop/server, bukan localhost.';
|
||||
}
|
||||
return fallback;
|
||||
}
|
||||
|
||||
@ -1 +1,393 @@
|
||||
export '../screens.dart';
|
||||
// ignore_for_file: use_build_context_synchronously
|
||||
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:go_router/go_router.dart';
|
||||
import 'package:shared_preferences/shared_preferences.dart';
|
||||
|
||||
import '../../app/injection_container.dart';
|
||||
import '../../core/network/api_client.dart';
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// RegisterScreen
|
||||
// ---------------------------------------------------------------------------
|
||||
//
|
||||
// Step 1: Pilih role (Guardian / User) dengan UI card besar.
|
||||
// • Guardian: icon perisai, "I will guide someone"
|
||||
// • User: icon mata, "I need navigation assistance"
|
||||
// Step 2: Form email, password, displayName.
|
||||
// Submit → POST /auth/register → simpan pending email → dialog sukses → /login.
|
||||
// Jika User: dialog menampilkan uniqueUserId untuk dibagikan ke Guardian.
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
class RegisterScreen extends StatefulWidget {
|
||||
const RegisterScreen({super.key});
|
||||
|
||||
@override
|
||||
State<RegisterScreen> createState() => _RegisterScreenState();
|
||||
}
|
||||
|
||||
class _RegisterScreenState extends State<RegisterScreen> {
|
||||
final _name = TextEditingController();
|
||||
final _email = TextEditingController();
|
||||
final _password = TextEditingController();
|
||||
bool _showPassword = false;
|
||||
String _role = 'USER';
|
||||
bool _loading = false;
|
||||
int _step = 0; // 0 = role selection, 1 = form
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
_name.dispose();
|
||||
_email.dispose();
|
||||
_password.dispose();
|
||||
super.dispose();
|
||||
}
|
||||
|
||||
Future<void> _register() async {
|
||||
if (_name.text.trim().isEmpty ||
|
||||
_email.text.trim().isEmpty ||
|
||||
_password.text.isEmpty) {
|
||||
_snack(context, 'Isi semua field terlebih dahulu.');
|
||||
return;
|
||||
}
|
||||
setState(() => _loading = true);
|
||||
try {
|
||||
final res = await sl<ApiClient>().dio.post('/auth/register', data: {
|
||||
'displayName': _name.text.trim(),
|
||||
'email': _email.text.trim(),
|
||||
'password': _password.text,
|
||||
'role': _role,
|
||||
});
|
||||
final data = Map<String, dynamic>.from(res.data['data'] as Map);
|
||||
if (!mounted) return;
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
await prefs.setString('pending_login_email', _email.text.trim());
|
||||
await _showRegisterSuccess(context, data);
|
||||
if (mounted) context.go('/login');
|
||||
} on DioException catch (e) {
|
||||
_snack(context, _friendlyDioMessage(e, fallback: 'Registrasi gagal'));
|
||||
} catch (e) {
|
||||
_snack(context, 'Registrasi gagal: $e');
|
||||
} finally {
|
||||
if (mounted) setState(() => _loading = false);
|
||||
}
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return _AuthFrame(
|
||||
title: _step == 0 ? 'Create Account' : 'Fill Your Details',
|
||||
subtitle: _step == 0
|
||||
? 'Who are you in the WalkGuide system?'
|
||||
: _role == 'USER'
|
||||
? 'User akan mendapat Unique ID untuk pairing.'
|
||||
: 'Guardian dapat monitor dan konfigurasi User.',
|
||||
child: _step == 0 ? _buildRoleStep(context) : _buildFormStep(context),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _buildRoleStep(BuildContext context) {
|
||||
return Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||
children: [
|
||||
_RoleCard(
|
||||
selected: _role == 'USER',
|
||||
icon: Icons.accessibility_new,
|
||||
title: 'User',
|
||||
subtitle: 'I need navigation assistance',
|
||||
onTap: () => setState(() => _role = 'USER'),
|
||||
),
|
||||
const SizedBox(height: 12),
|
||||
_RoleCard(
|
||||
selected: _role == 'GUARDIAN',
|
||||
icon: Icons.shield_outlined,
|
||||
title: 'Guardian',
|
||||
subtitle: 'I will guide someone',
|
||||
onTap: () => setState(() => _role = 'GUARDIAN'),
|
||||
),
|
||||
const SizedBox(height: 20),
|
||||
FilledButton.icon(
|
||||
onPressed: () => setState(() => _step = 1),
|
||||
icon: const Icon(Icons.arrow_forward),
|
||||
label: Text('Continue as $_role'),
|
||||
),
|
||||
TextButton(
|
||||
onPressed: () => context.go('/login'),
|
||||
child: const Text('Sudah punya akun')),
|
||||
],
|
||||
);
|
||||
}
|
||||
|
||||
Widget _buildFormStep(BuildContext context) {
|
||||
return Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||
children: [
|
||||
// Role indicator chip
|
||||
Container(
|
||||
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 6),
|
||||
decoration: BoxDecoration(
|
||||
color: _role == 'USER'
|
||||
? const Color(0xFFEFF6FF)
|
||||
: const Color(0xFFF0FDF4),
|
||||
borderRadius: BorderRadius.circular(999),
|
||||
),
|
||||
child: Row(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
Icon(
|
||||
_role == 'USER'
|
||||
? Icons.accessibility_new
|
||||
: Icons.shield_outlined,
|
||||
size: 16,
|
||||
color: _role == 'USER'
|
||||
? const Color(0xFF1A56DB)
|
||||
: const Color(0xFF16A34A),
|
||||
),
|
||||
const SizedBox(width: 6),
|
||||
Text(
|
||||
'Daftar sebagai $_role',
|
||||
style: TextStyle(
|
||||
fontWeight: FontWeight.w700,
|
||||
color: _role == 'USER'
|
||||
? const Color(0xFF1A56DB)
|
||||
: const Color(0xFF16A34A),
|
||||
),
|
||||
),
|
||||
const SizedBox(width: 6),
|
||||
GestureDetector(
|
||||
onTap: () => setState(() => _step = 0),
|
||||
child: const Icon(Icons.edit_outlined, size: 14),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
const SizedBox(height: 16),
|
||||
TextField(
|
||||
controller: _name,
|
||||
textInputAction: TextInputAction.next,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Display name',
|
||||
prefixIcon: Icon(Icons.person_outline),
|
||||
)),
|
||||
const SizedBox(height: 12),
|
||||
TextField(
|
||||
controller: _email,
|
||||
keyboardType: TextInputType.emailAddress,
|
||||
textInputAction: TextInputAction.next,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Email',
|
||||
prefixIcon: Icon(Icons.email_outlined),
|
||||
)),
|
||||
const SizedBox(height: 12),
|
||||
TextField(
|
||||
controller: _password,
|
||||
obscureText: !_showPassword,
|
||||
textInputAction: TextInputAction.done,
|
||||
onSubmitted: (_) => _register(),
|
||||
decoration: InputDecoration(
|
||||
labelText: 'Password',
|
||||
prefixIcon: const Icon(Icons.lock_outline),
|
||||
suffixIcon: IconButton(
|
||||
icon: Icon(
|
||||
_showPassword ? Icons.visibility_off : Icons.visibility),
|
||||
onPressed: () => setState(() => _showPassword = !_showPassword),
|
||||
),
|
||||
)),
|
||||
const SizedBox(height: 18),
|
||||
FilledButton.icon(
|
||||
onPressed: _loading ? null : _register,
|
||||
icon: _loading
|
||||
? const SizedBox(
|
||||
width: 18,
|
||||
height: 18,
|
||||
child: CircularProgressIndicator(strokeWidth: 2))
|
||||
: const Icon(Icons.person_add_alt_1),
|
||||
label: const Text('Register'),
|
||||
),
|
||||
TextButton(
|
||||
onPressed: () => context.go('/login'),
|
||||
child: const Text('Sudah punya akun')),
|
||||
],
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// _RoleCard
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
class _RoleCard extends StatelessWidget {
|
||||
final bool selected;
|
||||
final IconData icon;
|
||||
final String title;
|
||||
final String subtitle;
|
||||
final VoidCallback onTap;
|
||||
|
||||
const _RoleCard({
|
||||
required this.selected,
|
||||
required this.icon,
|
||||
required this.title,
|
||||
required this.subtitle,
|
||||
required this.onTap,
|
||||
});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return GestureDetector(
|
||||
onTap: onTap,
|
||||
child: AnimatedContainer(
|
||||
duration: const Duration(milliseconds: 180),
|
||||
padding: const EdgeInsets.all(16),
|
||||
decoration: BoxDecoration(
|
||||
color: selected ? const Color(0xFFEFF6FF) : Colors.white,
|
||||
borderRadius: BorderRadius.circular(14),
|
||||
border: Border.all(
|
||||
color: selected ? const Color(0xFF1A56DB) : const Color(0xFFE2E8F0),
|
||||
width: selected ? 2 : 1,
|
||||
),
|
||||
),
|
||||
child: Row(
|
||||
children: [
|
||||
Container(
|
||||
width: 48,
|
||||
height: 48,
|
||||
decoration: BoxDecoration(
|
||||
color: selected
|
||||
? const Color(0xFF1A56DB)
|
||||
: const Color(0xFFF1F5F9),
|
||||
borderRadius: BorderRadius.circular(12),
|
||||
),
|
||||
child: Icon(icon,
|
||||
color: selected ? Colors.white : const Color(0xFF64748B)),
|
||||
),
|
||||
const SizedBox(width: 14),
|
||||
Expanded(
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(title,
|
||||
style: const TextStyle(
|
||||
fontWeight: FontWeight.w800, fontSize: 16)),
|
||||
Text(subtitle,
|
||||
style: const TextStyle(
|
||||
color: Color(0xFF64748B), fontSize: 13)),
|
||||
],
|
||||
),
|
||||
),
|
||||
if (selected)
|
||||
const Icon(Icons.check_circle_rounded, color: Color(0xFF1A56DB)),
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Shared private widgets
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
class _AuthFrame extends StatelessWidget {
|
||||
final String title;
|
||||
final String subtitle;
|
||||
final Widget child;
|
||||
const _AuthFrame(
|
||||
{required this.title, required this.subtitle, required this.child});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
body: Center(
|
||||
child: SingleChildScrollView(
|
||||
padding: const EdgeInsets.all(24),
|
||||
child: ConstrainedBox(
|
||||
constraints: const BoxConstraints(maxWidth: 460),
|
||||
child: Card(
|
||||
elevation: 0,
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(18),
|
||||
side: const BorderSide(color: Color(0xFFE2E8F0))),
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(24),
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||
children: [
|
||||
const Icon(Icons.navigation_rounded,
|
||||
color: Color(0xFF1A56DB), size: 42),
|
||||
const SizedBox(height: 14),
|
||||
Text(title,
|
||||
textAlign: TextAlign.center,
|
||||
style: Theme.of(context)
|
||||
.textTheme
|
||||
.headlineSmall
|
||||
?.copyWith(fontWeight: FontWeight.w800)),
|
||||
const SizedBox(height: 4),
|
||||
Text(subtitle,
|
||||
textAlign: TextAlign.center,
|
||||
style: const TextStyle(color: Color(0xFF64748B))),
|
||||
const SizedBox(height: 22),
|
||||
child,
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Helpers
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
Future<void> _showRegisterSuccess(
|
||||
BuildContext context, Map<String, dynamic> data) async {
|
||||
final uniqueId = data['uniqueUserId']?.toString();
|
||||
final message = uniqueId == null || uniqueId.isEmpty
|
||||
? 'Registrasi berhasil. Silakan login.'
|
||||
: 'Registrasi berhasil!\n\nUnique User ID kamu:\n$uniqueId\n\nBagikan ID ini ke Guardian untuk pairing. Silakan login.';
|
||||
_snack(
|
||||
context,
|
||||
uniqueId == null
|
||||
? 'Registrasi berhasil.'
|
||||
: 'Registrasi berhasil. ID: $uniqueId');
|
||||
await showDialog<void>(
|
||||
context: context,
|
||||
builder: (dialogContext) => AlertDialog(
|
||||
title: const Text('Register Successful'),
|
||||
content: SelectableText(message),
|
||||
actions: [
|
||||
FilledButton(
|
||||
onPressed: () => Navigator.of(dialogContext).pop(),
|
||||
child: const Text('Login sekarang'),
|
||||
),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
void _snack(BuildContext context, String message) {
|
||||
if (context.mounted) {
|
||||
ScaffoldMessenger.of(context)
|
||||
.showSnackBar(SnackBar(content: Text(message)));
|
||||
}
|
||||
}
|
||||
|
||||
String _friendlyDioMessage(DioException e, {required String fallback}) {
|
||||
final data = e.response?.data;
|
||||
if (data is Map && data['message'] != null) return data['message'].toString();
|
||||
if (e.response?.statusCode == 409) {
|
||||
return 'Email sudah terdaftar. Gunakan email lain atau login.';
|
||||
}
|
||||
if (e.type == DioExceptionType.connectionTimeout ||
|
||||
e.type == DioExceptionType.receiveTimeout) {
|
||||
return 'Server terlalu lama merespons. Pastikan backend masih running.';
|
||||
}
|
||||
if (e.type == DioExceptionType.connectionError) {
|
||||
return 'Tidak bisa ke server. Di HP pakai IP server, bukan localhost.';
|
||||
}
|
||||
return fallback;
|
||||
}
|
||||
|
||||
@ -1 +1,136 @@
|
||||
export '../screens.dart';
|
||||
// ignore_for_file: use_build_context_synchronously
|
||||
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:go_router/go_router.dart';
|
||||
|
||||
import '../../app/injection_container.dart';
|
||||
import '../../core/storage/secure_storage.dart';
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// SplashScreen
|
||||
// ---------------------------------------------------------------------------
|
||||
//
|
||||
// Ditampilkan sesaat setelah user berhasil connect ke server (dari
|
||||
// ServerConnectScreen) atau saat app dibuka dan serverUrl sudah tersimpan.
|
||||
//
|
||||
// Logic:
|
||||
// 1. Delay singkat 500ms untuk animasi logo.
|
||||
// 2. Baca accessToken dari SecureStorage.
|
||||
// 3. Jika tidak ada → redirect /login.
|
||||
// 4. Jika ada → baca role → redirect /user/walkguide atau /guardian/dashboard.
|
||||
//
|
||||
// TTS "Welcome back, {displayName}" jika auto-login (token sudah ada).
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
class SplashScreen extends StatefulWidget {
|
||||
const SplashScreen({super.key});
|
||||
|
||||
@override
|
||||
State<SplashScreen> createState() => _SplashScreenState();
|
||||
}
|
||||
|
||||
class _SplashScreenState extends State<SplashScreen>
|
||||
with SingleTickerProviderStateMixin {
|
||||
late final AnimationController _animCtrl;
|
||||
late final Animation<double> _fadeAnim;
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
_animCtrl = AnimationController(
|
||||
vsync: this, duration: const Duration(milliseconds: 700));
|
||||
_fadeAnim = CurvedAnimation(parent: _animCtrl, curve: Curves.easeIn);
|
||||
_animCtrl.forward();
|
||||
_route();
|
||||
}
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
_animCtrl.dispose();
|
||||
super.dispose();
|
||||
}
|
||||
|
||||
Future<void> _route() async {
|
||||
try {
|
||||
// Animasi logo selalu tampil minimal 500ms agar tidak langsung flash.
|
||||
await Future.delayed(const Duration(milliseconds: 500));
|
||||
|
||||
final storage = sl<SecureStorage>();
|
||||
final token =
|
||||
await storage.getAccessToken().timeout(const Duration(seconds: 3));
|
||||
final role =
|
||||
await storage.getUserRole().timeout(const Duration(seconds: 3));
|
||||
|
||||
if (!mounted) return;
|
||||
|
||||
if (token == null || role == null) {
|
||||
context.go('/login');
|
||||
return;
|
||||
}
|
||||
|
||||
// Auto-login: arahkan ke home sesuai role.
|
||||
context.go(
|
||||
role == 'ROLE_GUARDIAN' ? '/guardian/dashboard' : '/user/walkguide');
|
||||
} catch (_) {
|
||||
if (mounted) context.go('/login');
|
||||
}
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
backgroundColor: const Color(0xFF1A56DB),
|
||||
body: Center(
|
||||
child: FadeTransition(
|
||||
opacity: _fadeAnim,
|
||||
child: Column(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
// Logo / icon
|
||||
Container(
|
||||
width: 100,
|
||||
height: 100,
|
||||
decoration: BoxDecoration(
|
||||
color: Colors.white.withOpacity(0.12),
|
||||
borderRadius: BorderRadius.circular(28),
|
||||
),
|
||||
child: const Icon(
|
||||
Icons.navigation_rounded,
|
||||
color: Colors.white,
|
||||
size: 60,
|
||||
),
|
||||
),
|
||||
const SizedBox(height: 24),
|
||||
const Text(
|
||||
'WalkGuide',
|
||||
style: TextStyle(
|
||||
color: Colors.white,
|
||||
fontSize: 34,
|
||||
fontWeight: FontWeight.w800,
|
||||
letterSpacing: -0.5,
|
||||
),
|
||||
),
|
||||
const SizedBox(height: 6),
|
||||
const Text(
|
||||
'AI-powered navigation for everyone',
|
||||
style: TextStyle(
|
||||
color: Colors.white70,
|
||||
fontSize: 13,
|
||||
),
|
||||
),
|
||||
const SizedBox(height: 48),
|
||||
const SizedBox(
|
||||
width: 28,
|
||||
height: 28,
|
||||
child: CircularProgressIndicator(
|
||||
color: Colors.white,
|
||||
strokeWidth: 2.5,
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
@ -1 +1,488 @@
|
||||
export '../screens.dart' show GuardianPairingScreen, UserPairingScreen;
|
||||
// ignore_for_file: use_build_context_synchronously, deprecated_member_use
|
||||
|
||||
import 'dart:async';
|
||||
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:go_router/go_router.dart';
|
||||
|
||||
import '../../app/injection_container.dart';
|
||||
import '../../core/network/api_client.dart';
|
||||
import '../../core/storage/secure_storage.dart';
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// UserPairingScreen
|
||||
// ---------------------------------------------------------------------------
|
||||
//
|
||||
// Ditampilkan ke akun ROLE_USER.
|
||||
// - Tampilkan uniqueUserId mereka (besar, bisa di-copy/share).
|
||||
// - Jika ada pending invite → tampilkan nama Guardian + tombol Accept / Reject.
|
||||
// - Jika sudah paired → tampilkan info Guardian + tombol Unpair.
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
class UserPairingScreen extends StatefulWidget {
|
||||
const UserPairingScreen({super.key});
|
||||
|
||||
@override
|
||||
State<UserPairingScreen> createState() => _UserPairingScreenState();
|
||||
}
|
||||
|
||||
class _UserPairingScreenState extends State<UserPairingScreen> {
|
||||
String? _uniqueId;
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
_loadUniqueId();
|
||||
}
|
||||
|
||||
Future<void> _loadUniqueId() async {
|
||||
var value = await sl<SecureStorage>().getUniqueUserId();
|
||||
if (value == null || value.isEmpty) {
|
||||
try {
|
||||
final res = await sl<ApiClient>()
|
||||
.dio
|
||||
.get('/user/profile')
|
||||
.timeout(const Duration(seconds: 5));
|
||||
final data = res.data['data'];
|
||||
if (data is Map) value = data['uniqueUserId']?.toString();
|
||||
} catch (_) {}
|
||||
}
|
||||
if (mounted) setState(() => _uniqueId = value);
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return _Page(
|
||||
title: 'Pairing',
|
||||
subtitle: 'Bagikan Unique ID ini ke Guardian untuk terhubung.',
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||
children: [
|
||||
if (_uniqueId == null || _uniqueId!.isEmpty)
|
||||
_InfoCard(
|
||||
title: 'Your Unique ID',
|
||||
value: 'Login sebagai User untuk melihat ID',
|
||||
icon: Icons.qr_code_2)
|
||||
else
|
||||
_InfoCard(
|
||||
title: 'Your Unique ID',
|
||||
value: _uniqueId!,
|
||||
icon: Icons.qr_code_2),
|
||||
const SizedBox(height: 16),
|
||||
_PairingStatusCard(allowUserResponse: true),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// GuardianPairingScreen
|
||||
// ---------------------------------------------------------------------------
|
||||
//
|
||||
// Ditampilkan ke akun ROLE_GUARDIAN.
|
||||
// - Input field 12-char User ID.
|
||||
// - Tombol "Send Invite".
|
||||
// - Status card: jika sudah paired → info User + tombol Unpair.
|
||||
// Jika pending → waiting state.
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
class GuardianPairingScreen extends StatefulWidget {
|
||||
const GuardianPairingScreen({super.key});
|
||||
|
||||
@override
|
||||
State<GuardianPairingScreen> createState() => _GuardianPairingScreenState();
|
||||
}
|
||||
|
||||
class _GuardianPairingScreenState extends State<GuardianPairingScreen> {
|
||||
final _id = TextEditingController();
|
||||
bool _loading = false;
|
||||
int _statusReload = 0;
|
||||
|
||||
Future<void> _invite() async {
|
||||
final uniqueId = _id.text.trim().toUpperCase();
|
||||
if (uniqueId.isEmpty || uniqueId.length != 12) {
|
||||
_snack(context, 'Unique ID harus 12 karakter dari akun User.');
|
||||
return;
|
||||
}
|
||||
setState(() => _loading = true);
|
||||
try {
|
||||
final res = await sl<ApiClient>().dio.post('/shared/pairing/invite',
|
||||
data: {'uniqueUserId': uniqueId}).timeout(const Duration(seconds: 8));
|
||||
_snack(
|
||||
context,
|
||||
res.data['message']?.toString() ??
|
||||
'Invite terkirim. Minta User buka menu Pairing lalu Accept.');
|
||||
setState(() => _statusReload++);
|
||||
} on DioException catch (e) {
|
||||
_snack(
|
||||
context,
|
||||
_friendlyDioMessage(e,
|
||||
fallback:
|
||||
'Invite gagal. Pastikan kamu login sebagai Guardian dan ID User benar.'));
|
||||
} on TimeoutException {
|
||||
_snack(context,
|
||||
'Server terlalu lama merespons invite. Coba Refresh status, jangan klik berkali-kali.');
|
||||
} catch (e) {
|
||||
_snack(context, 'Invite gagal: $e');
|
||||
} finally {
|
||||
if (mounted) setState(() => _loading = false);
|
||||
}
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return _Page(
|
||||
title: 'Pair User',
|
||||
subtitle: 'Masukkan 12 karakter Unique ID milik User.',
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||
children: [
|
||||
TextField(
|
||||
controller: _id,
|
||||
textCapitalization: TextCapitalization.characters,
|
||||
maxLength: 12,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Unique User ID',
|
||||
hintText: 'Contoh: AB1C2D3E4F5G',
|
||||
prefixIcon: Icon(Icons.link),
|
||||
)),
|
||||
FilledButton.icon(
|
||||
onPressed: _loading ? null : _invite,
|
||||
icon: _loading
|
||||
? const SizedBox(
|
||||
width: 18,
|
||||
height: 18,
|
||||
child: CircularProgressIndicator(strokeWidth: 2))
|
||||
: const Icon(Icons.link),
|
||||
label: Text(_loading ? 'Sending...' : 'Send Invite'),
|
||||
),
|
||||
const SizedBox(height: 20),
|
||||
_PairingStatusCard(key: ValueKey(_statusReload)),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// _PairingStatusCard (shared between both screens)
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
class _PairingStatusCard extends StatefulWidget {
|
||||
final bool allowUserResponse;
|
||||
const _PairingStatusCard({super.key, this.allowUserResponse = false});
|
||||
|
||||
@override
|
||||
State<_PairingStatusCard> createState() => _PairingStatusCardState();
|
||||
}
|
||||
|
||||
class _PairingStatusCardState extends State<_PairingStatusCard> {
|
||||
String _status = 'Mengecek status pairing...';
|
||||
bool _active = false;
|
||||
bool _loading = false;
|
||||
bool _responding = false;
|
||||
Map<String, dynamic>? _data;
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
_load();
|
||||
}
|
||||
|
||||
Future<void> _load() async {
|
||||
setState(() => _loading = true);
|
||||
try {
|
||||
final token = await sl<SecureStorage>().getAccessToken();
|
||||
if (token == null || token.isEmpty) {
|
||||
_active = false;
|
||||
_data = null;
|
||||
_status = 'Belum login. Login dulu supaya status pairing bisa dicek.';
|
||||
return;
|
||||
}
|
||||
final res = await sl<ApiClient>()
|
||||
.dio
|
||||
.get('/shared/pairing/status')
|
||||
.timeout(const Duration(seconds: 5));
|
||||
final data = res.data['data'];
|
||||
_data = data is Map ? Map<String, dynamic>.from(data) : null;
|
||||
_active = data is Map && data['status'] == 'ACTIVE';
|
||||
if (data is Map && data['status'] == 'ACTIVE') {
|
||||
_active = true;
|
||||
_status =
|
||||
'Sudah pairing dengan ${data['pairedWithName'] ?? data['pairedWithEmail'] ?? 'akun lain'}.';
|
||||
} else if (data is Map && data['status'] == 'PENDING') {
|
||||
_status = widget.allowUserResponse
|
||||
? 'Ada undangan pairing dari ${data['pairedWithName'] ?? data['pairedWithEmail'] ?? 'Guardian'}.'
|
||||
: 'Invite sudah terkirim. Tunggu User membuka menu Pairing lalu Accept.';
|
||||
} else {
|
||||
_status = 'Belum pairing. Bagikan Unique ID kamu ke Guardian.';
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
_active = false;
|
||||
_data = null;
|
||||
_status = _friendlyDioMessage(e,
|
||||
fallback:
|
||||
'Belum bisa mengecek server, tapi Unique ID tetap bisa dibagikan.');
|
||||
} on TimeoutException {
|
||||
_active = false;
|
||||
_data = null;
|
||||
_status =
|
||||
'Server terlalu lama merespons status pairing. Cek backend masih running dan URL server benar.';
|
||||
} catch (e) {
|
||||
_active = false;
|
||||
_data = null;
|
||||
_status = 'Status pairing belum bisa dicek: $e';
|
||||
} finally {
|
||||
if (mounted) setState(() => _loading = false);
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> _respond(bool accept) async {
|
||||
final pairingId = _data?['pairingId'];
|
||||
if (pairingId == null) {
|
||||
_snack(context, 'Tidak ada invite yang bisa direspons.');
|
||||
return;
|
||||
}
|
||||
setState(() => _responding = true);
|
||||
try {
|
||||
final res =
|
||||
await sl<ApiClient>().dio.post('/shared/pairing/respond', data: {
|
||||
'pairingId': pairingId,
|
||||
'accept': accept,
|
||||
}).timeout(const Duration(seconds: 8));
|
||||
_snack(
|
||||
context,
|
||||
res.data['message']?.toString() ??
|
||||
(accept ? 'Pairing diterima.' : 'Pairing ditolak.'));
|
||||
await _load();
|
||||
} on DioException catch (e) {
|
||||
_snack(context,
|
||||
_friendlyDioMessage(e, fallback: 'Gagal merespons pairing.'));
|
||||
} on TimeoutException {
|
||||
_snack(context, 'Server terlalu lama merespons pairing.');
|
||||
} finally {
|
||||
if (mounted) setState(() => _responding = false);
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> _unpair() async {
|
||||
final confirmed = await showDialog<bool>(
|
||||
context: context,
|
||||
builder: (ctx) => AlertDialog(
|
||||
title: const Text('Putus Pairing?'),
|
||||
content: const Text(
|
||||
'Semua konfigurasi voice command, shortcut, dan AI config akan dihapus.'),
|
||||
actions: [
|
||||
TextButton(
|
||||
onPressed: () => Navigator.of(ctx).pop(false),
|
||||
child: const Text('Batal')),
|
||||
FilledButton(
|
||||
style: FilledButton.styleFrom(
|
||||
backgroundColor: const Color(0xFFDC2626)),
|
||||
onPressed: () => Navigator.of(ctx).pop(true),
|
||||
child: const Text('Putus')),
|
||||
],
|
||||
),
|
||||
);
|
||||
if (confirmed != true) return;
|
||||
setState(() => _responding = true);
|
||||
try {
|
||||
await sl<ApiClient>()
|
||||
.dio
|
||||
.delete('/shared/pairing/unpair')
|
||||
.timeout(const Duration(seconds: 8));
|
||||
_snack(context, 'Pairing telah diputus.');
|
||||
await _load();
|
||||
} on DioException catch (e) {
|
||||
_snack(
|
||||
context, _friendlyDioMessage(e, fallback: 'Gagal memutus pairing.'));
|
||||
} finally {
|
||||
if (mounted) setState(() => _responding = false);
|
||||
}
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final pending = _data?['status'] == 'PENDING';
|
||||
return Container(
|
||||
padding: const EdgeInsets.all(16),
|
||||
decoration: BoxDecoration(
|
||||
color: _active ? const Color(0xFFF0FDF4) : const Color(0xFFFFFBEB),
|
||||
borderRadius: BorderRadius.circular(12),
|
||||
border: Border.all(
|
||||
color: _active ? const Color(0xFFBBF7D0) : const Color(0xFFFDE68A)),
|
||||
),
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||
children: [
|
||||
Row(
|
||||
children: [
|
||||
Icon(_active ? Icons.link : Icons.info_outline,
|
||||
color: _active
|
||||
? const Color(0xFF16A34A)
|
||||
: const Color(0xFFD97706)),
|
||||
const SizedBox(width: 12),
|
||||
Expanded(child: Text(_status)),
|
||||
IconButton(
|
||||
onPressed: _loading ? null : _load,
|
||||
icon: _loading
|
||||
? const SizedBox(
|
||||
width: 18,
|
||||
height: 18,
|
||||
child: CircularProgressIndicator(strokeWidth: 2))
|
||||
: const Icon(Icons.refresh)),
|
||||
],
|
||||
),
|
||||
if (widget.allowUserResponse && pending) ...[
|
||||
const SizedBox(height: 12),
|
||||
Row(
|
||||
children: [
|
||||
Expanded(
|
||||
child: FilledButton.icon(
|
||||
onPressed: _responding ? null : () => _respond(true),
|
||||
icon: const Icon(Icons.check),
|
||||
label: const Text('Accept'),
|
||||
),
|
||||
),
|
||||
const SizedBox(width: 10),
|
||||
Expanded(
|
||||
child: OutlinedButton.icon(
|
||||
onPressed: _responding ? null : () => _respond(false),
|
||||
icon: const Icon(Icons.close),
|
||||
label: const Text('Reject'),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
],
|
||||
if (_active) ...[
|
||||
const SizedBox(height: 12),
|
||||
OutlinedButton.icon(
|
||||
style: OutlinedButton.styleFrom(
|
||||
foregroundColor: const Color(0xFFDC2626),
|
||||
side: const BorderSide(color: Color(0xFFDC2626))),
|
||||
onPressed: _responding ? null : _unpair,
|
||||
icon: const Icon(Icons.link_off),
|
||||
label: const Text('Putus Pairing'),
|
||||
),
|
||||
],
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Shared private widgets
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
class _Page extends StatelessWidget {
|
||||
final String title;
|
||||
final String? subtitle;
|
||||
final Widget child;
|
||||
final List<Widget>? actions;
|
||||
|
||||
const _Page(
|
||||
{required this.title, required this.child, this.subtitle, this.actions});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return SafeArea(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(16),
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Row(
|
||||
children: [
|
||||
Expanded(
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(title,
|
||||
style: Theme.of(context)
|
||||
.textTheme
|
||||
.headlineSmall
|
||||
?.copyWith(fontWeight: FontWeight.w800)),
|
||||
if (subtitle != null)
|
||||
Text(subtitle!,
|
||||
style: const TextStyle(color: Color(0xFF64748B))),
|
||||
],
|
||||
),
|
||||
),
|
||||
...?actions,
|
||||
],
|
||||
),
|
||||
const SizedBox(height: 16),
|
||||
Expanded(child: child),
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class _InfoCard extends StatelessWidget {
|
||||
final String title;
|
||||
final String value;
|
||||
final IconData icon;
|
||||
const _InfoCard(
|
||||
{required this.title, required this.value, required this.icon});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Container(
|
||||
padding: const EdgeInsets.all(16),
|
||||
decoration: BoxDecoration(
|
||||
color: const Color(0xFFEFF6FF),
|
||||
borderRadius: BorderRadius.circular(12)),
|
||||
child: Row(
|
||||
children: [
|
||||
Icon(icon, color: const Color(0xFF1A56DB)),
|
||||
const SizedBox(width: 12),
|
||||
Expanded(
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(title),
|
||||
SelectableText(value,
|
||||
style: const TextStyle(
|
||||
fontSize: 22, fontWeight: FontWeight.w800))
|
||||
])),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Helpers
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
void _snack(BuildContext context, String message) {
|
||||
if (context.mounted) {
|
||||
ScaffoldMessenger.of(context)
|
||||
.showSnackBar(SnackBar(content: Text(message)));
|
||||
}
|
||||
}
|
||||
|
||||
String _friendlyDioMessage(DioException e, {required String fallback}) {
|
||||
final data = e.response?.data;
|
||||
if (data is Map && data['message'] != null) return data['message'].toString();
|
||||
if (e.response?.statusCode == 401) {
|
||||
return 'Sesi login habis. Logout lalu login ulang.';
|
||||
}
|
||||
if (e.response?.statusCode == 403) {
|
||||
return 'Role akun tidak cocok untuk fitur ini. Pastikan User/Guardian benar.';
|
||||
}
|
||||
if (e.type == DioExceptionType.connectionTimeout ||
|
||||
e.type == DioExceptionType.receiveTimeout) {
|
||||
return 'Server terlalu lama merespons. Pastikan backend masih running dan URL server benar.';
|
||||
}
|
||||
if (e.type == DioExceptionType.connectionError) {
|
||||
return 'Tidak bisa ke server. Di Chrome pakai http://localhost:8080. Di HP pakai IP laptop/server, bukan localhost.';
|
||||
}
|
||||
return fallback;
|
||||
}
|
||||
|
||||
@ -1 +1,205 @@
|
||||
export '../screens.dart';
|
||||
// ignore_for_file: use_build_context_synchronously
|
||||
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:go_router/go_router.dart';
|
||||
|
||||
import '../../app/injection_container.dart';
|
||||
import '../../core/constants/app_constants.dart';
|
||||
import '../../core/network/api_client.dart';
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// ServerConnectScreen
|
||||
// ---------------------------------------------------------------------------
|
||||
//
|
||||
// Gerbang pertama aplikasi.
|
||||
// Muncul HANYA jika SharedPreferences tidak punya serverUrl tersimpan.
|
||||
// Setelah berhasil connect, tidak akan muncul lagi kecuali user reset via
|
||||
// Settings → "Change Server".
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
class ServerConnectScreen extends StatefulWidget {
|
||||
const ServerConnectScreen({super.key});
|
||||
|
||||
@override
|
||||
State<ServerConnectScreen> createState() => _ServerConnectScreenState();
|
||||
}
|
||||
|
||||
class _ServerConnectScreenState extends State<ServerConnectScreen> {
|
||||
final _url = TextEditingController(text: 'http://202.46.28.160:8080');
|
||||
bool _loading = false;
|
||||
bool _ok = false;
|
||||
String? _message;
|
||||
|
||||
Future<void> _test() async {
|
||||
setState(() {
|
||||
_loading = true;
|
||||
_ok = false;
|
||||
_message = null;
|
||||
});
|
||||
try {
|
||||
final clean = AppConstants.normalizeServerUrl(_url.text);
|
||||
final res = await Dio(BaseOptions(
|
||||
connectTimeout: AppConstants.pingTimeout,
|
||||
receiveTimeout: AppConstants.pingTimeout,
|
||||
)).get('$clean/api/v1/auth/ping');
|
||||
_ok = res.statusCode == 200 && res.data['success'] == true;
|
||||
_message = _ok
|
||||
? 'Server aktif dan siap dipakai.'
|
||||
: 'Server merespons dengan format tidak valid.';
|
||||
} catch (e) {
|
||||
_message = 'Tidak bisa terhubung. Periksa URL dan jaringan.';
|
||||
} finally {
|
||||
if (mounted) setState(() => _loading = false);
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> _continue() async {
|
||||
final clean = AppConstants.normalizeServerUrl(_url.text);
|
||||
await AppConstants.setServerUrl(clean);
|
||||
await sl<ApiClient>().init(clean);
|
||||
if (mounted) context.go('/splash');
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return _AuthFrame(
|
||||
title: 'Connect to Server',
|
||||
subtitle: 'Masukkan URL backend WalkGuide yang diberikan dosen.',
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||
children: [
|
||||
TextField(
|
||||
controller: _url,
|
||||
keyboardType: TextInputType.url,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Server URL',
|
||||
hintText: 'http://202.46.28.160:8080',
|
||||
prefixIcon: Icon(Icons.dns_outlined),
|
||||
)),
|
||||
const SizedBox(height: 12),
|
||||
OutlinedButton.icon(
|
||||
onPressed: _loading ? null : _test,
|
||||
icon: _loading
|
||||
? const SizedBox(
|
||||
width: 18,
|
||||
height: 18,
|
||||
child: CircularProgressIndicator(strokeWidth: 2))
|
||||
: const Icon(Icons.wifi_tethering),
|
||||
label: const Text('Test Connection'),
|
||||
),
|
||||
if (_message != null) ...[
|
||||
const SizedBox(height: 12),
|
||||
_StatusBox(success: _ok, message: _message!),
|
||||
],
|
||||
if (_ok) ...[
|
||||
const SizedBox(height: 12),
|
||||
FilledButton.icon(
|
||||
onPressed: _continue,
|
||||
icon: const Icon(Icons.arrow_forward),
|
||||
label: const Text('Continue')),
|
||||
],
|
||||
const SizedBox(height: 24),
|
||||
const Center(
|
||||
child: Text(
|
||||
'v1.0.0 | For Testing Purposes Only',
|
||||
style: TextStyle(fontSize: 11, color: Color(0xFF94A3B8)),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Shared private widgets
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
class _AuthFrame extends StatelessWidget {
|
||||
final String title;
|
||||
final String subtitle;
|
||||
final Widget child;
|
||||
const _AuthFrame(
|
||||
{required this.title, required this.subtitle, required this.child});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
body: Center(
|
||||
child: SingleChildScrollView(
|
||||
padding: const EdgeInsets.all(24),
|
||||
child: ConstrainedBox(
|
||||
constraints: const BoxConstraints(maxWidth: 460),
|
||||
child: Card(
|
||||
elevation: 0,
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(18),
|
||||
side: const BorderSide(color: Color(0xFFE2E8F0))),
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(24),
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||
children: [
|
||||
const Icon(Icons.navigation_rounded,
|
||||
color: Color(0xFF1A56DB), size: 42),
|
||||
const SizedBox(height: 14),
|
||||
Text(title,
|
||||
textAlign: TextAlign.center,
|
||||
style: Theme.of(context)
|
||||
.textTheme
|
||||
.headlineSmall
|
||||
?.copyWith(fontWeight: FontWeight.w800)),
|
||||
const SizedBox(height: 4),
|
||||
Text(subtitle,
|
||||
textAlign: TextAlign.center,
|
||||
style: const TextStyle(color: Color(0xFF64748B))),
|
||||
const SizedBox(height: 22),
|
||||
child,
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class _StatusBox extends StatelessWidget {
|
||||
final bool success;
|
||||
final String message;
|
||||
const _StatusBox({required this.success, required this.message});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return DecoratedBox(
|
||||
decoration: BoxDecoration(
|
||||
color: success ? const Color(0xFFF0FDF4) : const Color(0xFFFEF2F2),
|
||||
borderRadius: BorderRadius.circular(10),
|
||||
),
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(12),
|
||||
child: Row(
|
||||
children: [
|
||||
Icon(
|
||||
success ? Icons.check_circle_outline : Icons.error_outline,
|
||||
color:
|
||||
success ? const Color(0xFF166534) : const Color(0xFF991B1B),
|
||||
size: 20,
|
||||
),
|
||||
const SizedBox(width: 8),
|
||||
Expanded(
|
||||
child: Text(message,
|
||||
style: TextStyle(
|
||||
color: success
|
||||
? const Color(0xFF166534)
|
||||
: const Color(0xFF991B1B))),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
@ -1 +1,572 @@
|
||||
export '../screens.dart';
|
||||
// ignore_for_file: use_build_context_synchronously, deprecated_member_use
|
||||
|
||||
import 'dart:async';
|
||||
import 'dart:convert';
|
||||
|
||||
import 'package:camera/camera.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter/services.dart';
|
||||
import 'package:go_router/go_router.dart';
|
||||
import 'package:shared_preferences/shared_preferences.dart';
|
||||
|
||||
import '../../app/injection_container.dart';
|
||||
import '../../core/ai/detection_export.dart';
|
||||
import '../../core/constants/app_constants.dart';
|
||||
import '../../core/network/api_client.dart';
|
||||
import '../../core/services/haptic_service.dart';
|
||||
import '../../core/services/location_reporter_service.dart';
|
||||
import '../../core/services/tts_service.dart';
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// WalkGuideScreen
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
class WalkGuideScreen extends StatefulWidget {
|
||||
const WalkGuideScreen({super.key});
|
||||
|
||||
@override
|
||||
State<WalkGuideScreen> createState() => _WalkGuideScreenState();
|
||||
}
|
||||
|
||||
class _WalkGuideScreenState extends State<WalkGuideScreen> {
|
||||
bool _active = false;
|
||||
String _status = 'Ready';
|
||||
CameraController? _camera;
|
||||
DetectionResult? _lastDetection;
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
_camera?.dispose();
|
||||
sl<LocationReporterService>().stop();
|
||||
super.dispose();
|
||||
}
|
||||
|
||||
Future<void> _toggle() async {
|
||||
final next = !_active;
|
||||
if (next) {
|
||||
await _startCamera();
|
||||
await sl<LocationReporterService>().start(walkGuideActive: true);
|
||||
} else {
|
||||
await _camera?.dispose();
|
||||
_camera = null;
|
||||
await sl<LocationReporterService>().stop();
|
||||
}
|
||||
setState(() {
|
||||
_active = next;
|
||||
_status = next ? 'Camera stream active. YOLO ready.' : 'Stopped';
|
||||
});
|
||||
await sl<ApiClient>()
|
||||
.dio
|
||||
.post(next ? '/user/walkguide/start' : '/user/walkguide/stop');
|
||||
sl<TtsService>().speak(next ? 'WalkGuide dimulai' : 'WalkGuide dihentikan');
|
||||
}
|
||||
|
||||
Future<void> _startCamera() async {
|
||||
if (_camera != null) return;
|
||||
try {
|
||||
final cameras = await availableCameras();
|
||||
if (cameras.isEmpty) return;
|
||||
final controller = CameraController(
|
||||
cameras.first, ResolutionPreset.medium,
|
||||
enableAudio: false);
|
||||
await controller.initialize();
|
||||
if (!mounted) {
|
||||
await controller.dispose();
|
||||
return;
|
||||
}
|
||||
setState(() => _camera = controller);
|
||||
} catch (_) {
|
||||
setState(() => _status = 'Camera unavailable. Demo mode active.');
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> _simulateObstacle() async {
|
||||
final detection = await sl<YoloDetector>().detectFallback();
|
||||
if (detection == null) return;
|
||||
_lastDetection = detection;
|
||||
await sl<ApiClient>().dio.post('/user/obstacle', data: {
|
||||
'label': detection.label,
|
||||
'confidence': detection.confidence,
|
||||
'direction': detection.directionName,
|
||||
'estimatedDist': detection.estimatedDistance,
|
||||
'lat': null,
|
||||
'lng': null,
|
||||
});
|
||||
await sl<HapticService>().obstacleClose();
|
||||
await sl<TtsService>().speakImmediate(detection.spokenId);
|
||||
setState(() => _status =
|
||||
'Obstacle: ${detection.label} ${detection.directionName} ${detection.estimatedDistance}');
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return _Page(
|
||||
title: 'WalkGuide',
|
||||
subtitle: 'On-device AI detection surface',
|
||||
actions: [
|
||||
IconButton(
|
||||
onPressed: () => context.go('/user/benchmark'),
|
||||
icon: const Icon(Icons.speed)),
|
||||
IconButton(
|
||||
onPressed: () => context.go('/user/pairing'),
|
||||
icon: const Icon(Icons.link)),
|
||||
],
|
||||
child: Column(
|
||||
children: [
|
||||
Expanded(
|
||||
child: Container(
|
||||
width: double.infinity,
|
||||
decoration: BoxDecoration(
|
||||
color: const Color(0xFF0F172A),
|
||||
borderRadius: BorderRadius.circular(16)),
|
||||
child: Stack(
|
||||
children: [
|
||||
if (_camera != null && _camera!.value.isInitialized)
|
||||
Positioned.fill(child: CameraPreview(_camera!))
|
||||
else
|
||||
const Center(
|
||||
child: Icon(Icons.videocam_outlined,
|
||||
color: Colors.white30, size: 96)),
|
||||
Positioned(
|
||||
top: 16,
|
||||
left: 16,
|
||||
child: _Pill(
|
||||
text: _active ? 'AI ACTIVE' : 'STANDBY',
|
||||
color: _active ? Colors.green : Colors.orange)),
|
||||
if (_lastDetection != null)
|
||||
Positioned(
|
||||
top: 64,
|
||||
left: 16,
|
||||
child: _Pill(
|
||||
text:
|
||||
'${_lastDetection!.label} ${_lastDetection!.directionName}',
|
||||
color: Colors.redAccent),
|
||||
),
|
||||
Positioned(
|
||||
left: 16,
|
||||
right: 16,
|
||||
bottom: 16,
|
||||
child: Text(_status,
|
||||
style: const TextStyle(
|
||||
color: Colors.white,
|
||||
fontSize: 18,
|
||||
fontWeight: FontWeight.w700))),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
const SizedBox(height: 14),
|
||||
Row(
|
||||
children: [
|
||||
Expanded(
|
||||
child: FilledButton.icon(
|
||||
onPressed: _toggle,
|
||||
icon: Icon(_active ? Icons.stop : Icons.play_arrow),
|
||||
label: Text(_active ? 'Stop' : 'Start'))),
|
||||
const SizedBox(width: 10),
|
||||
Expanded(
|
||||
child: OutlinedButton.icon(
|
||||
onPressed: _simulateObstacle,
|
||||
icon: const Icon(Icons.radar),
|
||||
label: const Text('Demo Detect'))),
|
||||
],
|
||||
),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// AiBenchmarkScreen
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
class AiBenchmarkScreen extends StatefulWidget {
|
||||
const AiBenchmarkScreen({super.key});
|
||||
|
||||
@override
|
||||
State<AiBenchmarkScreen> createState() => _AiBenchmarkScreenState();
|
||||
}
|
||||
|
||||
class _AiBenchmarkScreenState extends State<AiBenchmarkScreen> {
|
||||
static const _runsKey = 'ai_benchmark_runs';
|
||||
List<String> _models = const [];
|
||||
String _selectedModel = AppConstants.yoloModelPath;
|
||||
List<Map<String, dynamic>> _runs = const [];
|
||||
bool _running = false;
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
_load();
|
||||
}
|
||||
|
||||
Future<void> _load() async {
|
||||
final models = await _discoverTfliteModels();
|
||||
final selected = await AppConstants.getSelectedYoloModelPath();
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
final rawRuns = prefs.getStringList(_runsKey) ?? const [];
|
||||
setState(() {
|
||||
_models = models.isEmpty ? [selected] : models;
|
||||
_selectedModel = models.contains(selected) ? selected : _models.first;
|
||||
_runs = rawRuns
|
||||
.map((e) => Map<String, dynamic>.from(jsonDecode(e) as Map))
|
||||
.toList()
|
||||
.reversed
|
||||
.toList();
|
||||
});
|
||||
}
|
||||
|
||||
Future<void> _setModel(String? value) async {
|
||||
if (value == null) return;
|
||||
await AppConstants.setSelectedYoloModelPath(value);
|
||||
sl<YoloDetector>().dispose();
|
||||
await sl<YoloDetector>().init();
|
||||
setState(() => _selectedModel = value);
|
||||
_snack(context, 'Model aktif: ${value.split('/').last}');
|
||||
}
|
||||
|
||||
Future<void> _runBenchmark() async {
|
||||
setState(() => _running = true);
|
||||
final started = DateTime.now();
|
||||
final captureMs = await _measureCapture();
|
||||
|
||||
final inferenceWatch = Stopwatch()..start();
|
||||
String label = 'person';
|
||||
String direction = 'CENTER';
|
||||
String distance = 'Demo';
|
||||
var modelLoaded = false;
|
||||
try {
|
||||
await rootBundle.load(_selectedModel).timeout(const Duration(seconds: 3));
|
||||
modelLoaded = true;
|
||||
} catch (_) {}
|
||||
final detection = await sl<YoloDetector>().detectFallback();
|
||||
if (detection != null) {
|
||||
label = detection.label;
|
||||
direction = detection.directionName;
|
||||
distance = detection.estimatedDistance;
|
||||
}
|
||||
inferenceWatch.stop();
|
||||
|
||||
final notifWatch = Stopwatch()..start();
|
||||
final text = 'Obstacle $label di $direction, jarak $distance';
|
||||
notifWatch.stop();
|
||||
|
||||
final ttsWatch = Stopwatch()..start();
|
||||
try {
|
||||
await sl<TtsService>()
|
||||
.speakImmediate(text)
|
||||
.timeout(const Duration(seconds: 3));
|
||||
} catch (_) {}
|
||||
ttsWatch.stop();
|
||||
|
||||
final run = {
|
||||
'time': started.toIso8601String(),
|
||||
'model': _selectedModel,
|
||||
'modelLoaded': modelLoaded,
|
||||
'captureMs': captureMs,
|
||||
'inferenceMs': inferenceWatch.elapsedMilliseconds,
|
||||
'notificationMs': notifWatch.elapsedMicroseconds / 1000,
|
||||
'ttsMs': ttsWatch.elapsedMilliseconds,
|
||||
'label': label,
|
||||
'direction': direction,
|
||||
};
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
final next = [
|
||||
jsonEncode(run),
|
||||
...((prefs.getStringList(_runsKey) ?? const []).take(24))
|
||||
];
|
||||
await prefs.setStringList(_runsKey, next);
|
||||
if (mounted) {
|
||||
setState(() {
|
||||
_runs = [run, ..._runs].take(25).toList();
|
||||
_running = false;
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
Future<int> _measureCapture() async {
|
||||
final watch = Stopwatch()..start();
|
||||
CameraController? controller;
|
||||
try {
|
||||
final cameras =
|
||||
await availableCameras().timeout(const Duration(seconds: 3));
|
||||
if (cameras.isNotEmpty) {
|
||||
controller = CameraController(cameras.first, ResolutionPreset.low,
|
||||
enableAudio: false);
|
||||
await controller.initialize().timeout(const Duration(seconds: 5));
|
||||
await controller.takePicture().timeout(const Duration(seconds: 5));
|
||||
}
|
||||
} catch (_) {
|
||||
await Future<void>.delayed(const Duration(milliseconds: 16));
|
||||
} finally {
|
||||
await controller?.dispose();
|
||||
}
|
||||
watch.stop();
|
||||
return watch.elapsedMilliseconds;
|
||||
}
|
||||
|
||||
Future<void> _clearRuns() async {
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
await prefs.remove(_runsKey);
|
||||
setState(() => _runs = const []);
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final hasRealModel = _models.any((model) => model.endsWith('.tflite'));
|
||||
return _Page(
|
||||
title: 'AI Benchmark',
|
||||
subtitle: 'Capture, model, notification text, and TTS timing',
|
||||
child: ListView(
|
||||
children: [
|
||||
DropdownButtonFormField<String>(
|
||||
value: _selectedModel,
|
||||
decoration: const InputDecoration(labelText: 'Model file'),
|
||||
items: [
|
||||
for (final model in _models)
|
||||
DropdownMenuItem(
|
||||
value: model, child: Text(model.split('/').last))
|
||||
],
|
||||
onChanged: _setModel,
|
||||
),
|
||||
if (!hasRealModel) ...[
|
||||
const SizedBox(height: 10),
|
||||
const _StatusBox(
|
||||
success: false,
|
||||
message:
|
||||
'Belum ada file .tflite di assets/models. Taruh 3-5 model di folder itu, lalu restart app untuk muncul di dropdown.',
|
||||
),
|
||||
],
|
||||
const SizedBox(height: 12),
|
||||
FilledButton.icon(
|
||||
onPressed: _running ? null : _runBenchmark,
|
||||
icon: _running
|
||||
? const SizedBox(
|
||||
width: 18,
|
||||
height: 18,
|
||||
child: CircularProgressIndicator(strokeWidth: 2))
|
||||
: const Icon(Icons.play_arrow),
|
||||
label: Text(_running ? 'Running benchmark...' : 'Run benchmark'),
|
||||
),
|
||||
const SizedBox(height: 8),
|
||||
OutlinedButton.icon(
|
||||
onPressed: _clearRuns,
|
||||
icon: const Icon(Icons.delete_outline),
|
||||
label: const Text('Clear log')),
|
||||
const SizedBox(height: 16),
|
||||
for (final run in _runs) _BenchmarkCard(run: run),
|
||||
if (_runs.isEmpty)
|
||||
const _EmptyPanel(
|
||||
icon: Icons.speed,
|
||||
title: 'Belum Ada Log',
|
||||
message:
|
||||
'Klik Run benchmark untuk mencatat waktu capture, model/inference, text notification, dan TTS.',
|
||||
),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class _BenchmarkCard extends StatelessWidget {
|
||||
final Map<String, dynamic> run;
|
||||
const _BenchmarkCard({required this.run});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final time = DateTime.tryParse(run['time']?.toString() ?? '')?.toLocal();
|
||||
return Card(
|
||||
elevation: 0,
|
||||
margin: const EdgeInsets.only(bottom: 10),
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(12),
|
||||
side: const BorderSide(color: Color(0xFFE2E8F0))),
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(12),
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(
|
||||
time == null
|
||||
? 'Benchmark run'
|
||||
: '${_two(time.hour)}:${_two(time.minute)}:${_two(time.second)}',
|
||||
style: const TextStyle(fontWeight: FontWeight.w800)),
|
||||
const SizedBox(height: 8),
|
||||
Text(
|
||||
'Model: ${run['model'].toString().split('/').last} (${run['modelLoaded'] == true ? 'loaded' : 'fallback'})'),
|
||||
Text('Capture: ${run['captureMs']} ms'),
|
||||
Text('Model/inference: ${run['inferenceMs']} ms'),
|
||||
Text('Notification text: ${run['notificationMs']} ms'),
|
||||
Text('TTS start: ${run['ttsMs']} ms'),
|
||||
Text('Result: ${run['label']} ${run['direction']}'),
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Shared private widgets (scoped to this file)
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
class _Page extends StatelessWidget {
|
||||
final String title;
|
||||
final String? subtitle;
|
||||
final Widget child;
|
||||
final List<Widget>? actions;
|
||||
|
||||
const _Page(
|
||||
{required this.title, required this.child, this.subtitle, this.actions});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return SafeArea(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(16),
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Row(
|
||||
children: [
|
||||
Expanded(
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(title,
|
||||
style: Theme.of(context)
|
||||
.textTheme
|
||||
.headlineSmall
|
||||
?.copyWith(fontWeight: FontWeight.w800)),
|
||||
if (subtitle != null)
|
||||
Text(subtitle!,
|
||||
style: const TextStyle(color: Color(0xFF64748B))),
|
||||
],
|
||||
),
|
||||
),
|
||||
...?actions,
|
||||
],
|
||||
),
|
||||
const SizedBox(height: 16),
|
||||
Expanded(child: child),
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class _Pill extends StatelessWidget {
|
||||
final String text;
|
||||
final Color color;
|
||||
const _Pill({required this.text, required this.color});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return DecoratedBox(
|
||||
decoration: BoxDecoration(
|
||||
color: color.withOpacity(0.16),
|
||||
borderRadius: BorderRadius.circular(999),
|
||||
border: Border.all(color: color)),
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 7),
|
||||
child: Text(text,
|
||||
style: TextStyle(color: color, fontWeight: FontWeight.w800)),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class _StatusBox extends StatelessWidget {
|
||||
final bool success;
|
||||
final String message;
|
||||
const _StatusBox({required this.success, required this.message});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return DecoratedBox(
|
||||
decoration: BoxDecoration(
|
||||
color: success ? const Color(0xFFF0FDF4) : const Color(0xFFFEF2F2),
|
||||
borderRadius: BorderRadius.circular(10),
|
||||
),
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(12),
|
||||
child: Text(message,
|
||||
style: TextStyle(
|
||||
color: success
|
||||
? const Color(0xFF166534)
|
||||
: const Color(0xFF991B1B))),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class _EmptyPanel extends StatelessWidget {
|
||||
final IconData icon;
|
||||
final String title;
|
||||
final String message;
|
||||
final Widget? action;
|
||||
const _EmptyPanel(
|
||||
{required this.icon,
|
||||
required this.title,
|
||||
required this.message,
|
||||
this.action});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Container(
|
||||
width: double.infinity,
|
||||
constraints: const BoxConstraints(minHeight: 180),
|
||||
padding: const EdgeInsets.all(18),
|
||||
decoration: BoxDecoration(
|
||||
color: const Color(0xFFF8FAFC),
|
||||
borderRadius: BorderRadius.circular(12),
|
||||
border: Border.all(color: const Color(0xFFE2E8F0))),
|
||||
child: Column(
|
||||
mainAxisAlignment: MainAxisAlignment.center,
|
||||
children: [
|
||||
Icon(icon, color: const Color(0xFF64748B), size: 48),
|
||||
const SizedBox(height: 12),
|
||||
Text(title,
|
||||
style:
|
||||
const TextStyle(fontWeight: FontWeight.w800, fontSize: 18)),
|
||||
const SizedBox(height: 6),
|
||||
Text(message, textAlign: TextAlign.center),
|
||||
if (action != null) ...[
|
||||
const SizedBox(height: 12),
|
||||
action!,
|
||||
],
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Helpers
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
Future<List<String>> _discoverTfliteModels() async {
|
||||
try {
|
||||
final manifestRaw = await rootBundle.loadString('AssetManifest.json');
|
||||
final manifest = jsonDecode(manifestRaw) as Map<String, dynamic>;
|
||||
final models = manifest.keys
|
||||
.where((key) =>
|
||||
key.startsWith('assets/models/') && key.endsWith('.tflite'))
|
||||
.toList()
|
||||
..sort();
|
||||
return models;
|
||||
} catch (_) {
|
||||
return const [];
|
||||
}
|
||||
}
|
||||
|
||||
String _two(int value) => value.toString().padLeft(2, '0');
|
||||
|
||||
void _snack(BuildContext context, String message) {
|
||||
if (context.mounted) {
|
||||
ScaffoldMessenger.of(context)
|
||||
.showSnackBar(SnackBar(content: Text(message)));
|
||||
}
|
||||
}
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user