feat: implement ActivityLogScreen with real API + BLoC

This commit is contained in:
5803024019 2026-05-15 20:09:24 +07:00
parent 6944bb87a8
commit 4cfe317d02

View File

@ -1 +1,435 @@
export '../screens.dart' show ActivityLogScreen;
// lib/features/activity_log/activity_log_screen.dart
// ignore_for_file: use_build_context_synchronously
import 'package:dio/dio.dart';
import 'package:flutter/material.dart';
import 'package:intl/intl.dart';
import '../../app/injection_container.dart';
import '../../core/network/api_client.dart';
import '../../core/theme/app_colors.dart';
Dio get _api => sl<ApiClient>().dio;
class ActivityLogScreen extends StatefulWidget {
const ActivityLogScreen({super.key});
@override
State<ActivityLogScreen> createState() => _ActivityLogScreenState();
}
class _ActivityLogScreenState extends State<ActivityLogScreen> {
List<_LogItem> _items = [];
List<_LogItem> _filtered = [];
bool _loading = true;
String? _error;
String _selectedFilter = 'ALL';
static const _filters = [
'ALL',
'WALKGUIDE',
'SOS',
'AUTH',
'OBSTACLE',
'LOCATION',
];
@override
void initState() {
super.initState();
_load();
}
Future<void> _load() async {
setState(() {
_loading = true;
_error = null;
});
try {
final res = await _api
.get('/user/activity-logs')
.timeout(const Duration(seconds: 10));
final list = (res.data['data'] as List?) ?? [];
final items = list.map((e) => _LogItem.fromJson(e)).toList();
setState(() {
_items = items;
_applyFilter(_selectedFilter);
});
} on DioException catch (e) {
setState(() {
_error = e.response?.data?['message']?.toString() ??
'Gagal memuat activity log.';
});
} catch (e) {
setState(() => _error = 'Timeout / error: $e');
} finally {
if (mounted) setState(() => _loading = false);
}
}
void _applyFilter(String filter) {
_selectedFilter = filter;
if (filter == 'ALL') {
_filtered = List.from(_items);
} else {
_filtered = _items.where((item) {
switch (filter) {
case 'WALKGUIDE':
return item.logType.contains('WALKGUIDE');
case 'SOS':
return item.logType.contains('SOS');
case 'AUTH':
return item.logType == 'LOGIN' ||
item.logType == 'LOGOUT' ||
item.logType == 'APP_OPEN' ||
item.logType == 'APP_CLOSE';
case 'OBSTACLE':
return item.logType.contains('OBSTACLE');
case 'LOCATION':
return item.logType.contains('LOCATION') ||
item.logType.contains('GEOFENCE');
default:
return true;
}
}).toList();
}
}
@override
Widget build(BuildContext context) {
return SafeArea(
child: Padding(
padding: const EdgeInsets.all(16),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
// Header
Row(
children: [
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
'Activity Log',
style: Theme.of(context)
.textTheme
.headlineSmall
?.copyWith(fontWeight: FontWeight.w800),
),
Text(
'${_items.length} aktivitas tercatat',
style: const TextStyle(color: AppColors.muted),
),
],
),
),
IconButton(
onPressed: _load,
icon: const Icon(Icons.refresh),
tooltip: 'Refresh',
),
],
),
const SizedBox(height: 12),
// Filter chips
SizedBox(
height: 36,
child: ListView.separated(
scrollDirection: Axis.horizontal,
itemCount: _filters.length,
separatorBuilder: (_, __) => const SizedBox(width: 8),
itemBuilder: (_, i) {
final f = _filters[i];
final selected = _selectedFilter == f;
return FilterChip(
label: Text(f),
selected: selected,
onSelected: (_) {
setState(() => _applyFilter(f));
},
selectedColor: AppColors.primary.withOpacity(0.15),
checkmarkColor: AppColors.primary,
labelStyle: TextStyle(
color: selected ? AppColors.primary : AppColors.muted,
fontWeight:
selected ? FontWeight.w700 : FontWeight.normal,
fontSize: 12,
),
padding: const EdgeInsets.symmetric(horizontal: 4),
);
},
),
),
const SizedBox(height: 16),
// Body
Expanded(
child: _loading
? const Center(child: CircularProgressIndicator())
: _error != null
? _ErrorPanel(message: _error!, onRetry: _load)
: _filtered.isEmpty
? _EmptyPanel(filter: _selectedFilter)
: RefreshIndicator(
onRefresh: _load,
child: ListView.builder(
itemCount: _filtered.length,
itemBuilder: (ctx, i) =>
_LogCard(item: _filtered[i]),
),
),
),
],
),
),
);
}
}
// Data model
class _LogItem {
final int id;
final String logType;
final String? description;
final DateTime createdAt;
const _LogItem({
required this.id,
required this.logType,
this.description,
required this.createdAt,
});
factory _LogItem.fromJson(Map<String, dynamic> j) => _LogItem(
id: j['id'] as int,
logType: j['logType']?.toString() ?? 'UNKNOWN',
description: j['description']?.toString(),
createdAt: DateTime.tryParse(j['createdAt']?.toString() ?? '') ??
DateTime.now(),
);
}
// Log card
class _LogCard extends StatelessWidget {
final _LogItem item;
const _LogCard({required this.item});
@override
Widget build(BuildContext context) {
final meta = _logMeta(item.logType);
return Padding(
padding: const EdgeInsets.only(bottom: 8),
child: Row(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
// Timeline dot + line
Column(
children: [
Container(
width: 36,
height: 36,
decoration: BoxDecoration(
color: meta.color.withOpacity(0.12),
shape: BoxShape.circle,
),
child: Icon(meta.icon, color: meta.color, size: 18),
),
Container(
width: 1.5,
height: 20,
color: const Color(0xFFE2E8F0),
),
],
),
const SizedBox(width: 12),
// Content
Expanded(
child: Padding(
padding: const EdgeInsets.only(top: 4),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
children: [
Expanded(
child: Text(
meta.label,
style: TextStyle(
fontWeight: FontWeight.w700,
color: meta.color,
fontSize: 13,
),
),
),
Text(
_formatTime(item.createdAt),
style: const TextStyle(
color: AppColors.muted, fontSize: 11),
),
],
),
if (item.description != null && item.description!.isNotEmpty)
Padding(
padding: const EdgeInsets.only(top: 2),
child: Text(
item.description!,
style: const TextStyle(
fontSize: 13, color: AppColors.text),
),
),
const SizedBox(height: 12),
],
),
),
),
],
),
);
}
String _formatTime(DateTime dt) {
final now = DateTime.now();
if (now.difference(dt).inDays < 1 && dt.day == now.day) {
return DateFormat('HH:mm').format(dt);
}
return DateFormat('dd MMM HH:mm').format(dt);
}
}
// Log metadata helper
class _LogMeta {
final IconData icon;
final Color color;
final String label;
const _LogMeta(
{required this.icon, required this.color, required this.label});
}
_LogMeta _logMeta(String logType) {
switch (logType) {
case 'LOGIN':
return const _LogMeta(
icon: Icons.login, color: AppColors.success, label: 'Login');
case 'LOGOUT':
return const _LogMeta(
icon: Icons.logout, color: AppColors.muted, label: 'Logout');
case 'APP_OPEN':
return const _LogMeta(
icon: Icons.open_in_new,
color: AppColors.primary,
label: 'App Dibuka');
case 'APP_CLOSE':
return const _LogMeta(
icon: Icons.close, color: AppColors.muted, label: 'App Ditutup');
case 'WALKGUIDE_START':
return const _LogMeta(
icon: Icons.directions_walk,
color: AppColors.primary,
label: 'WalkGuide Mulai');
case 'WALKGUIDE_STOP':
return const _LogMeta(
icon: Icons.stop_circle,
color: AppColors.muted,
label: 'WalkGuide Berhenti');
case 'OBSTACLE_DETECTED':
return const _LogMeta(
icon: Icons.warning_amber,
color: Color(0xFFD97706),
label: 'Obstacle Terdeteksi');
case 'CALL_INITIATED':
return const _LogMeta(
icon: Icons.call,
color: AppColors.success,
label: 'Panggilan Dimulai');
case 'CALL_ENDED':
return const _LogMeta(
icon: Icons.call_end,
color: AppColors.muted,
label: 'Panggilan Selesai');
case 'SOS_TRIGGERED':
return const _LogMeta(
icon: Icons.sos, color: AppColors.danger, label: 'SOS Terkirim');
case 'SOS_ACKNOWLEDGED':
return const _LogMeta(
icon: Icons.check_circle,
color: AppColors.success,
label: 'SOS Diakui Guardian');
case 'LOCATION_UPDATE':
return const _LogMeta(
icon: Icons.location_on,
color: AppColors.primary,
label: 'Lokasi Diperbarui');
case 'GEOFENCE_EXIT':
return const _LogMeta(
icon: Icons.fence,
color: AppColors.danger,
label: 'Keluar Area Aman');
case 'GEOFENCE_ENTER':
return const _LogMeta(
icon: Icons.home, color: AppColors.success, label: 'Masuk Area Aman');
default:
return _LogMeta(
icon: Icons.circle, color: AppColors.muted, label: logType);
}
}
// Helper widgets
class _ErrorPanel extends StatelessWidget {
final String message;
final VoidCallback onRetry;
const _ErrorPanel({required this.message, required this.onRetry});
@override
Widget build(BuildContext context) {
return Center(
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
const Icon(Icons.wifi_off, size: 48, color: AppColors.muted),
const SizedBox(height: 12),
Text(message,
textAlign: TextAlign.center,
style: const TextStyle(color: AppColors.muted)),
const SizedBox(height: 16),
FilledButton.icon(
onPressed: onRetry,
icon: const Icon(Icons.refresh),
label: const Text('Coba lagi'),
),
],
),
);
}
}
class _EmptyPanel extends StatelessWidget {
final String filter;
const _EmptyPanel({required this.filter});
@override
Widget build(BuildContext context) {
return Center(
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
const Icon(Icons.history, size: 64, color: AppColors.muted),
const SizedBox(height: 12),
Text(
filter == 'ALL'
? 'Belum ada aktivitas'
: 'Tidak ada aktivitas "$filter"',
style: const TextStyle(
fontSize: 16,
fontWeight: FontWeight.w600,
color: AppColors.muted),
),
],
),
);
}
}