Initial Commit

pull/3/head
Drew Bednar 8 months ago
parent 6a29623470
commit 970748e974

@ -0,0 +1,8 @@
# Web Serial Console
Serial console access in the browser! This project uses and [experimental API](https://developer.mozilla.org/en-US/docs/Web/API/Web_Serial_API) available as of 08/06/2024 only on Chrome 89, Edge 89, or Opera 75.
## Building
## Use

@ -0,0 +1,16 @@
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Web Serial API Example</title>
<link rel="stylesheet" href="styles.css">
</head>
<body>
<h1>Web Serial Connect</h1>
<button id="connect">Connect to Serial Device</button>
<pre id="output"></pre>
<script src="script.js"></script>
</body>
</html>

@ -0,0 +1,8 @@
import time
i = 0
while True:
print("Iteration: {}".format(i))
i+=1
time.sleep(1)

@ -0,0 +1,37 @@
document.getElementById('connect').addEventListener('click', async () => {
// Feature detection
if ('serial' in navigator) {
try {
// Request a port and open a connection
const port = await navigator.serial.requestPort();
await port.open({ baudRate: 115200 });
// Create a text decoder to decode the bytes from the serial device
const decoder = new TextDecoderStream();
const inputDone = port.readable.pipeTo(decoder.writable);
const inputStream = decoder.readable;
// Read data from the serial device
const reader = inputStream.getReader();
const outputElement = document.getElementById('output');
outputElement.textContent = '';
while (true) {
const { value, done } = await reader.read();
if (done) {
// Allow the serial port to be closed later.
reader.releaseLock();
break;
}
// Print the output to the webpage
outputElement.textContent += value;
// Scroll to the bottom as new data comes in
outputElement.scrollTop = outputElement.scrollHeight;
}
} catch (error) {
console.error('There was an error:', error);
}
} else {
console.log('Web Serial API not supported in this browser.');
}
});

@ -0,0 +1,35 @@
document.getElementById('connect').addEventListener('click', async () => {
// Feature detection
if ('serial' in navigator) {
try {
// Request a port and open a connection
const port = await navigator.serial.requestPort();
await port.open({ baudRate: 115200 });
// Create a text decoder to decode the bytes from the serial device
const decoder = new TextDecoderStream();
const inputDone = port.readable.pipeTo(decoder.writable);
const inputStream = decoder.readable;
// Read data from the serial device
const reader = inputStream.getReader();
const outputElement = document.getElementById('output');
outputElement.textContent = '';
while (true) {
const { value, done } = await reader.read();
if (done) {
// Allow the serial port to be closed later.
reader.releaseLock();
break;
}
// Print the output to the webpage
outputElement.textContent += value;
}
} catch (error) {
console.error('There was an error:', error);
}
} else {
console.log('Web Serial API not supported in this browser.');
}
});

@ -0,0 +1,22 @@
body {
margin: 0;
font-family: Arial, sans-serif;
}
button {
margin: 20px;
}
#output {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 33vh; /* Top third of the screen */
background-color: #333; /* Slightly darker background */
color: #fff;
overflow-y: scroll;
padding: 10px;
box-sizing: border-box;
border-bottom: 1px solid #555;
}

@ -0,0 +1,61 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Web Serial Console</title>
<script src="https://cdn.tailwindcss.com"></script>
</head>
<body class="h-full m-0 p-0 bg-neutral-100">
<div class="flex flex-col h-screen" id="container">
<div class="p-5" id="top-half">
<h1 class="text-3xl font-bold mb-4">Web Serial Console</h1>
<div class="flex flex-col md:flex-row justify-between items-start md:items-center mb-4 space-y-4 md:space-y-0">
<div class="flex flex-col md:flex-row items-start md:items-center space-y-4 md:space-y-0 md:space-x-4">
<div class="flex flex-col md:flex-row items-start md:items-center space-y-4 md:space-y-0 md:space-x-4">
<label for="serial-select" class="text-sm font-medium leading-6 text-gray-900">Port:</label>
<select id="serial-select" name="serial-select" class="font-medium bg-white block w-full md:w-auto rounded-sm border-0 py-1.5 pl-3 pr-10 text-gray-900 ring-1 ring-inset ring-gray-300 focus:ring-2 focus:ring-red-500 sm:text-sm sm:leading-6">
<option value="">Select a device...&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</option>
</select>
<button id="add-port-button" class="ml-2 p-1 bg-gray-200 rounded-sm hover:bg-gray-300 focus:outline-none focus:ring-2 focus:ring-red-500 focus:ring-opacity-50">
<!-- <svg xmlns="http://www.w3.org/2000/svg" class="h-5 w-5 text-gray-500" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="4">
<path stroke-linecap="square" stroke-linejoin="round" d="M12 4v16m8-8H4" />
</svg> -->
<svg xmlns="http://www.w3.org/2000/svg" class="h-5 w-5 text-gray-500" fill="none" viewBox="0 0 24 24" strokeWidth={1.5} stroke="currentColor" className="size-6">
<path strokeLinecap="round" strokeLinejoin="round" d="M12 4.5v15m7.5-7.5h-15" />
</svg>
</button>
<button id="refresh-ports" class="ml-2 p-1 bg-gray-200 rounded-sm hover:bg-gray-300 focus:outline-none focus:ring-2 focus:ring-red-500 focus:ring-opacity-50">
<svg xmlns="http://www.w3.org/2000/svg" class="h-5 w-5 text-gray-600" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="7" >
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 4v5h.582m15.356 2A8.001 8.001 0 004.582 9m0 0H9m11 11v-5h-.581m0 0a8.003 8.003 0 01-15.357-2m15.357 2H15" />
</svg>
</button>
<label for="baud" class="text-sm font-medium leading-6 text-gray-900">Baud:</label>
<select id="baud" name="baud" class="font-medium bg-white block w-full md:w-auto rounded-sm border-0 py-1.5 pl-3 pr-10 text-gray-900 ring-1 ring-inset ring-gray-300 focus:ring-2 focus:ring-red-500 sm:text-sm sm:leading-6">
<option value="4800">4800</option>
<option value="9600">9600</option>
<option value="19200">19200</option>
<option value="38400">38400</option>
<option value="57600">57600</option>
<option value="115200" selected>115200</option>
<option value="230400">230400</option>
<option value="460800">460800</option>
<option value="">921600</option>
</select>
</div>
<button id="connect-button" class="px-4 py-1 bg-red-500 text-white rounded-sm hover:bg-red-600 focus:outline-none focus:ring-2 focus:ring-red-500 focus:ring-opacity-50">Connect</button>
</div>
<div class="flex flex-col md:flex-row items-start md:items-center space-y-4 md:space-y-0 md:space-x-4">
<label class="flex items-center space-x-2">
<input type="checkbox" id="autoscroll-checkbox" checked class="form-checkbox">
<span class="text-sm font-medium leading-6 text-gray-900">Autoscroll</span>
</label>
<button id="clear-button" class="px-4 py-1 bg-red-500 text-white rounded-sm hover:bg-red-600 focus:outline-none focus:ring-2 focus:ring-red-500 focus:ring-opacity-50">Clear</button>
</div>
</div>
</div>
<div class="font-mono rounded-sm bg-white shadow-xl h-[calc(50vh-70px)] overflow-y-auto border border-gray-300 p-2.5 bg-gray-200 mx-5 mb-5" id="scrollable-element"></div>
</div>
<script src="script.js"></script>
</body>
</html>

