[mirotalksfu] - #22 add real-time polls

هذا الالتزام موجود في:
Miroslav Pejic
2024-07-18 08:50:57 +02:00
الأصل 55c7da49cd
التزام e5785538ef
11 ملفات معدلة مع 589 إضافات و6 حذوفات

عرض الملف

@@ -61,6 +61,9 @@ module.exports = class Room {
this.rtmpFileStreamer = null;
this.rtmpUrlStreamer = null;
this.rtmp = config.server.rtmp || false;
// Polls
this.polls = [];
}
// ####################################################
@@ -87,10 +90,30 @@ module.exports = class Room {
survey: this.survey,
redirect: this.redirect,
videoAIEnabled: this.videoAIEnabled,
thereIsPolls: this.thereIsPolls(),
peers: JSON.stringify([...this.peers]),
};
}
// ##############################################
// POLLS
// ##############################################
thereIsPolls() {
return this.polls.length > 0;
}
getPolls() {
return this.polls;
}
convertPolls(polls) {
return polls.map((poll) => {
const voters = poll.voters ? Object.fromEntries(poll.voters.entries()) : {};
return { ...poll, voters };
});
}
// ##############################################
// RTMP from FILE
// ##############################################

عرض الملف

@@ -44,7 +44,7 @@ dependencies: {
* @license For commercial or closed source, contact us at license.mirotalk@gmail.com or purchase directly via CodeCanyon
* @license CodeCanyon: https://codecanyon.net/item/mirotalk-sfu-webrtc-realtime-video-conferences/40769970
* @author Miroslav Pejic - miroslav.pejic.85@gmail.com
* @version 1.4.99
* @version 1.5.10
*
*/
@@ -2385,6 +2385,97 @@ function startServer() {
log.debug('endRTMPfromURL - rtmpUrlStreamsCount ---->', rtmpUrlStreamsCount);
});
socket.on('createPoll', (dataObject) => {
if (!roomList.has(socket.room_id)) return;
const data = checkXSS(dataObject);
const { question, options } = data;
const room = roomList.get(socket.room_id);
const newPoll = {
question: question,
options: options,
voters: new Map(),
};
const roomPolls = room.getPolls();
roomPolls.push(newPoll);
room.sendToAll('updatePolls', room.convertPolls(roomPolls));
log.debug('[Poll] createPoll', roomPolls);
});
socket.on('vote', (dataObject) => {
if (!roomList.has(socket.room_id)) return;
const data = checkXSS(dataObject);
const room = roomList.get(socket.room_id);
const roomPolls = room.getPolls();
const poll = roomPolls[data.pollIndex];
if (poll) {
const peer_name = getPeerName(room, false) || socket.id;
poll.voters.set(peer_name, data.option);
room.sendToAll('updatePolls', room.convertPolls(roomPolls));
log.debug('[Poll] vote', roomPolls);
}
});
socket.on('updatePoll', () => {
if (!roomList.has(socket.room_id)) return;
const room = roomList.get(socket.room_id);
const roomPolls = room.getPolls();
if (roomPolls.length > 0) {
room.sendToAll('updatePolls', room.convertPolls(roomPolls));
log.debug('[Poll] updatePoll', roomPolls);
}
});
socket.on('editPoll', (dataObject) => {
if (!roomList.has(socket.room_id)) return;
const data = checkXSS(dataObject);
const { index, question, options } = data;
const room = roomList.get(socket.room_id);
const roomPolls = room.getPolls();
if (roomPolls[index]) {
roomPolls[index].question = question;
roomPolls[index].options = options;
room.sendToAll('updatePolls', roomPolls);
log.debug('[Poll] editPoll', roomPolls);
}
});
socket.on('deletePoll', async (data) => {
if (!roomList.has(socket.room_id)) return;
const { index, peer_name, peer_uuid } = checkXSS(data);
const isPresenter = await isPeerPresenter(socket.room_id, socket.id, peer_name, peer_uuid);
if (!isPresenter) return;
const room = roomList.get(socket.room_id);
const roomPolls = room.getPolls();
if (roomPolls[index]) {
roomPolls.splice(index, 1);
room.sendToAll('updatePolls', roomPolls);
}
});
socket.on('disconnect', async () => {
if (!roomList.has(socket.room_id)) return;

عرض الملف

@@ -355,6 +355,7 @@ module.exports = {
startScreenButton: true,
swapCameraButton: true,
chatButton: true,
pollButton: true,
raiseHandButton: true,
transcriptionButton: true,
whiteboardButton: true,

عرض الملف

@@ -1,6 +1,6 @@
{
"name": "mirotalksfu",
"version": "1.4.99",
"version": "1.5.10",
"description": "WebRTC SFU browser-based video calls",
"main": "Server.js",
"scripts": {
@@ -57,7 +57,7 @@
},
"dependencies": {
"@sentry/integrations": "7.114.0",
"@sentry/node": "8.17.0",
"@sentry/node": "8.18.0",
"axios": "^1.7.2",
"body-parser": "1.20.2",
"colors": "1.4.0",

167
public/css/Polls.css Normal file
عرض الملف

@@ -0,0 +1,167 @@
.pls-container {
z-index: 5;
position: absolute;
background: var(--body-bg);
padding: 20px;
border-radius: 8px;
border: var(--border);
box-shadow: 0 0 10px rgba(0, 0, 0, 0.1);
width: 100%;
height: 100%;
max-width: 600px;
max-height: 600px;
overflow-x: hidden;
}
.poll-header {
display: inline-flex;
text-align: left;
margin-bottom: 20px;
width: 100%;
cursor: move;
}
.poll-creation {
margin-bottom: 20px;
max-height: 480px;
overflow: hidden;
}
.poll-h1,
.poll-h2,
.poll-h3 {
margin-top: 5px;
color: #fff !important;
}
.form {
display: flex;
flex-direction: column;
max-height: 500px;
overflow-x: hidden;
}
.form-group {
margin-bottom: 15px;
}
.form-group label {
margin-bottom: 15px;
}
.form label {
margin-bottom: 10px;
font-weight: bold;
color: #fff;
}
.form input {
padding: 10px;
border: var(--border);
border-radius: 4px;
width: 100%;
box-sizing: border-box;
}
.poll-btns {
display: flex;
gap: 10px;
margin-top: 15px;
margin-bottom: 15px;
}
.poll-btn {
padding: 10px 15px;
background: var(--body-bg);
color: #fff;
border: none;
border-radius: 4px;
cursor: pointer;
transition: background 0.3s;
}
.poll-btn:hover {
background: var(--btns-bg-color) !important;
color: #fff;
}
.del-btn {
padding: 10px 15px;
background: #dc3545;
color: #fff;
border: none;
border-radius: 4px;
cursor: pointer;
transition: background 0.3s;
}
.del-btn:hover {
background: #c82333;
color: #fff;
}
.polls-container {
margin-top: 20px;
}
.poll {
background: var(--body-bg);
padding: 15px;
margin-bottom: 20px;
border-radius: 8px;
box-shadow: 0 0 10px rgba(0, 0, 0, 0.1);
}
.poll .options {
margin-top: 10px;
}
.poll .options div {
margin-bottom: 5px;
}
.poll ul {
list-style-type: none;
padding: 0;
margin-top: 10px;
}
.poll ul li {
background: var(--body-bg);
padding: 8px;
border: var(--border);
border-radius: 4px;
margin-bottom: 5px;
color: #fff;
}
.options input {
cursor: pointer;
}
.options label {
margin-left: 5px;
color: #fff;
}
.view-btn {
padding: 10px 15px;
background: var(--body-bg);
color: #fff;
border: none;
border-radius: 4px;
cursor: pointer;
transition: background 0.3s;
}
.view-btn:hover {
background: var(--btns-bg-color) !important;
color: #fff;
}
#pollCloseBtn {
position: absolute;
float: right;
right: 20px;
font-size: 1.6rem;
}

عرض الملف

@@ -1504,6 +1504,8 @@ z-index:
- 3 control buttons
- 4 whiteboard
- 5 chat group
- 5 polls
- 5 transcription
- 6 settings
- 7 participants/lobby
- 8 send receive progress

عرض الملف

@@ -11,7 +11,7 @@ if (location.href.substr(0, 5) !== 'https') location.href = 'https' + location.h
* @license For commercial or closed source, contact us at license.mirotalk@gmail.com or purchase directly via CodeCanyon
* @license CodeCanyon: https://codecanyon.net/item/mirotalk-sfu-webrtc-realtime-video-conferences/40769970
* @author Miroslav Pejic - miroslav.pejic.85@gmail.com
* @version 1.4.99
* @version 1.5.10
*
*/
@@ -176,6 +176,18 @@ const initMicrophoneSelect = getId('initMicrophoneSelect');
const speakerSelect = getId('speakerSelect');
const initSpeakerSelect = getId('initSpeakerSelect');
// ####################################################
// POLLS
// ####################################################
const createPollForm = getId('createPollForm');
const pollsContainer = getId('pollsContainer');
const addOptionButton = getId('addOptionButton');
const delOptionButton = getId('delOptionButton');
const optionsContainer = getId('optionsContainer');
const selectedOptions = {};
let pollOpen = false;
// ####################################################
// DYNAMIC SETTINGS
// ####################################################
@@ -333,6 +345,7 @@ function initClient() {
setTippy('chatShowParticipantsList', 'Toggle participants list', 'bottom');
setTippy('chatMaxButton', 'Maximize', 'bottom');
setTippy('chatMinButton', 'Minimize', 'bottom');
setTippy('pollCloseBtn', 'Close', 'bottom');
setTippy('participantsSaveBtn', 'Save participants info', 'bottom');
setTippy('participantsRaiseHandBtn', 'Toggle raise hands', 'bottom');
setTippy('participantsUnreadMessagesBtn', 'Toggle unread messages', 'bottom');
@@ -372,6 +385,7 @@ function refreshMainButtonsToolTipPlacement() {
setTippy('emojiRoomButton', 'Toggle emoji reaction', placement);
setTippy('swapCameraButton', 'Swap the camera', placement);
setTippy('chatButton', 'Toggle the chat', placement);
setTippy('pollButton', 'Toggle the poll', placement);
setTippy('transcriptionButton', 'Toggle transcription', placement);
setTippy('whiteboardButton', 'Toggle the whiteboard', placement);
setTippy('settingsButton', 'Toggle the settings', placement);
@@ -1267,6 +1281,7 @@ function roomIsReady() {
hide(tabRecordingBtn);
}
BUTTONS.main.chatButton && show(chatButton);
BUTTONS.main.pollButton && show(pollButton);
BUTTONS.main.raiseHandButton && show(raiseHandButton);
BUTTONS.main.emojiRoomButton && show(emojiRoomButton);
!BUTTONS.chat.chatSaveButton && hide(chatSaveButton);
@@ -1300,9 +1315,11 @@ function roomIsReady() {
hide(transcriptionTogglePinBtn);
hide(transcriptionMaxBtn);
hide(transcriptionMinBtn);
rc.pollMaximize();
} else {
rc.makeDraggable(emojiPickerContainer, emojiPickerHeader);
rc.makeDraggable(chatRoom, chatHeader);
rc.makeDraggable(pollRoom, pollHeader);
rc.makeDraggable(mySettings, mySettingsHeader);
rc.makeDraggable(whiteboard, whiteboardHeader);
rc.makeDraggable(sendFileDiv, imgShareSend);
@@ -1555,6 +1572,22 @@ function handleButtons() {
rc.toggleShowParticipants();
}
};
// Polls
pollButton.onclick = () => {
rc.togglePoll();
};
pollCloseBtn.onclick = () => {
rc.togglePoll();
};
addOptionButton.onclick = () => {
rc.pollAddOptions();
};
delOptionButton.onclick = () => {
rc.pollDeleteOptions();
};
createPollForm.onsubmit = (e) => {
rc.pollCreateForm(e);
};
transcriptionButton.onclick = () => {
transcription.toggle();
};
@@ -4240,7 +4273,7 @@ function showAbout() {
imageUrl: image.about,
customClass: { image: 'img-about' },
position: 'center',
title: 'WebRTC SFU v1.4.99',
title: 'WebRTC SFU v1.5.10',
html: `
<br />
<div id="about">

عرض الملف

@@ -9,7 +9,7 @@
* @license For commercial or closed source, contact us at license.mirotalk@gmail.com or purchase directly via CodeCanyon
* @license CodeCanyon: https://codecanyon.net/item/mirotalk-sfu-webrtc-realtime-video-conferences/40769970
* @author Miroslav Pejic - miroslav.pejic.85@gmail.com
* @version 1.4.99
* @version 1.5.10
*
*/
@@ -551,6 +551,10 @@ class RoomClient {
elemDisplay('tabRTMPStreamingBtn', false);
}
}
// There is polls
if (room.thereIsPolls) {
this.socket.emit('updatePoll');
}
}
// PARTICIPANTS
@@ -869,6 +873,7 @@ class RoomClient {
this.socket.on('errorRTMP', this.handleErrorRTMP);
this.socket.on('endRTMPfromURL', this.handleEndRTMPfromURL);
this.socket.on('errorRTMPfromURL', this.handleErrorRTMPfromURL);
this.socket.on('updatePolls', this.handleUpdatePolls);
}
// ####################################################
@@ -1039,6 +1044,10 @@ class RoomClient {
this.errorRTMPfromURL(data);
};
handleUpdatePolls = (data) => {
this.pollsUpdate(data);
};
// ####################################################
// SERVER AWAY/MAINTENANCE
// ####################################################
@@ -7730,6 +7739,216 @@ class RoomClient {
});
}
// ##############################################
// POOLS
// ##############################################
togglePoll() {
const pollRoom = this.getId('pollRoom');
pollRoom.classList.toggle('show');
if (!pollOpen) {
this.pollCenter();
this.sound('open');
}
pollOpen = !pollOpen;
}
pollCenter() {
const pollRoom = this.getId('pollRoom');
pollRoom.style.position = 'fixed';
pollRoom.style.transform = 'translate(-50%, -50%)';
pollRoom.style.top = '50%';
pollRoom.style.left = '50%';
}
pollMaximize() {
const pollRoom = this.getId('pollRoom');
pollRoom.style.maxHeight = '100vh';
pollRoom.style.maxWidth = '100vw';
}
pollsUpdate(polls) {
if (!pollOpen) this.togglePoll();
pollsContainer.innerHTML = '';
polls.forEach((poll, index) => {
const pollDiv = document.createElement('div');
pollDiv.className = 'poll';
const question = document.createElement('h3');
question.className = 'poll-h3';
question.textContent = poll.question;
pollDiv.appendChild(question);
const options = document.createElement('div');
options.className = 'options';
poll.options.forEach((option) => {
const optionDiv = document.createElement('div');
const input = document.createElement('input');
input.type = 'radio';
input.name = `poll${index}`;
input.value = option;
if (selectedOptions[index] === option) {
input.checked = true;
}
input.addEventListener('change', () => {
selectedOptions[index] = option;
this.socket.emit('vote', { pollIndex: index, option });
});
const label = document.createElement('label');
label.textContent = option;
optionDiv.appendChild(input);
optionDiv.appendChild(label);
options.appendChild(optionDiv);
});
pollDiv.appendChild(options);
// Only the presenters
// if (isPresenter) {
const pollButtonsDiv = document.createElement('div');
pollButtonsDiv.className = 'poll-btns';
// Toggle voters button
const toggleButton = document.createElement('button');
const toggleButtonIcon = document.createElement('i');
toggleButtonIcon.className = 'fas fa-users';
toggleButton.textContent = ' Toggle Voters';
toggleButton.className = 'view-btn';
// Append the icon to the button
toggleButton.insertBefore(toggleButtonIcon, toggleButton.firstChild);
toggleButton.addEventListener('click', () => {
votersList.style.display === 'none'
? (votersList.style.display = 'block')
: (votersList.style.display = 'none');
});
pollButtonsDiv.appendChild(toggleButton);
// Edit poll button using swal
const editPollButton = document.createElement('button');
const editPollButtonIcon = document.createElement('i');
editPollButtonIcon.className = 'fas fa-pen-to-square';
editPollButton.textContent = ' Edit Poll';
editPollButton.className = 'poll-btn';
editPollButton.insertBefore(editPollButtonIcon, editPollButton.firstChild);
editPollButton.addEventListener('click', () => {
Swal.fire({
allowOutsideClick: false,
allowEscapeKey: false,
background: swalBackground,
title: 'Edit Poll',
html: this.createPollInputs(poll),
focusConfirm: false,
showCancelButton: true,
confirmButtonText: 'Save',
cancelButtonText: 'Cancel',
cancelButtonColor: '#dc3545',
preConfirm: () => {
const newQuestion = document.getElementById('swal-input-question').value;
const newOptions = this.getPollOptions(poll.options.length);
this.socket.emit('editPoll', {
index,
question: newQuestion,
options: newOptions,
peer_name: this.peer_name,
peer_uuid: this.peer_uuid,
});
},
showClass: { popup: 'animate__animated animate__fadeInDown' },
hideClass: { popup: 'animate__animated animate__fadeOutUp' },
});
});
pollButtonsDiv.appendChild(editPollButton);
// Delete poll button
const deletePollButton = document.createElement('button');
const deletePollButtonIcon = document.createElement('i');
deletePollButtonIcon.className = 'fas fa-minus';
deletePollButton.textContent = ' Delete Poll';
deletePollButton.className = 'del-btn';
deletePollButton.insertBefore(deletePollButtonIcon, deletePollButton.firstChild);
deletePollButton.addEventListener('click', () => {
this.socket.emit('deletePoll', { index, peer_name: this.peer_name, peer_uuid: this.peer_uuid });
});
pollButtonsDiv.appendChild(deletePollButton);
// Append buttons to poll
pollDiv.appendChild(pollButtonsDiv);
// Create voter lists
const votersList = document.createElement('ul');
votersList.style.display = 'none';
for (const [user, vote] of Object.entries(poll.voters)) {
const voter = document.createElement('li');
voter.textContent = `${user}: ${vote}`;
votersList.appendChild(voter);
}
pollDiv.appendChild(votersList);
// }
pollsContainer.appendChild(pollDiv);
});
}
pollCreateForm(e) {
e.preventDefault();
const question = e.target.question.value;
const optionInputs = document.querySelectorAll('.option-input');
const options = Array.from(optionInputs).map((input) => input.value.trim());
this.socket.emit('createPoll', { question, options });
e.target.reset();
optionsContainer.innerHTML = '';
const initialOptionInput = document.createElement('input');
initialOptionInput.type = 'text';
initialOptionInput.name = 'option';
initialOptionInput.className = 'option-input';
initialOptionInput.required = true;
optionsContainer.appendChild(initialOptionInput);
}
pollAddOptions() {
const optionInput = document.createElement('input');
optionInput.type = 'text';
optionInput.name = 'option';
optionInput.className = 'option-input';
optionInput.required = true;
optionsContainer.appendChild(optionInput);
}
pollDeleteOptions() {
const optionInputs = document.querySelectorAll('.option-input');
if (optionInputs.length > 1) {
optionsContainer.removeChild(optionInputs[optionInputs.length - 1]);
}
}
createPollInputs(poll) {
const questionInput = `<input id="swal-input-question" class="swal2-input" value="${poll.question}">`;
const optionsInputs = poll.options
.map(
(option, i) => `
<input id="swal-input-option${i}" class="swal2-input" value="${option}">
`,
)
.join('');
return questionInput + optionsInputs;
}
getPollOptions(optionCount) {
const options = [];
for (let i = 0; i < optionCount; i++) {
options.push(document.getElementById(`swal-input-option${i}`).value);
}
return options;
}
sleep(ms) {
return new Promise((resolve) => setTimeout(resolve, ms));
}

عرض الملف

@@ -22,6 +22,7 @@ let BUTTONS = {
startScreenButton: true,
swapCameraButton: true,
chatButton: true,
pollButton: true,
raiseHandButton: true,
transcriptionButton: true,
whiteboardButton: true,

عرض الملف

@@ -28,6 +28,8 @@ const commands = {
chatOn: 'open the chat',
chatSend: 'send',
chatOff: 'close the chat',
pollOn: 'open the poll',
pollOff: 'close the poll',
toggleTr: 'toggle transcription',
whiteboardOn: 'open the whiteboard',
whiteboardOff: 'close the whiteboard',
@@ -186,6 +188,14 @@ function execVoiceCommands(transcript) {
printCommand(commands.chatOn);
chatButton.click();
break;
case commands.pollOn:
printCommand(commands.pollOn);
pollButton.click();
break;
case commands.pollOff:
printCommand(commands.pollOff);
pollCloseBtn.click();
break;
case commands.chatSend:
printCommand(commands.chatSend);
chatSendButton.click();

عرض الملف

@@ -42,6 +42,7 @@
<link rel="stylesheet" href="../css/Room.css" />
<link rel="stylesheet" href="../css/VideoGrid.css" />
<link rel="stylesheet" href="../css/GroupChat.css" />
<link rel="stylesheet" href="../css/Polls.css" />
<!-- https://cdnjs.com/libraries/font-awesome -->
@@ -178,6 +179,7 @@ access to use this app.
<button id="lowerHandButton" class="hidden"><i id="lowerHandIcon" class="fas fa-hand-paper"></i></button>
<button id="emojiRoomButton" class="hidden"><i class="fas fa-face-smile"></i></button>
<button id="chatButton" class="hidden"><i class="fas fa-comments"></i></button>
<button id="pollButton" class="hidden"><i class="fas fa-square-poll-horizontal"></i></button>
<button id="transcriptionButton" class="hidden"><i class="fas fa-closed-captioning"></i></button>
<button id="whiteboardButton" class="hidden"><i class="fas fa-chalkboard-teacher"></i></button>
<button id="settingsButton" class="hidden"><i class="fas fa-cogs"></i></button>
@@ -1436,6 +1438,40 @@ access to use this app.
</div>
</section>
<section id="pollRoom" class="pls-container fadein hidden">
<div id="pollHeader" class="poll-header">
<h2 class="poll-h2">Create a Poll</h2>
<button id="pollCloseBtn" class="fas fa-times"></button>
</div>
<div id="pollForm" class="poll-creation">
<form id="createPollForm" class="form">
<div class="form-group">
<label for="question">Question:</label>
<input type="text" id="question" name="question" required />
</div>
<div class="form-group">
<label for="options">Options:</label>
<div id="optionsContainer">
<input type="text" name="option" class="option-input" required />
</div>
<br />
<div class="poll-btns">
<button type="button" id="addOptionButton" class="poll-btn">
<i class="fas fa-plus"></i> Add Option
</button>
<button type="button" id="delOptionButton" class="del-btn">
<i class="fas fa-minus"></i> Delete Option
</button>
<button type="submit" id="addPollButton" class="poll-btn">
<i class="fas fa-square-poll-horizontal"></i> Create Poll
</button>
</div>
</div>
</form>
</div>
<div id="pollsContainer" class="polls-container"></div>
</section>
<section id="transcriptionRoom" class="transcription-room fadein">
<section id="transcriptionSection" class="transcription">
<header id="transcriptionHeader" class="transcription-header">