2018-06-14 09:14:32 +00:00
|
|
|
// @flow
|
|
|
|
|
2019-06-25 22:25:43 +00:00
|
|
|
import throttle from 'lodash/throttle';
|
|
|
|
import type { Dispatch } from 'redux';
|
|
|
|
|
2021-01-22 10:03:39 +00:00
|
|
|
import { NOTIFICATIONS_ENABLED, getFeatureFlag } from '../base/flags';
|
2021-09-02 12:47:11 +00:00
|
|
|
import { getParticipantCount } from '../base/participants/functions';
|
2021-01-22 10:03:39 +00:00
|
|
|
|
2017-07-28 17:56:49 +00:00
|
|
|
import {
|
2018-06-14 09:14:32 +00:00
|
|
|
CLEAR_NOTIFICATIONS,
|
2017-07-28 17:56:49 +00:00
|
|
|
HIDE_NOTIFICATION,
|
2021-09-10 11:05:16 +00:00
|
|
|
HIDE_RAISE_HAND_NOTIFICATIONS,
|
2017-08-02 18:15:55 +00:00
|
|
|
SET_NOTIFICATIONS_ENABLED,
|
2017-07-28 17:56:49 +00:00
|
|
|
SHOW_NOTIFICATION
|
|
|
|
} from './actionTypes';
|
2021-09-02 12:47:11 +00:00
|
|
|
import {
|
2021-12-17 12:39:15 +00:00
|
|
|
NOTIFICATION_ICON,
|
2021-11-24 11:05:27 +00:00
|
|
|
NOTIFICATION_TIMEOUT_TYPE,
|
2021-09-02 12:47:11 +00:00
|
|
|
NOTIFICATION_TIMEOUT,
|
|
|
|
NOTIFICATION_TYPE,
|
2021-12-02 13:17:07 +00:00
|
|
|
SILENT_JOIN_THRESHOLD,
|
|
|
|
SILENT_LEFT_THRESHOLD
|
2021-09-02 12:47:11 +00:00
|
|
|
} from './constants';
|
2017-07-28 17:56:49 +00:00
|
|
|
|
2021-11-24 11:05:27 +00:00
|
|
|
/**
|
|
|
|
* Function that returns notification timeout value based on notification timeout type.
|
|
|
|
*
|
|
|
|
* @param {string} type - Notification type.
|
|
|
|
* @param {Object} notificationTimeouts - Config notification timeouts.
|
|
|
|
* @returns {number}
|
|
|
|
*/
|
|
|
|
function getNotificationTimeout(type: ?string, notificationTimeouts: ?Object) {
|
|
|
|
if (type === NOTIFICATION_TIMEOUT_TYPE.SHORT) {
|
|
|
|
return notificationTimeouts?.short ?? NOTIFICATION_TIMEOUT.SHORT;
|
|
|
|
} else if (type === NOTIFICATION_TIMEOUT_TYPE.MEDIUM) {
|
|
|
|
return notificationTimeouts?.medium ?? NOTIFICATION_TIMEOUT.MEDIUM;
|
|
|
|
} else if (type === NOTIFICATION_TIMEOUT_TYPE.LONG) {
|
|
|
|
return notificationTimeouts?.long ?? NOTIFICATION_TIMEOUT.LONG;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NOTIFICATION_TIMEOUT.STICKY;
|
|
|
|
}
|
|
|
|
|
2018-06-14 09:14:32 +00:00
|
|
|
/**
|
|
|
|
* Clears (removes) all the notifications.
|
|
|
|
*
|
|
|
|
* @returns {{
|
|
|
|
* type: CLEAR_NOTIFICATIONS
|
|
|
|
* }}
|
|
|
|
*/
|
|
|
|
export function clearNotifications() {
|
|
|
|
return {
|
|
|
|
type: CLEAR_NOTIFICATIONS
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-07-28 17:56:49 +00:00
|
|
|
/**
|
|
|
|
* Removes the notification with the passed in id.
|
|
|
|
*
|
|
|
|
* @param {string} uid - The unique identifier for the notification to be
|
|
|
|
* removed.
|
|
|
|
* @returns {{
|
|
|
|
* type: HIDE_NOTIFICATION,
|
2021-06-23 11:23:44 +00:00
|
|
|
* uid: string
|
2017-07-28 17:56:49 +00:00
|
|
|
* }}
|
|
|
|
*/
|
2021-06-23 11:23:44 +00:00
|
|
|
export function hideNotification(uid: string) {
|
2017-07-28 17:56:49 +00:00
|
|
|
return {
|
|
|
|
type: HIDE_NOTIFICATION,
|
|
|
|
uid
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2021-09-10 11:05:16 +00:00
|
|
|
/**
|
|
|
|
* Removes the raise hand notifications.
|
|
|
|
*
|
|
|
|
* @returns {{
|
|
|
|
* type: HIDE_RAISE_HAND_NOTIFICATIONS
|
|
|
|
* }}
|
|
|
|
*/
|
|
|
|
export function hideRaiseHandNotifications() {
|
|
|
|
return {
|
|
|
|
type: HIDE_RAISE_HAND_NOTIFICATIONS
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-08-02 18:15:55 +00:00
|
|
|
/**
|
|
|
|
* Stops notifications from being displayed.
|
|
|
|
*
|
|
|
|
* @param {boolean} enabled - Whether or not notifications should display.
|
|
|
|
* @returns {{
|
|
|
|
* type: SET_NOTIFICATIONS_ENABLED,
|
|
|
|
* enabled: boolean
|
|
|
|
* }}
|
|
|
|
*/
|
2018-06-14 09:14:32 +00:00
|
|
|
export function setNotificationsEnabled(enabled: boolean) {
|
2017-08-02 18:15:55 +00:00
|
|
|
return {
|
|
|
|
type: SET_NOTIFICATIONS_ENABLED,
|
|
|
|
enabled
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-11-03 19:05:03 +00:00
|
|
|
/**
|
|
|
|
* Queues an error notification for display.
|
|
|
|
*
|
|
|
|
* @param {Object} props - The props needed to show the notification component.
|
2021-11-24 11:05:27 +00:00
|
|
|
* @param {string} type - Notification type.
|
2017-11-03 19:05:03 +00:00
|
|
|
* @returns {Object}
|
|
|
|
*/
|
2021-11-24 11:05:27 +00:00
|
|
|
export function showErrorNotification(props: Object, type: ?string) {
|
2017-12-11 18:33:09 +00:00
|
|
|
return showNotification({
|
2017-11-03 19:05:03 +00:00
|
|
|
...props,
|
|
|
|
appearance: NOTIFICATION_TYPE.ERROR
|
2021-11-24 11:05:27 +00:00
|
|
|
}, type);
|
2017-11-03 19:05:03 +00:00
|
|
|
}
|
|
|
|
|
2017-07-28 17:56:49 +00:00
|
|
|
/**
|
|
|
|
* Queues a notification for display.
|
|
|
|
*
|
|
|
|
* @param {Object} props - The props needed to show the notification component.
|
2021-11-24 11:05:27 +00:00
|
|
|
* @param {string} type - Notification type.
|
2021-01-11 10:52:52 +00:00
|
|
|
* @returns {Function}
|
2017-07-28 17:56:49 +00:00
|
|
|
*/
|
2021-11-24 11:05:27 +00:00
|
|
|
export function showNotification(props: Object = {}, type: ?string) {
|
2021-01-11 10:52:52 +00:00
|
|
|
return function(dispatch: Function, getState: Function) {
|
2021-11-24 11:05:27 +00:00
|
|
|
const { notifications, notificationTimeouts } = getState()['features/base/config'];
|
2021-01-22 10:03:39 +00:00
|
|
|
const enabledFlag = getFeatureFlag(getState(), NOTIFICATIONS_ENABLED, true);
|
|
|
|
|
|
|
|
const shouldDisplay = enabledFlag
|
|
|
|
&& (!notifications
|
|
|
|
|| notifications.includes(props.descriptionKey)
|
|
|
|
|| notifications.includes(props.titleKey));
|
2021-01-11 10:52:52 +00:00
|
|
|
|
|
|
|
if (shouldDisplay) {
|
|
|
|
return dispatch({
|
|
|
|
type: SHOW_NOTIFICATION,
|
|
|
|
props,
|
2021-11-24 11:05:27 +00:00
|
|
|
timeout: getNotificationTimeout(type, notificationTimeouts),
|
2021-06-23 11:23:44 +00:00
|
|
|
uid: props.uid || window.Date.now().toString()
|
2021-01-11 10:52:52 +00:00
|
|
|
});
|
|
|
|
}
|
2017-07-28 17:56:49 +00:00
|
|
|
};
|
|
|
|
}
|
2017-07-31 18:36:41 +00:00
|
|
|
|
2017-11-03 19:05:03 +00:00
|
|
|
/**
|
|
|
|
* Queues a warning notification for display.
|
|
|
|
*
|
|
|
|
* @param {Object} props - The props needed to show the notification component.
|
2021-11-24 11:05:27 +00:00
|
|
|
* @param {string} type - Notification type.
|
2017-11-03 19:05:03 +00:00
|
|
|
* @returns {Object}
|
|
|
|
*/
|
2021-11-24 11:05:27 +00:00
|
|
|
export function showWarningNotification(props: Object, type: ?string) {
|
|
|
|
|
2017-12-11 18:33:09 +00:00
|
|
|
return showNotification({
|
2017-11-03 19:05:03 +00:00
|
|
|
...props,
|
|
|
|
appearance: NOTIFICATION_TYPE.WARNING
|
2021-11-24 11:05:27 +00:00
|
|
|
}, type);
|
2017-11-03 19:05:03 +00:00
|
|
|
}
|
2019-06-25 22:25:43 +00:00
|
|
|
|
2021-12-17 12:39:15 +00:00
|
|
|
/**
|
|
|
|
* Queues a message notification for display.
|
|
|
|
*
|
|
|
|
* @param {Object} props - The props needed to show the notification component.
|
|
|
|
* @param {string} type - Notification type.
|
|
|
|
* @returns {Object}
|
|
|
|
*/
|
|
|
|
export function showMessageNotification(props: Object, type: ?string) {
|
|
|
|
return showNotification({
|
|
|
|
...props,
|
|
|
|
concatText: true,
|
|
|
|
titleKey: 'notify.chatMessages',
|
|
|
|
appearance: NOTIFICATION_TYPE.NORMAL,
|
|
|
|
icon: NOTIFICATION_ICON.MESSAGE
|
|
|
|
}, type);
|
|
|
|
}
|
|
|
|
|
2019-06-25 22:25:43 +00:00
|
|
|
/**
|
|
|
|
* An array of names of participants that have joined the conference. The array
|
|
|
|
* is replaced with an empty array as notifications are displayed.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @type {string[]}
|
|
|
|
*/
|
|
|
|
let joinedParticipantsNames = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A throttled internal function that takes the internal list of participant
|
|
|
|
* names, {@code joinedParticipantsNames}, and triggers the display of a
|
|
|
|
* notification informing of their joining.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @type {Function}
|
|
|
|
*/
|
2021-09-02 12:47:11 +00:00
|
|
|
const _throttledNotifyParticipantConnected = throttle((dispatch: Dispatch<any>, getState: Function) => {
|
|
|
|
const participantCount = getParticipantCount(getState());
|
|
|
|
|
|
|
|
// Skip join notifications altogether for large meetings.
|
|
|
|
if (participantCount > SILENT_JOIN_THRESHOLD) {
|
|
|
|
joinedParticipantsNames = [];
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-06-25 22:25:43 +00:00
|
|
|
const joinedParticipantsCount = joinedParticipantsNames.length;
|
|
|
|
|
|
|
|
let notificationProps;
|
|
|
|
|
|
|
|
if (joinedParticipantsCount >= 3) {
|
|
|
|
notificationProps = {
|
|
|
|
titleArguments: {
|
2021-09-02 12:46:16 +00:00
|
|
|
name: joinedParticipantsNames[0]
|
2019-06-25 22:25:43 +00:00
|
|
|
},
|
|
|
|
titleKey: 'notify.connectedThreePlusMembers'
|
|
|
|
};
|
|
|
|
} else if (joinedParticipantsCount === 2) {
|
|
|
|
notificationProps = {
|
|
|
|
titleArguments: {
|
|
|
|
first: joinedParticipantsNames[0],
|
|
|
|
second: joinedParticipantsNames[1]
|
|
|
|
},
|
|
|
|
titleKey: 'notify.connectedTwoMembers'
|
|
|
|
};
|
|
|
|
} else if (joinedParticipantsCount) {
|
|
|
|
notificationProps = {
|
|
|
|
titleArguments: {
|
|
|
|
name: joinedParticipantsNames[0]
|
|
|
|
},
|
|
|
|
titleKey: 'notify.connectedOneMember'
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if (notificationProps) {
|
|
|
|
dispatch(
|
2021-11-24 11:05:27 +00:00
|
|
|
showNotification(notificationProps, NOTIFICATION_TIMEOUT_TYPE.SHORT));
|
2019-06-25 22:25:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
joinedParticipantsNames = [];
|
|
|
|
|
2021-09-02 12:45:27 +00:00
|
|
|
}, 2000, { leading: false });
|
2019-06-25 22:25:43 +00:00
|
|
|
|
2021-12-02 13:17:07 +00:00
|
|
|
/**
|
|
|
|
* An array of names of participants that have left the conference. The array
|
|
|
|
* is replaced with an empty array as notifications are displayed.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @type {string[]}
|
|
|
|
*/
|
|
|
|
let leftParticipantsNames = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A throttled internal function that takes the internal list of participant
|
|
|
|
* names, {@code leftParticipantsNames}, and triggers the display of a
|
|
|
|
* notification informing of their leaving.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @type {Function}
|
|
|
|
*/
|
|
|
|
const _throttledNotifyParticipantLeft = throttle((dispatch: Dispatch<any>, getState: Function) => {
|
|
|
|
const participantCount = getParticipantCount(getState());
|
|
|
|
|
|
|
|
// Skip left notifications altogether for large meetings.
|
|
|
|
if (participantCount > SILENT_LEFT_THRESHOLD) {
|
|
|
|
leftParticipantsNames = [];
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const leftParticipantsCount = leftParticipantsNames.length;
|
|
|
|
|
|
|
|
let notificationProps;
|
|
|
|
|
|
|
|
if (leftParticipantsCount >= 3) {
|
|
|
|
notificationProps = {
|
|
|
|
titleArguments: {
|
|
|
|
name: leftParticipantsNames[0]
|
|
|
|
},
|
|
|
|
titleKey: 'notify.leftThreePlusMembers'
|
|
|
|
};
|
|
|
|
} else if (leftParticipantsCount === 2) {
|
|
|
|
notificationProps = {
|
|
|
|
titleArguments: {
|
|
|
|
first: leftParticipantsNames[0],
|
|
|
|
second: leftParticipantsNames[1]
|
|
|
|
},
|
|
|
|
titleKey: 'notify.leftTwoMembers'
|
|
|
|
};
|
|
|
|
} else if (leftParticipantsCount) {
|
|
|
|
notificationProps = {
|
|
|
|
titleArguments: {
|
|
|
|
name: leftParticipantsNames[0]
|
|
|
|
},
|
|
|
|
titleKey: 'notify.leftOneMember'
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if (notificationProps) {
|
|
|
|
dispatch(
|
|
|
|
showNotification(notificationProps, NOTIFICATION_TIMEOUT_TYPE.SHORT));
|
|
|
|
}
|
|
|
|
|
|
|
|
leftParticipantsNames = [];
|
|
|
|
|
|
|
|
}, 2000, { leading: false });
|
|
|
|
|
2019-06-25 22:25:43 +00:00
|
|
|
/**
|
|
|
|
* Queues the display of a notification of a participant having connected to
|
|
|
|
* the meeting. The notifications are batched so that quick consecutive
|
|
|
|
* connection events are shown in one notification.
|
|
|
|
*
|
|
|
|
* @param {string} displayName - The name of the participant that connected.
|
|
|
|
* @returns {Function}
|
|
|
|
*/
|
|
|
|
export function showParticipantJoinedNotification(displayName: string) {
|
2021-12-15 19:22:09 +00:00
|
|
|
joinedParticipantsNames.push(displayName);
|
2019-06-25 22:25:43 +00:00
|
|
|
|
2021-09-02 12:47:11 +00:00
|
|
|
return (dispatch: Dispatch<any>, getState: Function) => _throttledNotifyParticipantConnected(dispatch, getState);
|
2019-06-25 22:25:43 +00:00
|
|
|
}
|
2021-12-02 13:17:07 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Queues the display of a notification of a participant having left to
|
|
|
|
* the meeting. The notifications are batched so that quick consecutive
|
|
|
|
* connection events are shown in one notification.
|
|
|
|
*
|
|
|
|
* @param {string} displayName - The name of the participant that left.
|
|
|
|
* @returns {Function}
|
|
|
|
*/
|
|
|
|
export function showParticipantLeftNotification(displayName: string) {
|
2021-12-15 19:22:09 +00:00
|
|
|
leftParticipantsNames.push(displayName);
|
2021-12-02 13:17:07 +00:00
|
|
|
|
|
|
|
return (dispatch: Dispatch<any>, getState: Function) => _throttledNotifyParticipantLeft(dispatch, getState);
|
|
|
|
}
|