123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270 |
- import { AnyAction } from 'redux';
-
- import { IStore } from '../app/types';
- import { ENDPOINT_MESSAGE_RECEIVED } from '../base/conference/actionTypes';
- import MiddlewareRegistry from '../base/redux/MiddlewareRegistry';
-
- import {
- SET_REQUESTING_SUBTITLES,
- TOGGLE_REQUESTING_SUBTITLES
- } from './actionTypes';
- import {
- removeTranscriptMessage,
- updateTranscriptMessage
- } from './actions.any';
- import { notifyTranscriptionChunkReceived } from './functions';
-
-
- /**
- * The type of json-message which indicates that json carries a
- * transcription result.
- */
- const JSON_TYPE_TRANSCRIPTION_RESULT = 'transcription-result';
-
- /**
- * The type of json-message which indicates that json carries a
- * translation result.
- */
- const JSON_TYPE_TRANSLATION_RESULT = 'translation-result';
-
- /**
- * The local participant property which is used to set whether the local
- * participant wants to have a transcriber in the room.
- */
- const P_NAME_REQUESTING_TRANSCRIPTION = 'requestingTranscription';
-
- /**
- * The local participant property which is used to store the language
- * preference for translation for a participant.
- */
- const P_NAME_TRANSLATION_LANGUAGE = 'translation_language';
-
- /**
- * Time after which the rendered subtitles will be removed.
- */
- const REMOVE_AFTER_MS = 3000;
-
- /**
- * Stability factor for a transcription. We'll treat a transcript as stable
- * beyond this value.
- */
- const STABLE_TRANSCRIPTION_FACTOR = 0.85;
-
- /**
- * Middleware that catches actions related to transcript messages to be rendered
- * in {@link Captions}.
- *
- * @param {Store} store - The redux store.
- * @returns {Function}
- */
- MiddlewareRegistry.register(store => next => action => {
- switch (action.type) {
- case ENDPOINT_MESSAGE_RECEIVED:
- return _endpointMessageReceived(store, next, action);
-
- case TOGGLE_REQUESTING_SUBTITLES: {
- const state = store.getState()['features/subtitles'];
- const toggledValue = !state._requestingSubtitles;
-
- _requestingSubtitlesChange(store, toggledValue, state._language);
- break;
- }
- case SET_REQUESTING_SUBTITLES:
- _requestingSubtitlesChange(store, action.enabled, action.language);
- break;
- }
-
- return next(action);
- });
-
- /**
- * Notifies the feature transcription that the action
- * {@code ENDPOINT_MESSAGE_RECEIVED} is being dispatched within a specific redux
- * store.
- *
- * @param {Store} store - The redux store in which the specified {@code action}
- * is being dispatched.
- * @param {Dispatch} next - The redux {@code dispatch} function to
- * dispatch the specified {@code action} to the specified {@code store}.
- * @param {Action} action - The redux action {@code ENDPOINT_MESSAGE_RECEIVED}
- * which is being dispatched in the specified {@code store}.
- * @private
- * @returns {Object} The value returned by {@code next(action)}.
- */
- function _endpointMessageReceived(store: IStore, next: Function, action: AnyAction) {
- const { data: json } = action;
-
- if (![ JSON_TYPE_TRANSCRIPTION_RESULT, JSON_TYPE_TRANSLATION_RESULT ].includes(json?.type)) {
- return next(action);
- }
-
- const { dispatch, getState } = store;
- const state = getState();
- const language
- = state['features/base/conference'].conference
- ?.getLocalParticipantProperty(P_NAME_TRANSLATION_LANGUAGE);
- const { dumpTranscript, skipInterimTranscriptions } = state['features/base/config'].testing ?? {};
-
- const transcriptMessageID = json.message_id;
- const { name, id, avatar_url: avatarUrl } = json.participant;
- const participant = {
- avatarUrl,
- id,
- name
- };
-
- if (json.type === JSON_TYPE_TRANSLATION_RESULT && json.language === language) {
- // Displays final results in the target language if translation is
- // enabled.
-
- const newTranscriptMessage = {
- clearTimeOut: undefined,
- final: json.text,
- participant
- };
-
- _setClearerOnTranscriptMessage(dispatch, transcriptMessageID, newTranscriptMessage);
- dispatch(updateTranscriptMessage(transcriptMessageID, newTranscriptMessage));
- } else if (json.type === JSON_TYPE_TRANSCRIPTION_RESULT) {
- // Displays interim and final results without any translation if
- // translations are disabled.
-
- const { text } = json.transcript[0];
-
- // First, notify the external API.
- if (!(json.is_interim && skipInterimTranscriptions)) {
- const txt: any = {};
-
- if (!json.is_interim) {
- txt.final = text;
- } else if (json.stability > STABLE_TRANSCRIPTION_FACTOR) {
- txt.stable = text;
- } else {
- txt.unstable = text;
- }
-
- notifyTranscriptionChunkReceived(
- transcriptMessageID,
- json.language,
- participant,
- txt,
- store
- );
-
- if (navigator.product !== 'ReactNative') {
-
- // Dump transcript in a <transcript> element for debugging purposes.
- if (!json.is_interim && dumpTranscript) {
- try {
- let elem = document.body.getElementsByTagName('transcript')[0];
-
- // eslint-disable-next-line max-depth
- if (!elem) {
- elem = document.createElement('transcript');
- document.body.appendChild(elem);
- }
-
- elem.append(`${new Date(json.timestamp).toISOString()} ${participant.name}: ${text}`);
- } catch (_) {
- // Ignored.
- }
- }
- }
- }
-
- // If the user is not requesting transcriptions just bail.
- // Regex to filter out all possible country codes after language code:
- // this should catch all notations like 'en-GB' 'en_GB' and 'enGB'
- // and be independent of the country code length
- if (json.language.replace(/[-_A-Z].*/, '') !== language) {
- return next(action);
- }
-
- if (json.is_interim && skipInterimTranscriptions) {
- return next(action);
- }
-
- // We update the previous transcript message with the same
- // message ID or adds a new transcript message if it does not
- // exist in the map.
- const existingMessage = state['features/subtitles']._transcriptMessages.get(transcriptMessageID);
- const newTranscriptMessage: any = {
- clearTimeOut: existingMessage?.clearTimeOut,
- language,
- participant
- };
-
- _setClearerOnTranscriptMessage(dispatch, transcriptMessageID, newTranscriptMessage);
-
- // If this is final result, update the state as a final result
- // and start a count down to remove the subtitle from the state
- if (!json.is_interim) {
- newTranscriptMessage.final = text;
- } else if (json.stability > STABLE_TRANSCRIPTION_FACTOR) {
- // If the message has a high stability, we can update the
- // stable field of the state and remove the previously
- // unstable results
- newTranscriptMessage.stable = text;
- } else {
- // Otherwise, this result has an unstable result, which we
- // add to the state. The unstable result will be appended
- // after the stable part.
- newTranscriptMessage.unstable = text;
- }
-
- dispatch(updateTranscriptMessage(transcriptMessageID, newTranscriptMessage));
- }
-
- return next(action);
- }
-
- /**
- * Toggle the local property 'requestingTranscription'. This will cause Jicofo
- * and Jigasi to decide whether the transcriber needs to be in the room.
- *
- * @param {Store} store - The redux store.
- * @param {boolean} enabled - Whether subtitles should be enabled or not.
- * @param {string} language - The language to use for translation.
- * @private
- * @returns {void}
- */
- function _requestingSubtitlesChange(
- { getState }: IStore,
- enabled: boolean,
- language?: string | null) {
- const state = getState();
- const { conference } = state['features/base/conference'];
-
- conference?.setLocalParticipantProperty(
- P_NAME_REQUESTING_TRANSCRIPTION,
- enabled);
-
- if (enabled && language) {
- conference?.setLocalParticipantProperty(
- P_NAME_TRANSLATION_LANGUAGE,
- language.replace('translation-languages:', ''));
- }
- }
-
- /**
- * Set a timeout on a TranscriptMessage object so it clears itself when it's not
- * updated.
- *
- * @param {Function} dispatch - Dispatch remove action to store.
- * @param {string} transcriptMessageID - The id of the message to remove.
- * @param {Object} transcriptMessage - The message to remove.
- * @returns {void}
- */
- function _setClearerOnTranscriptMessage(
- dispatch: IStore['dispatch'],
- transcriptMessageID: string,
- transcriptMessage: { clearTimeOut?: number; }) {
- if (transcriptMessage.clearTimeOut) {
- clearTimeout(transcriptMessage.clearTimeOut);
- }
-
- transcriptMessage.clearTimeOut
- = window.setTimeout(
- () => dispatch(removeTranscriptMessage(transcriptMessageID)),
- REMOVE_AFTER_MS);
- }
|