feat: checkpoint API migration and dzikir UX updates

This commit is contained in:
Dwindi Ramadhana
2026-03-16 00:30:32 +07:00
parent c4696f2d9f
commit a049129a35
85 changed files with 4285 additions and 211 deletions

View File

@@ -19,6 +19,14 @@ class _QuranBookmarksScreenState extends State<QuranBookmarksScreen> {
bool _showLatin = true;
bool _showTerjemahan = true;
String _readingRoute(int surahId, int verseId) {
final isSimple =
Hive.box<AppSettings>(HiveBoxes.settings).get('default')?.simpleMode ??
false;
final base = isSimple ? '/quran' : '/tools/quran';
return '$base/$surahId?startVerse=$verseId';
}
@override
void initState() {
super.initState();
@@ -184,7 +192,7 @@ class _QuranBookmarksScreenState extends State<QuranBookmarksScreen> {
final dateStr = DateFormat('dd MMM yyyy, HH:mm').format(bookmark.savedAt);
return InkWell(
onTap: () => context.push('/tools/quran/${bookmark.surahId}?startVerse=${bookmark.verseId}'),
onTap: () => context.push(_readingRoute(bookmark.surahId, bookmark.verseId)),
borderRadius: BorderRadius.circular(16),
child: Container(
padding: const EdgeInsets.all(16),
@@ -252,6 +260,7 @@ class _QuranBookmarksScreenState extends State<QuranBookmarksScreen> {
style: const TextStyle(
fontFamily: 'Amiri',
fontSize: 22,
fontWeight: FontWeight.w400,
height: 1.8,
),
),
@@ -287,7 +296,8 @@ class _QuranBookmarksScreenState extends State<QuranBookmarksScreen> {
SizedBox(
width: double.infinity,
child: FilledButton.icon(
onPressed: () => context.push('/tools/quran/${bookmark.surahId}?startVerse=${bookmark.verseId}'),
onPressed: () =>
context.push(_readingRoute(bookmark.surahId, bookmark.verseId)),
icon: const Icon(LucideIcons.bookOpen, size: 18),
label: const Text('Lanjutkan Membaca'),
style: FilledButton.styleFrom(

View File

@@ -0,0 +1,773 @@
import 'package:flutter/material.dart';
import 'package:lucide_icons/lucide_icons.dart';
import '../../../app/theme/app_colors.dart';
import '../../../data/services/muslim_api_service.dart';
class QuranEnrichmentScreen extends StatefulWidget {
const QuranEnrichmentScreen({super.key});
@override
State<QuranEnrichmentScreen> createState() => _QuranEnrichmentScreenState();
}
class _QuranEnrichmentScreenState extends State<QuranEnrichmentScreen>
with SingleTickerProviderStateMixin {
late TabController _tabController;
final TextEditingController _searchController = TextEditingController();
final TextEditingController _pageController = TextEditingController(text: '1');
List<Map<String, dynamic>> _surahs = [];
List<Map<String, dynamic>> _searchResults = [];
List<Map<String, dynamic>> _tafsirItems = [];
List<Map<String, dynamic>> _asbabItems = [];
List<Map<String, dynamic>> _juzItems = [];
List<Map<String, dynamic>> _pageItems = [];
List<Map<String, dynamic>> _themeItems = [];
List<Map<String, dynamic>> _asmaItems = [];
int _selectedSurahId = 1;
int _selectedPage = 1;
bool _loadingInit = true;
bool _loadingSearch = false;
bool _loadingTafsir = false;
bool _loadingAsbab = false;
bool _loadingPage = false;
String? _error;
final Set<String> _expandedWordByWord = {};
final Map<String, List<Map<String, dynamic>>> _wordByWord = {};
final Set<String> _loadingWordByWord = {};
@override
void initState() {
super.initState();
_tabController = TabController(length: 7, vsync: this);
_bootstrap();
}
@override
void dispose() {
_tabController.dispose();
_searchController.dispose();
_pageController.dispose();
super.dispose();
}
Future<void> _bootstrap() async {
setState(() {
_loadingInit = true;
_error = null;
});
try {
final surahs = await MuslimApiService.instance.getAllSurahs();
final juz = await MuslimApiService.instance.getJuzList();
final themes = await MuslimApiService.instance.getThemes();
final asma = await MuslimApiService.instance.getAsmaulHusna();
if (!mounted) return;
setState(() {
_surahs = surahs;
_selectedSurahId = surahs.isNotEmpty
? ((surahs.first['nomor'] as int?) ?? 1)
: 1;
_juzItems = juz;
_themeItems = themes;
_asmaItems = asma;
_loadingInit = false;
});
await _loadTafsirForSelectedSurah();
await _loadAsbabForSelectedSurah();
await _loadPageAyah();
} catch (_) {
if (!mounted) return;
setState(() {
_loadingInit = false;
_error = 'Gagal memuat data enrichment';
});
}
}
Future<void> _runSearch() async {
final query = _searchController.text.trim();
if (query.isEmpty) {
setState(() => _searchResults = []);
return;
}
setState(() => _loadingSearch = true);
final result = await MuslimApiService.instance.searchAyah(query);
if (!mounted) return;
setState(() {
_searchResults = result;
_loadingSearch = false;
});
}
Future<void> _loadTafsirForSelectedSurah() async {
setState(() => _loadingTafsir = true);
final result =
await MuslimApiService.instance.getTafsirBySurah(_selectedSurahId);
if (!mounted) return;
setState(() {
_tafsirItems = result;
_loadingTafsir = false;
});
}
Future<void> _loadAsbabForSelectedSurah() async {
setState(() => _loadingAsbab = true);
final result = await MuslimApiService.instance.getAsbabBySurah(_selectedSurahId);
if (!mounted) return;
setState(() {
_asbabItems = result;
_loadingAsbab = false;
});
}
Future<void> _loadPageAyah() async {
setState(() => _loadingPage = true);
final page = int.tryParse(_pageController.text.trim()) ?? _selectedPage;
final safePage = page.clamp(1, 604);
final result = await MuslimApiService.instance.getAyahByPage(safePage);
if (!mounted) return;
setState(() {
_selectedPage = safePage;
_pageController.text = '$safePage';
_pageItems = result;
_loadingPage = false;
});
}
Future<void> _toggleWordByWord(Map<String, dynamic> ayah) async {
final surah = (ayah['surah'] as num?)?.toInt();
final ayahNum = (ayah['ayah'] as num?)?.toInt();
if (surah == null || ayahNum == null) return;
final key = '$surah:$ayahNum';
final expanded = _expandedWordByWord.contains(key);
if (expanded) {
setState(() => _expandedWordByWord.remove(key));
return;
}
if (_wordByWord.containsKey(key)) {
setState(() => _expandedWordByWord.add(key));
return;
}
setState(() {
_loadingWordByWord.add(key);
_expandedWordByWord.add(key);
});
final words = await MuslimApiService.instance.getWordByWord(surah, ayahNum);
if (!mounted) return;
setState(() {
_wordByWord[key] = words;
_loadingWordByWord.remove(key);
});
}
String _surahNameById(int surahId) {
for (final s in _surahs) {
if (s['nomor'] == surahId) {
return s['namaLatin']?.toString() ?? 'Surah $surahId';
}
}
return 'Surah $surahId';
}
@override
Widget build(BuildContext context) {
final isDark = Theme.of(context).brightness == Brightness.dark;
return Scaffold(
appBar: AppBar(
title: const Text('Quran Enrichment'),
actions: [
IconButton(
onPressed: _bootstrap,
icon: const Icon(LucideIcons.refreshCw),
tooltip: 'Muat ulang',
),
],
bottom: TabBar(
controller: _tabController,
isScrollable: true,
labelColor: AppColors.primary,
unselectedLabelColor: isDark
? AppColors.textSecondaryDark
: AppColors.textSecondaryLight,
indicatorColor: AppColors.primary,
tabs: const [
Tab(text: 'Cari'),
Tab(text: 'Tafsir'),
Tab(text: 'Asbab'),
Tab(text: 'Juz'),
Tab(text: 'Halaman'),
Tab(text: 'Tema'),
Tab(text: 'Asmaul Husna'),
],
),
),
body: _loadingInit
? const Center(child: CircularProgressIndicator())
: _error != null
? Center(
child: Text(
_error!,
style: TextStyle(
color: isDark
? AppColors.textSecondaryDark
: AppColors.textSecondaryLight,
),
),
)
: TabBarView(
controller: _tabController,
children: [
_buildSearchTab(context, isDark),
_buildTafsirTab(context, isDark),
_buildAsbabTab(context, isDark),
_buildJuzTab(context, isDark),
_buildPageTab(context, isDark),
_buildThemeTab(context, isDark),
_buildAsmaTab(context, isDark),
],
),
);
}
Widget _buildSearchTab(BuildContext context, bool isDark) {
return Column(
children: [
Padding(
padding: const EdgeInsets.fromLTRB(16, 12, 16, 8),
child: Row(
children: [
Expanded(
child: TextField(
controller: _searchController,
textInputAction: TextInputAction.search,
onSubmitted: (_) => _runSearch(),
decoration: InputDecoration(
hintText: 'Cari ayat, tema, atau kata kunci...',
prefixIcon: const Icon(LucideIcons.search),
border: OutlineInputBorder(
borderRadius: BorderRadius.circular(12),
),
),
),
),
const SizedBox(width: 8),
FilledButton(
onPressed: _runSearch,
child: const Text('Cari'),
),
],
),
),
Expanded(
child: _loadingSearch
? const Center(child: CircularProgressIndicator())
: _searchResults.isEmpty
? Center(
child: Text(
'Belum ada hasil pencarian',
style: TextStyle(
color: isDark
? AppColors.textSecondaryDark
: AppColors.textSecondaryLight,
),
),
)
: ListView.builder(
padding: const EdgeInsets.fromLTRB(16, 8, 16, 16),
itemCount: _searchResults.length,
itemBuilder: (context, index) {
final ayah = _searchResults[index];
final surahId = (ayah['surah'] as num?)?.toInt() ?? 0;
final ayahNum = (ayah['ayah'] as num?)?.toInt() ?? 0;
final key = '$surahId:$ayahNum';
final expanded = _expandedWordByWord.contains(key);
final words = _wordByWord[key] ?? const [];
final loadingWords = _loadingWordByWord.contains(key);
return Container(
margin: const EdgeInsets.only(bottom: 12),
padding: const EdgeInsets.all(14),
decoration: BoxDecoration(
color: isDark
? AppColors.surfaceDark
: AppColors.surfaceLight,
borderRadius: BorderRadius.circular(14),
border: Border.all(
color: isDark
? AppColors.primary.withValues(alpha: 0.1)
: AppColors.cream,
),
),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
mainAxisAlignment:
MainAxisAlignment.spaceBetween,
children: [
Text(
'${_surahNameById(surahId)} : $ayahNum',
style: const TextStyle(
fontWeight: FontWeight.w700,
color: AppColors.primary,
),
),
TextButton.icon(
onPressed: () => _toggleWordByWord(ayah),
icon: Icon(
expanded
? LucideIcons.chevronUp
: LucideIcons.languages,
size: 16,
),
label: Text(
expanded ? 'Tutup' : 'Per Kata',
),
),
],
),
const SizedBox(height: 8),
Align(
alignment: Alignment.centerRight,
child: Text(
ayah['arab']?.toString() ?? '',
textAlign: TextAlign.right,
style: const TextStyle(
fontFamily: 'Amiri',
fontSize: 24,
fontWeight: FontWeight.w400,
height: 1.8,
),
),
),
const SizedBox(height: 8),
Text(
ayah['text']?.toString() ?? '',
style: TextStyle(
color: isDark
? AppColors.textSecondaryDark
: AppColors.textSecondaryLight,
),
),
if (expanded) ...[
const SizedBox(height: 12),
if (loadingWords)
const Padding(
padding: EdgeInsets.symmetric(vertical: 8),
child: Center(
child: CircularProgressIndicator(),
),
)
else if (words.isEmpty)
Text(
'Data kata tidak tersedia untuk ayat ini.',
style: TextStyle(
color: isDark
? AppColors.textSecondaryDark
: AppColors.textSecondaryLight,
),
)
else
Wrap(
spacing: 8,
runSpacing: 8,
children: words.map((word) {
return Container(
padding: const EdgeInsets.all(10),
decoration: BoxDecoration(
color: AppColors.primary
.withValues(alpha: 0.08),
borderRadius:
BorderRadius.circular(10),
),
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
Text(
word['arab']?.toString() ?? '',
style: const TextStyle(
fontFamily: 'Amiri',
fontSize: 18,
fontWeight: FontWeight.w400,
),
),
const SizedBox(height: 2),
Text(
word['word']?.toString() ?? '',
style: const TextStyle(
fontWeight: FontWeight.w700,
fontSize: 12,
),
),
const SizedBox(height: 2),
Text(
word['indo']?.toString() ?? '',
style: TextStyle(
fontSize: 11,
color: isDark
? AppColors
.textSecondaryDark
: AppColors
.textSecondaryLight,
),
),
],
),
);
}).toList(),
),
],
],
),
);
},
),
),
],
);
}
Widget _buildTafsirTab(BuildContext context, bool isDark) {
return Column(
children: [
_buildSurahSelector(
onChanged: (value) {
setState(() => _selectedSurahId = value);
_loadTafsirForSelectedSurah();
},
),
Expanded(
child: _loadingTafsir
? const Center(child: CircularProgressIndicator())
: _tafsirItems.isEmpty
? _emptyText(isDark, 'Belum ada data tafsir untuk surah ini')
: ListView.builder(
padding: const EdgeInsets.fromLTRB(16, 8, 16, 16),
itemCount: _tafsirItems.length,
itemBuilder: (context, index) {
final item = _tafsirItems[index];
final ayah = item['nomorAyat']?.toString() ?? '-';
final wajiz = item['wajiz']?.toString() ?? '';
final tahlili = item['tahlili']?.toString() ?? '';
return _buildCard(
isDark,
title: 'Ayat $ayah',
body: '$wajiz\n\n$tahlili',
);
},
),
),
],
);
}
Widget _buildAsbabTab(BuildContext context, bool isDark) {
return Column(
children: [
_buildSurahSelector(
onChanged: (value) {
setState(() => _selectedSurahId = value);
_loadAsbabForSelectedSurah();
},
),
Expanded(
child: _loadingAsbab
? const Center(child: CircularProgressIndicator())
: _asbabItems.isEmpty
? _emptyText(
isDark,
'Belum ada data asbabun nuzul untuk surah ini',
)
: ListView.builder(
padding: const EdgeInsets.fromLTRB(16, 8, 16, 16),
itemCount: _asbabItems.length,
itemBuilder: (context, index) {
final item = _asbabItems[index];
final ayah = item['nomorAyat']?.toString() ?? '-';
return _buildCard(
isDark,
title: 'Ayat $ayah',
body: item['text']?.toString() ?? '',
);
},
),
),
],
);
}
Widget _buildJuzTab(BuildContext context, bool isDark) {
if (_juzItems.isEmpty) {
return _emptyText(isDark, 'Data juz tidak tersedia');
}
return ListView.builder(
padding: const EdgeInsets.fromLTRB(16, 12, 16, 16),
itemCount: _juzItems.length,
itemBuilder: (context, index) {
final item = _juzItems[index];
final number = item['number']?.toString() ?? '-';
final startName = item['name_start_id']?.toString() ?? '-';
final endName = item['name_end_id']?.toString() ?? '-';
final startVerse = item['verse_start']?.toString() ?? '-';
final endVerse = item['verse_end']?.toString() ?? '-';
return _buildCard(
isDark,
title: 'Juz $number',
body:
'Mulai: $startName ayat $startVerse\nSelesai: $endName ayat $endVerse',
);
},
);
}
Widget _buildPageTab(BuildContext context, bool isDark) {
return Column(
children: [
Padding(
padding: const EdgeInsets.fromLTRB(16, 12, 16, 8),
child: Row(
children: [
Expanded(
child: TextField(
controller: _pageController,
keyboardType: TextInputType.number,
decoration: InputDecoration(
labelText: 'Nomor Halaman (1-604)',
border: OutlineInputBorder(
borderRadius: BorderRadius.circular(12),
),
),
),
),
const SizedBox(width: 8),
FilledButton(
onPressed: _loadPageAyah,
child: const Text('Tampilkan'),
),
],
),
),
Expanded(
child: _loadingPage
? const Center(child: CircularProgressIndicator())
: _pageItems.isEmpty
? _emptyText(isDark, 'Tidak ada data untuk halaman ini')
: ListView.builder(
padding: const EdgeInsets.fromLTRB(16, 8, 16, 16),
itemCount: _pageItems.length,
itemBuilder: (context, index) {
final item = _pageItems[index];
final surahId = (item['surah'] as num?)?.toInt() ?? 0;
final ayah = item['ayah']?.toString() ?? '-';
return _buildCard(
isDark,
title: '${_surahNameById(surahId)} : $ayah',
body:
'${item['arab']?.toString() ?? ''}\n\n${item['text']?.toString() ?? ''}',
);
},
),
),
],
);
}
Widget _buildThemeTab(BuildContext context, bool isDark) {
if (_themeItems.isEmpty) {
return _emptyText(isDark, 'Data tema belum tersedia');
}
return ListView.builder(
padding: const EdgeInsets.fromLTRB(16, 12, 16, 16),
itemCount: _themeItems.length,
itemBuilder: (context, index) {
final item = _themeItems[index];
return _buildCard(
isDark,
title: 'Tema #${item['id'] ?? '-'}',
body: item['name']?.toString() ?? '',
);
},
);
}
Widget _buildAsmaTab(BuildContext context, bool isDark) {
if (_asmaItems.isEmpty) {
return _emptyText(isDark, 'Data Asmaul Husna tidak tersedia');
}
return ListView.builder(
padding: const EdgeInsets.fromLTRB(16, 12, 16, 16),
itemCount: _asmaItems.length,
itemBuilder: (context, index) {
final item = _asmaItems[index];
return Container(
margin: const EdgeInsets.only(bottom: 10),
padding: const EdgeInsets.all(14),
decoration: BoxDecoration(
color: isDark ? AppColors.surfaceDark : AppColors.surfaceLight,
borderRadius: BorderRadius.circular(12),
border: Border.all(
color: isDark
? AppColors.primary.withValues(alpha: 0.1)
: AppColors.cream,
),
),
child: Row(
children: [
Container(
width: 40,
height: 40,
alignment: Alignment.center,
decoration: BoxDecoration(
color: AppColors.primary.withValues(alpha: 0.1),
borderRadius: BorderRadius.circular(10),
),
child: Text(
'${item['id'] ?? '-'}',
style: const TextStyle(
fontWeight: FontWeight.w700,
color: AppColors.primary,
),
),
),
const SizedBox(width: 12),
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
item['arab']?.toString() ?? '',
style: const TextStyle(
fontFamily: 'Amiri',
fontSize: 22,
fontWeight: FontWeight.w400,
),
),
Text(
item['latin']?.toString() ?? '',
style: const TextStyle(
fontWeight: FontWeight.w700,
color: AppColors.primary,
),
),
const SizedBox(height: 2),
Text(
item['indo']?.toString() ?? '',
style: TextStyle(
color: isDark
? AppColors.textSecondaryDark
: AppColors.textSecondaryLight,
),
),
],
),
),
],
),
);
},
);
}
Widget _buildSurahSelector({required ValueChanged<int> onChanged}) {
if (_surahs.isEmpty) {
return Padding(
padding: const EdgeInsets.fromLTRB(16, 12, 16, 8),
child: _emptyText(
Theme.of(context).brightness == Brightness.dark,
'Data surah tidak tersedia',
),
);
}
return Padding(
padding: const EdgeInsets.fromLTRB(16, 12, 16, 8),
child: Container(
padding: const EdgeInsets.symmetric(horizontal: 12),
decoration: BoxDecoration(
border: Border.all(color: AppColors.primary.withValues(alpha: 0.2)),
borderRadius: BorderRadius.circular(12),
),
child: DropdownButtonHideUnderline(
child: DropdownButton<int>(
value: _selectedSurahId,
isExpanded: true,
items: _surahs.map((surah) {
final id = (surah['nomor'] as num?)?.toInt() ?? 1;
final name = surah['namaLatin']?.toString() ?? 'Surah $id';
return DropdownMenuItem<int>(
value: id,
child: Text('$id. $name'),
);
}).toList(),
onChanged: (value) {
if (value == null) return;
onChanged(value);
},
),
),
),
);
}
Widget _buildCard(bool isDark, {required String title, required String body}) {
return Container(
margin: const EdgeInsets.only(bottom: 10),
padding: const EdgeInsets.all(14),
decoration: BoxDecoration(
color: isDark ? AppColors.surfaceDark : AppColors.surfaceLight,
borderRadius: BorderRadius.circular(12),
border: Border.all(
color: isDark
? AppColors.primary.withValues(alpha: 0.1)
: AppColors.cream,
),
),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
title,
style: const TextStyle(
fontWeight: FontWeight.w700,
color: AppColors.primary,
),
),
const SizedBox(height: 8),
Text(body, style: const TextStyle(height: 1.5)),
],
),
);
}
Widget _emptyText(bool isDark, String text) {
return Center(
child: Text(
text,
textAlign: TextAlign.center,
style: TextStyle(
color: isDark
? AppColors.textSecondaryDark
: AppColors.textSecondaryLight,
),
),
);
}
}

