Ios Voip Kit

development_aamir
Aamir Muhammad 2 years ago
parent 364d1292a0
commit 23820e7e5e

@ -33,8 +33,8 @@ class ChatApiClient {
"isMobile": true,
"deviceToken": AppState().getIsHuawei ? AppState().getHuaweiPushToken : AppState().getDeviceToken,
"isHuaweiDevice": AppState().getIsHuawei,
//"platform" : "", // ios, android
//"voipToken": ""
"platform": Platform.isIOS ? "ios" : "android", // ios, android
"voipToken": Platform.isIOS ? AppState().iosVoipPlayerID : ""
},
);
@ -188,6 +188,7 @@ class ChatApiClient {
}
return imagesData;
}
// CallUser Login Token
Future<user.UserAutoLoginModel> getUserCallToken({required String userid}) async {
@ -216,9 +217,6 @@ class ChatApiClient {
// Call Decline On App Terminated State
Future<Response> callDecline({required int cUserID, required int tUserID, required String targetUsertoken}) async {
// var headers = {'Content-Type': 'application/json'};
// var request = http.Request('POST', Uri.parse('https://apiderichat.hmg.com/api/user/calldecline'));
Response response = await ApiClient().postJsonForResponse(
"${ApiConsts.chatLoginTokenUrl}calldecline",
{"currentUserId": cUserID, "targetUserId": tUserID, "secretKey": "derichatmobileuser", "targetUserToken": targetUsertoken},
@ -231,14 +229,18 @@ class ChatApiClient {
return response;
}
Future<Response> OneSignalVoip({required String deviceID}) async {
Future<String> oneSignalVoip(String value) async {
String id = "";
Response response = await ApiClient().postJsonForResponse(
"${ApiConsts.oneSignalCall}players",
{"app_id": ApiConsts.oneSignalAppID, "identifier": deviceID, "device_type": 0, "test_type": !kReleaseMode ? 0 : 1},
{"app_id": ApiConsts.oneSignalAppID, "identifier": value, "device_type": 0, "test_type": !kReleaseMode ? 1 : 0},
);
Map<String, dynamic> values = jsonDecode(response.body) as Map<String, dynamic>;
id = values["id"];
if (!kReleaseMode) {
print("res: " + response.body);
}
return response;
return id;
}
}

@ -193,4 +193,15 @@ class AppState {
}
bool cancelRequestTrancsection = true;
String _iosVoipPlayerID = "";
String get iosVoipPlayerID => _iosVoipPlayerID;
set setiosVoipPlayerID(String value) {
_iosVoipPlayerID = value;
}
}

@ -154,6 +154,15 @@ class ChatVoipCall {
// logger.log(Level.error, "API-EVENT-END");
}
Future<void> voipDeclineCall(IosCallPayload? _iosCallPayload ) async {
try {
ALM.UserAutoLoginModel model = await ChatApiClient().getUserCallToken(userid: _iosCallPayload!.callData!.split("-").first);
dynamic Res = await ChatApiClient().callDecline(cUserID: int.parse(_iosCallPayload!.incomingCallerId!), tUserID: int.parse(_iosCallPayload!.callData!.split("-").first), targetUsertoken: model.response!.token!);
} catch (err) {
print(err);
}
}
Future<HubConnection> makeHub({required IncomingCallModel sessionData}) async {
late HubConnection hc;
try {
@ -168,3 +177,4 @@ class ChatVoipCall {
}
}
}

@ -139,12 +139,14 @@ class IosCallPayload {
String? incomingCallType;
String? incomingCallerId;
String? incomingCallerName;
String? callData;
String? uuid;
IosCallPayload({
this.incomingCallType,
this.incomingCallerId,
this.incomingCallerName,
this.callData,
this.uuid,
});
@ -156,6 +158,7 @@ class IosCallPayload {
incomingCallType: json["incoming_call_type"],
incomingCallerId: json["incoming_caller_id"],
incomingCallerName: json["incoming_caller_name"],
callData: json["call_data"],
uuid: json["uuid"],
);
@ -163,6 +166,7 @@ class IosCallPayload {
"incoming_call_type": incomingCallType,
"incoming_caller_id": incomingCallerId,
"incoming_caller_name": incomingCallerName,
"call_data": callData,
"uuid": uuid,
};
}

@ -47,6 +47,9 @@ class ChatCallProvider with ChangeNotifier, DiagnosticableTreeMixin {
late BuildContext providerContext;
List<MediaDeviceInfo> devices = [];
var _videoDeviceId ;
void initCallListeners({required BuildContext context}) {
providerContext = context;
if (kDebugMode) {
@ -62,6 +65,7 @@ class ChatCallProvider with ChangeNotifier, DiagnosticableTreeMixin {
}
//Video Constraints
Map<String, Object> videoConstraints = {
"video": {
"mandatory": {
@ -69,6 +73,7 @@ class ChatCallProvider with ChangeNotifier, DiagnosticableTreeMixin {
"height": {"min": 720}
},
"optional": [
// {'sourceId': _videoDeviceId},
{
"width": {"max": 1280}
},
@ -80,7 +85,7 @@ class ChatCallProvider with ChangeNotifier, DiagnosticableTreeMixin {
"width": 1280, //420,//640,//1280,
"height": 720, //240//480//720
"audio": true,
};
};
// Audio Constraints
Map<String, Object> audioConstraints = {
@ -584,9 +589,12 @@ class ChatCallProvider with ChangeNotifier, DiagnosticableTreeMixin {
///////////////// Incoming Call ///////////////////////////////
Future<void> initStreams() async {
devices = await navigator.mediaDevices.enumerateDevices();
localVideoRenderer = RTCVideoRenderer();
remoteRenderer = RTCVideoRenderer();
await localVideoRenderer!.initialize();
_localStream ??= await navigator.mediaDevices.getUserMedia(isVideoCall ? videoConstraints : audioConstraints);
localVideoRenderer!.srcObject = _localStream;
await remoteRenderer!.initialize();

@ -107,7 +107,6 @@ class ChatProviderModel with ChangeNotifier, DiagnosticableTreeMixin {
} catch (e) {
disableChatForThisUser = true;
isUserOnline = false;
notifyListeners();
}
}

@ -8,10 +8,15 @@ import 'package:flutter/material.dart';
import 'package:flutter_callkit_incoming/flutter_callkit_incoming.dart';
import 'package:flutter_svg/flutter_svg.dart';
import 'package:flutter_webrtc/flutter_webrtc.dart';
import 'package:mohem_flutter_app/api/chat/chat_api_client.dart';
import 'package:mohem_flutter_app/app_state/app_state.dart';
import 'package:mohem_flutter_app/classes/colors.dart';
import 'package:mohem_flutter_app/classes/consts.dart';
import 'package:mohem_flutter_app/classes/utils.dart';
import 'package:mohem_flutter_app/extensions/int_extensions.dart';
import 'package:mohem_flutter_app/main.dart';
import 'package:mohem_flutter_app/models/chat/call.dart';
import 'package:mohem_flutter_app/models/chat/get_single_user_chat_list_model.dart';
import 'package:mohem_flutter_app/models/chat/get_user_login_token_model.dart';
import 'package:mohem_flutter_app/models/chat/incoming_call_model.dart';
import 'package:mohem_flutter_app/provider/chat_call_provider.dart';
@ -75,7 +80,47 @@ class _StartCallPageState extends State<StartCallPage> {
}
}
void startIosCall() {}
void startIosCall() async {
IosCallPayload _iosCallPayload = IosCallPayload.fromRawJson(await Utils.getStringFromPrefs("iosCallPayload"));
var userID = _iosCallPayload!.callData!.split("-").first;
var callType = _iosCallPayload!.callData!.split("-").last;
SingleUserChatModel inCallData = SingleUserChatModel(
targetUserName: _iosCallPayload.incomingCallerName,
chatEventId: 3,
targetUserId: int.parse(_iosCallPayload.incomingCallerId!),
currentUserId: int.parse(userID),
);
if (provider.isUserOnline) {
cProv.isUserOnline = provider.isUserOnline;
if (kDebugMode) {
print("====== Processing Incoming Call in Online State =========");
}
await cProv.startIncomingCallViaKit(inCallData: inCallData.toJson(), isVCall: callType == "video" ? true : false);
cProv.init();
isCallConnected = true;
} else {
if (kDebugMode) {
print("====== Processing Incoming Call =========");
}
cProv.isUserOnline = provider.isUserOnline;
UserAutoLoginModel userLoginResponse = await ChatApiClient().getUserCallToken(userid: userID);
if (userLoginResponse.response != null) {
AppState().setchatUserDetails = userLoginResponse;
Utils.saveStringFromPrefs("userLoginChatDetails", jsonEncode(userLoginResponse.response));
await cProv.startIncomingCallViaKit(inCallData: inCallData.toJson(), isVCall: callType == "video" ? true : false);
try {
AppState().setchatUserDetails = UserAutoLoginModel(response: userLoginResponse.response, errorResponses: null);
await provider.buildHubConnection(context: context, ccProvider: cProv).whenComplete(() {
cProv.init();
isCallConnected = true;
});
} catch (e) {
logger.w(e);
}
}
}
}
@override
Widget build(BuildContext context) {

@ -55,7 +55,7 @@ class LoginScreen extends StatefulWidget {
}
}
class _LoginScreenState extends State<LoginScreen> with WidgetsBindingObserver{
class _LoginScreenState extends State<LoginScreen> with WidgetsBindingObserver {
TextEditingController username = TextEditingController();
TextEditingController password = TextEditingController();
@ -65,7 +65,6 @@ class _LoginScreenState extends State<LoginScreen> with WidgetsBindingObserver{
late final FirebaseMessaging _firebaseMessaging;
IosCallPayload? _iosCallPayload;
bool _autoLogin = false;
bool? isAppOpenBySystem;
@ -81,7 +80,6 @@ class _LoginScreenState extends State<LoginScreen> with WidgetsBindingObserver{
final voIPKit = FlutterIOSVoIPKit.instance;
late Timer timeOutTimer;
@override
void initState() {
super.initState();
@ -98,38 +96,25 @@ class _LoginScreenState extends State<LoginScreen> with WidgetsBindingObserver{
setupVoIPCallBacks();
}
void _timeOut({
int seconds = 15,
}) async {
timeOutTimer = Timer(Duration(seconds: seconds), () async {
print('🎈 example: timeOut');
var incomingCallerName = await voIPKit.getIncomingCallerName();
voIPKit.unansweredIncomingCall(
skipLocalNotification: false,
missedCallTitle: '📞 Missed call',
missedCallBody: 'There was a call from $incomingCallerName',
);
});
}
// IOS Voip Call
void setupVoIPCallBacks() {
if (Platform.isIOS) {
voIPKit.getVoIPToken().then((value) {
print('🎈 example: getVoIPToken: $value');
ChatApiClient().OneSignalVoip(deviceID: value!);
voIPKit.getVoIPToken().then((value) async {
if (value != null) {
AppState().setiosVoipPlayerID = await ChatApiClient().oneSignalVoip(value!);
}
});
}
voIPKit.onDidUpdatePushToken = (String token) {
print('🎈 example: onDidUpdatePushToken: $token');
};
}
voIPKit.onDidReceiveIncomingPush = (
Map<String, dynamic> payload,
) async {
_iosCallPayload = IosCallPayload.fromJson(payload);
logger.d(_iosCallPayload!.incomingCallerId!.split("-").last);
print('🎈 example: onDidReceiveIncomingPush $payload');
isIncomingCall = true;
_timeOut();
};
@ -137,9 +122,8 @@ class _LoginScreenState extends State<LoginScreen> with WidgetsBindingObserver{
String uuid,
String callerId,
) async {
try {
var logText = "did reject call $callerId";
} catch (err) {}
await ChatVoipCall().voipDeclineCall(_iosCallPayload);
await voIPKit.endCall();
};
voIPKit.onDidAcceptIncomingCall = (
@ -147,33 +131,35 @@ class _LoginScreenState extends State<LoginScreen> with WidgetsBindingObserver{
String callerId,
) async {
var callerID = "did accept call $callerId";
debugPrint(callerID);
logger.d(_iosCallPayload!.incomingCallerId!.split("-").last);
debugPrint(_iosCallPayload!.incomingCallerId);
debugPrint(_iosCallPayload!.incomingCallerName);
debugPrint(_iosCallPayload!.incomingCallType);
timeOutTimer.cancel();
await connectCall();
await voIPKit.acceptIncomingCall(callerState: CallStateType.calling);
await voIPKit.callConnected();
Future.delayed(const Duration(milliseconds: 2500), () {
voIPKit.endCall().then((value) async {});
});
timeOutTimer.cancel();
};
}
void _timeOut() async {
timeOutTimer = Timer(const Duration(seconds: 25), () async {
var incomingCallerName = await voIPKit.getIncomingCallerName();
voIPKit.unansweredIncomingCall(
skipLocalNotification: false,
missedCallTitle: '📞 Missed call',
missedCallBody: 'There was a call from $incomingCallerName',
);
await ChatVoipCall().voipDeclineCall(_iosCallPayload);
await voIPKit.endCall();
});
}
Future<void> connectCall() async {
try {
UserAutoLoginModel userLoginResponse = await ChatApiClient().getUserCallToken(userid: _iosCallPayload!.incomingCallerId!.split("-").last);
if (userLoginResponse.response != null) {
AppState().setchatUserDetails = userLoginResponse;
Utils.saveStringFromPrefs("userLoginChatDetails", jsonEncode(userLoginResponse.response));
}
BuildContext context = AppRoutes.navigatorKey.currentContext!;
Utils.hideLoading(context);
Utils.saveStringFromPrefs("iosCallPayload", jsonEncode(_iosCallPayload));
var pageRoute = MaterialPageRoute(builder: (context) => StartCallPage());
Navigator.push(context, pageRoute).whenComplete(() {
checkFirebaseToken();
});
} catch (e) {
logger.d(e);
}
@ -206,13 +192,11 @@ class _LoginScreenState extends State<LoginScreen> with WidgetsBindingObserver{
// }
// }
Future<void> callListeners() async {
try {
print("Call Listeners Init");
FlutterCallkitIncoming.onEvent.listen((CallEvent? event) async {
switch (event!.event) {
case Event.actionCallIncoming:
break;
case Event.actionCallStart:
@ -239,10 +223,8 @@ class _LoginScreenState extends State<LoginScreen> with WidgetsBindingObserver{
Utils.saveStringFromPrefs("inComingCallData", "null");
FlutterCallkitIncoming.endAllCalls();
break;
}
print( '${event.toString()}\n');
print('${event.toString()}\n');
});
} on Exception {
logger.log(Level.info, "EXCEPTION-ON-EVENTS");
@ -289,9 +271,6 @@ class _LoginScreenState extends State<LoginScreen> with WidgetsBindingObserver{
}
}
@override
void dispose() {
super.dispose();

@ -76,7 +76,7 @@ class _OffersAndDiscountsDetailsState extends State<OffersAndDiscountsDetails> {
: getOffersList[0].titleEn!.toText22(isBold: true, color: const Color(0xff2B353E)).center,
Html(
data: AppState().isArabic(context) ? getOffersList[0].descriptionAr! : getOffersList[0].descriptionEn ?? "",
onLinkTap: (String? url, RenderContext context, Map<String, String> attributes, _) {
onLinkTap: (String? url, Map<String, String> attributes, _) {
launchUrl(Uri.parse(url!));
},
),

@ -87,7 +87,7 @@ dependencies:
signalr_netcore: ^1.3.3
logging: ^1.0.1
swipe_to: ^1.0.2
flutter_webrtc: ^0.9.20
flutter_webrtc: ^0.9.34
draggable_widget: ^2.0.0
flutter_local_notifications: any
flutter_callkit_incoming: ^2.0.0+1
@ -119,7 +119,6 @@ dependencies:
google_api_availability: ^3.0.1
flutter_ios_voip_kit: ^0.1.0
file: ^6.1.4
dependency_overrides:

Loading…
Cancel
Save