[mirotalksfu] - add transcription

هذا الالتزام موجود في:
Miroslav Pejic
2023-10-18 09:59:38 +02:00
الأصل aad20bf023
التزام 1b6de0da8e
10 ملفات معدلة مع 687 إضافات و52 حذوفات

عرض الملف

@@ -40,7 +40,7 @@ dependencies: {
* @license For commercial or closed source, contact us at license.mirotalk@gmail.com or purchase directly via CodeCanyon * @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 * @license CodeCanyon: https://codecanyon.net/item/mirotalk-sfu-webrtc-realtime-video-conferences/40769970
* @author Miroslav Pejic - miroslav.pejic.85@gmail.com * @author Miroslav Pejic - miroslav.pejic.85@gmail.com
* @version 1.0.9 * @version 1.1.0
* *
*/ */
@@ -608,18 +608,13 @@ function startServer() {
const data = checkXSS(dataObject); const data = checkXSS(dataObject);
log.debug('Cmd', data); log.debug('cmd', data);
const room = roomList.get(socket.room_id); const room = roomList.get(socket.room_id);
// cmd|foo|bar|.... switch (data.type) {
const words = data.split('|');
let cmd = words[0];
switch (cmd) {
case 'privacy': case 'privacy':
room.getPeers() room.getPeers().get(socket.id).updatePeerInfo({ type: data.type, status: data.active });
.get(socket.id)
.updatePeerInfo({ type: cmd, status: words[2] == 'true' });
break; break;
default: default:
break; break;

عرض الملف

@@ -1,6 +1,6 @@
{ {
"name": "mirotalksfu", "name": "mirotalksfu",
"version": "1.0.9", "version": "1.1.0",
"description": "WebRTC SFU browser-based video calls", "description": "WebRTC SFU browser-based video calls",
"main": "Server.js", "main": "Server.js",
"scripts": { "scripts": {
@@ -34,8 +34,8 @@
"author": "Miroslav Pejic", "author": "Miroslav Pejic",
"license": "AGPL-3.0", "license": "AGPL-3.0",
"dependencies": { "dependencies": {
"@sentry/integrations": "7.74.0", "@sentry/integrations": "7.74.1",
"@sentry/node": "7.74.0", "@sentry/node": "7.74.1",
"axios": "^1.5.1", "axios": "^1.5.1",
"body-parser": "1.20.2", "body-parser": "1.20.2",
"colors": "1.4.0", "colors": "1.4.0",
@@ -47,7 +47,7 @@
"mediasoup": "3.12.13", "mediasoup": "3.12.13",
"mediasoup-client": "3.6.101", "mediasoup-client": "3.6.101",
"ngrok": "^4.3.3", "ngrok": "^4.3.3",
"openai": "^4.12.1", "openai": "^4.12.4",
"qs": "6.11.2", "qs": "6.11.2",
"socket.io": "4.7.2", "socket.io": "4.7.2",
"swagger-ui-express": "5.0.0", "swagger-ui-express": "5.0.0",

عرض الملف

@@ -68,6 +68,11 @@
--btns-width: 320px; --btns-width: 320px;
--btns-flex-direction: row; --btns-flex-direction: row;
*/ */
--transcription-height: 680px;
--transcription-width: 420px;
--transcription-bg: radial-gradient(#393939, #000000);
--vmi-wh: 15vw; --vmi-wh: 15vw;
/* https://developer.mozilla.org/en-US/docs/Web/CSS/object-fit */ /* https://developer.mozilla.org/en-US/docs/Web/CSS/object-fit */
--videoObjFit: cover; --videoObjFit: cover;
@@ -515,6 +520,142 @@ th {
background-color: #ddd; background-color: #ddd;
} }
/*--------------------------------------------------------------
# Transcription Room
--------------------------------------------------------------*/
.transcription-room {
z-index: 5;
display: none;
position: fixed;
height: var(--transcription-height);
width: var(--transcription-width);
min-height: var(--transcription-height);
min-width: var(--transcription-width);
background: var(--transcription-bg);
border: var(--border);
border-radius: 10px;
box-shadow: var(--box-shadow);
resize: both;
overflow: hidden;
}
.transcription {
width: 100%;
height: 100%;
display: flex;
flex-flow: column wrap;
justify-content: space-between;
background: var(--transcription-bg);
}
/*--------------------------------------------------------------
# Transcription room header
--------------------------------------------------------------*/
.transcription-header {
display: flex;
justify-content: space-between;
padding: 10px;
background: rgba(0, 0, 0, 0.2);
color: #666;
cursor: move;
}
.transcription-header-title button,
.transcription-header-options button {
border: none;
transition: all 0.3s ease-in-out;
background: rgba(0, 0, 0, 0.2);
color: #fff;
border-radius: 5px;
transition: background 0.23s;
}
/*--------------------------------------------------------------
# Transcription room output area
--------------------------------------------------------------*/
.transcription-chat {
flex: 1;
overflow-y: auto;
padding: 10px;
background: var(--transcription-bg);
max-width: 100%;
}
.transcription-chat::-webkit-scrollbar {
width: 5px;
}
.transcription-chat::-webkit-scrollbar-track {
background: transparent;
}
.transcription-chat::-webkit-scrollbar-thumb {
background: black;
}
/*--------------------------------------------------------------
# Transcription room left side
--------------------------------------------------------------*/
.left-msg .msg-transcription-bubble {
background: var(--left-msg-bg);
border-bottom-left-radius: 0;
color: #fff;
width: var(--msger-bubble-width);
}
.left-msg .msg-transcription-img {
margin: 0px 5px 5px 0px;
}
/*--------------------------------------------------------------
# Transcription bubble text
--------------------------------------------------------------*/
.msg-transcription-bubble {
padding: 15px;
border-radius: 15px;
}
.msg-transcription-info {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 10px;
}
.msg-transcription-info-name {
margin-right: 10px;
font-weight: bold;
}
.msg-transcription-info-time {
font-size: 0.85em;
}
.msg-transcription-text {
display: inline;
overflow-wrap: break-word;
word-wrap: break-word;
hyphens: auto;
}
/*--------------------------------------------------------------
# Transcription room input area
--------------------------------------------------------------*/
.transcription-inputarea {
display: inline-flex;
padding: 10px;
border: none;
border: var(--border);
}
.transcription-inputarea select {
margin: 0 5px;
}
/*-------------------------------------------------------------- /*--------------------------------------------------------------
# Chat Room # Chat Room
--------------------------------------------------------------*/ --------------------------------------------------------------*/

عرض الملف

@@ -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 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 * @license CodeCanyon: https://codecanyon.net/item/mirotalk-sfu-webrtc-realtime-video-conferences/40769970
* @author Miroslav Pejic - miroslav.pejic.85@gmail.com * @author Miroslav Pejic - miroslav.pejic.85@gmail.com
* @version 1.0.9 * @version 1.1.0
* *
*/ */
@@ -137,6 +137,8 @@ let scriptProcessor = null;
const RoomURL = window.location.origin + '/join/' + room_id; const RoomURL = window.location.origin + '/join/' + room_id;
let transcription;
// #################################################### // ####################################################
// INIT ROOM // INIT ROOM
// #################################################### // ####################################################
@@ -208,6 +210,10 @@ function initClient() {
} }
setupWhiteboard(); setupWhiteboard();
initEnumerateDevices(); initEnumerateDevices();
// Transcription
transcription = new Transcription();
transcription.init();
} }
// #################################################### // ####################################################
@@ -232,6 +238,7 @@ function refreshMainButtonsToolTipPlacement() {
setTippy('roomEmojiPicker', 'Toggle emoji reaction', placement); setTippy('roomEmojiPicker', 'Toggle emoji reaction', placement);
setTippy('swapCameraButton', 'Swap the camera', placement); setTippy('swapCameraButton', 'Swap the camera', placement);
setTippy('chatButton', 'Toggle the chat', placement); setTippy('chatButton', 'Toggle the chat', placement);
setTippy('transcriptionButton', 'Toggle transcription', placement);
setTippy('participantsButton', 'Toggle participants', placement); setTippy('participantsButton', 'Toggle participants', placement);
setTippy('whiteboardButton', 'Toggle the whiteboard', placement); setTippy('whiteboardButton', 'Toggle the whiteboard', placement);
setTippy('settingsButton', 'Toggle the settings', placement); setTippy('settingsButton', 'Toggle the settings', placement);
@@ -917,6 +924,7 @@ function joinRoom(peer_name, room_id) {
isScreenAllowed, isScreenAllowed,
joinRoomWithScreen, joinRoomWithScreen,
isSpeechSynthesisSupported, isSpeechSynthesisSupported,
transcription,
roomIsReady, roomIsReady,
); );
handleRoomClientEvents(); handleRoomClientEvents();
@@ -943,11 +951,15 @@ function roomIsReady() {
BUTTONS.chat.chatMarkdownButton && show(chatMarkdownButton); BUTTONS.chat.chatMarkdownButton && show(chatMarkdownButton);
BUTTONS.chat.chatGPTButton && show(chatGPTButton); BUTTONS.chat.chatGPTButton && show(chatGPTButton);
BUTTONS.chat.chatShareFileButton && show(chatShareFileButton); BUTTONS.chat.chatShareFileButton && show(chatShareFileButton);
if (isWebkitSpeechRecognitionSupported && BUTTONS.chat.chatSpeechStartButton) {
show(chatSpeechStartButton); isWebkitSpeechRecognitionSupported && BUTTONS.chat.chatSpeechStartButton
} else { ? show(chatSpeechStartButton)
BUTTONS.chat.chatSpeechStartButton = false; : (BUTTONS.chat.chatSpeechStartButton = false);
}
transcription.isSupported() && BUTTONS.main.transcriptionButton
? show(transcriptionButton)
: (BUTTONS.main.transcriptionButton = false);
show(chatCleanTextButton); show(chatCleanTextButton);
show(chatPasteButton); show(chatPasteButton);
show(chatSendButton); show(chatSendButton);
@@ -957,6 +969,10 @@ function roomIsReady() {
hide(chatTogglePin); hide(chatTogglePin);
hide(chatMaxButton); hide(chatMaxButton);
hide(chatMinButton); hide(chatMinButton);
transcription.maximize();
hide(transcriptionTogglePinBtn);
hide(transcriptionMaxBtn);
hide(transcriptionMinBtn);
} else { } else {
rc.makeDraggable(emojiPickerContainer, emojiPickerHeader); rc.makeDraggable(emojiPickerContainer, emojiPickerHeader);
rc.makeDraggable(chatRoom, chatHeader); rc.makeDraggable(chatRoom, chatHeader);
@@ -966,6 +982,7 @@ function roomIsReady() {
rc.makeDraggable(sendFileDiv, imgShareSend); rc.makeDraggable(sendFileDiv, imgShareSend);
rc.makeDraggable(receiveFileDiv, imgShareReceive); rc.makeDraggable(receiveFileDiv, imgShareReceive);
rc.makeDraggable(lobby, lobbyHeader); rc.makeDraggable(lobby, lobbyHeader);
rc.makeDraggable(transcriptionRoom, transcriptionHeader);
if (navigator.getDisplayMedia || navigator.mediaDevices.getDisplayMedia) { if (navigator.getDisplayMedia || navigator.mediaDevices.getDisplayMedia) {
if (BUTTONS.main.startScreenButton) { if (BUTTONS.main.startScreenButton) {
show(startScreenButton); show(startScreenButton);
@@ -1150,6 +1167,30 @@ function handleButtons() {
chatButton.onclick = () => { chatButton.onclick = () => {
rc.toggleChat(); rc.toggleChat();
}; };
transcriptionButton.onclick = () => {
transcription.toggle();
};
transcriptionCloseBtn.onclick = () => {
transcription.toggle();
};
transcriptionTogglePinBtn.onclick = () => {
transcription.togglePinUnpin();
};
transcriptionMaxBtn.onclick = () => {
transcription.maximize();
};
transcriptionMinBtn.onclick = () => {
transcription.minimize();
};
transcriptionGhostBtn.onclick = () => {
transcription.toggleBg();
};
transcriptionSaveBtn.onclick = () => {
transcription.save();
};
transcriptionCleanBtn.onclick = () => {
transcription.delete();
};
chatGhostButton.onclick = (e) => { chatGhostButton.onclick = (e) => {
rc.chatToggleBg(); rc.chatToggleBg();
}; };
@@ -1195,10 +1236,16 @@ function handleButtons() {
fileShareButton.click(); fileShareButton.click();
}; };
chatSpeechStartButton.onclick = () => { chatSpeechStartButton.onclick = () => {
startSpeech(true); startSpeech();
}; };
chatSpeechStopButton.onclick = () => { chatSpeechStopButton.onclick = () => {
startSpeech(false); stopSpeech();
};
transcriptionSpeechStart.onclick = () => {
transcription.start();
};
transcriptionSpeechStop.onclick = () => {
transcription.stop();
}; };
fullScreenButton.onclick = () => { fullScreenButton.onclick = () => {
rc.toggleFullScreen(); rc.toggleFullScreen();
@@ -1787,7 +1834,11 @@ function handleRoomEmojiPicker() {
function sendEmojiToRoom(data) { function sendEmojiToRoom(data) {
console.log('Selected Emoji:', data.native); console.log('Selected Emoji:', data.native);
const cmd = `roomEmoji|${peer_name}|${data.native}`; const cmd = {
type: 'roomEmoji',
peer_name: peer_name,
emoji: data.native,
};
if (rc.thereAreParticipants()) { if (rc.thereAreParticipants()) {
rc.emitCmd(cmd); rc.emitCmd(cmd);
} }
@@ -2907,6 +2958,7 @@ function setTheme() {
case 'dark': case 'dark':
swalBackground = 'radial-gradient(#393939, #000000)'; swalBackground = 'radial-gradient(#393939, #000000)';
document.documentElement.style.setProperty('--body-bg', 'radial-gradient(#393939, #000000)'); document.documentElement.style.setProperty('--body-bg', 'radial-gradient(#393939, #000000)');
document.documentElement.style.setProperty('--transcription-bg', 'radial-gradient(#393939, #000000)');
document.documentElement.style.setProperty('--msger-bg', 'radial-gradient(#393939, #000000)'); document.documentElement.style.setProperty('--msger-bg', 'radial-gradient(#393939, #000000)');
document.documentElement.style.setProperty('--left-msg-bg', '#252d31'); document.documentElement.style.setProperty('--left-msg-bg', '#252d31');
document.documentElement.style.setProperty('--right-msg-bg', '#056162'); document.documentElement.style.setProperty('--right-msg-bg', '#056162');
@@ -2921,6 +2973,7 @@ function setTheme() {
case 'grey': case 'grey':
swalBackground = 'radial-gradient(#666, #333)'; swalBackground = 'radial-gradient(#666, #333)';
document.documentElement.style.setProperty('--body-bg', 'radial-gradient(#666, #333)'); document.documentElement.style.setProperty('--body-bg', 'radial-gradient(#666, #333)');
document.documentElement.style.setProperty('--transcription-bg', 'radial-gradient(#666, #333)');
document.documentElement.style.setProperty('--msger-bg', 'radial-gradient(#666, #333)'); document.documentElement.style.setProperty('--msger-bg', 'radial-gradient(#666, #333)');
document.documentElement.style.setProperty('--left-msg-bg', '#252d31'); document.documentElement.style.setProperty('--left-msg-bg', '#252d31');
document.documentElement.style.setProperty('--right-msg-bg', '#056162'); document.documentElement.style.setProperty('--right-msg-bg', '#056162');
@@ -2935,6 +2988,7 @@ function setTheme() {
case 'green': case 'green':
swalBackground = 'radial-gradient(#003934, #001E1A)'; swalBackground = 'radial-gradient(#003934, #001E1A)';
document.documentElement.style.setProperty('--body-bg', 'radial-gradient(#003934, #001E1A)'); document.documentElement.style.setProperty('--body-bg', 'radial-gradient(#003934, #001E1A)');
document.documentElement.style.setProperty('--transcription-bg', 'radial-gradient(#003934, #001E1A)');
document.documentElement.style.setProperty('--msger-bg', 'radial-gradient(#003934, #001E1A)'); document.documentElement.style.setProperty('--msger-bg', 'radial-gradient(#003934, #001E1A)');
document.documentElement.style.setProperty('--left-msg-bg', '#003934'); document.documentElement.style.setProperty('--left-msg-bg', '#003934');
document.documentElement.style.setProperty('--right-msg-bg', '#001E1A'); document.documentElement.style.setProperty('--right-msg-bg', '#001E1A');
@@ -2949,6 +3003,7 @@ function setTheme() {
case 'blue': case 'blue':
swalBackground = 'radial-gradient(#306bac, #141B41)'; swalBackground = 'radial-gradient(#306bac, #141B41)';
document.documentElement.style.setProperty('--body-bg', 'radial-gradient(#306bac, #141B41)'); document.documentElement.style.setProperty('--body-bg', 'radial-gradient(#306bac, #141B41)');
document.documentElement.style.setProperty('--transcription-bg', 'radial-gradient(#306bac, #141B41)');
document.documentElement.style.setProperty('--msger-bg', 'radial-gradient(#306bac, #141B41)'); document.documentElement.style.setProperty('--msger-bg', 'radial-gradient(#306bac, #141B41)');
document.documentElement.style.setProperty('--left-msg-bg', '#306bac'); document.documentElement.style.setProperty('--left-msg-bg', '#306bac');
document.documentElement.style.setProperty('--right-msg-bg', '#141B41'); document.documentElement.style.setProperty('--right-msg-bg', '#141B41');
@@ -2963,6 +3018,7 @@ function setTheme() {
case 'red': case 'red':
swalBackground = 'radial-gradient(#69140E, #3C1518)'; swalBackground = 'radial-gradient(#69140E, #3C1518)';
document.documentElement.style.setProperty('--body-bg', 'radial-gradient(#69140E, #3C1518)'); document.documentElement.style.setProperty('--body-bg', 'radial-gradient(#69140E, #3C1518)');
document.documentElement.style.setProperty('--transcription-bg', 'radial-gradient(#69140E, #3C1518)');
document.documentElement.style.setProperty('--msger-bg', 'radial-gradient(#69140E, #3C1518)'); document.documentElement.style.setProperty('--msger-bg', 'radial-gradient(#69140E, #3C1518)');
document.documentElement.style.setProperty('--left-msg-bg', '#69140E'); document.documentElement.style.setProperty('--left-msg-bg', '#69140E');
document.documentElement.style.setProperty('--right-msg-bg', '#3C1518'); document.documentElement.style.setProperty('--right-msg-bg', '#3C1518');

عرض الملف

@@ -9,7 +9,7 @@
* @license For commercial or closed source, contact us at license.mirotalk@gmail.com or purchase directly via CodeCanyon * @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 * @license CodeCanyon: https://codecanyon.net/item/mirotalk-sfu-webrtc-realtime-video-conferences/40769970
* @author Miroslav Pejic - miroslav.pejic.85@gmail.com * @author Miroslav Pejic - miroslav.pejic.85@gmail.com
* @version 1.0.9 * @version 1.1.0
* *
*/ */
@@ -136,6 +136,7 @@ class RoomClient {
isScreenAllowed, isScreenAllowed,
joinRoomWithScreen, joinRoomWithScreen,
isSpeechSynthesisSupported, isSpeechSynthesisSupported,
transcription,
successCallback, successCallback,
) { ) {
this.localAudioEl = localAudioEl; this.localAudioEl = localAudioEl;
@@ -196,6 +197,8 @@ class RoomClient {
this.RoomPassword = false; this.RoomPassword = false;
this.transcription = transcription;
// File transfer settings // File transfer settings
this.fileToSend = null; this.fileToSend = null;
this.fileReader = null; this.fileReader = null;
@@ -2734,6 +2737,9 @@ class RoomClient {
if (this.isChatPinned) { if (this.isChatPinned) {
this.chatPin(); this.chatPin();
} }
if (this.transcription.isPin()) {
this.transcription.pinned();
}
} }
adaptVideoObjectFit(index) { adaptVideoObjectFit(index) {
@@ -2782,7 +2788,11 @@ class RoomClient {
this.sound('click'); this.sound('click');
isVideoPrivacyActive = !isVideoPrivacyActive; isVideoPrivacyActive = !isVideoPrivacyActive;
this.setVideoPrivacyStatus(this.peer_id, isVideoPrivacyActive); this.setVideoPrivacyStatus(this.peer_id, isVideoPrivacyActive);
this.emitCmd(`privacy|${this.peer_id}|${isVideoPrivacyActive}`); this.emitCmd({
type: 'privacy',
peer_id: this.peer_id,
active: isVideoPrivacyActive,
});
}); });
} }
} }
@@ -2881,6 +2891,9 @@ class RoomClient {
} }
toggleChatPin() { toggleChatPin() {
if (transcription.isPin()) {
return userLog('info', 'Please unpin the transcription that appears to be currently pinned', 'top-end');
}
this.isChatPinned ? this.chatUnpin() : this.chatPin(); this.isChatPinned ? this.chatUnpin() : this.chatPin();
this.sound('click'); this.sound('click');
} }
@@ -4733,20 +4746,20 @@ class RoomClient {
// HANDLE COMMANDS // HANDLE COMMANDS
// #################################################### // ####################################################
emitCmd(data) { emitCmd(cmd) {
this.socket.emit('cmd', data); this.socket.emit('cmd', cmd);
} }
handleCmd(data) { handleCmd(cmd) {
// cmd|foo|bar|.... switch (cmd.type) {
const words = data.split('|');
let cmd = words[0];
switch (cmd) {
case 'privacy': case 'privacy':
this.setVideoPrivacyStatus(words[1], words[2] == 'true'); this.setVideoPrivacyStatus(cmd.peer_id, cmd.active);
break; break;
case 'roomEmoji': case 'roomEmoji':
this.handleRoomEmoji(words); this.handleRoomEmoji(cmd);
break;
case 'transcript':
this.transcription.handleTranscript(cmd);
break; break;
default: default:
break; break;
@@ -4754,7 +4767,7 @@ class RoomClient {
} }
} }
handleRoomEmoji(words, duration = 5000) { handleRoomEmoji(cmd, duration = 5000) {
const userEmoji = document.getElementById(`userEmoji`); const userEmoji = document.getElementById(`userEmoji`);
if (userEmoji) { if (userEmoji) {
const emojiDisplay = document.createElement('div'); const emojiDisplay = document.createElement('div');
@@ -4764,7 +4777,7 @@ class RoomClient {
emojiDisplay.style.color = '#FFF'; emojiDisplay.style.color = '#FFF';
emojiDisplay.style.backgroundColor = 'rgba(0, 0, 0, 0.2)'; emojiDisplay.style.backgroundColor = 'rgba(0, 0, 0, 0.2)';
emojiDisplay.style.borderRadius = '10px'; emojiDisplay.style.borderRadius = '10px';
emojiDisplay.innerText = `${words[2]} ${words[1]}`; emojiDisplay.innerText = `${cmd.emoji} ${cmd.peer_name}`;
userEmoji.appendChild(emojiDisplay); userEmoji.appendChild(emojiDisplay);
setTimeout(() => { setTimeout(() => {
emojiDisplay.remove(); emojiDisplay.remove();

عرض الملف

@@ -17,6 +17,7 @@ const BUTTONS = {
startScreenButton: true, startScreenButton: true,
swapCameraButton: true, swapCameraButton: true,
chatButton: true, chatButton: true,
transcriptionButton: true,
participantsButton: true, participantsButton: true,
whiteboardButton: true, whiteboardButton: true,
emojiRoomButton: true, emojiRoomButton: true,

عرض الملف

@@ -8,6 +8,11 @@ let isVoiceCommandSupported = browserLanguage.includes('en-');
const speechRecognition = window.SpeechRecognition || window.webkitSpeechRecognition; const speechRecognition = window.SpeechRecognition || window.webkitSpeechRecognition;
/**
* Enable real-time voice recognition in the chat, allowing you to execute commands using your voice.
* Note: Currently, it supports only the English language.
* TODO make it multi languages...
*/
const commands = { const commands = {
shareRoom: 'room', shareRoom: 'room',
hideMe: 'hide me', hideMe: 'hide me',
@@ -23,6 +28,7 @@ const commands = {
chatOn: 'open the chat', chatOn: 'open the chat',
chatSend: 'send', chatSend: 'send',
chatOff: 'close the chat', chatOff: 'close the chat',
toggleTr: 'toggle transcription',
chatGPTOn: 'open chatGPT', chatGPTOn: 'open chatGPT',
chatGPTOff: 'close chatGPT', chatGPTOff: 'close chatGPT',
whiteboardOn: 'open the whiteboard', whiteboardOn: 'open the whiteboard',
@@ -84,34 +90,37 @@ if (speechRecognition) {
console.log('Speech recognition', recognition); console.log('Speech recognition', recognition);
recognition.onstart = function () { recognition.onstart = function () {
console.log('Start speech recognition'); console.log('Speech recognition started');
hide(chatSpeechStartButton); hide(chatSpeechStartButton);
show(chatSpeechStopButton); show(chatSpeechStopButton);
setColor(chatSpeechStopButton, 'lime'); setColor(chatSpeechStopButton, 'lime');
userLog('info', 'Speech recognition started', 'top-end');
}; };
recognition.onresult = (e) => { recognition.onresult = (e) => {
let current = e.resultIndex; let current = e.resultIndex;
let transcript = e.results[current][0].transcript; let transcript = e.results[current][0].transcript;
if (transcript) {
if (transcript.trim().toLowerCase() != commands.chatSend) { if (transcript.trim().toLowerCase() != commands.chatSend) {
chatMessage.value = transcript; chatMessage.value = transcript;
} }
if (isVoiceCommandsEnabled && isVoiceCommandSupported) {
if (isVoiceCommandsEnabled && isVoiceCommandSupported) { execVoiceCommands(transcript);
execVoiceCommands(transcript); }
} }
}; };
recognition.onerror = function (event) { recognition.onerror = function (event) {
console.warn('Speech recognition error', event.error); console.error('Speech recognition error', event.error);
userLog('error', `Speech recognition error ${event.error}`, 'top-end', 6000);
}; };
recognition.onend = function () { recognition.onend = function () {
console.log('Stop speech recognition'); console.log('Speech recognition stopped');
show(chatSpeechStartButton); show(chatSpeechStartButton);
hide(chatSpeechStopButton); hide(chatSpeechStopButton);
setColor(chatSpeechStopButton, 'white'); setColor(chatSpeechStopButton, 'white');
userLog('info', 'Speech recognition stopped', 'top-end');
}; };
isWebkitSpeechRecognitionSupported = true; isWebkitSpeechRecognitionSupported = true;
@@ -120,13 +129,13 @@ if (speechRecognition) {
console.warn('This browser not supports webkitSpeechRecognition'); console.warn('This browser not supports webkitSpeechRecognition');
} }
function startSpeech(action) { function startSpeech() {
if (action) { recognition.lang = browserLanguage;
recognition.lang = browserLanguage; recognition.start();
recognition.start(); }
} else {
recognition.stop(); function stopSpeech() {
} recognition.stop();
} }
function execVoiceCommands(transcript) { function execVoiceCommands(transcript) {
@@ -187,6 +196,9 @@ function execVoiceCommands(transcript) {
printCommand(commands.chatOff); printCommand(commands.chatOff);
chatCloseButton.click(); chatCloseButton.click();
break; break;
case commands.toggleTr:
transcriptionButton.click();
break;
case commands.whiteboardOn: case commands.whiteboardOn:
printCommand(commands.whiteboardOn); printCommand(commands.whiteboardOn);
whiteboardButton.click(); whiteboardButton.click();
@@ -342,6 +354,8 @@ function execVoiceCommands(transcript) {
chatSpeechStopButton.click(); chatSpeechStopButton.click();
break; break;
// ... // ...
default:
break;
} }
} }

387
public/js/Transcription.js Normal file
عرض الملف

@@ -0,0 +1,387 @@
'use strict';
class Transcription {
constructor() {
this.languages = [
['Afrikaans', ['af-ZA']],
['Bahasa Indonesia', ['id-ID']],
['Bahasa Melayu', ['ms-MY']],
['Català', ['ca-ES']],
['Čeština', ['cs-CZ']],
['Deutsch', ['de-DE']],
[
'English',
['en-AU', 'Australia'],
['en-CA', 'Canada'],
['en-IN', 'India'],
['en-NZ', 'New Zealand'],
['en-ZA', 'South Africa'],
['en-GB', 'United Kingdom'],
['en-US', 'United States'],
],
[
'Español',
['es-AR', 'Argentina'],
['es-BO', 'Bolivia'],
['es-CL', 'Chile'],
['es-CO', 'Colombia'],
['es-CR', 'Costa Rica'],
['es-EC', 'Ecuador'],
['es-SV', 'El Salvador'],
['es-ES', 'España'],
['es-US', 'Estados Unidos'],
['es-GT', 'Guatemala'],
['es-HN', 'Honduras'],
['es-MX', 'México'],
['es-NI', 'Nicaragua'],
['es-PA', 'Panamá'],
['es-PY', 'Paraguay'],
['es-PE', 'Perú'],
['es-PR', 'Puerto Rico'],
['es-DO', 'República Dominicana'],
['es-UY', 'Uruguay'],
['es-VE', 'Venezuela'],
],
['Euskara', ['eu-ES']],
['Français', ['fr-FR']],
['Galego', ['gl-ES']],
['Hrvatski', ['hr_HR']],
['IsiZulu', ['zu-ZA']],
['Íslenska', ['is-IS']],
['Italiano', ['it-IT', 'Italia'], ['it-CH', 'Svizzera']],
['Magyar', ['hu-HU']],
['Nederlands', ['nl-NL']],
['Norsk bokmål', ['nb-NO']],
['Polski', ['pl-PL']],
['Português', ['pt-BR', 'Brasil'], ['pt-PT', 'Portugal']],
['Română', ['ro-RO']],
['Slovenčina', ['sk-SK']],
['Suomi', ['fi-FI']],
['Svenska', ['sv-SE']],
['Türkçe', ['tr-TR']],
['български', ['bg-BG']],
['Pусский', ['ru-RU']],
['Српски', ['sr-RS']],
['한국어', ['ko-KR']],
[
'中文',
['cmn-Hans-CN', '普通话 (中国大陆)'],
['cmn-Hans-HK', '普通话 (香港)'],
['cmn-Hant-TW', '中文 (台灣)'],
['yue-Hant-HK', '粵語 (香港)'],
],
['日本語', ['ja-JP']],
['Lingua latīna', ['la']],
];
this.speechTranscription = window.SpeechRecognition || window.webkitSpeechRecognition;
this.isTranscriptionSupported = false;
this.transcriptionRunning = false;
this.transcription;
this.transcripts = [];
this.isBgTransparent = false;
this.isPinned = false;
}
isSupported() {
return Boolean(this.speechTranscription);
}
init() {
if (this.isSupported()) {
this.handleLanguages();
this.transcription = new this.speechTranscription();
this.transcription.maxAlternatives = 1;
this.transcription.continuous = true;
this.transcription.lang = transcriptionDialect.value;
this.transcription.onstart = function () {
console.log('Transcription started');
transcriptionSpeechStart.style.display = 'none';
transcriptionSpeechStop.style.display = 'block';
setColor(transcriptionSpeechStatus, 'lime');
userLog('info', 'Transcription started', 'top-end');
};
this.transcription.onresult = (e) => {
const current = e.resultIndex;
const transcript = e.results[current][0].transcript;
const transcriptionData = {
type: 'transcript',
room_id: room_id,
peer_name: peer_name,
text_data: transcript,
time_stamp: new Date(),
};
if (transcript) {
this.sendTranscript(transcriptionData);
this.handleTranscript(transcriptionData);
}
};
this.transcription.onaudiostart = () => {
console.log('Transcription start to capture your voice');
};
this.transcription.onaudioend = () => {
console.log('Transcription stop to capture your voice');
};
this.transcription.onerror = function (event) {
userLog('error', `Transcription error ${event.error}`, 'top-end', 6000);
console.error('Transcription error', event.error);
};
this.transcription.onend = function () {
console.log('Transcription stopped');
transcriptionSpeechStop.style.display = 'none';
transcriptionSpeechStart.style.display = 'block';
setColor(transcriptionSpeechStatus, 'white');
userLog('info', 'Transcription stopped', 'top-end');
};
this.isTranscriptionSupported = true;
console.info('This Browser support Transcription');
} else {
console.warn(
'This browser not support Transcription, check out supported browsers: https://developer.mozilla.org/en-US/docs/Web/API/Web_Speech_API#browser_compatibility',
);
}
}
sendTranscript(transcriptionData) {
if (rc.thereAreParticipants()) {
//console.log('TRANSCRIPTION SEND', transcriptionData);
rc.emitCmd(transcriptionData);
}
}
handleTranscript(transcriptionData) {
console.log('TRANSCRIPTION TEXT', transcriptionData.text_data);
transcriptionData.text_data = filterXSS(transcriptionData.text_data);
transcriptionData.peer_name = filterXSS(transcriptionData.peer_name);
const { peer_name, text_data } = transcriptionData;
const time_stamp = rc.getTimeNow();
const avatar_image = rc.genAvatarSvg(peer_name, 32);
if (this.isHidden()) this.toggle();
const msgHTML = `
<div class="msg left-msg">
<img class="msg-transcription-img" src="${avatar_image}" />
<div class="msg-transcription-bubble">
<div class="msg-transcription-info">
<div class="msg-transcription-info-name">${peer_name} : ${time_stamp}</div>
</div>
<div class="msg-transcription-text">${text_data}</div>
</div>
</div>
`;
transcriptionChat.insertAdjacentHTML('beforeend', msgHTML);
transcriptionChat.scrollTop += 500;
this.transcripts.push({
time: time_stamp,
name: peer_name,
caption: text_data,
});
rc.sound('transcript');
}
isHidden() {
return Boolean(transcriptionRoom.style.display === 'none' || transcriptionRoom.style.display === '');
}
toggle() {
if (this.isHidden()) {
this.center();
transcriptionRoom.style.display = 'block';
rc.sound('open');
} else {
transcriptionRoom.style.display = 'none';
}
if (this.isPinned) this.unpinned();
}
toggleBg() {
this.isBgTransparent = !this.isBgTransparent;
this.isBgTransparent
? document.documentElement.style.setProperty('--transcription-bg', 'rgba(0, 0, 0, 0.100)')
: setTheme();
}
maximize() {
hide(transcriptionMaxBtn);
show(transcriptionMinBtn);
this.center();
document.documentElement.style.setProperty('--transcription-width', '100%');
document.documentElement.style.setProperty('--transcription-height', '100%');
}
minimize() {
hide(transcriptionMinBtn);
show(transcriptionMaxBtn);
if (this.isPinned) {
this.pinned();
} else {
this.center();
document.documentElement.style.setProperty('--transcription-width', '420px');
document.documentElement.style.setProperty('--transcription-height', '680px');
}
}
center() {
transcriptionRoom.style.position = 'fixed';
transcriptionRoom.style.transform = 'translate(-50%, -50%)';
transcriptionRoom.style.top = '50%';
transcriptionRoom.style.left = '50%';
}
togglePinUnpin() {
if (rc.isChatPinned) {
return userLog('info', 'Please unpin the chat that appears to be currently pinned', 'top-end');
}
this.isPinned ? this.unpinned() : this.pinned();
rc.sound('click');
}
isPin() {
return this.isPinned;
}
pinned() {
if (!rc.isVideoPinned) {
rc.videoMediaContainer.style.top = 0;
rc.videoMediaContainer.style.width = '75%';
rc.videoMediaContainer.style.height = '100%';
}
this.pin();
this.isPinned = true;
setColor(transcriptionTogglePinBtn, 'lime');
resizeVideoMedia();
transcriptionRoom.style.resize = 'none';
if (!rc.isMobileDevice) rc.makeUnDraggable(transcriptionRoom, transcriptionHeader);
}
pin() {
transcriptionRoom.style.position = 'absolute';
transcriptionRoom.style.top = 0;
transcriptionRoom.style.right = 0;
transcriptionRoom.style.left = null;
transcriptionRoom.style.transform = null;
document.documentElement.style.setProperty('--transcription-width', '25%');
document.documentElement.style.setProperty('--transcription-height', '100%');
}
unpinned() {
if (!rc.isVideoPinned) {
rc.videoMediaContainer.style.top = 0;
rc.videoMediaContainer.style.right = null;
rc.videoMediaContainer.style.width = '100%';
rc.videoMediaContainer.style.height = '100%';
}
document.documentElement.style.setProperty('--transcription-width', '420px');
document.documentElement.style.setProperty('--transcription-height', '680px');
hide(transcriptionMinBtn);
show(transcriptionMaxBtn);
this.center();
this.isPinned = false;
setColor(transcriptionTogglePinBtn, 'white');
resizeVideoMedia();
transcriptionRoom.style.resize = 'both';
if (!rc.isMobileDevice) rc.makeDraggable(transcriptionRoom, transcriptionHeader);
}
save() {
if (this.transcripts.length != 0) {
const a = document.createElement('a');
a.href = 'data:text/json;charset=utf-8,' + encodeURIComponent(JSON.stringify(this.transcripts, null, 1));
a.download = getDataTimeString() + room_id + '-TRANSCRIPTIONS.txt';
document.body.appendChild(a);
a.click();
document.body.removeChild(a);
rc.sound('download');
} else {
userLog('info', "There isn't transcriptions to save", 'top-end');
}
}
delete() {
if (this.transcripts.length != 0) {
Swal.fire({
background: swalBackground,
position: 'center',
title: 'Clean up all transcripts?',
imageUrl: image.delete,
showDenyButton: true,
confirmButtonText: `Yes`,
denyButtonText: `No`,
showClass: { popup: 'animate__animated animate__fadeInDown' },
hideClass: { popup: 'animate__animated animate__fadeOutUp' },
}).then((result) => {
if (result.isConfirmed) {
let captions = transcriptionChat.firstChild;
while (captions) {
transcriptionChat.removeChild(captions);
captions = transcriptionChat.firstChild;
}
this.transcripts = [];
rc.sound('delete');
}
});
} else {
userLog('info', "There isn't transcriptions to delete", 'top-end');
}
}
updateCountry() {
for (let i = transcriptionDialect.options.length - 1; i >= 0; i--) {
transcriptionDialect.remove(i);
}
let list = this.languages[transcriptionLanguage.selectedIndex];
for (let i = 1; i < list.length; i++) {
transcriptionDialect.options.add(new Option(list[i][1], list[i][0]));
}
transcriptionDialect.style.visibility = list[1].length == 1 ? 'hidden' : 'visible';
}
handleLanguages() {
for (let i = 0; i < this.languages.length; i++) {
transcriptionLanguage.options[i] = new Option(this.languages[i][0], i);
}
transcriptionLanguage.selectedIndex = 6;
this.updateCountry();
transcriptionDialect.selectedIndex = 6;
transcriptionLanguage.onchange = () => {
this.updateCountry();
};
}
start() {
try {
this.transcriptionRunning = true;
this.transcription.lang = transcriptionDialect.value;
this.selectEnabled(false);
this.transcription.start();
} catch (error) {
this.transcriptionRunning = false;
userLog('error', `Transcription start error ${error.message}`, 'top-end', 6000);
console.error('Transcription start error', error);
}
}
stop() {
this.transcriptionRunning = false;
this.selectEnabled();
this.transcription.stop();
}
selectEnabled(enabled = true) {
transcriptionLanguage.disabled = enabled;
transcriptionDialect.disabled = enabled;
}
}

ثنائية
public/sounds/transcript.wav Normal file

ملف ثنائي غير معروض.

عرض الملف

@@ -80,6 +80,7 @@
<script defer src="../js/Room.js"></script> <script defer src="../js/Room.js"></script>
<script defer src="../js/RoomClient.js"></script> <script defer src="../js/RoomClient.js"></script>
<script defer src="../js/SpeechRec.js"></script> <script defer src="../js/SpeechRec.js"></script>
<script defer src="../js/Transcription.js"></script>
<script defer src="../js/VideoGrid.js"></script> <script defer src="../js/VideoGrid.js"></script>
<script defer src="https://cdn.jsdelivr.net/npm/marked/marked.min.js"></script> <script defer src="https://cdn.jsdelivr.net/npm/marked/marked.min.js"></script>
<script defer src="https://cdn.jsdelivr.net/npm/detectrtc@1.4.1/DetectRTC.min.js"></script> <script defer src="https://cdn.jsdelivr.net/npm/detectrtc@1.4.1/DetectRTC.min.js"></script>
@@ -148,6 +149,7 @@ access to use this app.
<button id="lowerHandButton" class="hidden"><i id="lowerHandIcon" class="fas fa-hand-paper"></i></button> <button id="lowerHandButton" class="hidden"><i id="lowerHandIcon" class="fas fa-hand-paper"></i></button>
<button id="roomEmojiPicker" class="hidden"><i class="fas fa-face-smile"></i></button> <button id="roomEmojiPicker" class="hidden"><i class="fas fa-face-smile"></i></button>
<button id="chatButton" class="hidden"><i class="fas fa-comments"></i></button> <button id="chatButton" class="hidden"><i class="fas fa-comments"></i></button>
<button id="transcriptionButton" class="hidden"><i class="fas fa-closed-captioning"></i></button>
<button id="participantsButton" class="hidden"><i class="fas fa-users"></i></button> <button id="participantsButton" class="hidden"><i class="fas fa-users"></i></button>
<button id="whiteboardButton" class="hidden"><i class="fas fa-chalkboard-teacher"></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> <button id="settingsButton" class="hidden"><i class="fas fa-cogs"></i></button>
@@ -749,6 +751,32 @@ access to use this app.
</section> </section>
</section> </section>
<section id="transcriptionRoom" class="transcription-room fadein">
<section id="transcriptionSection" class="transcription">
<header id="transcriptionHeader" class="transcription-header">
<div class="transcription-header-title">
<button id="transcriptionCloseBtn" class="fas fa-times"></button>
<button id="transcriptionTogglePinBtn" class="fas fa-map-pin"></button>
<button id="transcriptionMaxBtn" class=""><i class="fas fa-expand"></i></button>
<button id="transcriptionMinBtn" class="hidden"><i class="fas fa-compress"></i></button>
<button id="transcriptionSpeechStatus" class="fas fa-microphone-alt"></button>
</div>
<div class="transcription-header-options">
<button id="transcriptionGhostBtn" class="fas fa-circle-half-stroke"></button>
<button id="transcriptionSaveBtn" class="fas fa-save"></button>
<button id="transcriptionCleanBtn" class="fas fa-trash"></button>
</div>
</header>
<main id="transcriptionChat" class="transcription-chat"></main>
<div class="transcription-inputarea">
<select id="transcriptionLanguage" class="form-select text-light bg-dark"></select>
<select id="transcriptionDialect" class="form-select text-light bg-dark"></select>
<button id="transcriptionSpeechStart" class=""><i class="fas fa-play"></i></button>
<button id="transcriptionSpeechStop" class="hidden"><i class="fas fa-stop"></i></button>
</div>
</section>
</section>
<section id="lobby" class="fadein center hidden"> <section id="lobby" class="fadein center hidden">
<header id="lobbyHeader" class="lobby-header"> <header id="lobbyHeader" class="lobby-header">
<div id="lobbyHeaderTitle" class="lobby-header-title">Lobby users</div> <div id="lobbyHeaderTitle" class="lobby-header-title">Lobby users</div>