@ -0,0 +1,72 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Scrollable Element with Autoscroll</title>
<style>
body, html {
height: 100%;
margin: 0;
padding: 0;
}
#container {
display: flex;
flex-direction: column;
height: 100%;
}
#top-half {
flex: 1;
padding: 20px;
}
#scrollable-element {
flex: 1;
overflow-y: auto;
border: 1px solid #ccc;
padding: 10px;
background-color: #e2e8f0;
}
</style>
</head>
<body>
<div id="container">
<div id="top-half">
<h1>Scrollable Element Demo</h1>
<label>
<input type="checkbox" id="autoscroll-checkbox" checked> Autoscroll
</label>
</div>
<div id="scrollable-element"></div>
</div>
<script>
const scrollableElement = document.getElementById('scrollable-element');
const autoscrollCheckbox = document.getElementById('autoscroll-checkbox');
let autoscroll = true;
autoscrollCheckbox.addEventListener('change', (e) => {
autoscroll = e.target.checked;
if (autoscroll) {
scrollToBottom();
}
});
function addText() {
const newText = document.createElement('p');
newText.textContent = `New text added at ${new Date().toLocaleTimeString()}`;
scrollableElement.appendChild(newText);
if (autoscroll) {
scrollToBottom();
}
}
function scrollToBottom() {
scrollableElement.scrollTop = scrollableElement.scrollHeight;
}
// Simulate adding text every 2 seconds
setInterval(addText, 2000);
</script>
</body>
</html>

@ -0,0 +1,68 @@
const scrollableElement = document.getElementById('scrollable-element');
const autoscrollCheckbox = document.getElementById('autoscroll-checkbox');
const clearButton = document.getElementById('clear-button');
const refreshPorts = document.getElementById('refresh-ports');
const select = document.getElementById('serial-select');
let autoscroll = true;
refreshPorts.addEventListener('click', async () => {
while (select.children.length > 1) {
select.removeChild(select.lastChild);
}
ports = await navigator.serial.getPorts();
console.log(ports)
ports.forEach(port => {
const option = document.createElement('option');
option.value = port;
const { usbVendorId, usbProductId } = port.getInfo();
option.text = `Device ${usbVendorId}:${usbProductId}`;
select.appendChild(option);
});
});
autoscrollCheckbox.addEventListener('change', (e) => {
autoscroll = e.target.checked;
if (autoscroll) {
scrollToBottom();
}
});
clearButton.addEventListener('click', () => {
while (scrollableElement.firstChild) {
scrollableElement.removeChild(scrollableElement.firstChild);
}
});
function addText() {
const newText = document.createElement('p');
newText.textContent = `New text added at ${new Date().toLocaleTimeString()}`;
scrollableElement.appendChild(newText);
if (autoscroll) {
scrollToBottom();
}
}
function scrollToBottom() {
scrollableElement.scrollTop = scrollableElement.scrollHeight;
}
async function updateSerialSelect(ports) {
const select = document.getElementById('serial-select');
// Clear existing options
select.innerHTML = '<option value="">Select a device...</option>';
ports.forEach(port => {
const option = document.createElement('option');
option.value = port;
option.text = `Serial device`;
select.appendChild(option);
});
}
// Simulate adding text every 2 seconds
setInterval(addText, 2000);