feat: implement ActivityLogScreen with real API + BLoC
This commit is contained in:
parent
6944bb87a8
commit
4cfe317d02
@ -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),
|
||||
),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user