Skip to main content

Integrate Voip call to your project

N|Solid

react-native-pitel-voip is package support for voip call.

Demo

register ougoing_call

Pitel Flow

When user make call from Pitel app, Pitel Server pushes a notification for all user login (who receives the call). When user "Accept" call, extension will re-register to receive call. Pitel Flow

Features

  • Register Extension
  • Call
  • Hangup
  • Turn on/off micro
  • Turn on/of speaker

Installation

  1. Install Packages Add pubspec.yaml:
yarn add react-native-pitel-voip
  1. Installing dependencies into a bare React Native project
yarn add react-native-callkeep@4.3.9 @react-native-firebase/app@18.1.0 @react-native-firebase/messaging@18.1.0 react-native-background-timer@2.4.1 react-native-get-random-values@1.9.0 react-native-incall-manager@4.0.1 react-native-svg@13.9.0 react-native-voip-push-notification@3.3.1 uuid@9.0.0 pitel-react-native-webrtc pitel-sdk-for-rn @react-native-async-storage/async-storage@1.19.1
  1. Pod install
cd ios
pod install
  1. Installing react-native-svg-transformer Follow document & setup react-native-svg-transformer
yarn add --dev react-native-svg-transformer
  1. Pushkit/ Push notification - Received VoIP and Wake app from Terminated State. Note Please check PUSH_NOTIF.md. setup Pushkit (for IOS), push notification (for Android).

  2. Configure Project

Android:

  • In file android/app/src/main/AndroidManifest.xml. Example
 <manifest...>
...
// Request permission
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.CAMERA" />
<uses-permission android:name="android.permission.RECORD_AUDIO" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
<uses-permission android:name="android.permission.BLUETOOTH" />
<uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS" />

...
// show when lock screen
<application
...
>
<activity
...
android:showOnLockScreen="true"
android:showWhenLocked="true"
android:turnScreenOn="true"
>
...
</activity>
</application>
</manifest>
  • In file android/gradle.properties
android.enableDexingArtifactTransform.desugaring=false

IOS

  • Request permission in file Info.plist
<key>NSMicrophoneUsageDescription</key>
<string>Need microphone access for voip call</string>
<array>
<string>fetch</string>
<string>processing</string>
<string>remote-notification</string>
<string>voip</string>
</array>
  • Make sure platform ios 12.0 in Podfile
platform :ios, '12.0'
target 'rn_pitel_demo' do
config = use_native_modules!

// Add this
pod 'Firebase', :modular_headers => true
pod 'FirebaseCoreInternal', :modular_headers => true
pod 'GoogleUtilities', :modular_headers => true
pod 'FirebaseCore', :modular_headers => true

Example

Please checkout repo github to get example

Usage

import { NotificationBackground } from "react-native-pitel-voip"; // Add this line
NotificationBackground(); // Add this line

AppRegistry.registerComponent(appName, () => App);
// Import this
import { PitelSDKProvider } from "react-native-pitel-voip";
import BackgroundTimer from "react-native-background-timer";
BackgroundTimer.start();

export default function App() {
return (
// Wrap your app with PitelSDKProvider
<PitelSDKProvider>...</PitelSDKProvider>
);
}
  • In file src/screens/home_screen/index.js Please follow example

Config sdkOption

const ext = `${EXTENSION}`;
const sipPass = `${EXTENSION_SIP_PASSWORD}`;
const appId = `${BUNDLE_ID}`;
const domainUrl = `${DOMAIN}`;

const sdkOptionsInit = {
sipDomain: `${DOMAIN}:${PORT}`,
wssServer: `${WSS_URL}`,
sipPassword: sipPass,
bundleId: appId, // Bundle id for IOS
packageId: appId, // Package id for Android
teamId: `${TEAM_ID}`, // Team id of Apple developer account
};
  • Register device token & remove device token
const _registerDeviceToken = async () => {
const fcmToken = await getFcmToken();
const deviceToken = Platform.OS == "android" ? fcmToken : iosPushToken;
await registerDeviceToken({
pn_token: deviceToken,
pn_type: Platform.OS == "android" ? "android" : "ios",
app_id: appId,
domain: domainUrl,
extension: ext,
app_mode: __DEV__ ? "dev" : "production",
fcm_token: fcmToken,
});
};

