635 lines
20 KiB
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_;
|
|
}
|
|
}
|
|
} |