feat: Enhance chapter list and TTS functionality
Build Android APK / build-apk (push) Failing after 4m37s

- Introduced ChapterListQuery and ChapterListPage classes for better chapter management.
- Updated chapterListProvider to handle pagination and canonical ID resolution.
- Improved ReaderScreen with enhanced TTS features, including auto-scroll to active paragraph and better handling of TTS state.
- Added TtsPlayerWidget with compact mode and improved UI for TTS controls.
- Enhanced TtsService to manage speech segments and background mode for TTS.
- Implemented battery optimization checks for TTS background mode on Android.
- Updated main.dart to ensure proper error handling in a zoned environment.
This commit is contained in:
2026-04-07 18:49:29 +07:00
parent 1afff18f4d
commit 6946083aee
27 changed files with 1590 additions and 157 deletions
+32 -11
View File
@@ -1,5 +1,26 @@
import 'package:equatable/equatable.dart';
int _toInt(dynamic value, {int fallback = 0}) {
if (value is int) return value;
if (value is num) return value.toInt();
if (value is String) return int.tryParse(value) ?? fallback;
return fallback;
}
DateTime _toDateTime(dynamic value) {
if (value is DateTime) return value;
if (value is String && value.isNotEmpty) {
final parsed = DateTime.tryParse(value);
if (parsed != null) return parsed;
}
return DateTime.fromMillisecondsSinceEpoch(0);
}
int? _toNullableInt(dynamic value) {
if (value == null) return null;
return _toInt(value);
}
class ChapterModel extends Equatable {
const ChapterModel({
required this.id,
@@ -36,18 +57,18 @@ class ChapterModel extends Equatable {
factory ChapterModel.fromJson(Map<String, dynamic> json) => ChapterModel(
id: json['id'] as String,
novelId: json['novelId'] as String,
number: (json['number'] as num).toInt(),
number: _toInt(json['number']),
title: json['title'] as String,
content: json['content'] as String,
views: (json['views'] as num?)?.toInt() ?? 0,
volumeNumber: json['volumeNumber'] as int?,
views: _toInt(json['views']),
volumeNumber: _toNullableInt(json['volumeNumber']),
volumeTitle: json['volumeTitle'] as String?,
volumeChapterNumber: json['volumeChapterNumber'] as int?,
volumeChapterNumber: _toNullableInt(json['volumeChapterNumber']),
prevChapterId: json['prevChapterId'] as String?,
prevChapterNumber: json['prevChapterNumber'] as int?,
prevChapterNumber: _toNullableInt(json['prevChapterNumber']),
nextChapterId: json['nextChapterId'] as String?,
nextChapterNumber: json['nextChapterNumber'] as int?,
createdAt: DateTime.parse(json['createdAt'] as String),
nextChapterNumber: _toNullableInt(json['nextChapterNumber']),
createdAt: _toDateTime(json['createdAt']),
);
@override
@@ -75,12 +96,12 @@ class ChapterListItem extends Equatable {
factory ChapterListItem.fromJson(Map<String, dynamic> json) => ChapterListItem(
id: json['id'] as String,
number: (json['number'] as num).toInt(),
number: _toInt(json['number']),
title: json['title'] as String,
volumeNumber: json['volumeNumber'] as int?,
volumeNumber: _toNullableInt(json['volumeNumber']),
volumeTitle: json['volumeTitle'] as String?,
volumeChapterNumber: json['volumeChapterNumber'] as int?,
createdAt: DateTime.parse(json['createdAt'] as String),
volumeChapterNumber: _toNullableInt(json['volumeChapterNumber']),
createdAt: _toDateTime(json['createdAt']),
);
@override