import React, { PureComponent } from 'react'; import PropTypes from 'prop-types'; import { defineMessages, injectIntl } from 'react-intl'; import { Meteor } from 'meteor/meteor'; import Auth from '/imports/ui/services/auth'; import ActivityReportService from '../activity-report/service'; import Button from '/imports/ui/components/button/component'; import allowRedirectToLogoutURL from './service'; import getFromUserSettings from '/imports/ui/services/users-settings'; import logoutRouteHandler from '/imports/utils/logoutRouteHandler'; import Rating from './rating/component'; import { styles } from './styles'; import logger from '/imports/startup/client/logger'; import Users from '/imports/api/users'; import Meetings from '/imports/api/meetings'; import AudioManager from '/imports/ui/services/audio-manager'; import { meetingIsBreakout } from '/imports/ui/components/app/service'; const intlMessage = defineMessages({ 410: { id: 'app.meeting.ended', description: 'message when meeting is ended', }, 403: { id: 'app.error.removed', description: 'Message to display when user is removed from the conference', }, 430: { id: 'app.error.meeting.ended', description: 'user logged conference', }, 'acl-not-allowed': { id: 'app.error.removed', description: 'Message to display when user is removed from the conference', }, messageEnded: { id: 'app.meeting.endedMessage', description: 'message saying to go back to home screen', }, messageEndedByUser: { id: 'app.meeting.endedByUserMessage', description: 'message informing who ended the meeting', }, messageEndedByNoModeratorSingular: { id: 'app.meeting.endedByNoModeratorMessageSingular', description: 'message informing that the meeting was ended due to no moderator present (singular)', }, messageEndedByNoModeratorPlural: { id: 'app.meeting.endedByNoModeratorMessagePlural', description: 'message informing that the meeting was ended due to no moderator present (plural)', }, buttonOkay: { id: 'app.meeting.endNotification.ok.label', description: 'label okay for button', }, title: { id: 'app.feedback.title', description: 'title for feedback screen', }, subtitle: { id: 'app.feedback.subtitle', description: 'subtitle for feedback screen', }, textarea: { id: 'app.feedback.textarea', description: 'placeholder for textarea', }, confirmDesc: { id: 'app.leaveConfirmation.confirmDesc', description: 'adds context to confim option', }, sendLabel: { id: 'app.feedback.sendFeedback', description: 'send feedback button label', }, sendDesc: { id: 'app.feedback.sendFeedbackDesc', description: 'adds context to send feedback option', }, duplicate_user_in_meeting_eject_reason: { id: 'app.meeting.logout.duplicateUserEjectReason', description: 'message for duplicate users', }, not_enough_permission_eject_reason: { id: 'app.meeting.logout.permissionEjectReason', description: 'message for whom was kicked by doing something without permission', }, user_requested_eject_reason: { id: 'app.meeting.logout.ejectedFromMeeting', description: 'message when the user is removed by someone', }, validate_token_failed_eject_reason: { id: 'app.meeting.logout.validateTokenFailedEjectReason', description: 'invalid auth token', }, user_inactivity_eject_reason: { id: 'app.meeting.logout.userInactivityEjectReason', description: 'message for whom was kicked by inactivity', }, open_activity_report_btn: { id: 'app.activity-report.clickHereToOpen', description: 'description of link to open activity report', }, }); const propTypes = { intl: PropTypes.shape({ formatMessage: PropTypes.func.isRequired, }).isRequired, code: PropTypes.string.isRequired, ejectedReason: PropTypes.string, endedReason: PropTypes.string, }; const defaultProps = { ejectedReason: null, endedReason: null, }; class MeetingEnded extends PureComponent { static getComment() { const textarea = document.getElementById('feedbackComment'); const comment = textarea.value; return comment; } constructor(props) { super(props); this.state = { selected: 0, dispatched: false, }; const user = Users.findOne({ userId: Auth.userID }); if (user) { this.localUserRole = user.role; } const meeting = Meetings.findOne({ id: user.meetingID }); if (meeting) { this.endWhenNoModeratorMinutes = meeting.durationProps.endWhenNoModeratorDelayInMinutes; const endedBy = Users.findOne({ userId: meeting.meetingEndedBy, }, { fields: { name: 1 } }); if (endedBy) { this.meetingEndedBy = endedBy.name; } } this.setSelectedStar = this.setSelectedStar.bind(this); this.confirmRedirect = this.confirmRedirect.bind(this); this.sendFeedback = this.sendFeedback.bind(this); this.shouldShowFeedback = this.shouldShowFeedback.bind(this); this.getEndingMessage = this.getEndingMessage.bind(this); AudioManager.exitAudio(); Meteor.disconnect(); } setSelectedStar(starNumber) { this.setState({ selected: starNumber, }); } shouldShowFeedback() { const { dispatched } = this.state; return getFromUserSettings('bbb_ask_for_feedback_on_logout', Meteor.settings.public.app.askForFeedbackOnLogout) && !dispatched; } confirmRedirect() { const { selected, } = this.state; if (selected <= 0) { if (meetingIsBreakout()) window.close(); if (allowRedirectToLogoutURL()) logoutRouteHandler(); } } getEndingMessage() { const { intl, code, endedReason } = this.props; if (endedReason && endedReason === 'ENDED_DUE_TO_NO_MODERATOR') { return this.endWhenNoModeratorMinutes === 1 ? intl.formatMessage(intlMessage.messageEndedByNoModeratorSingular) : intl.formatMessage(intlMessage.messageEndedByNoModeratorPlural, { 0: this.endWhenNoModeratorMinutes }); } if (this.meetingEndedBy) { return intl.formatMessage(intlMessage.messageEndedByUser, { 0: this.meetingEndedBy }); } if (intlMessage[code]) { return intl.formatMessage(intlMessage[code]); } return intl.formatMessage(intlMessage[430]); } sendFeedback() { const { selected, } = this.state; const { fullname } = Auth.credentials; const message = { rating: selected, userId: Auth.userID, userName: fullname, authToken: Auth.token, meetingId: Auth.meetingID, comment: MeetingEnded.getComment(), userRole: this.localUserRole, }; const url = './feedback'; const options = { method: 'POST', body: JSON.stringify(message), headers: { 'Content-Type': 'application/json', }, }; // client logger logger.info({ logCode: 'feedback_functionality', extraInfo: { feedback: message } }, 'Feedback component'); this.setState({ dispatched: true, }); if (allowRedirectToLogoutURL()) { const FEEDBACK_WAIT_TIME = 500; setTimeout(() => { fetch(url, options) .then(() => { logoutRouteHandler(); }) .catch(() => { logoutRouteHandler(); }); }, FEEDBACK_WAIT_TIME); } } renderNoFeedback() { const { intl, code, ejectedReason } = this.props; const logMessage = ejectedReason === 'user_requested_eject_reason' ? 'User removed from the meeting' : 'Meeting ended component, no feedback configured'; logger.info({ logCode: 'meeting_ended_code', extraInfo: { endedCode: code, reason: ejectedReason } }, logMessage); return (