2024-07-10 00:40:49 +02:00
|
|
|
const startTime = Date.now();
|
2024-07-08 01:00:09 +02:00
|
|
|
|
2024-07-14 16:57:05 +02:00
|
|
|
var conversation = null;
|
|
|
|
|
|
|
|
class Conversation {
|
2024-10-04 01:04:50 +02:00
|
|
|
constructor(name) {
|
2024-07-14 16:57:05 +02:00
|
|
|
this.messages = [];
|
2024-10-04 01:04:50 +02:00
|
|
|
this.name = name;
|
2024-07-14 16:57:05 +02:00
|
|
|
}
|
|
|
|
|
2024-10-06 19:30:58 +02:00
|
|
|
setInteractive(isInteractive) {
|
|
|
|
const children = document.getElementById("textbox").children;
|
|
|
|
for (let i = 0; i < children.length; i++) {
|
|
|
|
children[i].disabled = !isInteractive;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-04 01:04:50 +02:00
|
|
|
initialize(initialMessages) {
|
|
|
|
document.title = this.name;
|
|
|
|
document.getElementById("header-title").innerHTML = this.name;
|
2024-07-14 16:57:05 +02:00
|
|
|
|
2024-10-04 01:04:50 +02:00
|
|
|
this.messages = initialMessages;
|
2024-07-14 16:57:05 +02:00
|
|
|
}
|
|
|
|
|
2024-10-04 01:04:50 +02:00
|
|
|
// for the user to send their own messages
|
|
|
|
sendUserMessage(text) {
|
|
|
|
const message = new UserMessage(text);
|
|
|
|
message.updateStatus("sent");
|
2024-07-14 16:57:05 +02:00
|
|
|
setTimeout(() => {
|
2024-10-04 01:04:50 +02:00
|
|
|
message.updateStatus("delivered");
|
|
|
|
this.render();
|
|
|
|
}, 1000);
|
|
|
|
this.messages.push(message);
|
|
|
|
}
|
|
|
|
|
|
|
|
// update the current HTML based on messages
|
|
|
|
render() {
|
|
|
|
// clear stale HTML
|
|
|
|
getMessageList().innerHTML = "";
|
|
|
|
|
|
|
|
// render message elements
|
|
|
|
for (let i = 0; i < this.messages.length; i++)
|
|
|
|
{
|
|
|
|
const messageRoot = document.getElementById("messages");
|
|
|
|
const newMessage = this.messages[i];
|
|
|
|
messageRoot.appendChild(newMessage.getElement());
|
2024-07-14 16:57:05 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-09 02:55:13 +02:00
|
|
|
function getMessageList() {
|
|
|
|
return document.getElementById("messages");
|
|
|
|
}
|
|
|
|
|
2024-10-04 01:04:50 +02:00
|
|
|
class AgentMessage {
|
2024-10-06 17:27:02 +02:00
|
|
|
constructor(text, senderName) {
|
2024-10-04 01:04:50 +02:00
|
|
|
this.text = text;
|
2024-10-06 17:27:02 +02:00
|
|
|
this.senderName = senderName;
|
2024-07-14 16:57:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
getIsOurs() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-10-04 01:04:50 +02:00
|
|
|
getElement() {
|
|
|
|
const liElem = document.createElement("li");
|
2024-10-06 19:30:58 +02:00
|
|
|
liElem.className = "message";
|
2024-07-14 16:57:05 +02:00
|
|
|
|
2024-10-04 01:04:50 +02:00
|
|
|
const contentElem = document.createElement("span");
|
|
|
|
contentElem.className = "message-content rounded-rectangle theirs";
|
|
|
|
liElem.appendChild(contentElem);
|
2024-07-09 02:55:13 +02:00
|
|
|
|
2024-10-06 17:27:02 +02:00
|
|
|
if (this.senderName) {
|
|
|
|
const nameElem = document.createElement("h3");
|
|
|
|
nameElem.innerHTML = this.senderName;
|
|
|
|
contentElem.appendChild(nameElem);
|
|
|
|
}
|
|
|
|
|
2024-10-04 01:04:50 +02:00
|
|
|
const textElem = document.createElement("span");
|
|
|
|
textElem.className = "message-text";
|
|
|
|
textElem.innerHTML = this.text;
|
|
|
|
contentElem.appendChild(textElem);
|
2024-07-09 02:55:13 +02:00
|
|
|
|
2024-10-04 01:04:50 +02:00
|
|
|
return liElem;
|
|
|
|
}
|
|
|
|
}
|
2024-07-09 02:55:13 +02:00
|
|
|
|
2024-10-04 01:04:50 +02:00
|
|
|
class UserMessage {
|
|
|
|
constructor(text) {
|
|
|
|
this.createdTime = Date.now();
|
|
|
|
this.text = text;
|
|
|
|
this.status = "";
|
2024-07-09 02:55:13 +02:00
|
|
|
}
|
|
|
|
|
2024-07-14 16:57:05 +02:00
|
|
|
getIsOurs() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-10-04 01:04:50 +02:00
|
|
|
getElement() {
|
|
|
|
const liElem = document.createElement("li");
|
2024-10-06 19:30:58 +02:00
|
|
|
liElem.className = "message";
|
2024-07-14 16:57:05 +02:00
|
|
|
|
2024-10-04 01:04:50 +02:00
|
|
|
const contentElem = document.createElement("span");
|
|
|
|
contentElem.className = "message-content rounded-rectangle ours";
|
|
|
|
liElem.appendChild(contentElem);
|
2024-07-09 02:55:13 +02:00
|
|
|
|
2024-10-04 01:04:50 +02:00
|
|
|
const textElem = document.createElement("span");
|
|
|
|
textElem.className = "message-text";
|
|
|
|
textElem.innerHTML = this.text;
|
|
|
|
contentElem.appendChild(textElem);
|
|
|
|
|
|
|
|
const statusElem = document.createElement("p");
|
|
|
|
statusElem.className = "message-status";
|
|
|
|
statusElem.innerHTML = this.status;
|
|
|
|
liElem.appendChild(statusElem);
|
|
|
|
|
|
|
|
return liElem;
|
2024-07-09 02:55:13 +02:00
|
|
|
}
|
|
|
|
|
2024-10-04 01:04:50 +02:00
|
|
|
updateStatus(newStatus) {
|
|
|
|
this.status = newStatus;
|
2024-07-09 02:55:13 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-06 19:30:58 +02:00
|
|
|
class SystemMessage {
|
|
|
|
constructor(text) {
|
|
|
|
this.text = text;
|
|
|
|
}
|
|
|
|
|
|
|
|
getElement() {
|
|
|
|
const liElem = document.createElement("li");
|
|
|
|
liElem.className = "system-message";
|
|
|
|
liElem.innerHTML = this.text;
|
|
|
|
|
|
|
|
return liElem;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-09 22:51:28 +02:00
|
|
|
function setTypingIndicator(isTyping) {
|
|
|
|
document.getElementById("typing-indicator").innerHTML = isTyping
|
2024-10-02 02:13:37 +02:00
|
|
|
? `${conversation.contactName} is typing...`
|
2024-07-09 22:51:28 +02:00
|
|
|
: "";
|
|
|
|
}
|
2024-07-08 01:00:09 +02:00
|
|
|
|
2024-07-09 22:51:28 +02:00
|
|
|
// add the message at the index to the displayed messages
|
2024-07-14 16:57:05 +02:00
|
|
|
function addMessage(message) {
|
|
|
|
getMessageList().innerHTML += message.getHtml();
|
2024-07-10 15:16:33 +02:00
|
|
|
|
|
|
|
// scroll as far as we can so that messages aren't hidden
|
|
|
|
window.scrollTo(0, document.body.scrollHeight);
|
2024-07-08 01:00:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function updatePings() {
|
2024-10-04 01:04:50 +02:00
|
|
|
const title = conversation.name;
|
2024-07-14 16:57:05 +02:00
|
|
|
let newTitle = conversation.pings > 0
|
|
|
|
? `(${conversation.pings}) ${title}`
|
2024-07-08 01:00:09 +02:00
|
|
|
: title;
|
|
|
|
|
|
|
|
document.title = newTitle;
|
|
|
|
}
|
|
|
|
|
|
|
|
function clearPings() {
|
2024-07-14 16:57:05 +02:00
|
|
|
conversation.pings = 0;
|
2024-07-08 01:00:09 +02:00
|
|
|
updatePings();
|
|
|
|
}
|
|
|
|
|
2024-10-04 01:04:50 +02:00
|
|
|
// returns a decimal value between min and max
|
|
|
|
function getRandomInRange(min, max) {
|
2024-07-09 22:51:28 +02:00
|
|
|
const range = max - min;
|
|
|
|
return min + Math.random() * range;
|
|
|
|
}
|
|
|
|
|
2024-07-14 16:57:05 +02:00
|
|
|
function updateChat(message) {
|
|
|
|
addMessage(message);
|
2024-07-17 01:22:20 +02:00
|
|
|
const previewText = conversation.getTypedMessageText();
|
|
|
|
document.getElementById("textbox-input").value = previewText;
|
2024-07-10 00:40:49 +02:00
|
|
|
updatePings();
|
|
|
|
}
|
|
|
|
|
2024-07-09 02:55:13 +02:00
|
|
|
function pressSendButton() {
|
2024-10-04 01:04:50 +02:00
|
|
|
if (event.type == "keydown" && event.key != "Enter")
|
|
|
|
return;
|
|
|
|
|
2024-07-09 02:55:13 +02:00
|
|
|
// we have interacted with the page so remove all pings
|
|
|
|
clearPings();
|
2024-10-04 01:04:50 +02:00
|
|
|
|
|
|
|
// get the content of the text box
|
|
|
|
const textBox = document.getElementById("textbox-input");
|
|
|
|
const text = textBox.value;
|
|
|
|
if (!text)
|
|
|
|
return;
|
|
|
|
|
|
|
|
textBox.value = "";
|
|
|
|
|
|
|
|
conversation.sendUserMessage(text);
|
|
|
|
conversation.render();
|
|
|
|
|
|
|
|
// TODO: start process of receiving next message from server (or fake it for now)
|
2024-07-09 02:55:13 +02:00
|
|
|
}
|
|
|
|
|
2024-07-14 16:57:05 +02:00
|
|
|
function onMessageReceived(message) {
|
|
|
|
updateChat(message);
|
|
|
|
setTypingIndicator(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
function onMessageSent(message) {
|
|
|
|
updateChat(message);
|
|
|
|
}
|
|
|
|
|
2024-10-05 19:35:24 +02:00
|
|
|
// probably a bit hacky! but this saves having to do like, state or something in CSS?
|
|
|
|
// which probably is possible and probably would be the better way to do it, but that
|
|
|
|
// sounds like a bunch of learning i'm not SUPER in the mood for
|
|
|
|
function setVisibleOnMobile(element, isVisible) {
|
|
|
|
let classes = element.className.split().filter(c => c != "");
|
|
|
|
const invisibleClass = "invisible-on-mobile";
|
|
|
|
const visibleClass = "visible";
|
|
|
|
|
|
|
|
if (isVisible && !classes.includes(visibleClass)) {
|
|
|
|
const idx = classes.indexOf(invisibleClass);
|
|
|
|
if (idx != -1) {
|
|
|
|
classes.splice(idx, 1);
|
|
|
|
}
|
|
|
|
classes.push(visibleClass);
|
2024-10-06 17:27:02 +02:00
|
|
|
} else if (!isVisible && !classes.includes(invisibleClass)) {
|
2024-10-05 19:35:24 +02:00
|
|
|
const idx = classes.indexOf(visibleClass);
|
|
|
|
if (idx != -1) {
|
|
|
|
classes.splice(idx, 1);
|
2024-10-04 01:04:50 +02:00
|
|
|
}
|
2024-10-05 19:35:24 +02:00
|
|
|
classes.push(invisibleClass);
|
2024-10-04 01:04:50 +02:00
|
|
|
}
|
|
|
|
|
2024-10-05 19:35:24 +02:00
|
|
|
element.className = classes.join(" ");
|
|
|
|
}
|
2024-10-04 01:04:50 +02:00
|
|
|
|
2024-10-05 19:35:24 +02:00
|
|
|
function showSidePanel() {
|
|
|
|
// this function can only be called on mobile. the main conversation should be
|
|
|
|
// hidden and the side conversations panel should take up the whole screen.
|
|
|
|
|
|
|
|
const mainPanel = document.getElementById("main-panel");
|
|
|
|
const conversationListElem = document.getElementById("side-panel");
|
|
|
|
|
|
|
|
setVisibleOnMobile(mainPanel, false);
|
|
|
|
setVisibleOnMobile(conversationListElem, true);
|
2024-07-08 01:00:09 +02:00
|
|
|
}
|
|
|
|
|
2024-10-06 17:27:02 +02:00
|
|
|
function readConversationJson(path, callback) {
|
|
|
|
fetch(path)
|
|
|
|
.then(response => response.json())
|
|
|
|
.then(json => callback(json));
|
|
|
|
}
|
|
|
|
|
2024-10-05 19:35:24 +02:00
|
|
|
function showConversation(path) {
|
|
|
|
|
|
|
|
const mainPanel = document.getElementById("main-panel");
|
|
|
|
const conversationListElem = document.getElementById("side-panel");
|
|
|
|
|
|
|
|
setVisibleOnMobile(mainPanel, true);
|
|
|
|
setVisibleOnMobile(conversationListElem, false);
|
|
|
|
|
|
|
|
|
2024-10-06 17:27:02 +02:00
|
|
|
readConversationJson(path, json => {
|
2024-10-06 17:31:26 +02:00
|
|
|
conversation = new Conversation(json.title);
|
2024-10-06 17:27:02 +02:00
|
|
|
const jsonMessages = json.messages;
|
|
|
|
const participants = json.characters;
|
|
|
|
|
|
|
|
let initialMessages = [];
|
|
|
|
|
|
|
|
for (let i = 0; i < jsonMessages.length; i++) {
|
|
|
|
const data = jsonMessages[i];
|
|
|
|
const text = data.text;
|
2024-10-06 19:30:58 +02:00
|
|
|
if (data.character == -1) {
|
|
|
|
const message = new SystemMessage(text);
|
|
|
|
initialMessages.push(message);
|
|
|
|
} else if (data.character == 0) {
|
2024-10-06 17:27:02 +02:00
|
|
|
const message = new UserMessage(text);
|
|
|
|
message.updateStatus("delivered");
|
|
|
|
initialMessages.push(message);
|
|
|
|
} else {
|
|
|
|
const message = participants.length > 2
|
|
|
|
? new AgentMessage(text, participants[data.character])
|
|
|
|
: new AgentMessage(text);
|
|
|
|
initialMessages.push(message);
|
2024-10-05 19:35:24 +02:00
|
|
|
}
|
2024-10-06 17:27:02 +02:00
|
|
|
}
|
2024-10-05 19:35:24 +02:00
|
|
|
|
2024-10-06 17:27:02 +02:00
|
|
|
conversation.initialize(initialMessages);
|
2024-10-06 19:30:58 +02:00
|
|
|
conversation.setInteractive(json.interactive);
|
2024-10-06 17:27:02 +02:00
|
|
|
conversation.render();
|
|
|
|
});
|
2024-10-05 19:35:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function addConversationPreview(path) {
|
|
|
|
const listRoot = document.getElementById("side-panel");
|
|
|
|
|
2024-10-06 17:27:02 +02:00
|
|
|
readConversationJson(path, json => {
|
|
|
|
const messages = json.messages;
|
|
|
|
|
|
|
|
const elem = document.createElement("div");
|
|
|
|
elem.onclick = () => showConversation(path);
|
|
|
|
elem.className = "conversation";
|
2024-10-05 19:35:24 +02:00
|
|
|
|
2024-10-06 17:27:02 +02:00
|
|
|
const headerElem = document.createElement("h2");
|
2024-10-06 17:31:26 +02:00
|
|
|
headerElem.innerHTML = json.title;
|
2024-10-06 17:27:02 +02:00
|
|
|
elem.appendChild(headerElem);
|
2024-10-05 19:35:24 +02:00
|
|
|
|
2024-10-06 17:27:02 +02:00
|
|
|
const previewElem = document.createElement("span");
|
|
|
|
previewElem.innerHTML = messages[messages.length - 1].text;
|
|
|
|
elem.appendChild(previewElem);
|
2024-10-05 19:35:24 +02:00
|
|
|
|
2024-10-06 17:27:02 +02:00
|
|
|
listRoot.appendChild(elem);
|
|
|
|
});
|
2024-10-05 19:35:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function populateConversationList() {
|
|
|
|
const conversationFiles = [
|
|
|
|
"caesar.json",
|
|
|
|
"lucius.json",
|
2024-10-06 17:27:02 +02:00
|
|
|
"ides-of-march.json",
|
|
|
|
"lepidus.json"
|
2024-10-05 19:35:24 +02:00
|
|
|
];
|
|
|
|
|
|
|
|
for (let i = 0; i < conversationFiles.length; i++) {
|
|
|
|
const path = conversationFiles[i];
|
|
|
|
addConversationPreview(path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
setTypingIndicator(false);
|
|
|
|
populateConversationList();
|
|
|
|
|
2024-10-06 19:30:58 +02:00
|
|
|
showConversation("ides-of-march.json");
|
2024-10-04 01:04:50 +02:00
|
|
|
|