mirror of
https://github.com/vector-im/hydrogen-web.git
synced 2025-01-11 12:37:22 +01:00
WIP: expose streams, senders and receivers
This commit is contained in:
parent
36dc463d23
commit
2d4301fe5a
@ -75,6 +75,7 @@ export class CallTile extends SimpleTile {
|
|||||||
async join() {
|
async join() {
|
||||||
if (this.canJoin) {
|
if (this.canJoin) {
|
||||||
const mediaTracks = await this.platform.mediaDevices.getMediaTracks(false, true);
|
const mediaTracks = await this.platform.mediaDevices.getMediaTracks(false, true);
|
||||||
|
// const screenShareTrack = await this.platform.mediaDevices.getScreenShareTrack();
|
||||||
const localMedia = new LocalMedia().withTracks(mediaTracks);
|
const localMedia = new LocalMedia().withTracks(mediaTracks);
|
||||||
await this._call.join(localMedia);
|
await this._call.join(localMedia);
|
||||||
}
|
}
|
||||||
|
@ -15,37 +15,26 @@ limitations under the License.
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import {SDPStreamMetadataPurpose} from "./callEventTypes";
|
import {SDPStreamMetadataPurpose} from "./callEventTypes";
|
||||||
import {Track, AudioTrack, TrackType} from "../../platform/types/MediaDevices";
|
import {Stream} from "../../platform/types/MediaDevices";
|
||||||
import {SDPStreamMetadata} from "./callEventTypes";
|
import {SDPStreamMetadata} from "./callEventTypes";
|
||||||
|
|
||||||
export class LocalMedia {
|
export class LocalMedia {
|
||||||
constructor(
|
constructor(
|
||||||
public readonly cameraTrack?: Track,
|
public readonly userMedia?: Stream,
|
||||||
public readonly screenShareTrack?: Track,
|
public readonly screenShare?: Stream,
|
||||||
public readonly microphoneTrack?: AudioTrack,
|
|
||||||
public readonly dataChannelOptions?: RTCDataChannelInit,
|
public readonly dataChannelOptions?: RTCDataChannelInit,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
withTracks(tracks: Track[]) {
|
withUserMedia(stream: Stream) {
|
||||||
const cameraTrack = tracks.find(t => t.type === TrackType.Camera) ?? this.cameraTrack;
|
return new LocalMedia(stream, this.screenShare, this.dataChannelOptions);
|
||||||
const screenShareTrack = tracks.find(t => t.type === TrackType.ScreenShare) ?? this.screenShareTrack;
|
}
|
||||||
const microphoneTrack = tracks.find(t => t.type === TrackType.Microphone) ?? this.microphoneTrack;
|
|
||||||
if (cameraTrack && microphoneTrack && cameraTrack.streamId !== microphoneTrack.streamId) {
|
withScreenShare(stream: Stream) {
|
||||||
throw new Error("The camera and audio track should have the same stream id");
|
return new LocalMedia(this.userMedia, stream, this.dataChannelOptions);
|
||||||
}
|
|
||||||
return new LocalMedia(cameraTrack, screenShareTrack, microphoneTrack as AudioTrack, this.dataChannelOptions);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
withDataChannel(options: RTCDataChannelInit): LocalMedia {
|
withDataChannel(options: RTCDataChannelInit): LocalMedia {
|
||||||
return new LocalMedia(this.cameraTrack, this.screenShareTrack, this.microphoneTrack as AudioTrack, options);
|
return new LocalMedia(this.userMedia, this.screenShare, options);
|
||||||
}
|
|
||||||
|
|
||||||
get tracks(): Track[] {
|
|
||||||
const tracks: Track[] = [];
|
|
||||||
if (this.cameraTrack) { tracks.push(this.cameraTrack); }
|
|
||||||
if (this.screenShareTrack) { tracks.push(this.screenShareTrack); }
|
|
||||||
if (this.microphoneTrack) { tracks.push(this.microphoneTrack); }
|
|
||||||
return tracks;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
getSDPMetadata(): SDPStreamMetadata {
|
getSDPMetadata(): SDPStreamMetadata {
|
||||||
@ -54,8 +43,8 @@ export class LocalMedia {
|
|||||||
if (userMediaTrack) {
|
if (userMediaTrack) {
|
||||||
metadata[userMediaTrack.streamId] = {
|
metadata[userMediaTrack.streamId] = {
|
||||||
purpose: SDPStreamMetadataPurpose.Usermedia,
|
purpose: SDPStreamMetadataPurpose.Usermedia,
|
||||||
audio_muted: this.microphoneTrack?.muted ?? false,
|
audio_muted: this.microphoneTrack?.muted ?? true,
|
||||||
video_muted: this.cameraTrack?.muted ?? false,
|
video_muted: this.cameraTrack?.muted ?? true,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
if (this.screenShareTrack) {
|
if (this.screenShareTrack) {
|
||||||
@ -67,13 +56,12 @@ export class LocalMedia {
|
|||||||
}
|
}
|
||||||
|
|
||||||
clone() {
|
clone() {
|
||||||
// TODO: implement
|
return new LocalMedia(this.userMedia?.clone(), this.screenShare?.clone(), this.dataChannelOptions);
|
||||||
return this;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
dispose() {
|
dispose() {
|
||||||
this.cameraTrack?.stop();
|
this.userMedia?.audioTrack?.stop();
|
||||||
this.microphoneTrack?.stop();
|
this.userMedia?.videoTrack?.stop();
|
||||||
this.screenShareTrack?.stop();
|
this.screenShare?.videoTrack?.stop();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -701,8 +701,6 @@ export class PeerCall implements IDisposable {
|
|||||||
|
|
||||||
private updateRemoteSDPStreamMetadata(metadata: SDPStreamMetadata): void {
|
private updateRemoteSDPStreamMetadata(metadata: SDPStreamMetadata): void {
|
||||||
this.remoteSDPStreamMetadata = recursivelyAssign(this.remoteSDPStreamMetadata || {}, metadata, true);
|
this.remoteSDPStreamMetadata = recursivelyAssign(this.remoteSDPStreamMetadata || {}, metadata, true);
|
||||||
// will rerequest stream purpose for all tracks and set track.type accordingly
|
|
||||||
this.peerConnection.notifyStreamPurposeChanged();
|
|
||||||
for (const track of this.peerConnection.remoteTracks) {
|
for (const track of this.peerConnection.remoteTracks) {
|
||||||
const streamMetaData = this.remoteSDPStreamMetadata?.[track.streamId];
|
const streamMetaData = this.remoteSDPStreamMetadata?.[track.streamId];
|
||||||
if (streamMetaData) {
|
if (streamMetaData) {
|
||||||
@ -757,6 +755,8 @@ export class PeerCall implements IDisposable {
|
|||||||
this.iceDisconnectedTimeout?.abort();
|
this.iceDisconnectedTimeout?.abort();
|
||||||
this.iceDisconnectedTimeout = undefined;
|
this.iceDisconnectedTimeout = undefined;
|
||||||
this.setState(CallState.Connected, log);
|
this.setState(CallState.Connected, log);
|
||||||
|
const transceivers = this.peerConnection.peerConnection.getTransceivers();
|
||||||
|
console.log(transceivers);
|
||||||
} else if (state == 'failed') {
|
} else if (state == 'failed') {
|
||||||
this.iceDisconnectedTimeout?.abort();
|
this.iceDisconnectedTimeout?.abort();
|
||||||
this.iceDisconnectedTimeout = undefined;
|
this.iceDisconnectedTimeout = undefined;
|
||||||
|
@ -18,29 +18,32 @@ export interface MediaDevices {
|
|||||||
// filter out audiooutput
|
// filter out audiooutput
|
||||||
enumerate(): Promise<MediaDeviceInfo[]>;
|
enumerate(): Promise<MediaDeviceInfo[]>;
|
||||||
// to assign to a video element, we downcast to WrappedTrack and use the stream property.
|
// to assign to a video element, we downcast to WrappedTrack and use the stream property.
|
||||||
getMediaTracks(audio: true | MediaDeviceInfo, video: boolean | MediaDeviceInfo): Promise<Track[]>;
|
getMediaTracks(audio: true | MediaDeviceInfo, video: boolean | MediaDeviceInfo): Promise<Stream>;
|
||||||
getScreenShareTrack(): Promise<Track | undefined>;
|
getScreenShareTrack(): Promise<Stream | undefined>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum TrackType {
|
export interface Stream {
|
||||||
ScreenShare,
|
readonly audioTrack: AudioTrack | undefined;
|
||||||
Camera,
|
readonly videoTrack: Track | undefined;
|
||||||
Microphone,
|
readonly id: string;
|
||||||
|
clone(): Stream;
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum TrackKind {
|
||||||
|
Video = "video",
|
||||||
|
Audio = "audio"
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Track {
|
export interface Track {
|
||||||
get type(): TrackType;
|
readonly kind: TrackKind;
|
||||||
get label(): string;
|
readonly label: string;
|
||||||
get id(): string;
|
readonly id: string;
|
||||||
get streamId(): string;
|
readonly settings: MediaTrackSettings;
|
||||||
get settings(): MediaTrackSettings;
|
|
||||||
get muted(): boolean;
|
|
||||||
setMuted(muted: boolean): void;
|
|
||||||
stop(): void;
|
stop(): void;
|
||||||
clone(): Track;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface AudioTrack extends Track {
|
export interface AudioTrack extends Track {
|
||||||
|
// TODO: how to emit updates on this?
|
||||||
get isSpeaking(): boolean;
|
get isSpeaking(): boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -14,38 +14,62 @@ See the License for the specific language governing permissions and
|
|||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {Track, TrackType} from "./MediaDevices";
|
import {Track, Stream} from "./MediaDevices";
|
||||||
import {SDPStreamMetadataPurpose} from "../../matrix/calls/callEventTypes";
|
import {SDPStreamMetadataPurpose} from "../../matrix/calls/callEventTypes";
|
||||||
|
|
||||||
export interface WebRTC {
|
export interface WebRTC {
|
||||||
createPeerConnection(handler: PeerConnectionHandler, forceTURN: boolean, turnServers: RTCIceServer[], iceCandidatePoolSize): PeerConnection;
|
createPeerConnection(handler: PeerConnectionHandler, forceTURN: boolean, turnServers: RTCIceServer[], iceCandidatePoolSize): PeerConnection;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface StreamSender {
|
||||||
|
get stream(): Stream;
|
||||||
|
get audioSender(): TrackSender | undefined;
|
||||||
|
get videoSender(): TrackSender | undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface StreamReceiver {
|
||||||
|
get stream(): Stream;
|
||||||
|
get audioReceiver(): TrackReceiver | undefined;
|
||||||
|
get videoReceiver(): TrackReceiver | undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface TrackReceiver {
|
||||||
|
get track(): Track;
|
||||||
|
get enabled(): boolean;
|
||||||
|
enable(enabled: boolean); // this modifies the transceiver direction
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface TrackSender extends TrackReceiver {
|
||||||
|
/** replaces the track if possible without renegotiation. Can throw. */
|
||||||
|
replaceTrack(track: Track): Promise<void>;
|
||||||
|
/** make any needed adjustments to the sender or transceiver settings
|
||||||
|
* depending on the purpose, after adding the track to the connection */
|
||||||
|
prepareForPurpose(purpose: SDPStreamMetadataPurpose): void;
|
||||||
|
}
|
||||||
|
|
||||||
export interface PeerConnectionHandler {
|
export interface PeerConnectionHandler {
|
||||||
onIceConnectionStateChange(state: RTCIceConnectionState);
|
onIceConnectionStateChange(state: RTCIceConnectionState);
|
||||||
onLocalIceCandidate(candidate: RTCIceCandidate);
|
onLocalIceCandidate(candidate: RTCIceCandidate);
|
||||||
onIceGatheringStateChange(state: RTCIceGatheringState);
|
onIceGatheringStateChange(state: RTCIceGatheringState);
|
||||||
onRemoteTracksChanged(tracks: Track[]);
|
onRemoteStreamRemoved(stream: Stream);
|
||||||
|
onRemoteTracksAdded(receiver: TrackReceiver);
|
||||||
onRemoteDataChannel(dataChannel: any | undefined);
|
onRemoteDataChannel(dataChannel: any | undefined);
|
||||||
onNegotiationNeeded();
|
onNegotiationNeeded();
|
||||||
// request the type of incoming stream
|
|
||||||
getPurposeForStreamId(streamId: string): SDPStreamMetadataPurpose;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface PeerConnection {
|
export interface PeerConnection {
|
||||||
notifyStreamPurposeChanged(): void;
|
|
||||||
get remoteTracks(): Track[];
|
|
||||||
get iceGatheringState(): RTCIceGatheringState;
|
get iceGatheringState(): RTCIceGatheringState;
|
||||||
get signalingState(): RTCSignalingState;
|
get signalingState(): RTCSignalingState;
|
||||||
get localDescription(): RTCSessionDescription | undefined;
|
get localDescription(): RTCSessionDescription | undefined;
|
||||||
|
get localStreams(): ReadonlyArray<StreamSender>;
|
||||||
|
get remoteStreams(): ReadonlyArray<StreamReceiver>;
|
||||||
createOffer(): Promise<RTCSessionDescriptionInit>;
|
createOffer(): Promise<RTCSessionDescriptionInit>;
|
||||||
createAnswer(): Promise<RTCSessionDescriptionInit>;
|
createAnswer(): Promise<RTCSessionDescriptionInit>;
|
||||||
setLocalDescription(description?: RTCSessionDescriptionInit): Promise<void>;
|
setLocalDescription(description?: RTCSessionDescriptionInit): Promise<void>;
|
||||||
setRemoteDescription(description: RTCSessionDescriptionInit): Promise<void>;
|
setRemoteDescription(description: RTCSessionDescriptionInit): Promise<void>;
|
||||||
addIceCandidate(candidate: RTCIceCandidate): Promise<void>;
|
addIceCandidate(candidate: RTCIceCandidate): Promise<void>;
|
||||||
addTrack(track: Track): void;
|
addTrack(track: Track): TrackSender | undefined;
|
||||||
removeTrack(track: Track): boolean;
|
removeTrack(track: TrackSender): void;
|
||||||
replaceTrack(oldTrack: Track, newTrack: Track): Promise<boolean>;
|
|
||||||
createDataChannel(options: RTCDataChannelInit): any;
|
createDataChannel(options: RTCDataChannelInit): any;
|
||||||
dispose(): void;
|
dispose(): void;
|
||||||
close(): void;
|
close(): void;
|
||||||
|
@ -15,7 +15,7 @@ See the License for the specific language governing permissions and
|
|||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {MediaDevices as IMediaDevices, TrackType, Track, AudioTrack} from "../../types/MediaDevices";
|
import {MediaDevices as IMediaDevices, Stream, Track, TrackKind, AudioTrack} from "../../types/MediaDevices";
|
||||||
|
|
||||||
const POLLING_INTERVAL = 200; // ms
|
const POLLING_INTERVAL = 200; // ms
|
||||||
export const SPEAKING_THRESHOLD = -60; // dB
|
export const SPEAKING_THRESHOLD = -60; // dB
|
||||||
@ -28,22 +28,14 @@ export class MediaDevicesWrapper implements IMediaDevices {
|
|||||||
return this.mediaDevices.enumerateDevices();
|
return this.mediaDevices.enumerateDevices();
|
||||||
}
|
}
|
||||||
|
|
||||||
async getMediaTracks(audio: true | MediaDeviceInfo, video: boolean | MediaDeviceInfo): Promise<Track[]> {
|
async getMediaTracks(audio: true | MediaDeviceInfo, video: boolean | MediaDeviceInfo): Promise<Stream> {
|
||||||
const stream = await this.mediaDevices.getUserMedia(this.getUserMediaContraints(audio, video));
|
const stream = await this.mediaDevices.getUserMedia(this.getUserMediaContraints(audio, video));
|
||||||
const tracks = stream.getTracks().map(t => {
|
return new StreamWrapper(stream);
|
||||||
const type = t.kind === "audio" ? TrackType.Microphone : TrackType.Camera;
|
|
||||||
return wrapTrack(t, stream, type);
|
|
||||||
});
|
|
||||||
return tracks;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async getScreenShareTrack(): Promise<Track | undefined> {
|
async getScreenShareTrack(): Promise<Stream | undefined> {
|
||||||
const stream = await this.mediaDevices.getDisplayMedia(this.getScreenshareContraints());
|
const stream = await this.mediaDevices.getDisplayMedia(this.getScreenshareContraints());
|
||||||
const videoTrack = stream.getTracks().find(t => t.kind === "video");
|
return new StreamWrapper(stream);
|
||||||
if (videoTrack) {
|
|
||||||
return wrapTrack(videoTrack, stream, TrackType.ScreenShare);
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private getUserMediaContraints(audio: boolean | MediaDeviceInfo, video: boolean | MediaDeviceInfo): MediaStreamConstraints {
|
private getUserMediaContraints(audio: boolean | MediaDeviceInfo, video: boolean | MediaDeviceInfo): MediaStreamConstraints {
|
||||||
@ -78,43 +70,50 @@ export class MediaDevicesWrapper implements IMediaDevices {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function wrapTrack(track: MediaStreamTrack, stream: MediaStream, type: TrackType) {
|
export class StreamWrapper implements Stream {
|
||||||
if (track.kind === "audio") {
|
|
||||||
return new AudioTrackWrapper(track, stream, type);
|
public audioTrack: AudioTrackWrapper | undefined;
|
||||||
} else {
|
public videoTrack: TrackWrapper | undefined;
|
||||||
return new TrackWrapper(track, stream, type);
|
|
||||||
|
constructor(public readonly stream: MediaStream) {
|
||||||
|
for (const track of stream.getTracks()) {
|
||||||
|
this.update(track);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
get id(): string { return this.stream.id; }
|
||||||
|
|
||||||
|
clone(): Stream {
|
||||||
|
return new StreamWrapper(this.stream.clone());
|
||||||
|
}
|
||||||
|
|
||||||
|
update(track: MediaStreamTrack): TrackWrapper | undefined {
|
||||||
|
if (track.kind === "video") {
|
||||||
|
if (!this.videoTrack || track.id !== this.videoTrack.track.id) {
|
||||||
|
this.videoTrack = new TrackWrapper(track, this.stream);
|
||||||
|
return this.videoTrack;
|
||||||
|
}
|
||||||
|
} else if (track.kind === "audio") {
|
||||||
|
if (!this.audioTrack || track.id !== this.audioTrack.track.id) {
|
||||||
|
this.audioTrack = new AudioTrackWrapper(track, this.stream);
|
||||||
|
return this.audioTrack;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class TrackWrapper implements Track {
|
export class TrackWrapper implements Track {
|
||||||
constructor(
|
constructor(
|
||||||
public readonly track: MediaStreamTrack,
|
public readonly track: MediaStreamTrack,
|
||||||
public readonly stream: MediaStream,
|
public readonly stream: MediaStream
|
||||||
private _type: TrackType,
|
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
get type(): TrackType { return this._type; }
|
get kind(): TrackKind { return this.track.kind as TrackKind; }
|
||||||
get label(): string { return this.track.label; }
|
get label(): string { return this.track.label; }
|
||||||
get id(): string { return this.track.id; }
|
get id(): string { return this.track.id; }
|
||||||
get streamId(): string { return this.stream.id; }
|
|
||||||
get muted(): boolean { return this.track.muted; }
|
|
||||||
get settings(): MediaTrackSettings { return this.track.getSettings(); }
|
get settings(): MediaTrackSettings { return this.track.getSettings(); }
|
||||||
|
|
||||||
setMuted(muted: boolean): void {
|
stop() { this.track.stop(); }
|
||||||
this.track.enabled = !muted;
|
|
||||||
}
|
|
||||||
|
|
||||||
setType(type: TrackType): void {
|
|
||||||
this._type = type;
|
|
||||||
}
|
|
||||||
|
|
||||||
stop() {
|
|
||||||
this.track.stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
clone() {
|
|
||||||
return this.track.clone();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export class AudioTrackWrapper extends TrackWrapper {
|
export class AudioTrackWrapper extends TrackWrapper {
|
||||||
@ -127,8 +126,8 @@ export class AudioTrackWrapper extends TrackWrapper {
|
|||||||
private volumeLooperTimeout: number;
|
private volumeLooperTimeout: number;
|
||||||
private speakingVolumeSamples: number[];
|
private speakingVolumeSamples: number[];
|
||||||
|
|
||||||
constructor(track: MediaStreamTrack, stream: MediaStream, type: TrackType) {
|
constructor(track: MediaStreamTrack, stream: MediaStream) {
|
||||||
super(track, stream, type);
|
super(track, stream);
|
||||||
this.speakingVolumeSamples = new Array(SPEAKING_SAMPLE_COUNT).fill(-Infinity);
|
this.speakingVolumeSamples = new Array(SPEAKING_SAMPLE_COUNT).fill(-Infinity);
|
||||||
this.initVolumeMeasuring();
|
this.initVolumeMeasuring();
|
||||||
this.measureVolumeActivity(true);
|
this.measureVolumeActivity(true);
|
||||||
|
@ -14,9 +14,9 @@ See the License for the specific language governing permissions and
|
|||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {TrackWrapper, wrapTrack} from "./MediaDevices";
|
import {StreamWrapper, TrackWrapper, AudioTrackWrapper} from "./MediaDevices";
|
||||||
import {Track, TrackType} from "../../types/MediaDevices";
|
import {Stream, Track, AudioTrack, TrackKind} from "../../types/MediaDevices";
|
||||||
import {WebRTC, PeerConnectionHandler, DataChannel, PeerConnection} from "../../types/WebRTC";
|
import {WebRTC, PeerConnectionHandler, StreamSender, TrackSender, StreamReceiver, TrackReceiver, PeerConnection} from "../../types/WebRTC";
|
||||||
import {SDPStreamMetadataPurpose} from "../../../matrix/calls/callEventTypes";
|
import {SDPStreamMetadataPurpose} from "../../../matrix/calls/callEventTypes";
|
||||||
|
|
||||||
const POLLING_INTERVAL = 200; // ms
|
const POLLING_INTERVAL = 200; // ms
|
||||||
@ -29,11 +29,171 @@ export class DOMWebRTC implements WebRTC {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export class RemoteStreamWrapper extends StreamWrapper {
|
||||||
|
constructor(stream: MediaStream, private readonly emptyCallback: (stream: RemoteStreamWrapper) => void) {
|
||||||
|
super(stream);
|
||||||
|
this.stream.addEventListener("removetrack", this.onTrackRemoved);
|
||||||
|
}
|
||||||
|
|
||||||
|
onTrackRemoved = (evt: MediaStreamTrackEvent) => {
|
||||||
|
if (evt.track.id === this.audioTrack?.track.id) {
|
||||||
|
this.audioTrack = undefined;
|
||||||
|
} else if (evt.track.id === this.videoTrack?.track.id) {
|
||||||
|
this.videoTrack = undefined;
|
||||||
|
}
|
||||||
|
if (!this.audioTrack && !this.videoTrack) {
|
||||||
|
this.emptyCallback(this);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
dispose() {
|
||||||
|
this.stream.removeEventListener("removetrack", this.onTrackRemoved);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class DOMStreamSender implements StreamSender {
|
||||||
|
public audioSender: DOMTrackSender | undefined;
|
||||||
|
public videoSender: DOMTrackSender | undefined;
|
||||||
|
|
||||||
|
constructor(public readonly stream: StreamWrapper) {}
|
||||||
|
|
||||||
|
update(transceivers: ReadonlyArray<RTCRtpTransceiver>, sender: RTCRtpSender): DOMTrackSender | undefined {
|
||||||
|
const transceiver = transceivers.find(t => t.sender === sender);
|
||||||
|
if (transceiver && sender.track) {
|
||||||
|
const trackWrapper = this.stream.update(sender.track);
|
||||||
|
if (trackWrapper) {
|
||||||
|
if (trackWrapper.kind === TrackKind.Video) {
|
||||||
|
this.videoSender = new DOMTrackSender(trackWrapper, transceiver);
|
||||||
|
return this.videoSender;
|
||||||
|
} else {
|
||||||
|
this.audioSender = new DOMTrackSender(trackWrapper, transceiver);
|
||||||
|
return this.audioSender;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class DOMStreamReceiver implements StreamReceiver {
|
||||||
|
public audioReceiver: DOMTrackReceiver | undefined;
|
||||||
|
public videoReceiver: DOMTrackReceiver | undefined;
|
||||||
|
|
||||||
|
constructor(public readonly stream: RemoteStreamWrapper) {}
|
||||||
|
|
||||||
|
update(event: RTCTrackEvent): DOMTrackReceiver | undefined {
|
||||||
|
const {receiver} = event;
|
||||||
|
const {track} = receiver;
|
||||||
|
const trackWrapper = this.stream.update(track);
|
||||||
|
if (trackWrapper) {
|
||||||
|
if (trackWrapper.kind === TrackKind.Video) {
|
||||||
|
this.videoReceiver = new DOMTrackReceiver(trackWrapper, event.transceiver);
|
||||||
|
return this.videoReceiver;
|
||||||
|
} else {
|
||||||
|
this.audioReceiver = new DOMTrackReceiver(trackWrapper, event.transceiver);
|
||||||
|
return this.audioReceiver;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class DOMTrackSenderOrReceiver implements TrackReceiver {
|
||||||
|
constructor(
|
||||||
|
public readonly track: TrackWrapper,
|
||||||
|
public readonly transceiver: RTCRtpTransceiver,
|
||||||
|
private readonly exclusiveValue: RTCRtpTransceiverDirection,
|
||||||
|
private readonly excludedValue: RTCRtpTransceiverDirection
|
||||||
|
) {}
|
||||||
|
|
||||||
|
get enabled(): boolean {
|
||||||
|
return this.transceiver.currentDirection === "sendrecv" ||
|
||||||
|
this.transceiver.currentDirection === this.exclusiveValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
enable(enabled: boolean) {
|
||||||
|
if (enabled !== this.enabled) {
|
||||||
|
if (enabled) {
|
||||||
|
if (this.transceiver.currentDirection === "inactive") {
|
||||||
|
this.transceiver.direction = this.exclusiveValue;
|
||||||
|
} else {
|
||||||
|
this.transceiver.direction = "sendrecv";
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (this.transceiver.currentDirection === "sendrecv") {
|
||||||
|
this.transceiver.direction = this.excludedValue;
|
||||||
|
} else {
|
||||||
|
this.transceiver.direction = "inactive";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class DOMTrackReceiver extends DOMTrackSenderOrReceiver {
|
||||||
|
constructor(
|
||||||
|
track: TrackWrapper,
|
||||||
|
transceiver: RTCRtpTransceiver,
|
||||||
|
) {
|
||||||
|
super(track, transceiver, "recvonly", "sendonly");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class DOMTrackSender extends DOMTrackSenderOrReceiver {
|
||||||
|
constructor(
|
||||||
|
track: TrackWrapper,
|
||||||
|
transceiver: RTCRtpTransceiver,
|
||||||
|
) {
|
||||||
|
super(track, transceiver, "sendonly", "recvonly");
|
||||||
|
}
|
||||||
|
/** replaces the track if possible without renegotiation. Can throw. */
|
||||||
|
replaceTrack(track: Track): Promise<void> {
|
||||||
|
return this.transceiver.sender.replaceTrack(track ? (track as TrackWrapper).track : null);
|
||||||
|
}
|
||||||
|
|
||||||
|
prepareForPurpose(purpose: SDPStreamMetadataPurpose): void {
|
||||||
|
if (purpose === SDPStreamMetadataPurpose.Screenshare) {
|
||||||
|
this.getRidOfRTXCodecs();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method removes all video/rtx codecs from screensharing video
|
||||||
|
* transceivers. This is necessary since they can cause problems. Without
|
||||||
|
* this the following steps should produce an error:
|
||||||
|
* Chromium calls Firefox
|
||||||
|
* Firefox answers
|
||||||
|
* Firefox starts screen-sharing
|
||||||
|
* Chromium starts screen-sharing
|
||||||
|
* Call crashes for Chromium with:
|
||||||
|
* [96685:23:0518/162603.933321:ERROR:webrtc_video_engine.cc(3296)] RTX codec (PT=97) mapped to PT=96 which is not in the codec list.
|
||||||
|
* [96685:23:0518/162603.933377:ERROR:webrtc_video_engine.cc(1171)] GetChangedRecvParameters called without any video codecs.
|
||||||
|
* [96685:23:0518/162603.933430:ERROR:sdp_offer_answer.cc(4302)] Failed to set local video description recv parameters for m-section with mid='2'. (INVALID_PARAMETER)
|
||||||
|
*/
|
||||||
|
private getRidOfRTXCodecs(): void {
|
||||||
|
// RTCRtpReceiver.getCapabilities and RTCRtpSender.getCapabilities don't seem to be supported on FF
|
||||||
|
if (!RTCRtpReceiver.getCapabilities || !RTCRtpSender.getCapabilities) return;
|
||||||
|
|
||||||
|
const recvCodecs = RTCRtpReceiver.getCapabilities("video")?.codecs ?? [];
|
||||||
|
const sendCodecs = RTCRtpSender.getCapabilities("video")?.codecs ?? [];
|
||||||
|
const codecs = [...sendCodecs, ...recvCodecs];
|
||||||
|
|
||||||
|
for (const codec of codecs) {
|
||||||
|
if (codec.mimeType === "video/rtx") {
|
||||||
|
const rtxCodecIndex = codecs.indexOf(codec);
|
||||||
|
codecs.splice(rtxCodecIndex, 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (this.transceiver.sender.track?.kind === "video" ||
|
||||||
|
this.transceiver.receiver.track?.kind === "video") {
|
||||||
|
this.transceiver.setCodecPreferences(codecs);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
class DOMPeerConnection implements PeerConnection {
|
class DOMPeerConnection implements PeerConnection {
|
||||||
private readonly peerConnection: RTCPeerConnection;
|
private readonly peerConnection: RTCPeerConnection;
|
||||||
private readonly handler: PeerConnectionHandler;
|
private readonly handler: PeerConnectionHandler;
|
||||||
//private dataChannelWrapper?: DOMDataChannel;
|
public readonly localStreams: DOMStreamSender[];
|
||||||
private _remoteTracks: TrackWrapper[] = [];
|
public readonly remoteStreams: DOMStreamReceiver[];
|
||||||
|
|
||||||
constructor(handler: PeerConnectionHandler, forceTURN: boolean, turnServers: RTCIceServer[], iceCandidatePoolSize) {
|
constructor(handler: PeerConnectionHandler, forceTURN: boolean, turnServers: RTCIceServer[], iceCandidatePoolSize) {
|
||||||
this.handler = handler;
|
this.handler = handler;
|
||||||
@ -45,7 +205,6 @@ class DOMPeerConnection implements PeerConnection {
|
|||||||
this.registerHandler();
|
this.registerHandler();
|
||||||
}
|
}
|
||||||
|
|
||||||
get remoteTracks(): Track[] { return this._remoteTracks; }
|
|
||||||
get iceGatheringState(): RTCIceGatheringState { return this.peerConnection.iceGatheringState; }
|
get iceGatheringState(): RTCIceGatheringState { return this.peerConnection.iceGatheringState; }
|
||||||
get localDescription(): RTCSessionDescription | undefined { return this.peerConnection.localDescription ?? undefined; }
|
get localDescription(): RTCSessionDescription | undefined { return this.peerConnection.localDescription ?? undefined; }
|
||||||
get signalingState(): RTCSignalingState { return this.peerConnection.signalingState; }
|
get signalingState(): RTCSignalingState { return this.peerConnection.signalingState; }
|
||||||
@ -74,48 +233,26 @@ class DOMPeerConnection implements PeerConnection {
|
|||||||
return this.peerConnection.close();
|
return this.peerConnection.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
addTrack(track: Track): void {
|
addTrack(track: Track): DOMTrackSender | undefined {
|
||||||
if (!(track instanceof TrackWrapper)) {
|
if (!(track instanceof TrackWrapper)) {
|
||||||
throw new Error("Not a TrackWrapper");
|
throw new Error("Not a TrackWrapper");
|
||||||
}
|
}
|
||||||
this.peerConnection.addTrack(track.track, track.stream);
|
const sender = this.peerConnection.addTrack(track.track, track.stream);
|
||||||
if (track.type === TrackType.ScreenShare) {
|
let streamSender: DOMStreamSender | undefined = this.localStreams.find(s => s.stream.id === track.stream.id);
|
||||||
this.getRidOfRTXCodecs(track);
|
if (!streamSender) {
|
||||||
|
streamSender = new DOMStreamSender(new StreamWrapper(track.stream));
|
||||||
|
this.localStreams.push(streamSender);
|
||||||
}
|
}
|
||||||
|
const trackSender = streamSender.update(this.peerConnection.getTransceivers(), sender);
|
||||||
|
return trackSender;
|
||||||
}
|
}
|
||||||
|
|
||||||
removeTrack(track: Track): boolean {
|
removeTrack(sender: TrackSender): void {
|
||||||
if (!(track instanceof TrackWrapper)) {
|
if (!(sender instanceof DOMTrackSender)) {
|
||||||
throw new Error("Not a TrackWrapper");
|
throw new Error("Not a DOMTrackSender");
|
||||||
}
|
|
||||||
const sender = this.peerConnection.getSenders().find(s => s.track === track.track);
|
|
||||||
if (sender) {
|
|
||||||
this.peerConnection.removeTrack(sender);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
async replaceTrack(oldTrack: Track, newTrack: Track): Promise<boolean> {
|
|
||||||
if (!(oldTrack instanceof TrackWrapper) || !(newTrack instanceof TrackWrapper)) {
|
|
||||||
throw new Error("Not a TrackWrapper");
|
|
||||||
}
|
|
||||||
const sender = this.peerConnection.getSenders().find(s => s.track === oldTrack.track);
|
|
||||||
if (sender) {
|
|
||||||
await sender.replaceTrack(newTrack.track);
|
|
||||||
if (newTrack.type === TrackType.ScreenShare) {
|
|
||||||
this.getRidOfRTXCodecs(newTrack);
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
notifyStreamPurposeChanged(): void {
|
|
||||||
for (const track of this.remoteTracks) {
|
|
||||||
const wrapper = track as TrackWrapper;
|
|
||||||
wrapper.setType(this.getRemoteTrackType(wrapper.track, wrapper.streamId));
|
|
||||||
}
|
}
|
||||||
|
this.peerConnection.removeTrack((sender as DOMTrackSender).transceiver.sender);
|
||||||
|
// TODO: update localStreams
|
||||||
}
|
}
|
||||||
|
|
||||||
createDataChannel(options: RTCDataChannelInit): any {
|
createDataChannel(options: RTCDataChannelInit): any {
|
||||||
@ -170,6 +307,9 @@ class DOMPeerConnection implements PeerConnection {
|
|||||||
|
|
||||||
dispose(): void {
|
dispose(): void {
|
||||||
this.deregisterHandler();
|
this.deregisterHandler();
|
||||||
|
for (const r of this.remoteStreams) {
|
||||||
|
r.stream.dispose();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private handleLocalIceCandidate(event: RTCPeerConnectionIceEvent) {
|
private handleLocalIceCandidate(event: RTCPeerConnectionIceEvent) {
|
||||||
@ -187,67 +327,28 @@ class DOMPeerConnection implements PeerConnection {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
onRemoteStreamEmpty = (stream: RemoteStreamWrapper): void => {
|
||||||
|
const idx = this.remoteStreams.findIndex(r => r.stream === stream);
|
||||||
|
if (idx !== -1) {
|
||||||
|
this.remoteStreams.splice(idx, 1);
|
||||||
|
this.handler.onRemoteStreamRemoved(stream);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private handleRemoteTrack(evt: RTCTrackEvent) {
|
private handleRemoteTrack(evt: RTCTrackEvent) {
|
||||||
// TODO: unit test this code somehow
|
if (evt.streams.length !== 0) {
|
||||||
// the tracks on the new stream (with their stream)
|
throw new Error("track in multiple streams is not supported");
|
||||||
const updatedTracks = evt.streams.flatMap(stream => stream.getTracks().map(track => {return {stream, track};}));
|
|
||||||
// of the tracks we already know about, filter the ones that aren't in the new stream
|
|
||||||
const withoutRemovedTracks = this._remoteTracks.filter(t => updatedTracks.some(ut => t.track.id === ut.track.id));
|
|
||||||
// of the new tracks, filter the ones that we didn't already knew about
|
|
||||||
const addedTracks = updatedTracks.filter(ut => !this._remoteTracks.some(t => t.track.id === ut.track.id));
|
|
||||||
// wrap them
|
|
||||||
const wrappedAddedTracks = addedTracks.map(t => wrapTrack(t.track, t.stream, this.getRemoteTrackType(t.track, t.stream.id)));
|
|
||||||
// and concat the tracks for other streams with the added tracks
|
|
||||||
this._remoteTracks = withoutRemovedTracks.concat(...wrappedAddedTracks);
|
|
||||||
this.handler.onRemoteTracksChanged(this.remoteTracks);
|
|
||||||
}
|
|
||||||
|
|
||||||
private getRemoteTrackType(track: MediaStreamTrack, streamId: string): TrackType {
|
|
||||||
if (track.kind === "video") {
|
|
||||||
const purpose = this.handler.getPurposeForStreamId(streamId);
|
|
||||||
return purpose === SDPStreamMetadataPurpose.Usermedia ? TrackType.Camera : TrackType.ScreenShare;
|
|
||||||
} else {
|
|
||||||
return TrackType.Microphone;
|
|
||||||
}
|
}
|
||||||
}
|
const stream = evt.streams[0];
|
||||||
|
const transceivers = this.peerConnection.getTransceivers();
|
||||||
/**
|
let streamReceiver: DOMStreamReceiver | undefined = this.remoteStreams.find(r => r.stream.id === stream.id);
|
||||||
* This method removes all video/rtx codecs from screensharing video
|
if (!streamReceiver) {
|
||||||
* transceivers. This is necessary since they can cause problems. Without
|
streamReceiver = new DOMStreamReceiver(new RemoteStreamWrapper(stream, this.onRemoteStreamEmpty));
|
||||||
* this the following steps should produce an error:
|
this.remoteStreams.push(streamReceiver);
|
||||||
* Chromium calls Firefox
|
|
||||||
* Firefox answers
|
|
||||||
* Firefox starts screen-sharing
|
|
||||||
* Chromium starts screen-sharing
|
|
||||||
* Call crashes for Chromium with:
|
|
||||||
* [96685:23:0518/162603.933321:ERROR:webrtc_video_engine.cc(3296)] RTX codec (PT=97) mapped to PT=96 which is not in the codec list.
|
|
||||||
* [96685:23:0518/162603.933377:ERROR:webrtc_video_engine.cc(1171)] GetChangedRecvParameters called without any video codecs.
|
|
||||||
* [96685:23:0518/162603.933430:ERROR:sdp_offer_answer.cc(4302)] Failed to set local video description recv parameters for m-section with mid='2'. (INVALID_PARAMETER)
|
|
||||||
*/
|
|
||||||
private getRidOfRTXCodecs(screensharingTrack: TrackWrapper): void {
|
|
||||||
// RTCRtpReceiver.getCapabilities and RTCRtpSender.getCapabilities don't seem to be supported on FF
|
|
||||||
if (!RTCRtpReceiver.getCapabilities || !RTCRtpSender.getCapabilities) return;
|
|
||||||
|
|
||||||
const recvCodecs = RTCRtpReceiver.getCapabilities("video")?.codecs ?? [];
|
|
||||||
const sendCodecs = RTCRtpSender.getCapabilities("video")?.codecs ?? [];
|
|
||||||
const codecs = [...sendCodecs, ...recvCodecs];
|
|
||||||
|
|
||||||
for (const codec of codecs) {
|
|
||||||
if (codec.mimeType === "video/rtx") {
|
|
||||||
const rtxCodecIndex = codecs.indexOf(codec);
|
|
||||||
codecs.splice(rtxCodecIndex, 1);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
const trackReceiver = streamReceiver.update(evt);
|
||||||
for (const trans of this.peerConnection.getTransceivers()) {
|
if (trackReceiver) {
|
||||||
if (trans.sender.track === screensharingTrack.track &&
|
this.handler.onRemoteTracksAdded(trackReceiver);
|
||||||
(
|
|
||||||
trans.sender.track?.kind === "video" ||
|
|
||||||
trans.receiver.track?.kind === "video"
|
|
||||||
)
|
|
||||||
) {
|
|
||||||
trans.setCodecPreferences(codecs);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user