Files
reader-app/lib/core/network/api_client.dart
T
virtus 6946083aee
Build Android APK / build-apk (push) Failing after 4m37s
feat: Enhance chapter list and TTS functionality
- 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.
2026-04-07 18:49:29 +07:00

76 lines
2.3 KiB
Dart

import 'package:dio/dio.dart';
import 'package:flutter/foundation.dart';
import '../storage/secure_store.dart';
class ApiClient {
ApiClient({
required String baseUrl,
required SecureStore secureStore,
this.onSessionExpired,
}) : _secureStore = secureStore,
dio = Dio(
BaseOptions(
baseUrl: baseUrl,
connectTimeout: const Duration(seconds: 20),
receiveTimeout: const Duration(seconds: 20),
headers: const {'Content-Type': 'application/json'},
),
) {
dio.interceptors.add(
InterceptorsWrapper(
onRequest: (options, handler) async {
debugPrint('[API] ${options.method} ${options.baseUrl}${options.path}');
final token = await _secureStore.getAccessToken();
if (token != null && token.isNotEmpty) {
options.headers['Authorization'] = 'Bearer $token';
}
handler.next(options);
},
onResponse: (response, handler) {
debugPrint(
'[API][OK] ${response.requestOptions.method} '
'${response.requestOptions.baseUrl}${response.requestOptions.path} '
'-> ${response.statusCode}',
);
handler.next(response);
},
onError: (error, handler) {
final statusCode = error.response?.statusCode;
final path = error.requestOptions.path;
if ((statusCode == 401 || statusCode == 403) && !_isAuthEndpoint(path)) {
_handleSessionExpired();
}
debugPrint(
'[API][ERROR] ${error.requestOptions.method} ${error.requestOptions.baseUrl}${error.requestOptions.path} '
'-> ${error.type}: ${error.message}',
);
handler.next(error);
},
),
);
}
final Dio dio;
final SecureStore _secureStore;
final VoidCallback? onSessionExpired;
DateTime? _lastSessionExpiredAt;
bool _isAuthEndpoint(String path) {
return path.contains('/api/auth/mobile-login');
}
void _handleSessionExpired() {
final now = DateTime.now();
if (_lastSessionExpiredAt != null &&
now.difference(_lastSessionExpiredAt!) < const Duration(seconds: 2)) {
return;
}
_lastSessionExpiredAt = now;
_secureStore.clear();
onSessionExpired?.call();
}
}