mirror of
https://gitcode.com/GitHub_Trending/ji/jitsi-meet.git
synced 2026-05-14 03:32:31 +00:00
Unfortunately, as the Jitsi Meet development evolved the routing mechanism became more complex and thre logic ended up spread across multiple parts of the codebase, which made it hard to follow and extend. This change aims to fix that by rewriting the routing logic and centralizing it in (pretty much) a single place, with no implicit inter-dependencies. In order to arrive there, however, some extra changes were needed, which were not caught early enough and are thus part of this change: - JitsiMeetJS initialization is now synchronous: there is nothing async about it, and the only async requirement (Temasys support) was lifted. See [0]. - WebRTC support can be detected early: building on top of the above, WebRTC support can now be detected immediately, so take advantage of this to simplify how we handle unsupported browsers. See [0]. The new router takes decissions based on the Redux state at the time of invocation. A route can be represented by either a component or a URl reference, with the latter taking precedence. On mobile, obviously, there is no concept of URL reference so routing is based solely on components. [0]: https://github.com/jitsi/lib-jitsi-meet/pull/779
201 lines
5.9 KiB
JavaScript
201 lines
5.9 KiB
JavaScript
import { API_ID } from '../../../modules/API/constants';
|
|
import {
|
|
PostMessageTransportBackend,
|
|
Transport
|
|
} from '../../../modules/transport';
|
|
|
|
import {
|
|
getAudioOutputDeviceId,
|
|
setAudioInputDevice,
|
|
setAudioOutputDevice,
|
|
setVideoInputDevice
|
|
} from '../base/devices';
|
|
import { i18next } from '../base/i18n';
|
|
import JitsiMeetJS from '../base/lib-jitsi-meet';
|
|
|
|
import { SET_DEVICE_SELECTION_POPUP_DATA } from './actionTypes';
|
|
import { getDeviceSelectionDialogProps } from './functions';
|
|
|
|
/**
|
|
* Opens a popup window with the device selection dialog in it.
|
|
*
|
|
* @returns {Function}
|
|
*/
|
|
export function openDeviceSelectionPopup() {
|
|
return (dispatch, getState) => {
|
|
const { popupDialogData } = getState()['features/device-selection'];
|
|
|
|
if (popupDialogData) {
|
|
popupDialogData.popup.focus();
|
|
|
|
return;
|
|
}
|
|
|
|
// API_ID will always be defined because the iframe api is enabled
|
|
const scope = `dialog_${API_ID}`;
|
|
const url = `${
|
|
window.location.origin}/static/deviceSelectionPopup.html#scope=${
|
|
encodeURIComponent(JSON.stringify(scope))}`;
|
|
const popup
|
|
= window.open(
|
|
url,
|
|
'device-selection-popup',
|
|
'toolbar=no,scrollbars=no,resizable=no,width=720,height=458');
|
|
|
|
popup.addEventListener('DOMContentLoaded', () => {
|
|
popup.init(i18next);
|
|
});
|
|
|
|
const transport = new Transport({
|
|
backend: new PostMessageTransportBackend({
|
|
postisOptions: {
|
|
scope,
|
|
window: popup
|
|
}
|
|
})
|
|
});
|
|
|
|
transport.on('request',
|
|
_processRequest.bind(undefined, dispatch, getState));
|
|
transport.on('event', event => {
|
|
if (event.type === 'devices-dialog' && event.name === 'close') {
|
|
popup.close();
|
|
transport.dispose();
|
|
dispatch(_setDeviceSelectionPopupData());
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
});
|
|
|
|
dispatch(_setDeviceSelectionPopupData({
|
|
popup,
|
|
transport
|
|
}));
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Processes device requests from external applications.
|
|
*
|
|
* @param {Dispatch} dispatch - The redux {@code dispatch} function.
|
|
* @param {Function} getState - The redux function that gets/retrieves the redux
|
|
* state.
|
|
* @param {Object} request - The request to be processed.
|
|
* @param {Function} responseCallback - The callback that will send the
|
|
* response.
|
|
* @returns {boolean}
|
|
*/
|
|
function _processRequest(dispatch, getState, request, responseCallback) { // eslint-disable-line max-len, max-params
|
|
if (request.type === 'devices') {
|
|
const state = getState();
|
|
const settings = state['features/base/settings'];
|
|
|
|
switch (request.name) {
|
|
case 'isDeviceListAvailable':
|
|
responseCallback(JitsiMeetJS.mediaDevices.isDeviceListAvailable());
|
|
break;
|
|
case 'isDeviceChangeAvailable':
|
|
responseCallback(
|
|
JitsiMeetJS.mediaDevices.isDeviceChangeAvailable(
|
|
request.deviceType));
|
|
break;
|
|
case 'isMultipleAudioInputSupported':
|
|
responseCallback(JitsiMeetJS.isMultipleAudioInputSupported());
|
|
break;
|
|
case 'getCurrentDevices':
|
|
responseCallback({
|
|
audioInput: settings.micDeviceId,
|
|
audioOutput: getAudioOutputDeviceId(),
|
|
videoInput: settings.cameraDeviceId
|
|
});
|
|
break;
|
|
case 'getAvailableDevices':
|
|
responseCallback(getState()['features/base/devices']);
|
|
break;
|
|
case 'setDevice': {
|
|
let action;
|
|
const { device } = request;
|
|
|
|
switch (device.kind) {
|
|
case 'audioinput':
|
|
action = setAudioInputDevice;
|
|
break;
|
|
case 'audiooutput':
|
|
action = setAudioOutputDevice;
|
|
break;
|
|
case 'videoinput':
|
|
action = setVideoInputDevice;
|
|
break;
|
|
default:
|
|
|
|
}
|
|
dispatch(action(device.id));
|
|
responseCallback(true);
|
|
break;
|
|
}
|
|
default:
|
|
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Sets information about device selection popup in the store.
|
|
*
|
|
* @param {Object} popupDialogData - Information about the popup.
|
|
* @param {Object} popupDialog.popup - The popup object returned from
|
|
* window.open.
|
|
* @param {Object} popupDialogData.transport - The transport instance used for
|
|
* communication with the popup window.
|
|
* @returns {{
|
|
* type: SET_DEVICE_SELECTION_POPUP_DATA,
|
|
* popupDialogData: Object
|
|
* }}
|
|
*/
|
|
function _setDeviceSelectionPopupData(popupDialogData) {
|
|
return {
|
|
type: SET_DEVICE_SELECTION_POPUP_DATA,
|
|
popupDialogData
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Submits the settings related to device selection.
|
|
*
|
|
* @param {Object} newState - The new settings.
|
|
* @returns {Function}
|
|
*/
|
|
export function submitDeviceSelectionTab(newState) {
|
|
return (dispatch, getState) => {
|
|
const currentState = getDeviceSelectionDialogProps(getState());
|
|
|
|
if (newState.selectedVideoInputId
|
|
&& newState.selectedVideoInputId
|
|
!== currentState.selectedVideoInputId) {
|
|
dispatch(
|
|
setVideoInputDevice(newState.selectedVideoInputId));
|
|
}
|
|
|
|
if (newState.selectedAudioInputId
|
|
&& newState.selectedAudioInputId
|
|
!== currentState.selectedAudioInputId) {
|
|
dispatch(
|
|
setAudioInputDevice(newState.selectedAudioInputId));
|
|
}
|
|
|
|
if (newState.selectedAudioOutputId
|
|
&& newState.selectedAudioOutputId
|
|
!== currentState.selectedAudioOutputId) {
|
|
dispatch(
|
|
setAudioOutputDevice(newState.selectedAudioOutputId));
|
|
}
|
|
};
|
|
}
|