Compare commits

...

5 Commits

View File

@ -37,10 +37,9 @@
{% include "dialog_help.html" %} {% include "dialog_help.html" %}
{% include "dialog_online.html" %} {% include "dialog_online.html" %}
<script type="module"> <script type="module">
import { app } from "/app.js"; import {app} from "/app.js";
import { Schedule } from "/schedule.js";
// --- Cache selectors --- // --- Cache selectors ---
const chatInputField = document.querySelector("chat-input"); const chatInputField = document.querySelector("chat-input");
const messagesContainer = document.querySelector(".chat-messages"); const messagesContainer = document.querySelector(".chat-messages");
const chatArea = document.querySelector(".chat-area"); const chatArea = document.querySelector(".chat-area");
@ -99,21 +98,61 @@ setInterval(() => requestIdleCallback(updateTimes), 30000);
// --- Paste & drag/drop uploads --- // --- Paste & drag/drop uploads ---
const textBox = chatInputField.textarea; const textBox = chatInputField.textarea;
textBox.addEventListener("paste", async (e) => {
try { function uploadDataTransfer(dt) {
const clipboardItems = await navigator.clipboard.read(); if (dt.items.length > 0) {
const dt = new DataTransfer(); const uploadButton = chatInputField.fileUploadGrid;
for (const item of clipboardItems) {
for (const type of item.types.filter(t => !t.startsWith('text/'))) { for (const item of dt.items) {
const blob = await item.getType(type); if (item.kind === "file") {
dt.items.add(new File([blob], "image.png", { type })); const file = item.getAsFile();
if (file) {
uploadButton.uploadsStarted++
uploadButton.createTile(file)
} else {
console.error("Failed to get file from DataTransferItem");
}
} }
} }
if (dt.items.length > 0) { }
const uploadButton = chatInputField.uploadButton; }
const input = uploadButton.shadowRoot.querySelector('.file-input'); textBox.addEventListener("paste", async (e) => {
input.files = dt.files; try {
await uploadButton.uploadFiles(); console.log("Pasted data:", e.clipboardData);
if (e.clipboardData.types.every(v => v.startsWith("text/"))) {
const codeType = e.clipboardData.types.find(t => !t.startsWith('text/plain') && !t.startsWith('text/html') && !t.startsWith('text/rtf'));
const probablyCode = codeType ||e.clipboardData.types.some(t => !t.startsWith('text/plain'));
for (const item of e.clipboardData.items) {
if (item.kind === "string" && item.type === "text/plain") {
e.preventDefault();
item.getAsString(text => {
const value = chatInputField.value;
if (probablyCode) {
let code = text;
const minIndentDepth = code.split('\n').reduce((acc, line) => {
if (!line.trim()) return acc;
const match = line.match(/^(\s*)/);
return match ? Math.min(acc, match[1].length) : acc;
}, 9000);
code = code.split('\n').map(line => line.slice(minIndentDepth)).join('\n')
text = `\`\`\`${codeType?.split('/')?.[1] ?? ''}\n${code}\n\`\`\`\n`;
}
const area = chatInputField.textarea
if(area){
const start = area.selectionStart
if ("\n" !== value[start - 1]) {
text = `\n${text}`;
}
document.execCommand('insertText', false, text);
}
});
break;
}
}
} else {
uploadDataTransfer(e.clipboardData);
} }
} catch (error) { } catch (error) {
console.error("Failed to read clipboard contents: ", error); console.error("Failed to read clipboard contents: ", error);
@ -121,13 +160,7 @@ textBox.addEventListener("paste", async (e) => {
}); });
chatArea.addEventListener("drop", async (e) => { chatArea.addEventListener("drop", async (e) => {
e.preventDefault(); e.preventDefault();
const dt = e.dataTransfer; uploadDataTransfer(e.dataTransfer);
if (dt.items.length > 0) {
const uploadButton = chatInputField.uploadButton;
const input = uploadButton.shadowRoot.querySelector('.file-input');
input.files = dt.files;
await uploadButton.uploadFiles();
}
}); });
chatArea.addEventListener("dragover", e => { chatArea.addEventListener("dragover", e => {
e.preventDefault(); e.preventDefault();