2024-04-27 08:57:01 +02:00
|
|
|
(async () => {
|
|
|
|
// check notification permission
|
|
|
|
// This is useful to alert people that they should do something
|
|
|
|
await Notification.requestPermission()
|
|
|
|
})()
|
|
|
|
|
2024-04-27 12:08:10 +02:00
|
|
|
let channels = {}
|
2024-04-27 13:27:27 +02:00
|
|
|
let messages = {}
|
2024-04-27 12:08:10 +02:00
|
|
|
let selected_channel_id = null
|
|
|
|
|
2024-04-27 08:57:01 +02:00
|
|
|
/**
|
|
|
|
* Display a new notification with the given title and the given body.
|
|
|
|
* @param title The title of the notification
|
|
|
|
* @param body The body of the notification
|
|
|
|
* @param timeout The time (in milliseconds) after that the notification automatically closes. 0 to make indefinite. Default to 5000 ms.
|
|
|
|
* @return Notification
|
|
|
|
*/
|
|
|
|
function showNotification(title, body, timeout = 5000) {
|
|
|
|
let notif = new Notification(title, {'body': body, 'icon': "/static/tfjm.svg"})
|
|
|
|
if (timeout)
|
|
|
|
setTimeout(() => notif.close(), timeout)
|
|
|
|
return notif
|
|
|
|
}
|
|
|
|
|
2024-04-27 12:08:10 +02:00
|
|
|
function selectChannel(channel_id) {
|
|
|
|
let channel = channels[channel_id]
|
|
|
|
if (!channel) {
|
|
|
|
console.error('Channel not found:', channel_id)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
selected_channel_id = channel_id
|
|
|
|
|
|
|
|
let channelTitle = document.getElementById('channel-title')
|
|
|
|
channelTitle.innerText = channel['name']
|
|
|
|
|
|
|
|
let messageInput = document.getElementById('input-message')
|
|
|
|
messageInput.disabled = !channel['write_access']
|
2024-04-27 13:27:27 +02:00
|
|
|
|
|
|
|
redrawMessages()
|
2024-04-27 12:08:10 +02:00
|
|
|
}
|
|
|
|
|
2024-04-27 12:59:50 +02:00
|
|
|
function sendMessage() {
|
|
|
|
let messageInput = document.getElementById('input-message')
|
|
|
|
let message = messageInput.value
|
|
|
|
messageInput.value = ''
|
|
|
|
|
|
|
|
if (!message) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
socket.send(JSON.stringify({
|
|
|
|
'type': 'send_message',
|
|
|
|
'channel_id': selected_channel_id,
|
|
|
|
'content': message,
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
2024-04-27 12:08:10 +02:00
|
|
|
function setChannels(new_channels) {
|
|
|
|
channels = {}
|
|
|
|
for (let channel of new_channels) {
|
|
|
|
channels[channel['id']] = channel
|
2024-04-27 13:27:27 +02:00
|
|
|
if (!messages[channel['id']])
|
|
|
|
messages[channel['id']] = []
|
|
|
|
|
|
|
|
socket.send(JSON.stringify({
|
|
|
|
'type': 'fetch_messages',
|
|
|
|
'channel_id': channel['id'],
|
|
|
|
}))
|
2024-04-27 12:08:10 +02:00
|
|
|
}
|
|
|
|
|
2024-04-27 13:27:27 +02:00
|
|
|
if (new_channels && (!selected_channel_id || !channels[selected_channel_id]))
|
2024-04-27 12:08:10 +02:00
|
|
|
selectChannel(Object.keys(channels)[0])
|
|
|
|
}
|
|
|
|
|
2024-04-27 12:59:50 +02:00
|
|
|
function receiveMessage(message) {
|
2024-04-27 13:27:27 +02:00
|
|
|
messages[message['channel_id']].push(message)
|
|
|
|
redrawMessages()
|
|
|
|
}
|
|
|
|
|
|
|
|
function fetchMessages(data) {
|
|
|
|
let channel_id = data['channel_id']
|
|
|
|
let new_messages = data['messages']
|
2024-04-27 12:59:50 +02:00
|
|
|
|
2024-04-27 13:27:27 +02:00
|
|
|
if (!messages[channel_id])
|
|
|
|
messages[channel_id] = []
|
2024-04-27 12:59:50 +02:00
|
|
|
|
2024-04-27 13:27:27 +02:00
|
|
|
for (let message of new_messages) {
|
|
|
|
messages[channel_id].push(message)
|
|
|
|
}
|
|
|
|
|
|
|
|
redrawMessages()
|
|
|
|
}
|
2024-04-27 12:59:50 +02:00
|
|
|
|
2024-04-27 13:27:27 +02:00
|
|
|
function redrawMessages() {
|
|
|
|
let messageList = document.getElementById('message-list')
|
|
|
|
messageList.innerHTML = ''
|
|
|
|
|
|
|
|
let lastMessage = null
|
|
|
|
let lastContentDiv = null
|
|
|
|
|
|
|
|
for (let message of messages[selected_channel_id]) {
|
|
|
|
if (lastMessage && lastMessage['author'] === message['author']) {
|
|
|
|
let lastTimestamp = new Date(lastMessage['timestamp'])
|
|
|
|
let newTimestamp = new Date(message['timestamp'])
|
|
|
|
if ((newTimestamp - lastTimestamp) / 1000 < 60 * 10) {
|
|
|
|
let messageContentDiv = document.createElement('div')
|
|
|
|
messageContentDiv.innerText = message['content']
|
|
|
|
lastContentDiv.appendChild(messageContentDiv)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let messageElement = document.createElement('li')
|
|
|
|
messageElement.classList.add('list-group-item')
|
|
|
|
messageList.appendChild(messageElement)
|
|
|
|
|
|
|
|
let authorDiv = document.createElement('div')
|
|
|
|
messageElement.appendChild(authorDiv)
|
|
|
|
|
|
|
|
let authorSpan = document.createElement('span')
|
|
|
|
authorSpan.classList.add('text-muted', 'fw-bold')
|
|
|
|
authorSpan.innerText = message['author']
|
|
|
|
authorDiv.appendChild(authorSpan)
|
|
|
|
|
|
|
|
let dateSpan = document.createElement('span')
|
|
|
|
dateSpan.classList.add('text-muted', 'float-end')
|
|
|
|
dateSpan.innerText = new Date(message['timestamp']).toLocaleString()
|
|
|
|
authorDiv.appendChild(dateSpan)
|
|
|
|
|
|
|
|
let contentDiv = document.createElement('div')
|
|
|
|
messageElement.appendChild(contentDiv)
|
|
|
|
|
|
|
|
let messageContentDiv = document.createElement('div')
|
|
|
|
messageContentDiv.innerText = message['content']
|
|
|
|
contentDiv.appendChild(messageContentDiv)
|
|
|
|
|
|
|
|
lastMessage = message
|
|
|
|
lastContentDiv = contentDiv
|
|
|
|
}
|
2024-04-27 12:59:50 +02:00
|
|
|
}
|
|
|
|
|
2024-04-27 08:57:01 +02:00
|
|
|
document.addEventListener('DOMContentLoaded', () => {
|
|
|
|
/**
|
|
|
|
* Process the received data from the server.
|
|
|
|
* @param data The received message
|
|
|
|
*/
|
|
|
|
function processMessage(data) {
|
2024-04-27 12:08:10 +02:00
|
|
|
switch (data['type']) {
|
|
|
|
case 'fetch_channels':
|
|
|
|
setChannels(data['channels'])
|
|
|
|
break
|
2024-04-27 12:59:50 +02:00
|
|
|
case 'send_message':
|
|
|
|
receiveMessage(data)
|
|
|
|
break
|
2024-04-27 13:27:27 +02:00
|
|
|
case 'fetch_messages':
|
|
|
|
fetchMessages(data)
|
|
|
|
break
|
2024-04-27 12:08:10 +02:00
|
|
|
default:
|
|
|
|
console.log(data)
|
|
|
|
console.error('Unknown message type:', data['type'])
|
|
|
|
break
|
|
|
|
}
|
2024-04-27 08:57:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function setupSocket(nextDelay = 1000) {
|
|
|
|
// Open a global websocket
|
|
|
|
socket = new WebSocket(
|
|
|
|
(document.location.protocol === 'https:' ? 'wss' : 'ws') + '://' + window.location.host + '/ws/chat/'
|
|
|
|
)
|
|
|
|
|
|
|
|
// Listen on websockets and process messages from the server
|
|
|
|
socket.addEventListener('message', e => {
|
|
|
|
// Parse received data as JSON
|
|
|
|
const data = JSON.parse(e.data)
|
|
|
|
|
|
|
|
processMessage(data)
|
|
|
|
})
|
|
|
|
|
|
|
|
// Manage errors
|
|
|
|
socket.addEventListener('close', e => {
|
|
|
|
console.error('Chat socket closed unexpectedly, restarting…')
|
|
|
|
setTimeout(() => setupSocket(2 * nextDelay), nextDelay)
|
|
|
|
})
|
2024-04-27 09:53:55 +02:00
|
|
|
|
|
|
|
socket.addEventListener('open', e => {
|
|
|
|
socket.send(JSON.stringify({
|
|
|
|
'type': 'fetch_channels',
|
|
|
|
}))
|
|
|
|
})
|
2024-04-27 08:57:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
setupSocket()
|
|
|
|
})
|