@sbhjt-gr/react-native-webrtc 137.0.4 → 137.0.5-palabra.10
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +28 -7
- package/android/build.gradle +1 -0
- package/android/src/main/java/com/oney/WebRTCModule/WebRTCModule.java +73 -123
- package/android/src/main/java/com/oney/WebRTCModule/palabra/PalabraClient.java +464 -0
- package/android/src/main/java/com/oney/WebRTCModule/palabra/PalabraConfig.java +17 -0
- package/android/src/main/java/com/oney/WebRTCModule/palabra/PalabraListener.java +7 -0
- package/ios/RCTWebRTC/PalabraAudioSink.h +13 -0
- package/ios/RCTWebRTC/PalabraAudioSink.m +18 -0
- package/ios/RCTWebRTC/PalabraClient.h +36 -0
- package/ios/RCTWebRTC/PalabraClient.m +584 -0
- package/ios/RCTWebRTC/WebRTCModule+Palabra.h +4 -0
- package/ios/RCTWebRTC/WebRTCModule+Palabra.m +83 -0
- package/ios/RCTWebRTC/WebRTCModule+RTCMediaStream.m +0 -147
- package/ios/RCTWebRTC/WebRTCModule.h +7 -3
- package/ios/RCTWebRTC/WebRTCModule.m +3 -4
- package/lib/commonjs/EventEmitter.js +1 -21
- package/lib/commonjs/EventEmitter.js.map +1 -1
- package/lib/commonjs/MediaStreamTrack.js +15 -51
- package/lib/commonjs/MediaStreamTrack.js.map +1 -1
- package/lib/commonjs/index.js.map +1 -1
- package/lib/module/EventEmitter.js +1 -20
- package/lib/module/EventEmitter.js.map +1 -1
- package/lib/module/MediaStreamTrack.js +16 -52
- package/lib/module/MediaStreamTrack.js.map +1 -1
- package/lib/module/index.js.map +1 -1
- package/lib/typescript/EventEmitter.d.ts +0 -1
- package/lib/typescript/MediaStreamTrack.d.ts +13 -17
- package/lib/typescript/index.d.ts +2 -2
- package/livekit-react-native-webrtc.podspec +3 -3
- package/package.json +4 -4
- package/src/EventEmitter.ts +3 -25
- package/src/MediaStreamTrack.ts +39 -91
- package/src/index.ts +3 -1
package/README.md
CHANGED
|
@@ -1,13 +1,34 @@
|
|
|
1
1
|
[<img src="https://avatars.githubusercontent.com/u/42463376" alt="React Native WebRTC" style="height: 6em;" />](https://github.com/react-native-webrtc/react-native-webrtc)
|
|
2
2
|
|
|
3
|
-
# React-Native-WebRTC
|
|
4
|
-
[
|
|
4
|
+
[](https://www.npmjs.com/package/@livekit/react-native-webrtc)
|
|
5
5
|
[](https://react-native-webrtc.discourse.group/)
|
|
6
6
|
|
|
7
|
-
A WebRTC module for React Native.
|
|
7
|
+
A WebRTC module for React Native with native Palabra AI translation support.
|
|
8
8
|
|
|
9
9
|
> [!NOTE]
|
|
10
|
-
> This is a fork
|
|
10
|
+
> This is a fork with integrated Palabra translation for real-time speech-to-speech translation during WebRTC calls.
|
|
11
|
+
> Used by WiLang app for video/voice call translation.
|
|
12
|
+
|
|
13
|
+
## Palabra Translation
|
|
14
|
+
|
|
15
|
+
This fork adds native Palabra AI integration for translating remote peer audio in real-time:
|
|
16
|
+
|
|
17
|
+
```javascript
|
|
18
|
+
import { startPalabraTranslation, stopPalabraTranslation } from '@livekit/react-native-webrtc';
|
|
19
|
+
|
|
20
|
+
await startPalabraTranslation(
|
|
21
|
+
peerConnectionId,
|
|
22
|
+
remoteAudioTrackId,
|
|
23
|
+
PALABRA_CLIENT_ID,
|
|
24
|
+
PALABRA_CLIENT_SECRET,
|
|
25
|
+
'en',
|
|
26
|
+
'es',
|
|
27
|
+
'https://api.palabra.ai'
|
|
28
|
+
);
|
|
29
|
+
|
|
30
|
+
await stopPalabraTranslation();
|
|
31
|
+
```
|
|
11
32
|
|
|
12
33
|
## Feature Overview
|
|
13
34
|
|
|
@@ -52,9 +73,9 @@ Software encode/decode factories have been enabled by default.
|
|
|
52
73
|
Use one of the following preferred package install methods to immediately get going.
|
|
53
74
|
Don't forget to follow platform guides below to cover any extra required steps.
|
|
54
75
|
|
|
55
|
-
**npm:** `npm install @
|
|
56
|
-
**yarn:** `yarn add @
|
|
57
|
-
**pnpm:** `pnpm install @
|
|
76
|
+
**npm:** `npm install @livekit/react-native-webrtc --save`
|
|
77
|
+
**yarn:** `yarn add @livekit/react-native-webrtc`
|
|
78
|
+
**pnpm:** `pnpm install @livekit/react-native-webrtc`
|
|
58
79
|
|
|
59
80
|
## Guides
|
|
60
81
|
|
package/android/build.gradle
CHANGED
|
@@ -23,11 +23,13 @@ import com.facebook.react.module.annotations.ReactModule;
|
|
|
23
23
|
import com.facebook.react.modules.core.DeviceEventManagerModule;
|
|
24
24
|
import com.oney.WebRTCModule.webrtcutils.H264AndSoftwareVideoDecoderFactory;
|
|
25
25
|
import com.oney.WebRTCModule.webrtcutils.H264AndSoftwareVideoEncoderFactory;
|
|
26
|
+
import com.oney.WebRTCModule.palabra.PalabraClient;
|
|
27
|
+
import com.oney.WebRTCModule.palabra.PalabraConfig;
|
|
28
|
+
import com.oney.WebRTCModule.palabra.PalabraListener;
|
|
26
29
|
|
|
27
30
|
import org.webrtc.AddIceObserver;
|
|
28
31
|
import org.webrtc.AudioProcessingFactory;
|
|
29
32
|
import org.webrtc.AudioTrack;
|
|
30
|
-
import org.webrtc.AudioTrackSink;
|
|
31
33
|
import org.webrtc.CryptoOptions;
|
|
32
34
|
import org.webrtc.EglBase;
|
|
33
35
|
import org.webrtc.IceCandidate;
|
|
@@ -53,17 +55,12 @@ import org.webrtc.VideoTrack;
|
|
|
53
55
|
import org.webrtc.audio.AudioDeviceModule;
|
|
54
56
|
import org.webrtc.audio.JavaAudioDeviceModule;
|
|
55
57
|
|
|
56
|
-
import java.nio.ByteBuffer;
|
|
57
|
-
import java.nio.ByteOrder;
|
|
58
|
-
import java.nio.IntBuffer;
|
|
59
|
-
import java.nio.ShortBuffer;
|
|
60
58
|
import java.util.ArrayList;
|
|
61
59
|
import java.util.HashMap;
|
|
62
60
|
import java.util.List;
|
|
63
61
|
import java.util.Map;
|
|
64
62
|
import java.util.Objects;
|
|
65
63
|
import java.util.concurrent.Callable;
|
|
66
|
-
import java.util.concurrent.ConcurrentHashMap;
|
|
67
64
|
import java.util.concurrent.ExecutionException;
|
|
68
65
|
|
|
69
66
|
@ReactModule(name = "WebRTCModule")
|
|
@@ -78,10 +75,11 @@ public class WebRTCModule extends ReactContextBaseJavaModule {
|
|
|
78
75
|
// Need to expose the peer connection codec factories here to get capabilities
|
|
79
76
|
private final SparseArray<PeerConnectionObserver> mPeerConnectionObservers;
|
|
80
77
|
final Map<String, MediaStream> localStreams;
|
|
81
|
-
private final Map<String, AudioTrackSink> audioSinkMap = new ConcurrentHashMap<>();
|
|
82
78
|
|
|
83
79
|
private final GetUserMediaImpl getUserMediaImpl;
|
|
84
80
|
|
|
81
|
+
private PalabraClient palabraClient;
|
|
82
|
+
|
|
85
83
|
public WebRTCModule(ReactApplicationContext reactContext) {
|
|
86
84
|
super(reactContext);
|
|
87
85
|
|
|
@@ -486,10 +484,6 @@ public class WebRTCModule extends ReactContextBaseJavaModule {
|
|
|
486
484
|
return pco.remoteTracks.get(trackId);
|
|
487
485
|
}
|
|
488
486
|
|
|
489
|
-
private String audioSinkKey(int pcId, String trackId) {
|
|
490
|
-
return pcId + ":" + trackId;
|
|
491
|
-
}
|
|
492
|
-
|
|
493
487
|
MediaStreamTrack getLocalTrack(String trackId) {
|
|
494
488
|
return getUserMediaImpl.getTrack(trackId);
|
|
495
489
|
}
|
|
@@ -964,118 +958,6 @@ public class WebRTCModule extends ReactContextBaseJavaModule {
|
|
|
964
958
|
});
|
|
965
959
|
}
|
|
966
960
|
|
|
967
|
-
@ReactMethod
|
|
968
|
-
public void mediaStreamTrackSetPlaybackEnabled(int pcId, String id, boolean enabled) {
|
|
969
|
-
ThreadUtils.runOnExecutor(() -> {
|
|
970
|
-
MediaStreamTrack track = getTrack(pcId, id);
|
|
971
|
-
if (track == null) {
|
|
972
|
-
Log.d(TAG, "mediaStreamTrackSetPlaybackEnabled() could not find track " + id);
|
|
973
|
-
return;
|
|
974
|
-
}
|
|
975
|
-
|
|
976
|
-
if (!(track instanceof AudioTrack)) {
|
|
977
|
-
Log.d(TAG, "mediaStreamTrackSetPlaybackEnabled() track is not an AudioTrack!");
|
|
978
|
-
return;
|
|
979
|
-
}
|
|
980
|
-
|
|
981
|
-
((AudioTrack) track).setVolume(enabled ? 1.0 : 0.0);
|
|
982
|
-
});
|
|
983
|
-
}
|
|
984
|
-
|
|
985
|
-
private class AudioSamplesInterceptor implements AudioTrackSink {
|
|
986
|
-
private final String trackId;
|
|
987
|
-
|
|
988
|
-
AudioSamplesInterceptor(String trackId) {
|
|
989
|
-
this.trackId = trackId;
|
|
990
|
-
}
|
|
991
|
-
|
|
992
|
-
@Override
|
|
993
|
-
public void onData(
|
|
994
|
-
ByteBuffer audioData,
|
|
995
|
-
int bitsPerSample,
|
|
996
|
-
int sampleRate,
|
|
997
|
-
int numberOfChannels,
|
|
998
|
-
int numberOfFrames,
|
|
999
|
-
long absoluteCaptureTimestampMs) {
|
|
1000
|
-
WritableMap params = Arguments.createMap();
|
|
1001
|
-
params.putString("trackId", trackId);
|
|
1002
|
-
params.putArray("samples", convertSamples(audioData, bitsPerSample));
|
|
1003
|
-
params.putInt("sampleRate", sampleRate);
|
|
1004
|
-
params.putInt("channels", numberOfChannels);
|
|
1005
|
-
params.putInt("bitsPerSample", bitsPerSample);
|
|
1006
|
-
params.putInt("framesPerBuffer", numberOfFrames);
|
|
1007
|
-
params.putDouble("timestamp", (double) absoluteCaptureTimestampMs);
|
|
1008
|
-
|
|
1009
|
-
sendEvent("audioSamples", params);
|
|
1010
|
-
}
|
|
1011
|
-
|
|
1012
|
-
private WritableArray convertSamples(ByteBuffer audioData, int bitsPerSample) {
|
|
1013
|
-
WritableArray samples = Arguments.createArray();
|
|
1014
|
-
if (audioData == null) {
|
|
1015
|
-
return samples;
|
|
1016
|
-
}
|
|
1017
|
-
|
|
1018
|
-
ByteBuffer buffer = audioData.slice().order(ByteOrder.LITTLE_ENDIAN);
|
|
1019
|
-
|
|
1020
|
-
if (bitsPerSample == 16) {
|
|
1021
|
-
ShortBuffer shortBuffer = buffer.asShortBuffer();
|
|
1022
|
-
while (shortBuffer.hasRemaining()) {
|
|
1023
|
-
samples.pushDouble(shortBuffer.get());
|
|
1024
|
-
}
|
|
1025
|
-
} else if (bitsPerSample == 32) {
|
|
1026
|
-
IntBuffer intBuffer = buffer.asIntBuffer();
|
|
1027
|
-
while (intBuffer.hasRemaining()) {
|
|
1028
|
-
samples.pushDouble(intBuffer.get());
|
|
1029
|
-
}
|
|
1030
|
-
} else {
|
|
1031
|
-
while (buffer.hasRemaining()) {
|
|
1032
|
-
samples.pushDouble(buffer.get());
|
|
1033
|
-
}
|
|
1034
|
-
}
|
|
1035
|
-
|
|
1036
|
-
return samples;
|
|
1037
|
-
}
|
|
1038
|
-
}
|
|
1039
|
-
|
|
1040
|
-
@ReactMethod
|
|
1041
|
-
public void mediaStreamTrackEnableAudioSink(int pcId, String id, boolean enabled) {
|
|
1042
|
-
ThreadUtils.runOnExecutor(() -> {
|
|
1043
|
-
String sinkKey = audioSinkKey(pcId, id);
|
|
1044
|
-
|
|
1045
|
-
if (enabled) {
|
|
1046
|
-
if (audioSinkMap.containsKey(sinkKey)) {
|
|
1047
|
-
return;
|
|
1048
|
-
}
|
|
1049
|
-
|
|
1050
|
-
MediaStreamTrack track = getTrack(pcId, id);
|
|
1051
|
-
if (track == null) {
|
|
1052
|
-
Log.d(TAG, "mediaStreamTrackEnableAudioSink() could not find track " + id);
|
|
1053
|
-
return;
|
|
1054
|
-
}
|
|
1055
|
-
|
|
1056
|
-
if (!(track instanceof AudioTrack)) {
|
|
1057
|
-
Log.d(TAG, "mediaStreamTrackEnableAudioSink() track is not an AudioTrack!");
|
|
1058
|
-
return;
|
|
1059
|
-
}
|
|
1060
|
-
|
|
1061
|
-
AudioSamplesInterceptor sink = new AudioSamplesInterceptor(id);
|
|
1062
|
-
((AudioTrack) track).addSink(sink);
|
|
1063
|
-
audioSinkMap.put(sinkKey, sink);
|
|
1064
|
-
} else {
|
|
1065
|
-
AudioTrackSink sink = audioSinkMap.remove(sinkKey);
|
|
1066
|
-
|
|
1067
|
-
if (sink == null) {
|
|
1068
|
-
return;
|
|
1069
|
-
}
|
|
1070
|
-
|
|
1071
|
-
MediaStreamTrack track = getTrack(pcId, id);
|
|
1072
|
-
if (track instanceof AudioTrack) {
|
|
1073
|
-
((AudioTrack) track).removeSink(sink);
|
|
1074
|
-
}
|
|
1075
|
-
}
|
|
1076
|
-
});
|
|
1077
|
-
}
|
|
1078
|
-
|
|
1079
961
|
/**
|
|
1080
962
|
* This serializes the transceivers current direction and mid and returns them
|
|
1081
963
|
* for update when an sdp negotiation/renegotiation happens
|
|
@@ -1690,4 +1572,72 @@ public class WebRTCModule extends ReactContextBaseJavaModule {
|
|
|
1690
1572
|
public void removeListeners(Integer count) {
|
|
1691
1573
|
// Keep: Required for RN built in Event Emitter Calls.
|
|
1692
1574
|
}
|
|
1575
|
+
|
|
1576
|
+
@ReactMethod
|
|
1577
|
+
public void startPalabraTranslation(int peerConnectionId, String trackId, String clientId,
|
|
1578
|
+
String clientSecret, String sourceLang, String targetLang,
|
|
1579
|
+
String apiUrl, Promise promise) {
|
|
1580
|
+
PeerConnectionObserver pco = mPeerConnectionObservers.get(peerConnectionId);
|
|
1581
|
+
if (pco == null) {
|
|
1582
|
+
promise.reject("E_INVALID", "pc_not_found");
|
|
1583
|
+
return;
|
|
1584
|
+
}
|
|
1585
|
+
|
|
1586
|
+
MediaStreamTrack track = pco.remoteTracks.get(trackId);
|
|
1587
|
+
if (track == null) {
|
|
1588
|
+
promise.reject("E_INVALID", "track_not_found");
|
|
1589
|
+
return;
|
|
1590
|
+
}
|
|
1591
|
+
|
|
1592
|
+
if (!(track instanceof AudioTrack)) {
|
|
1593
|
+
promise.reject("E_INVALID", "not_audio_track");
|
|
1594
|
+
return;
|
|
1595
|
+
}
|
|
1596
|
+
|
|
1597
|
+
AudioTrack audioTrack = (AudioTrack) track;
|
|
1598
|
+
|
|
1599
|
+
if (palabraClient != null) {
|
|
1600
|
+
palabraClient.stop();
|
|
1601
|
+
}
|
|
1602
|
+
|
|
1603
|
+
PalabraConfig config = new PalabraConfig(clientId, clientSecret, sourceLang, targetLang, apiUrl);
|
|
1604
|
+
palabraClient = new PalabraClient(getReactApplicationContext(), config);
|
|
1605
|
+
palabraClient.setListener(new PalabraListener() {
|
|
1606
|
+
@Override
|
|
1607
|
+
public void onTranscription(String text, String lang, boolean isFinal) {
|
|
1608
|
+
WritableMap params = Arguments.createMap();
|
|
1609
|
+
params.putString("text", text);
|
|
1610
|
+
params.putString("lang", lang);
|
|
1611
|
+
params.putBoolean("isFinal", isFinal);
|
|
1612
|
+
sendEvent("palabraTranscription", params);
|
|
1613
|
+
}
|
|
1614
|
+
|
|
1615
|
+
@Override
|
|
1616
|
+
public void onConnectionState(String state) {
|
|
1617
|
+
WritableMap params = Arguments.createMap();
|
|
1618
|
+
params.putString("state", state);
|
|
1619
|
+
sendEvent("palabraConnectionState", params);
|
|
1620
|
+
}
|
|
1621
|
+
|
|
1622
|
+
@Override
|
|
1623
|
+
public void onError(int code, String message) {
|
|
1624
|
+
WritableMap params = Arguments.createMap();
|
|
1625
|
+
params.putInt("code", code);
|
|
1626
|
+
params.putString("message", message);
|
|
1627
|
+
sendEvent("palabraError", params);
|
|
1628
|
+
}
|
|
1629
|
+
});
|
|
1630
|
+
|
|
1631
|
+
palabraClient.start(audioTrack);
|
|
1632
|
+
promise.resolve(null);
|
|
1633
|
+
}
|
|
1634
|
+
|
|
1635
|
+
@ReactMethod
|
|
1636
|
+
public void stopPalabraTranslation(Promise promise) {
|
|
1637
|
+
if (palabraClient != null) {
|
|
1638
|
+
palabraClient.stop();
|
|
1639
|
+
palabraClient = null;
|
|
1640
|
+
}
|
|
1641
|
+
promise.resolve(null);
|
|
1642
|
+
}
|
|
1693
1643
|
}
|