123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452 |
- using System;
- using System.Collections;
- using System.Collections.Generic;
- using System.Linq;
- using UnityEngine;
- using UnityEngine.UI;
- namespace Unity.WebRTC.Samples
- {
- class VideoReceiveSample : MonoBehaviour
- {
- #pragma warning disable 0649
- [SerializeField] private Button callButton;
- [SerializeField] private Button hangUpButton;
- [SerializeField] private Button addTracksButton;
- [SerializeField] private Button removeTracksButton;
- [SerializeField] private Toggle useWebCamToggle;
- [SerializeField] private Dropdown webCamListDropdown;
- [SerializeField] private Toggle useMicToggle;
- [SerializeField] private Dropdown micListDropdown;
- [SerializeField] private Camera cam;
- [SerializeField] private AudioClip clip;
- [SerializeField] private RawImage sourceImage;
- [SerializeField] private AudioSource sourceAudio;
- [SerializeField] private RawImage receiveImage;
- [SerializeField] private AudioSource receiveAudio;
- [SerializeField] private Transform rotateObject;
- #pragma warning restore 0649
- private RTCPeerConnection _pc1, _pc2;
- private List<RTCRtpSender> pc1Senders;
- private VideoStreamTrack videoStreamTrack;
- private AudioStreamTrack audioStreamTrack;
- private MediaStream receiveAudioStream, receiveVideoStream;
- private DelegateOnIceConnectionChange pc1OnIceConnectionChange;
- private DelegateOnIceConnectionChange pc2OnIceConnectionChange;
- private DelegateOnIceCandidate pc1OnIceCandidate;
- private DelegateOnIceCandidate pc2OnIceCandidate;
- private DelegateOnTrack pc2Ontrack;
- private DelegateOnNegotiationNeeded pc1OnNegotiationNeeded;
- private WebCamTexture webCamTexture;
- private void Awake()
- {
- WebRTC.Initialize(WebRTCSettings.LimitTextureSize);
- callButton.onClick.AddListener(Call);
- hangUpButton.onClick.AddListener(HangUp);
- addTracksButton.onClick.AddListener(AddTracks);
- removeTracksButton.onClick.AddListener(RemoveTracks);
- useWebCamToggle.onValueChanged.AddListener(SwitchUseWebCam);
- webCamListDropdown.options = WebCamTexture.devices.Select(x => new Dropdown.OptionData(x.name)).ToList();
- useMicToggle.onValueChanged.AddListener(SwitchUseMic);
- micListDropdown.options = Microphone.devices.Select(x => new Dropdown.OptionData(x)).ToList();
- }
- private void OnDestroy()
- {
- if (webCamTexture != null)
- {
- webCamTexture.Stop();
- webCamTexture = null;
- }
- WebRTC.Dispose();
- }
- private void Start()
- {
- pc1Senders = new List<RTCRtpSender>();
- callButton.interactable = true;
- hangUpButton.interactable = false;
- pc1OnIceConnectionChange = state => { OnIceConnectionChange(_pc1, state); };
- pc2OnIceConnectionChange = state => { OnIceConnectionChange(_pc2, state); };
- pc1OnIceCandidate = candidate => { OnIceCandidate(_pc1, candidate); };
- pc2OnIceCandidate = candidate => { OnIceCandidate(_pc2, candidate); };
- pc2Ontrack = e =>
- {
- if (e.Track is VideoStreamTrack video)
- {
- video.OnVideoReceived += tex =>
- {
- receiveImage.texture = tex;
- };
- }
- if (e.Track is AudioStreamTrack audioTrack)
- {
- receiveAudio.SetTrack(audioTrack);
- receiveAudio.loop = true;
- receiveAudio.Play();
- }
- };
- pc1OnNegotiationNeeded = () => { StartCoroutine(PeerNegotiationNeeded(_pc1)); };
- StartCoroutine(WebRTC.Update());
- }
- private void Update()
- {
- if (rotateObject != null)
- {
- rotateObject.Rotate(1, 2, 3);
- }
- }
- private static RTCConfiguration GetSelectedSdpSemantics()
- {
- RTCConfiguration config = default;
- config.iceServers = new[] {new RTCIceServer {urls = new[] {"stun:stun.l.google.com:19302"}}};
- return config;
- }
- private void OnIceConnectionChange(RTCPeerConnection pc, RTCIceConnectionState state)
- {
- switch (state)
- {
- case RTCIceConnectionState.New:
- Debug.Log($"{GetName(pc)} IceConnectionState: New");
- break;
- case RTCIceConnectionState.Checking:
- Debug.Log($"{GetName(pc)} IceConnectionState: Checking");
- break;
- case RTCIceConnectionState.Closed:
- Debug.Log($"{GetName(pc)} IceConnectionState: Closed");
- break;
- case RTCIceConnectionState.Completed:
- Debug.Log($"{GetName(pc)} IceConnectionState: Completed");
- break;
- case RTCIceConnectionState.Connected:
- Debug.Log($"{GetName(pc)} IceConnectionState: Connected");
- break;
- case RTCIceConnectionState.Disconnected:
- Debug.Log($"{GetName(pc)} IceConnectionState: Disconnected");
- break;
- case RTCIceConnectionState.Failed:
- Debug.Log($"{GetName(pc)} IceConnectionState: Failed");
- break;
- case RTCIceConnectionState.Max:
- Debug.Log($"{GetName(pc)} IceConnectionState: Max");
- break;
- default:
- throw new ArgumentOutOfRangeException(nameof(state), state, null);
- }
- }
- IEnumerator PeerNegotiationNeeded(RTCPeerConnection pc)
- {
- Debug.Log($"{GetName(pc)} createOffer start");
- var op = pc.CreateOffer();
- yield return op;
- if (!op.IsError)
- {
- if (pc.SignalingState != RTCSignalingState.Stable)
- {
- Debug.LogError($"{GetName(pc)} signaling state is not stable.");
- yield break;
- }
- yield return StartCoroutine(OnCreateOfferSuccess(pc, op.Desc));
- }
- else
- {
- OnCreateSessionDescriptionError(op.Error);
- }
- }
- private void AddTracks()
- {
- var videoSender = _pc1.AddTrack(videoStreamTrack);
- pc1Senders.Add(videoSender);
- pc1Senders.Add(_pc1.AddTrack(audioStreamTrack));
- if (WebRTCSettings.UseVideoCodec != null)
- {
- var codecs = new[] {WebRTCSettings.UseVideoCodec};
- var transceiver = _pc1.GetTransceivers().First(t => t.Sender == videoSender);
- transceiver.SetCodecPreferences(codecs);
- }
- addTracksButton.interactable = false;
- removeTracksButton.interactable = true;
- }
- private void RemoveTracks()
- {
- var transceivers = _pc1.GetTransceivers();
- foreach (var transceiver in transceivers)
- {
- if(transceiver.Sender != null)
- {
- transceiver.Stop();
- _pc1.RemoveTrack(transceiver.Sender);
- }
- }
- pc1Senders.Clear();
- addTracksButton.interactable = true;
- removeTracksButton.interactable = false;
- }
- private void SwitchUseWebCam(bool isOn)
- {
- webCamListDropdown.interactable = isOn;
- }
- private void SwitchUseMic(bool isOn)
- {
- micListDropdown.interactable = isOn;
- }
- private void Call()
- {
- useWebCamToggle.interactable = false;
- webCamListDropdown.interactable = false;
- useMicToggle.interactable = false;
- micListDropdown.interactable = false;
- callButton.interactable = false;
- hangUpButton.interactable = true;
- addTracksButton.interactable = true;
- removeTracksButton.interactable = false;
- Debug.Log("GetSelectedSdpSemantics");
- var configuration = GetSelectedSdpSemantics();
- _pc1 = new RTCPeerConnection(ref configuration);
- Debug.Log("Created local peer connection object pc1");
- _pc1.OnIceCandidate = pc1OnIceCandidate;
- _pc1.OnIceConnectionChange = pc1OnIceConnectionChange;
- _pc1.OnNegotiationNeeded = pc1OnNegotiationNeeded;
- _pc2 = new RTCPeerConnection(ref configuration);
- Debug.Log("Created remote peer connection object pc2");
- _pc2.OnIceCandidate = pc2OnIceCandidate;
- _pc2.OnIceConnectionChange = pc2OnIceConnectionChange;
- _pc2.OnTrack = pc2Ontrack;
- CaptureAudioStart();
- StartCoroutine(CaptureVideoStart());
- }
- private void CaptureAudioStart()
- {
- if (!useMicToggle.isOn)
- {
- sourceAudio.clip = clip;
- sourceAudio.loop = true;
- sourceAudio.Play();
- audioStreamTrack = new AudioStreamTrack(sourceAudio);
- return;
- }
- var deviceName = Microphone.devices[micListDropdown.value];
- Microphone.GetDeviceCaps(deviceName, out int minFreq, out int maxFreq);
- var micClip = Microphone.Start(deviceName, true, 1, 48000);
- // set the latency to “0” samples before the audio starts to play.
- while (!(Microphone.GetPosition(deviceName) > 0)) {}
- sourceAudio.clip = micClip;
- sourceAudio.loop = true;
- sourceAudio.Play();
- audioStreamTrack = new AudioStreamTrack(sourceAudio);
- }
- private IEnumerator CaptureVideoStart()
- {
- if (!useWebCamToggle.isOn)
- {
- videoStreamTrack = cam.CaptureStreamTrack(WebRTCSettings.StreamSize.x, WebRTCSettings.StreamSize.y);
- sourceImage.texture = cam.targetTexture;
- yield break;
- }
- if (WebCamTexture.devices.Length == 0)
- {
- Debug.LogFormat("WebCam device not found");
- yield break;
- }
- yield return Application.RequestUserAuthorization(UserAuthorization.WebCam);
- if (!Application.HasUserAuthorization(UserAuthorization.WebCam))
- {
- Debug.LogFormat("authorization for using the device is denied");
- yield break;
- }
- WebCamDevice userCameraDevice = WebCamTexture.devices[webCamListDropdown.value];
- webCamTexture = new WebCamTexture(userCameraDevice.name, WebRTCSettings.StreamSize.x, WebRTCSettings.StreamSize.y, 30);
- webCamTexture.Play();
- yield return new WaitUntil(() => webCamTexture.didUpdateThisFrame);
- videoStreamTrack = new VideoStreamTrack(webCamTexture);
- sourceImage.texture = webCamTexture;
- }
- private void HangUp()
- {
- if (webCamTexture != null)
- {
- webCamTexture.Stop();
- webCamTexture = null;
- }
- receiveAudioStream?.Dispose();
- receiveAudioStream = null;
- receiveVideoStream?.Dispose();
- receiveVideoStream = null;
- videoStreamTrack?.Dispose();
- videoStreamTrack = null;
- audioStreamTrack?.Dispose();
- audioStreamTrack = null;
- Debug.Log("Close local/remote peer connection");
- _pc1?.Dispose();
- _pc2?.Dispose();
- _pc1 = null;
- _pc2 = null;
- sourceImage.texture = null;
- sourceAudio.Stop();
- sourceAudio.clip = null;
- receiveImage.texture = null;
- receiveAudio.Stop();
- receiveAudio.clip = null;
- useWebCamToggle.interactable = true;
- webCamListDropdown.interactable = useWebCamToggle.isOn;
- useMicToggle.interactable = true;
- micListDropdown.interactable = useMicToggle.isOn;
- callButton.interactable = true;
- hangUpButton.interactable = false;
- addTracksButton.interactable = false;
- removeTracksButton.interactable = false;
- }
- private void OnIceCandidate(RTCPeerConnection pc, RTCIceCandidate candidate)
- {
- GetOtherPc(pc).AddIceCandidate(candidate);
- Debug.Log($"{GetName(pc)} ICE candidate:\n {candidate.Candidate}");
- }
- private string GetName(RTCPeerConnection pc)
- {
- return (pc == _pc1) ? "pc1" : "pc2";
- }
- private RTCPeerConnection GetOtherPc(RTCPeerConnection pc)
- {
- return (pc == _pc1) ? _pc2 : _pc1;
- }
- private IEnumerator OnCreateOfferSuccess(RTCPeerConnection pc, RTCSessionDescription desc)
- {
- Debug.Log($"Offer from {GetName(pc)}\n{desc.sdp}");
- Debug.Log($"{GetName(pc)} setLocalDescription start");
- var op = pc.SetLocalDescription(ref desc);
- yield return op;
- if (!op.IsError)
- {
- OnSetLocalSuccess(pc);
- }
- else
- {
- var error = op.Error;
- OnSetSessionDescriptionError(ref error);
- }
- var otherPc = GetOtherPc(pc);
- Debug.Log($"{GetName(otherPc)} setRemoteDescription start");
- var op2 = otherPc.SetRemoteDescription(ref desc);
- yield return op2;
- if (!op2.IsError)
- {
- OnSetRemoteSuccess(otherPc);
- }
- else
- {
- var error = op2.Error;
- OnSetSessionDescriptionError(ref error);
- }
- Debug.Log($"{GetName(otherPc)} createAnswer start");
- // Since the 'remote' side has no media stream we need
- // to pass in the right constraints in order for it to
- // accept the incoming offer of audio and video.
- var op3 = otherPc.CreateAnswer();
- yield return op3;
- if (!op3.IsError)
- {
- yield return OnCreateAnswerSuccess(otherPc, op3.Desc);
- }
- else
- {
- OnCreateSessionDescriptionError(op3.Error);
- }
- }
- private void OnSetLocalSuccess(RTCPeerConnection pc)
- {
- Debug.Log($"{GetName(pc)} SetLocalDescription complete");
- }
- static void OnSetSessionDescriptionError(ref RTCError error)
- {
- Debug.LogError($"Error Detail Type: {error.message}");
- }
- private void OnSetRemoteSuccess(RTCPeerConnection pc)
- {
- Debug.Log($"{GetName(pc)} SetRemoteDescription complete");
- }
- IEnumerator OnCreateAnswerSuccess(RTCPeerConnection pc, RTCSessionDescription desc)
- {
- Debug.Log($"Answer from {GetName(pc)}:\n{desc.sdp}");
- Debug.Log($"{GetName(pc)} setLocalDescription start");
- var op = pc.SetLocalDescription(ref desc);
- yield return op;
- if (!op.IsError)
- {
- OnSetLocalSuccess(pc);
- }
- else
- {
- var error = op.Error;
- OnSetSessionDescriptionError(ref error);
- }
- var otherPc = GetOtherPc(pc);
- Debug.Log($"{GetName(otherPc)} setRemoteDescription start");
- var op2 = otherPc.SetRemoteDescription(ref desc);
- yield return op2;
- if (!op2.IsError)
- {
- OnSetRemoteSuccess(otherPc);
- }
- else
- {
- var error = op2.Error;
- OnSetSessionDescriptionError(ref error);
- }
- }
- private static void OnCreateSessionDescriptionError(RTCError error)
- {
- Debug.LogError($"Error Detail Type: {error.message}");
- }
- }
- }
|