mirror of
https://github.com/element-hq/element-web.git
synced 2024-12-17 06:21:56 +03:00
98 lines
3.6 KiB
TypeScript
98 lines
3.6 KiB
TypeScript
|
/*
|
||
|
Copyright 2021 Šimon Brandner <simon.bra.ag@gmail.com>
|
||
|
|
||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
you may not use this file except in compliance with the License.
|
||
|
You may obtain a copy of the License at
|
||
|
|
||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||
|
|
||
|
Unless required by applicable law or agreed to in writing, software
|
||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
See the License for the specific language governing permissions and
|
||
|
limitations under the License.
|
||
|
*/
|
||
|
|
||
|
import React, {createRef} from 'react';
|
||
|
import { CallFeed, CallFeedEvent } from 'matrix-js-sdk/src/webrtc/callFeed';
|
||
|
import { logger } from 'matrix-js-sdk/src/logger';
|
||
|
import CallMediaHandler from "../../../CallMediaHandler";
|
||
|
|
||
|
interface IProps {
|
||
|
feed: CallFeed,
|
||
|
}
|
||
|
|
||
|
export default class AudioFeed extends React.Component<IProps> {
|
||
|
private element = createRef<HTMLAudioElement>();
|
||
|
|
||
|
componentDidMount() {
|
||
|
this.props.feed.addListener(CallFeedEvent.NewStream, this.onNewStream);
|
||
|
this.playMedia();
|
||
|
}
|
||
|
|
||
|
componentWillUnmount() {
|
||
|
this.props.feed.removeListener(CallFeedEvent.NewStream, this.onNewStream);
|
||
|
this.stopMedia();
|
||
|
}
|
||
|
|
||
|
private playMedia() {
|
||
|
const element = this.element.current;
|
||
|
const audioOutput = CallMediaHandler.getAudioOutput();
|
||
|
|
||
|
if (audioOutput) {
|
||
|
try {
|
||
|
// This seems quite unreliable in Chrome, although I haven't yet managed to make a jsfiddle where
|
||
|
// it fails.
|
||
|
// It seems reliable if you set the sink ID after setting the srcObject and then set the sink ID
|
||
|
// back to the default after the call is over - Dave
|
||
|
element.setSinkId(audioOutput);
|
||
|
} catch (e) {
|
||
|
console.error("Couldn't set requested audio output device: using default", e);
|
||
|
logger.warn("Couldn't set requested audio output device: using default", e);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
element.muted = false;
|
||
|
element.srcObject = this.props.feed.stream;
|
||
|
element.autoplay = true;
|
||
|
|
||
|
try {
|
||
|
// A note on calling methods on media elements:
|
||
|
// We used to have queues per media element to serialise all calls on those elements.
|
||
|
// The reason given for this was that load() and play() were racing. However, we now
|
||
|
// never call load() explicitly so this seems unnecessary. However, serialising every
|
||
|
// operation was causing bugs where video would not resume because some play command
|
||
|
// had got stuck and all media operations were queued up behind it. If necessary, we
|
||
|
// should serialise the ones that need to be serialised but then be able to interrupt
|
||
|
// them with another load() which will cancel the pending one, but since we don't call
|
||
|
// load() explicitly, it shouldn't be a problem. - Dave
|
||
|
element.play()
|
||
|
} catch (e) {
|
||
|
logger.info("Failed to play media element with feed", this.props.feed, e);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private stopMedia() {
|
||
|
const element = this.element.current;
|
||
|
|
||
|
element.pause();
|
||
|
element.src = null;
|
||
|
|
||
|
// As per comment in componentDidMount, setting the sink ID back to the
|
||
|
// default once the call is over makes setSinkId work reliably. - Dave
|
||
|
// Since we are not using the same element anymore, the above doesn't
|
||
|
// seem to be necessary - Šimon
|
||
|
}
|
||
|
|
||
|
private onNewStream = () => {
|
||
|
this.playMedia();
|
||
|
};
|
||
|
|
||
|
render() {
|
||
|
return (
|
||
|
<audio ref={this.element} />
|
||
|
);
|
||
|
}
|
||
|
}
|