mirror of
https://github.com/PR0M3TH3AN/bitvid.git
synced 2025-09-08 06:58:43 +00:00
462 lines
15 KiB
HTML
462 lines
15 KiB
HTML
<!DOCTYPE html>
|
||
<html lang="en">
|
||
<head>
|
||
<meta charset="utf-8" />
|
||
<title>bitvid Content Appeals Form</title>
|
||
<!-- Link to your main stylesheet -->
|
||
<link rel="stylesheet" href="../../css/style.css" />
|
||
<style>
|
||
/* Override for form page to match modal field styling */
|
||
|
||
/* Remove width constraints from body so our container can be full width */
|
||
body {
|
||
background-color: transparent;
|
||
color: #fff;
|
||
font-family: system-ui, -apple-system, sans-serif;
|
||
margin: 20px;
|
||
}
|
||
|
||
/* Override the .container to use the full available width */
|
||
.container {
|
||
width: 100%;
|
||
max-width: 100%;
|
||
margin: 0;
|
||
padding: 0;
|
||
}
|
||
|
||
/* Card-like container for the form, similar to modal-content */
|
||
.form-container {
|
||
background-color: #111827; /* Tailwind's bg-gray-900 */
|
||
padding: 1.5rem;
|
||
border-radius: 0.5rem;
|
||
box-shadow: var(--shadow-md);
|
||
}
|
||
|
||
h1,
|
||
h2 {
|
||
color: #fff;
|
||
}
|
||
|
||
/* Labels in a light gray */
|
||
label {
|
||
display: block;
|
||
margin-top: 1em;
|
||
font-weight: bold;
|
||
color: #e5e7eb; /* Tailwind's text-gray-200 */
|
||
}
|
||
|
||
/* Input, textarea, and select mimic modal field styles */
|
||
input,
|
||
textarea,
|
||
select {
|
||
width: 100%;
|
||
margin-bottom: 0.75em;
|
||
background-color: #1f2937; /* Tailwind's bg-gray-800 */
|
||
color: #f3f4f6; /* Tailwind's text-gray-100 */
|
||
border: 1px solid #374151; /* Tailwind's border-gray-700 */
|
||
padding: 0.5em;
|
||
border-radius: 0.375rem; /* rounded-md */
|
||
box-sizing: border-box;
|
||
}
|
||
|
||
input:focus,
|
||
textarea:focus,
|
||
select:focus {
|
||
border-color: #3b82f6; /* blue-500 */
|
||
outline: none;
|
||
box-shadow: 0 0 0 1px #3b82f6;
|
||
}
|
||
|
||
/* Button styled similarly to modal publish button */
|
||
button {
|
||
padding: 0.5em 1em;
|
||
background: #3b82f6; /* blue-500 */
|
||
color: #fff;
|
||
border: none;
|
||
border-radius: 0.375rem;
|
||
cursor: pointer;
|
||
}
|
||
|
||
/* Status log area */
|
||
#status {
|
||
margin-top: 1em;
|
||
padding: 0.5em;
|
||
background: #111827;
|
||
white-space: pre-wrap;
|
||
min-height: 80px;
|
||
border-radius: 0.25rem;
|
||
}
|
||
.status-line {
|
||
margin: 0.25em 0;
|
||
}
|
||
.error {
|
||
color: #f87171; /* a red tint */
|
||
}
|
||
.success {
|
||
color: #3b82f6; /* blue-500 */
|
||
}
|
||
.warn {
|
||
color: #facc15; /* a yellow tone */
|
||
}
|
||
|
||
/* Custom Scrollbar styling for WebKit browsers */
|
||
::-webkit-scrollbar {
|
||
width: 8px;
|
||
height: 8px;
|
||
}
|
||
::-webkit-scrollbar-track {
|
||
background: transparent;
|
||
}
|
||
::-webkit-scrollbar-thumb {
|
||
background-color: #3b82f6;
|
||
border-radius: 4px;
|
||
}
|
||
/* Custom Scrollbar styling for Firefox */
|
||
* {
|
||
scrollbar-width: thin;
|
||
scrollbar-color: #3b82f6 transparent;
|
||
}
|
||
</style>
|
||
<!-- Load nostr‑tools v2.10.4 -->
|
||
<script src="https://cdn.jsdelivr.net/npm/nostr-tools@2.10.4/lib/nostr.bundle.min.js"></script>
|
||
</head>
|
||
<body>
|
||
<div class="container">
|
||
<div class="form-container">
|
||
<p>
|
||
If you believe your content was unfairly blocked or restricted on
|
||
bitvid, please complete this form. Appeals will be reviewed manually,
|
||
and decisions will be communicated back to you.
|
||
</p>
|
||
<form id="dm-form">
|
||
<!-- NPUB input removed as the recipient is now set by the code -->
|
||
<h2>1. User Information</h2>
|
||
<label for="contactMethod">Contact Method (if applicable):</label>
|
||
<input
|
||
type="text"
|
||
id="contactMethod"
|
||
placeholder="Nostr DM, email, or other"
|
||
/>
|
||
|
||
<h2>2. Content Details</h2>
|
||
<label for="videoTitle">Title of the Video:</label>
|
||
<input
|
||
type="text"
|
||
id="videoTitle"
|
||
placeholder="Enter the exact title"
|
||
/>
|
||
|
||
<label for="magnetLink">Magnet Link:</label>
|
||
<input
|
||
type="text"
|
||
id="magnetLink"
|
||
placeholder="Enter the magnet link"
|
||
/>
|
||
|
||
<label for="submissionDate">Date of Content Submission:</label>
|
||
<input type="date" id="submissionDate" />
|
||
|
||
<h2>3. Reason for Appeal</h2>
|
||
<label for="reasonBlocked"
|
||
>Why do you believe your content was unfairly blocked?</label
|
||
>
|
||
<textarea
|
||
id="reasonBlocked"
|
||
rows="3"
|
||
placeholder="Explain in detail"
|
||
></textarea>
|
||
|
||
<label for="fitsGuidelines"
|
||
>Does your content fit within bitvid's Community Guidelines?</label
|
||
>
|
||
<select id="fitsGuidelines">
|
||
<option value="">Select an option</option>
|
||
<option value="Yes">Yes</option>
|
||
<option value="No">No</option>
|
||
</select>
|
||
|
||
<label for="guidelinesCited"
|
||
>If yes, which guideline(s) support your appeal?</label
|
||
>
|
||
<textarea
|
||
id="guidelinesCited"
|
||
rows="2"
|
||
placeholder="Cite the specific guidelines"
|
||
></textarea>
|
||
|
||
<label for="editedContent"
|
||
>Was this content edited after being blocked?</label
|
||
>
|
||
<select id="editedContent">
|
||
<option value="">Select an option</option>
|
||
<option value="Yes">Yes</option>
|
||
<option value="No">No</option>
|
||
</select>
|
||
|
||
<label for="changesMade">If yes, what changes were made?</label>
|
||
<textarea
|
||
id="changesMade"
|
||
rows="2"
|
||
placeholder="Describe the modifications"
|
||
></textarea>
|
||
|
||
<h2>4. Additional Context</h2>
|
||
<label for="misunderstanding"
|
||
>Was there any misunderstanding or misclassification?</label
|
||
>
|
||
<textarea
|
||
id="misunderstanding"
|
||
rows="2"
|
||
placeholder="Provide context"
|
||
></textarea>
|
||
|
||
<label for="externalReferences"
|
||
>Are there external references that validate your appeal?</label
|
||
>
|
||
<textarea
|
||
id="externalReferences"
|
||
rows="2"
|
||
placeholder="Links, citations, or additional info"
|
||
></textarea>
|
||
|
||
<h2>5. Declaration</h2>
|
||
<p>
|
||
By submitting this appeal, you confirm that:
|
||
<br />- You are the original creator or authorized representative of
|
||
the content. <br />- Your appeal is submitted in good faith and
|
||
aligns with bitvid’s policies. <br />- You understand that final
|
||
decisions are at the discretion of bitvid’s moderation process.
|
||
</p>
|
||
<label for="signature">Signature (Digital or Written):</label>
|
||
<input type="text" id="signature" placeholder="Your signature" />
|
||
|
||
<label for="declarationDate">Date:</label>
|
||
<input type="date" id="declarationDate" />
|
||
|
||
<button type="submit">Submit Appeal</button>
|
||
</form>
|
||
<div id="status"></div>
|
||
</div>
|
||
</div>
|
||
<script>
|
||
document.addEventListener("DOMContentLoaded", () => {
|
||
// Logging functions for both on-page and console output.
|
||
function log(msg, type = "info") {
|
||
const div = document.createElement("div");
|
||
div.classList.add("status-line");
|
||
if (type === "error") div.classList.add("error");
|
||
if (type === "success") div.classList.add("success");
|
||
if (type === "warn") div.classList.add("warn");
|
||
div.textContent = msg;
|
||
document.getElementById("status").appendChild(div);
|
||
console.log(`[${type.toUpperCase()}] ${msg}`);
|
||
}
|
||
function clear() {
|
||
document.getElementById("status").innerHTML = "";
|
||
}
|
||
if (!window.NostrTools) {
|
||
log("NostrTools not loaded. Check console or ad-blockers.", "error");
|
||
return;
|
||
}
|
||
const {
|
||
generateSecretKey,
|
||
getPublicKey,
|
||
finalizeEvent,
|
||
nip04,
|
||
nip19,
|
||
SimplePool,
|
||
Relay,
|
||
} = window.NostrTools;
|
||
|
||
// Set the recipient's NPUB (your personal NPUB) here.
|
||
const recipientNpub =
|
||
"npub13yarr7j6vjqjjkahd63dmr27curypehx45ucue286ac7sft27y0srnpmpe";
|
||
|
||
const RELAYS = [
|
||
"wss://relay.snort.social",
|
||
"wss://relay.damus.io",
|
||
"wss://relay.primal.net",
|
||
];
|
||
const pool = new SimplePool();
|
||
|
||
document
|
||
.getElementById("dm-form")
|
||
.addEventListener("submit", async (ev) => {
|
||
ev.preventDefault();
|
||
clear();
|
||
try {
|
||
// Note: The customer's NPUB input is removed.
|
||
// Use the recipientNpub constant to get the public key.
|
||
|
||
const contactMethod = document
|
||
.getElementById("contactMethod")
|
||
.value.trim();
|
||
const videoTitle = document
|
||
.getElementById("videoTitle")
|
||
.value.trim();
|
||
const magnetLink = document
|
||
.getElementById("magnetLink")
|
||
.value.trim();
|
||
const submissionDate = document
|
||
.getElementById("submissionDate")
|
||
.value.trim();
|
||
const reasonBlocked = document
|
||
.getElementById("reasonBlocked")
|
||
.value.trim();
|
||
const fitsGuidelines = document
|
||
.getElementById("fitsGuidelines")
|
||
.value.trim();
|
||
const guidelinesCited = document
|
||
.getElementById("guidelinesCited")
|
||
.value.trim();
|
||
const editedContent = document
|
||
.getElementById("editedContent")
|
||
.value.trim();
|
||
const changesMade = document
|
||
.getElementById("changesMade")
|
||
.value.trim();
|
||
const misunderstanding = document
|
||
.getElementById("misunderstanding")
|
||
.value.trim();
|
||
const externalReferences = document
|
||
.getElementById("externalReferences")
|
||
.value.trim();
|
||
const signature = document
|
||
.getElementById("signature")
|
||
.value.trim();
|
||
const declarationDate = document
|
||
.getElementById("declarationDate")
|
||
.value.trim();
|
||
|
||
// Construct the appeal content.
|
||
const appealContent = `
|
||
# **bitvid Content Appeals Form**
|
||
|
||
**1. User Information**
|
||
- **Contact Method:** ${contactMethod || "N/A"}
|
||
|
||
**2. Content Details**
|
||
- **Title of the Video:** ${videoTitle}
|
||
- **Magnet Link:** ${magnetLink}
|
||
- **Date of Content Submission:** ${submissionDate}
|
||
|
||
**3. Reason for Appeal**
|
||
- **Why do you believe your content was unfairly blocked?**
|
||
${reasonBlocked}
|
||
- **Does your content fit within bitvid's Community Guidelines?**
|
||
${fitsGuidelines}
|
||
- **If yes, which guideline(s) support your appeal?**
|
||
${guidelinesCited}
|
||
- **Was this content edited after being blocked?**
|
||
${editedContent}
|
||
- **If yes, what changes were made?**
|
||
${changesMade}
|
||
|
||
**4. Additional Context**
|
||
- **Was there any misunderstanding or misclassification?**
|
||
${misunderstanding}
|
||
- **Are there external references that validate your appeal?**
|
||
${externalReferences}
|
||
|
||
**5. Declaration**
|
||
By submitting this appeal, you confirm that:
|
||
- You are the original creator or an authorized representative of the content.
|
||
- Your appeal is submitted in good faith and aligns with bitvid’s policies.
|
||
- You understand that final decisions are at the discretion of bitvid’s moderation process.
|
||
|
||
**Signature (Digital or Written):** ${signature}
|
||
**Date:** ${declarationDate}
|
||
|
||
---
|
||
|
||
**Processing Time:** Appeals will be reviewed within **7-14 days**. If additional information is required, you will be contacted via your provided contact method.
|
||
|
||
For further questions, reach out through bitvid’s Nostr support channels.
|
||
`.trim();
|
||
|
||
log("[DEBUG] Constructed appeal content:\n" + appealContent);
|
||
|
||
// Decode the recipient NPUB to get the public key.
|
||
log("Decoding recipient npub...");
|
||
const decoded = nip19.decode(recipientNpub);
|
||
log("[DEBUG] Decoded npub: " + JSON.stringify(decoded));
|
||
if (decoded.type !== "npub") {
|
||
throw new Error("Decoded type is not npub.");
|
||
}
|
||
const targetPubHex = decoded.data;
|
||
log("Recipient pubkey: " + targetPubHex.slice(0, 16) + "...");
|
||
|
||
// Generate an ephemeral key pair.
|
||
log("Generating ephemeral key...");
|
||
const ephemeralPriv = generateSecretKey();
|
||
const ephemeralPubHex = getPublicKey(ephemeralPriv);
|
||
log("Ephemeral pubkey: " + ephemeralPubHex.slice(0, 16) + "...");
|
||
|
||
// Encrypt the appeal content.
|
||
log("Encrypting appeal content (nip04)...");
|
||
const ciphertext = await nip04.encrypt(
|
||
ephemeralPriv,
|
||
targetPubHex,
|
||
appealContent
|
||
);
|
||
log("[DEBUG] Ciphertext: " + ciphertext);
|
||
log("Encryption done.");
|
||
|
||
// Build the event template.
|
||
const now = Math.floor(Date.now() / 1000);
|
||
const eventTemplate = {
|
||
kind: 4,
|
||
created_at: now,
|
||
tags: [["p", targetPubHex]],
|
||
content: ciphertext,
|
||
};
|
||
log(
|
||
"[DEBUG] Event template before finalizing: " +
|
||
JSON.stringify(eventTemplate)
|
||
);
|
||
|
||
// Finalize the event.
|
||
const event = finalizeEvent(eventTemplate, ephemeralPriv);
|
||
log("[DEBUG] Final event: " + JSON.stringify(event));
|
||
|
||
// Publish the event to all relays.
|
||
log("Publishing the appeal to relays...");
|
||
await Promise.any(pool.publish(RELAYS, event));
|
||
log("At least one relay accepted the event.", "success");
|
||
|
||
// Subscribe to each relay.
|
||
for (const url of RELAYS) {
|
||
log("Connecting to " + url + " for subscription...");
|
||
const relay = await Relay.connect(url);
|
||
relay.subscribe([{ authors: [ephemeralPubHex], kinds: [4] }], {
|
||
onEvent(foundEvent) {
|
||
if (foundEvent.id === event.id) {
|
||
log(
|
||
"[" +
|
||
url +
|
||
"] => Found our appeal in storage! ID: " +
|
||
foundEvent.id.slice(0, 8) +
|
||
"...",
|
||
"success"
|
||
);
|
||
}
|
||
},
|
||
onEose() {
|
||
relay.close();
|
||
},
|
||
});
|
||
}
|
||
|
||
log(
|
||
"Done. If the logs show that at least one relay accepted the event and the appeal was found in storage, a moderator will review your appeal within 7-14 days."
|
||
);
|
||
} catch (err) {
|
||
log("Error: " + err.message, "error");
|
||
console.error(err);
|
||
}
|
||
});
|
||
});
|
||
</script>
|
||
</body>
|
||
</html>
|