263 lines
13 KiB
HTML
263 lines
13 KiB
HTML
<!--
|
|
~ This file is part of ImageFrame.
|
|
~
|
|
~ Copyright (C) 2025. LoohpJames <jamesloohp@gmail.com>
|
|
~ Copyright (C) 2025. Contributors
|
|
~
|
|
~ This program is free software: you can redistribute it and/or modify
|
|
~ it under the terms of the GNU General Public License as published by
|
|
~ the Free Software Foundation, either version 3 of the License, or
|
|
~ (at your option) any later version.
|
|
~
|
|
~ This program is distributed in the hope that it will be useful,
|
|
~ but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
~ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
~ GNU General Public License for more details.
|
|
~
|
|
~ You should have received a copy of the GNU General Public License
|
|
~ along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
-->
|
|
|
|
<!DOCTYPE html>
|
|
<html lang="en">
|
|
<head>
|
|
<meta charset="UTF-8">
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
<title>ImageFrame Upload</title>
|
|
<link href="https://fonts.cdnfonts.com/css/minecraftia" rel="stylesheet">
|
|
<link rel="icon" type="image/x-icon" href="imageframe.png">
|
|
<style>
|
|
* {
|
|
font-family: 'Minecraftia', sans-serif;
|
|
}
|
|
body {
|
|
display: block;
|
|
justify-content: center;
|
|
align-items: center;
|
|
height: 90vh;
|
|
min-height: 90vh;
|
|
text-align: center;
|
|
background-image: url("background.webp");
|
|
}
|
|
.title {
|
|
color: white;
|
|
margin-bottom: 0;
|
|
margin-top: 30px;
|
|
}
|
|
.expire {
|
|
margin: 0;
|
|
font-size: 14px;
|
|
color: #AAAAAA;
|
|
}
|
|
.name {
|
|
color: #FFFF55;
|
|
}
|
|
.container {
|
|
margin: 30px auto 30px auto;
|
|
height: 80%;
|
|
text-align: center;
|
|
background: #D2BF82;
|
|
padding: 20px;
|
|
border-radius: 10px;
|
|
aspect-ratio: 1;
|
|
box-shadow: 0 0 10px rgba(0, 0, 0, 0.1);
|
|
}
|
|
.drop-zone {
|
|
width: 100%;
|
|
height: calc(100% - 45px);
|
|
background: #A25430;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
text-align: center;
|
|
margin-bottom: 12px;
|
|
cursor: pointer;
|
|
position: relative;
|
|
overflow: hidden;
|
|
color: white;
|
|
box-shadow: inset 0 0 3px black;
|
|
}
|
|
.drop-zone img {
|
|
max-width: 100%;
|
|
max-height: 100%;
|
|
display: none;
|
|
position: absolute;
|
|
object-fit: cover;
|
|
}
|
|
button {
|
|
background-image: url();
|
|
border-color: #AAA #565656 #565656 #AAA;
|
|
padding: 7px 20px;
|
|
font-size: 15px;
|
|
cursor: pointer;
|
|
text-shadow: 3px 3px #4C4C4C;
|
|
outline: 2px solid #000;
|
|
color: white;
|
|
}
|
|
button:hover:enabled {
|
|
background-image: url();
|
|
border-color: #BDC6FF #59639A #59639A #BDC6FF;
|
|
}
|
|
button:disabled {
|
|
cursor: not-allowed;
|
|
filter: brightness(0.5);
|
|
}
|
|
.closed-container {
|
|
text-align: center;
|
|
height: 100%;
|
|
align-content: center;
|
|
}
|
|
.closed-icon {
|
|
width: 120px;
|
|
}
|
|
.success-title {
|
|
color: #55FF55;
|
|
margin: 10px;
|
|
}
|
|
.error-title {
|
|
color: #FF5555;
|
|
margin: 10px;
|
|
}
|
|
@media screen and (max-width: 700px) {
|
|
.container {
|
|
aspect-ratio: unset;
|
|
}
|
|
}
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<h2 class="title">ImageFrame Upload<label id="titlePlayer"></label></h2>
|
|
<p class="expire" id="expire">This link expires in ...</p>
|
|
<div class="container" id="container">
|
|
<div class="drop-zone" id="dropZone">
|
|
Drop image here or click to select
|
|
<img id="imagePreview" alt="Image Preview" src="">
|
|
</div>
|
|
<input type="file" id="fileInput" accept="image/png, image/jpeg, image/gif" hidden>
|
|
<button id="uploadBtn" disabled>Upload</button>
|
|
</div>
|
|
|
|
<script>
|
|
const urlParams = new URLSearchParams(window.location.search);
|
|
const user = urlParams.get('user');
|
|
const uploadId = urlParams.get('id');
|
|
const player = urlParams.get('player');
|
|
const expire = urlParams.get('expire');
|
|
|
|
function showExpired() {
|
|
const forPlayer = player ? ` for <span class="name">${player}</span>` : "";
|
|
document.body.innerHTML = `
|
|
<div class="closed-container">
|
|
<img src="imageframe.png" alt="ImageFrame" class="closed-icon">
|
|
<h2 class="error-title">This link${forPlayer} had expired!<br>Please create a new one in-game.</h2>
|
|
</div>
|
|
`;
|
|
}
|
|
function showError() {
|
|
const forPlayer = player ? ` for <span class="name">${player}</span>` : "";
|
|
document.body.innerHTML = `
|
|
<div class="closed-container">
|
|
<img src="imageframe.png" alt="ImageFrame" class="closed-icon">
|
|
<h2 class="error-title">There is an error while uploading${forPlayer}.<br>Please create a new one in-game.</h2>
|
|
</div>
|
|
`;
|
|
}
|
|
function showSuccess() {
|
|
const forPlayer = player ? ` for <span class="name">${player}</span>` : "";
|
|
document.body.innerHTML = `
|
|
<div class="closed-container">
|
|
<img src="imageframe.png" alt="ImageFrame" class="closed-icon">
|
|
<h1 class="success-title">Upload successful${forPlayer}!<br>Please continue in-game!</h1>
|
|
</div>
|
|
`;
|
|
}
|
|
|
|
if (player) {
|
|
document.getElementById('titlePlayer').innerHTML = ` for <span class="name">${player}</span>`;
|
|
}
|
|
if (expire) {
|
|
const element = document.getElementById('expire');
|
|
const taskId = setInterval(() => {
|
|
const now = Date.now();
|
|
const diff = Math.floor((expire - now) / 1000);
|
|
if (diff < 0) {
|
|
clearInterval(taskId);
|
|
showExpired();
|
|
return;
|
|
}
|
|
const minutes = Math.floor(diff / 60);
|
|
const seconds = (diff % 60).toString().padStart(2, '0');
|
|
element.innerHTML = minutes > 0 ? `This link expires in ${minutes}m ${seconds}s` : `This link expires in ${seconds}s`;
|
|
}, 500);
|
|
}
|
|
|
|
const dropZone = document.getElementById("dropZone");
|
|
const fileInput = document.getElementById("fileInput");
|
|
const uploadBtn = document.getElementById("uploadBtn");
|
|
const imagePreview = document.getElementById("imagePreview");
|
|
let selectedFile = null;
|
|
|
|
dropZone.addEventListener("click", () => fileInput.click());
|
|
fileInput.addEventListener("change", handleFileSelect);
|
|
dropZone.addEventListener("dragover", (e) => {
|
|
e.preventDefault();
|
|
dropZone.classList.add("highlight");
|
|
});
|
|
dropZone.addEventListener("dragleave", () => dropZone.classList.remove("highlight"));
|
|
dropZone.addEventListener("drop", (e) => {
|
|
e.preventDefault();
|
|
dropZone.classList.remove("highlight");
|
|
if (e.dataTransfer.files.length) {
|
|
handleFileSelect({ target: { files: e.dataTransfer.files } });
|
|
}
|
|
});
|
|
|
|
function handleFileSelect(event) {
|
|
const file = event.target.files[0];
|
|
if (file && ["image/png", "image/jpeg", "image/gif", "image/webp"].includes(file.type)) {
|
|
selectedFile = file;
|
|
uploadBtn.disabled = false;
|
|
|
|
const reader = new FileReader();
|
|
reader.onload = function(e) {
|
|
imagePreview.src = e.target.result;
|
|
imagePreview.style.display = "block";
|
|
};
|
|
reader.readAsDataURL(file);
|
|
} else {
|
|
alert("Only PNG, JPG, and GIF files are allowed.");
|
|
fileInput.value = "";
|
|
selectedFile = null;
|
|
uploadBtn.disabled = true;
|
|
imagePreview.style.display = "none";
|
|
}
|
|
}
|
|
|
|
uploadBtn.addEventListener("click", () => {
|
|
if (!selectedFile) return;
|
|
const formData = new FormData();
|
|
formData.append("image", selectedFile);
|
|
|
|
fetch(`/upload?user=${user}&id=${uploadId}`, {
|
|
method: "POST",
|
|
body: formData
|
|
})
|
|
.then(response => response.json())
|
|
.then(data => {
|
|
if (data.hasOwnProperty("error")) {
|
|
showError();
|
|
console.log("Upload failed: " + data.error);
|
|
} else {
|
|
showSuccess();
|
|
console.log("Upload successful: " + data.message);
|
|
}
|
|
})
|
|
.catch(error => {
|
|
showError();
|
|
console.log("Upload failed: " + error);
|
|
});
|
|
});
|
|
</script>
|
|
</body>
|
|
</html>
|