MIF_E31230892/sim_mobile/lib/features/profil/profil_page.dart

635 lines
20 KiB
Dart

// lib/features/profil/profil_page.dart
import 'dart:convert';
import 'package:flutter/material.dart';
import 'package:flutter/foundation.dart';
import 'package:shared_preferences/shared_preferences.dart';
import '../../core/api/api_service.dart';
class ProfilPage extends StatefulWidget {
const ProfilPage({super.key});
@override
State<ProfilPage> createState() => _ProfilPageState();
}
class _ProfilPageState extends State<ProfilPage> {
final _api = ApiService();
Map<String, dynamic>? _santriData;
bool _isLoading = true;
@override
void initState() {
super.initState();
_loadProfile();
}
Future<void> _loadProfile() async {
setState(() => _isLoading = true);
// Ambil dari cache dulu untuk UX yang cepat
final prefs = await SharedPreferences.getInstance();
final cachedData = prefs.getString('santri_data');
if (cachedData != null) {
if (mounted) {
setState(() {
_santriData = json.decode(cachedData);
_isLoading = false;
});
}
}
// Refresh dari API di background
final result = await _api.getProfile();
if (mounted && result['success'] == true && result['data'] != null) {
setState(() {
_santriData = result['data'];
_isLoading = false;
});
} else if (mounted && _santriData == null) {
setState(() => _isLoading = false);
}
}
@override
Widget build(BuildContext context) {
return Scaffold(
backgroundColor: Colors.grey[100],
appBar: AppBar(
title: const Text('Profil Santri'),
backgroundColor: const Color(0xFF6FBA9D),
foregroundColor: Colors.white,
elevation: 0,
),
body: _isLoading && _santriData == null
? const Center(child: CircularProgressIndicator())
: RefreshIndicator(
onRefresh: _loadProfile,
child: SingleChildScrollView(
physics: const AlwaysScrollableScrollPhysics(),
child: Column(
children: [
// Header dengan foto
_buildHeader(),
// Content
Padding(
padding: const EdgeInsets.all(12),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
// Informasi Dasar
_buildSectionCard(
title: 'Informasi Dasar',
icon: Icons.info_outline,
children: [
_buildInfoRow('ID Santri', _santriData?['id_santri']),
_buildInfoRow('NIS', _santriData?['nis']),
_buildInfoRow('Nama Lengkap', _santriData?['nama_lengkap']),
_buildInfoRow('Jenis Kelamin', _santriData?['jenis_kelamin']),
_buildInfoRow('Status', _santriData?['status'], isLast: true),
],
),
const SizedBox(height: 12),
// Kelas yang Diikuti
if (_santriData?['kelas_list'] != null &&
(_santriData!['kelas_list'] as List).isNotEmpty)
_buildKelasListSection(),
if (_santriData?['kelas_list'] != null &&
(_santriData!['kelas_list'] as List).isNotEmpty)
const SizedBox(height: 12),
// Alamat & Asal
_buildSectionCard(
title: 'Alamat & Asal',
icon: Icons.location_on_outlined,
children: [
_buildInfoRow('Alamat Santri', _santriData?['alamat_santri'],
isMultiline: true),
_buildInfoRow('Daerah Asal', _santriData?['daerah_asal'],
isLast: true),
],
),
const SizedBox(height: 12),
// Data Orang Tua / Wali
_buildSectionCard(
title: 'Data Orang Tua / Wali',
icon: Icons.family_restroom,
children: [
_buildInfoRow('Nama Orang Tua', _santriData?['nama_orang_tua']),
_buildInfoRow('Nomor HP Orang Tua', _santriData?['nomor_hp_ortu'],
isLast: true),
],
),
const SizedBox(height: 19),
],
),
),
],
),
),
),
);
}
Widget _buildHeader() {
final namaLengkap = _santriData?['nama_lengkap'] ?? 'Nama Santri';
final idSantri = _santriData?['id_santri'] ?? '-';
final status = _santriData?['status'] ?? 'Aktif';
final fotoUrl = _santriData?['foto_url'];
return Container(
width: double.infinity,
decoration: const BoxDecoration(
gradient: LinearGradient(
begin: Alignment.topCenter,
end: Alignment.bottomCenter,
colors: [
Color(0xFF6FBA9D),
Color(0xFF4D987B),
],
),
borderRadius: BorderRadius.only(
bottomLeft: Radius.circular(32),
bottomRight: Radius.circular(32),
),
),
child: Padding(
padding: const EdgeInsets.fromLTRB(19, 0, 19, 24),
child: Column(
children: [
// Avatar dengan foto
Container(
padding: const EdgeInsets.all(2),
decoration: BoxDecoration(
shape: BoxShape.circle,
border: Border.all(color: Colors.white, width: 2),
),
child: CircleAvatar(
radius: 50,
backgroundColor: Colors.white,
child: _buildFotoWidget(fotoUrl),
),
),
const SizedBox(height: 12),
// Nama
Text(
namaLengkap,
style: const TextStyle(
fontSize: 17,
fontWeight: FontWeight.bold,
color: Colors.white,
),
textAlign: TextAlign.center,
),
const SizedBox(height: 2),
// ID Santri
Text(
idSantri,
style: const TextStyle(
fontSize: 11,
color: Colors.white70,
),
),
const SizedBox(height: 9),
// Primary Kelas Badge
_buildPrimaryKelasBadge(),
const SizedBox(height: 7),
// Status Badge
Container(
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 5),
decoration: BoxDecoration(
color: status == 'Aktif' ? Colors.green : Colors.orange,
borderRadius: BorderRadius.circular(15),
),
child: Text(
status,
style: const TextStyle(
fontSize: 9,
fontWeight: FontWeight.bold,
color: Colors.white,
),
),
),
],
),
),
);
}
/// Widget foto profil dengan fallback ke icon
Widget _buildFotoWidget(String? fotoUrl) {
if (fotoUrl == null || fotoUrl.isEmpty) {
return Icon(
Icons.person,
size: 39,
color: const Color(0xFF6FBA9D).withValues(alpha: 0.7),
);
}
// Fix localhost: Chrome pakai localhost, Android emulator pakai 10.0.2.2
// final fixedUrl = kIsWeb
// ? fotoUrl
// : fotoUrl.replaceFirst('http://localhost', 'http://192.168.100.71');
final fixedUrl = fotoUrl;
debugPrint('🖼️ Foto URL: $fixedUrl');
return ClipOval(
child: Image.network(
fixedUrl,
width: 100,
height: 100,
fit: BoxFit.cover,
loadingBuilder: (context, child, loadingProgress) {
if (loadingProgress == null) return child;
return const SizedBox(
width: 30,
height: 30,
child: CircularProgressIndicator(
strokeWidth: 2,
color: Color(0xFF6FBA9D),
),
);
},
errorBuilder: (context, error, stackTrace) {
debugPrint('🔴 Foto error: $error');
return Icon(
Icons.person,
size: 39,
color: const Color(0xFF6FBA9D).withValues(alpha: 0.7),
);
},
),
);
}
Widget _buildSectionCard({
required String title,
required IconData icon,
required List<Widget> children,
}) {
return Card(
elevation: 2,
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(12),
),
child: Padding(
padding: const EdgeInsets.all(12),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
// Section Title
Row(
children: [
Icon(icon, size: 15, color: const Color(0xFF6FBA9D)),
const SizedBox(width: 7),
Text(
title,
style: const TextStyle(
fontSize: 12,
fontWeight: FontWeight.bold,
color: Color(0xFF6FBA9D),
),
),
],
),
const Divider(height: 19),
// Content
...children,
],
),
),
);
}
Widget _buildInfoRow(String label, String? value,
{bool isMultiline = false, bool isLast = false}) {
return LayoutBuilder(
builder: (context, constraints) {
final labelWidth = constraints.maxWidth * 0.35;
return Padding(
padding: EdgeInsets.only(bottom: isLast ? 0 : 9),
child: Row(
crossAxisAlignment:
isMultiline ? CrossAxisAlignment.start : CrossAxisAlignment.center,
children: [
SizedBox(
width: labelWidth,
child: Text(
label,
style: TextStyle(
fontSize: 11,
color: Colors.grey[600],
),
maxLines: 2,
overflow: TextOverflow.ellipsis,
),
),
const SizedBox(width: 7),
Expanded(
child: Text(
value ?? '-',
style: const TextStyle(
fontSize: 11,
fontWeight: FontWeight.w600,
),
maxLines: isMultiline ? 4 : 2,
overflow: TextOverflow.ellipsis,
),
),
],
),
);
},
);
}
// ==========================================
// METHODS FOR MULTI-CLASS DISPLAY
// ==========================================
/// Build primary kelas badge with class count
Widget _buildPrimaryKelasBadge() {
final kelasName = _santriData?['kelas'] ?? '-';
final kelasList = _santriData?['kelas_list'] as List?;
// Count total kelas
int totalKelas = 0;
if (kelasList != null) {
for (var kelompok in kelasList) {
final kelasDalam = kelompok['kelas'] as List? ?? [];
totalKelas += kelasDalam.length;
}
}
return Column(
children: [
// Primary class badge
Container(
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 7),
decoration: BoxDecoration(
color: Colors.white.withValues(alpha: 0.2),
borderRadius: BorderRadius.circular(15),
border: Border.all(color: Colors.white.withValues(alpha: 0.3), width: 1.5),
),
child: Row(
mainAxisSize: MainAxisSize.min,
children: [
const Icon(Icons.school, color: Colors.white, size: 12),
const SizedBox(width: 5),
Text(
kelasName,
style: const TextStyle(
fontSize: 11,
fontWeight: FontWeight.bold,
color: Colors.white,
),
),
],
),
),
// Class count hint
if (totalKelas > 1) ...[
const SizedBox(height: 5),
Text(
'+${totalKelas - 1} kelas lainnya',
style: TextStyle(
fontSize: 8,
color: Colors.white.withValues(alpha: 0.8),
),
),
],
],
);
}
/// Build kelas list section with ExpansionTile grouped by kelompok
Widget _buildKelasListSection() {
final kelasList = _santriData?['kelas_list'] as List? ?? [];
if (kelasList.isEmpty) {
return _buildSectionCard(
title: 'Kelas yang Diikuti',
icon: Icons.class_,
children: [
Center(
child: Padding(
padding: const EdgeInsets.all(12),
child: Text(
'Belum mengikuti kelas apapun',
style: TextStyle(
color: Colors.grey[600],
fontSize: 11,
),
),
),
),
],
);
}
return Card(
elevation: 2,
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(12),
),
child: Padding(
padding: const EdgeInsets.all(12),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
// Section Title
Row(
children: [
const Icon(Icons.class_, size: 15, color: Color(0xFF6FBA9D)),
const SizedBox(width: 7),
const Text(
'Kelas yang Diikuti',
style: TextStyle(
fontSize: 12,
fontWeight: FontWeight.bold,
color: Color(0xFF6FBA9D),
),
),
],
),
const Divider(height: 19),
// Kelompok list
...kelasList.asMap().entries.map((entry) {
final index = entry.key;
final kelompok = entry.value;
final kelompokName = kelompok['kelompok_name'] ?? 'Unknown';
final kelasItems = kelompok['kelas'] as List? ?? [];
return Column(
children: [
if (index > 0) const SizedBox(height: 7),
_buildKelompokExpansionTile(kelompokName, kelasItems),
],
);
}),
],
),
),
);
}
/// Build ExpansionTile for each kelompok
Widget _buildKelompokExpansionTile(String kelompokName, List kelasItems) {
final color = _getKelompokColor(kelompokName);
final icon = _getKelompokIcon(kelompokName);
return Container(
decoration: BoxDecoration(
border: Border.all(color: color.withValues(alpha: 0.3)),
borderRadius: BorderRadius.circular(9),
),
child: Theme(
data: Theme.of(context).copyWith(dividerColor: Colors.transparent),
child: ExpansionTile(
tilePadding: const EdgeInsets.symmetric(horizontal: 9, vertical: 2),
childrenPadding: const EdgeInsets.fromLTRB(9, 0, 9, 7),
leading: Container(
padding: const EdgeInsets.all(7),
decoration: BoxDecoration(
color: color.withValues(alpha: 0.1),
borderRadius: BorderRadius.circular(7),
),
child: Icon(icon, color: color, size: 15),
),
title: Text(
kelompokName,
style: TextStyle(
fontSize: 11,
fontWeight: FontWeight.bold,
color: color,
),
),
subtitle: Text(
'${kelasItems.length} kelas',
style: TextStyle(
fontSize: 9,
color: Colors.grey[600],
),
),
children: kelasItems.map((kelas) {
final namaKelas = kelas['nama_kelas'] ?? '-';
final kodeKelas = kelas['kode_kelas'] ?? '-';
final isPrimary = kelas['is_primary'] == true;
return Container(
margin: const EdgeInsets.only(top: 7),
padding: const EdgeInsets.all(9),
decoration: BoxDecoration(
color: isPrimary
? color.withValues(alpha: 0.1)
: Colors.grey.withValues(alpha: 0.05),
borderRadius: BorderRadius.circular(7),
border: isPrimary
? Border.all(color: color.withValues(alpha: 0.3), width: 1.5)
: null,
),
child: Row(
children: [
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
namaKelas,
style: TextStyle(
fontSize: 11,
fontWeight: isPrimary ? FontWeight.bold : FontWeight.w600,
color: isPrimary ? color : Colors.black87,
),
),
const SizedBox(height: 2),
Text(
kodeKelas,
style: TextStyle(
fontSize: 8,
color: Colors.grey[600],
),
),
],
),
),
if (isPrimary) ...[
const SizedBox(width: 7),
Container(
padding: const EdgeInsets.symmetric(horizontal: 7, vertical: 2),
decoration: BoxDecoration(
color: const Color(0xFFfbbf24),
borderRadius: BorderRadius.circular(7),
),
child: const Row(
mainAxisSize: MainAxisSize.min,
children: [
Icon(Icons.star, color: Colors.white, size: 9),
SizedBox(width: 2),
Text(
'Utama',
style: TextStyle(
fontSize: 8,
fontWeight: FontWeight.bold,
color: Colors.white,
),
),
],
),
),
],
],
),
);
}).toList(),
),
),
);
}
/// Get color for kelompok
Color _getKelompokColor(String kelompokName) {
final name = kelompokName.toLowerCase();
if (name.contains('pb') || name.contains('pondok')) {
return const Color(0xFF3b82f6); // Blue
} else if (name.contains('lambatan')) {
return const Color(0xFFfb923c); // Orange
} else if (name.contains('cepatan')) {
return const Color(0xFF10b981); // Green
} else if (name.contains('tahfidz') || name.contains('tahfid')) {
return const Color(0xFF6FBA9D); // Green
} else if (name.contains('hadist') || name.contains('hadis')) {
return const Color(0xFF14b8a6); // Teal
} else {
return const Color(0xFF6b7280); // Gray
}
}
/// Get icon for kelompok
IconData _getKelompokIcon(String kelompokName) {
final name = kelompokName.toLowerCase();
if (name.contains('pb') || name.contains('pondok')) {
return Icons.school;
} else if (name.contains('lambatan')) {
return Icons.menu_book;
} else if (name.contains('cepatan')) {
return Icons.speed;
} else if (name.contains('tahfidz') || name.contains('tahfid')) {
return Icons.auto_stories;
} else if (name.contains('hadist') || name.contains('hadis')) {
return Icons.import_contacts;
} else {
return Icons.class_;
}
}
}