fixed scrolling and buffer times- oh yeah also huge backend overhual

This commit is contained in:
ari melody 2023-09-30 15:52:54 +01:00
parent 1eeb632ff2
commit 524969370d
Signed by: ari
GPG key ID: CF99829C92678188
3 changed files with 250 additions and 99 deletions

View file

@ -1,11 +1,22 @@
var buffer = "";
var send_buffer = "";
var recv_buffer = [];
var send_buffer = [];
var content;
var mobile_input;
var client;
var my_colour = false;
var pre_buffer_chars = 0;
var term_interval = 10;
var ready = false;
const DATA_TYPES = {
text: 0,
colour: 1,
buffer: 2,
backspace: 3,
backword: 4,
arrow: 5,
};
function start() {
console.log("%chello, world!", "color: #b7fd49; font-size: 3rem; font-weight: bold");
@ -34,74 +45,115 @@ to help you feel a little more comfortable, i've prepared some commands for you:
content = document.getElementById("content");
mobile_input = document.getElementById("mobile-input");
content.addEventListener("click", () => {
content.addEventListener("touchend", () => {
mobile_input.focus();
});
buffer += "Connecting to the server...";
add_system_message("Connecting to the server...");
setTimeout(connect, 500);
setInterval(() => {
if (send_buffer.length > 0) {
const data = JSON.stringify(send_buffer[0]);
client.send(data);
send_buffer = send_buffer.slice(1);
}
}, 1000 / 600);
loop();
}
function loop() {
if (buffer.length > 0) {
const char = buffer.substring(0, 1);
insert_text(char);
buffer = buffer.substring(1);
}
if (send_buffer.length > 0) {
const char = send_buffer.substring(0, 1);
client.send(char);
send_buffer = send_buffer.substring(1);
}
mobile_input.value = content.innerText;
setTimeout(loop, term_interval);
setTimeout(loop, 1000 / 60);
}
function connect() {
client = new WebSocket("wss://" + window.location.host);
client.addEventListener('open', () => {
// insert_text('\x00');
buffer += "\nConnection successful.\n\n";
buffer += "=== BEGIN SESSION ===\n\n";
add_system_message(`\nConnection successful.\n\n`);
add_system_message(`=== BEGIN SESSION ===\n\n`);
new_caret();
});
client.addEventListener('message', event => {
buffer += event.data;
if (pre_buffer_chars == 0) {
pre_buffer_chars = content.innerText.length + buffer.length;
}
});
client.addEventListener('message', event => { handle_message(JSON.parse(event.data)) });
client.addEventListener('close', () => {
insert_text("\n\n[CONNECTION LOST, PLEASE REFRESH]");
add_system_message(`\n[CONNECTION LOST, PLEASE REFRESH]\n`);
});
}
function insert_text(text) {
const carat = content.querySelector("#carat");
if (carat) carat.remove();
function add_system_message(text) {
const span = document.createElement("span");
span.classList.add('sticky');
span.innerText = text;
content.appendChild(span);
if (text == "\x00") {
content.innerText = "";
pre_buffer_chars = 0;
} else if (text == "\b") {
if (content.innerText.length > pre_buffer_chars) {
new_caret();
}
function handle_message(data) {
if (!data.type && data.type != 0) return;
const is_at_bottom = content.scrollTop == content.scrollTopMax;
switch (data.type) {
case DATA_TYPES.colour:
my_colour = data.colour;
console.log(`%cColour has been changed to ${my_colour}`, `color: ${my_colour}`);
break;
case DATA_TYPES.backspace:
content.querySelectorAll("#caret").forEach(caret => caret.remove());
/*
const last_child = content.lastChild;
if (last_child.classList.contains('sticky')) break;
last_child.remove();
*/
if (content.innerText.length <= pre_buffer_chars) {
break;
}
content.innerText = content.innerText.slice(0, content.innerText.length - 1);
}
} else {
content.innerText += text;
break;
case DATA_TYPES.text:
/*
const span = document.createElement("span");
if (data.colour) span.style.color = data.colour;
if (data.sticky) span.classList.add('sticky');
span.innerText = data.text;
content.appendChild(span);
*/
content.innerText += data.text;
break;
case DATA_TYPES.buffer:
content.innerText += data.data;
break;
/*
data.data.forEach(block => {
handle_message(block);
});
*/
}
const new_carat = document.createElement("div");
new_carat.id = "carat";
content.appendChild(new_carat);
if (pre_buffer_chars == 0) {
pre_buffer_chars = content.innerText.length;
}
new_caret();
if (is_at_bottom) content.scrollTop = content.scrollTopMax;
}
function new_caret() {
content.querySelectorAll("#caret").forEach(caret => caret.remove());
const new_caret = document.createElement("div");
new_caret.id = "caret";
if (my_colour) {
new_caret.style.backgroundColor = my_colour;
}
content.appendChild(new_caret);
}
function handle_input(event) {
@ -109,38 +161,77 @@ function handle_input(event) {
event.preventDefault();
}
if (event.key == "Backspace") {
if (event.ctrlKey && send_buffer.length == 0) {
const last_space = content.innerText.lastIndexOf(" ");
const last_newline = content.innerText.lastIndexOf("\n");
var break_at = last_space;
if (last_newline > last_space) {
break_at = last_newline;
}
const word_length = content.innerText.length - break_at;
for (let i = 0; i < word_length; i++) {
send_buffer += '\b';
switch (event.key) {
case "Backspace":
if (event.ctrlKey) {
if (send_buffer.length > 0) return;
/*
send_buffer.push({
type: DATA_TYPES.backword,
});
*/
var break_point = content.innerText.lastIndexOf(" ");
const last_newline = content.innerText.lastIndexOf("\n");
if (last_newline > break_point) break_point = last_newline;
const count = content.innerText.length - break_point;
for (var i = 0; i < count; i++) {
send_buffer.push({
type: DATA_TYPES.backspace,
});
}
return;
}
send_buffer.push({
type: DATA_TYPES.backspace,
});
return;
case "Enter":
send_buffer.push({
type: DATA_TYPES.text,
text: "\n",
});
return;
case "ArrowUp":
send_buffer.push({
type: DATA_TYPES.arrow,
dir: "up",
});
return;
case "ArrowDown":
send_buffer.push({
type: DATA_TYPES.arrow,
dir: "down",
});
return;
case "ArrowLeft":
send_buffer.push({
type: DATA_TYPES.arrow,
dir: "left",
});
return;
case "ArrowRight":
send_buffer.push({
type: DATA_TYPES.arrow,
dir: "right",
});
return;
}
send_buffer += '\b';
return;
}
if (event.key == "Enter") {
send_buffer += '\n';
return;
}
if (event.key.length > 1) {
// server will discard text over 1 character, anyway
return;
}
if (event.ctrlKey) {
return;
}
send_buffer += event.key;
content.scrollTop = content.scrollHeight;
send_buffer.push({
type: DATA_TYPES.text,
text: event.key,
});
content.scrollTop = content.scrollTopMax;
}
function handle_paste(event) {
@ -151,8 +242,11 @@ function handle_paste(event) {
}
const paste = (event.clipboardData || window.clipboardData).getData("text");
send_buffer += paste;
content.scrollTop = content.scrollHeight;
send_buffer.push({
type: DATA_TYPES.text,
text: paste,
});
content.scrollTop = content.scrollTopMax;
}
const PALETTE = {

View file

@ -28,16 +28,16 @@ pre#content {
text-shadow: 0 0 1em, 0 0 3em;
}
div#carat {
div#caret {
width: .5em;
height: .9em;
display: inline-block;
background: var(--colour);
transform: translateY(1px);
animation: linear .5s infinite forwards carat-blink;
animation: linear .5s infinite forwards caret-blink;
}
@keyframes carat-blink {
@keyframes caret-blink {
from {
opacity: 1;
}

View file

@ -20,7 +20,16 @@ const MIME_TYPES = {
svg: "image/svg+xml",
};
const motds = [
const DATA_TYPES = {
text: 0,
colour: 1,
buffer: 2,
backspace: 3,
backword: 4,
arrow: 5,
};
const MOTDS = [
"hello, world!",
"all your TTY are belong to us.",
"TIP: got a linux system low on storage? try running `sudo rm -rf /`!",
@ -46,7 +55,7 @@ let sockets = [];
let buffer = "";
const MAX_BUFFER_SIZE = 10240;
const MAX_MESSAGE_LENGTH = 64;
const MAX_MESSAGE_LENGTH = 1024;
async function get_file(url) {
const paths = [STATIC_PATH, url];
@ -76,14 +85,31 @@ const server = https.createServer(config, async (req, res) => {
const wss = new Websocket.Server({ server });
wss.on('connection', socket => {
socket.send(`${banner}/* ${motds[Math.floor(Math.random() * motds.length)]} */\n\n`);
socket.send(buffer);
/*
socket.colour = generate_colour();
socket.send(JSON.stringify({
type: DATA_TYPES.colour,
colour: socket.colour,
}));
*/
socket.send(JSON.stringify({
type: DATA_TYPES.text,
text: `${banner}/* ${MOTDS[Math.floor(Math.random() * MOTDS.length)]} */\n\n`,
colour: false,
sticky: true,
}));
if (buffer) {
socket.send(JSON.stringify({
type: DATA_TYPES.buffer,
data: buffer,
}));
}
sockets.push(socket);
// console.log(`new connection.\n\tcurrent connections: ${sockets.length}`);
socket.on('message', handle_message);
socket.on('message', event => { handle_message(JSON.parse(event), socket) });
socket.on('close', () => {
sockets = sockets.filter(s => s !== socket);
@ -91,38 +117,69 @@ wss.on('connection', socket => {
});
});
function handle_message(msg) {
if (msg.length > MAX_MESSAGE_LENGTH) {
return;
}
if (msg == '\b') {
buffer = buffer.slice(0, buffer.length - 1);
send_text('\b');
return;
} else if (buffer.length >= MAX_BUFFER_SIZE) {
return;
}
if (msg == '\n') {
buffer += '\n';
send_text('\n');
return;
function handle_message(data, user) {
switch (data.type) {
case DATA_TYPES.backword:
var break_point = buffer.lastIndexOf(" ");
const last_newline = buffer.lastIndexOf("\n");
if (last_newline > break_point) break_point = last_newline;
buffer = buffer.substring(0, break_point);
for (var i = 0; i < buffer.length - break_point; i++) {
broadcast(JSON.stringify({
type: DATA_TYPES.backspace,
}));
}
case DATA_TYPES.backspace:
buffer = buffer.substring(0, buffer.length - 1);
broadcast(JSON.stringify({
type: DATA_TYPES.backspace,
}));
return;
case DATA_TYPES.text:
if (buffer.length >= MAX_BUFFER_SIZE) {
return;
}
if (data.text.length > MAX_MESSAGE_LENGTH) {
return;
}
block = {
type: DATA_TYPES.text,
text: data.text,
colour: user.colour,
};
buffer += data.text;
broadcast(JSON.stringify(block));
}
buffer += msg.toString();
send_text(msg.toString());
/*
if (buffer.length > MAX_BUFFER_SIZE) {
buffer = buffer.slice(buffer.length - MAX_BUFFER_SIZE, buffer.length);
send_as_server(`\n\nSERVER: This channel's maximum buffer length has been hit (${MAX_BUFFER_SIZE}).\n` +
`You will need to make more room, or the server will have to be restarted.\n` +
`Apologies for the inconvenience!`)
}
*/
}
function generate_colour() {
let result = '#';
let hexref = '0123456789abcdef';
for (let i = 0; i < 6; i++) {
result += hexref.charAt(Math.floor(Math.random() * hexref.length * .75) + 4);
}
return result;
}
server.listen(PORT, () => {
console.log(`OpenTerminal is now LIVE on https://127.0.0.1:${PORT}!`);
});
function send_text(text) {
sockets.forEach(s => s.send(text));
function send_as_server(message) {
broadcast(JSON.stringify({
type: DATA_TYPES.text,
text: message,
colour: "#ffffff",
}));
}
function broadcast(data) {
sockets.forEach(s => s.send(data));
}