taxgilde/lib/consts/details_webscokect.dart
2026-04-11 10:21:31 +05:30

241 lines
6.0 KiB
Dart

import 'dart:async';
import 'dart:convert';
import 'package:flutter/foundation.dart';
import 'package:web_socket_channel/web_socket_channel.dart';
/// Common WebSocket for user-based communication (not chatId-based)
class DetailsWebscokect {
static final DetailsWebscokect _instance =
DetailsWebscokect._internal();
factory DetailsWebscokect() => _instance;
DetailsWebscokect._internal();
WebSocketChannel? _channel;
StreamSubscription? _subscription;
Timer? _pingTimer;
Timer? _reconnectTimer;
bool _isConnected = false;
bool _isIntentionalClose = false;
int _reconnectAttempts = 0;
static const int _maxReconnectAttempts = 5;
static const Duration _reconnectDelay = Duration(seconds: 3);
// Callbacks
Function(Map<String, dynamic>)? onMessageReceived;
Function()? onConnected;
Function()? onDisconnected;
Function(dynamic error)? onError;
String? _userId;
String? _wsUrl;
/// Check connection status
bool get isConnected => _isConnected;
/// Current user ID
String? get currentUserId => _userId;
/// Connect WS using LOCAL USER ID
Future<void> connect({
required String userId,
String baseUrl = "wss://taxglide.amrithaa.net:443/reverb",
String appKey = "2yj0lyzc9ylw2h03ts6i",
}) async {
try {
await disconnect();
_userId = userId;
_wsUrl = "$baseUrl/app/$appKey?protocol=7&client=js&version=1.0";
_isIntentionalClose = false;
_reconnectAttempts = 0;
await _establishConnection();
} catch (e) {
debugPrint("❌ Connection error: $e");
onError?.call(e);
_scheduleReconnect();
}
}
Future<void> _establishConnection() async {
try {
debugPrint("🔌 Connecting to: $_wsUrl");
_channel = WebSocketChannel.connect(Uri.parse(_wsUrl!));
_subscription = _channel!.stream.listen(
_handleMessage,
onError: _handleError,
onDone: _handleDisconnection,
cancelOnError: false,
);
await Future.delayed(const Duration(milliseconds: 500));
_isConnected = true;
_reconnectAttempts = 0;
debugPrint("✅ WebSocket connected");
_subscribeToUserChannel();
_startPingTimer();
onConnected?.call();
} catch (e) {
debugPrint("❌ Failed to establish connection: $e");
_isConnected = false;
rethrow;
}
}
/// Handle incoming messages
void _handleMessage(dynamic data) {
try {
debugPrint("📨 Received: $data");
final Map<String, dynamic> message = json.decode(data);
if (message['event'] == 'pusher:ping') {
_sendPong();
return;
}
if (message['event'] == 'pusher:connection_established') {
debugPrint("🔗 Pusher connection established");
return;
}
if (message['event'] == 'pusher_internal:subscription_succeeded') {
debugPrint("✅ User channel subscribed");
return;
}
if (message['event'] != null &&
!message['event'].toString().startsWith('pusher')) {
onMessageReceived?.call(message);
}
} catch (e) {
debugPrint("❌ Error parsing message: $e");
onError?.call(e);
}
}
void _handleError(dynamic error) {
debugPrint("❌ WebSocket error: $error");
_isConnected = false;
onError?.call(error);
if (!_isIntentionalClose) {
_scheduleReconnect();
}
}
void _handleDisconnection() {
debugPrint("🔌 Disconnected");
_isConnected = false;
_stopPingTimer();
onDisconnected?.call();
if (!_isIntentionalClose) {
_scheduleReconnect();
}
}
/// Subscribe to USER CHANNEL
void _subscribeToUserChannel() {
if (_userId == null) return;
final subscribeMessage = {
'event': 'pusher:subscribe',
'data': {'channel': 'user-chat-notification.$_userId'},
};
_sendMessage(subscribeMessage);
debugPrint("📡 Subscribing to user.$_userId");
}
void _sendPong() {
final pong = {'event': 'pusher:pong', 'data': {}};
_sendMessage(pong);
}
void _startPingTimer() {
_pingTimer?.cancel();
_pingTimer = Timer.periodic(const Duration(seconds: 30), (timer) {
if (_isConnected) debugPrint("💓 WS heartbeat");
});
}
void _stopPingTimer() {
_pingTimer?.cancel();
_pingTimer = null;
}
void _scheduleReconnect() {
if (_isIntentionalClose) return;
if (_reconnectAttempts >= _maxReconnectAttempts) {
onError?.call(
"Failed to reconnect after $_maxReconnectAttempts attempts",
);
return;
}
_reconnectTimer?.cancel();
_reconnectAttempts++;
debugPrint(
"🔄 Retry $_reconnectAttempts in ${_reconnectDelay.inSeconds}s...",
);
_reconnectTimer = Timer(_reconnectDelay, () async {
if (!_isIntentionalClose && _wsUrl != null && _userId != null) {
await _establishConnection();
}
});
}
/// Send message
void _sendMessage(Map<String, dynamic> message) {
if (!_isConnected || _channel == null) {
debugPrint("⚠️ Cannot send: Not connected");
return;
}
try {
final jsonMsg = json.encode(message);
_channel!.sink.add(jsonMsg);
debugPrint("📤 Sent: $jsonMsg");
} catch (e) {
debugPrint("❌ Send error: $e");
}
}
/// Send event to backend
void sendEvent({
required String eventName,
required Map<String, dynamic> data,
}) {
final msg = {
'event': eventName,
'data': data,
'channel': 'user-chat-notification.$_userId',
};
_sendMessage(msg);
}
Future<void> disconnect() async {
_isIntentionalClose = true;
_reconnectTimer?.cancel();
_stopPingTimer();
await _subscription?.cancel();
await _channel?.sink.close();
_channel = null;
_subscription = null;
_isConnected = false;
_userId = null;
debugPrint("🔌 WS disconnected intentionally");
}
}