const _removeDeviceToken = async () => {
const fcmToken = await getFcmToken();
const deviceToken = Platform.OS == "android" ? fcmToken : iosPushToken;
removeDeviceToken({
pn_token: deviceToken,
domain: domainUrl,
extension: ext,
});
};
  • Wrap your hone screen component with PitelSDK
return (
<PitelSDK
sdkOptionsInit={sdkOptionsInit}
iosPushToken={iosPushToken}
setSdkOptions={setSdkOptions}
>
<HomeScreenComponent
navigation={navigation}
sdkOptions={sdkOptions}
handleRegisterToken={_registerDeviceToken}
handleRemoveToken={_removeDeviceToken}
setIOSPushToken={setIOSPushToken}
/>
</PitelSDK>
);

Properties

PropDescriptionTypeDefault
sdkOptionsInityour extension info use to loginObjectRequired
iosPushTokenios device voip push tokenStringRequired
setSdkOptionsset sdkOption when your extension login successFunctionRequired
  • In file src/screens/home_screen/home_screen.js Example
// Register your extension to PBX
const {
callState,
receivedPhoneNumber,
registerState,

setCallState,
registerFunc,
} = useRegister({
sdkOptions: sdkOptions,
setPitelSDK: setPitelSDK,
extension: ext,
});

return (
<PitelCallNotif
callkitSetup={callkitSetup}
pitelSDK={pitelSDK}
setCallState={setCallState}
callState={callState}
isLogin={isLogin}
isCallOut={isCallOut}
setCallID={setCallID}
sdkOptions={sdkOptions}
registerFunc={registerFunc}
setIsCallOut={setIsCallOut}
onCreated={handleCreated}
onReceived={handleReceived}
onHangup={handleHangup}
onIOSToken={(iosToken) => {
setIOSPushToken(iosToken);
}}
>
...
</PitelCallNotif>
);

Properties

PropDescriptionTypeDefault
pitelSDKpitelSDK get from params of routeObjectRequired
callkitSetupset information for callkit request permissionObjectRequired
setCallStateset call status() => voidRequired
isLoginapp login/logout statusboolRequired
isCallOutcall direction statusboolRequired
setCallIDset call direction() => voidRequired
sdkOptionsreceived sdkOptions when extension login successObjectRequired
registerFuncregister extension() => voidRequired
setIsCallOutset call direction is call out() => voidRequired
onCreatedmake outgoing call() => voidRequired
onReceivedreceived incoming call() => voidRequired
onHangupset hang up() => voidRequired
onIOSTokenios voip push notificationStringRequired
  • In file src/screens/call_screen/index.js Example
import React, { useState, useContext } from "react";
import { PitelCallKit, PitelSDKContext } from "react-native-pitel-voip";

export const CallScreen = ({ route, navigation }) => {
const [mute, setMute] = useState(false);
const [speaker, setSpeaker] = useState(false);
const { pitelSDK } = useContext(PitelSDKContext);

const { phoneNumber, direction, callState, callID } = route.params;

return (
<PitelCallKit
pitelSDK={pitelSDK}
callState={callState}
phoneNumber={phoneNumber}
direction={direction}
microState={mute}
speakerState={speaker}
callID={callID}
onHangup={() => {
pitelSDK.hangup();
}}
onMicro={() => {
setMute(!mute);
}}
onSpeaker={() => {
setSpeaker(!speaker);
}}
/>
);
};
PropDescriptionTypeDefault
pitelSDKpitelSDK when extension login successObjectRequired
callStatecall statusStringRequired
directioncall directionStringRequired
microStatemicrophone statusboolRequired
speakerStateloudspeaker statusboolRequired
callIDincoming call idStringRequired
onHanguphang up when end callFunctionRequired
onMicroon/off microphoneFunctionRequired
onSpeakeron/off loud speakerFunctionRequired

How to test

Using tryit to test voip call connection & conversation Link: https://tryit.jssip.net/ Setting:

  1. Access to link https://tryit.jssip.net/
  2. Enter extension: example 102
  3. Click Setting icon
  4. Enter information to input field tryit
  5. Save
  6. Click icon -> to connect