chore: update dependencies and enhance connectivity service
- Added `connectivity_plus` dependency to manage network connectivity status. - Updated `pubspec.yaml` to include the new dependency version. - Enhanced `ConnectivityService` to utilize `connectivity_plus` for improved connectivity monitoring and handling. - Refactored connectivity checks and state management for better performance and reliability.
This commit is contained in:
@@ -34,6 +34,8 @@ PODS:
|
|||||||
- DKImagePickerController/PhotoGallery
|
- DKImagePickerController/PhotoGallery
|
||||||
- Flutter
|
- Flutter
|
||||||
- Flutter (1.0.0)
|
- Flutter (1.0.0)
|
||||||
|
- flutter_local_notifications (0.0.1):
|
||||||
|
- Flutter
|
||||||
- flutter_native_splash (2.4.3):
|
- flutter_native_splash (2.4.3):
|
||||||
- Flutter
|
- Flutter
|
||||||
- flutter_secure_storage (6.0.0):
|
- flutter_secure_storage (6.0.0):
|
||||||
@@ -82,6 +84,7 @@ PODS:
|
|||||||
DEPENDENCIES:
|
DEPENDENCIES:
|
||||||
- file_picker (from `.symlinks/plugins/file_picker/ios`)
|
- file_picker (from `.symlinks/plugins/file_picker/ios`)
|
||||||
- Flutter (from `Flutter`)
|
- Flutter (from `Flutter`)
|
||||||
|
- flutter_local_notifications (from `.symlinks/plugins/flutter_local_notifications/ios`)
|
||||||
- flutter_native_splash (from `.symlinks/plugins/flutter_native_splash/ios`)
|
- flutter_native_splash (from `.symlinks/plugins/flutter_native_splash/ios`)
|
||||||
- flutter_secure_storage (from `.symlinks/plugins/flutter_secure_storage/ios`)
|
- flutter_secure_storage (from `.symlinks/plugins/flutter_secure_storage/ios`)
|
||||||
- flutter_tts (from `.symlinks/plugins/flutter_tts/ios`)
|
- flutter_tts (from `.symlinks/plugins/flutter_tts/ios`)
|
||||||
@@ -111,6 +114,8 @@ EXTERNAL SOURCES:
|
|||||||
:path: ".symlinks/plugins/file_picker/ios"
|
:path: ".symlinks/plugins/file_picker/ios"
|
||||||
Flutter:
|
Flutter:
|
||||||
:path: Flutter
|
:path: Flutter
|
||||||
|
flutter_local_notifications:
|
||||||
|
:path: ".symlinks/plugins/flutter_local_notifications/ios"
|
||||||
flutter_native_splash:
|
flutter_native_splash:
|
||||||
:path: ".symlinks/plugins/flutter_native_splash/ios"
|
:path: ".symlinks/plugins/flutter_native_splash/ios"
|
||||||
flutter_secure_storage:
|
flutter_secure_storage:
|
||||||
@@ -149,6 +154,7 @@ SPEC CHECKSUMS:
|
|||||||
DKPhotoGallery: b3834fecb755ee09a593d7c9e389d8b5d6deed60
|
DKPhotoGallery: b3834fecb755ee09a593d7c9e389d8b5d6deed60
|
||||||
file_picker: a0560bc09d61de87f12d246fc47d2119e6ef37be
|
file_picker: a0560bc09d61de87f12d246fc47d2119e6ef37be
|
||||||
Flutter: cabc95a1d2626b1b06e7179b784ebcf0c0cde467
|
Flutter: cabc95a1d2626b1b06e7179b784ebcf0c0cde467
|
||||||
|
flutter_local_notifications: a5a732f069baa862e728d839dd2ebb904737effb
|
||||||
flutter_native_splash: c32d145d68aeda5502d5f543ee38c192065986cf
|
flutter_native_splash: c32d145d68aeda5502d5f543ee38c192065986cf
|
||||||
flutter_secure_storage: 1ed9476fba7e7a782b22888f956cce43e2c62f13
|
flutter_secure_storage: 1ed9476fba7e7a782b22888f956cce43e2c62f13
|
||||||
flutter_tts: b88dbc8655d3dc961bc4a796e4e16a4cc1795833
|
flutter_tts: b88dbc8655d3dc961bc4a796e4e16a4cc1795833
|
||||||
|
|||||||
@@ -122,11 +122,26 @@ class BackgroundStreamingHandler: NSObject {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private func saveStreamStates(_ states: [[String: Any]]) {
|
private func saveStreamStates(_ states: [[String: Any]]) {
|
||||||
UserDefaults.standard.set(states, forKey: "ConduitActiveStreams")
|
do {
|
||||||
|
let jsonData = try JSONSerialization.data(withJSONObject: states, options: [])
|
||||||
|
UserDefaults.standard.set(jsonData, forKey: "ConduitActiveStreams")
|
||||||
|
} catch {
|
||||||
|
print("BackgroundStreamingHandler: Failed to serialize stream states: \(error)")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private func recoverStreamStates() -> [[String: Any]] {
|
private func recoverStreamStates() -> [[String: Any]] {
|
||||||
return UserDefaults.standard.array(forKey: "ConduitActiveStreams") as? [[String: Any]] ?? []
|
guard let jsonData = UserDefaults.standard.data(forKey: "ConduitActiveStreams") else {
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
do {
|
||||||
|
if let states = try JSONSerialization.jsonObject(with: jsonData, options: []) as? [[String: Any]] {
|
||||||
|
return states
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
print("BackgroundStreamingHandler: Failed to deserialize stream states: \(error)")
|
||||||
|
}
|
||||||
|
return []
|
||||||
}
|
}
|
||||||
|
|
||||||
deinit {
|
deinit {
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import 'dart:async';
|
import 'dart:async';
|
||||||
|
|
||||||
|
import 'package:connectivity_plus/connectivity_plus.dart';
|
||||||
import 'package:dio/dio.dart';
|
import 'package:dio/dio.dart';
|
||||||
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
||||||
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
||||||
@@ -11,46 +12,171 @@ part 'connectivity_service.g.dart';
|
|||||||
enum ConnectivityStatus { online, offline, checking }
|
enum ConnectivityStatus { online, offline, checking }
|
||||||
|
|
||||||
class ConnectivityService {
|
class ConnectivityService {
|
||||||
final Dio _dio;
|
ConnectivityService(this._dio, this._ref, [Connectivity? connectivity])
|
||||||
final Ref _ref;
|
: _connectivity = connectivity ?? Connectivity() {
|
||||||
Timer? _connectivityTimer;
|
|
||||||
final _connectivityController =
|
|
||||||
StreamController<ConnectivityStatus>.broadcast();
|
|
||||||
ConnectivityStatus _lastStatus = ConnectivityStatus.checking;
|
|
||||||
int _recentFailures = 0;
|
|
||||||
Duration _interval = const Duration(seconds: 10);
|
|
||||||
int _lastLatencyMs = -1;
|
|
||||||
|
|
||||||
ConnectivityService(this._dio, this._ref) {
|
|
||||||
_startConnectivityMonitoring();
|
_startConnectivityMonitoring();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
final Dio _dio;
|
||||||
|
final Ref _ref;
|
||||||
|
final Connectivity _connectivity;
|
||||||
|
|
||||||
|
final _connectivityController =
|
||||||
|
StreamController<ConnectivityStatus>.broadcast();
|
||||||
|
|
||||||
|
Timer? _initialCheckTimer;
|
||||||
|
Timer? _pollTimer;
|
||||||
|
StreamSubscription<List<ConnectivityResult>>? _connectivitySubscription;
|
||||||
|
Completer<void>? _activeCheck;
|
||||||
|
List<ConnectivityResult>? _lastConnectivityResults;
|
||||||
|
|
||||||
|
ConnectivityStatus _lastStatus = ConnectivityStatus.checking;
|
||||||
|
Duration _interval = const Duration(seconds: 10);
|
||||||
|
int _recentFailures = 0;
|
||||||
|
int _lastLatencyMs = -1;
|
||||||
|
bool _hasNetwork = true;
|
||||||
|
bool _queuedImmediateCheck = false;
|
||||||
|
|
||||||
Stream<ConnectivityStatus> get connectivityStream =>
|
Stream<ConnectivityStatus> get connectivityStream =>
|
||||||
_connectivityController.stream;
|
_connectivityController.stream;
|
||||||
ConnectivityStatus get currentStatus => _lastStatus;
|
ConnectivityStatus get currentStatus => _lastStatus;
|
||||||
int get lastLatencyMs => _lastLatencyMs;
|
int get lastLatencyMs => _lastLatencyMs;
|
||||||
|
|
||||||
/// Stream that emits true when connected, false when offline
|
|
||||||
Stream<bool> get isConnected =>
|
Stream<bool> get isConnected =>
|
||||||
connectivityStream.map((status) => status == ConnectivityStatus.online);
|
connectivityStream.map((status) => status == ConnectivityStatus.online);
|
||||||
|
|
||||||
/// Check if currently connected
|
|
||||||
bool get isCurrentlyConnected => _lastStatus == ConnectivityStatus.online;
|
bool get isCurrentlyConnected => _lastStatus == ConnectivityStatus.online;
|
||||||
|
|
||||||
void _startConnectivityMonitoring() {
|
void _startConnectivityMonitoring() {
|
||||||
// Initial check after a brief delay to avoid showing offline during startup
|
_initialCheckTimer = Timer(const Duration(milliseconds: 800), () {
|
||||||
Timer(const Duration(milliseconds: 800), () {
|
unawaited(_runConnectivityCheck(force: true));
|
||||||
_checkConnectivity();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// Check periodically; interval adapts to recent failures
|
_connectivitySubscription = _connectivity.onConnectivityChanged.listen((
|
||||||
_connectivityTimer = Timer.periodic(_interval, (_) {
|
results,
|
||||||
_checkConnectivity();
|
) {
|
||||||
|
unawaited(_handleConnectivityChange(results));
|
||||||
|
});
|
||||||
|
|
||||||
|
unawaited(
|
||||||
|
_connectivity.checkConnectivity().then(
|
||||||
|
(results) => _handleConnectivityChange(results),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _runConnectivityCheck({bool force = false}) async {
|
||||||
|
if (_connectivityController.isClosed) return;
|
||||||
|
|
||||||
|
if (!_hasNetwork) {
|
||||||
|
_lastLatencyMs = -1;
|
||||||
|
_updateStatus(ConnectivityStatus.offline);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
_initialCheckTimer?.cancel();
|
||||||
|
_initialCheckTimer = null;
|
||||||
|
_cancelScheduledPoll();
|
||||||
|
|
||||||
|
final existingCheck = _activeCheck;
|
||||||
|
if (existingCheck != null) {
|
||||||
|
if (force) {
|
||||||
|
_queuedImmediateCheck = true;
|
||||||
|
}
|
||||||
|
await existingCheck.future;
|
||||||
|
if (force && _queuedImmediateCheck) {
|
||||||
|
_queuedImmediateCheck = false;
|
||||||
|
await _runConnectivityCheck(force: false);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
final completer = Completer<void>();
|
||||||
|
_activeCheck = completer;
|
||||||
|
|
||||||
|
if (_lastStatus != ConnectivityStatus.checking) {
|
||||||
|
_updateStatus(ConnectivityStatus.checking);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await _checkConnectivity();
|
||||||
|
} finally {
|
||||||
|
completer.complete();
|
||||||
|
_activeCheck = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_queuedImmediateCheck) {
|
||||||
|
_queuedImmediateCheck = false;
|
||||||
|
await _runConnectivityCheck(force: false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
_scheduleNextPoll();
|
||||||
|
}
|
||||||
|
|
||||||
|
void _scheduleNextPoll() {
|
||||||
|
if (_connectivityController.isClosed || !_hasNetwork) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
_pollTimer = Timer(_interval, () {
|
||||||
|
_pollTimer = null;
|
||||||
|
unawaited(_runConnectivityCheck());
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void _cancelScheduledPoll() {
|
||||||
|
_pollTimer?.cancel();
|
||||||
|
_pollTimer = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool _haveSameConnectivity(
|
||||||
|
List<ConnectivityResult> previous,
|
||||||
|
List<ConnectivityResult> current,
|
||||||
|
) {
|
||||||
|
if (identical(previous, current)) return true;
|
||||||
|
if (previous.length != current.length) return false;
|
||||||
|
final previousSet = previous.toSet();
|
||||||
|
final currentSet = current.toSet();
|
||||||
|
if (previousSet.length != currentSet.length) return false;
|
||||||
|
for (final value in previousSet) {
|
||||||
|
if (!currentSet.contains(value)) return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _handleConnectivityChange(
|
||||||
|
List<ConnectivityResult> results,
|
||||||
|
) async {
|
||||||
|
if (_connectivityController.isClosed) return;
|
||||||
|
|
||||||
|
final previousResults = _lastConnectivityResults;
|
||||||
|
_lastConnectivityResults = results;
|
||||||
|
final hadNetwork = _hasNetwork;
|
||||||
|
_hasNetwork = results.any((result) => result != ConnectivityResult.none);
|
||||||
|
|
||||||
|
if (!_hasNetwork) {
|
||||||
|
_lastLatencyMs = -1;
|
||||||
|
_queuedImmediateCheck = false;
|
||||||
|
_cancelScheduledPoll();
|
||||||
|
_initialCheckTimer?.cancel();
|
||||||
|
_initialCheckTimer = null;
|
||||||
|
_updateStatus(ConnectivityStatus.offline);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
final networkTypeChanged = previousResults == null
|
||||||
|
? true
|
||||||
|
: !_haveSameConnectivity(previousResults, results);
|
||||||
|
|
||||||
|
if (!hadNetwork ||
|
||||||
|
_lastStatus == ConnectivityStatus.offline ||
|
||||||
|
networkTypeChanged) {
|
||||||
|
unawaited(_runConnectivityCheck(force: true));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Future<void> _checkConnectivity() async {
|
Future<void> _checkConnectivity() async {
|
||||||
// Don't check connectivity if service is disposed
|
|
||||||
if (_connectivityController.isClosed) return;
|
if (_connectivityController.isClosed) return;
|
||||||
|
|
||||||
final serverReachability = await _probeActiveServer();
|
final serverReachability = await _probeActiveServer();
|
||||||
@@ -75,7 +201,6 @@ class ConnectivityService {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// No configured server to probe; assume usable connectivity so setup flows continue.
|
|
||||||
_lastLatencyMs = -1;
|
_lastLatencyMs = -1;
|
||||||
_updateStatus(ConnectivityStatus.online);
|
_updateStatus(ConnectivityStatus.online);
|
||||||
}
|
}
|
||||||
@@ -83,13 +208,11 @@ class ConnectivityService {
|
|||||||
void _updateStatus(ConnectivityStatus status) {
|
void _updateStatus(ConnectivityStatus status) {
|
||||||
if (_lastStatus != status) {
|
if (_lastStatus != status) {
|
||||||
_lastStatus = status;
|
_lastStatus = status;
|
||||||
// Only add to stream if controller is not closed
|
|
||||||
if (!_connectivityController.isClosed) {
|
if (!_connectivityController.isClosed) {
|
||||||
_connectivityController.add(status);
|
_connectivityController.add(status);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Adapt polling interval based on recent failures to reduce battery/CPU
|
|
||||||
if (status == ConnectivityStatus.offline) {
|
if (status == ConnectivityStatus.offline) {
|
||||||
_recentFailures = (_recentFailures + 1).clamp(0, 10);
|
_recentFailures = (_recentFailures + 1).clamp(0, 10);
|
||||||
} else if (status == ConnectivityStatus.online) {
|
} else if (status == ConnectivityStatus.online) {
|
||||||
@@ -104,25 +227,25 @@ class ConnectivityService {
|
|||||||
|
|
||||||
if (newInterval != _interval) {
|
if (newInterval != _interval) {
|
||||||
_interval = newInterval;
|
_interval = newInterval;
|
||||||
_connectivityTimer?.cancel();
|
_cancelScheduledPoll();
|
||||||
// Only create new timer if service is not disposed
|
if (_lastStatus != ConnectivityStatus.offline && _hasNetwork) {
|
||||||
if (!_connectivityController.isClosed) {
|
_scheduleNextPoll();
|
||||||
_connectivityTimer = Timer.periodic(
|
|
||||||
_interval,
|
|
||||||
(_) => _checkConnectivity(),
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<bool> checkConnectivity() async {
|
Future<bool> checkConnectivity() async {
|
||||||
await _checkConnectivity();
|
await _runConnectivityCheck(force: true);
|
||||||
return _lastStatus == ConnectivityStatus.online;
|
return _lastStatus == ConnectivityStatus.online;
|
||||||
}
|
}
|
||||||
|
|
||||||
void dispose() {
|
void dispose() {
|
||||||
_connectivityTimer?.cancel();
|
_initialCheckTimer?.cancel();
|
||||||
_connectivityTimer = null;
|
_initialCheckTimer = null;
|
||||||
|
_cancelScheduledPoll();
|
||||||
|
_connectivitySubscription?.cancel();
|
||||||
|
_connectivitySubscription = null;
|
||||||
|
_activeCheck = null;
|
||||||
if (!_connectivityController.isClosed) {
|
if (!_connectivityController.isClosed) {
|
||||||
_connectivityController.close();
|
_connectivityController.close();
|
||||||
}
|
}
|
||||||
@@ -151,14 +274,15 @@ class ConnectivityService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Future<bool?> _probeBaseEndpoint(
|
Future<bool?> _probeBaseEndpoint(
|
||||||
Uri uri, {
|
Uri baseUri, {
|
||||||
bool updateLatency = false,
|
bool updateLatency = false,
|
||||||
}) async {
|
}) async {
|
||||||
try {
|
try {
|
||||||
final start = DateTime.now();
|
final start = DateTime.now();
|
||||||
|
final healthUri = baseUri.resolve('/health');
|
||||||
final response = await _dio
|
final response = await _dio
|
||||||
.getUri(
|
.getUri(
|
||||||
uri,
|
healthUri,
|
||||||
options: Options(
|
options: Options(
|
||||||
method: 'GET',
|
method: 'GET',
|
||||||
sendTimeout: const Duration(seconds: 3),
|
sendTimeout: const Duration(seconds: 3),
|
||||||
@@ -175,7 +299,6 @@ class ConnectivityService {
|
|||||||
}
|
}
|
||||||
return isHealthy;
|
return isHealthy;
|
||||||
} catch (_) {
|
} catch (_) {
|
||||||
// Treat as unreachable.
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -205,27 +328,22 @@ class ConnectivityService {
|
|||||||
}
|
}
|
||||||
if (parsed == null) return null;
|
if (parsed == null) return null;
|
||||||
|
|
||||||
// Return the base URL directly instead of resolving to /health
|
|
||||||
return parsed;
|
return parsed;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Providers
|
|
||||||
final connectivityServiceProvider = Provider<ConnectivityService>((ref) {
|
final connectivityServiceProvider = Provider<ConnectivityService>((ref) {
|
||||||
// Create a Dio instance with custom headers from the active server config
|
|
||||||
final activeServer = ref.watch(activeServerProvider);
|
final activeServer = ref.watch(activeServerProvider);
|
||||||
|
|
||||||
return activeServer.maybeWhen(
|
return activeServer.maybeWhen(
|
||||||
data: (server) {
|
data: (server) {
|
||||||
if (server == null) {
|
if (server == null) {
|
||||||
// No server configured, use lightweight Dio
|
|
||||||
final dio = Dio();
|
final dio = Dio();
|
||||||
final service = ConnectivityService(dio, ref);
|
final service = ConnectivityService(dio, ref);
|
||||||
ref.onDispose(() => service.dispose());
|
ref.onDispose(() => service.dispose());
|
||||||
return service;
|
return service;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create Dio with custom headers from server config
|
|
||||||
final dio = Dio(
|
final dio = Dio(
|
||||||
BaseOptions(
|
BaseOptions(
|
||||||
baseUrl: server.url,
|
baseUrl: server.url,
|
||||||
@@ -234,7 +352,6 @@ final connectivityServiceProvider = Provider<ConnectivityService>((ref) {
|
|||||||
followRedirects: true,
|
followRedirects: true,
|
||||||
maxRedirects: 5,
|
maxRedirects: 5,
|
||||||
validateStatus: (status) => status != null && status < 400,
|
validateStatus: (status) => status != null && status < 400,
|
||||||
// Add custom headers from server config
|
|
||||||
headers: server.customHeaders.isNotEmpty
|
headers: server.customHeaders.isNotEmpty
|
||||||
? Map<String, String>.from(server.customHeaders)
|
? Map<String, String>.from(server.customHeaders)
|
||||||
: null,
|
: null,
|
||||||
@@ -246,7 +363,6 @@ final connectivityServiceProvider = Provider<ConnectivityService>((ref) {
|
|||||||
return service;
|
return service;
|
||||||
},
|
},
|
||||||
orElse: () {
|
orElse: () {
|
||||||
// No server data available, use lightweight Dio
|
|
||||||
final dio = Dio();
|
final dio = Dio();
|
||||||
final service = ConnectivityService(dio, ref);
|
final service = ConnectivityService(dio, ref);
|
||||||
ref.onDispose(() => service.dispose());
|
ref.onDispose(() => service.dispose());
|
||||||
@@ -280,17 +396,12 @@ class ConnectivityStatusNotifier extends _$ConnectivityStatusNotifier {
|
|||||||
}
|
}
|
||||||
|
|
||||||
final isOnlineProvider = Provider<bool>((ref) {
|
final isOnlineProvider = Provider<bool>((ref) {
|
||||||
// In reviewer mode, treat app as online to enable flows
|
|
||||||
final reviewerMode = ref.watch(reviewerModeProvider);
|
final reviewerMode = ref.watch(reviewerModeProvider);
|
||||||
if (reviewerMode) return true;
|
if (reviewerMode) return true;
|
||||||
final status = ref.watch(connectivityStatusProvider);
|
final status = ref.watch(connectivityStatusProvider);
|
||||||
return status.when(
|
return status.when(
|
||||||
data: (status) => status != ConnectivityStatus.offline,
|
data: (status) => status != ConnectivityStatus.offline,
|
||||||
loading: () => true, // Assume online while checking
|
loading: () => true,
|
||||||
error: (_, _) =>
|
error: (error, _) => true,
|
||||||
true, // Assume online on error to avoid false offline states
|
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Dio provider (if not already defined elsewhere)
|
|
||||||
// Removed unused Dio provider to avoid confusion. Use ApiService instead.
|
|
||||||
|
|||||||
24
pubspec.lock
24
pubspec.lock
@@ -225,6 +225,22 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "1.19.1"
|
version: "1.19.1"
|
||||||
|
connectivity_plus:
|
||||||
|
dependency: "direct main"
|
||||||
|
description:
|
||||||
|
name: connectivity_plus
|
||||||
|
sha256: "33bae12a398f841c6cda09d1064212957265869104c478e5ad51e2fb26c3973c"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "7.0.0"
|
||||||
|
connectivity_plus_platform_interface:
|
||||||
|
dependency: transitive
|
||||||
|
description:
|
||||||
|
name: connectivity_plus_platform_interface
|
||||||
|
sha256: "42657c1715d48b167930d5f34d00222ac100475f73d10162ddf43e714932f204"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "2.0.1"
|
||||||
convert:
|
convert:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -917,6 +933,14 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "1.0.0"
|
version: "1.0.0"
|
||||||
|
nm:
|
||||||
|
dependency: transitive
|
||||||
|
description:
|
||||||
|
name: nm
|
||||||
|
sha256: "2c9aae4127bdc8993206464fcc063611e0e36e72018696cd9631023a31b24254"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "0.5.0"
|
||||||
node_preamble:
|
node_preamble:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
|
|||||||
@@ -68,6 +68,7 @@ dependencies:
|
|||||||
share_handler: ^0.0.19
|
share_handler: ^0.0.19
|
||||||
riverpod_annotation: ^3.0.0
|
riverpod_annotation: ^3.0.0
|
||||||
flutter_local_notifications: ^19.4.2
|
flutter_local_notifications: ^19.4.2
|
||||||
|
connectivity_plus: ^7.0.0
|
||||||
|
|
||||||
# Clipboard functionality is available through flutter/services (part of Flutter SDK)
|
# Clipboard functionality is available through flutter/services (part of Flutter SDK)
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user