View File

@@ -9,14 +9,11 @@ import 'package:lucide_icons/lucide_icons.dart';
import 'package:cached_network_image/cached_network_image.dart';
import 'package:url_launcher/url_launcher.dart';
import '../../../app/theme/app_colors.dart';
import '../../../data/services/equran_service.dart';
import '../../../data/services/muslim_api_service.dart';
import '../../../data/services/unsplash_service.dart';
import 'package:hive_flutter/hive_flutter.dart';
import '../../../data/local/hive_boxes.dart';
import '../../../data/local/models/app_settings.dart';
/// Quran Murattal (audio player) screen.
/// Implements full Surah playback using just_audio and EQuran v2 API.
/// Implements full Surah playback using just_audio.
class QuranMurattalScreen extends ConsumerStatefulWidget {
final String surahId;
final String? initialQariId;
@@ -77,7 +74,7 @@ class _QuranMurattalScreenState extends ConsumerState<QuranMurattalScreen> {
Future<void> _initDataAndPlayer() async {
final surahNum = int.tryParse(widget.surahId) ?? 1;
final data = await EQuranService.instance.getSurah(surahNum);
final data = await MuslimApiService.instance.getSurah(surahNum);
if (data != null && mounted) {
setState(() {
@@ -186,7 +183,10 @@ class _QuranMurattalScreenState extends ConsumerState<QuranMurattalScreen> {
void _navigateToSurahNumber(int surahNum, {bool autoplay = false}) {
if (surahNum >= 1 && surahNum <= 114) {
context.pushReplacement('/tools/quran/$surahNum/murattal?qariId=$_selectedQariId&autoplay=$autoplay');
final base = widget.isSimpleModeTab ? '/quran' : '/tools/quran';
context.pushReplacement(
'$base/$surahNum/murattal?qariId=$_selectedQariId&autoplay=$autoplay',
);
}
}
@@ -219,7 +219,7 @@ class _QuranMurattalScreenState extends ConsumerState<QuranMurattalScreen> {
),
),
const SizedBox(height: 16),
...EQuranService.qariNames.entries.map((entry) {
...MuslimApiService.qariNames.entries.map((entry) {
final isSelected = entry.key == _selectedQariId;
return ListTile(
leading: Icon(
@@ -287,7 +287,7 @@ class _QuranMurattalScreenState extends ConsumerState<QuranMurattalScreen> {
const SizedBox(height: 8),
Expanded(
child: FutureBuilder<List<Map<String, dynamic>>>(
future: EQuranService.instance.getAllSurahs(),
future: MuslimApiService.instance.getAllSurahs(),
builder: (context, snapshot) {
if (!snapshot.hasData) {
return const Center(child: CircularProgressIndicator());
@@ -339,7 +339,7 @@ class _QuranMurattalScreenState extends ConsumerState<QuranMurattalScreen> {
Navigator.pop(context);
if (!isCurrentSurah) {
context.pushReplacement(
'/tools/quran/$surahNum/murattal?qariId=$_selectedQariId',
'${widget.isSimpleModeTab ? '/quran' : '/tools/quran'}/$surahNum/murattal?qariId=$_selectedQariId',
);
}
},
@@ -360,8 +360,6 @@ class _QuranMurattalScreenState extends ConsumerState<QuranMurattalScreen> {
@override
Widget build(BuildContext context) {
final isDark = Theme.of(context).brightness == Brightness.dark;
final box = Hive.box<AppSettings>(HiveBoxes.settings);
final isSimpleMode = box.get('default')?.simpleMode ?? false;
final surahName = _surahData?['namaLatin'] ?? 'Surah ${widget.surahId}';
final hasPhoto = _unsplashPhoto != null;
@@ -519,7 +517,7 @@ class _QuranMurattalScreenState extends ConsumerState<QuranMurattalScreen> {
const SizedBox(height: 32),
// Qari name
Text(
EQuranService.qariNames[_selectedQariId] ?? 'Memuat...',
MuslimApiService.qariNames[_selectedQariId] ?? 'Memuat...',
style: TextStyle(
fontSize: 18,
fontWeight: FontWeight.w700,
@@ -742,7 +740,7 @@ class _QuranMurattalScreenState extends ConsumerState<QuranMurattalScreen> {
color: _unsplashPhoto != null ? Colors.white : AppColors.primary),
const SizedBox(width: 8),
Text(
EQuranService.qariNames[_selectedQariId] ?? 'Ganti Qari',
MuslimApiService.qariNames[_selectedQariId] ?? 'Ganti Qari',
style: TextStyle(
fontSize: 12,
fontWeight: FontWeight.w600,

View File

@@ -12,7 +12,7 @@ import '../../../data/local/models/quran_bookmark.dart';
import '../../../data/local/models/app_settings.dart';
import '../../../data/local/models/daily_worship_log.dart';
import '../../../data/local/models/tilawah_log.dart';
import '../../../data/services/equran_service.dart';
import '../../../data/services/muslim_api_service.dart';
import '../../../core/providers/tilawah_tracking_provider.dart';
class QuranReadingScreen extends ConsumerStatefulWidget {
@@ -151,7 +151,8 @@ class _QuranReadingScreenState extends ConsumerState<QuranReadingScreen> {
Future<void> _loadSurah() async {
final surahNum = int.tryParse(widget.surahId) ?? 1;
final data = await EQuranService.instance.getSurah(surahNum);
final data = await MuslimApiService.instance.getSurah(surahNum);
if (!mounted) return;
if (data != null) {
setState(() {
_surah = data;
@@ -356,7 +357,9 @@ class _QuranReadingScreenState extends ConsumerState<QuranReadingScreen> {
),
);
}
} Future<void> _showEndTrackingDialog(TilawahSession session, int endVerseId) async {
}
Future<void> _showEndTrackingDialog(TilawahSession session, int endVerseId) async {
final endSurahId = _surah!['nomor'] ?? 1;
final endSurahName = _surah!['namaLatin'] ?? '';
@@ -367,26 +370,30 @@ class _QuranReadingScreenState extends ConsumerState<QuranReadingScreen> {
calculatedAyat = (endVerseId - session.startVerseId).abs() + 1;
} else {
// Cross surah calculation
final allSurahs = await EQuranService.instance.getAllSurahs();
final allSurahs = await MuslimApiService.instance.getAllSurahs();
if (allSurahs.isNotEmpty) {
int startSurahIdx = allSurahs.indexWhere((s) => s['nomor'] == session.startSurahId);
int endSurahIdx = allSurahs.indexWhere((s) => s['nomor'] == endSurahId);
// Ensure chronological calculation
if (startSurahIdx > endSurahIdx) {
final tempIdx = startSurahIdx; startSurahIdx = endSurahIdx; endSurahIdx = tempIdx;
if (startSurahIdx < 0 || endSurahIdx < 0) {
calculatedAyat = (endVerseId - session.startVerseId).abs() + 1;
} else {
// Ensure chronological calculation
if (startSurahIdx > endSurahIdx) {
final tempIdx = startSurahIdx; startSurahIdx = endSurahIdx; endSurahIdx = tempIdx;
}
final startSurahData = allSurahs[startSurahIdx];
final int totalAyatInStart = (startSurahData['jumlahAyat'] as num?)?.toInt() ?? 1;
calculatedAyat += (totalAyatInStart - session.startVerseId) + 1; // Ayats inside StartSurah
for (int i = startSurahIdx + 1; i < endSurahIdx; i++) {
calculatedAyat += (allSurahs[i]['jumlahAyat'] as int? ?? 0); // Intermediate Surahs
}
calculatedAyat += endVerseId; // Ayats inside EndSurah
}
final startSurahData = allSurahs[startSurahIdx];
final int totalAyatInStart = (startSurahData['jumlahAyat'] as num?)?.toInt() ?? 1;
calculatedAyat += (totalAyatInStart - session.startVerseId) + 1; // Ayats inside StartSurah
for (int i = startSurahIdx + 1; i < endSurahIdx; i++) {
calculatedAyat += (allSurahs[i]['jumlahAyat'] as int? ?? 0); // Intermediate Surahs
}
calculatedAyat += endVerseId; // Ayats inside EndSurah
} else {
calculatedAyat = 1; // Fallback
}
@@ -572,6 +579,11 @@ class _QuranReadingScreenState extends ConsumerState<QuranReadingScreen> {
],
),
actions: [
IconButton(
icon: const Icon(LucideIcons.headphones),
tooltip: 'Murattal Surah',
onPressed: _navigateToMurattal,
),
IconButton(
icon: Icon(
LucideIcons.brain,
@@ -620,6 +632,7 @@ class _QuranReadingScreenState extends ConsumerState<QuranReadingScreen> {
style: TextStyle(
fontFamily: 'Amiri',
fontSize: 26,
fontWeight: FontWeight.w400,
),
),
const SizedBox(height: 4),
@@ -802,6 +815,7 @@ class _QuranReadingScreenState extends ConsumerState<QuranReadingScreen> {
style: const TextStyle(
fontFamily: 'Amiri',
fontSize: 26,
fontWeight: FontWeight.w400,
height: 2.0,
),
),

View File

@@ -7,7 +7,7 @@ import '../../../app/theme/app_colors.dart';
import '../../../data/local/hive_boxes.dart';
import '../../../data/local/models/app_settings.dart';
import '../../../data/local/models/quran_bookmark.dart';
import '../../../data/services/equran_service.dart';
import '../../../data/services/muslim_api_service.dart';
class QuranScreen extends ConsumerStatefulWidget {
final bool isSimpleModeTab;
@@ -36,7 +36,8 @@ class _QuranScreenState extends ConsumerState<QuranScreen> {
}
Future<void> _loadSurahs() async {
final data = await EQuranService.instance.getAllSurahs();
final data = await MuslimApiService.instance.getAllSurahs();
if (!mounted) return;
setState(() {
_surahs = data;
_loading = false;
@@ -100,8 +101,6 @@ class _QuranScreenState extends ConsumerState<QuranScreen> {
@override
Widget build(BuildContext context) {
final isDark = Theme.of(context).brightness == Brightness.dark;
final box = Hive.box<AppSettings>(HiveBoxes.settings);
final isSimpleMode = box.get('default')?.simpleMode ?? false;
final filtered = _searchQuery.isEmpty
? _surahs
: _surahs
@@ -119,7 +118,15 @@ class _QuranScreenState extends ConsumerState<QuranScreen> {
actions: [
IconButton(
icon: const Icon(LucideIcons.bookmark),
onPressed: () => context.push('/tools/quran/bookmarks'),
onPressed: () => context.push(widget.isSimpleModeTab
? '/quran/bookmarks'
: '/tools/quran/bookmarks'),
),
IconButton(
icon: const Icon(LucideIcons.sparkles),
onPressed: () => context.push(widget.isSimpleModeTab
? '/quran/enrichment'
: '/tools/quran/enrichment'),
),
IconButton(
icon: const Icon(LucideIcons.settings2),
@@ -198,8 +205,9 @@ class _QuranScreenState extends ConsumerState<QuranScreen> {
final hasLastRead = box.values.any((b) => b.isLastRead && b.surahId == number);
return ListTile(
onTap: () =>
context.push('/tools/quran/$number'),
onTap: () => context.push(widget.isSimpleModeTab
? '/quran/$number'
: '/tools/quran/$number'),
contentPadding: const EdgeInsets.symmetric(
horizontal: 0, vertical: 6),
leading: Container(
@@ -250,6 +258,7 @@ class _QuranScreenState extends ConsumerState<QuranScreen> {
style: const TextStyle(
fontFamily: 'Amiri',
fontSize: 18,
fontWeight: FontWeight.w400,
),
),
);