Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix "system default" option for media devices #5955

Open
wants to merge 18 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion src/components/bone-mute-state-indicator.js
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ AFRAME.registerComponent("bone-mute-state-indicator", {

updateMuteState() {
if (!this.modelLoaded) return;
const muted = !APP.mediaDevicesManager.isMicEnabled;
const muted = !APP.dialog.isMicEnabled;
this.mutedBone.position.y = muted ? this.data.onPos : this.data.offPos;
this.unmutedBone.position.y = !muted ? this.data.onPos : this.data.offPos;
this.mutedBone.matrixNeedsUpdate = true;
Expand Down
2 changes: 1 addition & 1 deletion src/components/in-world-hud.js
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@ AFRAME.registerComponent("in-world-hud", {
};

this.onMicClick = () => {
APP.mediaDevicesManager.toggleMic();
APP.dialog.toggleMicrophone();
};

this.onSpawnClick = () => {
Expand Down
7 changes: 4 additions & 3 deletions src/components/mute-mic.js
Original file line number Diff line number Diff line change
Expand Up @@ -52,16 +52,17 @@ AFRAME.registerComponent("mute-mic", {
},

onToggle: function () {
APP.mediaDevicesManager.toggleMic();
APP.dialog.toggleMicrophone();

if (!this.el.sceneEl.is("entered")) return;
this.el.sceneEl.systems["hubs-systems"].soundEffectsSystem.playSoundOneShot(SOUND_TOGGLE_MIC);
},

onMute: function () {
APP.mediaDevicesManager.micEnabled = false;
APP.dialog.enableMicrophone(false);
},

onUnmute: function () {
APP.mediaDevicesManager.micEnabled = true;
APP.dialog.enableMicrophone(true);
}
});
7 changes: 6 additions & 1 deletion src/hub.js
Original file line number Diff line number Diff line change
Expand Up @@ -764,6 +764,11 @@ document.addEventListener("DOMContentLoaded", async () => {
const authChannel = new AuthChannel(store);
const hubChannel = new HubChannel(store, hubId);
window.APP.hubChannel = hubChannel;
hubChannel.addEventListener("permissions_updated", () => {
if (!hubChannel.can("voice_chat")) {
APP.dialog.enableMicrophone(false);
}
});

const entryManager = new SceneEntryManager(hubChannel, authChannel, history);
window.APP.entryManager = entryManager;
Expand Down Expand Up @@ -1397,7 +1402,7 @@ document.addEventListener("DOMContentLoaded", async () => {

hubPhxChannel.on("mute", ({ session_id }) => {
if (session_id === NAF.clientId) {
APP.mediaDevicesManager.micEnabled = false;
APP.dialog.enableMicrophone(false);
}
});

Expand Down
18 changes: 14 additions & 4 deletions src/react-components/preferences-screen.js
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,12 @@ import styles from "../assets/stylesheets/preferences-screen.scss";
import { AVAILABLE_LOCALES } from "../assets/locales/locale_config";
import { themes } from "../utils/theme";
import MediaDevicesManager from "../utils/media-devices-manager";
import { MediaDevices, MediaDevicesEvents, PermissionStatus } from "../utils/media-devices-utils";
import {
DEFAULT_MEDIA_DEVICE_OPTION,
MediaDevices,
MediaDevicesEvents,
PermissionStatus
} from "../utils/media-devices-utils";
import { Slider } from "./input/Slider";
import {
addOrientationChangeListener,
Expand Down Expand Up @@ -877,25 +882,30 @@ class PreferencesScreen extends Component {

this.mediaDevicesManager = APP.mediaDevicesManager;

const defaultMediaDeviceOption = {
value: DEFAULT_MEDIA_DEVICE_OPTION.value,
text: DEFAULT_MEDIA_DEVICE_OPTION.label
};

this.state = {
category: CATEGORY_AUDIO,
toastHeight: "150px",
preferredMic: {
key: "preferredMic",
prefType: PREFERENCE_LIST_ITEM_TYPE.SELECT,
options: [{ value: "none", text: "None" }],
options: [defaultMediaDeviceOption],
disabled: !canVoiceChat
},
preferredCamera: {
key: "preferredCamera",
prefType: PREFERENCE_LIST_ITEM_TYPE.SELECT,
options: [{ value: "none", text: "None" }]
options: [defaultMediaDeviceOption]
},
...(MediaDevicesManager.isAudioOutputSelectEnabled && {
preferredSpeakers: {
key: "preferredSpeakers",
prefType: PREFERENCE_LIST_ITEM_TYPE.SELECT,
options: [{ value: "none", text: "None" }]
options: [defaultMediaDeviceOption]
}
}),
canVoiceChat
Expand Down
4 changes: 2 additions & 2 deletions src/react-components/room/useMicrophoneStatus.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ import { MediaDevices, MediaDevicesEvents } from "../../utils/media-devices-util

export function useMicrophoneStatus(scene) {
const mediaDevicesManager = APP.mediaDevicesManager;
const [isMicMuted, setIsMicMuted] = useState(!mediaDevicesManager.isMicEnabled);
const [isMicMuted, setIsMicMuted] = useState(!APP.dialog.isMicEnabled);
const [isMicEnabled, setIsMicEnabled] = useState(APP.mediaDevicesManager.isMicShared);
const [permissionStatus, setPermissionsStatus] = useState(
mediaDevicesManager.getPermissionsStatus(MediaDevices.MICROPHONE)
Expand Down Expand Up @@ -41,7 +41,7 @@ export function useMicrophoneStatus(scene) {

const toggleMute = useCallback(() => {
if (mediaDevicesManager.isMicShared) {
mediaDevicesManager.toggleMic();
APP.dialog.toggleMicrophone();
} else {
mediaDevicesManager.startMicShare({ unmute: true });
}
Expand Down
2 changes: 1 addition & 1 deletion src/scene-entry-manager.js
Original file line number Diff line number Diff line change
Expand Up @@ -131,7 +131,7 @@ export default class SceneEntryManager {

this.scene.addState("entered");

APP.mediaDevicesManager.micEnabled = !muteOnEntry;
APP.dialog.enableMicrophone(!muteOnEntry);
};

whenSceneLoaded = callback => {
Expand Down
2 changes: 1 addition & 1 deletion src/storage/store.js
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ const OAUTH_FLOW_CREDENTIALS_KEY = "ret-oauth-flow-account-credentials";
const validator = new Validator();
import { EventTarget } from "event-target-shim";
import { fetchRandomDefaultAvatarId, generateRandomName } from "../utils/identity.js";
import { NO_DEVICE_ID } from "../utils/media-devices-utils.js";
import { NO_DEVICE_ID } from "../utils/media-devices-utils";
import { AAModes } from "../constants";

const defaultMaterialQuality = (function () {
Expand Down
3 changes: 2 additions & 1 deletion src/systems/audio-system.js
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ function performDelayedReconnect(gainNode) {

import * as sdpTransform from "sdp-transform";
import MediaDevicesManager from "../utils/media-devices-manager";
import { NO_DEVICE_ID } from "../utils/media-devices-utils";

function isThreeAudio(node) {
return node instanceof THREE.Audio || node instanceof THREE.PositionalAudio;
Expand Down Expand Up @@ -218,7 +219,7 @@ export class AudioSystem {

if (MediaDevicesManager.isAudioOutputSelectEnabled && APP.mediaDevicesManager) {
const sinkId = APP.mediaDevicesManager.selectedSpeakersDeviceId;
const isDefault = sinkId === APP.mediaDevicesManager.defaultOutputDeviceId;
const isDefault = sinkId === NO_DEVICE_ID;
if ((!this.outputMediaAudio && isDefault) || sinkId === this.outputMediaAudio?.sinkId) return;
const sink = isDefault ? this._sceneEl.audioListener.getInput() : this.audioDestination;
this.mixer[SourceType.AVATAR_AUDIO_SOURCE].disconnect();
Expand Down
139 changes: 34 additions & 105 deletions src/utils/media-devices-manager.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,13 @@
import { EventEmitter } from "eventemitter3";
import { MediaDevicesEvents, PermissionStatus, MediaDevices, NO_DEVICE_ID } from "./media-devices-utils";
import {
MediaDevicesEvents,
PermissionStatus,
MediaDevices,
NO_DEVICE_ID,
optionFor,
getValidMediaDevices,
DEFAULT_MEDIA_DEVICE_OPTION
} from "./media-devices-utils";
import { detectOS, detect } from "detect-browser";
import { isIOS as detectIOS } from "./is-mobile";

Expand Down Expand Up @@ -33,8 +41,7 @@ export default class MediaDevicesManager extends EventEmitter {
this._micDevices = [];
this._videoDevices = [];
this._outputDevices = [];
this._deviceId = null;
this._audioTrack = null;
this.audioTrack = null;
this.audioSystem = audioSystem;
this._mediaStream = audioSystem.outboundStream;
this._permissionsStatus = {
Expand All @@ -46,8 +53,6 @@ export default class MediaDevicesManager extends EventEmitter {

this.onDeviceChange = this.onDeviceChange.bind(this);
navigator.mediaDevices.addEventListener("devicechange", this.onDeviceChange);
this.onPermissionsUpdated = this.onPermissionsUpdated.bind(this);
APP.hubChannel.addEventListener("permissions_updated", this.onPermissionsUpdated);
}

static get isAudioOutputSelectEnabled() {
Expand All @@ -58,44 +63,16 @@ export default class MediaDevicesManager extends EventEmitter {
return audioInputSelectEnabled;
}

get deviceId() {
return this._deviceId;
}

set deviceId(deviceId) {
this._deviceId = deviceId;
}

get audioTrack() {
return this._audioTrack;
}

set audioTrack(audioTrack) {
this._audioTrack = audioTrack;
}

get defaultInputDeviceId() {
return this._micDevices.length > 0 ? this._micDevices[0].value : NO_DEVICE_ID;
}

get defaultOutputDeviceId() {
return this._outputDevices.length > 0 ? this._outputDevices[0].value : NO_DEVICE_ID;
}

get defaultVideoDeviceId() {
return this._videoDevices.length > 0 ? this._videoDevices[0].value : NO_DEVICE_ID;
}

get micDevicesOptions() {
return this._micDevices.length > 0 ? this._micDevices : [{ value: NO_DEVICE_ID, label: "None" }];
return [DEFAULT_MEDIA_DEVICE_OPTION, ...this._micDevices];
}

get videoDevicesOptions() {
return this._videoDevices.length > 0 ? this._videoDevices : [{ value: NO_DEVICE_ID, label: "None" }];
return [DEFAULT_MEDIA_DEVICE_OPTION, ...this._videoDevices];
}

get outputDevicesOptions() {
return this._outputDevices.length > 0 ? this._outputDevices : [{ value: NO_DEVICE_ID, label: "None" }];
return [DEFAULT_MEDIA_DEVICE_OPTION, ...this._outputDevices];
}

get mediaStream() {
Expand All @@ -122,7 +99,7 @@ export default class MediaDevicesManager extends EventEmitter {
const exists = this._outputDevices.some(device => {
return device.value === preferredSpeakers;
});
return exists ? preferredSpeakers : this.defaultOutputDeviceId;
return exists ? preferredSpeakers : NO_DEVICE_ID;
}

get isMicShared() {
Expand All @@ -145,75 +122,37 @@ export default class MediaDevicesManager extends EventEmitter {
});
}

set micEnabled(enabled) {
APP.dialog.enableMicrophone(enabled);
}

get isMicEnabled() {
return APP.dialog.isMicEnabled;
}

toggleMic() {
APP.dialog.toggleMicrophone();
}

getPermissionsStatus(type) {
return this._permissionsStatus[type];
}

onPermissionsUpdated = () => {
if (!APP.hubChannel.can("voice_chat")) {
APP.dialog.enableMicrophone(false);
}
};

onDeviceChange = () => {
this.fetchMediaDevices().then(() => {
this.changeAudioOutput(this.selectedSpeakersDeviceId);
this.emit(MediaDevicesEvents.DEVICE_CHANGE, null);
});
};

updatePermissions() {
const micStatus = this._micDevices.length === 0 ? PermissionStatus.PROMPT : PermissionStatus.GRANTED;
this._permissionsStatus[MediaDevices.MICROPHONE] = micStatus;
this.emit(MediaDevicesEvents.PERMISSIONS_STATUS_CHANGED, {
mediaDevice: MediaDevices.MICROPHONE,
status: micStatus
});
const videoStatus = this._videoDevices.length === 0 ? PermissionStatus.PROMPT : PermissionStatus.GRANTED;
this._permissionsStatus[MediaDevices.CAMERA] = videoStatus;
this.emit(MediaDevicesEvents.PERMISSIONS_STATUS_CHANGED, {
mediaDevice: MediaDevices.CAMERA,
status: videoStatus
});
const speakersStatus = this._micDevices.length === 0 ? PermissionStatus.PROMPT : PermissionStatus.GRANTED;
this._permissionsStatus[MediaDevices.SPEAKERS] = speakersStatus;
updatePermissionStatus(mediaDevice, shouldPrompt) {
const status = shouldPrompt ? PermissionStatus.PROMPT : PermissionStatus.GRANTED;
this._permissionsStatus[mediaDevice] = status;
this.emit(MediaDevicesEvents.PERMISSIONS_STATUS_CHANGED, {
mediaDevice: MediaDevices.SPEAKERS,
status: speakersStatus
mediaDevice,
status
});
}

async fetchMediaDevices() {
console.log("Fetching media devices");
return new Promise(resolve => {
navigator.mediaDevices.enumerateDevices().then(mediaDevices => {
mediaDevices = mediaDevices.filter(d => d.label !== "");
this._micDevices = mediaDevices
.filter(d => d.deviceId !== "default" && d.kind === "audioinput")
.map(d => ({ value: d.deviceId, label: d.label || `Mic Device (${d.deviceId.substring(0, 9)})` }));
this._videoDevices = mediaDevices
.filter(d => d.deviceId !== "default" && d.kind === "videoinput")
.map(d => ({ value: d.deviceId, label: d.label || `Camera Device (${d.deviceId.substring(0, 9)})` }));
if (MediaDevicesManager.isAudioOutputSelectEnabled) {
this._outputDevices = mediaDevices
.filter(d => d.deviceId !== "default" && d.kind === "audiooutput")
.map(d => ({ value: d.deviceId, label: d.label || `Audio Output (${d.deviceId.substring(0, 9)})` }));
}
this.updatePermissions();
resolve();
});
return getValidMediaDevices().then(mediaDevices => {
this._micDevices = mediaDevices.filter(d => d.kind === "audioinput").map(optionFor);
this._videoDevices = mediaDevices.filter(d => d.kind === "videoinput").map(optionFor);
if (MediaDevicesManager.isAudioOutputSelectEnabled) {
this._outputDevices = mediaDevices.filter(d => d.kind === "audiooutput").map(optionFor);
this.updatePermissionStatus(MediaDevices.SPEAKERS, this._outputDevices.length === 0);
}
this.updatePermissionStatus(MediaDevices.MICROPHONE, this._micDevices.length === 0);
this.updatePermissionStatus(MediaDevices.CAMERA, this._videoDevices.length === 0);
});
}

Expand Down Expand Up @@ -297,7 +236,7 @@ export default class MediaDevicesManager extends EventEmitter {
this.audioTrack = newStream.getAudioTracks()[0];
this.audioTrack.addEventListener("ended", async () => {
this._scene.emit(MediaDevicesEvents.MIC_SHARE_ENDED);
this.startMicShare({ unmute: this.isMicEnabled });
this.startMicShare({ unmute: APP.dialog.isMicEnabled });
});

if (/Oculus/.test(navigator.userAgent)) {
Expand Down Expand Up @@ -437,30 +376,20 @@ export default class MediaDevicesManager extends EventEmitter {
}

deviceIdForMicDeviceLabel(label) {
return this._micDevices.filter(d => d.label === label).map(d => d.value)[0] || this.defaultInputDeviceId;
const match = this.micDevicesOptions.find(d => d.label === label);
return (match && match.value) || NO_DEVICE_ID;
}

deviceIdForSpeakersDeviceLabel(label) {
return this._outputDevices.filter(d => d.label === label).map(d => d.value)[0] || this.defaultOutputDeviceId;
const match = this.outputDevicesOptions.find(d => d.label === label);
return (match && match.value) || NO_DEVICE_ID;
}

micLabelForDeviceId(deviceId) {
return this._micDevices.filter(d => d.value === deviceId).map(d => d.label)[0];
}

speakersLabelForDeviceId(deviceId) {
return this._outputDevices.filter(d => d.value === deviceId).map(d => d.label)[0];
return this.micDevicesOptions.find(d => d.value === deviceId).label;
}

hasHmdMicrophone() {
return !!this.state._micDevices.find(d => HMD_MIC_REGEXES.find(r => d.label.match(r)));
}

videoDeviceIdForMicLabel(label) {
return this._videoDevices.filter(d => d.label === label).map(d => d.value)[0];
}

videoLabelForDeviceId(deviceId) {
return this._videoDevices.filter(d => d.value === deviceId).map(d => d.label)[0];
}
}
Loading